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