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 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();
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 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
				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 已提交
168
		printf("%*s%s ", indent - 1, "", _(menu->prompt->text));
L
Linus Torvalds 已提交
169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
		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");
190
		if (menu_has_help(menu))
L
Linus Torvalds 已提交
191 192
			printf("/?");
		printf("] ");
193 194
		if (!conf_askvalue(sym, sym_get_string_value(sym)))
			return 0;
L
Linus Torvalds 已提交
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 226
		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:
227
		print_help(menu);
L
Linus Torvalds 已提交
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 254
	}
}

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

	while (1) {
		int cnt, def;

E
EGRY Gabor 已提交
265
		printf("%*s%s\n", indent - 1, "", _(menu_get_prompt(menu)));
L
Linus Torvalds 已提交
266 267
		def_sym = sym_get_choice_value(sym);
		cnt = def = 0;
268
		line[0] = 0;
L
Linus Torvalds 已提交
269 270 271 272
		for (child = menu->list; child; child = child->next) {
			if (!menu_is_visible(child))
				continue;
			if (!child->sym) {
E
EGRY Gabor 已提交
273
				printf("%*c %s\n", indent, '*', _(menu_get_prompt(child)));
L
Linus Torvalds 已提交
274 275 276 277 278 279 280 281
				continue;
			}
			cnt++;
			if (child->sym == def_sym) {
				def = cnt;
				printf("%*c", indent, '>');
			} else
				printf("%*c", indent, ' ');
E
EGRY Gabor 已提交
282
			printf(" %d. %s", cnt, _(menu_get_prompt(child)));
L
Linus Torvalds 已提交
283 284 285
			if (child->sym->name)
				printf(" (%s)", child->sym->name);
			if (!sym_has_value(child->sym))
E
EGRY Gabor 已提交
286
				printf(_(" (NEW)"));
L
Linus Torvalds 已提交
287 288
			printf("\n");
		}
E
EGRY Gabor 已提交
289
		printf(_("%*schoice"), indent - 1, "");
L
Linus Torvalds 已提交
290 291 292 293 294
		if (cnt == 1) {
			printf("[1]: 1\n");
			goto conf_childs;
		}
		printf("[1-%d", cnt);
295
		if (menu_has_help(menu))
L
Linus Torvalds 已提交
296 297 298
			printf("?");
		printf("]: ");
		switch (input_mode) {
S
Sam Ravnborg 已提交
299 300
		case oldconfig:
		case silentoldconfig:
L
Linus Torvalds 已提交
301 302 303 304 305 306
			if (!is_new) {
				cnt = def;
				printf("%d\n", cnt);
				break;
			}
			check_stdin();
S
Sam Ravnborg 已提交
307
		case oldaskconfig:
L
Linus Torvalds 已提交
308
			fflush(stdout);
309
			xfgets(line, 128, stdin);
L
Linus Torvalds 已提交
310 311
			strip(line);
			if (line[0] == '?') {
312
				print_help(menu);
L
Linus Torvalds 已提交
313 314 315 316 317 318 319 320 321
				continue;
			}
			if (!line[0])
				cnt = def;
			else if (isdigit(line[0]))
				cnt = atoi(line);
			else
				continue;
			break;
322 323
		default:
			break;
L
Linus Torvalds 已提交
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;
335
		if (line[0] && line[strlen(line) - 1] == '?') {
336
			print_help(child);
L
Linus Torvalds 已提交
337 338 339
			continue;
		}
		sym_set_choice_value(sym, child->sym);
340
		for (child = child->list; child; child = child->next) {
L
Linus Torvalds 已提交
341
			indent += 2;
342
			conf(child);
L
Linus Torvalds 已提交
343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364
			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 已提交
365
			if ((input_mode == silentoldconfig ||
366
			     input_mode == listnewconfig ||
367
			     input_mode == oldnoconfig) &&
368
			    rootEntry != menu) {
L
Linus Torvalds 已提交
369 370 371 372 373 374 375 376
				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 已提交
377
					indent, '*', _(prompt),
L
Linus Torvalds 已提交
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 422
					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;
423 424 425
	if (sym && !sym_has_value(sym)) {
		if (sym_is_changable(sym) ||
		    (sym_is_choice(sym) && sym_get_tristate_value(sym) == yes)) {
426 427
			if (input_mode == listnewconfig) {
				if (sym->name && !sym_is_choice_value(sym)) {
428
					printf("%s%s\n", CONFIG_, sym->name);
429
				}
430
			} else if (input_mode != oldnoconfig) {
431 432 433 434 435
				if (!conf_cnt++)
					printf(_("*\n* Restart config...\n*\n"));
				rootEntry = menu_get_parent_menu(menu);
				conf(rootEntry);
			}
L
Linus Torvalds 已提交
436 437 438 439 440 441 442
		}
	}

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

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

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

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

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

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

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

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

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