conf.c 13.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5
/*
 * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
 * Released under the terms of the GNU GPL v2.0.
 */

E
EGRY Gabor 已提交
6
#include <locale.h>
L
Linus Torvalds 已提交
7
#include <ctype.h>
8
#include <stdio.h>
9
#include <stdlib.h>
L
Linus Torvalds 已提交
10 11
#include <string.h>
#include <time.h>
12
#include <unistd.h>
S
Sam Ravnborg 已提交
13
#include <getopt.h>
L
Linus Torvalds 已提交
14
#include <sys/stat.h>
15
#include <sys/time.h>
L
Linus Torvalds 已提交
16 17 18 19 20 21 22

#define LKC_DIRECT_LINK
#include "lkc.h"

static void conf(struct menu *menu);
static void check_conf(struct menu *menu);

S
Sam Ravnborg 已提交
23 24 25 26 27 28 29
enum input_mode {
	oldaskconfig,
	silentoldconfig,
	oldconfig,
	allnoconfig,
	allyesconfig,
	allmodconfig,
S
Sam Ravnborg 已提交
30
	alldefconfig,
S
Sam Ravnborg 已提交
31 32
	randconfig,
	defconfig,
S
Sam Ravnborg 已提交
33
	savedefconfig,
34
	listnewconfig,
35
	oldnoconfig,
S
Sam Ravnborg 已提交
36 37
} input_mode = oldaskconfig;

L
Linus Torvalds 已提交
38 39 40 41
char *defconfig_file;

static int indent = 1;
static int valid_stdin = 1;
42
static int sync_kconfig;
L
Linus Torvalds 已提交
43
static int conf_cnt;
J
J.A. Magallon 已提交
44
static char line[128];
L
Linus Torvalds 已提交
45 46
static struct menu *rootEntry;

47
static void print_help(struct menu *menu)
48
{
49 50 51 52 53 54
	struct gstr help = str_new();

	menu_get_ext_help(menu, &help);

	printf("\n%s\n", str_get(&help));
	str_free(&help);
55 56
}

J
J.A. Magallon 已提交
57
static void strip(char *str)
L
Linus Torvalds 已提交
58
{
J
J.A. Magallon 已提交
59
	char *p = str;
L
Linus Torvalds 已提交
60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
	int l;

	while ((isspace(*p)))
		p++;
	l = strlen(p);
	if (p != str)
		memmove(str, p, l + 1);
	if (!l)
		return;
	p = str + l - 1;
	while ((isspace(*p)))
		*p-- = 0;
}

static void check_stdin(void)
{
76
	if (!valid_stdin) {
77 78 79
		printf(_("aborted!\n\n"));
		printf(_("Console input/output is redirected. "));
		printf(_("Run 'make oldconfig' to update configuration.\n\n"));
L
Linus Torvalds 已提交
80 81 82 83
		exit(1);
	}
}

84
static int conf_askvalue(struct symbol *sym, const char *def)
L
Linus Torvalds 已提交
85 86 87 88
{
	enum symbol_type type = sym_get_type(sym);

	if (!sym_has_value(sym))
E
EGRY Gabor 已提交
89
		printf(_("(NEW) "));
L
Linus Torvalds 已提交
90 91 92 93 94 95 96 97

	line[0] = '\n';
	line[1] = 0;

	if (!sym_is_changable(sym)) {
		printf("%s\n", def);
		line[0] = '\n';
		line[1] = 0;
98
		return 0;
L
Linus Torvalds 已提交
99 100 101
	}

	switch (input_mode) {
S
Sam Ravnborg 已提交
102 103
	case oldconfig:
	case silentoldconfig:
L
Linus Torvalds 已提交
104 105
		if (sym_has_value(sym)) {
			printf("%s\n", def);
106
			return 0;
L
Linus Torvalds 已提交
107 108
		}
		check_stdin();
109
		/* fall through */
S
Sam Ravnborg 已提交
110
	case oldaskconfig:
L
Linus Torvalds 已提交
111
		fflush(stdout);
112
		xfgets(line, 128, stdin);
113
		return 1;
L
Linus Torvalds 已提交
114 115 116 117 118 119 120 121 122
	default:
		break;
	}

	switch (type) {
	case S_INT:
	case S_HEX:
	case S_STRING:
		printf("%s\n", def);
123
		return 1;
L
Linus Torvalds 已提交
124 125 126 127
	default:
		;
	}
	printf("%s", line);
128
	return 1;
L
Linus Torvalds 已提交
129 130
}

T
Trevor Keith 已提交
131
static int conf_string(struct menu *menu)
L
Linus Torvalds 已提交
132 133
{
	struct symbol *sym = menu->sym;
134
	const char *def;
L
Linus Torvalds 已提交
135 136

	while (1) {
E
EGRY Gabor 已提交
137
		printf("%*s%s ", indent - 1, "", _(menu->prompt->text));
L
Linus Torvalds 已提交
138 139 140 141
		printf("(%s) ", sym->name);
		def = sym_get_string_value(sym);
		if (sym_get_string_value(sym))
			printf("[%s] ", def);
142 143
		if (!conf_askvalue(sym, def))
			return 0;
L
Linus Torvalds 已提交
144 145 146 147 148 149
		switch (line[0]) {
		case '\n':
			break;
		case '?':
			/* print help */
			if (line[1] == '\n') {
150
				print_help(menu);
L
Linus Torvalds 已提交
151 152 153
				def = NULL;
				break;
			}
154
			/* fall through */
L
Linus Torvalds 已提交
155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
		default:
			line[strlen(line)-1] = 0;
			def = line;
		}
		if (def && sym_set_string_value(sym, def))
			return 0;
	}
}

static int conf_sym(struct menu *menu)
{
	struct symbol *sym = menu->sym;
	tristate oldval, newval;

	while (1) {
E
EGRY Gabor 已提交
170
		printf("%*s%s ", indent - 1, "", _(menu->prompt->text));
L
Linus Torvalds 已提交
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191
		if (sym->name)
			printf("(%s) ", sym->name);
		putchar('[');
		oldval = sym_get_tristate_value(sym);
		switch (oldval) {
		case no:
			putchar('N');
			break;
		case mod:
			putchar('M');
			break;
		case yes:
			putchar('Y');
			break;
		}
		if (oldval != no && sym_tristate_within_range(sym, no))
			printf("/n");
		if (oldval != mod && sym_tristate_within_range(sym, mod))
			printf("/m");
		if (oldval != yes && sym_tristate_within_range(sym, yes))
			printf("/y");
192
		if (menu_has_help(menu))
L
Linus Torvalds 已提交
193 194
			printf("/?");
		printf("] ");
195 196
		if (!conf_askvalue(sym, sym_get_string_value(sym)))
			return 0;
L
Linus Torvalds 已提交
197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
		strip(line);

		switch (line[0]) {
		case 'n':
		case 'N':
			newval = no;
			if (!line[1] || !strcmp(&line[1], "o"))
				break;
			continue;
		case 'm':
		case 'M':
			newval = mod;
			if (!line[1])
				break;
			continue;
		case 'y':
		case 'Y':
			newval = yes;
			if (!line[1] || !strcmp(&line[1], "es"))
				break;
			continue;
		case 0:
			newval = oldval;
			break;
		case '?':
			goto help;
		default:
			continue;
		}
		if (sym_set_tristate_value(sym, newval))
			return 0;
help:
229
		print_help(menu);
L
Linus Torvalds 已提交
230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256
	}
}

static int conf_choice(struct menu *menu)
{
	struct symbol *sym, *def_sym;
	struct menu *child;
	bool is_new;

	sym = menu->sym;
	is_new = !sym_has_value(sym);
	if (sym_is_changable(sym)) {
		conf_sym(menu);
		sym_calc_value(sym);
		switch (sym_get_tristate_value(sym)) {
		case no:
			return 1;
		case mod:
			return 0;
		case yes:
			break;
		}
	} else {
		switch (sym_get_tristate_value(sym)) {
		case no:
			return 1;
		case mod:
E
EGRY Gabor 已提交
257
			printf("%*s%s\n", indent - 1, "", _(menu_get_prompt(menu)));
L
Linus Torvalds 已提交
258 259 260 261 262 263 264 265 266
			return 0;
		case yes:
			break;
		}
	}

	while (1) {
		int cnt, def;

E
EGRY Gabor 已提交
267
		printf("%*s%s\n", indent - 1, "", _(menu_get_prompt(menu)));
L
Linus Torvalds 已提交
268 269
		def_sym = sym_get_choice_value(sym);
		cnt = def = 0;
270
		line[0] = 0;
L
Linus Torvalds 已提交
271 272 273 274
		for (child = menu->list; child; child = child->next) {
			if (!menu_is_visible(child))
				continue;
			if (!child->sym) {
E
EGRY Gabor 已提交
275
				printf("%*c %s\n", indent, '*', _(menu_get_prompt(child)));
L
Linus Torvalds 已提交
276 277 278 279 280 281 282 283
				continue;
			}
			cnt++;
			if (child->sym == def_sym) {
				def = cnt;
				printf("%*c", indent, '>');
			} else
				printf("%*c", indent, ' ');
E
EGRY Gabor 已提交
284
			printf(" %d. %s", cnt, _(menu_get_prompt(child)));
L
Linus Torvalds 已提交
285 286 287
			if (child->sym->name)
				printf(" (%s)", child->sym->name);
			if (!sym_has_value(child->sym))
E
EGRY Gabor 已提交
288
				printf(_(" (NEW)"));
L
Linus Torvalds 已提交
289 290
			printf("\n");
		}
E
EGRY Gabor 已提交
291
		printf(_("%*schoice"), indent - 1, "");
L
Linus Torvalds 已提交
292 293 294 295 296
		if (cnt == 1) {
			printf("[1]: 1\n");
			goto conf_childs;
		}
		printf("[1-%d", cnt);
297
		if (menu_has_help(menu))
L
Linus Torvalds 已提交
298 299 300
			printf("?");
		printf("]: ");
		switch (input_mode) {
S
Sam Ravnborg 已提交
301 302
		case oldconfig:
		case silentoldconfig:
L
Linus Torvalds 已提交
303 304 305 306 307 308
			if (!is_new) {
				cnt = def;
				printf("%d\n", cnt);
				break;
			}
			check_stdin();
309
			/* fall through */
S
Sam Ravnborg 已提交
310
		case oldaskconfig:
L
Linus Torvalds 已提交
311
			fflush(stdout);
312
			xfgets(line, 128, stdin);
L
Linus Torvalds 已提交
313 314
			strip(line);
			if (line[0] == '?') {
315
				print_help(menu);
L
Linus Torvalds 已提交
316 317 318 319 320 321 322 323 324
				continue;
			}
			if (!line[0])
				cnt = def;
			else if (isdigit(line[0]))
				cnt = atoi(line);
			else
				continue;
			break;
325 326
		default:
			break;
L
Linus Torvalds 已提交
327 328 329 330 331 332 333 334 335 336 337
		}

	conf_childs:
		for (child = menu->list; child; child = child->next) {
			if (!child->sym || !menu_is_visible(child))
				continue;
			if (!--cnt)
				break;
		}
		if (!child)
			continue;
338
		if (line[0] && line[strlen(line) - 1] == '?') {
339
			print_help(child);
L
Linus Torvalds 已提交
340 341 342
			continue;
		}
		sym_set_choice_value(sym, child->sym);
343
		for (child = child->list; child; child = child->next) {
L
Linus Torvalds 已提交
344
			indent += 2;
345
			conf(child);
L
Linus Torvalds 已提交
346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367
			indent -= 2;
		}
		return 1;
	}
}

static void conf(struct menu *menu)
{
	struct symbol *sym;
	struct property *prop;
	struct menu *child;

	if (!menu_is_visible(menu))
		return;

	sym = menu->sym;
	prop = menu->prompt;
	if (prop) {
		const char *prompt;

		switch (prop->type) {
		case P_MENU:
S
Sam Ravnborg 已提交
368
			if ((input_mode == silentoldconfig ||
369
			     input_mode == listnewconfig ||
370
			     input_mode == oldnoconfig) &&
371
			    rootEntry != menu) {
L
Linus Torvalds 已提交
372 373 374
				check_conf(menu);
				return;
			}
375
			/* fall through */
L
Linus Torvalds 已提交
376 377 378 379 380
		case P_COMMENT:
			prompt = menu_get_prompt(menu);
			if (prompt)
				printf("%*c\n%*c %s\n%*c\n",
					indent, '*',
E
EGRY Gabor 已提交
381
					indent, '*', _(prompt),
L
Linus Torvalds 已提交
382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426
					indent, '*');
		default:
			;
		}
	}

	if (!sym)
		goto conf_childs;

	if (sym_is_choice(sym)) {
		conf_choice(menu);
		if (sym->curr.tri != mod)
			return;
		goto conf_childs;
	}

	switch (sym->type) {
	case S_INT:
	case S_HEX:
	case S_STRING:
		conf_string(menu);
		break;
	default:
		conf_sym(menu);
		break;
	}

conf_childs:
	if (sym)
		indent += 2;
	for (child = menu->list; child; child = child->next)
		conf(child);
	if (sym)
		indent -= 2;
}

static void check_conf(struct menu *menu)
{
	struct symbol *sym;
	struct menu *child;

	if (!menu_is_visible(menu))
		return;

	sym = menu->sym;
427 428 429
	if (sym && !sym_has_value(sym)) {
		if (sym_is_changable(sym) ||
		    (sym_is_choice(sym) && sym_get_tristate_value(sym) == yes)) {
430 431
			if (input_mode == listnewconfig) {
				if (sym->name && !sym_is_choice_value(sym)) {
432
					printf("%s%s\n", CONFIG_, sym->name);
433
				}
434
			} else if (input_mode != oldnoconfig) {
435 436 437 438 439
				if (!conf_cnt++)
					printf(_("*\n* Restart config...\n*\n"));
				rootEntry = menu_get_parent_menu(menu);
				conf(rootEntry);
			}
L
Linus Torvalds 已提交
440 441 442 443 444 445 446
		}
	}

	for (child = menu->list; child; child = child->next)
		check_conf(child);
}

S
Sam Ravnborg 已提交
447 448 449 450 451
static struct option long_opts[] = {
	{"oldaskconfig",    no_argument,       NULL, oldaskconfig},
	{"oldconfig",       no_argument,       NULL, oldconfig},
	{"silentoldconfig", no_argument,       NULL, silentoldconfig},
	{"defconfig",       optional_argument, NULL, defconfig},
S
Sam Ravnborg 已提交
452
	{"savedefconfig",   required_argument, NULL, savedefconfig},
S
Sam Ravnborg 已提交
453 454 455
	{"allnoconfig",     no_argument,       NULL, allnoconfig},
	{"allyesconfig",    no_argument,       NULL, allyesconfig},
	{"allmodconfig",    no_argument,       NULL, allmodconfig},
S
Sam Ravnborg 已提交
456
	{"alldefconfig",    no_argument,       NULL, alldefconfig},
S
Sam Ravnborg 已提交
457
	{"randconfig",      no_argument,       NULL, randconfig},
458
	{"listnewconfig",   no_argument,       NULL, listnewconfig},
459
	{"oldnoconfig",     no_argument,       NULL, oldnoconfig},
S
Sam Ravnborg 已提交
460 461 462
	{NULL, 0, NULL, 0}
};

L
Linus Torvalds 已提交
463 464
int main(int ac, char **av)
{
465
	int opt;
L
Linus Torvalds 已提交
466 467 468
	const char *name;
	struct stat tmpstat;

E
EGRY Gabor 已提交
469 470 471 472
	setlocale(LC_ALL, "");
	bindtextdomain(PACKAGE, LOCALEDIR);
	textdomain(PACKAGE);

473
	while ((opt = getopt_long(ac, av, "", long_opts, NULL)) != -1) {
S
Sam Ravnborg 已提交
474
		input_mode = (enum input_mode)opt;
475
		switch (opt) {
S
Sam Ravnborg 已提交
476
		case silentoldconfig:
477
			sync_kconfig = 1;
L
Linus Torvalds 已提交
478
			break;
S
Sam Ravnborg 已提交
479
		case defconfig:
S
Sam Ravnborg 已提交
480
		case savedefconfig:
481
			defconfig_file = optarg;
L
Linus Torvalds 已提交
482
			break;
S
Sam Ravnborg 已提交
483
		case randconfig:
484 485 486 487 488 489 490 491 492 493 494 495
		{
			struct timeval now;
			unsigned int seed;

			/*
			 * Use microseconds derived seed,
			 * compensate for systems where it may be zero
			 */
			gettimeofday(&now, NULL);

			seed = (unsigned int)((now.tv_sec + 1) * (now.tv_usec + 1));
			srand(seed);
L
Linus Torvalds 已提交
496
			break;
497
		}
S
Sam Ravnborg 已提交
498
		case '?':
E
EGRY Gabor 已提交
499
			fprintf(stderr, _("See README for usage info\n"));
500
			exit(1);
S
Sam Ravnborg 已提交
501
			break;
L
Linus Torvalds 已提交
502 503
		}
	}
504
	if (ac == optind) {
505
		printf(_("%s: Kconfig file missing\n"), av[0]);
506
		exit(1);
L
Linus Torvalds 已提交
507
	}
508
	name = av[optind];
L
Linus Torvalds 已提交
509 510
	conf_parse(name);
	//zconfdump(stdout);
511
	if (sync_kconfig) {
512 513
		name = conf_get_configname();
		if (stat(name, &tmpstat)) {
514
			fprintf(stderr, _("***\n"
515
				"*** Configuration file \"%s\" not found!\n"
516 517 518
				"***\n"
				"*** Please run some configurator (e.g. \"make oldconfig\" or\n"
				"*** \"make menuconfig\" or \"make xconfig\").\n"
519
				"***\n"), name);
520 521 522 523
			exit(1);
		}
	}

L
Linus Torvalds 已提交
524
	switch (input_mode) {
S
Sam Ravnborg 已提交
525
	case defconfig:
L
Linus Torvalds 已提交
526 527 528
		if (!defconfig_file)
			defconfig_file = conf_get_default_confname();
		if (conf_read(defconfig_file)) {
E
EGRY Gabor 已提交
529
			printf(_("***\n"
L
Linus Torvalds 已提交
530
				"*** Can't find default configuration \"%s\"!\n"
E
EGRY Gabor 已提交
531
				"***\n"), defconfig_file);
L
Linus Torvalds 已提交
532 533 534
			exit(1);
		}
		break;
S
Sam Ravnborg 已提交
535
	case savedefconfig:
S
Sam Ravnborg 已提交
536 537 538
	case silentoldconfig:
	case oldaskconfig:
	case oldconfig:
539
	case listnewconfig:
540
	case oldnoconfig:
L
Linus Torvalds 已提交
541 542
		conf_read(NULL);
		break;
S
Sam Ravnborg 已提交
543 544 545
	case allnoconfig:
	case allyesconfig:
	case allmodconfig:
S
Sam Ravnborg 已提交
546
	case alldefconfig:
S
Sam Ravnborg 已提交
547
	case randconfig:
548 549
		name = getenv("KCONFIG_ALLCONFIG");
		if (name && !stat(name, &tmpstat)) {
550
			conf_read_simple(name, S_DEF_USER);
551 552 553
			break;
		}
		switch (input_mode) {
S
Sam Ravnborg 已提交
554 555 556
		case allnoconfig:	name = "allno.config"; break;
		case allyesconfig:	name = "allyes.config"; break;
		case allmodconfig:	name = "allmod.config"; break;
S
Sam Ravnborg 已提交
557
		case alldefconfig:	name = "alldef.config"; break;
S
Sam Ravnborg 已提交
558
		case randconfig:	name = "allrandom.config"; break;
559 560 561
		default: break;
		}
		if (!stat(name, &tmpstat))
562
			conf_read_simple(name, S_DEF_USER);
563
		else if (!stat("all.config", &tmpstat))
564
			conf_read_simple("all.config", S_DEF_USER);
565
		break;
L
Linus Torvalds 已提交
566 567 568
	default:
		break;
	}
569 570 571 572 573 574

	if (sync_kconfig) {
		if (conf_get_changed()) {
			name = getenv("KCONFIG_NOSILENTUPDATE");
			if (name && *name) {
				fprintf(stderr,
575
					_("\n*** The configuration requires explicit update.\n\n"));
576 577 578 579 580 581
				return 1;
			}
		}
		valid_stdin = isatty(0) && isatty(1) && isatty(2);
	}

582
	switch (input_mode) {
S
Sam Ravnborg 已提交
583
	case allnoconfig:
584 585
		conf_set_all_new_symbols(def_no);
		break;
S
Sam Ravnborg 已提交
586
	case allyesconfig:
587 588
		conf_set_all_new_symbols(def_yes);
		break;
S
Sam Ravnborg 已提交
589
	case allmodconfig:
590 591
		conf_set_all_new_symbols(def_mod);
		break;
S
Sam Ravnborg 已提交
592 593 594
	case alldefconfig:
		conf_set_all_new_symbols(def_default);
		break;
S
Sam Ravnborg 已提交
595
	case randconfig:
596 597
		conf_set_all_new_symbols(def_random);
		break;
S
Sam Ravnborg 已提交
598
	case defconfig:
599 600
		conf_set_all_new_symbols(def_default);
		break;
S
Sam Ravnborg 已提交
601 602
	case savedefconfig:
		break;
S
Sam Ravnborg 已提交
603
	case oldaskconfig:
604 605
		rootEntry = &rootmenu;
		conf(&rootmenu);
S
Sam Ravnborg 已提交
606
		input_mode = silentoldconfig;
607
		/* fall through */
S
Sam Ravnborg 已提交
608
	case oldconfig:
609
	case listnewconfig:
610
	case oldnoconfig:
S
Sam Ravnborg 已提交
611
	case silentoldconfig:
612 613 614 615
		/* Update until a loop caused no more changes */
		do {
			conf_cnt = 0;
			check_conf(&rootmenu);
616
		} while (conf_cnt &&
617
			 (input_mode != listnewconfig &&
618
			  input_mode != oldnoconfig));
619 620
		break;
	}
L
Linus Torvalds 已提交
621

622 623 624 625 626
	if (sync_kconfig) {
		/* silentoldconfig is used during the build so we shall update autoconf.
		 * All other commands are only used to generate a config.
		 */
		if (conf_get_changed() && conf_write(NULL)) {
627
			fprintf(stderr, _("\n*** Error during writing of the configuration.\n\n"));
628 629 630
			exit(1);
		}
		if (conf_write_autoconf()) {
631
			fprintf(stderr, _("\n*** Error during update of the configuration.\n\n"));
632 633
			return 1;
		}
S
Sam Ravnborg 已提交
634 635 636 637 638 639
	} else if (input_mode == savedefconfig) {
		if (conf_write_defconfig(defconfig_file)) {
			fprintf(stderr, _("n*** Error while saving defconfig to: %s\n\n"),
			        defconfig_file);
			return 1;
		}
640
	} else if (input_mode != listnewconfig) {
641
		if (conf_write(NULL)) {
642
			fprintf(stderr, _("\n*** Error during writing of the configuration.\n\n"));
643 644
			exit(1);
		}
R
Roman Zippel 已提交
645
	}
646
	return 0;
L
Linus Torvalds 已提交
647
}
648 649 650 651 652 653 654 655 656 657 658
/*
 * Helper function to facilitate fgets() by Jean Sacren.
 */
void xfgets(str, size, in)
	char *str;
	int size;
	FILE *in;
{
	if (fgets(str, size, in) == NULL)
		fprintf(stderr, "\nError in reading or end of file.\n");
}