conf.c 12.7 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,
33
	listnewconfig,
34
	oldnoconfig,
S
Sam Ravnborg 已提交
35 36
} input_mode = oldaskconfig;

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

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

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

	menu_get_ext_help(menu, &help);

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

J
J.A. Magallon 已提交
56
static void strip(char *str)
L
Linus Torvalds 已提交
57
{
J
J.A. Magallon 已提交
58
	char *p = str;
L
Linus Torvalds 已提交
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
	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)
{
75
	if (!valid_stdin) {
76 77 78
		printf(_("aborted!\n\n"));
		printf(_("Console input/output is redirected. "));
		printf(_("Run 'make oldconfig' to update configuration.\n\n"));
L
Linus Torvalds 已提交
79 80 81 82
		exit(1);
	}
}

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

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

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

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

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

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

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

	while (1) {
E
EGRY Gabor 已提交
135
		printf("%*s%s ", indent - 1, "", _(menu->prompt->text));
L
Linus Torvalds 已提交
136 137 138 139
		printf("(%s) ", sym->name);
		def = sym_get_string_value(sym);
		if (sym_get_string_value(sym))
			printf("[%s] ", def);
140 141
		if (!conf_askvalue(sym, def))
			return 0;
L
Linus Torvalds 已提交
142 143 144 145 146 147
		switch (line[0]) {
		case '\n':
			break;
		case '?':
			/* print help */
			if (line[1] == '\n') {
148
				print_help(menu);
L
Linus Torvalds 已提交
149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
				def = NULL;
				break;
			}
		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 已提交
167
		printf("%*s%s ", indent - 1, "", _(menu->prompt->text));
L
Linus Torvalds 已提交
168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188
		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");
189
		if (menu_has_help(menu))
L
Linus Torvalds 已提交
190 191
			printf("/?");
		printf("] ");
192 193
		if (!conf_askvalue(sym, sym_get_string_value(sym)))
			return 0;
L
Linus Torvalds 已提交
194 195 196 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
		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:
226
		print_help(menu);
L
Linus Torvalds 已提交
227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
	}
}

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 已提交
254
			printf("%*s%s\n", indent - 1, "", _(menu_get_prompt(menu)));
L
Linus Torvalds 已提交
255 256 257 258 259 260 261 262 263
			return 0;
		case yes:
			break;
		}
	}

	while (1) {
		int cnt, def;

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

	conf_childs:
		for (child = menu->list; child; child = child->next) {
			if (!child->sym || !menu_is_visible(child))
				continue;
			if (!--cnt)
				break;
		}
		if (!child)
			continue;
		if (line[strlen(line) - 1] == '?') {
335
			print_help(child);
L
Linus Torvalds 已提交
336 337 338
			continue;
		}
		sym_set_choice_value(sym, child->sym);
339
		for (child = child->list; child; child = child->next) {
L
Linus Torvalds 已提交
340
			indent += 2;
341
			conf(child);
L
Linus Torvalds 已提交
342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363
			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 已提交
364
			if ((input_mode == silentoldconfig ||
365
			     input_mode == listnewconfig ||
366
			     input_mode == oldnoconfig) &&
367
			    rootEntry != menu) {
L
Linus Torvalds 已提交
368 369 370 371 372 373 374 375
				check_conf(menu);
				return;
			}
		case P_COMMENT:
			prompt = menu_get_prompt(menu);
			if (prompt)
				printf("%*c\n%*c %s\n%*c\n",
					indent, '*',
E
EGRY Gabor 已提交
376
					indent, '*', _(prompt),
L
Linus Torvalds 已提交
377 378 379 380 381 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
					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;
422 423 424
	if (sym && !sym_has_value(sym)) {
		if (sym_is_changable(sym) ||
		    (sym_is_choice(sym) && sym_get_tristate_value(sym) == yes)) {
425 426 427
			if (input_mode == listnewconfig) {
				if (sym->name && !sym_is_choice_value(sym)) {
					printf("CONFIG_%s\n", sym->name);
428 429 430 431 432 433 434
				}
			} else {
				if (!conf_cnt++)
					printf(_("*\n* Restart config...\n*\n"));
				rootEntry = menu_get_parent_menu(menu);
				conf(rootEntry);
			}
L
Linus Torvalds 已提交
435 436 437 438 439 440 441
		}
	}

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

S
Sam Ravnborg 已提交
442 443 444 445 446 447 448 449
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},
	{"allnoconfig",     no_argument,       NULL, allnoconfig},
	{"allyesconfig",    no_argument,       NULL, allyesconfig},
	{"allmodconfig",    no_argument,       NULL, allmodconfig},
S
Sam Ravnborg 已提交
450
	{"alldefconfig",    no_argument,       NULL, alldefconfig},
S
Sam Ravnborg 已提交
451
	{"randconfig",      no_argument,       NULL, randconfig},
452
	{"listnewconfig",   no_argument,       NULL, listnewconfig},
453
	{"oldnoconfig",     no_argument,       NULL, oldnoconfig},
S
Sam Ravnborg 已提交
454 455 456
	{NULL, 0, NULL, 0}
};

L
Linus Torvalds 已提交
457 458
int main(int ac, char **av)
{
459
	int opt;
L
Linus Torvalds 已提交
460 461 462
	const char *name;
	struct stat tmpstat;

E
EGRY Gabor 已提交
463 464 465 466
	setlocale(LC_ALL, "");
	bindtextdomain(PACKAGE, LOCALEDIR);
	textdomain(PACKAGE);

S
Sam Ravnborg 已提交
467 468
	while ((opt = getopt_long_only(ac, av, "", long_opts, NULL)) != -1) {
		input_mode = (enum input_mode)opt;
469
		switch (opt) {
S
Sam Ravnborg 已提交
470
		case silentoldconfig:
471
			sync_kconfig = 1;
L
Linus Torvalds 已提交
472
			break;
S
Sam Ravnborg 已提交
473
		case defconfig:
474
			defconfig_file = optarg;
L
Linus Torvalds 已提交
475
			break;
S
Sam Ravnborg 已提交
476
		case randconfig:
477 478 479 480 481 482 483 484 485 486 487 488
		{
			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 已提交
489
			break;
490
		}
S
Sam Ravnborg 已提交
491
		case '?':
E
EGRY Gabor 已提交
492
			fprintf(stderr, _("See README for usage info\n"));
493
			exit(1);
S
Sam Ravnborg 已提交
494
			break;
L
Linus Torvalds 已提交
495 496
		}
	}
497
	if (ac == optind) {
498
		printf(_("%s: Kconfig file missing\n"), av[0]);
499
		exit(1);
L
Linus Torvalds 已提交
500
	}
501
	name = av[optind];
L
Linus Torvalds 已提交
502 503
	conf_parse(name);
	//zconfdump(stdout);
504
	if (sync_kconfig) {
505 506
		name = conf_get_configname();
		if (stat(name, &tmpstat)) {
507 508
			fprintf(stderr, _("***\n"
				"*** You have not yet configured your kernel!\n"
509
				"*** (missing kernel config file \"%s\")\n"
510 511 512
				"***\n"
				"*** Please run some configurator (e.g. \"make oldconfig\" or\n"
				"*** \"make menuconfig\" or \"make xconfig\").\n"
513
				"***\n"), name);
514 515 516 517
			exit(1);
		}
	}

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

	if (sync_kconfig) {
		if (conf_get_changed()) {
			name = getenv("KCONFIG_NOSILENTUPDATE");
			if (name && *name) {
				fprintf(stderr,
					_("\n*** Kernel configuration requires explicit update.\n\n"));
				return 1;
			}
		}
		valid_stdin = isatty(0) && isatty(1) && isatty(2);
	}

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

613 614 615 616 617 618 619 620 621 622 623 624
	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)) {
			fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
			exit(1);
		}
		if (conf_write_autoconf()) {
			fprintf(stderr, _("\n*** Error during update of the kernel configuration.\n\n"));
			return 1;
		}
625
	} else if (input_mode != listnewconfig) {
626 627 628 629
		if (conf_write(NULL)) {
			fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
			exit(1);
		}
R
Roman Zippel 已提交
630
	}
631
	return 0;
L
Linus Torvalds 已提交
632
}