conf.c 13.2 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

#include "lkc.h"

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

S
Sam Ravnborg 已提交
22 23 24 25 26 27 28
enum input_mode {
	oldaskconfig,
	silentoldconfig,
	oldconfig,
	allnoconfig,
	allyesconfig,
	allmodconfig,
S
Sam Ravnborg 已提交
29
	alldefconfig,
S
Sam Ravnborg 已提交
30 31
	randconfig,
	defconfig,
S
Sam Ravnborg 已提交
32
	savedefconfig,
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();
108
		/* fall through */
S
Sam Ravnborg 已提交
109
	case oldaskconfig:
L
Linus Torvalds 已提交
110
		fflush(stdout);
111
		xfgets(line, 128, stdin);
112
		return 1;
L
Linus Torvalds 已提交
113 114 115 116 117 118 119 120 121
	default:
		break;
	}

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

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

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

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

	while (1) {
		int cnt, def;

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

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

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

S
Sam Ravnborg 已提交
446 447 448 449 450
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 已提交
451
	{"savedefconfig",   required_argument, NULL, savedefconfig},
S
Sam Ravnborg 已提交
452 453 454
	{"allnoconfig",     no_argument,       NULL, allnoconfig},
	{"allyesconfig",    no_argument,       NULL, allyesconfig},
	{"allmodconfig",    no_argument,       NULL, allmodconfig},
S
Sam Ravnborg 已提交
455
	{"alldefconfig",    no_argument,       NULL, alldefconfig},
S
Sam Ravnborg 已提交
456
	{"randconfig",      no_argument,       NULL, randconfig},
457
	{"listnewconfig",   no_argument,       NULL, listnewconfig},
458
	{"oldnoconfig",     no_argument,       NULL, oldnoconfig},
S
Sam Ravnborg 已提交
459 460 461
	{NULL, 0, NULL, 0}
};

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

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

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

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

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

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

621 622 623 624 625
	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)) {
626
			fprintf(stderr, _("\n*** Error during writing of the configuration.\n\n"));
627 628 629
			exit(1);
		}
		if (conf_write_autoconf()) {
630
			fprintf(stderr, _("\n*** Error during update of the configuration.\n\n"));
631 632
			return 1;
		}
S
Sam Ravnborg 已提交
633 634 635 636 637 638
	} 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;
		}
639
	} else if (input_mode != listnewconfig) {
640
		if (conf_write(NULL)) {
641
			fprintf(stderr, _("\n*** Error during writing of the configuration.\n\n"));
642 643
			exit(1);
		}
R
Roman Zippel 已提交
644
	}
645
	return 0;
L
Linus Torvalds 已提交
646
}
647 648 649 650 651 652 653 654 655 656 657
/*
 * 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");
}