Re: [PATCH] nconfig V3

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Sat, Nov 29, 2008 at 11:45:48PM +0200, Nir Tzachar wrote:
> Changes since last version:
> 
> 1) Added cursor to input dialog boxes.
> 2) ? key is bound to help messages.
> 3) Removed the "()" logic of hot keys. Now, the first upper cased letter of an
>    item is the hot key.
> 4) Source was split into two parts; for gui logic (nconf.gui.c) and kconfig
>    interface (nconf.c).
> 5) Shared code with mconf.c has been moved to share.c .
> 6) General improvements (e.g., all text is centered, etc.)
> 
> Open issues:
> 
> 1) The problem Willy reported with pressing the keypad too quickly. As I cannot
>    reproduce this, I would appreciate a recheck.
> 2) The color scheme.
> 
> Regarding the color scheme, I think we should first agree that the patch is
> worth including before we procceed. Tweaking the colors will be the easy part ;)

Thanks Nir!

I will take a closer look later the coming week.
I have a few kbuild bits pending which I need to look at
first. But with sparc,sparc64 merge done this is
next on the agenda and then nconfig.

	Sam


> 
> ---
>  scripts/kconfig/Makefile            |   23 +-
>  scripts/kconfig/conf.c              |  123 ++--
>  scripts/kconfig/confdata.c          |   11 +-
>  scripts/kconfig/lkc.h               |    2 +-
>  scripts/kconfig/mconf.c             |   76 +--
>  scripts/kconfig/menu.c              |    2 +-
>  scripts/kconfig/nconf.c             | 1378 +++++++++++++++++++++++++++++++++++
>  scripts/kconfig/nconf.gui.c         |  653 +++++++++++++++++
>  scripts/kconfig/nconf.h             |   86 +++
>  scripts/kconfig/shared.c            |   82 +++
>  scripts/kconfig/shared.h            |   13 +
>  scripts/kconfig/zconf.tab.c_shipped |    2 +-
>  scripts/kconfig/zconf.y             |    2 +-
>  13 files changed, 2306 insertions(+), 147 deletions(-)
>  create mode 100644 scripts/kconfig/nconf.c
>  create mode 100644 scripts/kconfig/nconf.gui.c
>  create mode 100644 scripts/kconfig/nconf.h
>  create mode 100644 scripts/kconfig/shared.c
>  create mode 100644 scripts/kconfig/shared.h
> 
> diff --git a/scripts/kconfig/Makefile b/scripts/kconfig/Makefile
> index fa8c2dd..58626b0 100644
> --- a/scripts/kconfig/Makefile
> +++ b/scripts/kconfig/Makefile
> @@ -4,11 +4,7 @@
>  
>  PHONY += oldconfig xconfig gconfig menuconfig config silentoldconfig update-po-config
>  
> -ifdef KBUILD_KCONFIG
> -Kconfig := $(KBUILD_KCONFIG)
> -else
>  Kconfig := arch/$(SRCARCH)/Kconfig
> -endif
>  
>  xconfig: $(obj)/qconf
>  	$< $(Kconfig)
> @@ -19,6 +15,9 @@ gconfig: $(obj)/gconf
>  menuconfig: $(obj)/mconf
>  	$< $(Kconfig)
>  
> +nconfig: $(obj)/nconf
> +	$< $(Kconfig)
> +
>  config: $(obj)/conf
>  	$< $(Kconfig)
>  
> @@ -79,6 +78,7 @@ endif
>  # Help text used by make help
>  help:
>  	@echo  '  config	  - Update current config utilising a line-oriented program'
> +	@echo  '  nconfig	  - Update current config utilising a ncurses menu based program'
>  	@echo  '  menuconfig	  - Update current config utilising a menu based program'
>  	@echo  '  xconfig	  - Update current config utilising a QT based front-end'
>  	@echo  '  gconfig	  - Update current config utilising a GTK based front-end'
> @@ -106,6 +106,8 @@ HOST_EXTRACFLAGS += -DLOCALE
>  # conf:	  Used for defconfig, oldconfig and related targets
>  # mconf:  Used for the mconfig target.
>  #         Utilizes the lxdialog package
> +# nconf:  Used for the nconfig target.
> +#         Utilizes ncurses
>  # qconf:  Used for the xconfig target
>  #         Based on QT which needs to be installed to compile it
>  # gconf:  Used for the gconfig target
> @@ -116,7 +118,8 @@ lxdialog := lxdialog/checklist.o lxdialog/util.o lxdialog/inputbox.o
>  lxdialog += lxdialog/textbox.o lxdialog/yesno.o lxdialog/menubox.o
>  
>  conf-objs	:= conf.o  zconf.tab.o
> -mconf-objs	:= mconf.o zconf.tab.o $(lxdialog)
> +mconf-objs	:= mconf.o zconf.tab.o shared.o $(lxdialog)
> +nconf-objs	:= nconf.o zconf.tab.o shared.o nconf.gui.o 
>  kxgettext-objs	:= kxgettext.o zconf.tab.o
>  
>  hostprogs-y := conf qconf gconf kxgettext
> @@ -125,6 +128,10 @@ ifeq ($(MAKECMDGOALS),menuconfig)
>  	hostprogs-y += mconf
>  endif
>  
> +ifeq ($(MAKECMDGOALS),nconfig)
> +	hostprogs-y += nconf
> +endif
> +
>  ifeq ($(MAKECMDGOALS),xconfig)
>  	qconf-target := 1
>  endif
> @@ -144,7 +151,7 @@ endif
>  
>  clean-files	:= lkc_defs.h qconf.moc .tmp_qtcheck \
>  		   .tmp_gtkcheck zconf.tab.c lex.zconf.c zconf.hash.c gconf.glade.h
> -clean-files     += mconf qconf gconf
> +clean-files     += mconf qconf gconf nconf
>  clean-files     += config.pot linux.pot
>  
>  # Check that we have the required ncurses stuff installed for lxdialog (menuconfig)
> @@ -162,6 +169,10 @@ HOST_EXTRACFLAGS += $(shell $(CONFIG_SHELL) $(srctree)/$(src)/check.sh $(HOSTCC)
>  HOSTCFLAGS_lex.zconf.o	:= -I$(src)
>  HOSTCFLAGS_zconf.tab.o	:= -I$(src)
>  
> +HOSTLOADLIBES_nconf	= -lmenu -lpanel -lncurses 
> +HOSTCFLAGS_nconf.o	=
> +LDFLAGS_nconf.o		=
> +
>  HOSTLOADLIBES_qconf	= $(KC_QT_LIBS) -ldl
>  HOSTCXXFLAGS_qconf.o	= $(KC_QT_CFLAGS) -D LKC_DIRECT_LINK
>  
> diff --git a/scripts/kconfig/conf.c b/scripts/kconfig/conf.c
> index 3e1057f..36b5eed 100644
> --- a/scripts/kconfig/conf.c
> +++ b/scripts/kconfig/conf.c
> @@ -32,7 +32,6 @@ char *defconfig_file;
>  
>  static int indent = 1;
>  static int valid_stdin = 1;
> -static int sync_kconfig;
>  static int conf_cnt;
>  static char line[128];
>  static struct menu *rootEntry;
> @@ -66,7 +65,7 @@ static void strip(char *str)
>  
>  static void check_stdin(void)
>  {
> -	if (!valid_stdin) {
> +	if (!valid_stdin && input_mode == ask_silent) {
>  		printf(_("aborted!\n\n"));
>  		printf(_("Console input/output is redirected. "));
>  		printf(_("Run 'make oldconfig' to update configuration.\n\n"));
> @@ -428,6 +427,43 @@ static void check_conf(struct menu *menu)
>  		check_conf(child);
>  }
>  
> +static void conf_do_update(void)
> +{
> +	/* Update until a loop caused no more changes */
> +	do {
> +		conf_cnt = 0;
> +		check_conf(&rootmenu);
> +	} while (conf_cnt);
> +}
> +
> +static int conf_silent_update(void)
> +{
> +	const char *name;
> +
> +	if (conf_get_changed()) {
> +		name = getenv("KCONFIG_NOSILENTUPDATE");
> +		if (name && *name) {
> +			fprintf(stderr,
> +			_("\n*** Kernel configuration requires explicit update.\n\n"));
> +			return 1;
> +		}
> +		conf_do_update();
> +	}
> +	return 0;
> +}
> +
> +static int conf_update(void)
> +{
> +	rootEntry = &rootmenu;
> +	conf(&rootmenu);
> +	if (input_mode == ask_all) {
> +		input_mode = ask_silent;
> +		valid_stdin = 1;
> +	}
> +	conf_do_update();
> +	return 0;
> +}
> +
>  int main(int ac, char **av)
>  {
>  	int opt;
> @@ -441,11 +477,11 @@ int main(int ac, char **av)
>  	while ((opt = getopt(ac, av, "osdD:nmyrh")) != -1) {
>  		switch (opt) {
>  		case 'o':
> -			input_mode = ask_silent;
> +			input_mode = ask_new;
>  			break;
>  		case 's':
>  			input_mode = ask_silent;
> -			sync_kconfig = 1;
> +			valid_stdin = isatty(0) && isatty(1) && isatty(2);
>  			break;
>  		case 'd':
>  			input_mode = set_default;
> @@ -483,19 +519,6 @@ int main(int ac, char **av)
>  	name = av[optind];
>  	conf_parse(name);
>  	//zconfdump(stdout);
> -	if (sync_kconfig) {
> -		if (stat(".config", &tmpstat)) {
> -			fprintf(stderr, _("***\n"
> -				"*** You have not yet configured your kernel!\n"
> -				"*** (missing kernel .config file)\n"
> -				"***\n"
> -				"*** Please run some configurator (e.g. \"make oldconfig\" or\n"
> -				"*** \"make menuconfig\" or \"make xconfig\").\n"
> -				"***\n"));
> -			exit(1);
> -		}
> -	}
> -
>  	switch (input_mode) {
>  	case set_default:
>  		if (!defconfig_file)
> @@ -508,6 +531,16 @@ int main(int ac, char **av)
>  		}
>  		break;
>  	case ask_silent:
> +		if (stat(".config", &tmpstat)) {
> +			printf(_("***\n"
> +				"*** You have not yet configured your kernel!\n"
> +				"*** (missing kernel .config file)\n"
> +				"***\n"
> +				"*** Please run some configurator (e.g. \"make oldconfig\" or\n"
> +				"*** \"make menuconfig\" or \"make xconfig\").\n"
> +				"***\n"));
> +			exit(1);
> +		}
>  	case ask_all:
>  	case ask_new:
>  		conf_read(NULL);
> @@ -536,19 +569,6 @@ int main(int ac, char **av)
>  	default:
>  		break;
>  	}
> -
> -	if (sync_kconfig) {
> -		if (conf_get_changed()) {
> -			name = getenv("KCONFIG_NOSILENTUPDATE");
> -			if (name && *name) {
> -				fprintf(stderr,
> -					_("\n*** Kernel configuration requires explicit update.\n\n"));
> -				return 1;
> -			}
> -		}
> -		valid_stdin = isatty(0) && isatty(1) && isatty(2);
> -	}
> -
>  	switch (input_mode) {
>  	case set_no:
>  		conf_set_all_new_symbols(def_no);
> @@ -565,38 +585,27 @@ int main(int ac, char **av)
>  	case set_default:
>  		conf_set_all_new_symbols(def_default);
>  		break;
> +	case ask_silent:
>  	case ask_new:
> +		if (conf_silent_update())
> +			exit(1);
> +		break;
>  	case ask_all:
> -		rootEntry = &rootmenu;
> -		conf(&rootmenu);
> -		input_mode = ask_silent;
> -		/* fall through */
> -	case ask_silent:
> -		/* Update until a loop caused no more changes */
> -		do {
> -			conf_cnt = 0;
> -			check_conf(&rootmenu);
> -		} while (conf_cnt);
> +		if (conf_update())
> +			exit(1);
>  		break;
>  	}
>  
> -	if (sync_kconfig) {
> -		/* silentoldconfig is used during the build so we shall update autoconf.
> -		 * All other commands are only used to generate a config.
> -		 */
> -		if (conf_get_changed() && conf_write(NULL)) {
> -			fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
> -			exit(1);
> -		}
> -		if (conf_write_autoconf()) {
> -			fprintf(stderr, _("\n*** Error during update of the kernel configuration.\n\n"));
> -			return 1;
> -		}
> -	} else {
> -		if (conf_write(NULL)) {
> -			fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
> -			exit(1);
> -		}
> +	if (conf_write(NULL)) {
> +		fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
> +		exit(1);
> +	}
> +	/* ask_silent is used during the build so we shall update autoconf.
> +	 * All other commands are only used to generate a config.
> +	 */
> +	if (input_mode == ask_silent && conf_write_autoconf()) {
> +		fprintf(stderr, _("\n*** Error during writing of the kernel configuration.\n\n"));
> +		return 1;
>  	}
>  	return 0;
>  }
> diff --git a/scripts/kconfig/confdata.c b/scripts/kconfig/confdata.c
> index 830d9ea..df6a188 100644
> --- a/scripts/kconfig/confdata.c
> +++ b/scripts/kconfig/confdata.c
> @@ -222,10 +222,8 @@ load:
>  				continue;
>  			if (def == S_DEF_USER) {
>  				sym = sym_find(line + 9);
> -				if (!sym) {
> -					sym_add_change_count(1);
> +				if (!sym)
>  					break;
> -				}
>  			} else {
>  				sym = sym_lookup(line + 9, 0);
>  				if (sym->type == S_UNKNOWN)
> @@ -261,10 +259,8 @@ load:
>  			}
>  			if (def == S_DEF_USER) {
>  				sym = sym_find(line + 7);
> -				if (!sym) {
> -					sym_add_change_count(1);
> +				if (!sym)
>  					break;
> -				}
>  			} else {
>  				sym = sym_lookup(line + 7, 0);
>  				if (sym->type == S_UNKNOWN)
> @@ -852,7 +848,8 @@ void conf_set_all_new_symbols(enum conf_def_mode mode)
>  
>  	}
>  
> -	sym_clear_all_valid();
> +	if (modules_sym)
> +		sym_calc_value(modules_sym);
>  
>  	if (mode != def_random)
>  		return;
> diff --git a/scripts/kconfig/lkc.h b/scripts/kconfig/lkc.h
> index 4a9af6f..0e9c477 100644
> --- a/scripts/kconfig/lkc.h
> +++ b/scripts/kconfig/lkc.h
> @@ -83,7 +83,7 @@ void conf_set_all_new_symbols(enum conf_def_mode mode);
>  void kconfig_load(void);
>  
>  /* menu.c */
> -void menu_init(void);
> +void _menu_init(void);
>  void menu_warn(struct menu *menu, const char *fmt, ...);
>  struct menu *menu_add_menu(void);
>  void menu_end_menu(void);
> diff --git a/scripts/kconfig/mconf.c b/scripts/kconfig/mconf.c
> index 6841e95..96dfcab 100644
> --- a/scripts/kconfig/mconf.c
> +++ b/scripts/kconfig/mconf.c
> @@ -6,6 +6,8 @@
>   * 2002-11-06 Petr Baudis <pasky@xxxxxx>
>   *
>   * i18n, 2005, Arnaldo Carvalho de Melo <acme@xxxxxxxxxxxxxxxx>
> + *
> + * ncurses interface, 2008, Nir Tzachar <nir@xxxxxxxxxxx>
>   */
>  
>  #include <ctype.h>
> @@ -20,6 +22,7 @@
>  
>  #define LKC_DIRECT_LINK
>  #include "lkc.h"
> +#include "shared.h"
>  #include "lxdialog/dialog.h"
>  
>  static const char mconf_readme[] = N_(
> @@ -284,78 +287,6 @@ static void show_textbox(const char *title, const char *text, int r, int c);
>  static void show_helptext(const char *title, const char *text);
>  static void show_help(struct menu *menu);
>  
> -static void get_prompt_str(struct gstr *r, struct property *prop)
> -{
> -	int i, j;
> -	struct menu *submenu[8], *menu;
> -
> -	str_printf(r, _("Prompt: %s\n"), _(prop->text));
> -	str_printf(r, _("  Defined at %s:%d\n"), prop->menu->file->name,
> -		prop->menu->lineno);
> -	if (!expr_is_yes(prop->visible.expr)) {
> -		str_append(r, _("  Depends on: "));
> -		expr_gstr_print(prop->visible.expr, r);
> -		str_append(r, "\n");
> -	}
> -	menu = prop->menu->parent;
> -	for (i = 0; menu != &rootmenu && i < 8; menu = menu->parent)
> -		submenu[i++] = menu;
> -	if (i > 0) {
> -		str_printf(r, _("  Location:\n"));
> -		for (j = 4; --i >= 0; j += 2) {
> -			menu = submenu[i];
> -			str_printf(r, "%*c-> %s", j, ' ', _(menu_get_prompt(menu)));
> -			if (menu->sym) {
> -				str_printf(r, " (%s [=%s])", menu->sym->name ?
> -					menu->sym->name : _("<choice>"),
> -					sym_get_string_value(menu->sym));
> -			}
> -			str_append(r, "\n");
> -		}
> -	}
> -}
> -
> -static void get_symbol_str(struct gstr *r, struct symbol *sym)
> -{
> -	bool hit;
> -	struct property *prop;
> -
> -	if (sym && sym->name)
> -		str_printf(r, "Symbol: %s [=%s]\n", sym->name,
> -		                                    sym_get_string_value(sym));
> -	for_all_prompts(sym, prop)
> -		get_prompt_str(r, prop);
> -	hit = false;
> -	for_all_properties(sym, prop, P_SELECT) {
> -		if (!hit) {
> -			str_append(r, "  Selects: ");
> -			hit = true;
> -		} else
> -			str_printf(r, " && ");
> -		expr_gstr_print(prop->expr, r);
> -	}
> -	if (hit)
> -		str_append(r, "\n");
> -	if (sym->rev_dep.expr) {
> -		str_append(r, _("  Selected by: "));
> -		expr_gstr_print(sym->rev_dep.expr, r);
> -		str_append(r, "\n");
> -	}
> -	str_append(r, "\n\n");
> -}
> -
> -static struct gstr get_relations_str(struct symbol **sym_arr)
> -{
> -	struct symbol *sym;
> -	struct gstr res = str_new();
> -	int i;
> -
> -	for (i = 0; sym_arr && (sym = sym_arr[i]); i++)
> -		get_symbol_str(&res, sym);
> -	if (!i)
> -		str_append(&res, _("No matches found.\n"));
> -	return res;
> -}
>  
>  static char filename[PATH_MAX+1];
>  static void set_config_filename(const char *config_filename)
> @@ -378,7 +309,6 @@ static void set_config_filename(const char *config_filename)
>  		filename[sizeof(filename)-1] = '\0';
>  }
>  
> -
>  static void search_conf(void)
>  {
>  	struct symbol **sym_arr;
> diff --git a/scripts/kconfig/menu.c b/scripts/kconfig/menu.c
> index 07ff8d1..ade19a3 100644
> --- a/scripts/kconfig/menu.c
> +++ b/scripts/kconfig/menu.c
> @@ -35,7 +35,7 @@ static void prop_warn(struct property *prop, const char *fmt, ...)
>  	va_end(ap);
>  }
>  
> -void menu_init(void)
> +void _menu_init(void)
>  {
>  	current_entry = current_menu = &rootmenu;
>  	last_entry_ptr = &rootmenu.list;
> diff --git a/scripts/kconfig/nconf.c b/scripts/kconfig/nconf.c
> new file mode 100644
> index 0000000..cb64236
> --- /dev/null
> +++ b/scripts/kconfig/nconf.c
> @@ -0,0 +1,1378 @@
> +/*
> + * Copyright (C) 2008 Nir Tzachar <nir.tzachar@xxxxxxxxx?
> + * Released under the terms of the GNU GPL v2.0.
> + *
> + * Derived from menuconfig.
> + *
> + */
> +#define LKC_DIRECT_LINK
> +#include "lkc.h"
> +#include "shared.h"
> +#include "nconf.h"
> +
> +static const char nconf_readme[] = N_(
> +"Overview\n"
> +"--------\n"
> +"Some kernel features may be built directly into the kernel.\n"
> +"Some may be made into loadable runtime modules.  Some features\n"
> +"may be completely removed altogether.  There are also certain\n"
> +"kernel parameters which are not really features, but must be\n"
> +"entered in as decimal or hexadecimal numbers or possibly text.\n"
> +"\n"
> +"Menu items beginning with following braces represent features that\n"
> +"  [ ] can be built in or removed\n"
> +"  < > can be built in, modularized or removed\n"
> +"  { } can be built in or modularized (selected by other feature)\n"
> +"  - - are selected by other feature,\n"
> +"while *, M or whitespace inside braces means to build in, build as\n"
> +"a module or to exclude the feature respectively.\n"
> +"\n"
> +"To change any of these features, highlight it with the cursor\n"
> +"keys and press <Y> to build it in, <M> to make it a module or\n"
> +"<N> to removed it.  You may also press the <Space Bar> to cycle\n"
> +"through the available options (ie. Y->N->M->Y).\n"
> +"\n"
> +"Some additional keyboard hints:\n"
> +"\n"
> +"Menus\n"
> +"----------\n"
> +"o  Use the Up/Down arrow keys (cursor keys) to highlight the item\n"
> +"   you wish to change or submenu wish to select and press <Enter>.\n"
> +"   Submenus are designated by \"--->\".\n"
> +"\n"
> +"   Shortcut: Press the option's highlighted letter (hotkey).\n"
> +"             Pressing a hotkey more than once will sequence\n"
> +"             through all visible items which use that hotkey.\n"
> +"\n"
> +"   You may also use the <PAGE UP> and <PAGE DOWN> keys to scroll\n"
> +"   unseen options into view.\n"
> +"\n"
> +"o  To exit a menu use the cursor keys to highlight the <Exit> button\n"
> +"   and press <ENTER>, or just press <ESC>.\n"
> +"\n"
> +"   Also, the <TAB> and cursor keys will cycle between <Select>,\n"
> +"   <Exit>, <Help> and <Instructions> \n"
> +"\n"
> +"o  To get help with an item, use the cursor keys to highlight <Help>\n"
> +"   and Press <ENTER>.\n"
> +"\n"
> +"   Shortcut: Press <h> or <?>.\n"
> +"\n"
> +"\n"
> +"Radiolists  (Choice lists)\n"
> +"-----------\n"
> +"o  Use the cursor keys to select the option you wish to set and press\n"
> +"   <S> or the <SPACE BAR>.\n"
> +"\n"
> +"   Shortcut: Press the first letter of the option you wish to set then\n"
> +"             press <S> or <SPACE BAR>.\n"
> +"\n"
> +"o  To see available help for the item, use the cursor keys to highlight\n"
> +"   <Help> and Press <ENTER>.\n"
> +"\n"
> +"   Shortcut: Press <H> or <?>.\n"
> +"\n"
> +"   Also, the <TAB> and cursor keys will cycle between <Select>, \n"
> +"   <Exit>, <Help> and <Instructions>\n"
> +"\n"
> +"\n"
> +"Data Entry\n"
> +"-----------\n"
> +"o  Enter the requested information and press <ENTER>\n"
> +"   If you are entering hexadecimal values, it is not necessary to\n"
> +"   add the '0x' prefix to the entry.\n"
> +"\n"
> +"o  For help, use the <TAB> or cursor keys to highlight the help option\n"
> +"   and press <ENTER>.  You can try <TAB><H> as well.\n"
> +"\n"
> +"\n"
> +"Text Box    (Help Window)\n"
> +"--------\n"
> +"o  Use the cursor keys to scroll up/down/left/right.  The VI editor\n"
> +"   keys h,j,k,l function here as do <SPACE BAR> for those\n"
> +"   who are familiar with less and lynx.\n"
> +"\n"
> +"o  Press <Enter> or <Esc> to exit.\n"
> +"\n"
> +"\n"
> +"Alternate Configuration Files\n"
> +"-----------------------------\n"
> +"nconfig supports the use of alternate configuration files for\n"
> +"those who, for various reasons, find it necessary to switch\n"
> +"between different kernel configurations.\n"
> +"\n"
> +"At the end of the main menu you will find two options.  One is\n"
> +"for saving the current configuration to a file of your choosing.\n"
> +"The other option is for loading a previously saved alternate\n"
> +"configuration.\n"
> +"\n"
> +"Even if you don't use alternate configuration files, but you\n"
> +"find during a nconfig session that you have completely messed\n"
> +"up your settings, you may use the \"Load Alternate...\" option to\n"
> +"restore your previously saved settings from \".config\" without\n"
> +"restarting nconfig.\n"
> +"\n"
> +"Other information\n"
> +"-----------------\n"
> +"If you use nconfig in an XTERM window make sure you have your\n"
> +"$TERM variable set to point to a xterm definition which supports color.\n"
> +"Otherwise, nconfig will look rather bad.  nconfig will not\n"
> +"display correctly in a RXVT window because rxvt displays only one\n"
> +"intensity of color, bright.\n"
> +"\n"
> +"nconfig will display larger menus on screens or xterms which are\n"
> +"set to display more than the standard 25 row by 80 column geometry.\n"
> +"In order for this to work, the \"stty size\" command must be able to\n"
> +"display the screen's current row and column geometry.  I STRONGLY\n"
> +"RECOMMEND that you make sure you do NOT have the shell variables\n"
> +"LINES and COLUMNS exported into your environment.  Some distributions\n"
> +"export those variables via /etc/profile.  Some ncurses programs can\n"
> +"become confused when those variables (LINES & COLUMNS) don't reflect\n"
> +"the true screen size.\n"
> +"\n"
> +"Optional personality available\n"
> +"------------------------------\n"
> +"If you prefer to have all of the kernel options listed in a single\n"
> +"menu, rather than the default multimenu hierarchy, run the nconfig\n"
> +"with NCONFIG_MODE environment variable set to single_menu. Example:\n"
> +"\n"
> +"make NCONFIG_MODE = single_menu nconfig\n"
> +"\n"
> +"<Enter> will then unroll the appropriate category, or enfold it if it\n"
> +"is already unrolled.\n"
> +"\n"
> +"Note that this mode can eventually be a little more CPU expensive\n"
> +"(especially with a larger number of unrolled categories) than the\n"
> +"default mode.\n"
> +"\n"),
> +menu_instructions[] = N_(
> +	" Arrow keys navigate the menu.\n"
> +	" <Enter> selects submenus --->.\n"
> +	" Letters in () are hotkeys.\n"
> +	" Pressing <Y> includes, <N> excludes, <M> modularizes features.\n"
> +	" Press <Esc> to go back one menu, <?> or <h> for Help, </> for Search.\n"
> +	" Legend: [*] built-in  [ ] excluded  <M> module  < > module capable.\n"
> +	" <Esc> allways leaves the current window\n"),
> +radiolist_instructions[] = N_(
> +	" Use the arrow keys to navigate this window or\n"
> +	" press the hotkey of the item you wish to select\n"
> +	" followed by the <SPACE BAR>.\n"
> +	" Press <?> or <h> for additional information about this option.\n"),
> +inputbox_instructions_int[] = N_(
> +	"Please enter a decimal value.\n"
> +	"Fractions will not be accepted.\n"
> +	"Press <RETURN> to accept, <ESC> to cancel."),
> +inputbox_instructions_hex[] = N_(
> +	"Please enter a hexadecimal value.\n"
> +	"Press <RETURN> to accept, <ESC> to cancel."),
> +inputbox_instructions_string[] = N_(
> +	"Please enter a string value.\n"
> +	"Press <RETURN> to accept, <ESC> to cancel."),
> +setmod_text[] = N_(
> +	"This feature depends on another which\n"
> +	"has been configured as a module.\n"
> +	"As a result, this feature will be built as a module."),
> +nohelp_text[] = N_(
> +	"There is no help available for this kernel option.\n"),
> +load_config_text[] = N_(
> +	"Enter the name of the configuration file you wish to load.\n"
> +	"Accept the name shown to restore the configuration you\n"
> +	"last retrieved.  Leave blank to abort."),
> +load_config_help[] = N_(
> +	"\n"
> +	"For various reasons, one may wish to keep several different kernel\n"
> +	"configurations available on a single machine.\n"
> +	"\n"
> +	"If you have saved a previous configuration in a file other than the\n"
> +	"kernel's default, entering the name of the file here will allow you\n"
> +	"to modify that configuration.\n"
> +	"\n"
> +	"If you are uncertain, then you have probably never used alternate\n"
> +	"configuration files.  You should therefor leave this blank to abort.\n"),
> +save_config_text[] = N_(
> +	"Enter a filename to which this configuration should be saved\n"
> +	"as an alternate.  Leave blank to abort."),
> +save_config_help[] = N_(
> +	"\n"
> +	"For various reasons, one may wish to keep different kernel\n"
> +	"configurations available on a single machine.\n"
> +	"\n"
> +	"Entering a file name here will allow you to later retrieve, modify\n"
> +	"and use the current configuration as an alternate to whatever\n"
> +	"configuration options you have selected at that time.\n"
> +	"\n"
> +	"If you are uncertain what all this means then you should probably\n"
> +	"leave this blank.\n"),
> +search_help[] = N_(
> +	"\n"
> +	"Search for CONFIG_ symbols and display their relations.\n"
> +	"Regular expressions are allowed.\n"
> +	"Example: search for \"^FOO\"\n"
> +	"Result:\n"
> +	"-----------------------------------------------------------------\n"
> +	"Symbol: FOO [ = m]\n"
> +	"Prompt: Foo bus is used to drive the bar HW\n"
> +	"Defined at drivers/pci/Kconfig:47\n"
> +	"Depends on: X86_LOCAL_APIC && X86_IO_APIC || IA64\n"
> +	"Location:\n"
> +	"  -> Bus options (PCI, PCMCIA, EISA, MCA, ISA)\n"
> +	"    -> PCI support (PCI [ = y])\n"
> +	"      -> PCI access mode (<choice> [ = y])\n"
> +	"Selects: LIBCRC32\n"
> +	"Selected by: BAR\n"
> +	"-----------------------------------------------------------------\n"
> +	"o The line 'Prompt:' shows the text used in the menu structure for\n"
> +	"  this CONFIG_ symbol\n"
> +	"o The 'Defined at' line tell at what file / line number the symbol\n"
> +	"  is defined\n"
> +	"o The 'Depends on:' line tell what symbols needs to be defined for\n"
> +	"  this symbol to be visible in the menu (selectable)\n"
> +	"o The 'Location:' lines tell where in the menu structure this symbol\n"
> +	"  is located\n"
> +	"    A location followed by a [ = y] indicate that this is a selectable\n"
> +	"    menu item - and current value is displayed inside brackets.\n"
> +	"o The 'Selects:' line tell what symbol will be automatically\n"
> +	"  selected if this symbol is selected (y or m)\n"
> +	"o The 'Selected by' line tell what symbol has selected this symbol\n"
> +	"\n"
> +	"Only relevant lines are shown.\n"
> +	"\n\n"
> +	"Search examples:\n"
> +	"Examples: USB	 = > find all CONFIG_ symbols containing USB\n"
> +	"          ^USB => find all CONFIG_ symbols starting with USB\n"
> +	"          USB$ => find all CONFIG_ symbols ending with USB\n"
> +	"\n");
> +
> +struct mitem {
> +	char str[256];
> +	char tag;
> +	void *usrptr;
> +	int is_hot;
> +};
> +
> +#define MAX_MENU_ITEMS 4096
> +static int indent;
> +static struct menu *current_menu;
> +static int child_count;
> +static int single_menu_mode;
> +/* the window in which all information appears */
> +static WINDOW *main_window;
> +/* the largest size of the menu window */
> +static int mwin_max_lines;
> +static int mwin_max_cols;
> +/* the window in which we show option buttons */
> +static WINDOW *btns_window;
> +static MENU *curses_menu;
> +static MENU *btns_menu;
> +static ITEM *curses_menu_items[MAX_MENU_ITEMS];
> +static struct mitem k_menu_items[MAX_MENU_ITEMS];
> +static int items_num;
> +/* the currently selected button */
> +static enum current_btn_t { BTN_SELECT = 0, BTN_EXIT, BTN_HELP, BTN_INSTS,
> +	BTN_MAX} current_btn = BTN_SELECT;
> +static ITEM *btns_items[BTN_MAX];
> +const char *current_instructions = menu_instructions;
> +
> +static char *btns[] = {
> +	"   <Select>   ",
> +	"    <Exit>    ",
> +	"    <Help>    ",
> +	"<Instructions>",
> +	NULL
> +};
> +
> +/* this array is used to implement hot keys. it is updated in item_make and
> + * resetted in clean_items. It would be better to use a hash, but lets keep it
> + * simple... */
> +#define MAX_SAME_KEY MAX_MENU_ITEMS
> +struct {
> +	int count;
> +	int ptrs[MAX_MENU_ITEMS];
> +} hotkeys[1<<(sizeof(char)*8)];
> +
> +static void conf(struct menu *menu);
> +static void conf_choice(struct menu *menu);
> +static void conf_string(struct menu *menu);
> +static void conf_load(void);
> +static void conf_save(void);
> +static void show_help(struct menu *menu);
> +
> +static void update_current_btn(void)
> +{
> +	int index = item_index(current_item(btns_menu));
> +	switch (index) {
> +	case 0:
> +		current_btn = BTN_SELECT;
> +		break;
> +	case 1:
> +		current_btn = BTN_EXIT;
> +		break;
> +	case 2:
> +		current_btn = BTN_HELP;
> +		break;
> +	case 3:
> +		current_btn = BTN_INSTS;
> +		break;
> +	}
> +}
> +
> +
> +static void clean_items(void)
> +{
> +	int i;
> +	for (i = 0; curses_menu_items[i]; i++)
> +		free_item(curses_menu_items[i]);
> +	bzero(curses_menu_items, sizeof(curses_menu_items));
> +	bzero(k_menu_items, sizeof(k_menu_items));
> +	bzero(hotkeys, sizeof(hotkeys));
> +	items_num = 0;
> +}
> +
> +/* return the index of the next hot item, or -1 if no such item exists */
> +int get_next_hot(int c)
> +{
> +	static int hot_index;
> +	static int hot_char;
> +
> +	if (c < 0 || c > 255 || hotkeys[c].count <= 0)
> +		return -1;
> +
> +	if (hot_char == c) {
> +		hot_index = (hot_index+1)%hotkeys[c].count;
> +		return hotkeys[c].ptrs[hot_index];
> +	} else {
> +		hot_char = c;
> +		hot_index = 0;
> +		return hotkeys[c].ptrs[0];
> +	}
> +}
> +
> +/* can the char c be a hot key? no, if c is a common shortcut used elsewhere */
> +int canbhot(char c)
> +{
> +	return isalnum(c) && c != 'm' && c != 'M' && c != 'h' && c != 'H' &&
> +		c != 'n' && c != 'N' && c != 's' && c != '?';
> +}
> +
> +/* check if str already contains a hot key. */
> +int is_hot(int index)
> +{
> +	return k_menu_items[index].is_hot;
> +}
> +
> +/* find the first possible hot key, and mark it.
> + * index is the index of the item in the menu
> + * return 0 on success*/
> +int make_hot(char *dest, int len, const char *org, int index)
> +{
> +	int position = -1;
> +	int i;
> +	int tmp;
> +	int c;
> +	int org_len = strlen(org);
> +
> +	if (org == NULL || is_hot(index))
> +		return 1;
> +
> +	/* make sure not to make hot keys out of markers.
> +	 * find where to start looking for a hot key
> +	 */
> +	i = 0;
> +	/* skip white space */
> +	while (i < org_len && org[i] == ' ')
> +		i++;
> +	if (i == org_len)
> +		return -1;
> +	/* if encountering '(' or '<' or '[', find the match and look from there
> +	 **/
> +	if (org[i] == '[' || org[i] == '<' || org[i] == '(') {
> +		i++;
> +		for (; i < org_len; i++)
> +			if (org[i] == ']' || org[i] == '>' || org[i] == ')')
> +				break;
> +	}
> +	if (i == org_len)
> +		return -1;
> +	for (; i < org_len; i++) {
> +		if (canbhot(org[i]) && org[i-1] != '<' && org[i-1] != '(') {
> +			position = i;
> +			break;
> +		}
> +	}
> +	if (position == -1)
> +		return 1;
> +
> +	/* ok, char at org[position] should be a hot key to this item */
> +	c = tolower(org[position]);
> +	tmp = hotkeys[c].count;
> +	hotkeys[c].ptrs[tmp] = index;
> +	hotkeys[c].count++;
> +	/*
> +	snprintf(dest, len, "%.*s(%c)%s", position, org, org[position],
> +			&org[position+1]);
> +	*/
> +	/* make org[position] uppercase, and all leading letter small case */
> +	strncpy(dest, org, len);
> +	for (i=0; i<position; i++)
> +		dest[i] = tolower(dest[i]);
> +	dest[position] = toupper(dest[position]);
> +	k_menu_items[index].is_hot = 1;
> +	return 0;
> +}
> +
> +/* Make a new item. Add a hotkey mark in the first possible letter.
> + * As ncurses does not allow any attributes inside menue item, we use mark the
> + * hot key as the first capitalized letter in the string */
> +void item_make(void *usrptr, char tag, const char *fmt, ...)
> +{
> +	va_list ap;
> +	char tmp_str[256];
> +
> +	if (items_num > MAX_MENU_ITEMS-1)
> +		return;
> +
> +	bzero(&k_menu_items[items_num], sizeof(k_menu_items[0]));
> +	k_menu_items[items_num].tag = tag;
> +	k_menu_items[items_num].usrptr = usrptr;
> +
> +	va_start(ap, fmt);
> +	vsnprintf(tmp_str, sizeof(tmp_str), fmt, ap);
> +	va_end(ap);
> +	if (make_hot(k_menu_items[items_num].str,
> +		sizeof(k_menu_items[items_num].str), tmp_str, items_num) != 0)
> +		strncpy(k_menu_items[items_num].str,
> +			tmp_str,
> +			sizeof(k_menu_items[items_num].str));
> +
> +	curses_menu_items[items_num] = new_item(
> +			k_menu_items[items_num].str,
> +			k_menu_items[items_num].str);
> +	set_item_userptr(curses_menu_items[items_num],
> +			&k_menu_items[items_num]);
> +	items_num++;
> +	curses_menu_items[items_num] = NULL;
> +}
> +
> +/* very hackish. adds a string to the last item added */
> +void item_add_str(const char *fmt, ...)
> +{
> +	va_list ap;
> +	int index = items_num-1;
> +	char new_str[256];
> +	char tmp_str[256];
> +
> +	if (index < 0)
> +		return;
> +
> +	va_start(ap, fmt);
> +	vsnprintf(new_str, sizeof(new_str), fmt, ap);
> +	va_end(ap);
> +	snprintf(tmp_str, sizeof(tmp_str), "%s%s",
> +		k_menu_items[index].str, new_str);
> +	if (make_hot(k_menu_items[index].str,
> +		sizeof(k_menu_items[index].str), tmp_str, index) != 0)
> +		strncpy(k_menu_items[index].str,
> +			tmp_str,
> +			sizeof(k_menu_items[index].str));
> +
> +	free_item(curses_menu_items[index]);
> +	curses_menu_items[index] = new_item(
> +			k_menu_items[index].str,
> +			k_menu_items[index].str);
> +	set_item_userptr(curses_menu_items[index],
> +			&k_menu_items[index]);
> +}
> +
> +/* get the tag of the currently selected item */
> +char item_tag(void)
> +{
> +	ITEM *cur;
> +	struct mitem *mcur;
> +
> +	cur = current_item(curses_menu);
> +	if (cur == NULL)
> +		return 0;
> +	mcur = (struct mitem *) item_userptr(cur);
> +	return mcur->tag;
> +}
> +
> +int curses_item_index(void)
> +{
> +	return	item_index(current_item(curses_menu));
> +}
> +
> +void *item_data(void)
> +{
> +	ITEM *cur;
> +	struct mitem *mcur;
> +
> +	cur = current_item(curses_menu);
> +	mcur = (struct mitem *) item_userptr(cur);
> +	return mcur->usrptr;
> +
> +}
> +
> +int item_is_tag(char tag)
> +{
> +	return item_tag() == tag;
> +}
> +
> +static char filename[PATH_MAX+1];
> +const char *set_config_filename(const char *config_filename)
> +{
> +	static char menu_backtitle[PATH_MAX+128];
> +	int size;
> +	struct symbol *sym;
> +
> +	sym = sym_lookup("KERNELVERSION", 0);
> +	sym_calc_value(sym);
> +	size = snprintf(menu_backtitle, sizeof(menu_backtitle),
> +			_("%s - Linux Kernel v%s Configuration"),
> +			config_filename, sym_get_string_value(sym));
> +	if (size >= sizeof(menu_backtitle))
> +		menu_backtitle[sizeof(menu_backtitle)-1] = '\0';
> +
> +	size = snprintf(filename, sizeof(filename), "%s", config_filename);
> +	if (size >= sizeof(filename))
> +		filename[sizeof(filename)-1] = '\0';
> +	return menu_backtitle;
> +}
> +
> +
> +static void search_conf(void)
> +{
> +	struct symbol **sym_arr;
> +	struct gstr res;
> +	char dialog_input_result[100];
> +	char *dialog_input;
> +	int dres;
> +again:
> +	dres = dialog_inputbox(main_window,
> +			_("Search Configuration Parameter"),
> +		      _("Enter CONFIG_ (sub)string to search for "
> +			"(with or without \"CONFIG\")"),
> +		      "", dialog_input_result, 99);
> +	switch (dres) {
> +	case 0:
> +		break;
> +	case 1:
> +		show_scroll_win(main_window,
> +			_("Search Configuration"), search_help);
> +		goto again;
> +	default:
> +		return;
> +	}
> +
> +	/* strip CONFIG_ if necessary */
> +	dialog_input = dialog_input_result;
> +	if (strncasecmp(dialog_input_result, "CONFIG_", 7) == 0)
> +		dialog_input += 7;
> +
> +	sym_arr = sym_re_search(dialog_input);
> +	res = get_relations_str(sym_arr);
> +	free(sym_arr);
> +	show_scroll_win(main_window,
> +		_("Search Results"), str_get(&res));
> +	str_free(&res);
> +}
> +
> +
> +static void build_conf(struct menu *menu)
> +{
> +	struct symbol *sym;
> +	struct property *prop;
> +	struct menu *child;
> +	int type, tmp, doint = 2;
> +	tristate val;
> +	char ch;
> +
> +
> +	if (!menu_is_visible(menu))
> +		return;
> +
> +	sym = menu->sym;
> +	prop = menu->prompt;
> +	if (!sym) {
> +		if (prop && menu != current_menu) {
> +			const char *prompt = menu_get_prompt(menu);
> +			switch (prop->type) {
> +			case P_MENU:
> +				child_count++;
> +				prompt = _(prompt);
> +				if (single_menu_mode) {
> +					item_make(menu, 'm',
> +						"%s%*c%s",
> +						menu->data ? "-->" : "++>",
> +						indent + 1, ' ', prompt);
> +				} else
> +					item_make(menu, 'm',
> +							"   %*c%s  --->",
> +							indent + 1,
> +							' ', prompt);
> +
> +				if (single_menu_mode && menu->data)
> +					goto conf_childs;
> +				return;
> +			case P_COMMENT:
> +				if (prompt) {
> +					child_count++;
> +					item_make(menu, ':',
> +							"   %*c*** %s ***",
> +							indent + 1, ' ',
> +							_(prompt));
> +				}
> +				break;
> +			default:
> +				if (prompt) {
> +					child_count++;
> +					item_make(menu, ':', "---%*c%s",
> +							indent + 1, ' ',
> +							_(prompt));
> +				}
> +			}
> +		} else
> +			doint = 0;
> +		goto conf_childs;
> +	}
> +
> +	type = sym_get_type(sym);
> +	if (sym_is_choice(sym)) {
> +		struct symbol *def_sym = sym_get_choice_value(sym);
> +		struct menu *def_menu = NULL;
> +
> +		child_count++;
> +		for (child = menu->list; child; child = child->next) {
> +			if (menu_is_visible(child) && child->sym == def_sym)
> +				def_menu = child;
> +		}
> +
> +		val = sym_get_tristate_value(sym);
> +		if (sym_is_changable(sym)) {
> +			switch (type) {
> +			case S_BOOLEAN:
> +				item_make(menu, 't', "[%c]",
> +						val == no ? ' ' : '*');
> +				break;
> +			case S_TRISTATE:
> +				switch (val) {
> +				case yes:
> +					ch = '*';
> +					break;
> +				case mod:
> +					ch = 'M';
> +					break;
> +				default:
> +					ch = ' ';
> +					break;
> +				}
> +				item_make(menu, 't', "<%c>", ch);
> +				break;
> +			}
> +		} else {
> +			item_make(menu, def_menu ? 't' : ':', "   ");
> +		}
> +
> +		item_add_str("%*c%s", indent + 1,
> +				' ', _(menu_get_prompt(menu)));
> +		if (val == yes) {
> +			if (def_menu) {
> +				item_add_str(" (%s)",
> +					_(menu_get_prompt(def_menu)));
> +				item_add_str("  --->");
> +				if (def_menu->list) {
> +					indent += 2;
> +					build_conf(def_menu);
> +					indent -= 2;
> +				}
> +			}
> +			return;
> +		}
> +	} else {
> +		if (menu == current_menu) {
> +			item_make(menu, ':',
> +				"---%*c%s", indent + 1,
> +				' ', _(menu_get_prompt(menu)));
> +			goto conf_childs;
> +		}
> +		child_count++;
> +		val = sym_get_tristate_value(sym);
> +		if (sym_is_choice_value(sym) && val == yes) {
> +			item_make(menu, ':', "   ");
> +		} else {
> +			switch (type) {
> +			case S_BOOLEAN:
> +				if (sym_is_changable(sym))
> +					item_make(menu, 't', "[%c]",
> +						val == no ? ' ' : '*');
> +				else
> +					item_make(menu, 't', "-%c-",
> +						val == no ? ' ' : '*');
> +				break;
> +			case S_TRISTATE:
> +				switch (val) {
> +				case yes:
> +					ch = '*';
> +					break;
> +				case mod:
> +					ch = 'M';
> +					break;
> +				default:
> +					ch = ' ';
> +					break;
> +				}
> +				if (sym_is_changable(sym)) {
> +					if (sym->rev_dep.tri == mod)
> +						item_make(menu,
> +							't', "{%c}", ch);
> +					else
> +						item_make(menu,
> +							't', "<%c>", ch);
> +				} else
> +					item_make(menu, 't', "-%c-", ch);
> +				break;
> +			default:
> +				tmp = 2 + strlen(sym_get_string_value(sym));
> +				item_make(menu, 's', "(%s)",
> +					sym_get_string_value(sym));
> +				tmp = indent - tmp + 4;
> +				if (tmp < 0)
> +					tmp = 0;
> +				item_add_str("%*c%s%s", tmp, ' ',
> +					_(menu_get_prompt(menu)),
> +					(sym_has_value(sym) ||
> +					 !sym_is_changable(sym)) ? "" :
> +					   _(" (NEW)"));
> +				goto conf_childs;
> +			}
> +		}
> +		item_add_str("%*c%s%s", indent + 1, ' ',
> +			_(menu_get_prompt(menu)),
> +			(sym_has_value(sym) || !sym_is_changable(sym)) ?
> +			  "" : _(" (NEW)"));
> +		if (menu->prompt->type == P_MENU) {
> +			item_add_str("  --->");
> +			return;
> +		}
> +	}
> +
> +conf_childs:
> +	indent += doint;
> +	for (child = menu->list; child; child = child->next)
> +		build_conf(child);
> +	indent -= doint;
> +}
> +
> +static void reset_menu(void)
> +{
> +	unpost_menu(curses_menu);
> +	unpost_menu(btns_menu);
> +	clean_items();
> +}
> +
> +/* adjust the menu to show this item.
> + * prefer not to scroll the menu if possible*/
> +static void center_item(int selected_index)
> +{
> +	int toprow;
> +	int maxy, maxx;
> +
> +	scale_menu(curses_menu, &maxy, &maxx);
> +	toprow = top_row(curses_menu);
> +	if (selected_index >= toprow && selected_index < toprow+maxy) {
> +		/* we can only move the selected item. no need to scroll */
> +		set_current_item(curses_menu,
> +			curses_menu_items[selected_index]);
> +	} else {
> +		toprow = max(selected_index-maxy/2, 0);
> +		if (toprow >= item_count(curses_menu)-maxy)
> +			toprow = item_count(curses_menu)-mwin_max_lines;
> +		set_top_row(curses_menu, toprow);
> +		set_current_item(curses_menu,
> +			curses_menu_items[selected_index]);
> +	}
> +	post_menu(curses_menu);
> +	refresh_all_windows(main_window);
> +}
> +
> +/* this function assumes reset_menu has been called before */
> +static void show_menu(const char *prompt, const char *instructions,
> +		int selected_index)
> +{
> +	int maxx, maxy;
> +	WINDOW *menu_window;
> +
> +	current_instructions = instructions;
> +
> +	wclear(main_window);
> +
> +	wattrset(main_window, attributes[MAIN_MENU_BOX]);
> +	box(main_window, 0, 0);
> +	wattrset(main_window, attributes[MAIN_MENU_HEADING]);
> +	mvwprintw(main_window, 0, 3, " %s ", prompt);
> +
> +	set_menu_items(curses_menu, curses_menu_items);
> +
> +	/* position the menu at the middle of the screen */
> +	scale_menu(curses_menu, &maxy, &maxx);
> +	maxx = min(maxx, mwin_max_cols);
> +	menu_window = derwin(main_window,
> +			mwin_max_lines,
> +			maxx,
> +			2,
> +			(COLS-maxx)/2);
> +	keypad(menu_window, TRUE);
> +	set_menu_win(curses_menu, menu_window);
> +	set_menu_sub(curses_menu, menu_window);
> +
> +	/* must reassert this after changing items, otherwise returns to a
> +	 * default of 16
> +	 */
> +	set_menu_format(curses_menu, mwin_max_lines, 1);
> +	center_item(selected_index);
> +
> +	/* move lower menu btn to <select> */
> +	set_current_item(btns_menu, btns_items[BTN_SELECT]);
> +
> +	/* Post the menu */
> +	post_menu(curses_menu);
> +	post_menu(btns_menu);
> +	refresh_all_windows(main_window);
> +}
> +
> +
> +static void conf(struct menu *menu)
> +{
> +	char pattern[256];
> +	struct menu *submenu;
> +	const char *prompt = menu_get_prompt(menu);
> +	struct symbol *sym;
> +	struct menu *active_menu = NULL;
> +	int res;
> +	int current_index = 0;
> +
> +	bzero(pattern, sizeof(pattern));
> +
> +	while (1) {
> +		reset_menu();
> +		current_menu = menu;
> +		build_conf(menu);
> +		if (!child_count)
> +			break;
> +
> +		if (menu == &rootmenu) {
> +			item_make(NULL, 0, "--- ");
> +			item_make(NULL, 'L',
> +				_("    Load an Alternate Configuration File"));
> +			item_make(NULL, 'S',
> +				_("    Save an Alternate Configuration File"));
> +		}
> +
> +		show_menu(prompt ? _(prompt) : _("Main Menu"),
> +				_(menu_instructions),
> +				current_index);
> +		while ((res = wgetch(menu_win(curses_menu)))) {
> +			switch (res) {
> +			case KEY_DOWN:
> +				menu_driver(curses_menu, REQ_DOWN_ITEM);
> +				break;
> +			case KEY_UP:
> +				menu_driver(curses_menu, REQ_UP_ITEM);
> +				break;
> +			case KEY_LEFT:
> +				menu_driver(btns_menu, REQ_LEFT_ITEM);
> +				break;
> +			case KEY_RIGHT:
> +			case '\t':
> +				menu_driver(btns_menu, REQ_RIGHT_ITEM);
> +				break;
> +			case KEY_NPAGE:
> +				menu_driver(curses_menu, REQ_SCR_DPAGE);
> +				break;
> +			case KEY_PPAGE:
> +				menu_driver(curses_menu, REQ_SCR_UPAGE);
> +				break;
> +			}
> +			if (res == 10 || res == 27 ||  res == 'h' ||
> +			    res == 32 || res == 'n' || res == 'y' ||
> +			    res == 'm' || res == '/' || res == '?')
> +				break;
> +			else if (canbhot(res)) {
> +				/* check for hot keys: */
> +				int tmp = get_next_hot(res);
> +				if (tmp != -1)
> +					center_item(tmp);
> +			}
> +			refresh_all_windows(main_window);
> +		}
> +
> +		/* update selected btn */
> +		update_current_btn();
> +		refresh_all_windows(main_window);
> +
> +		/* if ESC or btn_exit */
> +		if (res == 27 || (res == 10 && current_btn == BTN_EXIT))
> +			break;
> +
> +		/* remember location in the menu */
> +		current_index = curses_item_index();
> +
> +		if (!item_tag())
> +			continue;
> +
> +		submenu = (struct menu *) item_data();
> +		active_menu = (struct menu *)item_data();
> +		if (submenu)
> +			sym = submenu->sym;
> +		else
> +			sym = NULL;
> +
> +		switch (res) {
> +		case 10: /* ENTER WAS PRESSED */
> +			/* check for btn help!! */
> +			if (current_btn == BTN_HELP) {
> +				if (sym)
> +					show_help(submenu);
> +				else
> +					show_scroll_win(main_window,
> +						_("README"),
> +						_(nconf_readme));
> +				break;
> +			} else if (current_btn == BTN_INSTS) {
> +				show_scroll_win(main_window,
> +					_("Instructions"),
> +					_(current_instructions));
> +				break;
> +			}
> +
> +			switch (item_tag()) {
> +			case 'm':
> +				if (single_menu_mode)
> +					submenu->data =
> +						(void *) (long) !submenu->data;
> +				else
> +					conf(submenu);
> +				break;
> +			case 't':
> +				if (sym_is_choice(sym) &&
> +					sym_get_tristate_value(sym) == yes)
> +						conf_choice(submenu);
> +				else if (submenu->prompt->type == P_MENU)
> +					conf(submenu);
> +				break;
> +			case 's':
> +				conf_string(submenu);
> +				break;
> +			case 'L':
> +				conf_load();
> +				break;
> +			case 'S':
> +				conf_save();
> +				break;
> +			}
> +			break;
> +		case 'h':
> +		case '?':
> +			if (sym)
> +				show_help(submenu);
> +			else
> +				show_scroll_win(main_window,
> +					_("README"), _(nconf_readme));
> +			break;
> +		case 'y':
> +			if (item_is_tag('t')) {
> +				if (sym_set_tristate_value(sym, yes))
> +					break;
> +				if (sym_set_tristate_value(sym, mod))
> +					btn_dialog(main_window, setmod_text, 0);
> +			}
> +			break;
> +		case 'n':
> +			if (item_is_tag('t'))
> +				sym_set_tristate_value(sym, no);
> +			break;
> +		case 'm':
> +			if (item_is_tag('t'))
> +				sym_set_tristate_value(sym, mod);
> +			break;
> +		case ' ':
> +			if (item_is_tag('t'))
> +				sym_toggle_tristate_value(sym);
> +			else if (item_is_tag('m'))
> +				conf(submenu);
> +			break;
> +		case '/':
> +			search_conf();
> +			break;
> +		}
> +	}
> +}
> +
> +static void show_help(struct menu *menu)
> +{
> +	struct gstr help = str_new();
> +	struct symbol *sym = menu->sym;
> +
> +	if (menu_has_help(menu)) {
> +		if (sym->name) {
> +			str_printf(&help, "CONFIG_%s:\n\n", sym->name);
> +			str_append(&help, _(menu_get_help(menu)));
> +			str_append(&help, "\n");
> +		}
> +	} else {
> +		str_append(&help, nohelp_text);
> +	}
> +	get_symbol_str(&help, sym);
> +	show_scroll_win(main_window, _(menu_get_prompt(menu)), str_get(&help));
> +	str_free(&help);
> +}
> +
> +static void conf_choice(struct menu *menu)
> +{
> +	const char *prompt = _(menu_get_prompt(menu));
> +	struct menu *child;
> +	struct symbol *active;
> +	int selected_index = 0;
> +	int res, i = 0;
> +
> +	active = sym_get_choice_value(menu->sym);
> +	/* this is mostly duplicated from the conf() function. */
> +	while (1) {
> +		reset_menu();
> +
> +		for (i = 0, child = menu->list; child; child = child->next) {
> +			if (!menu_is_visible(child))
> +				continue;
> +
> +			if (child->sym == sym_get_choice_value(menu->sym))
> +				item_make(child, ':', "<X> %s",
> +					_(menu_get_prompt(child)));
> +			else
> +				item_make(child, ':', "    %s",
> +					_(menu_get_prompt(child)));
> +			if (child->sym == active)
> +				selected_index = i;
> +			i++;
> +		}
> +		show_menu(prompt ? _(prompt) : _("Choice Menu"),
> +				_(radiolist_instructions),
> +				selected_index);
> +		while ((res = wgetch(menu_win(curses_menu)))) {
> +			switch (res) {
> +			case KEY_DOWN:
> +				menu_driver(curses_menu, REQ_DOWN_ITEM);
> +				break;
> +			case KEY_UP:
> +				menu_driver(curses_menu, REQ_UP_ITEM);
> +				break;
> +			case KEY_LEFT:
> +				menu_driver(btns_menu, REQ_LEFT_ITEM);
> +				break;
> +			case KEY_RIGHT:
> +			case '\t':
> +				menu_driver(btns_menu, REQ_RIGHT_ITEM);
> +				break;
> +			case KEY_NPAGE:
> +				menu_driver(curses_menu, REQ_SCR_DPAGE);
> +				break;
> +			case KEY_PPAGE:
> +				menu_driver(curses_menu, REQ_SCR_UPAGE);
> +				break;
> +			}
> +			if (res == 10 || res == 27 || res == 'h'
> +				|| res == ' ' || res == 's' || res == '?')
> +				break;
> +			else if (canbhot(res)) {
> +				/* check for hot keys: */
> +				int tmp = get_next_hot(res);
> +				if (tmp != -1)
> +					center_item(tmp);
> +			}
> +			refresh_all_windows(main_window);
> +		}
> +		/* update selected btn */
> +		update_current_btn();
> +
> +		/* if ESC or btn_exit */
> +		if (res == 27 || (res == 10 && current_btn == BTN_EXIT))
> +			return;
> +
> +		child = item_data();
> +		switch (res) {
> +		case ' ':
> +		case 's':
> +		case  10:
> +			if (current_btn == BTN_HELP) {
> +				show_help(child);
> +				break;
> +			} else if (current_btn == BTN_INSTS) {
> +				show_scroll_win(main_window, _("Instructions"),
> +						_(current_instructions));
> +				break;
> +			}
> +			sym_set_tristate_value(child->sym, yes);
> +			return;
> +		case 'h':
> +		case '?':
> +			show_help(child);
> +			active = child->sym;
> +			break;
> +		case KEY_EXIT:
> +			return;
> +		}
> +	}
> +}
> +
> +static void conf_string(struct menu *menu)
> +{
> +	const char *prompt = menu_get_prompt(menu);
> +	char dialog_input_result[256];
> +
> +	while (1) {
> +		int res;
> +		const char *heading;
> +
> +		switch (sym_get_type(menu->sym)) {
> +		case S_INT:
> +			heading = _(inputbox_instructions_int);
> +			break;
> +		case S_HEX:
> +			heading = _(inputbox_instructions_hex);
> +			break;
> +		case S_STRING:
> +			heading = _(inputbox_instructions_string);
> +			break;
> +		default:
> +			heading = _("Internal nconf error!");
> +		}
> +		res = dialog_inputbox(main_window,
> +				prompt ? _(prompt) : _("Main Menu"),
> +			      heading,
> +			      sym_get_string_value(menu->sym),
> +			      dialog_input_result,
> +			      sizeof(dialog_input_result));
> +		switch (res) {
> +		case 0:
> +			if (sym_set_string_value(menu->sym,
> +				dialog_input_result))
> +					return;
> +			btn_dialog(main_window,
> +				_("You have made an invalid entry."), 0);
> +			break;
> +		case 1:
> +			show_help(menu);
> +			break;
> +		case KEY_EXIT:
> +			return;
> +		}
> +	}
> +}
> +
> +static void conf_load(void)
> +{
> +	char dialog_input_result[256];
> +	while (1) {
> +		int res;
> +		res = dialog_inputbox(main_window,
> +				NULL, load_config_text,
> +				filename,
> +				dialog_input_result,
> +				sizeof(dialog_input_result));
> +		switch (res) {
> +		case 0:
> +			if (!dialog_input_result[0])
> +				return;
> +			if (!conf_read(dialog_input_result)) {
> +				set_config_filename(dialog_input_result);
> +				sym_set_change_count(1);
> +				return;
> +			}
> +			btn_dialog(main_window, _("File does not exist!"), 0);
> +			break;
> +		case 1:
> +			show_scroll_win(main_window,
> +				_("Load Alternate Configuration"),
> +				load_config_help);
> +			break;
> +		case KEY_EXIT:
> +			return;
> +		}
> +	}
> +}
> +
> +static void conf_save(void)
> +{
> +	char dialog_input_result[256];
> +	while (1) {
> +		int res;
> +		res = dialog_inputbox(main_window,
> +				NULL, save_config_text,
> +				filename,
> +				dialog_input_result,
> +				sizeof(dialog_input_result));
> +		switch (res) {
> +		case 0:
> +			if (!dialog_input_result[0])
> +				return;
> +			if (!conf_write(dialog_input_result)) {
> +				set_config_filename(dialog_input_result);
> +				return;
> +			}
> +			btn_dialog(main_window, _("Can't create file! "
> +				"Probably a nonexistent directory."), 0);
> +			break;
> +		case 1:
> +			show_scroll_win(main_window,
> +				_("Save Alternate Configuration"),
> +				save_config_help);
> +			break;
> +		case KEY_EXIT:
> +			return;
> +		}
> +	}
> +}
> +
> +void setup_windows(void)
> +{
> +	int btn_lines, btn_cols;
> +
> +	if (main_window != NULL)
> +		delwin(main_window);
> +
> +	/* set up the menu and menu window */
> +	main_window = newwin(LINES-2, COLS-2, 2, 1);
> +	keypad(main_window, TRUE);
> +	mwin_max_lines = LINES-7;
> +	mwin_max_cols = COLS-6;
> +
> +	scale_menu(btns_menu, &btn_lines, &btn_cols);
> +	btn_cols = min(btn_cols, COLS-16);
> +	btns_window = derwin(main_window,
> +		1,
> +		btn_cols,
> +		LINES-4,
> +		(COLS-2-btn_cols)/2);
> +	set_menu_win(btns_menu, btns_window);
> +	set_menu_sub(btns_menu, btns_window);
> +
> +	/* panels order is from bottom to top */
> +	new_panel(main_window);
> +}
> +
> +int main(int ac, char **av)
> +{
> +	char *mode;
> +	const char *title;
> +	int res;
> +	int i;
> +
> +
> +	setlocale(LC_ALL, "");
> +	bindtextdomain(PACKAGE, LOCALEDIR);
> +	textdomain(PACKAGE);
> +
> +	conf_parse(av[1]);
> +	conf_read(NULL);
> +
> +	mode = getenv("NCONFIG_MODE");
> +	if (mode) {
> +		if (!strcasecmp(mode, "single_menu"))
> +			single_menu_mode = 1;
> +	}
> +
> +	/* Initialize curses */
> +	initscr();
> +	cbreak();
> +	noecho();
> +	raw();
> +	keypad(stdscr, TRUE);
> +	clear();
> +	curs_set(0);
> +
> +	if (COLS < 60 || LINES < 20) {
> +		endwin();
> +		printf("Your terminal should have at "
> +			"least 20 lines and 60 columns\n");
> +		return 1;
> +	}
> +
> +	/* set color theme */
> +	set_colors();
> +
> +	notimeout(stdscr, TRUE);
> +	/* notimeout does not seem to work!! */
> +	ESCDELAY = 0;
> +
> +	title = set_config_filename(conf_get_configname());
> +	print_in_middle(stdscr, 1, 0, COLS, title, attributes[MAIN_HEADING]);
> +
> +	/* set btns menu */
> +	for (i = 0; i < BTN_MAX; i++)
> +		btns_items[i] = new_item(btns[i], "");
> +	btns_items[BTN_MAX] = NULL;
> +	btns_menu = new_menu(btns_items);
> +	set_menu_format(btns_menu, 1, BTN_MAX);
> +	menu_opts_off(btns_menu, O_SHOWDESC);
> +	menu_opts_on(btns_menu, O_SHOWMATCH);
> +	menu_opts_on(btns_menu, O_ONEVALUE);
> +	menu_opts_off(btns_menu, O_NONCYCLIC);
> +	set_menu_mark(btns_menu, "");
> +	set_menu_fore(btns_menu, attributes[BOTTOM_MENU_FORE]);
> +	set_menu_back(btns_menu, attributes[BOTTOM_MENU_BACK]);
> +
> +	curses_menu = new_menu(curses_menu_items);
> +	menu_opts_off(curses_menu, O_SHOWDESC);
> +	menu_opts_on(curses_menu, O_SHOWMATCH);
> +	menu_opts_on(curses_menu, O_ONEVALUE);
> +	menu_opts_on(curses_menu, O_NONCYCLIC);
> +	set_menu_mark(curses_menu, " ");
> +	set_menu_fore(curses_menu, attributes[MAIN_MENU_FORE]);
> +	set_menu_back(curses_menu, attributes[MAIN_MENU_BACK]);
> +
> +	setup_windows();
> +
> +	do {
> +		conf(&rootmenu);
> +		clear();
> +		if (conf_get_changed())
> +			res = btn_dialog(main_window,
> +					_("Do you wish to save your "
> +					     "new kernel configuration?\n"
> +					     "<ESC><ESC> to continue."),
> +					2,
> +					"   <save>   ",
> +					"<don't save>");
> +		else
> +			res = -1;
> +	} while (res == KEY_EXIT);
> +	unpost_menu(curses_menu);
> +	unpost_menu(btns_menu);
> +	free_menu(curses_menu);
> +	free_menu(btns_menu);
> +	delwin(main_window);
> +	endwin();
> +
> +
> +	switch (res) {
> +	case 0:
> +		if (conf_write(filename)) {
> +			fprintf(stderr, _("\n\n"
> +			"Error during writing of the kernel configuration.\n"
> +			"Your kernel configuration changes were NOT saved."
> +			"\n\n"));
> +			return 1;
> +		}
> +	case -1:
> +		printf(_("\n\n"
> +		"*** End of Linux kernel configuration.\n"
> +		"*** Execute 'make' to build the kernel or try 'make help'."
> +		"\n\n"));
> +		break;
> +	default:
> +		fprintf(stderr, _("\n\n"
> +			"Your kernel configuration changes were NOT saved."
> +			"\n\n"));
> +	}
> +
> +	return 0;
> +}
> +
> diff --git a/scripts/kconfig/nconf.gui.c b/scripts/kconfig/nconf.gui.c
> new file mode 100644
> index 0000000..dd39628
> --- /dev/null
> +++ b/scripts/kconfig/nconf.gui.c
> @@ -0,0 +1,653 @@
> +/*
> + * Copyright (C) 2008 Nir Tzachar <nir.tzachar@xxxxxxxxx?
> + * Released under the terms of the GNU GPL v2.0.
> + *
> + * Derived from menuconfig.
> + *
> + */
> +#include "nconf.h"
> +
> +/* a list of all the different widgets we use */
> +attributes_t attributes[ATTR_MAX+1] = {0};
> +
> +/* available colors:
> +   COLOR_BLACK   0
> +   COLOR_RED     1
> +   COLOR_GREEN   2
> +   COLOR_YELLOW  3
> +   COLOR_BLUE    4
> +   COLOR_MAGENTA 5
> +   COLOR_CYAN    6
> +   COLOR_WHITE   7
> +   */
> +void set_normal_colors(void)
> +{
> +	init_pair(NORMAL, COLOR_WHITE, COLOR_BLACK);
> +	init_pair(MAIN_HEADING, COLOR_MAGENTA, COLOR_BLACK);
> +
> +	/* FORE is for the selected item */
> +	init_pair(MAIN_MENU_FORE, COLOR_CYAN, COLOR_BLACK);
> +	/* BACK for all the rest */
> +	init_pair(MAIN_MENU_BACK, COLOR_CYAN, COLOR_BLACK);
> +	init_pair(MAIN_MENU_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(MAIN_MENU_BOX, COLOR_YELLOW, COLOR_BLACK);
> +
> +	init_pair(INSTS_WIN_TEXT, COLOR_BLUE, COLOR_BLACK);
> +	init_pair(INSTS_WIN_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(INSTS_WIN_BOX, COLOR_YELLOW, COLOR_BLACK);
> +
> +	init_pair(SCROLLWIN_TEXT, COLOR_WHITE, COLOR_BLACK);
> +	init_pair(SCROLLWIN_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(SCROLLWIN_BOX, COLOR_YELLOW, COLOR_BLACK);
> +
> +	init_pair(BOTTOM_MENU_FORE, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(BOTTOM_MENU_BACK, COLOR_WHITE, COLOR_BLACK);
> +
> +	init_pair(DIALOG_TEXT, COLOR_RED, COLOR_BLACK);
> +	init_pair(DIALOG_BOX, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(DIALOG_MENU_FORE, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(DIALOG_MENU_BACK, COLOR_WHITE, COLOR_BLACK);
> +
> +	init_pair(INPUT_BOX, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(INPUT_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(INPUT_TEXT, COLOR_RED, COLOR_BLACK);
> +	init_pair(INPUT_FIELD, COLOR_BLUE, COLOR_BLACK);
> +}
> +
> +void rgb_colors(void)
> +{
> +	int start_color = COLOR_BLACK+1;
> +
> +/* make a new color pair, given foureground and background rgb vals
> + * values for rgb are from 0 to 1000 */
> +#define mkcolor(attr, fr, fg, fb, br, bg, bb) {\
> +	init_color(start_color, fr, fg, fb);\
> +	init_color(start_color+1, br, bg, bb);\
> +	init_pair(attr, start_color, start_color+1); }
> +
> +	mkcolor(NORMAL, 1000, 0, 0, 0, 0, 0);
> +	init_pair(MAIN_HEADING, COLOR_MAGENTA, COLOR_BLACK);
> +
> +	/* FORE is for the selected item */
> +	init_pair(MAIN_MENU_FORE, COLOR_CYAN, COLOR_BLACK);
> +	/* BACK for all the rest */
> +	init_pair(MAIN_MENU_BACK, COLOR_CYAN, COLOR_BLACK);
> +	init_pair(MAIN_MENU_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(MAIN_MENU_BOX, COLOR_YELLOW, COLOR_BLACK);
> +
> +	init_pair(INSTS_WIN_TEXT, COLOR_BLUE, COLOR_BLACK);
> +	init_pair(INSTS_WIN_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(INSTS_WIN_BOX, COLOR_YELLOW, COLOR_BLACK);
> +
> +	init_pair(SCROLLWIN_TEXT, COLOR_RED, COLOR_BLACK);
> +	init_pair(SCROLLWIN_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(SCROLLWIN_BOX, COLOR_YELLOW, COLOR_BLACK);
> +
> +	init_pair(BOTTOM_MENU_FORE, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(BOTTOM_MENU_BACK, COLOR_WHITE, COLOR_BLACK);
> +
> +	init_pair(DIALOG_TEXT, COLOR_RED, COLOR_BLACK);
> +	init_pair(DIALOG_BOX, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(DIALOG_MENU_FORE, COLOR_MAGENTA, COLOR_BLACK);
> +	init_pair(DIALOG_MENU_BACK, COLOR_WHITE, COLOR_BLACK);
> +
> +	init_pair(INPUT_BOX, COLOR_YELLOW, COLOR_BLACK);
> +	init_pair(INPUT_HEADING, COLOR_GREEN, COLOR_BLACK);
> +	init_pair(INPUT_TEXT, COLOR_RED, COLOR_BLACK);
> +	init_pair(INPUT_FIELD, COLOR_BLUE, COLOR_BLACK);
> +
> +}
> +
> +/* available attributes:
> +   A_NORMAL        Normal display (no highlight)
> +   A_STANDOUT      Best highlighting mode of the terminal.
> +   A_UNDERLINE     Underlining
> +   A_REVERSE       Reverse video
> +   A_BLINK         Blinking
> +   A_DIM           Half bright
> +   A_BOLD          Extra bright or bold
> +   A_PROTECT       Protected mode
> +   A_INVIS         Invisible or blank mode
> +   A_ALTCHARSET    Alternate character set
> +   A_CHARTEXT      Bit-mask to extract a character
> +   COLOR_PAIR(n)   Color-pair number n
> +   */
> +void normal_color_theme(void)
> +{
> +/* automatically add color... */
> +#define mkattr(name, attr) { attributes[name] = attr | COLOR_PAIR(name); }
> +	mkattr(NORMAL, NORMAL);
> +	mkattr(MAIN_HEADING, A_BOLD | A_UNDERLINE);
> +
> +	mkattr(MAIN_MENU_FORE, A_STANDOUT);
> +	mkattr(MAIN_MENU_BACK, A_NORMAL);
> +	mkattr(MAIN_MENU_HEADING, A_BOLD);
> +	mkattr(MAIN_MENU_BOX, A_NORMAL);
> +
> +	mkattr(INSTS_WIN_TEXT, A_NORMAL);
> +	mkattr(INSTS_WIN_HEADING, A_BOLD);
> +	mkattr(INSTS_WIN_BOX, A_NORMAL);
> +
> +	mkattr(SCROLLWIN_TEXT, A_NORMAL);
> +	mkattr(SCROLLWIN_HEADING, A_BOLD);
> +	mkattr(SCROLLWIN_BOX, A_BOLD);
> +
> +	mkattr(BOTTOM_MENU_FORE, A_STANDOUT);
> +	mkattr(BOTTOM_MENU_BACK, A_BOLD);
> +
> +	mkattr(DIALOG_TEXT, A_BOLD);
> +	mkattr(DIALOG_BOX, A_BOLD);
> +	mkattr(DIALOG_MENU_FORE, A_STANDOUT);
> +	mkattr(DIALOG_MENU_BACK, A_BOLD);
> +
> +	mkattr(INPUT_BOX, A_BOLD);
> +	mkattr(INPUT_HEADING, A_BOLD);
> +	mkattr(INPUT_TEXT, A_NORMAL);
> +	mkattr(INPUT_FIELD, A_UNDERLINE);
> +}
> +
> +void no_colors_theme(void)
> +{
> +/* automatically add highlight, no color */
> +#define mkattrn(name, attr) { attributes[name] = attr; }
> +
> +	mkattrn(NORMAL, NORMAL);
> +	mkattrn(MAIN_HEADING, A_BOLD | A_UNDERLINE);
> +
> +	mkattrn(MAIN_MENU_FORE, A_STANDOUT);
> +	mkattrn(MAIN_MENU_BACK, A_NORMAL);
> +	mkattrn(MAIN_MENU_HEADING, A_BOLD);
> +	mkattrn(MAIN_MENU_BOX, A_NORMAL);
> +
> +	mkattrn(INSTS_WIN_TEXT, A_NORMAL);
> +	mkattrn(INSTS_WIN_HEADING, A_BOLD);
> +	mkattrn(INSTS_WIN_BOX, A_NORMAL);
> +
> +	mkattrn(SCROLLWIN_TEXT, A_NORMAL);
> +	mkattrn(SCROLLWIN_HEADING, A_BOLD);
> +	mkattrn(SCROLLWIN_BOX, A_BOLD);
> +
> +	mkattrn(BOTTOM_MENU_FORE, A_STANDOUT);
> +	mkattrn(BOTTOM_MENU_BACK, A_BOLD);
> +
> +	mkattrn(DIALOG_TEXT, A_NORMAL);
> +	mkattrn(DIALOG_BOX, A_BOLD);
> +	mkattrn(DIALOG_MENU_FORE, A_STANDOUT);
> +	mkattrn(DIALOG_MENU_BACK, A_BOLD);
> +
> +	mkattrn(INPUT_BOX, A_BOLD);
> +	mkattrn(INPUT_HEADING, A_BOLD);
> +	mkattrn(INPUT_TEXT, A_NORMAL);
> +	mkattrn(INPUT_FIELD, A_UNDERLINE);
> +}
> +
> +void set_colors()
> +{
> +	if (has_colors()) {
> +		start_color();
> +		if (can_change_color())
> +			rgb_colors();
> +		else
> +			set_normal_colors();
> +		normal_color_theme();
> +	} else {
> +		/* give deafults */
> +		no_colors_theme();
> +	}
> +}
> +
> +
> +/* this changes the windows attributes !!! */
> +void print_in_middle(WINDOW *win,
> +		int starty,
> +		int startx,
> +		int width,
> +		const char *string,
> +		chtype color)
> +{	int length, x, y;
> +	float temp;
> +
> +
> +	if (win == NULL)
> +		win = stdscr;
> +	getyx(win, y, x);
> +	if (startx != 0)
> +		x = startx;
> +	if (starty != 0)
> +		y = starty;
> +	if (width == 0)
> +		width = 80;
> +
> +	length = strlen(string);
> +	temp = (width - length) / 2;
> +	x = startx + (int)temp;
> +	wattrset(win, color);
> +	mvwprintw(win, y, x, "%s", string);
> +	refresh();
> +}
> +
> +int get_line_no(const char *text)
> +{
> +	int i;
> +	int total = 1;
> +
> +	if (!text)
> +		return 0;
> +
> +	for (i = 0; text[i] != '\0'; i++)
> +		if (text[i] == '\n')
> +			total++;
> +	return total;
> +}
> +
> +const char *get_line(const char *text, int line_no)
> +{
> +	int i;
> +	int lines = 0;
> +
> +	if (!text)
> +		return 0;
> +
> +	for (i = 0; text[i] != '\0' && lines < line_no; i++)
> +		if (text[i] == '\n')
> +			lines++;
> +	return text+i;
> +}
> +
> +int get_line_length(const char *line)
> +{
> +	int res = 0;
> +	while (*line != '\0' && *line != '\n') {
> +		line++;
> +		res++;
> +	}
> +	return res;
> +}
> +
> +/* print all lines to the window. */
> +void fill_window(WINDOW *win, const char *text)
> +{
> +	int x, y;
> +	int total_lines = get_line_no(text);
> +	int i;
> +
> +	getmaxyx(win, y, x);
> +	/* do not go over end of line */
> +	total_lines = min(total_lines, y);
> +	for (i = 0; i < total_lines; i++) {
> +		char tmp[x+10];
> +		const char *line = get_line(text, i);
> +		int len = get_line_length(line);
> +		strncpy(tmp, line, min(len, x));
> +		tmp[len] = '\0';
> +		mvwprintw(win, i, 0, tmp);
> +	}
> +}
> +
> +/* get the message, and buttons.
> + * each button must be a char*
> + * return the selected button
> + *
> + * this dialog is used for 2 different things:
> + * 1) show a text box, no buttons.
> + * 2) show a dialog, with horizontal buttons
> + */
> +int btn_dialog(WINDOW *main_window, const char *msg, int btn_num, ...)
> +{
> +	va_list ap;
> +	char *btn;
> +	int btns_width = 0;
> +	int msg_lines = 0;
> +	int msg_width = 0;
> +	int total_width;
> +	int win_rows = 0;
> +	WINDOW *win;
> +	WINDOW *msg_win;
> +	WINDOW *menu_win;
> +	PANEL *panel;
> +	MENU *menu;
> +	ITEM *btns[btn_num+1];
> +	int i, x, y;
> +	int res = -1;
> +
> +
> +	va_start(ap, btn_num);
> +	for (i = 0; i < btn_num; i++) {
> +		btn = va_arg(ap, char *);
> +		btns[i] = new_item(btn, "");
> +		btns_width += strlen(btn)+1;
> +	}
> +	va_end(ap);
> +	btns[btn_num] = NULL;
> +
> +	/* find the widest line of msg: */
> +	msg_lines = get_line_no(msg);
> +	for (i = 0; i < msg_lines; i++) {
> +		const char *line = get_line(msg, i);
> +		int len = get_line_length(line);
> +		if (msg_width < len)
> +			msg_width = len;
> +	}
> +
> +	total_width = max(msg_width, btns_width);
> +	/* place dialog in middle of screen */
> +	y = (LINES-(msg_lines+4))/2;
> +	x = (COLS-(total_width+4))/2;
> +
> +
> +	/* create the windows */
> +	if (btn_num > 0)
> +		win_rows = msg_lines+4;
> +	else
> +		win_rows = msg_lines+2;
> +
> +	win = newwin(win_rows, total_width+4, y, x);
> +	keypad(win, TRUE);
> +	menu_win = derwin(win, 1, btns_width, win_rows-2,
> +			1+(total_width+2-btns_width)/2);
> +	menu = new_menu(btns);
> +	msg_win = derwin(win, win_rows-2, msg_width, 1,
> +			1+(total_width+2-msg_width)/2);
> +
> +	set_menu_fore(menu, attributes[DIALOG_MENU_FORE]);
> +	set_menu_back(menu, attributes[DIALOG_MENU_BACK]);
> +
> +	wattrset(win, attributes[DIALOG_BOX]);
> +	box(win, 0, 0);
> +
> +	/* print message */
> +	wattrset(msg_win, attributes[DIALOG_TEXT]);
> +	fill_window(msg_win, msg);
> +
> +	set_menu_win(menu, win);
> +	set_menu_sub(menu, menu_win);
> +	set_menu_format(menu, 1, btn_num);
> +	menu_opts_off(menu, O_SHOWDESC);
> +	menu_opts_on(menu, O_SHOWMATCH);
> +	menu_opts_on(menu, O_ONEVALUE);
> +	menu_opts_on(menu, O_NONCYCLIC);
> +	set_menu_mark(menu, "");
> +	post_menu(menu);
> +
> +
> +	/* create panels */
> +	panel = new_panel(win);
> +
> +	touchwin(win);
> +	refresh_all_windows(main_window);
> +	while ((res = wgetch(win))) {
> +		switch (res) {
> +		case KEY_LEFT:
> +			menu_driver(menu, REQ_LEFT_ITEM);
> +			break;
> +		case KEY_RIGHT:
> +			menu_driver(menu, REQ_RIGHT_ITEM);
> +			break;
> +		case 10: /* ENTER */
> +			break;
> +		case 27: /* ESCAPE */
> +			break;
> +		}
> +		touchwin(win);
> +		refresh_all_windows(main_window);
> +
> +		if (res == 10) {
> +			res = item_index(current_item(menu));
> +			break;
> +		} else if (res == 27) {
> +			res = -1;
> +			break;
> +		}
> +	}
> +
> +	unpost_menu(menu);
> +	free_menu(menu);
> +	for (i = 0; i < btn_num; i++)
> +		free_item(btns[i]);
> +
> +	del_panel(panel);
> +	delwin(win);
> +	return res;
> +}
> +
> +int dialog_inputbox(WINDOW *main_window,
> +		const char *title, const char *prompt,
> +		const char *init, char *result, int result_len)
> +{
> +	int prompt_lines = 0;
> +	int prompt_width = 0;
> +	WINDOW *win;
> +	WINDOW *prompt_win;
> +	WINDOW *form_win;
> +	PANEL *panel;
> +	int i, x, y;
> +	int res = -1;
> +	int cursor_position = strlen(init);
> +
> +
> +	/* find the widest line of msg: */
> +	prompt_lines = get_line_no(prompt);
> +	for (i = 0; i < prompt_lines; i++) {
> +		const char *line = get_line(prompt, i);
> +		int len = get_line_length(line);
> +		prompt_width = max(prompt_width, len);
> +	}
> +
> +	if (title)
> +		prompt_width = max(prompt_width, strlen(title));
> +
> +	/* place dialog in middle of screen */
> +	y = (LINES-(prompt_lines+4))/2;
> +	x = (COLS-(prompt_width+4))/2;
> +
> +	strncpy(result, init, result_len);
> +
> +	/* create the windows */
> +	win = newwin(prompt_lines+6, prompt_width+7, y, x);
> +	prompt_win = derwin(win, prompt_lines+1, prompt_width, 2, 2);
> +	form_win = derwin(win, 1, prompt_width, prompt_lines+3, 2);
> +	keypad(form_win, TRUE);
> +	wattrset(form_win, attributes[INPUT_FIELD]);
> +
> +	wattrset(win, attributes[INPUT_BOX]);
> +	box(win, 0, 0);
> +	wattrset(win, attributes[INPUT_HEADING]);
> +	if (title)
> +		mvwprintw(win, 0, 3, "%s", title);
> +
> +	/* print message */
> +	wattrset(prompt_win, attributes[INPUT_TEXT]);
> +	fill_window(prompt_win, prompt);
> +
> +	mvwprintw(form_win, 0, 0, "%*s", prompt_width, " ");
> +	mvwprintw(form_win, 0, 0, "%s", result);
> +
> +	/* create panels */
> +	panel = new_panel(win);
> +
> +	/* show the cursor */
> +	curs_set(1);
> +
> +	touchwin(win);
> +	refresh_all_windows(main_window);
> +	while ((res = wgetch(form_win))) {
> +		int len = strlen(result);
> +		switch (res) {
> +		case 10: /* ENTER */
> +		case 27: /* ESCAPE */
> +		case '?':
> +			break;
> +		case 127:
> +		case KEY_BACKSPACE:
> +			if (cursor_position > 0) {
> +				memmove(&result[cursor_position-1],
> +					&result[cursor_position],
> +					len-cursor_position+1);
> +				cursor_position--;
> +			}
> +			break;
> +		case KEY_DC:
> +			if (cursor_position >= 0 && cursor_position < len) {
> +				memmove(&result[cursor_position],
> +					&result[cursor_position+1],
> +					len-cursor_position+1);
> +			}
> +			break;
> +		case KEY_UP:
> +		case KEY_RIGHT:
> +			if (cursor_position < len &&
> +			     cursor_position < min(result_len, prompt_width))
> +				cursor_position++;
> +			break;
> +		case KEY_DOWN:
> +		case KEY_LEFT:
> +			if (cursor_position > 0)
> +				cursor_position--;
> +			break;
> +		default:
> +			if ((isgraph(res) || isspace(res)) &&
> +			    len-2 < result_len) {
> +				/* insert the char at the proper position */
> +				memmove(&result[cursor_position+1],
> +					&result[cursor_position],
> +					len+1);
> +				result[cursor_position] = res;
> +				cursor_position++;
> +			} else {
> +				mvprintw(0, 0, "unknow key: %d\n", res);
> +			}
> +			break;
> +		}
> +		wmove(form_win, 0, 0);
> +		wclrtoeol(form_win);
> +		mvwprintw(form_win, 0, 0, "%*s", prompt_width, " ");
> +		mvwprintw(form_win, 0, 0, "%s", result);
> +		wmove(form_win, 0, cursor_position);
> +		touchwin(win);
> +		refresh_all_windows(main_window);
> +
> +		if (res == 10) {
> +			res = 0;
> +			break;
> +		} else if (res == 27) {
> +			res = KEY_EXIT;
> +			break;
> +		} else if (res == '?') {
> +			res = 1;
> +			break;
> +		}
> +	}
> +
> +	/* hide the cursor */
> +	curs_set(0);
> +	del_panel(panel);
> +	delwin(win);
> +	return res;
> +}
> +
> +/* refresh all windows in the correct order */
> +void refresh_all_windows(WINDOW *main_window)
> +{
> +	update_panels();
> +	touchwin(main_window);
> +	refresh();
> +}
> +
> +/* layman's scrollble window... */
> +void show_scroll_win(WINDOW *main_window,
> +		const char *title, 
> +		const char *text)
> +{
> +	int res;
> +	int total_lines = get_line_no(text);
> +	int x, y;
> +	int start_x = 0, start_y = 0;
> +	int text_lines = 0, text_cols = 0;
> +	int total_cols = 0;
> +	int win_cols = 0;
> +	int win_lines = 0;
> +	int i = 0;
> +	WINDOW *win;
> +	WINDOW *pad;
> +	PANEL *panel;
> +
> +	/* find the widest line of msg: */
> +	total_lines = get_line_no(text);
> +	for (i = 0; i < total_lines; i++) {
> +		const char *line = get_line(text, i);
> +		int len = get_line_length(line);
> +		total_cols = max(total_cols, len+2);
> +	}
> +
> +	/* create the pad */
> +	pad = newpad(total_lines+10, total_cols+10);
> +	wattrset(pad, attributes[SCROLLWIN_TEXT]);
> +	fill_window(pad, text);
> +
> +	win_lines = min(total_lines+2, LINES-2);
> +	win_cols = min(total_cols+2, COLS-2);
> +	text_lines = max(win_lines-2, 0);
> +	text_cols = max(win_cols-2, 0);
> +
> +	/* place window in middle of screen */
> +	y = (LINES-win_lines)/2;
> +	x = (COLS-win_cols)/2;
> +
> +	win = newwin(win_lines, win_cols, y, x);
> +	keypad(win, TRUE);
> +	/* show the help in the help window, and show the help panel */
> +	wattrset(win, attributes[SCROLLWIN_BOX]);
> +	box(win, 0, 0);
> +	wattrset(win, attributes[SCROLLWIN_HEADING]);
> +	mvwprintw(win, 0, 3, " %s ", title);
> +	panel = new_panel(win);
> +
> +	/* handle scrolling */
> +	do {
> +
> +		copywin(pad, win, start_y, start_x, 2, 2, text_lines,
> +				text_cols, 0);
> +		wrefresh(win);
> +
> +		res = wgetch(win);
> +		switch (res) {
> +		case KEY_NPAGE:
> +		case ' ':
> +			start_y += text_lines-2;
> +			break;
> +		case KEY_PPAGE:
> +			start_y -= text_lines+2;
> +			break;
> +		case KEY_DOWN:
> +		case 'j':
> +			start_y++;
> +			break;
> +		case KEY_UP:
> +		case 'k':
> +			start_y--;
> +			break;
> +		case KEY_LEFT:
> +		case 'h':
> +			start_x--;
> +			break;
> +		case KEY_RIGHT:
> +		case 'l':
> +			start_x++;
> +			break;
> +		}
> +		if (res == 10 || res == 27)
> +			break;
> +		if (start_y < 0)
> +			start_y = 0;
> +		if (start_y >= total_lines-text_lines)
> +			start_y = total_lines-text_lines;
> +		if (start_x < 0)
> +			start_x = 0;
> +		if (start_x >= total_cols-text_cols)
> +			start_x = total_cols-text_cols;
> +	} while (res);
> +
> +	del_panel(panel);
> +	delwin(win);
> +	refresh_all_windows(main_window);
> +}
> diff --git a/scripts/kconfig/nconf.h b/scripts/kconfig/nconf.h
> new file mode 100644
> index 0000000..39ed395
> --- /dev/null
> +++ b/scripts/kconfig/nconf.h
> @@ -0,0 +1,86 @@
> +/*
> + * Copyright (C) 2008 Nir Tzachar <nir.tzachar@xxxxxxxxx?
> + * Released under the terms of the GNU GPL v2.0.
> + *
> + * Derived from menuconfig.
> + *
> + */
> +
> +#include <ctype.h>
> +#include <errno.h>
> +#include <fcntl.h>
> +#include <limits.h>
> +#include <stdarg.h>
> +#include <stdlib.h>
> +#include <string.h>
> +#include <unistd.h>
> +#include <locale.h>
> +#include <curses.h>
> +#include <menu.h>
> +#include <panel.h>
> +#include <form.h>
> +
> +#include <stdio.h>
> +#include <time.h>
> +#include <sys/time.h>
> +
> +#include "ncurses.h"
> +
> +#define max(a,b) ({\
> +		typeof(a) _a = a;\
> +		typeof(b) _b = b;\
> +		_a > _b ? _a : _b; })
> +
> +#define min(a,b) ({\
> +		typeof(a) _a = a;\
> +		typeof(b) _b = b;\
> +		_a < _b ? _a : _b; })
> +
> +typedef enum {
> +	NORMAL = 1,
> +	MAIN_HEADING,
> +	MAIN_MENU_BOX,
> +	MAIN_MENU_FORE,
> +	MAIN_MENU_BACK,
> +	MAIN_MENU_HEADING,
> +	INSTS_WIN_BOX,
> +	INSTS_WIN_HEADING,
> +	INSTS_WIN_TEXT,
> +	SCROLLWIN_TEXT,
> +	SCROLLWIN_HEADING,
> +	SCROLLWIN_BOX,
> +	BOTTOM_MENU_FORE,
> +	BOTTOM_MENU_BACK,
> +	DIALOG_TEXT,
> +	DIALOG_MENU_FORE,
> +	DIALOG_MENU_BACK,
> +	DIALOG_BOX,
> +	INPUT_BOX,
> +	INPUT_HEADING,
> +	INPUT_TEXT,
> +	INPUT_FIELD,
> +	ATTR_MAX
> +} attributes_t;
> +extern attributes_t attributes[];
> +
> +void set_colors(void);
> +
> +/* this changes the windows attributes !!! */
> +void print_in_middle(WINDOW *win,
> +	int starty,
> +	int startx,
> +	int width,
> +	const char *string,
> +	chtype color);
> +int get_line_length(const char *line);
> +int get_line_no(const char *text);
> +const char *get_line(const char *text, int line_no);
> +void fill_window(WINDOW *win, const char *text);
> +int btn_dialog(WINDOW *main_window, const char *msg, int btn_num, ...);
> +int dialog_inputbox(WINDOW *main_window,
> +	const char *title, const char *prompt,
> +	const char *init, char *result, int result_len);
> +void refresh_all_windows(WINDOW *main_window);
> +void show_scroll_win(WINDOW *main_window,
> +	const char *title, 
> +	const char *text);
> diff --git a/scripts/kconfig/shared.c b/scripts/kconfig/shared.c
> new file mode 100644
> index 0000000..b4f3f4b
> --- /dev/null
> +++ b/scripts/kconfig/shared.c
> @@ -0,0 +1,82 @@
> +/*
> + * Copyright (C) 2008 Nir Tzachar <nir.tzachar@xxxxxxxxx?
> + * Released under the terms of the GNU GPL v2.0.
> + *
> + * Derived from menuconfig.
> + *
> + */
> +
> +#include "shared.h"
> +
> +void get_prompt_str(struct gstr *r, struct property *prop)
> +{
> +	int i, j;
> +	struct menu *submenu[8], *menu;
> +
> +	str_printf(r, _("Prompt: %s\n"), _(prop->text));
> +	str_printf(r, _("  Defined at %s:%d\n"), prop->menu->file->name,
> +		prop->menu->lineno);
> +	if (!expr_is_yes(prop->visible.expr)) {
> +		str_append(r, _("  Depends on: "));
> +		expr_gstr_print(prop->visible.expr, r);
> +		str_append(r, "\n");
> +	}
> +	menu = prop->menu->parent;
> +	for (i = 0; menu != &rootmenu && i < 8; menu = menu->parent)
> +		submenu[i++] = menu;
> +	if (i > 0) {
> +		str_printf(r, _("  Location:\n"));
> +		for (j = 4; --i >= 0; j += 2) {
> +			menu = submenu[i];
> +			str_printf(r, "%*c-> %s", j, ' ', _(menu_get_prompt(menu)));
> +			if (menu->sym) {
> +				str_printf(r, " (%s [=%s])", menu->sym->name ?
> +					menu->sym->name : _("<choice>"),
> +					sym_get_string_value(menu->sym));
> +			}
> +			str_append(r, "\n");
> +		}
> +	}
> +}
> +
> +void get_symbol_str(struct gstr *r, struct symbol *sym)
> +{
> +	bool hit;
> +	struct property *prop;
> +
> +	if (sym && sym->name)
> +		str_printf(r, "Symbol: %s [=%s]\n", sym->name,
> +		                                    sym_get_string_value(sym));
> +	for_all_prompts(sym, prop)
> +		get_prompt_str(r, prop);
> +	hit = false;
> +	for_all_properties(sym, prop, P_SELECT) {
> +		if (!hit) {
> +			str_append(r, "  Selects: ");
> +			hit = true;
> +		} else
> +			str_printf(r, " && ");
> +		expr_gstr_print(prop->expr, r);
> +	}
> +	if (hit)
> +		str_append(r, "\n");
> +	if (sym->rev_dep.expr) {
> +		str_append(r, _("  Selected by: "));
> +		expr_gstr_print(sym->rev_dep.expr, r);
> +		str_append(r, "\n");
> +	}
> +	str_append(r, "\n\n");
> +}
> +
> +struct gstr get_relations_str(struct symbol **sym_arr)
> +{
> +	struct symbol *sym;
> +	struct gstr res = str_new();
> +	int i;
> +
> +	for (i = 0; sym_arr && (sym = sym_arr[i]); i++)
> +		get_symbol_str(&res, sym);
> +	if (!i)
> +		str_append(&res, _("No matches found.\n"));
> +	return res;
> +}
> diff --git a/scripts/kconfig/shared.h b/scripts/kconfig/shared.h
> new file mode 100644
> index 0000000..935dcaa
> --- /dev/null
> +++ b/scripts/kconfig/shared.h
> @@ -0,0 +1,13 @@
> +/*
> + * Copyright (C) 2008 Nir Tzachar <nir.tzachar@xxxxxxxxx?
> + * Released under the terms of the GNU GPL v2.0.
> + *
> + * Derived from menuconfig.
> + *
> + */
> +#define LKC_DIRECT_LINK
> +#include "lkc.h"
> +
> +void get_prompt_str(struct gstr *r, struct property *prop);
> +void get_symbol_str(struct gstr *r, struct symbol *sym);
> +struct gstr get_relations_str(struct symbol **sym_arr);
> diff --git a/scripts/kconfig/zconf.tab.c_shipped b/scripts/kconfig/zconf.tab.c_shipped
> index 95df833..ef5b369 100644
> --- a/scripts/kconfig/zconf.tab.c_shipped
> +++ b/scripts/kconfig/zconf.tab.c_shipped
> @@ -2255,7 +2255,7 @@ void conf_parse(const char *name)
>  	zconf_initscan(name);
>  
>  	sym_init();
> -	menu_init();
> +	_menu_init();
>  	modules_sym = sym_lookup(NULL, 0);
>  	modules_sym->type = S_BOOLEAN;
>  	modules_sym->flags |= SYMBOL_AUTO;
> diff --git a/scripts/kconfig/zconf.y b/scripts/kconfig/zconf.y
> index 9710b82..cbb72fd 100644
> --- a/scripts/kconfig/zconf.y
> +++ b/scripts/kconfig/zconf.y
> @@ -472,7 +472,7 @@ void conf_parse(const char *name)
>  	zconf_initscan(name);
>  
>  	sym_init();
> -	menu_init();
> +	_menu_init();
>  	modules_sym = sym_lookup(NULL, 0);
>  	modules_sym->type = S_BOOLEAN;
>  	modules_sym->flags |= SYMBOL_AUTO;
> -- 
> 1.5.6.4
> 
--
To unsubscribe from this list: send the line "unsubscribe linux-kbuild" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Linux&nblp;USB Development]     [Linux Media]     [Video for Linux]     [Linux Audio Users]     [Yosemite Secrets]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux