Linux Audio

Check our new training course

Linux kernel drivers training

Mar 31-Apr 9, 2025, special US time zones
Register
Loading...
v5.14.15
   1// SPDX-License-Identifier: GPL-2.0
   2/*
   3 * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
   4 *
   5 * Introduced single menu mode (show all sub-menus in one large tree).
   6 * 2002-11-06 Petr Baudis <pasky@ucw.cz>
   7 *
   8 * i18n, 2005, Arnaldo Carvalho de Melo <acme@conectiva.com.br>
   9 */
  10
  11#include <ctype.h>
  12#include <errno.h>
  13#include <fcntl.h>
  14#include <limits.h>
  15#include <stdarg.h>
  16#include <stdlib.h>
  17#include <string.h>
  18#include <strings.h>
  19#include <signal.h>
  20#include <unistd.h>
  21
  22#include "lkc.h"
  23#include "lxdialog/dialog.h"
  24
  25#define JUMP_NB			9
  26
  27static const char mconf_readme[] =
  28"Overview\n"
  29"--------\n"
  30"This interface lets you select features and parameters for the build.\n"
  31"Features can either be built-in, modularized, or ignored. Parameters\n"
  32"must be entered in as decimal or hexadecimal numbers or text.\n"
  33"\n"
  34"Menu items beginning with following braces represent features that\n"
  35"  [ ] can be built in or removed\n"
  36"  < > can be built in, modularized or removed\n"
  37"  { } can be built in or modularized (selected by other feature)\n"
  38"  - - are selected by other feature,\n"
  39"while *, M or whitespace inside braces means to build in, build as\n"
  40"a module or to exclude the feature respectively.\n"
  41"\n"
  42"To change any of these features, highlight it with the cursor\n"
  43"keys and press <Y> to build it in, <M> to make it a module or\n"
  44"<N> to remove it.  You may also press the <Space Bar> to cycle\n"
  45"through the available options (i.e. Y->N->M->Y).\n"
  46"\n"
  47"Some additional keyboard hints:\n"
  48"\n"
  49"Menus\n"
  50"----------\n"
  51"o  Use the Up/Down arrow keys (cursor keys) to highlight the item you\n"
  52"   wish to change or the submenu you wish to select and press <Enter>.\n"
  53"   Submenus are designated by \"--->\", empty ones by \"----\".\n"
  54"\n"
  55"   Shortcut: Press the option's highlighted letter (hotkey).\n"
  56"             Pressing a hotkey more than once will sequence\n"
  57"             through all visible items which use that hotkey.\n"
  58"\n"
  59"   You may also use the <PAGE UP> and <PAGE DOWN> keys to scroll\n"
  60"   unseen options into view.\n"
  61"\n"
  62"o  To exit a menu use the cursor keys to highlight the <Exit> button\n"
  63"   and press <ENTER>.\n"
  64"\n"
  65"   Shortcut: Press <ESC><ESC> or <E> or <X> if there is no hotkey\n"
  66"             using those letters.  You may press a single <ESC>, but\n"
  67"             there is a delayed response which you may find annoying.\n"
  68"\n"
  69"   Also, the <TAB> and cursor keys will cycle between <Select>,\n"
  70"   <Exit>, <Help>, <Save>, and <Load>.\n"
  71"\n"
  72"o  To get help with an item, use the cursor keys to highlight <Help>\n"
  73"   and press <ENTER>.\n"
  74"\n"
  75"   Shortcut: Press <H> or <?>.\n"
  76"\n"
  77"o  To toggle the display of hidden options, press <Z>.\n"
  78"\n"
  79"\n"
  80"Radiolists  (Choice lists)\n"
  81"-----------\n"
  82"o  Use the cursor keys to select the option you wish to set and press\n"
  83"   <S> or the <SPACE BAR>.\n"
  84"\n"
  85"   Shortcut: Press the first letter of the option you wish to set then\n"
  86"             press <S> or <SPACE BAR>.\n"
  87"\n"
  88"o  To see available help for the item, use the cursor keys to highlight\n"
  89"   <Help> and Press <ENTER>.\n"
  90"\n"
  91"   Shortcut: Press <H> or <?>.\n"
  92"\n"
  93"   Also, the <TAB> and cursor keys will cycle between <Select> and\n"
  94"   <Help>\n"
  95"\n"
  96"\n"
  97"Data Entry\n"
  98"-----------\n"
  99"o  Enter the requested information and press <ENTER>\n"
 100"   If you are entering hexadecimal values, it is not necessary to\n"
 101"   add the '0x' prefix to the entry.\n"
 102"\n"
 103"o  For help, use the <TAB> or cursor keys to highlight the help option\n"
 104"   and press <ENTER>.  You can try <TAB><H> as well.\n"
 105"\n"
 106"\n"
 107"Text Box    (Help Window)\n"
 108"--------\n"
 109"o  Use the cursor keys to scroll up/down/left/right.  The VI editor\n"
 110"   keys h,j,k,l function here as do <u>, <d>, <SPACE BAR> and <B> for\n"
 111"   those who are familiar with less and lynx.\n"
 112"\n"
 113"o  Press <E>, <X>, <q>, <Enter> or <Esc><Esc> to exit.\n"
 114"\n"
 115"\n"
 116"Alternate Configuration Files\n"
 117"-----------------------------\n"
 118"Menuconfig supports the use of alternate configuration files for\n"
 119"those who, for various reasons, find it necessary to switch\n"
 120"between different configurations.\n"
 121"\n"
 122"The <Save> button will let you save the current configuration to\n"
 123"a file of your choosing.  Use the <Load> button to load a previously\n"
 124"saved alternate configuration.\n"
 125"\n"
 126"Even if you don't use alternate configuration files, but you find\n"
 127"during a Menuconfig session that you have completely messed up your\n"
 128"settings, you may use the <Load> button to restore your previously\n"
 129"saved settings from \".config\" without restarting Menuconfig.\n"
 130"\n"
 131"Other information\n"
 132"-----------------\n"
 133"If you use Menuconfig in an XTERM window, make sure you have your\n"
 134"$TERM variable set to point to an xterm definition which supports\n"
 135"color.  Otherwise, Menuconfig will look rather bad.  Menuconfig will\n"
 136"not display correctly in an RXVT window because rxvt displays only one\n"
 137"intensity of color, bright.\n"
 138"\n"
 139"Menuconfig will display larger menus on screens or xterms which are\n"
 140"set to display more than the standard 25 row by 80 column geometry.\n"
 141"In order for this to work, the \"stty size\" command must be able to\n"
 142"display the screen's current row and column geometry.  I STRONGLY\n"
 143"RECOMMEND that you make sure you do NOT have the shell variables\n"
 144"LINES and COLUMNS exported into your environment.  Some distributions\n"
 145"export those variables via /etc/profile.  Some ncurses programs can\n"
 146"become confused when those variables (LINES & COLUMNS) don't reflect\n"
 147"the true screen size.\n"
 148"\n"
 149"Optional personality available\n"
 150"------------------------------\n"
 151"If you prefer to have all of the options listed in a single menu,\n"
 152"rather than the default multimenu hierarchy, run the menuconfig with\n"
 153"MENUCONFIG_MODE environment variable set to single_menu. Example:\n"
 154"\n"
 155"make MENUCONFIG_MODE=single_menu menuconfig\n"
 156"\n"
 157"<Enter> will then unroll the appropriate category, or enfold it if it\n"
 158"is already unrolled.\n"
 159"\n"
 160"Note that this mode can eventually be a little more CPU expensive\n"
 161"(especially with a larger number of unrolled categories) than the\n"
 162"default mode.\n"
 163"\n"
 164"Different color themes available\n"
 165"--------------------------------\n"
 166"It is possible to select different color themes using the variable\n"
 167"MENUCONFIG_COLOR. To select a theme use:\n"
 168"\n"
 169"make MENUCONFIG_COLOR=<theme> menuconfig\n"
 170"\n"
 171"Available themes are\n"
 172" mono       => selects colors suitable for monochrome displays\n"
 173" blackbg    => selects a color scheme with black background\n"
 174" classic    => theme with blue background. The classic look\n"
 175" bluetitle  => an LCD friendly version of classic. (default)\n"
 176"\n",
 177menu_instructions[] =
 178	"Arrow keys navigate the menu.  "
 179	"<Enter> selects submenus ---> (or empty submenus ----).  "
 180	"Highlighted letters are hotkeys.  "
 181	"Pressing <Y> includes, <N> excludes, <M> modularizes features.  "
 182	"Press <Esc><Esc> to exit, <?> for Help, </> for Search.  "
 183	"Legend: [*] built-in  [ ] excluded  <M> module  < > module capable",
 184radiolist_instructions[] =
 185	"Use the arrow keys to navigate this window or "
 186	"press the hotkey of the item you wish to select "
 187	"followed by the <SPACE BAR>. "
 188	"Press <?> for additional information about this option.",
 189inputbox_instructions_int[] =
 190	"Please enter a decimal value. "
 191	"Fractions will not be accepted.  "
 192	"Use the <TAB> key to move from the input field to the buttons below it.",
 193inputbox_instructions_hex[] =
 194	"Please enter a hexadecimal value. "
 195	"Use the <TAB> key to move from the input field to the buttons below it.",
 196inputbox_instructions_string[] =
 197	"Please enter a string value. "
 198	"Use the <TAB> key to move from the input field to the buttons below it.",
 199setmod_text[] =
 200	"This feature depends on another which has been configured as a module.\n"
 201	"As a result, this feature will be built as a module.",
 202load_config_text[] =
 203	"Enter the name of the configuration file you wish to load.  "
 204	"Accept the name shown to restore the configuration you "
 205	"last retrieved.  Leave blank to abort.",
 206load_config_help[] =
 207	"\n"
 208	"For various reasons, one may wish to keep several different\n"
 209	"configurations available on a single machine.\n"
 210	"\n"
 211	"If you have saved a previous configuration in a file other than the\n"
 212	"default one, entering its name here will allow you to modify that\n"
 213	"configuration.\n"
 214	"\n"
 215	"If you are uncertain, then you have probably never used alternate\n"
 216	"configuration files. You should therefore leave this blank to abort.\n",
 217save_config_text[] =
 218	"Enter a filename to which this configuration should be saved "
 219	"as an alternate.  Leave blank to abort.",
 220save_config_help[] =
 221	"\n"
 222	"For various reasons, one may wish to keep different configurations\n"
 223	"available on a single machine.\n"
 224	"\n"
 225	"Entering a file name here will allow you to later retrieve, modify\n"
 226	"and use the current configuration as an alternate to whatever\n"
 227	"configuration options you have selected at that time.\n"
 228	"\n"
 229	"If you are uncertain what all this means then you should probably\n"
 230	"leave this blank.\n",
 231search_help[] =
 232	"\n"
 233	"Search for symbols and display their relations.\n"
 234	"Regular expressions are allowed.\n"
 235	"Example: search for \"^FOO\"\n"
 236	"Result:\n"
 237	"-----------------------------------------------------------------\n"
 238	"Symbol: FOO [=m]\n"
 239	"Type  : tristate\n"
 240	"Prompt: Foo bus is used to drive the bar HW\n"
 241	"  Location:\n"
 242	"    -> Bus options (PCI, PCMCIA, EISA, ISA)\n"
 243	"      -> PCI support (PCI [=y])\n"
 244	"(1)     -> PCI access mode (<choice> [=y])\n"
 245	"  Defined at drivers/pci/Kconfig:47\n"
 246	"  Depends on: X86_LOCAL_APIC && X86_IO_APIC || IA64\n"
 247	"  Selects: LIBCRC32\n"
 248	"  Selected by: BAR [=n]\n"
 249	"-----------------------------------------------------------------\n"
 250	"o The line 'Type:' shows the type of the configuration option for\n"
 251	"  this symbol (bool, tristate, string, ...)\n"
 252	"o The line 'Prompt:' shows the text used in the menu structure for\n"
 253	"  this symbol\n"
 254	"o The 'Defined at' line tells at what file / line number the symbol\n"
 255	"  is defined\n"
 256	"o The 'Depends on:' line tells what symbols need to be defined for\n"
 257	"  this symbol to be visible in the menu (selectable)\n"
 258	"o The 'Location:' lines tells where in the menu structure this symbol\n"
 259	"  is located\n"
 260	"    A location followed by a [=y] indicates that this is a\n"
 261	"    selectable menu item - and the current value is displayed inside\n"
 262	"    brackets.\n"
 263	"    Press the key in the (#) prefix to jump directly to that\n"
 264	"    location. You will be returned to the current search results\n"
 265	"    after exiting this new menu.\n"
 266	"o The 'Selects:' line tells what symbols will be automatically\n"
 267	"  selected if this symbol is selected (y or m)\n"
 268	"o The 'Selected by' line tells what symbol has selected this symbol\n"
 269	"\n"
 270	"Only relevant lines are shown.\n"
 271	"\n\n"
 272	"Search examples:\n"
 273	"Examples: USB	=> find all symbols containing USB\n"
 274	"          ^USB => find all symbols starting with USB\n"
 275	"          USB$ => find all symbols ending with USB\n"
 276	"\n";
 277
 278static int indent;
 279static struct menu *current_menu;
 280static int child_count;
 281static int single_menu_mode;
 282static int show_all_options;
 283static int save_and_exit;
 284static int silent;
 285
 286static void conf(struct menu *menu, struct menu *active_menu);
 287static void conf_choice(struct menu *menu);
 288static void conf_string(struct menu *menu);
 289static void conf_load(void);
 290static void conf_save(void);
 291static int show_textbox_ext(const char *title, char *text, int r, int c,
 292			    int *keys, int *vscroll, int *hscroll,
 293			    update_text_fn update_text, void *data);
 294static void show_textbox(const char *title, const char *text, int r, int c);
 295static void show_helptext(const char *title, const char *text);
 296static void show_help(struct menu *menu);
 297
 298static char filename[PATH_MAX+1];
 299static void set_config_filename(const char *config_filename)
 300{
 301	static char menu_backtitle[PATH_MAX+128];
 
 302
 303	snprintf(menu_backtitle, sizeof(menu_backtitle), "%s - %s",
 304		 config_filename, rootmenu.prompt->text);
 
 
 305	set_dialog_backtitle(menu_backtitle);
 306
 307	snprintf(filename, sizeof(filename), "%s", config_filename);
 
 
 308}
 309
 310struct subtitle_part {
 311	struct list_head entries;
 312	const char *text;
 313};
 314static LIST_HEAD(trail);
 315
 316static struct subtitle_list *subtitles;
 317static void set_subtitle(void)
 318{
 319	struct subtitle_part *sp;
 320	struct subtitle_list *pos, *tmp;
 321
 322	for (pos = subtitles; pos != NULL; pos = tmp) {
 323		tmp = pos->next;
 324		free(pos);
 325	}
 326
 327	subtitles = NULL;
 328	list_for_each_entry(sp, &trail, entries) {
 329		if (sp->text) {
 330			if (pos) {
 331				pos->next = xcalloc(1, sizeof(*pos));
 332				pos = pos->next;
 333			} else {
 334				subtitles = pos = xcalloc(1, sizeof(*pos));
 335			}
 336			pos->text = sp->text;
 337		}
 338	}
 339
 340	set_dialog_subtitles(subtitles);
 341}
 342
 343static void reset_subtitle(void)
 344{
 345	struct subtitle_list *pos, *tmp;
 346
 347	for (pos = subtitles; pos != NULL; pos = tmp) {
 348		tmp = pos->next;
 349		free(pos);
 350	}
 351	subtitles = NULL;
 352	set_dialog_subtitles(subtitles);
 353}
 354
 355struct search_data {
 356	struct list_head *head;
 357	struct menu **targets;
 358	int *keys;
 359};
 360
 361static void update_text(char *buf, size_t start, size_t end, void *_data)
 362{
 363	struct search_data *data = _data;
 364	struct jump_key *pos;
 365	int k = 0;
 366
 367	list_for_each_entry(pos, data->head, entries) {
 368		if (pos->offset >= start && pos->offset < end) {
 369			char header[4];
 370
 371			if (k < JUMP_NB) {
 372				int key = '0' + (pos->index % JUMP_NB) + 1;
 373
 374				sprintf(header, "(%c)", key);
 375				data->keys[k] = key;
 376				data->targets[k] = pos->target;
 377				k++;
 378			} else {
 379				sprintf(header, "   ");
 380			}
 381
 382			memcpy(buf + pos->offset, header, sizeof(header) - 1);
 383		}
 384	}
 385	data->keys[k] = 0;
 386}
 387
 388static void search_conf(void)
 389{
 390	struct symbol **sym_arr;
 391	struct gstr res;
 392	struct gstr title;
 393	char *dialog_input;
 394	int dres, vscroll = 0, hscroll = 0;
 395	bool again;
 396	struct gstr sttext;
 397	struct subtitle_part stpart;
 398
 399	title = str_new();
 400	str_printf( &title, "Enter (sub)string or regexp to search for "
 401			      "(with or without \"%s\")", CONFIG_);
 402
 403again:
 404	dialog_clear();
 405	dres = dialog_inputbox("Search Configuration Parameter",
 406			      str_get(&title),
 407			      10, 75, "");
 408	switch (dres) {
 409	case 0:
 410		break;
 411	case 1:
 412		show_helptext("Search Configuration", search_help);
 413		goto again;
 414	default:
 415		str_free(&title);
 416		return;
 417	}
 418
 419	/* strip the prefix if necessary */
 420	dialog_input = dialog_input_result;
 421	if (strncasecmp(dialog_input_result, CONFIG_, strlen(CONFIG_)) == 0)
 422		dialog_input += strlen(CONFIG_);
 423
 424	sttext = str_new();
 425	str_printf(&sttext, "Search (%s)", dialog_input_result);
 426	stpart.text = str_get(&sttext);
 427	list_add_tail(&stpart.entries, &trail);
 428
 429	sym_arr = sym_re_search(dialog_input);
 430	do {
 431		LIST_HEAD(head);
 432		struct menu *targets[JUMP_NB];
 433		int keys[JUMP_NB + 1], i;
 434		struct search_data data = {
 435			.head = &head,
 436			.targets = targets,
 437			.keys = keys,
 438		};
 439		struct jump_key *pos, *tmp;
 440
 441		res = get_relations_str(sym_arr, &head);
 442		set_subtitle();
 443		dres = show_textbox_ext("Search Results", (char *)
 444					str_get(&res), 0, 0, keys, &vscroll,
 445					&hscroll, &update_text, (void *)
 446					&data);
 447		again = false;
 448		for (i = 0; i < JUMP_NB && keys[i]; i++)
 449			if (dres == keys[i]) {
 450				conf(targets[i]->parent, targets[i]);
 451				again = true;
 452			}
 453		str_free(&res);
 454		list_for_each_entry_safe(pos, tmp, &head, entries)
 455			free(pos);
 456	} while (again);
 457	free(sym_arr);
 458	str_free(&title);
 459	list_del(trail.prev);
 460	str_free(&sttext);
 461}
 462
 463static void build_conf(struct menu *menu)
 464{
 465	struct symbol *sym;
 466	struct property *prop;
 467	struct menu *child;
 468	int type, tmp, doint = 2;
 469	tristate val;
 470	char ch;
 471	bool visible;
 472
 473	/*
 474	 * note: menu_is_visible() has side effect that it will
 475	 * recalc the value of the symbol.
 476	 */
 477	visible = menu_is_visible(menu);
 478	if (show_all_options && !menu_has_prompt(menu))
 479		return;
 480	else if (!show_all_options && !visible)
 481		return;
 482
 483	sym = menu->sym;
 484	prop = menu->prompt;
 485	if (!sym) {
 486		if (prop && menu != current_menu) {
 487			const char *prompt = menu_get_prompt(menu);
 488			switch (prop->type) {
 489			case P_MENU:
 490				child_count++;
 491				if (single_menu_mode) {
 492					item_make("%s%*c%s",
 493						  menu->data ? "-->" : "++>",
 494						  indent + 1, ' ', prompt);
 495				} else
 496					item_make("   %*c%s  %s",
 497						  indent + 1, ' ', prompt,
 498						  menu_is_empty(menu) ? "----" : "--->");
 499				item_set_tag('m');
 500				item_set_data(menu);
 501				if (single_menu_mode && menu->data)
 502					goto conf_childs;
 503				return;
 504			case P_COMMENT:
 505				if (prompt) {
 506					child_count++;
 507					item_make("   %*c*** %s ***", indent + 1, ' ', prompt);
 508					item_set_tag(':');
 509					item_set_data(menu);
 510				}
 511				break;
 512			default:
 513				if (prompt) {
 514					child_count++;
 515					item_make("---%*c%s", indent + 1, ' ', prompt);
 516					item_set_tag(':');
 517					item_set_data(menu);
 518				}
 519			}
 520		} else
 521			doint = 0;
 522		goto conf_childs;
 523	}
 524
 525	type = sym_get_type(sym);
 526	if (sym_is_choice(sym)) {
 527		struct symbol *def_sym = sym_get_choice_value(sym);
 528		struct menu *def_menu = NULL;
 529
 530		child_count++;
 531		for (child = menu->list; child; child = child->next) {
 532			if (menu_is_visible(child) && child->sym == def_sym)
 533				def_menu = child;
 534		}
 535
 536		val = sym_get_tristate_value(sym);
 537		if (sym_is_changeable(sym)) {
 538			switch (type) {
 539			case S_BOOLEAN:
 540				item_make("[%c]", val == no ? ' ' : '*');
 541				break;
 542			case S_TRISTATE:
 543				switch (val) {
 544				case yes: ch = '*'; break;
 545				case mod: ch = 'M'; break;
 546				default:  ch = ' '; break;
 547				}
 548				item_make("<%c>", ch);
 549				break;
 550			}
 551			item_set_tag('t');
 552			item_set_data(menu);
 553		} else {
 554			item_make("   ");
 555			item_set_tag(def_menu ? 't' : ':');
 556			item_set_data(menu);
 557		}
 558
 559		item_add_str("%*c%s", indent + 1, ' ', menu_get_prompt(menu));
 560		if (val == yes) {
 561			if (def_menu) {
 562				item_add_str(" (%s)", menu_get_prompt(def_menu));
 563				item_add_str("  --->");
 564				if (def_menu->list) {
 565					indent += 2;
 566					build_conf(def_menu);
 567					indent -= 2;
 568				}
 569			}
 570			return;
 571		}
 572	} else {
 573		if (menu == current_menu) {
 574			item_make("---%*c%s", indent + 1, ' ', menu_get_prompt(menu));
 575			item_set_tag(':');
 576			item_set_data(menu);
 577			goto conf_childs;
 578		}
 579		child_count++;
 580		val = sym_get_tristate_value(sym);
 581		if (sym_is_choice_value(sym) && val == yes) {
 582			item_make("   ");
 583			item_set_tag(':');
 584			item_set_data(menu);
 585		} else {
 586			switch (type) {
 587			case S_BOOLEAN:
 588				if (sym_is_changeable(sym))
 589					item_make("[%c]", val == no ? ' ' : '*');
 590				else
 591					item_make("-%c-", val == no ? ' ' : '*');
 592				item_set_tag('t');
 593				item_set_data(menu);
 594				break;
 595			case S_TRISTATE:
 596				switch (val) {
 597				case yes: ch = '*'; break;
 598				case mod: ch = 'M'; break;
 599				default:  ch = ' '; break;
 600				}
 601				if (sym_is_changeable(sym)) {
 602					if (sym->rev_dep.tri == mod)
 603						item_make("{%c}", ch);
 604					else
 605						item_make("<%c>", ch);
 606				} else
 607					item_make("-%c-", ch);
 608				item_set_tag('t');
 609				item_set_data(menu);
 610				break;
 611			default:
 612				tmp = 2 + strlen(sym_get_string_value(sym)); /* () = 2 */
 613				item_make("(%s)", sym_get_string_value(sym));
 614				tmp = indent - tmp + 4;
 615				if (tmp < 0)
 616					tmp = 0;
 617				item_add_str("%*c%s%s", tmp, ' ', menu_get_prompt(menu),
 618					     (sym_has_value(sym) || !sym_is_changeable(sym)) ?
 619					     "" : " (NEW)");
 620				item_set_tag('s');
 621				item_set_data(menu);
 622				goto conf_childs;
 623			}
 624		}
 625		item_add_str("%*c%s%s", indent + 1, ' ', menu_get_prompt(menu),
 626			  (sym_has_value(sym) || !sym_is_changeable(sym)) ?
 627			  "" : " (NEW)");
 628		if (menu->prompt->type == P_MENU) {
 629			item_add_str("  %s", menu_is_empty(menu) ? "----" : "--->");
 630			return;
 631		}
 632	}
 633
 634conf_childs:
 635	indent += doint;
 636	for (child = menu->list; child; child = child->next)
 637		build_conf(child);
 638	indent -= doint;
 639}
 640
 641static void conf(struct menu *menu, struct menu *active_menu)
 642{
 643	struct menu *submenu;
 644	const char *prompt = menu_get_prompt(menu);
 645	struct subtitle_part stpart;
 646	struct symbol *sym;
 647	int res;
 648	int s_scroll = 0;
 649
 650	if (menu != &rootmenu)
 651		stpart.text = menu_get_prompt(menu);
 652	else
 653		stpart.text = NULL;
 654	list_add_tail(&stpart.entries, &trail);
 655
 656	while (1) {
 657		item_reset();
 658		current_menu = menu;
 659		build_conf(menu);
 660		if (!child_count)
 661			break;
 662		set_subtitle();
 663		dialog_clear();
 664		res = dialog_menu(prompt ? prompt : "Main Menu",
 665				  menu_instructions,
 666				  active_menu, &s_scroll);
 667		if (res == 1 || res == KEY_ESC || res == -ERRDISPLAYTOOSMALL)
 668			break;
 669		if (item_count() != 0) {
 670			if (!item_activate_selected())
 671				continue;
 672			if (!item_tag())
 673				continue;
 674		}
 675		submenu = item_data();
 676		active_menu = item_data();
 677		if (submenu)
 678			sym = submenu->sym;
 679		else
 680			sym = NULL;
 681
 682		switch (res) {
 683		case 0:
 684			switch (item_tag()) {
 685			case 'm':
 686				if (single_menu_mode)
 687					submenu->data = (void *) (long) !submenu->data;
 688				else
 689					conf(submenu, NULL);
 690				break;
 691			case 't':
 692				if (sym_is_choice(sym) && sym_get_tristate_value(sym) == yes)
 693					conf_choice(submenu);
 694				else if (submenu->prompt->type == P_MENU)
 695					conf(submenu, NULL);
 696				break;
 697			case 's':
 698				conf_string(submenu);
 699				break;
 700			}
 701			break;
 702		case 2:
 703			if (sym)
 704				show_help(submenu);
 705			else {
 706				reset_subtitle();
 707				show_helptext("README", mconf_readme);
 708			}
 709			break;
 710		case 3:
 711			reset_subtitle();
 712			conf_save();
 713			break;
 714		case 4:
 715			reset_subtitle();
 716			conf_load();
 717			break;
 718		case 5:
 719			if (item_is_tag('t')) {
 720				if (sym_set_tristate_value(sym, yes))
 721					break;
 722				if (sym_set_tristate_value(sym, mod))
 723					show_textbox(NULL, setmod_text, 6, 74);
 724			}
 725			break;
 726		case 6:
 727			if (item_is_tag('t'))
 728				sym_set_tristate_value(sym, no);
 729			break;
 730		case 7:
 731			if (item_is_tag('t'))
 732				sym_set_tristate_value(sym, mod);
 733			break;
 734		case 8:
 735			if (item_is_tag('t'))
 736				sym_toggle_tristate_value(sym);
 737			else if (item_is_tag('m'))
 738				conf(submenu, NULL);
 739			break;
 740		case 9:
 741			search_conf();
 742			break;
 743		case 10:
 744			show_all_options = !show_all_options;
 745			break;
 746		}
 747	}
 748
 749	list_del(trail.prev);
 750}
 751
 752static int show_textbox_ext(const char *title, char *text, int r, int c, int
 753			    *keys, int *vscroll, int *hscroll, update_text_fn
 754			    update_text, void *data)
 755{
 756	dialog_clear();
 757	return dialog_textbox(title, text, r, c, keys, vscroll, hscroll,
 758			      update_text, data);
 759}
 760
 761static void show_textbox(const char *title, const char *text, int r, int c)
 762{
 763	show_textbox_ext(title, (char *) text, r, c, (int []) {0}, NULL, NULL,
 764			 NULL, NULL);
 765}
 766
 767static void show_helptext(const char *title, const char *text)
 768{
 769	show_textbox(title, text, 0, 0);
 770}
 771
 772static void conf_message_callback(const char *s)
 773{
 774	if (save_and_exit) {
 775		if (!silent)
 776			printf("%s", s);
 777	} else {
 778		show_textbox(NULL, s, 6, 60);
 779	}
 780}
 781
 782static void show_help(struct menu *menu)
 783{
 784	struct gstr help = str_new();
 785
 786	help.max_width = getmaxx(stdscr) - 10;
 787	menu_get_ext_help(menu, &help);
 788
 789	show_helptext(menu_get_prompt(menu), str_get(&help));
 790	str_free(&help);
 791}
 792
 793static void conf_choice(struct menu *menu)
 794{
 795	const char *prompt = menu_get_prompt(menu);
 796	struct menu *child;
 797	struct symbol *active;
 798
 799	active = sym_get_choice_value(menu->sym);
 800	while (1) {
 801		int res;
 802		int selected;
 803		item_reset();
 804
 805		current_menu = menu;
 806		for (child = menu->list; child; child = child->next) {
 807			if (!menu_is_visible(child))
 808				continue;
 809			if (child->sym)
 810				item_make("%s", menu_get_prompt(child));
 811			else {
 812				item_make("*** %s ***", menu_get_prompt(child));
 813				item_set_tag(':');
 814			}
 815			item_set_data(child);
 816			if (child->sym == active)
 817				item_set_selected(1);
 818			if (child->sym == sym_get_choice_value(menu->sym))
 819				item_set_tag('X');
 820		}
 821		dialog_clear();
 822		res = dialog_checklist(prompt ? prompt : "Main Menu",
 823					radiolist_instructions,
 824					MENUBOX_HEIGTH_MIN,
 825					MENUBOX_WIDTH_MIN,
 826					CHECKLIST_HEIGTH_MIN);
 827		selected = item_activate_selected();
 828		switch (res) {
 829		case 0:
 830			if (selected) {
 831				child = item_data();
 832				if (!child->sym)
 833					break;
 834
 835				sym_set_tristate_value(child->sym, yes);
 836			}
 837			return;
 838		case 1:
 839			if (selected) {
 840				child = item_data();
 841				show_help(child);
 842				active = child->sym;
 843			} else
 844				show_help(menu);
 845			break;
 846		case KEY_ESC:
 847			return;
 848		case -ERRDISPLAYTOOSMALL:
 849			return;
 850		}
 851	}
 852}
 853
 854static void conf_string(struct menu *menu)
 855{
 856	const char *prompt = menu_get_prompt(menu);
 857
 858	while (1) {
 859		int res;
 860		const char *heading;
 861
 862		switch (sym_get_type(menu->sym)) {
 863		case S_INT:
 864			heading = inputbox_instructions_int;
 865			break;
 866		case S_HEX:
 867			heading = inputbox_instructions_hex;
 868			break;
 869		case S_STRING:
 870			heading = inputbox_instructions_string;
 871			break;
 872		default:
 873			heading = "Internal mconf error!";
 874		}
 875		dialog_clear();
 876		res = dialog_inputbox(prompt ? prompt : "Main Menu",
 877				      heading, 10, 75,
 878				      sym_get_string_value(menu->sym));
 879		switch (res) {
 880		case 0:
 881			if (sym_set_string_value(menu->sym, dialog_input_result))
 882				return;
 883			show_textbox(NULL, "You have made an invalid entry.", 5, 43);
 884			break;
 885		case 1:
 886			show_help(menu);
 887			break;
 888		case KEY_ESC:
 889			return;
 890		}
 891	}
 892}
 893
 894static void conf_load(void)
 895{
 896
 897	while (1) {
 898		int res;
 899		dialog_clear();
 900		res = dialog_inputbox(NULL, load_config_text,
 901				      11, 55, filename);
 902		switch(res) {
 903		case 0:
 904			if (!dialog_input_result[0])
 905				return;
 906			if (!conf_read(dialog_input_result)) {
 907				set_config_filename(dialog_input_result);
 908				conf_set_changed(true);
 909				return;
 910			}
 911			show_textbox(NULL, "File does not exist!", 5, 38);
 912			break;
 913		case 1:
 914			show_helptext("Load Alternate Configuration", load_config_help);
 915			break;
 916		case KEY_ESC:
 917			return;
 918		}
 919	}
 920}
 921
 922static void conf_save(void)
 923{
 924	while (1) {
 925		int res;
 926		dialog_clear();
 927		res = dialog_inputbox(NULL, save_config_text,
 928				      11, 55, filename);
 929		switch(res) {
 930		case 0:
 931			if (!dialog_input_result[0])
 932				return;
 933			if (!conf_write(dialog_input_result)) {
 934				set_config_filename(dialog_input_result);
 935				return;
 936			}
 937			show_textbox(NULL, "Can't create file!", 5, 60);
 938			break;
 939		case 1:
 940			show_helptext("Save Alternate Configuration", save_config_help);
 941			break;
 942		case KEY_ESC:
 943			return;
 944		}
 945	}
 946}
 947
 948static int handle_exit(void)
 949{
 950	int res;
 951
 952	save_and_exit = 1;
 953	reset_subtitle();
 954	dialog_clear();
 955	if (conf_get_changed())
 956		res = dialog_yesno(NULL,
 957				   "Do you wish to save your new configuration?\n"
 958				     "(Press <ESC><ESC> to continue kernel configuration.)",
 959				   6, 60);
 960	else
 961		res = -1;
 962
 963	end_dialog(saved_x, saved_y);
 964
 965	switch (res) {
 966	case 0:
 967		if (conf_write(filename)) {
 968			fprintf(stderr, "\n\n"
 969					  "Error while writing of the configuration.\n"
 970					  "Your configuration changes were NOT saved."
 971					  "\n\n");
 972			return 1;
 973		}
 974		conf_write_autoconf(0);
 975		/* fall through */
 976	case -1:
 977		if (!silent)
 978			printf("\n\n"
 979				 "*** End of the configuration.\n"
 980				 "*** Execute 'make' to start the build or try 'make help'."
 981				 "\n\n");
 982		res = 0;
 983		break;
 984	default:
 985		if (!silent)
 986			fprintf(stderr, "\n\n"
 987					  "Your configuration changes were NOT saved."
 988					  "\n\n");
 989		if (res != KEY_ESC)
 990			res = 0;
 991	}
 992
 993	return res;
 994}
 995
 996static void sig_handler(int signo)
 997{
 998	exit(handle_exit());
 999}
1000
1001int main(int ac, char **av)
1002{
1003	char *mode;
1004	int res;
1005
1006	signal(SIGINT, sig_handler);
1007
1008	if (ac > 1 && strcmp(av[1], "-s") == 0) {
1009		silent = 1;
1010		/* Silence conf_read() until the real callback is set up */
1011		conf_set_message_callback(NULL);
1012		av++;
1013	}
1014	conf_parse(av[1]);
1015	conf_read(NULL);
1016
1017	mode = getenv("MENUCONFIG_MODE");
1018	if (mode) {
1019		if (!strcasecmp(mode, "single_menu"))
1020			single_menu_mode = 1;
1021	}
1022
1023	if (init_dialog(NULL)) {
1024		fprintf(stderr, "Your display is too small to run Menuconfig!\n");
1025		fprintf(stderr, "It must be at least 19 lines by 80 columns.\n");
1026		return 1;
1027	}
1028
1029	set_config_filename(conf_get_configname());
1030	conf_set_message_callback(conf_message_callback);
1031	do {
1032		conf(&rootmenu, NULL);
1033		res = handle_exit();
1034	} while (res == KEY_ESC);
1035
1036	return res;
1037}
v5.4
   1// SPDX-License-Identifier: GPL-2.0
   2/*
   3 * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
   4 *
   5 * Introduced single menu mode (show all sub-menus in one large tree).
   6 * 2002-11-06 Petr Baudis <pasky@ucw.cz>
   7 *
   8 * i18n, 2005, Arnaldo Carvalho de Melo <acme@conectiva.com.br>
   9 */
  10
  11#include <ctype.h>
  12#include <errno.h>
  13#include <fcntl.h>
  14#include <limits.h>
  15#include <stdarg.h>
  16#include <stdlib.h>
  17#include <string.h>
 
  18#include <signal.h>
  19#include <unistd.h>
  20
  21#include "lkc.h"
  22#include "lxdialog/dialog.h"
  23
 
 
  24static const char mconf_readme[] =
  25"Overview\n"
  26"--------\n"
  27"This interface lets you select features and parameters for the build.\n"
  28"Features can either be built-in, modularized, or ignored. Parameters\n"
  29"must be entered in as decimal or hexadecimal numbers or text.\n"
  30"\n"
  31"Menu items beginning with following braces represent features that\n"
  32"  [ ] can be built in or removed\n"
  33"  < > can be built in, modularized or removed\n"
  34"  { } can be built in or modularized (selected by other feature)\n"
  35"  - - are selected by other feature,\n"
  36"while *, M or whitespace inside braces means to build in, build as\n"
  37"a module or to exclude the feature respectively.\n"
  38"\n"
  39"To change any of these features, highlight it with the cursor\n"
  40"keys and press <Y> to build it in, <M> to make it a module or\n"
  41"<N> to remove it.  You may also press the <Space Bar> to cycle\n"
  42"through the available options (i.e. Y->N->M->Y).\n"
  43"\n"
  44"Some additional keyboard hints:\n"
  45"\n"
  46"Menus\n"
  47"----------\n"
  48"o  Use the Up/Down arrow keys (cursor keys) to highlight the item you\n"
  49"   wish to change or the submenu you wish to select and press <Enter>.\n"
  50"   Submenus are designated by \"--->\", empty ones by \"----\".\n"
  51"\n"
  52"   Shortcut: Press the option's highlighted letter (hotkey).\n"
  53"             Pressing a hotkey more than once will sequence\n"
  54"             through all visible items which use that hotkey.\n"
  55"\n"
  56"   You may also use the <PAGE UP> and <PAGE DOWN> keys to scroll\n"
  57"   unseen options into view.\n"
  58"\n"
  59"o  To exit a menu use the cursor keys to highlight the <Exit> button\n"
  60"   and press <ENTER>.\n"
  61"\n"
  62"   Shortcut: Press <ESC><ESC> or <E> or <X> if there is no hotkey\n"
  63"             using those letters.  You may press a single <ESC>, but\n"
  64"             there is a delayed response which you may find annoying.\n"
  65"\n"
  66"   Also, the <TAB> and cursor keys will cycle between <Select>,\n"
  67"   <Exit>, <Help>, <Save>, and <Load>.\n"
  68"\n"
  69"o  To get help with an item, use the cursor keys to highlight <Help>\n"
  70"   and press <ENTER>.\n"
  71"\n"
  72"   Shortcut: Press <H> or <?>.\n"
  73"\n"
  74"o  To toggle the display of hidden options, press <Z>.\n"
  75"\n"
  76"\n"
  77"Radiolists  (Choice lists)\n"
  78"-----------\n"
  79"o  Use the cursor keys to select the option you wish to set and press\n"
  80"   <S> or the <SPACE BAR>.\n"
  81"\n"
  82"   Shortcut: Press the first letter of the option you wish to set then\n"
  83"             press <S> or <SPACE BAR>.\n"
  84"\n"
  85"o  To see available help for the item, use the cursor keys to highlight\n"
  86"   <Help> and Press <ENTER>.\n"
  87"\n"
  88"   Shortcut: Press <H> or <?>.\n"
  89"\n"
  90"   Also, the <TAB> and cursor keys will cycle between <Select> and\n"
  91"   <Help>\n"
  92"\n"
  93"\n"
  94"Data Entry\n"
  95"-----------\n"
  96"o  Enter the requested information and press <ENTER>\n"
  97"   If you are entering hexadecimal values, it is not necessary to\n"
  98"   add the '0x' prefix to the entry.\n"
  99"\n"
 100"o  For help, use the <TAB> or cursor keys to highlight the help option\n"
 101"   and press <ENTER>.  You can try <TAB><H> as well.\n"
 102"\n"
 103"\n"
 104"Text Box    (Help Window)\n"
 105"--------\n"
 106"o  Use the cursor keys to scroll up/down/left/right.  The VI editor\n"
 107"   keys h,j,k,l function here as do <u>, <d>, <SPACE BAR> and <B> for\n"
 108"   those who are familiar with less and lynx.\n"
 109"\n"
 110"o  Press <E>, <X>, <q>, <Enter> or <Esc><Esc> to exit.\n"
 111"\n"
 112"\n"
 113"Alternate Configuration Files\n"
 114"-----------------------------\n"
 115"Menuconfig supports the use of alternate configuration files for\n"
 116"those who, for various reasons, find it necessary to switch\n"
 117"between different configurations.\n"
 118"\n"
 119"The <Save> button will let you save the current configuration to\n"
 120"a file of your choosing.  Use the <Load> button to load a previously\n"
 121"saved alternate configuration.\n"
 122"\n"
 123"Even if you don't use alternate configuration files, but you find\n"
 124"during a Menuconfig session that you have completely messed up your\n"
 125"settings, you may use the <Load> button to restore your previously\n"
 126"saved settings from \".config\" without restarting Menuconfig.\n"
 127"\n"
 128"Other information\n"
 129"-----------------\n"
 130"If you use Menuconfig in an XTERM window, make sure you have your\n"
 131"$TERM variable set to point to an xterm definition which supports\n"
 132"color.  Otherwise, Menuconfig will look rather bad.  Menuconfig will\n"
 133"not display correctly in an RXVT window because rxvt displays only one\n"
 134"intensity of color, bright.\n"
 135"\n"
 136"Menuconfig will display larger menus on screens or xterms which are\n"
 137"set to display more than the standard 25 row by 80 column geometry.\n"
 138"In order for this to work, the \"stty size\" command must be able to\n"
 139"display the screen's current row and column geometry.  I STRONGLY\n"
 140"RECOMMEND that you make sure you do NOT have the shell variables\n"
 141"LINES and COLUMNS exported into your environment.  Some distributions\n"
 142"export those variables via /etc/profile.  Some ncurses programs can\n"
 143"become confused when those variables (LINES & COLUMNS) don't reflect\n"
 144"the true screen size.\n"
 145"\n"
 146"Optional personality available\n"
 147"------------------------------\n"
 148"If you prefer to have all of the options listed in a single menu,\n"
 149"rather than the default multimenu hierarchy, run the menuconfig with\n"
 150"MENUCONFIG_MODE environment variable set to single_menu. Example:\n"
 151"\n"
 152"make MENUCONFIG_MODE=single_menu menuconfig\n"
 153"\n"
 154"<Enter> will then unroll the appropriate category, or enfold it if it\n"
 155"is already unrolled.\n"
 156"\n"
 157"Note that this mode can eventually be a little more CPU expensive\n"
 158"(especially with a larger number of unrolled categories) than the\n"
 159"default mode.\n"
 160"\n"
 161"Different color themes available\n"
 162"--------------------------------\n"
 163"It is possible to select different color themes using the variable\n"
 164"MENUCONFIG_COLOR. To select a theme use:\n"
 165"\n"
 166"make MENUCONFIG_COLOR=<theme> menuconfig\n"
 167"\n"
 168"Available themes are\n"
 169" mono       => selects colors suitable for monochrome displays\n"
 170" blackbg    => selects a color scheme with black background\n"
 171" classic    => theme with blue background. The classic look\n"
 172" bluetitle  => an LCD friendly version of classic. (default)\n"
 173"\n",
 174menu_instructions[] =
 175	"Arrow keys navigate the menu.  "
 176	"<Enter> selects submenus ---> (or empty submenus ----).  "
 177	"Highlighted letters are hotkeys.  "
 178	"Pressing <Y> includes, <N> excludes, <M> modularizes features.  "
 179	"Press <Esc><Esc> to exit, <?> for Help, </> for Search.  "
 180	"Legend: [*] built-in  [ ] excluded  <M> module  < > module capable",
 181radiolist_instructions[] =
 182	"Use the arrow keys to navigate this window or "
 183	"press the hotkey of the item you wish to select "
 184	"followed by the <SPACE BAR>. "
 185	"Press <?> for additional information about this option.",
 186inputbox_instructions_int[] =
 187	"Please enter a decimal value. "
 188	"Fractions will not be accepted.  "
 189	"Use the <TAB> key to move from the input field to the buttons below it.",
 190inputbox_instructions_hex[] =
 191	"Please enter a hexadecimal value. "
 192	"Use the <TAB> key to move from the input field to the buttons below it.",
 193inputbox_instructions_string[] =
 194	"Please enter a string value. "
 195	"Use the <TAB> key to move from the input field to the buttons below it.",
 196setmod_text[] =
 197	"This feature depends on another which has been configured as a module.\n"
 198	"As a result, this feature will be built as a module.",
 199load_config_text[] =
 200	"Enter the name of the configuration file you wish to load.  "
 201	"Accept the name shown to restore the configuration you "
 202	"last retrieved.  Leave blank to abort.",
 203load_config_help[] =
 204	"\n"
 205	"For various reasons, one may wish to keep several different\n"
 206	"configurations available on a single machine.\n"
 207	"\n"
 208	"If you have saved a previous configuration in a file other than the\n"
 209	"default one, entering its name here will allow you to modify that\n"
 210	"configuration.\n"
 211	"\n"
 212	"If you are uncertain, then you have probably never used alternate\n"
 213	"configuration files. You should therefore leave this blank to abort.\n",
 214save_config_text[] =
 215	"Enter a filename to which this configuration should be saved "
 216	"as an alternate.  Leave blank to abort.",
 217save_config_help[] =
 218	"\n"
 219	"For various reasons, one may wish to keep different configurations\n"
 220	"available on a single machine.\n"
 221	"\n"
 222	"Entering a file name here will allow you to later retrieve, modify\n"
 223	"and use the current configuration as an alternate to whatever\n"
 224	"configuration options you have selected at that time.\n"
 225	"\n"
 226	"If you are uncertain what all this means then you should probably\n"
 227	"leave this blank.\n",
 228search_help[] =
 229	"\n"
 230	"Search for symbols and display their relations.\n"
 231	"Regular expressions are allowed.\n"
 232	"Example: search for \"^FOO\"\n"
 233	"Result:\n"
 234	"-----------------------------------------------------------------\n"
 235	"Symbol: FOO [=m]\n"
 236	"Type  : tristate\n"
 237	"Prompt: Foo bus is used to drive the bar HW\n"
 238	"  Location:\n"
 239	"    -> Bus options (PCI, PCMCIA, EISA, ISA)\n"
 240	"      -> PCI support (PCI [=y])\n"
 241	"(1)     -> PCI access mode (<choice> [=y])\n"
 242	"  Defined at drivers/pci/Kconfig:47\n"
 243	"  Depends on: X86_LOCAL_APIC && X86_IO_APIC || IA64\n"
 244	"  Selects: LIBCRC32\n"
 245	"  Selected by: BAR [=n]\n"
 246	"-----------------------------------------------------------------\n"
 247	"o The line 'Type:' shows the type of the configuration option for\n"
 248	"  this symbol (bool, tristate, string, ...)\n"
 249	"o The line 'Prompt:' shows the text used in the menu structure for\n"
 250	"  this symbol\n"
 251	"o The 'Defined at' line tells at what file / line number the symbol\n"
 252	"  is defined\n"
 253	"o The 'Depends on:' line tells what symbols need to be defined for\n"
 254	"  this symbol to be visible in the menu (selectable)\n"
 255	"o The 'Location:' lines tells where in the menu structure this symbol\n"
 256	"  is located\n"
 257	"    A location followed by a [=y] indicates that this is a\n"
 258	"    selectable menu item - and the current value is displayed inside\n"
 259	"    brackets.\n"
 260	"    Press the key in the (#) prefix to jump directly to that\n"
 261	"    location. You will be returned to the current search results\n"
 262	"    after exiting this new menu.\n"
 263	"o The 'Selects:' line tells what symbols will be automatically\n"
 264	"  selected if this symbol is selected (y or m)\n"
 265	"o The 'Selected by' line tells what symbol has selected this symbol\n"
 266	"\n"
 267	"Only relevant lines are shown.\n"
 268	"\n\n"
 269	"Search examples:\n"
 270	"Examples: USB	=> find all symbols containing USB\n"
 271	"          ^USB => find all symbols starting with USB\n"
 272	"          USB$ => find all symbols ending with USB\n"
 273	"\n";
 274
 275static int indent;
 276static struct menu *current_menu;
 277static int child_count;
 278static int single_menu_mode;
 279static int show_all_options;
 280static int save_and_exit;
 281static int silent;
 282
 283static void conf(struct menu *menu, struct menu *active_menu);
 284static void conf_choice(struct menu *menu);
 285static void conf_string(struct menu *menu);
 286static void conf_load(void);
 287static void conf_save(void);
 288static int show_textbox_ext(const char *title, char *text, int r, int c,
 289			    int *keys, int *vscroll, int *hscroll,
 290			    update_text_fn update_text, void *data);
 291static void show_textbox(const char *title, const char *text, int r, int c);
 292static void show_helptext(const char *title, const char *text);
 293static void show_help(struct menu *menu);
 294
 295static char filename[PATH_MAX+1];
 296static void set_config_filename(const char *config_filename)
 297{
 298	static char menu_backtitle[PATH_MAX+128];
 299	int size;
 300
 301	size = snprintf(menu_backtitle, sizeof(menu_backtitle),
 302			"%s - %s", config_filename, rootmenu.prompt->text);
 303	if (size >= sizeof(menu_backtitle))
 304		menu_backtitle[sizeof(menu_backtitle)-1] = '\0';
 305	set_dialog_backtitle(menu_backtitle);
 306
 307	size = snprintf(filename, sizeof(filename), "%s", config_filename);
 308	if (size >= sizeof(filename))
 309		filename[sizeof(filename)-1] = '\0';
 310}
 311
 312struct subtitle_part {
 313	struct list_head entries;
 314	const char *text;
 315};
 316static LIST_HEAD(trail);
 317
 318static struct subtitle_list *subtitles;
 319static void set_subtitle(void)
 320{
 321	struct subtitle_part *sp;
 322	struct subtitle_list *pos, *tmp;
 323
 324	for (pos = subtitles; pos != NULL; pos = tmp) {
 325		tmp = pos->next;
 326		free(pos);
 327	}
 328
 329	subtitles = NULL;
 330	list_for_each_entry(sp, &trail, entries) {
 331		if (sp->text) {
 332			if (pos) {
 333				pos->next = xcalloc(1, sizeof(*pos));
 334				pos = pos->next;
 335			} else {
 336				subtitles = pos = xcalloc(1, sizeof(*pos));
 337			}
 338			pos->text = sp->text;
 339		}
 340	}
 341
 342	set_dialog_subtitles(subtitles);
 343}
 344
 345static void reset_subtitle(void)
 346{
 347	struct subtitle_list *pos, *tmp;
 348
 349	for (pos = subtitles; pos != NULL; pos = tmp) {
 350		tmp = pos->next;
 351		free(pos);
 352	}
 353	subtitles = NULL;
 354	set_dialog_subtitles(subtitles);
 355}
 356
 357struct search_data {
 358	struct list_head *head;
 359	struct menu **targets;
 360	int *keys;
 361};
 362
 363static void update_text(char *buf, size_t start, size_t end, void *_data)
 364{
 365	struct search_data *data = _data;
 366	struct jump_key *pos;
 367	int k = 0;
 368
 369	list_for_each_entry(pos, data->head, entries) {
 370		if (pos->offset >= start && pos->offset < end) {
 371			char header[4];
 372
 373			if (k < JUMP_NB) {
 374				int key = '0' + (pos->index % JUMP_NB) + 1;
 375
 376				sprintf(header, "(%c)", key);
 377				data->keys[k] = key;
 378				data->targets[k] = pos->target;
 379				k++;
 380			} else {
 381				sprintf(header, "   ");
 382			}
 383
 384			memcpy(buf + pos->offset, header, sizeof(header) - 1);
 385		}
 386	}
 387	data->keys[k] = 0;
 388}
 389
 390static void search_conf(void)
 391{
 392	struct symbol **sym_arr;
 393	struct gstr res;
 394	struct gstr title;
 395	char *dialog_input;
 396	int dres, vscroll = 0, hscroll = 0;
 397	bool again;
 398	struct gstr sttext;
 399	struct subtitle_part stpart;
 400
 401	title = str_new();
 402	str_printf( &title, "Enter (sub)string or regexp to search for "
 403			      "(with or without \"%s\")", CONFIG_);
 404
 405again:
 406	dialog_clear();
 407	dres = dialog_inputbox("Search Configuration Parameter",
 408			      str_get(&title),
 409			      10, 75, "");
 410	switch (dres) {
 411	case 0:
 412		break;
 413	case 1:
 414		show_helptext("Search Configuration", search_help);
 415		goto again;
 416	default:
 417		str_free(&title);
 418		return;
 419	}
 420
 421	/* strip the prefix if necessary */
 422	dialog_input = dialog_input_result;
 423	if (strncasecmp(dialog_input_result, CONFIG_, strlen(CONFIG_)) == 0)
 424		dialog_input += strlen(CONFIG_);
 425
 426	sttext = str_new();
 427	str_printf(&sttext, "Search (%s)", dialog_input_result);
 428	stpart.text = str_get(&sttext);
 429	list_add_tail(&stpart.entries, &trail);
 430
 431	sym_arr = sym_re_search(dialog_input);
 432	do {
 433		LIST_HEAD(head);
 434		struct menu *targets[JUMP_NB];
 435		int keys[JUMP_NB + 1], i;
 436		struct search_data data = {
 437			.head = &head,
 438			.targets = targets,
 439			.keys = keys,
 440		};
 441		struct jump_key *pos, *tmp;
 442
 443		res = get_relations_str(sym_arr, &head);
 444		set_subtitle();
 445		dres = show_textbox_ext("Search Results", (char *)
 446					str_get(&res), 0, 0, keys, &vscroll,
 447					&hscroll, &update_text, (void *)
 448					&data);
 449		again = false;
 450		for (i = 0; i < JUMP_NB && keys[i]; i++)
 451			if (dres == keys[i]) {
 452				conf(targets[i]->parent, targets[i]);
 453				again = true;
 454			}
 455		str_free(&res);
 456		list_for_each_entry_safe(pos, tmp, &head, entries)
 457			free(pos);
 458	} while (again);
 459	free(sym_arr);
 460	str_free(&title);
 461	list_del(trail.prev);
 462	str_free(&sttext);
 463}
 464
 465static void build_conf(struct menu *menu)
 466{
 467	struct symbol *sym;
 468	struct property *prop;
 469	struct menu *child;
 470	int type, tmp, doint = 2;
 471	tristate val;
 472	char ch;
 473	bool visible;
 474
 475	/*
 476	 * note: menu_is_visible() has side effect that it will
 477	 * recalc the value of the symbol.
 478	 */
 479	visible = menu_is_visible(menu);
 480	if (show_all_options && !menu_has_prompt(menu))
 481		return;
 482	else if (!show_all_options && !visible)
 483		return;
 484
 485	sym = menu->sym;
 486	prop = menu->prompt;
 487	if (!sym) {
 488		if (prop && menu != current_menu) {
 489			const char *prompt = menu_get_prompt(menu);
 490			switch (prop->type) {
 491			case P_MENU:
 492				child_count++;
 493				if (single_menu_mode) {
 494					item_make("%s%*c%s",
 495						  menu->data ? "-->" : "++>",
 496						  indent + 1, ' ', prompt);
 497				} else
 498					item_make("   %*c%s  %s",
 499						  indent + 1, ' ', prompt,
 500						  menu_is_empty(menu) ? "----" : "--->");
 501				item_set_tag('m');
 502				item_set_data(menu);
 503				if (single_menu_mode && menu->data)
 504					goto conf_childs;
 505				return;
 506			case P_COMMENT:
 507				if (prompt) {
 508					child_count++;
 509					item_make("   %*c*** %s ***", indent + 1, ' ', prompt);
 510					item_set_tag(':');
 511					item_set_data(menu);
 512				}
 513				break;
 514			default:
 515				if (prompt) {
 516					child_count++;
 517					item_make("---%*c%s", indent + 1, ' ', prompt);
 518					item_set_tag(':');
 519					item_set_data(menu);
 520				}
 521			}
 522		} else
 523			doint = 0;
 524		goto conf_childs;
 525	}
 526
 527	type = sym_get_type(sym);
 528	if (sym_is_choice(sym)) {
 529		struct symbol *def_sym = sym_get_choice_value(sym);
 530		struct menu *def_menu = NULL;
 531
 532		child_count++;
 533		for (child = menu->list; child; child = child->next) {
 534			if (menu_is_visible(child) && child->sym == def_sym)
 535				def_menu = child;
 536		}
 537
 538		val = sym_get_tristate_value(sym);
 539		if (sym_is_changeable(sym)) {
 540			switch (type) {
 541			case S_BOOLEAN:
 542				item_make("[%c]", val == no ? ' ' : '*');
 543				break;
 544			case S_TRISTATE:
 545				switch (val) {
 546				case yes: ch = '*'; break;
 547				case mod: ch = 'M'; break;
 548				default:  ch = ' '; break;
 549				}
 550				item_make("<%c>", ch);
 551				break;
 552			}
 553			item_set_tag('t');
 554			item_set_data(menu);
 555		} else {
 556			item_make("   ");
 557			item_set_tag(def_menu ? 't' : ':');
 558			item_set_data(menu);
 559		}
 560
 561		item_add_str("%*c%s", indent + 1, ' ', menu_get_prompt(menu));
 562		if (val == yes) {
 563			if (def_menu) {
 564				item_add_str(" (%s)", menu_get_prompt(def_menu));
 565				item_add_str("  --->");
 566				if (def_menu->list) {
 567					indent += 2;
 568					build_conf(def_menu);
 569					indent -= 2;
 570				}
 571			}
 572			return;
 573		}
 574	} else {
 575		if (menu == current_menu) {
 576			item_make("---%*c%s", indent + 1, ' ', menu_get_prompt(menu));
 577			item_set_tag(':');
 578			item_set_data(menu);
 579			goto conf_childs;
 580		}
 581		child_count++;
 582		val = sym_get_tristate_value(sym);
 583		if (sym_is_choice_value(sym) && val == yes) {
 584			item_make("   ");
 585			item_set_tag(':');
 586			item_set_data(menu);
 587		} else {
 588			switch (type) {
 589			case S_BOOLEAN:
 590				if (sym_is_changeable(sym))
 591					item_make("[%c]", val == no ? ' ' : '*');
 592				else
 593					item_make("-%c-", val == no ? ' ' : '*');
 594				item_set_tag('t');
 595				item_set_data(menu);
 596				break;
 597			case S_TRISTATE:
 598				switch (val) {
 599				case yes: ch = '*'; break;
 600				case mod: ch = 'M'; break;
 601				default:  ch = ' '; break;
 602				}
 603				if (sym_is_changeable(sym)) {
 604					if (sym->rev_dep.tri == mod)
 605						item_make("{%c}", ch);
 606					else
 607						item_make("<%c>", ch);
 608				} else
 609					item_make("-%c-", ch);
 610				item_set_tag('t');
 611				item_set_data(menu);
 612				break;
 613			default:
 614				tmp = 2 + strlen(sym_get_string_value(sym)); /* () = 2 */
 615				item_make("(%s)", sym_get_string_value(sym));
 616				tmp = indent - tmp + 4;
 617				if (tmp < 0)
 618					tmp = 0;
 619				item_add_str("%*c%s%s", tmp, ' ', menu_get_prompt(menu),
 620					     (sym_has_value(sym) || !sym_is_changeable(sym)) ?
 621					     "" : " (NEW)");
 622				item_set_tag('s');
 623				item_set_data(menu);
 624				goto conf_childs;
 625			}
 626		}
 627		item_add_str("%*c%s%s", indent + 1, ' ', menu_get_prompt(menu),
 628			  (sym_has_value(sym) || !sym_is_changeable(sym)) ?
 629			  "" : " (NEW)");
 630		if (menu->prompt->type == P_MENU) {
 631			item_add_str("  %s", menu_is_empty(menu) ? "----" : "--->");
 632			return;
 633		}
 634	}
 635
 636conf_childs:
 637	indent += doint;
 638	for (child = menu->list; child; child = child->next)
 639		build_conf(child);
 640	indent -= doint;
 641}
 642
 643static void conf(struct menu *menu, struct menu *active_menu)
 644{
 645	struct menu *submenu;
 646	const char *prompt = menu_get_prompt(menu);
 647	struct subtitle_part stpart;
 648	struct symbol *sym;
 649	int res;
 650	int s_scroll = 0;
 651
 652	if (menu != &rootmenu)
 653		stpart.text = menu_get_prompt(menu);
 654	else
 655		stpart.text = NULL;
 656	list_add_tail(&stpart.entries, &trail);
 657
 658	while (1) {
 659		item_reset();
 660		current_menu = menu;
 661		build_conf(menu);
 662		if (!child_count)
 663			break;
 664		set_subtitle();
 665		dialog_clear();
 666		res = dialog_menu(prompt ? prompt : "Main Menu",
 667				  menu_instructions,
 668				  active_menu, &s_scroll);
 669		if (res == 1 || res == KEY_ESC || res == -ERRDISPLAYTOOSMALL)
 670			break;
 671		if (item_count() != 0) {
 672			if (!item_activate_selected())
 673				continue;
 674			if (!item_tag())
 675				continue;
 676		}
 677		submenu = item_data();
 678		active_menu = item_data();
 679		if (submenu)
 680			sym = submenu->sym;
 681		else
 682			sym = NULL;
 683
 684		switch (res) {
 685		case 0:
 686			switch (item_tag()) {
 687			case 'm':
 688				if (single_menu_mode)
 689					submenu->data = (void *) (long) !submenu->data;
 690				else
 691					conf(submenu, NULL);
 692				break;
 693			case 't':
 694				if (sym_is_choice(sym) && sym_get_tristate_value(sym) == yes)
 695					conf_choice(submenu);
 696				else if (submenu->prompt->type == P_MENU)
 697					conf(submenu, NULL);
 698				break;
 699			case 's':
 700				conf_string(submenu);
 701				break;
 702			}
 703			break;
 704		case 2:
 705			if (sym)
 706				show_help(submenu);
 707			else {
 708				reset_subtitle();
 709				show_helptext("README", mconf_readme);
 710			}
 711			break;
 712		case 3:
 713			reset_subtitle();
 714			conf_save();
 715			break;
 716		case 4:
 717			reset_subtitle();
 718			conf_load();
 719			break;
 720		case 5:
 721			if (item_is_tag('t')) {
 722				if (sym_set_tristate_value(sym, yes))
 723					break;
 724				if (sym_set_tristate_value(sym, mod))
 725					show_textbox(NULL, setmod_text, 6, 74);
 726			}
 727			break;
 728		case 6:
 729			if (item_is_tag('t'))
 730				sym_set_tristate_value(sym, no);
 731			break;
 732		case 7:
 733			if (item_is_tag('t'))
 734				sym_set_tristate_value(sym, mod);
 735			break;
 736		case 8:
 737			if (item_is_tag('t'))
 738				sym_toggle_tristate_value(sym);
 739			else if (item_is_tag('m'))
 740				conf(submenu, NULL);
 741			break;
 742		case 9:
 743			search_conf();
 744			break;
 745		case 10:
 746			show_all_options = !show_all_options;
 747			break;
 748		}
 749	}
 750
 751	list_del(trail.prev);
 752}
 753
 754static int show_textbox_ext(const char *title, char *text, int r, int c, int
 755			    *keys, int *vscroll, int *hscroll, update_text_fn
 756			    update_text, void *data)
 757{
 758	dialog_clear();
 759	return dialog_textbox(title, text, r, c, keys, vscroll, hscroll,
 760			      update_text, data);
 761}
 762
 763static void show_textbox(const char *title, const char *text, int r, int c)
 764{
 765	show_textbox_ext(title, (char *) text, r, c, (int []) {0}, NULL, NULL,
 766			 NULL, NULL);
 767}
 768
 769static void show_helptext(const char *title, const char *text)
 770{
 771	show_textbox(title, text, 0, 0);
 772}
 773
 774static void conf_message_callback(const char *s)
 775{
 776	if (save_and_exit) {
 777		if (!silent)
 778			printf("%s", s);
 779	} else {
 780		show_textbox(NULL, s, 6, 60);
 781	}
 782}
 783
 784static void show_help(struct menu *menu)
 785{
 786	struct gstr help = str_new();
 787
 788	help.max_width = getmaxx(stdscr) - 10;
 789	menu_get_ext_help(menu, &help);
 790
 791	show_helptext(menu_get_prompt(menu), str_get(&help));
 792	str_free(&help);
 793}
 794
 795static void conf_choice(struct menu *menu)
 796{
 797	const char *prompt = menu_get_prompt(menu);
 798	struct menu *child;
 799	struct symbol *active;
 800
 801	active = sym_get_choice_value(menu->sym);
 802	while (1) {
 803		int res;
 804		int selected;
 805		item_reset();
 806
 807		current_menu = menu;
 808		for (child = menu->list; child; child = child->next) {
 809			if (!menu_is_visible(child))
 810				continue;
 811			if (child->sym)
 812				item_make("%s", menu_get_prompt(child));
 813			else {
 814				item_make("*** %s ***", menu_get_prompt(child));
 815				item_set_tag(':');
 816			}
 817			item_set_data(child);
 818			if (child->sym == active)
 819				item_set_selected(1);
 820			if (child->sym == sym_get_choice_value(menu->sym))
 821				item_set_tag('X');
 822		}
 823		dialog_clear();
 824		res = dialog_checklist(prompt ? prompt : "Main Menu",
 825					radiolist_instructions,
 826					MENUBOX_HEIGTH_MIN,
 827					MENUBOX_WIDTH_MIN,
 828					CHECKLIST_HEIGTH_MIN);
 829		selected = item_activate_selected();
 830		switch (res) {
 831		case 0:
 832			if (selected) {
 833				child = item_data();
 834				if (!child->sym)
 835					break;
 836
 837				sym_set_tristate_value(child->sym, yes);
 838			}
 839			return;
 840		case 1:
 841			if (selected) {
 842				child = item_data();
 843				show_help(child);
 844				active = child->sym;
 845			} else
 846				show_help(menu);
 847			break;
 848		case KEY_ESC:
 849			return;
 850		case -ERRDISPLAYTOOSMALL:
 851			return;
 852		}
 853	}
 854}
 855
 856static void conf_string(struct menu *menu)
 857{
 858	const char *prompt = menu_get_prompt(menu);
 859
 860	while (1) {
 861		int res;
 862		const char *heading;
 863
 864		switch (sym_get_type(menu->sym)) {
 865		case S_INT:
 866			heading = inputbox_instructions_int;
 867			break;
 868		case S_HEX:
 869			heading = inputbox_instructions_hex;
 870			break;
 871		case S_STRING:
 872			heading = inputbox_instructions_string;
 873			break;
 874		default:
 875			heading = "Internal mconf error!";
 876		}
 877		dialog_clear();
 878		res = dialog_inputbox(prompt ? prompt : "Main Menu",
 879				      heading, 10, 75,
 880				      sym_get_string_value(menu->sym));
 881		switch (res) {
 882		case 0:
 883			if (sym_set_string_value(menu->sym, dialog_input_result))
 884				return;
 885			show_textbox(NULL, "You have made an invalid entry.", 5, 43);
 886			break;
 887		case 1:
 888			show_help(menu);
 889			break;
 890		case KEY_ESC:
 891			return;
 892		}
 893	}
 894}
 895
 896static void conf_load(void)
 897{
 898
 899	while (1) {
 900		int res;
 901		dialog_clear();
 902		res = dialog_inputbox(NULL, load_config_text,
 903				      11, 55, filename);
 904		switch(res) {
 905		case 0:
 906			if (!dialog_input_result[0])
 907				return;
 908			if (!conf_read(dialog_input_result)) {
 909				set_config_filename(dialog_input_result);
 910				sym_set_change_count(1);
 911				return;
 912			}
 913			show_textbox(NULL, "File does not exist!", 5, 38);
 914			break;
 915		case 1:
 916			show_helptext("Load Alternate Configuration", load_config_help);
 917			break;
 918		case KEY_ESC:
 919			return;
 920		}
 921	}
 922}
 923
 924static void conf_save(void)
 925{
 926	while (1) {
 927		int res;
 928		dialog_clear();
 929		res = dialog_inputbox(NULL, save_config_text,
 930				      11, 55, filename);
 931		switch(res) {
 932		case 0:
 933			if (!dialog_input_result[0])
 934				return;
 935			if (!conf_write(dialog_input_result)) {
 936				set_config_filename(dialog_input_result);
 937				return;
 938			}
 939			show_textbox(NULL, "Can't create file!", 5, 60);
 940			break;
 941		case 1:
 942			show_helptext("Save Alternate Configuration", save_config_help);
 943			break;
 944		case KEY_ESC:
 945			return;
 946		}
 947	}
 948}
 949
 950static int handle_exit(void)
 951{
 952	int res;
 953
 954	save_and_exit = 1;
 955	reset_subtitle();
 956	dialog_clear();
 957	if (conf_get_changed())
 958		res = dialog_yesno(NULL,
 959				   "Do you wish to save your new configuration?\n"
 960				     "(Press <ESC><ESC> to continue kernel configuration.)",
 961				   6, 60);
 962	else
 963		res = -1;
 964
 965	end_dialog(saved_x, saved_y);
 966
 967	switch (res) {
 968	case 0:
 969		if (conf_write(filename)) {
 970			fprintf(stderr, "\n\n"
 971					  "Error while writing of the configuration.\n"
 972					  "Your configuration changes were NOT saved."
 973					  "\n\n");
 974			return 1;
 975		}
 976		conf_write_autoconf(0);
 977		/* fall through */
 978	case -1:
 979		if (!silent)
 980			printf("\n\n"
 981				 "*** End of the configuration.\n"
 982				 "*** Execute 'make' to start the build or try 'make help'."
 983				 "\n\n");
 984		res = 0;
 985		break;
 986	default:
 987		if (!silent)
 988			fprintf(stderr, "\n\n"
 989					  "Your configuration changes were NOT saved."
 990					  "\n\n");
 991		if (res != KEY_ESC)
 992			res = 0;
 993	}
 994
 995	return res;
 996}
 997
 998static void sig_handler(int signo)
 999{
1000	exit(handle_exit());
1001}
1002
1003int main(int ac, char **av)
1004{
1005	char *mode;
1006	int res;
1007
1008	signal(SIGINT, sig_handler);
1009
1010	if (ac > 1 && strcmp(av[1], "-s") == 0) {
1011		silent = 1;
1012		/* Silence conf_read() until the real callback is set up */
1013		conf_set_message_callback(NULL);
1014		av++;
1015	}
1016	conf_parse(av[1]);
1017	conf_read(NULL);
1018
1019	mode = getenv("MENUCONFIG_MODE");
1020	if (mode) {
1021		if (!strcasecmp(mode, "single_menu"))
1022			single_menu_mode = 1;
1023	}
1024
1025	if (init_dialog(NULL)) {
1026		fprintf(stderr, "Your display is too small to run Menuconfig!\n");
1027		fprintf(stderr, "It must be at least 19 lines by 80 columns.\n");
1028		return 1;
1029	}
1030
1031	set_config_filename(conf_get_configname());
1032	conf_set_message_callback(conf_message_callback);
1033	do {
1034		conf(&rootmenu, NULL);
1035		res = handle_exit();
1036	} while (res == KEY_ESC);
1037
1038	return res;
1039}