diff options
Diffstat (limited to 'scripts')
41 files changed, 1319 insertions, 849 deletions
diff --git a/scripts/Makefile.build b/scripts/Makefile.build index de9836eee8bb..67fb4530a6ff 100644 --- a/scripts/Makefile.build +++ b/scripts/Makefile.build | |||
@@ -83,10 +83,12 @@ ifneq ($(strip $(obj-y) $(obj-m) $(obj-n) $(obj-) $(lib-target)),) | |||
83 | builtin-target := $(obj)/built-in.o | 83 | builtin-target := $(obj)/built-in.o |
84 | endif | 84 | endif |
85 | 85 | ||
86 | modorder-target := $(obj)/modules.order | ||
87 | |||
86 | # We keep a list of all modules in $(MODVERDIR) | 88 | # We keep a list of all modules in $(MODVERDIR) |
87 | 89 | ||
88 | __build: $(if $(KBUILD_BUILTIN),$(builtin-target) $(lib-target) $(extra-y)) \ | 90 | __build: $(if $(KBUILD_BUILTIN),$(builtin-target) $(lib-target) $(extra-y)) \ |
89 | $(if $(KBUILD_MODULES),$(obj-m)) \ | 91 | $(if $(KBUILD_MODULES),$(obj-m) $(modorder-target)) \ |
90 | $(subdir-ym) $(always) | 92 | $(subdir-ym) $(always) |
91 | @: | 93 | @: |
92 | 94 | ||
@@ -101,6 +103,10 @@ ifneq ($(KBUILD_CHECKSRC),0) | |||
101 | endif | 103 | endif |
102 | endif | 104 | endif |
103 | 105 | ||
106 | # Do section mismatch analysis for each module/built-in.o | ||
107 | ifdef CONFIG_DEBUG_SECTION_MISMATCH | ||
108 | cmd_secanalysis = ; scripts/mod/modpost $@ | ||
109 | endif | ||
104 | 110 | ||
105 | # Compile C sources (.c) | 111 | # Compile C sources (.c) |
106 | # --------------------------------------------------------------------------- | 112 | # --------------------------------------------------------------------------- |
@@ -266,7 +272,8 @@ ifdef builtin-target | |||
266 | quiet_cmd_link_o_target = LD $@ | 272 | quiet_cmd_link_o_target = LD $@ |
267 | # If the list of objects to link is empty, just create an empty built-in.o | 273 | # If the list of objects to link is empty, just create an empty built-in.o |
268 | cmd_link_o_target = $(if $(strip $(obj-y)),\ | 274 | cmd_link_o_target = $(if $(strip $(obj-y)),\ |
269 | $(LD) $(ld_flags) -r -o $@ $(filter $(obj-y), $^),\ | 275 | $(LD) $(ld_flags) -r -o $@ $(filter $(obj-y), $^) \ |
276 | $(cmd_secanalysis),\ | ||
270 | rm -f $@; $(AR) rcs $@) | 277 | rm -f $@; $(AR) rcs $@) |
271 | 278 | ||
272 | $(builtin-target): $(obj-y) FORCE | 279 | $(builtin-target): $(obj-y) FORCE |
@@ -276,6 +283,19 @@ targets += $(builtin-target) | |||
276 | endif # builtin-target | 283 | endif # builtin-target |
277 | 284 | ||
278 | # | 285 | # |
286 | # Rule to create modules.order file | ||
287 | # | ||
288 | # Create commands to either record .ko file or cat modules.order from | ||
289 | # a subdirectory | ||
290 | modorder-cmds = \ | ||
291 | $(foreach m, $(modorder), \ | ||
292 | $(if $(filter %/modules.order, $m), \ | ||
293 | cat $m;, echo kernel/$m;)) | ||
294 | |||
295 | $(modorder-target): $(subdir-ym) FORCE | ||
296 | $(Q)(cat /dev/null; $(modorder-cmds)) > $@ | ||
297 | |||
298 | # | ||
279 | # Rule to compile a set of .o files into one .a file | 299 | # Rule to compile a set of .o files into one .a file |
280 | # | 300 | # |
281 | ifdef lib-target | 301 | ifdef lib-target |
@@ -301,7 +321,7 @@ $($(subst $(obj)/,,$(@:.o=-objs))) \ | |||
301 | $($(subst $(obj)/,,$(@:.o=-y)))), $^) | 321 | $($(subst $(obj)/,,$(@:.o=-y)))), $^) |
302 | 322 | ||
303 | quiet_cmd_link_multi-y = LD $@ | 323 | quiet_cmd_link_multi-y = LD $@ |
304 | cmd_link_multi-y = $(LD) $(ld_flags) -r -o $@ $(link_multi_deps) | 324 | cmd_link_multi-y = $(LD) $(ld_flags) -r -o $@ $(link_multi_deps) $(cmd_secanalysis) |
305 | 325 | ||
306 | quiet_cmd_link_multi-m = LD [M] $@ | 326 | quiet_cmd_link_multi-m = LD [M] $@ |
307 | cmd_link_multi-m = $(cmd_link_multi-y) | 327 | cmd_link_multi-m = $(cmd_link_multi-y) |
diff --git a/scripts/Makefile.lib b/scripts/Makefile.lib index 3c5e88bfecf1..8e440233c27d 100644 --- a/scripts/Makefile.lib +++ b/scripts/Makefile.lib | |||
@@ -25,6 +25,11 @@ lib-y := $(filter-out $(obj-y), $(sort $(lib-y) $(lib-m))) | |||
25 | # o if we encounter foo/ in $(obj-m), remove it from $(obj-m) | 25 | # o if we encounter foo/ in $(obj-m), remove it from $(obj-m) |
26 | # and add the directory to the list of dirs to descend into: $(subdir-m) | 26 | # and add the directory to the list of dirs to descend into: $(subdir-m) |
27 | 27 | ||
28 | # Determine modorder. | ||
29 | # Unfortunately, we don't have information about ordering between -y | ||
30 | # and -m subdirs. Just put -y's first. | ||
31 | modorder := $(patsubst %/,%/modules.order, $(filter %/, $(obj-y)) $(obj-m:.o=.ko)) | ||
32 | |||
28 | __subdir-y := $(patsubst %/,%,$(filter %/, $(obj-y))) | 33 | __subdir-y := $(patsubst %/,%,$(filter %/, $(obj-y))) |
29 | subdir-y += $(__subdir-y) | 34 | subdir-y += $(__subdir-y) |
30 | __subdir-m := $(patsubst %/,%,$(filter %/, $(obj-m))) | 35 | __subdir-m := $(patsubst %/,%,$(filter %/, $(obj-m))) |
@@ -64,6 +69,7 @@ real-objs-m := $(foreach m, $(obj-m), $(if $(strip $($(m:.o=-objs)) $($(m:.o=-y) | |||
64 | extra-y := $(addprefix $(obj)/,$(extra-y)) | 69 | extra-y := $(addprefix $(obj)/,$(extra-y)) |
65 | always := $(addprefix $(obj)/,$(always)) | 70 | always := $(addprefix $(obj)/,$(always)) |
66 | targets := $(addprefix $(obj)/,$(targets)) | 71 | targets := $(addprefix $(obj)/,$(targets)) |
72 | modorder := $(addprefix $(obj)/,$(modorder)) | ||
67 | obj-y := $(addprefix $(obj)/,$(obj-y)) | 73 | obj-y := $(addprefix $(obj)/,$(obj-y)) |
68 | obj-m := $(addprefix $(obj)/,$(obj-m)) | 74 | obj-m := $(addprefix $(obj)/,$(obj-m)) |
69 | lib-y := $(addprefix $(obj)/,$(lib-y)) | 75 | lib-y := $(addprefix $(obj)/,$(lib-y)) |
diff --git a/scripts/Makefile.modinst b/scripts/Makefile.modinst index f0ff248f5e6f..efa5d940e632 100644 --- a/scripts/Makefile.modinst +++ b/scripts/Makefile.modinst | |||
@@ -21,7 +21,7 @@ quiet_cmd_modules_install = INSTALL $@ | |||
21 | 21 | ||
22 | # Modules built outside the kernel source tree go into extra by default | 22 | # Modules built outside the kernel source tree go into extra by default |
23 | INSTALL_MOD_DIR ?= extra | 23 | INSTALL_MOD_DIR ?= extra |
24 | ext-mod-dir = $(INSTALL_MOD_DIR)$(subst $(KBUILD_EXTMOD),,$(@D)) | 24 | ext-mod-dir = $(INSTALL_MOD_DIR)$(subst $(patsubst %/,%,$(KBUILD_EXTMOD)),,$(@D)) |
25 | 25 | ||
26 | modinst_dir = $(if $(KBUILD_EXTMOD),$(ext-mod-dir),kernel/$(@D)) | 26 | modinst_dir = $(if $(KBUILD_EXTMOD),$(ext-mod-dir),kernel/$(@D)) |
27 | 27 | ||
diff --git a/scripts/Makefile.modpost b/scripts/Makefile.modpost index d988f5d21e3d..65e707e1ffc3 100644 --- a/scripts/Makefile.modpost +++ b/scripts/Makefile.modpost | |||
@@ -62,6 +62,7 @@ modpost = scripts/mod/modpost \ | |||
62 | $(if $(KBUILD_EXTMOD),-i,-o) $(kernelsymfile) \ | 62 | $(if $(KBUILD_EXTMOD),-i,-o) $(kernelsymfile) \ |
63 | $(if $(KBUILD_EXTMOD),-I $(modulesymfile)) \ | 63 | $(if $(KBUILD_EXTMOD),-I $(modulesymfile)) \ |
64 | $(if $(KBUILD_EXTMOD),-o $(modulesymfile)) \ | 64 | $(if $(KBUILD_EXTMOD),-o $(modulesymfile)) \ |
65 | $(if $(CONFIG_DEBUG_SECTION_MISMATCH),,-S) \ | ||
65 | $(if $(KBUILD_EXTMOD)$(KBUILD_MODPOST_WARN),-w) | 66 | $(if $(KBUILD_EXTMOD)$(KBUILD_MODPOST_WARN),-w) |
66 | 67 | ||
67 | quiet_cmd_modpost = MODPOST $(words $(filter-out vmlinux FORCE, $^)) modules | 68 | quiet_cmd_modpost = MODPOST $(words $(filter-out vmlinux FORCE, $^)) modules |
diff --git a/scripts/basic/docproc.c b/scripts/basic/docproc.c index 0e4bd5459df4..35bdc68b6e66 100644 --- a/scripts/basic/docproc.c +++ b/scripts/basic/docproc.c | |||
@@ -30,6 +30,7 @@ | |||
30 | * !Ifilename | 30 | * !Ifilename |
31 | * !Dfilename | 31 | * !Dfilename |
32 | * !Ffilename | 32 | * !Ffilename |
33 | * !Pfilename | ||
33 | * | 34 | * |
34 | */ | 35 | */ |
35 | 36 | ||
@@ -57,6 +58,7 @@ FILEONLY *symbolsonly; | |||
57 | typedef void FILELINE(char * file, char * line); | 58 | typedef void FILELINE(char * file, char * line); |
58 | FILELINE * singlefunctions; | 59 | FILELINE * singlefunctions; |
59 | FILELINE * entity_system; | 60 | FILELINE * entity_system; |
61 | FILELINE * docsection; | ||
60 | 62 | ||
61 | #define MAXLINESZ 2048 | 63 | #define MAXLINESZ 2048 |
62 | #define MAXFILES 250 | 64 | #define MAXFILES 250 |
@@ -65,6 +67,7 @@ FILELINE * entity_system; | |||
65 | #define DOCBOOK "-docbook" | 67 | #define DOCBOOK "-docbook" |
66 | #define FUNCTION "-function" | 68 | #define FUNCTION "-function" |
67 | #define NOFUNCTION "-nofunction" | 69 | #define NOFUNCTION "-nofunction" |
70 | #define NODOCSECTIONS "-no-doc-sections" | ||
68 | 71 | ||
69 | char *srctree; | 72 | char *srctree; |
70 | 73 | ||
@@ -231,13 +234,14 @@ void docfunctions(char * filename, char * type) | |||
231 | 234 | ||
232 | for (i=0; i <= symfilecnt; i++) | 235 | for (i=0; i <= symfilecnt; i++) |
233 | symcnt += symfilelist[i].symbolcnt; | 236 | symcnt += symfilelist[i].symbolcnt; |
234 | vec = malloc((2 + 2 * symcnt + 2) * sizeof(char*)); | 237 | vec = malloc((2 + 2 * symcnt + 3) * sizeof(char *)); |
235 | if (vec == NULL) { | 238 | if (vec == NULL) { |
236 | perror("docproc: "); | 239 | perror("docproc: "); |
237 | exit(1); | 240 | exit(1); |
238 | } | 241 | } |
239 | vec[idx++] = KERNELDOC; | 242 | vec[idx++] = KERNELDOC; |
240 | vec[idx++] = DOCBOOK; | 243 | vec[idx++] = DOCBOOK; |
244 | vec[idx++] = NODOCSECTIONS; | ||
241 | for (i=0; i < symfilecnt; i++) { | 245 | for (i=0; i < symfilecnt; i++) { |
242 | struct symfile * sym = &symfilelist[i]; | 246 | struct symfile * sym = &symfilelist[i]; |
243 | for (j=0; j < sym->symbolcnt; j++) { | 247 | for (j=0; j < sym->symbolcnt; j++) { |
@@ -287,12 +291,36 @@ void singfunc(char * filename, char * line) | |||
287 | } | 291 | } |
288 | 292 | ||
289 | /* | 293 | /* |
294 | * Insert specific documentation section from a file. | ||
295 | * Call kernel-doc with the following parameters: | ||
296 | * kernel-doc -docbook -function "doc section" filename | ||
297 | */ | ||
298 | void docsect(char *filename, char *line) | ||
299 | { | ||
300 | char *vec[6]; /* kerneldoc -docbook -function "section" file NULL */ | ||
301 | char *s; | ||
302 | |||
303 | for (s = line; *s; s++) | ||
304 | if (*s == '\n') | ||
305 | *s = '\0'; | ||
306 | |||
307 | vec[0] = KERNELDOC; | ||
308 | vec[1] = DOCBOOK; | ||
309 | vec[2] = FUNCTION; | ||
310 | vec[3] = line; | ||
311 | vec[4] = filename; | ||
312 | vec[5] = NULL; | ||
313 | exec_kernel_doc(vec); | ||
314 | } | ||
315 | |||
316 | /* | ||
290 | * Parse file, calling action specific functions for: | 317 | * Parse file, calling action specific functions for: |
291 | * 1) Lines containing !E | 318 | * 1) Lines containing !E |
292 | * 2) Lines containing !I | 319 | * 2) Lines containing !I |
293 | * 3) Lines containing !D | 320 | * 3) Lines containing !D |
294 | * 4) Lines containing !F | 321 | * 4) Lines containing !F |
295 | * 5) Default lines - lines not matching the above | 322 | * 5) Lines containing !P |
323 | * 6) Default lines - lines not matching the above | ||
296 | */ | 324 | */ |
297 | void parse_file(FILE *infile) | 325 | void parse_file(FILE *infile) |
298 | { | 326 | { |
@@ -326,6 +354,15 @@ void parse_file(FILE *infile) | |||
326 | s++; | 354 | s++; |
327 | singlefunctions(line +2, s); | 355 | singlefunctions(line +2, s); |
328 | break; | 356 | break; |
357 | case 'P': | ||
358 | /* filename */ | ||
359 | while (*s && !isspace(*s)) s++; | ||
360 | *s++ = '\0'; | ||
361 | /* DOC: section name */ | ||
362 | while (isspace(*s)) | ||
363 | s++; | ||
364 | docsection(line + 2, s); | ||
365 | break; | ||
329 | default: | 366 | default: |
330 | defaultline(line); | 367 | defaultline(line); |
331 | } | 368 | } |
@@ -372,6 +409,7 @@ int main(int argc, char *argv[]) | |||
372 | externalfunctions = find_export_symbols; | 409 | externalfunctions = find_export_symbols; |
373 | symbolsonly = find_export_symbols; | 410 | symbolsonly = find_export_symbols; |
374 | singlefunctions = noaction2; | 411 | singlefunctions = noaction2; |
412 | docsection = noaction2; | ||
375 | parse_file(infile); | 413 | parse_file(infile); |
376 | 414 | ||
377 | /* Rewind to start from beginning of file again */ | 415 | /* Rewind to start from beginning of file again */ |
@@ -381,6 +419,7 @@ int main(int argc, char *argv[]) | |||
381 | externalfunctions = extfunc; | 419 | externalfunctions = extfunc; |
382 | symbolsonly = printline; | 420 | symbolsonly = printline; |
383 | singlefunctions = singfunc; | 421 | singlefunctions = singfunc; |
422 | docsection = docsect; | ||
384 | 423 | ||
385 | parse_file(infile); | 424 | parse_file(infile); |
386 | } | 425 | } |
@@ -394,6 +433,7 @@ int main(int argc, char *argv[]) | |||
394 | externalfunctions = adddep; | 433 | externalfunctions = adddep; |
395 | symbolsonly = adddep; | 434 | symbolsonly = adddep; |
396 | singlefunctions = adddep2; | 435 | singlefunctions = adddep2; |
436 | docsection = adddep2; | ||
397 | parse_file(infile); | 437 | parse_file(infile); |
398 | printf("\n"); | 438 | printf("\n"); |
399 | } | 439 | } |
diff --git a/scripts/decodecode b/scripts/decodecode index 1e1a8f620c47..235d3938529d 100644 --- a/scripts/decodecode +++ b/scripts/decodecode | |||
@@ -6,7 +6,19 @@ | |||
6 | # e.g., to decode an i386 oops on an x86_64 system, use: | 6 | # e.g., to decode an i386 oops on an x86_64 system, use: |
7 | # AFLAGS=--32 decodecode < 386.oops | 7 | # AFLAGS=--32 decodecode < 386.oops |
8 | 8 | ||
9 | T=`mktemp` | 9 | cleanup() { |
10 | rm -f $T $T.s $T.o | ||
11 | exit 1 | ||
12 | } | ||
13 | |||
14 | die() { | ||
15 | echo "$@" | ||
16 | exit 1 | ||
17 | } | ||
18 | |||
19 | trap cleanup EXIT | ||
20 | |||
21 | T=`mktemp` || die "cannot create temp file" | ||
10 | code= | 22 | code= |
11 | 23 | ||
12 | while read i ; do | 24 | while read i ; do |
@@ -20,6 +32,7 @@ esac | |||
20 | done | 32 | done |
21 | 33 | ||
22 | if [ -z "$code" ]; then | 34 | if [ -z "$code" ]; then |
35 | rm $T | ||
23 | exit | 36 | exit |
24 | fi | 37 | fi |
25 | 38 | ||
@@ -48,4 +61,4 @@ echo -n " .byte 0x" > $T.s | |||
48 | echo $code >> $T.s | 61 | echo $code >> $T.s |
49 | as $AFLAGS -o $T.o $T.s | 62 | as $AFLAGS -o $T.o $T.s |
50 | objdump -S $T.o | 63 | objdump -S $T.o |
51 | rm $T.o $T.s | 64 | rm $T $T.s $T.o |
diff --git a/scripts/gcc-version.sh b/scripts/gcc-version.sh index a5121a6d8949..cc767b388baf 100644 --- a/scripts/gcc-version.sh +++ b/scripts/gcc-version.sh | |||
@@ -9,7 +9,10 @@ | |||
9 | # gcc-2.95.3, `030301' for gcc-3.3.1, etc. | 9 | # gcc-2.95.3, `030301' for gcc-3.3.1, etc. |
10 | # | 10 | # |
11 | 11 | ||
12 | if [[ $1 = "-p" ]] ; then with_patchlevel=1; shift; fi | 12 | if [ "$1" = "-p" ] ; then |
13 | with_patchlevel=1; | ||
14 | shift; | ||
15 | fi | ||
13 | 16 | ||
14 | compiler="$*" | 17 | compiler="$*" |
15 | 18 | ||
diff --git a/scripts/genksyms/genksyms.c b/scripts/genksyms/genksyms.c index 511023b430a8..dca5e0dd09bf 100644 --- a/scripts/genksyms/genksyms.c +++ b/scripts/genksyms/genksyms.c | |||
@@ -440,17 +440,21 @@ void error_with_pos(const char *fmt, ...) | |||
440 | 440 | ||
441 | static void genksyms_usage(void) | 441 | static void genksyms_usage(void) |
442 | { | 442 | { |
443 | fputs("Usage:\n" "genksyms [-dDwqhV] > /path/to/.tmp_obj.ver\n" "\n" | 443 | fputs("Usage:\n" "genksyms [-adDTwqhV] > /path/to/.tmp_obj.ver\n" "\n" |
444 | #ifdef __GNU_LIBRARY__ | 444 | #ifdef __GNU_LIBRARY__ |
445 | " -a, --arch Select architecture\n" | ||
445 | " -d, --debug Increment the debug level (repeatable)\n" | 446 | " -d, --debug Increment the debug level (repeatable)\n" |
446 | " -D, --dump Dump expanded symbol defs (for debugging only)\n" | 447 | " -D, --dump Dump expanded symbol defs (for debugging only)\n" |
448 | " -T, --dump-types file Dump expanded types into file (for debugging only)\n" | ||
447 | " -w, --warnings Enable warnings\n" | 449 | " -w, --warnings Enable warnings\n" |
448 | " -q, --quiet Disable warnings (default)\n" | 450 | " -q, --quiet Disable warnings (default)\n" |
449 | " -h, --help Print this message\n" | 451 | " -h, --help Print this message\n" |
450 | " -V, --version Print the release version\n" | 452 | " -V, --version Print the release version\n" |
451 | #else /* __GNU_LIBRARY__ */ | 453 | #else /* __GNU_LIBRARY__ */ |
454 | " -a Select architecture\n" | ||
452 | " -d Increment the debug level (repeatable)\n" | 455 | " -d Increment the debug level (repeatable)\n" |
453 | " -D Dump expanded symbol defs (for debugging only)\n" | 456 | " -D Dump expanded symbol defs (for debugging only)\n" |
457 | " -T file Dump expanded types into file (for debugging only)\n" | ||
454 | " -w Enable warnings\n" | 458 | " -w Enable warnings\n" |
455 | " -q Disable warnings (default)\n" | 459 | " -q Disable warnings (default)\n" |
456 | " -h Print this message\n" | 460 | " -h Print this message\n" |
@@ -477,10 +481,10 @@ int main(int argc, char **argv) | |||
477 | {0, 0, 0, 0} | 481 | {0, 0, 0, 0} |
478 | }; | 482 | }; |
479 | 483 | ||
480 | while ((o = getopt_long(argc, argv, "a:dwqVDT:k:p:", | 484 | while ((o = getopt_long(argc, argv, "a:dwqVDT:h", |
481 | &long_opts[0], NULL)) != EOF) | 485 | &long_opts[0], NULL)) != EOF) |
482 | #else /* __GNU_LIBRARY__ */ | 486 | #else /* __GNU_LIBRARY__ */ |
483 | while ((o = getopt(argc, argv, "a:dwqVDT:k:p:")) != EOF) | 487 | while ((o = getopt(argc, argv, "a:dwqVDT:h")) != EOF) |
484 | #endif /* __GNU_LIBRARY__ */ | 488 | #endif /* __GNU_LIBRARY__ */ |
485 | switch (o) { | 489 | switch (o) { |
486 | case 'a': | 490 | case 'a': |
diff --git a/scripts/kconfig/Makefile b/scripts/kconfig/Makefile index 1ad6f7fc490a..32e8c5a227c3 100644 --- a/scripts/kconfig/Makefile +++ b/scripts/kconfig/Makefile | |||
@@ -24,22 +24,25 @@ oldconfig: $(obj)/conf | |||
24 | silentoldconfig: $(obj)/conf | 24 | silentoldconfig: $(obj)/conf |
25 | $< -s $(Kconfig) | 25 | $< -s $(Kconfig) |
26 | 26 | ||
27 | # Create new linux.po file | 27 | # Create new linux.pot file |
28 | # Adjust charset to UTF-8 in .po file to accept UTF-8 in Kconfig files | 28 | # Adjust charset to UTF-8 in .po file to accept UTF-8 in Kconfig files |
29 | # The symlink is used to repair a deficiency in arch/um | 29 | # The symlink is used to repair a deficiency in arch/um |
30 | update-po-config: $(obj)/kxgettext | 30 | update-po-config: $(obj)/kxgettext $(obj)/gconf.glade.h |
31 | xgettext --default-domain=linux \ | 31 | $(Q)echo " GEN config" |
32 | $(Q)xgettext --default-domain=linux \ | ||
32 | --add-comments --keyword=_ --keyword=N_ \ | 33 | --add-comments --keyword=_ --keyword=N_ \ |
33 | --from-code=UTF-8 \ | 34 | --from-code=UTF-8 \ |
34 | --files-from=scripts/kconfig/POTFILES.in \ | 35 | --files-from=scripts/kconfig/POTFILES.in \ |
35 | --output $(obj)/config.pot | 36 | --output $(obj)/config.pot |
36 | $(Q)sed -i s/CHARSET/UTF-8/ $(obj)/config.pot | 37 | $(Q)sed -i s/CHARSET/UTF-8/ $(obj)/config.pot |
37 | $(Q)ln -fs Kconfig.i386 arch/um/Kconfig.arch | 38 | $(Q)ln -fs Kconfig.i386 arch/um/Kconfig.arch |
38 | (for i in `ls arch/`; \ | 39 | $(Q)(for i in `ls arch/`; \ |
39 | do \ | 40 | do \ |
40 | $(obj)/kxgettext arch/$$i/Kconfig; \ | 41 | echo " GEN $$i"; \ |
41 | done ) >> $(obj)/config.pot | 42 | $(obj)/kxgettext arch/$$i/Kconfig \ |
42 | msguniq --sort-by-file --to-code=UTF-8 $(obj)/config.pot \ | 43 | >> $(obj)/config.pot; \ |
44 | done ) | ||
45 | $(Q)msguniq --sort-by-file --to-code=UTF-8 $(obj)/config.pot \ | ||
43 | --output $(obj)/linux.pot | 46 | --output $(obj)/linux.pot |
44 | $(Q)rm -f arch/um/Kconfig.arch | 47 | $(Q)rm -f arch/um/Kconfig.arch |
45 | $(Q)rm -f $(obj)/config.pot | 48 | $(Q)rm -f $(obj)/config.pot |
@@ -93,12 +96,6 @@ HOST_LOADLIBES = $(shell $(CONFIG_SHELL) $(check-lxdialog) -ldflags $(HOSTCC)) | |||
93 | 96 | ||
94 | HOST_EXTRACFLAGS += -DLOCALE | 97 | HOST_EXTRACFLAGS += -DLOCALE |
95 | 98 | ||
96 | PHONY += $(obj)/dochecklxdialog | ||
97 | $(obj)/dochecklxdialog: | ||
98 | $(Q)$(CONFIG_SHELL) $(check-lxdialog) -check $(HOSTCC) $(HOST_LOADLIBES) | ||
99 | |||
100 | always := dochecklxdialog | ||
101 | |||
102 | 99 | ||
103 | # =========================================================================== | 100 | # =========================================================================== |
104 | # Shared Makefile for the various kconfig executables: | 101 | # Shared Makefile for the various kconfig executables: |
@@ -142,8 +139,17 @@ gconf-objs := gconf.o kconfig_load.o zconf.tab.o | |||
142 | endif | 139 | endif |
143 | 140 | ||
144 | clean-files := lkc_defs.h qconf.moc .tmp_qtcheck \ | 141 | clean-files := lkc_defs.h qconf.moc .tmp_qtcheck \ |
145 | .tmp_gtkcheck zconf.tab.c lex.zconf.c zconf.hash.c | 142 | .tmp_gtkcheck zconf.tab.c lex.zconf.c zconf.hash.c gconf.glade.h |
146 | clean-files += mconf qconf gconf | 143 | clean-files += mconf qconf gconf |
144 | clean-files += config.pot linux.pot | ||
145 | |||
146 | # Check that we have the required ncurses stuff installed for lxdialog (menuconfig) | ||
147 | PHONY += $(obj)/dochecklxdialog | ||
148 | $(addprefix $(obj)/,$(lxdialog)): $(obj)/dochecklxdialog | ||
149 | $(obj)/dochecklxdialog: | ||
150 | $(Q)$(CONFIG_SHELL) $(check-lxdialog) -check $(HOSTCC) $(HOST_EXTRACFLAGS) $(HOST_LOADLIBES) | ||
151 | |||
152 | always := dochecklxdialog | ||
147 | 153 | ||
148 | # Add environment specific flags | 154 | # Add environment specific flags |
149 | HOST_EXTRACFLAGS += $(shell $(CONFIG_SHELL) $(srctree)/$(src)/check.sh $(HOSTCC) $(HOSTCFLAGS)) | 155 | HOST_EXTRACFLAGS += $(shell $(CONFIG_SHELL) $(srctree)/$(src)/check.sh $(HOSTCC) $(HOSTCFLAGS)) |
@@ -248,6 +254,9 @@ $(obj)/%.moc: $(src)/%.h | |||
248 | $(obj)/lkc_defs.h: $(src)/lkc_proto.h | 254 | $(obj)/lkc_defs.h: $(src)/lkc_proto.h |
249 | sed < $< > $@ 's/P(\([^,]*\),.*/#define \1 (\*\1_p)/' | 255 | sed < $< > $@ 's/P(\([^,]*\),.*/#define \1 (\*\1_p)/' |
250 | 256 | ||
257 | # Extract gconf menu items for I18N support | ||
258 | $(obj)/gconf.glade.h: $(obj)/gconf.glade | ||
259 | intltool-extract --type=gettext/glade $(obj)/gconf.glade | ||
251 | 260 | ||
252 | ### | 261 | ### |
253 | # The following requires flex/bison/gperf | 262 | # The following requires flex/bison/gperf |
diff --git a/scripts/kconfig/POTFILES.in b/scripts/kconfig/POTFILES.in index cc94e46a79e8..967457396990 100644 --- a/scripts/kconfig/POTFILES.in +++ b/scripts/kconfig/POTFILES.in | |||
@@ -1,5 +1,12 @@ | |||
1 | scripts/kconfig/lxdialog/checklist.c | ||
2 | scripts/kconfig/lxdialog/inputbox.c | ||
3 | scripts/kconfig/lxdialog/menubox.c | ||
4 | scripts/kconfig/lxdialog/textbox.c | ||
5 | scripts/kconfig/lxdialog/util.c | ||
6 | scripts/kconfig/lxdialog/yesno.c | ||
1 | scripts/kconfig/mconf.c | 7 | scripts/kconfig/mconf.c |
2 | scripts/kconfig/conf.c | 8 | scripts/kconfig/conf.c |
3 | scripts/kconfig/confdata.c | 9 | scripts/kconfig/confdata.c |
4 | scripts/kconfig/gconf.c | 10 | scripts/kconfig/gconf.c |
11 | scripts/kconfig/gconf.glade.h | ||
5 | scripts/kconfig/qconf.cc | 12 | scripts/kconfig/qconf.cc |
diff --git a/scripts/kconfig/conf.c b/scripts/kconfig/conf.c index 8d6f17490c5e..fda63136ae68 100644 --- a/scripts/kconfig/conf.c +++ b/scripts/kconfig/conf.c | |||
@@ -3,12 +3,13 @@ | |||
3 | * Released under the terms of the GNU GPL v2.0. | 3 | * Released under the terms of the GNU GPL v2.0. |
4 | */ | 4 | */ |
5 | 5 | ||
6 | #include <locale.h> | ||
6 | #include <ctype.h> | 7 | #include <ctype.h> |
7 | #include <stdlib.h> | ||
8 | #include <stdio.h> | 8 | #include <stdio.h> |
9 | #include <stdlib.h> | ||
9 | #include <string.h> | 10 | #include <string.h> |
10 | #include <unistd.h> | ||
11 | #include <time.h> | 11 | #include <time.h> |
12 | #include <unistd.h> | ||
12 | #include <sys/stat.h> | 13 | #include <sys/stat.h> |
13 | 14 | ||
14 | #define LKC_DIRECT_LINK | 15 | #define LKC_DIRECT_LINK |
@@ -40,7 +41,7 @@ static char nohelp_text[] = N_("Sorry, no help available for this option yet.\n" | |||
40 | static const char *get_help(struct menu *menu) | 41 | static const char *get_help(struct menu *menu) |
41 | { | 42 | { |
42 | if (menu_has_help(menu)) | 43 | if (menu_has_help(menu)) |
43 | return menu_get_help(menu); | 44 | return _(menu_get_help(menu)); |
44 | else | 45 | else |
45 | return nohelp_text; | 46 | return nohelp_text; |
46 | } | 47 | } |
@@ -78,7 +79,7 @@ static int conf_askvalue(struct symbol *sym, const char *def) | |||
78 | tristate val; | 79 | tristate val; |
79 | 80 | ||
80 | if (!sym_has_value(sym)) | 81 | if (!sym_has_value(sym)) |
81 | printf("(NEW) "); | 82 | printf(_("(NEW) ")); |
82 | 83 | ||
83 | line[0] = '\n'; | 84 | line[0] = '\n'; |
84 | line[1] = 0; | 85 | line[1] = 0; |
@@ -160,7 +161,7 @@ static int conf_askvalue(struct symbol *sym, const char *def) | |||
160 | } | 161 | } |
161 | case set_random: | 162 | case set_random: |
162 | do { | 163 | do { |
163 | val = (tristate)(random() % 3); | 164 | val = (tristate)(rand() % 3); |
164 | } while (!sym_tristate_within_range(sym, val)); | 165 | } while (!sym_tristate_within_range(sym, val)); |
165 | switch (val) { | 166 | switch (val) { |
166 | case no: line[0] = 'n'; break; | 167 | case no: line[0] = 'n'; break; |
@@ -183,7 +184,7 @@ int conf_string(struct menu *menu) | |||
183 | const char *def; | 184 | const char *def; |
184 | 185 | ||
185 | while (1) { | 186 | while (1) { |
186 | printf("%*s%s ", indent - 1, "", menu->prompt->text); | 187 | printf("%*s%s ", indent - 1, "", _(menu->prompt->text)); |
187 | printf("(%s) ", sym->name); | 188 | printf("(%s) ", sym->name); |
188 | def = sym_get_string_value(sym); | 189 | def = sym_get_string_value(sym); |
189 | if (sym_get_string_value(sym)) | 190 | if (sym_get_string_value(sym)) |
@@ -216,7 +217,7 @@ static int conf_sym(struct menu *menu) | |||
216 | tristate oldval, newval; | 217 | tristate oldval, newval; |
217 | 218 | ||
218 | while (1) { | 219 | while (1) { |
219 | printf("%*s%s ", indent - 1, "", menu->prompt->text); | 220 | printf("%*s%s ", indent - 1, "", _(menu->prompt->text)); |
220 | if (sym->name) | 221 | if (sym->name) |
221 | printf("(%s) ", sym->name); | 222 | printf("(%s) ", sym->name); |
222 | type = sym_get_type(sym); | 223 | type = sym_get_type(sym); |
@@ -306,7 +307,7 @@ static int conf_choice(struct menu *menu) | |||
306 | case no: | 307 | case no: |
307 | return 1; | 308 | return 1; |
308 | case mod: | 309 | case mod: |
309 | printf("%*s%s\n", indent - 1, "", menu_get_prompt(menu)); | 310 | printf("%*s%s\n", indent - 1, "", _(menu_get_prompt(menu))); |
310 | return 0; | 311 | return 0; |
311 | case yes: | 312 | case yes: |
312 | break; | 313 | break; |
@@ -316,7 +317,7 @@ static int conf_choice(struct menu *menu) | |||
316 | while (1) { | 317 | while (1) { |
317 | int cnt, def; | 318 | int cnt, def; |
318 | 319 | ||
319 | printf("%*s%s\n", indent - 1, "", menu_get_prompt(menu)); | 320 | printf("%*s%s\n", indent - 1, "", _(menu_get_prompt(menu))); |
320 | def_sym = sym_get_choice_value(sym); | 321 | def_sym = sym_get_choice_value(sym); |
321 | cnt = def = 0; | 322 | cnt = def = 0; |
322 | line[0] = 0; | 323 | line[0] = 0; |
@@ -324,7 +325,7 @@ static int conf_choice(struct menu *menu) | |||
324 | if (!menu_is_visible(child)) | 325 | if (!menu_is_visible(child)) |
325 | continue; | 326 | continue; |
326 | if (!child->sym) { | 327 | if (!child->sym) { |
327 | printf("%*c %s\n", indent, '*', menu_get_prompt(child)); | 328 | printf("%*c %s\n", indent, '*', _(menu_get_prompt(child))); |
328 | continue; | 329 | continue; |
329 | } | 330 | } |
330 | cnt++; | 331 | cnt++; |
@@ -333,14 +334,14 @@ static int conf_choice(struct menu *menu) | |||
333 | printf("%*c", indent, '>'); | 334 | printf("%*c", indent, '>'); |
334 | } else | 335 | } else |
335 | printf("%*c", indent, ' '); | 336 | printf("%*c", indent, ' '); |
336 | printf(" %d. %s", cnt, menu_get_prompt(child)); | 337 | printf(" %d. %s", cnt, _(menu_get_prompt(child))); |
337 | if (child->sym->name) | 338 | if (child->sym->name) |
338 | printf(" (%s)", child->sym->name); | 339 | printf(" (%s)", child->sym->name); |
339 | if (!sym_has_value(child->sym)) | 340 | if (!sym_has_value(child->sym)) |
340 | printf(" (NEW)"); | 341 | printf(_(" (NEW)")); |
341 | printf("\n"); | 342 | printf("\n"); |
342 | } | 343 | } |
343 | printf("%*schoice", indent - 1, ""); | 344 | printf(_("%*schoice"), indent - 1, ""); |
344 | if (cnt == 1) { | 345 | if (cnt == 1) { |
345 | printf("[1]: 1\n"); | 346 | printf("[1]: 1\n"); |
346 | goto conf_childs; | 347 | goto conf_childs; |
@@ -375,7 +376,7 @@ static int conf_choice(struct menu *menu) | |||
375 | break; | 376 | break; |
376 | case set_random: | 377 | case set_random: |
377 | if (is_new) | 378 | if (is_new) |
378 | def = (random() % cnt) + 1; | 379 | def = (rand() % cnt) + 1; |
379 | case set_default: | 380 | case set_default: |
380 | case set_yes: | 381 | case set_yes: |
381 | case set_mod: | 382 | case set_mod: |
@@ -399,9 +400,9 @@ static int conf_choice(struct menu *menu) | |||
399 | continue; | 400 | continue; |
400 | } | 401 | } |
401 | sym_set_choice_value(sym, child->sym); | 402 | sym_set_choice_value(sym, child->sym); |
402 | if (child->list) { | 403 | for (child = child->list; child; child = child->next) { |
403 | indent += 2; | 404 | indent += 2; |
404 | conf(child->list); | 405 | conf(child); |
405 | indent -= 2; | 406 | indent -= 2; |
406 | } | 407 | } |
407 | return 1; | 408 | return 1; |
@@ -433,7 +434,7 @@ static void conf(struct menu *menu) | |||
433 | if (prompt) | 434 | if (prompt) |
434 | printf("%*c\n%*c %s\n%*c\n", | 435 | printf("%*c\n%*c %s\n%*c\n", |
435 | indent, '*', | 436 | indent, '*', |
436 | indent, '*', prompt, | 437 | indent, '*', _(prompt), |
437 | indent, '*'); | 438 | indent, '*'); |
438 | default: | 439 | default: |
439 | ; | 440 | ; |
@@ -495,12 +496,16 @@ static void check_conf(struct menu *menu) | |||
495 | 496 | ||
496 | int main(int ac, char **av) | 497 | int main(int ac, char **av) |
497 | { | 498 | { |
498 | int i = 1; | 499 | int opt; |
499 | const char *name; | 500 | const char *name; |
500 | struct stat tmpstat; | 501 | struct stat tmpstat; |
501 | 502 | ||
502 | if (ac > i && av[i][0] == '-') { | 503 | setlocale(LC_ALL, ""); |
503 | switch (av[i++][1]) { | 504 | bindtextdomain(PACKAGE, LOCALEDIR); |
505 | textdomain(PACKAGE); | ||
506 | |||
507 | while ((opt = getopt(ac, av, "osdD:nmyrh")) != -1) { | ||
508 | switch (opt) { | ||
504 | case 'o': | 509 | case 'o': |
505 | input_mode = ask_new; | 510 | input_mode = ask_new; |
506 | break; | 511 | break; |
@@ -513,12 +518,7 @@ int main(int ac, char **av) | |||
513 | break; | 518 | break; |
514 | case 'D': | 519 | case 'D': |
515 | input_mode = set_default; | 520 | input_mode = set_default; |
516 | defconfig_file = av[i++]; | 521 | defconfig_file = optarg; |
517 | if (!defconfig_file) { | ||
518 | printf(_("%s: No default config file specified\n"), | ||
519 | av[0]); | ||
520 | exit(1); | ||
521 | } | ||
522 | break; | 522 | break; |
523 | case 'n': | 523 | case 'n': |
524 | input_mode = set_no; | 524 | input_mode = set_no; |
@@ -531,19 +531,22 @@ int main(int ac, char **av) | |||
531 | break; | 531 | break; |
532 | case 'r': | 532 | case 'r': |
533 | input_mode = set_random; | 533 | input_mode = set_random; |
534 | srandom(time(NULL)); | 534 | srand(time(NULL)); |
535 | break; | 535 | break; |
536 | case 'h': | 536 | case 'h': |
537 | case '?': | 537 | printf(_("See README for usage info\n")); |
538 | fprintf(stderr, "See README for usage info\n"); | ||
539 | exit(0); | 538 | exit(0); |
539 | break; | ||
540 | default: | ||
541 | fprintf(stderr, _("See README for usage info\n")); | ||
542 | exit(1); | ||
540 | } | 543 | } |
541 | } | 544 | } |
542 | name = av[i]; | 545 | if (ac == optind) { |
543 | if (!name) { | ||
544 | printf(_("%s: Kconfig file missing\n"), av[0]); | 546 | printf(_("%s: Kconfig file missing\n"), av[0]); |
545 | exit(1); | 547 | exit(1); |
546 | } | 548 | } |
549 | name = av[optind]; | ||
547 | conf_parse(name); | 550 | conf_parse(name); |
548 | //zconfdump(stdout); | 551 | //zconfdump(stdout); |
549 | switch (input_mode) { | 552 | switch (input_mode) { |
@@ -551,9 +554,9 @@ int main(int ac, char **av) | |||
551 | if (!defconfig_file) | 554 | if (!defconfig_file) |
552 | defconfig_file = conf_get_default_confname(); | 555 | defconfig_file = conf_get_default_confname(); |
553 | if (conf_read(defconfig_file)) { | 556 | if (conf_read(defconfig_file)) { |
554 | printf("***\n" | 557 | printf(_("***\n" |
555 | "*** Can't find default configuration \"%s\"!\n" | 558 | "*** Can't find default configuration \"%s\"!\n" |
556 | "***\n", defconfig_file); | 559 | "***\n"), defconfig_file); |
557 | exit(1); | 560 | exit(1); |
558 | } | 561 | } |
559 | break; | 562 | break; |
diff --git a/scripts/kconfig/confdata.c b/scripts/kconfig/confdata.c index e0f402f3b75d..ee5fe943d58d 100644 --- a/scripts/kconfig/confdata.c +++ b/scripts/kconfig/confdata.c | |||
@@ -232,8 +232,7 @@ load: | |||
232 | sym->type = S_BOOLEAN; | 232 | sym->type = S_BOOLEAN; |
233 | } | 233 | } |
234 | if (sym->flags & def_flags) { | 234 | if (sym->flags & def_flags) { |
235 | conf_warning("trying to reassign symbol %s", sym->name); | 235 | conf_warning("override: reassigning to symbol %s", sym->name); |
236 | break; | ||
237 | } | 236 | } |
238 | switch (sym->type) { | 237 | switch (sym->type) { |
239 | case S_BOOLEAN: | 238 | case S_BOOLEAN: |
@@ -272,8 +271,7 @@ load: | |||
272 | sym->type = S_OTHER; | 271 | sym->type = S_OTHER; |
273 | } | 272 | } |
274 | if (sym->flags & def_flags) { | 273 | if (sym->flags & def_flags) { |
275 | conf_warning("trying to reassign symbol %s", sym->name); | 274 | conf_warning("override: reassigning to symbol %s", sym->name); |
276 | break; | ||
277 | } | 275 | } |
278 | if (conf_set_sym_val(sym, def, def_flags, p)) | 276 | if (conf_set_sym_val(sym, def, def_flags, p)) |
279 | continue; | 277 | continue; |
@@ -297,14 +295,12 @@ load: | |||
297 | } | 295 | } |
298 | break; | 296 | break; |
299 | case yes: | 297 | case yes: |
300 | if (cs->def[def].tri != no) { | 298 | if (cs->def[def].tri != no) |
301 | conf_warning("%s creates inconsistent choice state", sym->name); | 299 | conf_warning("override: %s changes choice state", sym->name); |
302 | cs->flags &= ~def_flags; | 300 | cs->def[def].val = sym; |
303 | } else | ||
304 | cs->def[def].val = sym; | ||
305 | break; | 301 | break; |
306 | } | 302 | } |
307 | cs->def[def].tri = E_OR(cs->def[def].tri, sym->def[def].tri); | 303 | cs->def[def].tri = EXPR_OR(cs->def[def].tri, sym->def[def].tri); |
308 | } | 304 | } |
309 | } | 305 | } |
310 | fclose(in); | 306 | fclose(in); |
@@ -316,7 +312,7 @@ load: | |||
316 | 312 | ||
317 | int conf_read(const char *name) | 313 | int conf_read(const char *name) |
318 | { | 314 | { |
319 | struct symbol *sym; | 315 | struct symbol *sym, *choice_sym; |
320 | struct property *prop; | 316 | struct property *prop; |
321 | struct expr *e; | 317 | struct expr *e; |
322 | int i, flags; | 318 | int i, flags; |
@@ -357,9 +353,9 @@ int conf_read(const char *name) | |||
357 | */ | 353 | */ |
358 | prop = sym_get_choice_prop(sym); | 354 | prop = sym_get_choice_prop(sym); |
359 | flags = sym->flags; | 355 | flags = sym->flags; |
360 | for (e = prop->expr; e; e = e->left.expr) | 356 | expr_list_for_each_sym(prop->expr, e, choice_sym) |
361 | if (e->right.sym->visible != no) | 357 | if (choice_sym->visible != no) |
362 | flags &= e->right.sym->flags; | 358 | flags &= choice_sym->flags; |
363 | sym->flags &= flags | ~SYMBOL_DEF_USER; | 359 | sym->flags &= flags | ~SYMBOL_DEF_USER; |
364 | } | 360 | } |
365 | 361 | ||
diff --git a/scripts/kconfig/expr.c b/scripts/kconfig/expr.c index 6f98dbfe70cf..579ece4fa584 100644 --- a/scripts/kconfig/expr.c +++ b/scripts/kconfig/expr.c | |||
@@ -87,7 +87,7 @@ struct expr *expr_copy(struct expr *org) | |||
87 | break; | 87 | break; |
88 | case E_AND: | 88 | case E_AND: |
89 | case E_OR: | 89 | case E_OR: |
90 | case E_CHOICE: | 90 | case E_LIST: |
91 | e->left.expr = expr_copy(org->left.expr); | 91 | e->left.expr = expr_copy(org->left.expr); |
92 | e->right.expr = expr_copy(org->right.expr); | 92 | e->right.expr = expr_copy(org->right.expr); |
93 | break; | 93 | break; |
@@ -217,7 +217,7 @@ int expr_eq(struct expr *e1, struct expr *e2) | |||
217 | expr_free(e2); | 217 | expr_free(e2); |
218 | trans_count = old_count; | 218 | trans_count = old_count; |
219 | return res; | 219 | return res; |
220 | case E_CHOICE: | 220 | case E_LIST: |
221 | case E_RANGE: | 221 | case E_RANGE: |
222 | case E_NONE: | 222 | case E_NONE: |
223 | /* panic */; | 223 | /* panic */; |
@@ -648,7 +648,7 @@ struct expr *expr_transform(struct expr *e) | |||
648 | case E_EQUAL: | 648 | case E_EQUAL: |
649 | case E_UNEQUAL: | 649 | case E_UNEQUAL: |
650 | case E_SYMBOL: | 650 | case E_SYMBOL: |
651 | case E_CHOICE: | 651 | case E_LIST: |
652 | break; | 652 | break; |
653 | default: | 653 | default: |
654 | e->left.expr = expr_transform(e->left.expr); | 654 | e->left.expr = expr_transform(e->left.expr); |
@@ -932,7 +932,7 @@ struct expr *expr_trans_compare(struct expr *e, enum expr_type type, struct symb | |||
932 | break; | 932 | break; |
933 | case E_SYMBOL: | 933 | case E_SYMBOL: |
934 | return expr_alloc_comp(type, e->left.sym, sym); | 934 | return expr_alloc_comp(type, e->left.sym, sym); |
935 | case E_CHOICE: | 935 | case E_LIST: |
936 | case E_RANGE: | 936 | case E_RANGE: |
937 | case E_NONE: | 937 | case E_NONE: |
938 | /* panic */; | 938 | /* panic */; |
@@ -955,14 +955,14 @@ tristate expr_calc_value(struct expr *e) | |||
955 | case E_AND: | 955 | case E_AND: |
956 | val1 = expr_calc_value(e->left.expr); | 956 | val1 = expr_calc_value(e->left.expr); |
957 | val2 = expr_calc_value(e->right.expr); | 957 | val2 = expr_calc_value(e->right.expr); |
958 | return E_AND(val1, val2); | 958 | return EXPR_AND(val1, val2); |
959 | case E_OR: | 959 | case E_OR: |
960 | val1 = expr_calc_value(e->left.expr); | 960 | val1 = expr_calc_value(e->left.expr); |
961 | val2 = expr_calc_value(e->right.expr); | 961 | val2 = expr_calc_value(e->right.expr); |
962 | return E_OR(val1, val2); | 962 | return EXPR_OR(val1, val2); |
963 | case E_NOT: | 963 | case E_NOT: |
964 | val1 = expr_calc_value(e->left.expr); | 964 | val1 = expr_calc_value(e->left.expr); |
965 | return E_NOT(val1); | 965 | return EXPR_NOT(val1); |
966 | case E_EQUAL: | 966 | case E_EQUAL: |
967 | sym_calc_value(e->left.sym); | 967 | sym_calc_value(e->left.sym); |
968 | sym_calc_value(e->right.sym); | 968 | sym_calc_value(e->right.sym); |
@@ -1000,9 +1000,9 @@ int expr_compare_type(enum expr_type t1, enum expr_type t2) | |||
1000 | if (t2 == E_OR) | 1000 | if (t2 == E_OR) |
1001 | return 1; | 1001 | return 1; |
1002 | case E_OR: | 1002 | case E_OR: |
1003 | if (t2 == E_CHOICE) | 1003 | if (t2 == E_LIST) |
1004 | return 1; | 1004 | return 1; |
1005 | case E_CHOICE: | 1005 | case E_LIST: |
1006 | if (t2 == 0) | 1006 | if (t2 == 0) |
1007 | return 1; | 1007 | return 1; |
1008 | default: | 1008 | default: |
@@ -1034,12 +1034,18 @@ void expr_print(struct expr *e, void (*fn)(void *, struct symbol *, const char * | |||
1034 | expr_print(e->left.expr, fn, data, E_NOT); | 1034 | expr_print(e->left.expr, fn, data, E_NOT); |
1035 | break; | 1035 | break; |
1036 | case E_EQUAL: | 1036 | case E_EQUAL: |
1037 | fn(data, e->left.sym, e->left.sym->name); | 1037 | if (e->left.sym->name) |
1038 | fn(data, e->left.sym, e->left.sym->name); | ||
1039 | else | ||
1040 | fn(data, NULL, "<choice>"); | ||
1038 | fn(data, NULL, "="); | 1041 | fn(data, NULL, "="); |
1039 | fn(data, e->right.sym, e->right.sym->name); | 1042 | fn(data, e->right.sym, e->right.sym->name); |
1040 | break; | 1043 | break; |
1041 | case E_UNEQUAL: | 1044 | case E_UNEQUAL: |
1042 | fn(data, e->left.sym, e->left.sym->name); | 1045 | if (e->left.sym->name) |
1046 | fn(data, e->left.sym, e->left.sym->name); | ||
1047 | else | ||
1048 | fn(data, NULL, "<choice>"); | ||
1043 | fn(data, NULL, "!="); | 1049 | fn(data, NULL, "!="); |
1044 | fn(data, e->right.sym, e->right.sym->name); | 1050 | fn(data, e->right.sym, e->right.sym->name); |
1045 | break; | 1051 | break; |
@@ -1053,11 +1059,11 @@ void expr_print(struct expr *e, void (*fn)(void *, struct symbol *, const char * | |||
1053 | fn(data, NULL, " && "); | 1059 | fn(data, NULL, " && "); |
1054 | expr_print(e->right.expr, fn, data, E_AND); | 1060 | expr_print(e->right.expr, fn, data, E_AND); |
1055 | break; | 1061 | break; |
1056 | case E_CHOICE: | 1062 | case E_LIST: |
1057 | fn(data, e->right.sym, e->right.sym->name); | 1063 | fn(data, e->right.sym, e->right.sym->name); |
1058 | if (e->left.expr) { | 1064 | if (e->left.expr) { |
1059 | fn(data, NULL, " ^ "); | 1065 | fn(data, NULL, " ^ "); |
1060 | expr_print(e->left.expr, fn, data, E_CHOICE); | 1066 | expr_print(e->left.expr, fn, data, E_LIST); |
1061 | } | 1067 | } |
1062 | break; | 1068 | break; |
1063 | case E_RANGE: | 1069 | case E_RANGE: |
diff --git a/scripts/kconfig/expr.h b/scripts/kconfig/expr.h index a195986eec6f..9d4cba1c001d 100644 --- a/scripts/kconfig/expr.h +++ b/scripts/kconfig/expr.h | |||
@@ -25,14 +25,13 @@ struct file { | |||
25 | 25 | ||
26 | #define FILE_BUSY 0x0001 | 26 | #define FILE_BUSY 0x0001 |
27 | #define FILE_SCANNED 0x0002 | 27 | #define FILE_SCANNED 0x0002 |
28 | #define FILE_PRINTED 0x0004 | ||
29 | 28 | ||
30 | typedef enum tristate { | 29 | typedef enum tristate { |
31 | no, mod, yes | 30 | no, mod, yes |
32 | } tristate; | 31 | } tristate; |
33 | 32 | ||
34 | enum expr_type { | 33 | enum expr_type { |
35 | E_NONE, E_OR, E_AND, E_NOT, E_EQUAL, E_UNEQUAL, E_CHOICE, E_SYMBOL, E_RANGE | 34 | E_NONE, E_OR, E_AND, E_NOT, E_EQUAL, E_UNEQUAL, E_LIST, E_SYMBOL, E_RANGE |
36 | }; | 35 | }; |
37 | 36 | ||
38 | union expr_data { | 37 | union expr_data { |
@@ -45,9 +44,12 @@ struct expr { | |||
45 | union expr_data left, right; | 44 | union expr_data left, right; |
46 | }; | 45 | }; |
47 | 46 | ||
48 | #define E_OR(dep1, dep2) (((dep1)>(dep2))?(dep1):(dep2)) | 47 | #define EXPR_OR(dep1, dep2) (((dep1)>(dep2))?(dep1):(dep2)) |
49 | #define E_AND(dep1, dep2) (((dep1)<(dep2))?(dep1):(dep2)) | 48 | #define EXPR_AND(dep1, dep2) (((dep1)<(dep2))?(dep1):(dep2)) |
50 | #define E_NOT(dep) (2-(dep)) | 49 | #define EXPR_NOT(dep) (2-(dep)) |
50 | |||
51 | #define expr_list_for_each_sym(l, e, s) \ | ||
52 | for (e = (l); e && (s = e->right.sym); e = e->left.expr) | ||
51 | 53 | ||
52 | struct expr_value { | 54 | struct expr_value { |
53 | struct expr *expr; | 55 | struct expr *expr; |
@@ -86,7 +88,6 @@ struct symbol { | |||
86 | #define SYMBOL_CHECK 0x0008 | 88 | #define SYMBOL_CHECK 0x0008 |
87 | #define SYMBOL_CHOICE 0x0010 | 89 | #define SYMBOL_CHOICE 0x0010 |
88 | #define SYMBOL_CHOICEVAL 0x0020 | 90 | #define SYMBOL_CHOICEVAL 0x0020 |
89 | #define SYMBOL_PRINTED 0x0040 | ||
90 | #define SYMBOL_VALID 0x0080 | 91 | #define SYMBOL_VALID 0x0080 |
91 | #define SYMBOL_OPTIONAL 0x0100 | 92 | #define SYMBOL_OPTIONAL 0x0100 |
92 | #define SYMBOL_WRITE 0x0200 | 93 | #define SYMBOL_WRITE 0x0200 |
@@ -105,7 +106,8 @@ struct symbol { | |||
105 | #define SYMBOL_HASHMASK 0xff | 106 | #define SYMBOL_HASHMASK 0xff |
106 | 107 | ||
107 | enum prop_type { | 108 | enum prop_type { |
108 | P_UNKNOWN, P_PROMPT, P_COMMENT, P_MENU, P_DEFAULT, P_CHOICE, P_SELECT, P_RANGE | 109 | P_UNKNOWN, P_PROMPT, P_COMMENT, P_MENU, P_DEFAULT, P_CHOICE, |
110 | P_SELECT, P_RANGE, P_ENV | ||
109 | }; | 111 | }; |
110 | 112 | ||
111 | struct property { | 113 | struct property { |
diff --git a/scripts/kconfig/gconf.c b/scripts/kconfig/gconf.c index 262908cfc2ac..199b22bb49e2 100644 --- a/scripts/kconfig/gconf.c +++ b/scripts/kconfig/gconf.c | |||
@@ -119,8 +119,6 @@ const char *dbg_print_flags(int val) | |||
119 | strcat(buf, "choice/"); | 119 | strcat(buf, "choice/"); |
120 | if (val & SYMBOL_CHOICEVAL) | 120 | if (val & SYMBOL_CHOICEVAL) |
121 | strcat(buf, "choiceval/"); | 121 | strcat(buf, "choiceval/"); |
122 | if (val & SYMBOL_PRINTED) | ||
123 | strcat(buf, "printed/"); | ||
124 | if (val & SYMBOL_VALID) | 122 | if (val & SYMBOL_VALID) |
125 | strcat(buf, "valid/"); | 123 | strcat(buf, "valid/"); |
126 | if (val & SYMBOL_OPTIONAL) | 124 | if (val & SYMBOL_OPTIONAL) |
@@ -457,14 +455,18 @@ static void text_insert_help(struct menu *menu) | |||
457 | { | 455 | { |
458 | GtkTextBuffer *buffer; | 456 | GtkTextBuffer *buffer; |
459 | GtkTextIter start, end; | 457 | GtkTextIter start, end; |
460 | const char *prompt = menu_get_prompt(menu); | 458 | const char *prompt = _(menu_get_prompt(menu)); |
461 | gchar *name; | 459 | gchar *name; |
462 | const char *help; | 460 | const char *help; |
463 | 461 | ||
464 | help = _(menu_get_help(menu)); | 462 | help = menu_get_help(menu); |
463 | |||
464 | /* Gettextize if the help text not empty */ | ||
465 | if ((help != 0) && (help[0] != 0)) | ||
466 | help = _(help); | ||
465 | 467 | ||
466 | if (menu->sym && menu->sym->name) | 468 | if (menu->sym && menu->sym->name) |
467 | name = g_strdup_printf(_(menu->sym->name)); | 469 | name = g_strdup_printf(menu->sym->name); |
468 | else | 470 | else |
469 | name = g_strdup(""); | 471 | name = g_strdup(""); |
470 | 472 | ||
@@ -1171,7 +1173,7 @@ static gchar **fill_row(struct menu *menu) | |||
1171 | bzero(row, sizeof(row)); | 1173 | bzero(row, sizeof(row)); |
1172 | 1174 | ||
1173 | row[COL_OPTION] = | 1175 | row[COL_OPTION] = |
1174 | g_strdup_printf("%s %s", menu_get_prompt(menu), | 1176 | g_strdup_printf("%s %s", _(menu_get_prompt(menu)), |
1175 | sym && sym_has_value(sym) ? "(NEW)" : ""); | 1177 | sym && sym_has_value(sym) ? "(NEW)" : ""); |
1176 | 1178 | ||
1177 | if (show_all && !menu_is_visible(menu)) | 1179 | if (show_all && !menu_is_visible(menu)) |
@@ -1221,7 +1223,7 @@ static gchar **fill_row(struct menu *menu) | |||
1221 | 1223 | ||
1222 | if (def_menu) | 1224 | if (def_menu) |
1223 | row[COL_VALUE] = | 1225 | row[COL_VALUE] = |
1224 | g_strdup(menu_get_prompt(def_menu)); | 1226 | g_strdup(_(menu_get_prompt(def_menu))); |
1225 | } | 1227 | } |
1226 | if (sym->flags & SYMBOL_CHOICEVAL) | 1228 | if (sym->flags & SYMBOL_CHOICEVAL) |
1227 | row[COL_BTNRAD] = GINT_TO_POINTER(TRUE); | 1229 | row[COL_BTNRAD] = GINT_TO_POINTER(TRUE); |
diff --git a/scripts/kconfig/lex.zconf.c_shipped b/scripts/kconfig/lex.zconf.c_shipped index a065d5a57c01..bed0f4e2d2f7 100644 --- a/scripts/kconfig/lex.zconf.c_shipped +++ b/scripts/kconfig/lex.zconf.c_shipped | |||
@@ -1275,6 +1275,11 @@ YY_RULE_SETUP | |||
1275 | case 32: | 1275 | case 32: |
1276 | YY_RULE_SETUP | 1276 | YY_RULE_SETUP |
1277 | { | 1277 | { |
1278 | while (zconfleng) { | ||
1279 | if ((zconftext[zconfleng-1] != ' ') && (zconftext[zconfleng-1] != '\t')) | ||
1280 | break; | ||
1281 | zconfleng--; | ||
1282 | } | ||
1278 | append_string(zconftext, zconfleng); | 1283 | append_string(zconftext, zconfleng); |
1279 | if (!first_ts) | 1284 | if (!first_ts) |
1280 | first_ts = last_ts; | 1285 | first_ts = last_ts; |
diff --git a/scripts/kconfig/lkc.h b/scripts/kconfig/lkc.h index 8a07ee4f6bd4..4bc68f20a73c 100644 --- a/scripts/kconfig/lkc.h +++ b/scripts/kconfig/lkc.h | |||
@@ -44,6 +44,7 @@ extern "C" { | |||
44 | 44 | ||
45 | #define T_OPT_MODULES 1 | 45 | #define T_OPT_MODULES 1 |
46 | #define T_OPT_DEFCONFIG_LIST 2 | 46 | #define T_OPT_DEFCONFIG_LIST 2 |
47 | #define T_OPT_ENV 3 | ||
47 | 48 | ||
48 | struct kconf_id { | 49 | struct kconf_id { |
49 | int name; | 50 | int name; |
@@ -74,6 +75,7 @@ void kconfig_load(void); | |||
74 | 75 | ||
75 | /* menu.c */ | 76 | /* menu.c */ |
76 | void menu_init(void); | 77 | void menu_init(void); |
78 | void menu_warn(struct menu *menu, const char *fmt, ...); | ||
77 | struct menu *menu_add_menu(void); | 79 | struct menu *menu_add_menu(void); |
78 | void menu_end_menu(void); | 80 | void menu_end_menu(void); |
79 | void menu_add_entry(struct symbol *sym); | 81 | void menu_add_entry(struct symbol *sym); |
@@ -103,6 +105,8 @@ void str_printf(struct gstr *gs, const char *fmt, ...); | |||
103 | const char *str_get(struct gstr *gs); | 105 | const char *str_get(struct gstr *gs); |
104 | 106 | ||
105 | /* symbol.c */ | 107 | /* symbol.c */ |
108 | extern struct expr *sym_env_list; | ||
109 | |||
106 | void sym_init(void); | 110 | void sym_init(void); |
107 | void sym_clear_all_valid(void); | 111 | void sym_clear_all_valid(void); |
108 | void sym_set_all_changed(void); | 112 | void sym_set_all_changed(void); |
@@ -110,6 +114,7 @@ void sym_set_changed(struct symbol *sym); | |||
110 | struct symbol *sym_check_deps(struct symbol *sym); | 114 | struct symbol *sym_check_deps(struct symbol *sym); |
111 | struct property *prop_alloc(enum prop_type type, struct symbol *sym); | 115 | struct property *prop_alloc(enum prop_type type, struct symbol *sym); |
112 | struct symbol *prop_get_symbol(struct property *prop); | 116 | struct symbol *prop_get_symbol(struct property *prop); |
117 | struct property *sym_get_env_prop(struct symbol *sym); | ||
113 | 118 | ||
114 | static inline tristate sym_get_tristate_value(struct symbol *sym) | 119 | static inline tristate sym_get_tristate_value(struct symbol *sym) |
115 | { | 120 | { |
diff --git a/scripts/kconfig/lxdialog/check-lxdialog.sh b/scripts/kconfig/lxdialog/check-lxdialog.sh index 9681476b96e7..62e1e02126e6 100644 --- a/scripts/kconfig/lxdialog/check-lxdialog.sh +++ b/scripts/kconfig/lxdialog/check-lxdialog.sh | |||
@@ -36,14 +36,16 @@ trap "rm -f $tmp" 0 1 2 3 15 | |||
36 | 36 | ||
37 | # Check if we can link to ncurses | 37 | # Check if we can link to ncurses |
38 | check() { | 38 | check() { |
39 | echo "main() {}" | $cc -xc - -o $tmp 2> /dev/null | 39 | echo -e " #include CURSES_LOC \n main() {}" | |
40 | $cc -xc - -o $tmp 2> /dev/null | ||
40 | if [ $? != 0 ]; then | 41 | if [ $? != 0 ]; then |
41 | echo " *** Unable to find the ncurses libraries." 1>&2 | 42 | echo " *** Unable to find the ncurses libraries or the" 1>&2 |
42 | echo " *** make menuconfig require the ncurses libraries" 1>&2 | 43 | echo " *** required header files." 1>&2 |
43 | echo " *** " 1>&2 | 44 | echo " *** 'make menuconfig' requires the ncurses libraries." 1>&2 |
44 | echo " *** Install ncurses (ncurses-devel) and try again" 1>&2 | 45 | echo " *** " 1>&2 |
45 | echo " *** " 1>&2 | 46 | echo " *** Install ncurses (ncurses-devel) and try again." 1>&2 |
46 | exit 1 | 47 | echo " *** " 1>&2 |
48 | exit 1 | ||
47 | fi | 49 | fi |
48 | } | 50 | } |
49 | 51 | ||
diff --git a/scripts/kconfig/lxdialog/checklist.c b/scripts/kconfig/lxdialog/checklist.c index cf697080dddd..b2a878c936d6 100644 --- a/scripts/kconfig/lxdialog/checklist.c +++ b/scripts/kconfig/lxdialog/checklist.c | |||
@@ -97,8 +97,8 @@ static void print_buttons(WINDOW * dialog, int height, int width, int selected) | |||
97 | int x = width / 2 - 11; | 97 | int x = width / 2 - 11; |
98 | int y = height - 2; | 98 | int y = height - 2; |
99 | 99 | ||
100 | print_button(dialog, "Select", y, x, selected == 0); | 100 | print_button(dialog, gettext("Select"), y, x, selected == 0); |
101 | print_button(dialog, " Help ", y, x + 14, selected == 1); | 101 | print_button(dialog, gettext(" Help "), y, x + 14, selected == 1); |
102 | 102 | ||
103 | wmove(dialog, y, x + 1 + 14 * selected); | 103 | wmove(dialog, y, x + 1 + 14 * selected); |
104 | wrefresh(dialog); | 104 | wrefresh(dialog); |
diff --git a/scripts/kconfig/lxdialog/dialog.h b/scripts/kconfig/lxdialog/dialog.h index 7e17eba75ae8..b5211fce0d94 100644 --- a/scripts/kconfig/lxdialog/dialog.h +++ b/scripts/kconfig/lxdialog/dialog.h | |||
@@ -26,6 +26,12 @@ | |||
26 | #include <string.h> | 26 | #include <string.h> |
27 | #include <stdbool.h> | 27 | #include <stdbool.h> |
28 | 28 | ||
29 | #ifndef KBUILD_NO_NLS | ||
30 | # include <libintl.h> | ||
31 | #else | ||
32 | # define gettext(Msgid) ((const char *) (Msgid)) | ||
33 | #endif | ||
34 | |||
29 | #ifdef __sun__ | 35 | #ifdef __sun__ |
30 | #define CURS_MACROS | 36 | #define CURS_MACROS |
31 | #endif | 37 | #endif |
@@ -187,10 +193,9 @@ int item_is_tag(char tag); | |||
187 | int on_key_esc(WINDOW *win); | 193 | int on_key_esc(WINDOW *win); |
188 | int on_key_resize(void); | 194 | int on_key_resize(void); |
189 | 195 | ||
190 | void init_dialog(const char *backtitle); | 196 | int init_dialog(const char *backtitle); |
191 | void set_dialog_backtitle(const char *backtitle); | 197 | void set_dialog_backtitle(const char *backtitle); |
192 | void reset_dialog(void); | 198 | void end_dialog(int x, int y); |
193 | void end_dialog(void); | ||
194 | void attr_clear(WINDOW * win, int height, int width, chtype attr); | 199 | void attr_clear(WINDOW * win, int height, int width, chtype attr); |
195 | void dialog_clear(void); | 200 | void dialog_clear(void); |
196 | void print_autowrap(WINDOW * win, const char *prompt, int width, int y, int x); | 201 | void print_autowrap(WINDOW * win, const char *prompt, int width, int y, int x); |
diff --git a/scripts/kconfig/lxdialog/inputbox.c b/scripts/kconfig/lxdialog/inputbox.c index 05e72066b359..4946bd02b46d 100644 --- a/scripts/kconfig/lxdialog/inputbox.c +++ b/scripts/kconfig/lxdialog/inputbox.c | |||
@@ -31,8 +31,8 @@ static void print_buttons(WINDOW * dialog, int height, int width, int selected) | |||
31 | int x = width / 2 - 11; | 31 | int x = width / 2 - 11; |
32 | int y = height - 2; | 32 | int y = height - 2; |
33 | 33 | ||
34 | print_button(dialog, " Ok ", y, x, selected == 0); | 34 | print_button(dialog, gettext(" Ok "), y, x, selected == 0); |
35 | print_button(dialog, " Help ", y, x + 14, selected == 1); | 35 | print_button(dialog, gettext(" Help "), y, x + 14, selected == 1); |
36 | 36 | ||
37 | wmove(dialog, y, x + 1 + 14 * selected); | 37 | wmove(dialog, y, x + 1 + 14 * selected); |
38 | wrefresh(dialog); | 38 | wrefresh(dialog); |
diff --git a/scripts/kconfig/lxdialog/menubox.c b/scripts/kconfig/lxdialog/menubox.c index 0d83159d9012..fa9d633f293c 100644 --- a/scripts/kconfig/lxdialog/menubox.c +++ b/scripts/kconfig/lxdialog/menubox.c | |||
@@ -157,9 +157,9 @@ static void print_buttons(WINDOW * win, int height, int width, int selected) | |||
157 | int x = width / 2 - 16; | 157 | int x = width / 2 - 16; |
158 | int y = height - 2; | 158 | int y = height - 2; |
159 | 159 | ||
160 | print_button(win, "Select", y, x, selected == 0); | 160 | print_button(win, gettext("Select"), y, x, selected == 0); |
161 | print_button(win, " Exit ", y, x + 12, selected == 1); | 161 | print_button(win, gettext(" Exit "), y, x + 12, selected == 1); |
162 | print_button(win, " Help ", y, x + 24, selected == 2); | 162 | print_button(win, gettext(" Help "), y, x + 24, selected == 2); |
163 | 163 | ||
164 | wmove(win, y, x + 1 + 12 * selected); | 164 | wmove(win, y, x + 1 + 12 * selected); |
165 | wrefresh(win); | 165 | wrefresh(win); |
diff --git a/scripts/kconfig/lxdialog/textbox.c b/scripts/kconfig/lxdialog/textbox.c index fabfc1ad789d..c704712d0227 100644 --- a/scripts/kconfig/lxdialog/textbox.c +++ b/scripts/kconfig/lxdialog/textbox.c | |||
@@ -114,7 +114,7 @@ do_resize: | |||
114 | 114 | ||
115 | print_title(dialog, title, width); | 115 | print_title(dialog, title, width); |
116 | 116 | ||
117 | print_button(dialog, " Exit ", height - 2, width / 2 - 4, TRUE); | 117 | print_button(dialog, gettext(" Exit "), height - 2, width / 2 - 4, TRUE); |
118 | wnoutrefresh(dialog); | 118 | wnoutrefresh(dialog); |
119 | getyx(dialog, cur_y, cur_x); /* Save cursor position */ | 119 | getyx(dialog, cur_y, cur_x); /* Save cursor position */ |
120 | 120 | ||
diff --git a/scripts/kconfig/lxdialog/util.c b/scripts/kconfig/lxdialog/util.c index a1bddefe73d0..86d95cca46a7 100644 --- a/scripts/kconfig/lxdialog/util.c +++ b/scripts/kconfig/lxdialog/util.c | |||
@@ -266,31 +266,41 @@ void dialog_clear(void) | |||
266 | /* | 266 | /* |
267 | * Do some initialization for dialog | 267 | * Do some initialization for dialog |
268 | */ | 268 | */ |
269 | void init_dialog(const char *backtitle) | 269 | int init_dialog(const char *backtitle) |
270 | { | 270 | { |
271 | dlg.backtitle = backtitle; | 271 | int height, width; |
272 | color_setup(getenv("MENUCONFIG_COLOR")); | 272 | |
273 | } | 273 | initscr(); /* Init curses */ |
274 | getmaxyx(stdscr, height, width); | ||
275 | if (height < 19 || width < 80) { | ||
276 | endwin(); | ||
277 | return -ERRDISPLAYTOOSMALL; | ||
278 | } | ||
274 | 279 | ||
275 | void set_dialog_backtitle(const char *backtitle) | ||
276 | { | ||
277 | dlg.backtitle = backtitle; | 280 | dlg.backtitle = backtitle; |
278 | } | 281 | color_setup(getenv("MENUCONFIG_COLOR")); |
279 | 282 | ||
280 | void reset_dialog(void) | ||
281 | { | ||
282 | initscr(); /* Init curses */ | ||
283 | keypad(stdscr, TRUE); | 283 | keypad(stdscr, TRUE); |
284 | cbreak(); | 284 | cbreak(); |
285 | noecho(); | 285 | noecho(); |
286 | dialog_clear(); | 286 | dialog_clear(); |
287 | |||
288 | return 0; | ||
289 | } | ||
290 | |||
291 | void set_dialog_backtitle(const char *backtitle) | ||
292 | { | ||
293 | dlg.backtitle = backtitle; | ||
287 | } | 294 | } |
288 | 295 | ||
289 | /* | 296 | /* |
290 | * End using dialog functions. | 297 | * End using dialog functions. |
291 | */ | 298 | */ |
292 | void end_dialog(void) | 299 | void end_dialog(int x, int y) |
293 | { | 300 | { |
301 | /* move cursor back to original position */ | ||
302 | move(y, x); | ||
303 | refresh(); | ||
294 | endwin(); | 304 | endwin(); |
295 | } | 305 | } |
296 | 306 | ||
diff --git a/scripts/kconfig/lxdialog/yesno.c b/scripts/kconfig/lxdialog/yesno.c index ee0a04e3e012..4e6e8090c20b 100644 --- a/scripts/kconfig/lxdialog/yesno.c +++ b/scripts/kconfig/lxdialog/yesno.c | |||
@@ -29,8 +29,8 @@ static void print_buttons(WINDOW * dialog, int height, int width, int selected) | |||
29 | int x = width / 2 - 10; | 29 | int x = width / 2 - 10; |
30 | int y = height - 2; | 30 | int y = height - 2; |
31 | 31 | ||
32 | print_button(dialog, " Yes ", y, x, selected == 0); | 32 | print_button(dialog, gettext(" Yes "), y, x, selected == 0); |
33 | print_button(dialog, " No ", y, x + 13, selected == 1); | 33 | print_button(dialog, gettext(" No "), y, x + 13, selected == 1); |
34 | 34 | ||
35 | wmove(dialog, y, x + 1 + 13 * selected); | 35 | wmove(dialog, y, x + 1 + 13 * selected); |
36 | wrefresh(dialog); | 36 | wrefresh(dialog); |
diff --git a/scripts/kconfig/mconf.c b/scripts/kconfig/mconf.c index 47e226fdedd7..50e61c411bc0 100644 --- a/scripts/kconfig/mconf.c +++ b/scripts/kconfig/mconf.c | |||
@@ -8,17 +8,13 @@ | |||
8 | * i18n, 2005, Arnaldo Carvalho de Melo <acme@conectiva.com.br> | 8 | * i18n, 2005, Arnaldo Carvalho de Melo <acme@conectiva.com.br> |
9 | */ | 9 | */ |
10 | 10 | ||
11 | #include <sys/ioctl.h> | ||
12 | #include <sys/wait.h> | ||
13 | #include <ctype.h> | 11 | #include <ctype.h> |
14 | #include <errno.h> | 12 | #include <errno.h> |
15 | #include <fcntl.h> | 13 | #include <fcntl.h> |
16 | #include <limits.h> | 14 | #include <limits.h> |
17 | #include <signal.h> | ||
18 | #include <stdarg.h> | 15 | #include <stdarg.h> |
19 | #include <stdlib.h> | 16 | #include <stdlib.h> |
20 | #include <string.h> | 17 | #include <string.h> |
21 | #include <termios.h> | ||
22 | #include <unistd.h> | 18 | #include <unistd.h> |
23 | #include <locale.h> | 19 | #include <locale.h> |
24 | 20 | ||
@@ -275,8 +271,6 @@ search_help[] = N_( | |||
275 | "\n"); | 271 | "\n"); |
276 | 272 | ||
277 | static int indent; | 273 | static int indent; |
278 | static struct termios ios_org; | ||
279 | static int rows = 0, cols = 0; | ||
280 | static struct menu *current_menu; | 274 | static struct menu *current_menu; |
281 | static int child_count; | 275 | static int child_count; |
282 | static int single_menu_mode; | 276 | static int single_menu_mode; |
@@ -290,51 +284,16 @@ static void show_textbox(const char *title, const char *text, int r, int c); | |||
290 | static void show_helptext(const char *title, const char *text); | 284 | static void show_helptext(const char *title, const char *text); |
291 | static void show_help(struct menu *menu); | 285 | static void show_help(struct menu *menu); |
292 | 286 | ||
293 | static void init_wsize(void) | ||
294 | { | ||
295 | struct winsize ws; | ||
296 | char *env; | ||
297 | |||
298 | if (!ioctl(STDIN_FILENO, TIOCGWINSZ, &ws)) { | ||
299 | rows = ws.ws_row; | ||
300 | cols = ws.ws_col; | ||
301 | } | ||
302 | |||
303 | if (!rows) { | ||
304 | env = getenv("LINES"); | ||
305 | if (env) | ||
306 | rows = atoi(env); | ||
307 | if (!rows) | ||
308 | rows = 24; | ||
309 | } | ||
310 | if (!cols) { | ||
311 | env = getenv("COLUMNS"); | ||
312 | if (env) | ||
313 | cols = atoi(env); | ||
314 | if (!cols) | ||
315 | cols = 80; | ||
316 | } | ||
317 | |||
318 | if (rows < 19 || cols < 80) { | ||
319 | fprintf(stderr, N_("Your display is too small to run Menuconfig!\n")); | ||
320 | fprintf(stderr, N_("It must be at least 19 lines by 80 columns.\n")); | ||
321 | exit(1); | ||
322 | } | ||
323 | |||
324 | rows -= 4; | ||
325 | cols -= 5; | ||
326 | } | ||
327 | |||
328 | static void get_prompt_str(struct gstr *r, struct property *prop) | 287 | static void get_prompt_str(struct gstr *r, struct property *prop) |
329 | { | 288 | { |
330 | int i, j; | 289 | int i, j; |
331 | struct menu *submenu[8], *menu; | 290 | struct menu *submenu[8], *menu; |
332 | 291 | ||
333 | str_printf(r, "Prompt: %s\n", prop->text); | 292 | str_printf(r, _("Prompt: %s\n"), _(prop->text)); |
334 | str_printf(r, " Defined at %s:%d\n", prop->menu->file->name, | 293 | str_printf(r, _(" Defined at %s:%d\n"), prop->menu->file->name, |
335 | prop->menu->lineno); | 294 | prop->menu->lineno); |
336 | if (!expr_is_yes(prop->visible.expr)) { | 295 | if (!expr_is_yes(prop->visible.expr)) { |
337 | str_append(r, " Depends on: "); | 296 | str_append(r, _(" Depends on: ")); |
338 | expr_gstr_print(prop->visible.expr, r); | 297 | expr_gstr_print(prop->visible.expr, r); |
339 | str_append(r, "\n"); | 298 | str_append(r, "\n"); |
340 | } | 299 | } |
@@ -342,13 +301,13 @@ static void get_prompt_str(struct gstr *r, struct property *prop) | |||
342 | for (i = 0; menu != &rootmenu && i < 8; menu = menu->parent) | 301 | for (i = 0; menu != &rootmenu && i < 8; menu = menu->parent) |
343 | submenu[i++] = menu; | 302 | submenu[i++] = menu; |
344 | if (i > 0) { | 303 | if (i > 0) { |
345 | str_printf(r, " Location:\n"); | 304 | str_printf(r, _(" Location:\n")); |
346 | for (j = 4; --i >= 0; j += 2) { | 305 | for (j = 4; --i >= 0; j += 2) { |
347 | menu = submenu[i]; | 306 | menu = submenu[i]; |
348 | str_printf(r, "%*c-> %s", j, ' ', menu_get_prompt(menu)); | 307 | str_printf(r, "%*c-> %s", j, ' ', _(menu_get_prompt(menu))); |
349 | if (menu->sym) { | 308 | if (menu->sym) { |
350 | str_printf(r, " (%s [=%s])", menu->sym->name ? | 309 | str_printf(r, " (%s [=%s])", menu->sym->name ? |
351 | menu->sym->name : "<choice>", | 310 | menu->sym->name : _("<choice>"), |
352 | sym_get_string_value(menu->sym)); | 311 | sym_get_string_value(menu->sym)); |
353 | } | 312 | } |
354 | str_append(r, "\n"); | 313 | str_append(r, "\n"); |
@@ -378,7 +337,7 @@ static void get_symbol_str(struct gstr *r, struct symbol *sym) | |||
378 | if (hit) | 337 | if (hit) |
379 | str_append(r, "\n"); | 338 | str_append(r, "\n"); |
380 | if (sym->rev_dep.expr) { | 339 | if (sym->rev_dep.expr) { |
381 | str_append(r, " Selected by: "); | 340 | str_append(r, _(" Selected by: ")); |
382 | expr_gstr_print(sym->rev_dep.expr, r); | 341 | expr_gstr_print(sym->rev_dep.expr, r); |
383 | str_append(r, "\n"); | 342 | str_append(r, "\n"); |
384 | } | 343 | } |
@@ -394,7 +353,7 @@ static struct gstr get_relations_str(struct symbol **sym_arr) | |||
394 | for (i = 0; sym_arr && (sym = sym_arr[i]); i++) | 353 | for (i = 0; sym_arr && (sym = sym_arr[i]); i++) |
395 | get_symbol_str(&res, sym); | 354 | get_symbol_str(&res, sym); |
396 | if (!i) | 355 | if (!i) |
397 | str_append(&res, "No matches found.\n"); | 356 | str_append(&res, _("No matches found.\n")); |
398 | return res; | 357 | return res; |
399 | } | 358 | } |
400 | 359 | ||
@@ -474,6 +433,7 @@ static void build_conf(struct menu *menu) | |||
474 | switch (prop->type) { | 433 | switch (prop->type) { |
475 | case P_MENU: | 434 | case P_MENU: |
476 | child_count++; | 435 | child_count++; |
436 | prompt = _(prompt); | ||
477 | if (single_menu_mode) { | 437 | if (single_menu_mode) { |
478 | item_make("%s%*c%s", | 438 | item_make("%s%*c%s", |
479 | menu->data ? "-->" : "++>", | 439 | menu->data ? "-->" : "++>", |
@@ -489,7 +449,7 @@ static void build_conf(struct menu *menu) | |||
489 | case P_COMMENT: | 449 | case P_COMMENT: |
490 | if (prompt) { | 450 | if (prompt) { |
491 | child_count++; | 451 | child_count++; |
492 | item_make(" %*c*** %s ***", indent + 1, ' ', prompt); | 452 | item_make(" %*c*** %s ***", indent + 1, ' ', _(prompt)); |
493 | item_set_tag(':'); | 453 | item_set_tag(':'); |
494 | item_set_data(menu); | 454 | item_set_data(menu); |
495 | } | 455 | } |
@@ -497,7 +457,7 @@ static void build_conf(struct menu *menu) | |||
497 | default: | 457 | default: |
498 | if (prompt) { | 458 | if (prompt) { |
499 | child_count++; | 459 | child_count++; |
500 | item_make("---%*c%s", indent + 1, ' ', prompt); | 460 | item_make("---%*c%s", indent + 1, ' ', _(prompt)); |
501 | item_set_tag(':'); | 461 | item_set_tag(':'); |
502 | item_set_data(menu); | 462 | item_set_data(menu); |
503 | } | 463 | } |
@@ -541,10 +501,10 @@ static void build_conf(struct menu *menu) | |||
541 | item_set_data(menu); | 501 | item_set_data(menu); |
542 | } | 502 | } |
543 | 503 | ||
544 | item_add_str("%*c%s", indent + 1, ' ', menu_get_prompt(menu)); | 504 | item_add_str("%*c%s", indent + 1, ' ', _(menu_get_prompt(menu))); |
545 | if (val == yes) { | 505 | if (val == yes) { |
546 | if (def_menu) { | 506 | if (def_menu) { |
547 | item_add_str(" (%s)", menu_get_prompt(def_menu)); | 507 | item_add_str(" (%s)", _(menu_get_prompt(def_menu))); |
548 | item_add_str(" --->"); | 508 | item_add_str(" --->"); |
549 | if (def_menu->list) { | 509 | if (def_menu->list) { |
550 | indent += 2; | 510 | indent += 2; |
@@ -556,7 +516,7 @@ static void build_conf(struct menu *menu) | |||
556 | } | 516 | } |
557 | } else { | 517 | } else { |
558 | if (menu == current_menu) { | 518 | if (menu == current_menu) { |
559 | item_make("---%*c%s", indent + 1, ' ', menu_get_prompt(menu)); | 519 | item_make("---%*c%s", indent + 1, ' ', _(menu_get_prompt(menu))); |
560 | item_set_tag(':'); | 520 | item_set_tag(':'); |
561 | item_set_data(menu); | 521 | item_set_data(menu); |
562 | goto conf_childs; | 522 | goto conf_childs; |
@@ -599,17 +559,17 @@ static void build_conf(struct menu *menu) | |||
599 | tmp = indent - tmp + 4; | 559 | tmp = indent - tmp + 4; |
600 | if (tmp < 0) | 560 | if (tmp < 0) |
601 | tmp = 0; | 561 | tmp = 0; |
602 | item_add_str("%*c%s%s", tmp, ' ', menu_get_prompt(menu), | 562 | item_add_str("%*c%s%s", tmp, ' ', _(menu_get_prompt(menu)), |
603 | (sym_has_value(sym) || !sym_is_changable(sym)) ? | 563 | (sym_has_value(sym) || !sym_is_changable(sym)) ? |
604 | "" : " (NEW)"); | 564 | "" : _(" (NEW)")); |
605 | item_set_tag('s'); | 565 | item_set_tag('s'); |
606 | item_set_data(menu); | 566 | item_set_data(menu); |
607 | goto conf_childs; | 567 | goto conf_childs; |
608 | } | 568 | } |
609 | } | 569 | } |
610 | item_add_str("%*c%s%s", indent + 1, ' ', menu_get_prompt(menu), | 570 | item_add_str("%*c%s%s", indent + 1, ' ', _(menu_get_prompt(menu)), |
611 | (sym_has_value(sym) || !sym_is_changable(sym)) ? | 571 | (sym_has_value(sym) || !sym_is_changable(sym)) ? |
612 | "" : " (NEW)"); | 572 | "" : _(" (NEW)")); |
613 | if (menu->prompt->type == P_MENU) { | 573 | if (menu->prompt->type == P_MENU) { |
614 | item_add_str(" --->"); | 574 | item_add_str(" --->"); |
615 | return; | 575 | return; |
@@ -647,7 +607,7 @@ static void conf(struct menu *menu) | |||
647 | item_set_tag('S'); | 607 | item_set_tag('S'); |
648 | } | 608 | } |
649 | dialog_clear(); | 609 | dialog_clear(); |
650 | res = dialog_menu(prompt ? prompt : _("Main Menu"), | 610 | res = dialog_menu(prompt ? _(prompt) : _("Main Menu"), |
651 | _(menu_instructions), | 611 | _(menu_instructions), |
652 | active_menu, &s_scroll); | 612 | active_menu, &s_scroll); |
653 | if (res == 1 || res == KEY_ESC || res == -ERRDISPLAYTOOSMALL) | 613 | if (res == 1 || res == KEY_ESC || res == -ERRDISPLAYTOOSMALL) |
@@ -694,7 +654,7 @@ static void conf(struct menu *menu) | |||
694 | if (sym) | 654 | if (sym) |
695 | show_help(submenu); | 655 | show_help(submenu); |
696 | else | 656 | else |
697 | show_helptext("README", _(mconf_readme)); | 657 | show_helptext(_("README"), _(mconf_readme)); |
698 | break; | 658 | break; |
699 | case 3: | 659 | case 3: |
700 | if (item_is_tag('t')) { | 660 | if (item_is_tag('t')) { |
@@ -752,13 +712,13 @@ static void show_help(struct menu *menu) | |||
752 | str_append(&help, nohelp_text); | 712 | str_append(&help, nohelp_text); |
753 | } | 713 | } |
754 | get_symbol_str(&help, sym); | 714 | get_symbol_str(&help, sym); |
755 | show_helptext(menu_get_prompt(menu), str_get(&help)); | 715 | show_helptext(_(menu_get_prompt(menu)), str_get(&help)); |
756 | str_free(&help); | 716 | str_free(&help); |
757 | } | 717 | } |
758 | 718 | ||
759 | static void conf_choice(struct menu *menu) | 719 | static void conf_choice(struct menu *menu) |
760 | { | 720 | { |
761 | const char *prompt = menu_get_prompt(menu); | 721 | const char *prompt = _(menu_get_prompt(menu)); |
762 | struct menu *child; | 722 | struct menu *child; |
763 | struct symbol *active; | 723 | struct symbol *active; |
764 | 724 | ||
@@ -772,7 +732,7 @@ static void conf_choice(struct menu *menu) | |||
772 | for (child = menu->list; child; child = child->next) { | 732 | for (child = menu->list; child; child = child->next) { |
773 | if (!menu_is_visible(child)) | 733 | if (!menu_is_visible(child)) |
774 | continue; | 734 | continue; |
775 | item_make("%s", menu_get_prompt(child)); | 735 | item_make("%s", _(menu_get_prompt(child))); |
776 | item_set_data(child); | 736 | item_set_data(child); |
777 | if (child->sym == active) | 737 | if (child->sym == active) |
778 | item_set_selected(1); | 738 | item_set_selected(1); |
@@ -780,7 +740,7 @@ static void conf_choice(struct menu *menu) | |||
780 | item_set_tag('X'); | 740 | item_set_tag('X'); |
781 | } | 741 | } |
782 | dialog_clear(); | 742 | dialog_clear(); |
783 | res = dialog_checklist(prompt ? prompt : _("Main Menu"), | 743 | res = dialog_checklist(prompt ? _(prompt) : _("Main Menu"), |
784 | _(radiolist_instructions), | 744 | _(radiolist_instructions), |
785 | 15, 70, 6); | 745 | 15, 70, 6); |
786 | selected = item_activate_selected(); | 746 | selected = item_activate_selected(); |
@@ -826,10 +786,10 @@ static void conf_string(struct menu *menu) | |||
826 | heading = _(inputbox_instructions_string); | 786 | heading = _(inputbox_instructions_string); |
827 | break; | 787 | break; |
828 | default: | 788 | default: |
829 | heading = "Internal mconf error!"; | 789 | heading = _("Internal mconf error!"); |
830 | } | 790 | } |
831 | dialog_clear(); | 791 | dialog_clear(); |
832 | res = dialog_inputbox(prompt ? prompt : _("Main Menu"), | 792 | res = dialog_inputbox(prompt ? _(prompt) : _("Main Menu"), |
833 | heading, 10, 75, | 793 | heading, 10, 75, |
834 | sym_get_string_value(menu->sym)); | 794 | sym_get_string_value(menu->sym)); |
835 | switch (res) { | 795 | switch (res) { |
@@ -900,13 +860,9 @@ static void conf_save(void) | |||
900 | } | 860 | } |
901 | } | 861 | } |
902 | 862 | ||
903 | static void conf_cleanup(void) | ||
904 | { | ||
905 | tcsetattr(1, TCSAFLUSH, &ios_org); | ||
906 | } | ||
907 | |||
908 | int main(int ac, char **av) | 863 | int main(int ac, char **av) |
909 | { | 864 | { |
865 | int saved_x, saved_y; | ||
910 | char *mode; | 866 | char *mode; |
911 | int res; | 867 | int res; |
912 | 868 | ||
@@ -923,11 +879,13 @@ int main(int ac, char **av) | |||
923 | single_menu_mode = 1; | 879 | single_menu_mode = 1; |
924 | } | 880 | } |
925 | 881 | ||
926 | tcgetattr(1, &ios_org); | 882 | getyx(stdscr, saved_y, saved_x); |
927 | atexit(conf_cleanup); | 883 | if (init_dialog(NULL)) { |
928 | init_wsize(); | 884 | fprintf(stderr, N_("Your display is too small to run Menuconfig!\n")); |
929 | reset_dialog(); | 885 | fprintf(stderr, N_("It must be at least 19 lines by 80 columns.\n")); |
930 | init_dialog(NULL); | 886 | return 1; |
887 | } | ||
888 | |||
931 | set_config_filename(conf_get_configname()); | 889 | set_config_filename(conf_get_configname()); |
932 | do { | 890 | do { |
933 | conf(&rootmenu); | 891 | conf(&rootmenu); |
@@ -941,7 +899,7 @@ int main(int ac, char **av) | |||
941 | else | 899 | else |
942 | res = -1; | 900 | res = -1; |
943 | } while (res == KEY_ESC); | 901 | } while (res == KEY_ESC); |
944 | end_dialog(); | 902 | end_dialog(saved_x, saved_y); |
945 | 903 | ||
946 | switch (res) { | 904 | switch (res) { |
947 | case 0: | 905 | case 0: |
diff --git a/scripts/kconfig/menu.c b/scripts/kconfig/menu.c index f9d0d91a3fe4..fdad17367f61 100644 --- a/scripts/kconfig/menu.c +++ b/scripts/kconfig/menu.c | |||
@@ -15,7 +15,7 @@ static struct menu **last_entry_ptr; | |||
15 | struct file *file_list; | 15 | struct file *file_list; |
16 | struct file *current_file; | 16 | struct file *current_file; |
17 | 17 | ||
18 | static void menu_warn(struct menu *menu, const char *fmt, ...) | 18 | void menu_warn(struct menu *menu, const char *fmt, ...) |
19 | { | 19 | { |
20 | va_list ap; | 20 | va_list ap; |
21 | va_start(ap, fmt); | 21 | va_start(ap, fmt); |
@@ -172,6 +172,9 @@ void menu_add_option(int token, char *arg) | |||
172 | else if (sym_defconfig_list != current_entry->sym) | 172 | else if (sym_defconfig_list != current_entry->sym) |
173 | zconf_error("trying to redefine defconfig symbol"); | 173 | zconf_error("trying to redefine defconfig symbol"); |
174 | break; | 174 | break; |
175 | case T_OPT_ENV: | ||
176 | prop_add_env(arg); | ||
177 | break; | ||
175 | } | 178 | } |
176 | } | 179 | } |
177 | 180 | ||
@@ -239,9 +242,11 @@ void menu_finalize(struct menu *parent) | |||
239 | for (menu = parent->list; menu; menu = menu->next) { | 242 | for (menu = parent->list; menu; menu = menu->next) { |
240 | if (menu->sym) { | 243 | if (menu->sym) { |
241 | current_entry = parent; | 244 | current_entry = parent; |
242 | menu_set_type(menu->sym->type); | 245 | if (sym->type == S_UNKNOWN) |
246 | menu_set_type(menu->sym->type); | ||
243 | current_entry = menu; | 247 | current_entry = menu; |
244 | menu_set_type(sym->type); | 248 | if (menu->sym->type == S_UNKNOWN) |
249 | menu_set_type(sym->type); | ||
245 | break; | 250 | break; |
246 | } | 251 | } |
247 | } | 252 | } |
@@ -326,12 +331,42 @@ void menu_finalize(struct menu *parent) | |||
326 | "values not supported"); | 331 | "values not supported"); |
327 | } | 332 | } |
328 | current_entry = menu; | 333 | current_entry = menu; |
329 | menu_set_type(sym->type); | 334 | if (menu->sym->type == S_UNKNOWN) |
335 | menu_set_type(sym->type); | ||
336 | /* Non-tristate choice values of tristate choices must | ||
337 | * depend on the choice being set to Y. The choice | ||
338 | * values' dependencies were propagated to their | ||
339 | * properties above, so the change here must be re- | ||
340 | * propagated. */ | ||
341 | if (sym->type == S_TRISTATE && menu->sym->type != S_TRISTATE) { | ||
342 | basedep = expr_alloc_comp(E_EQUAL, sym, &symbol_yes); | ||
343 | basedep = expr_alloc_and(basedep, menu->dep); | ||
344 | basedep = expr_eliminate_dups(basedep); | ||
345 | menu->dep = basedep; | ||
346 | for (prop = menu->sym->prop; prop; prop = prop->next) { | ||
347 | if (prop->menu != menu) | ||
348 | continue; | ||
349 | dep = expr_alloc_and(expr_copy(basedep), | ||
350 | prop->visible.expr); | ||
351 | dep = expr_eliminate_dups(dep); | ||
352 | dep = expr_trans_bool(dep); | ||
353 | prop->visible.expr = dep; | ||
354 | if (prop->type == P_SELECT) { | ||
355 | struct symbol *es = prop_get_symbol(prop); | ||
356 | dep2 = expr_alloc_symbol(menu->sym); | ||
357 | dep = expr_alloc_and(dep2, | ||
358 | expr_copy(dep)); | ||
359 | dep = expr_alloc_or(es->rev_dep.expr, dep); | ||
360 | dep = expr_eliminate_dups(dep); | ||
361 | es->rev_dep.expr = dep; | ||
362 | } | ||
363 | } | ||
364 | } | ||
330 | menu_add_symbol(P_CHOICE, sym, NULL); | 365 | menu_add_symbol(P_CHOICE, sym, NULL); |
331 | prop = sym_get_choice_prop(sym); | 366 | prop = sym_get_choice_prop(sym); |
332 | for (ep = &prop->expr; *ep; ep = &(*ep)->left.expr) | 367 | for (ep = &prop->expr; *ep; ep = &(*ep)->left.expr) |
333 | ; | 368 | ; |
334 | *ep = expr_alloc_one(E_CHOICE, NULL); | 369 | *ep = expr_alloc_one(E_LIST, NULL); |
335 | (*ep)->right.sym = menu->sym; | 370 | (*ep)->right.sym = menu->sym; |
336 | } | 371 | } |
337 | if (menu->list && (!menu->prompt || !menu->prompt->text)) { | 372 | if (menu->list && (!menu->prompt || !menu->prompt->text)) { |
@@ -394,9 +429,9 @@ bool menu_is_visible(struct menu *menu) | |||
394 | const char *menu_get_prompt(struct menu *menu) | 429 | const char *menu_get_prompt(struct menu *menu) |
395 | { | 430 | { |
396 | if (menu->prompt) | 431 | if (menu->prompt) |
397 | return _(menu->prompt->text); | 432 | return menu->prompt->text; |
398 | else if (menu->sym) | 433 | else if (menu->sym) |
399 | return _(menu->sym->name); | 434 | return menu->sym->name; |
400 | return NULL; | 435 | return NULL; |
401 | } | 436 | } |
402 | 437 | ||
diff --git a/scripts/kconfig/qconf.cc b/scripts/kconfig/qconf.cc index b9bb32dfd628..5d0fd38b089b 100644 --- a/scripts/kconfig/qconf.cc +++ b/scripts/kconfig/qconf.cc | |||
@@ -114,7 +114,7 @@ void ConfigItem::updateMenu(void) | |||
114 | 114 | ||
115 | sym = menu->sym; | 115 | sym = menu->sym; |
116 | prop = menu->prompt; | 116 | prop = menu->prompt; |
117 | prompt = QString::fromLocal8Bit(menu_get_prompt(menu)); | 117 | prompt = _(menu_get_prompt(menu)); |
118 | 118 | ||
119 | if (prop) switch (prop->type) { | 119 | if (prop) switch (prop->type) { |
120 | case P_MENU: | 120 | case P_MENU: |
@@ -208,7 +208,7 @@ void ConfigItem::updateMenu(void) | |||
208 | break; | 208 | break; |
209 | } | 209 | } |
210 | if (!sym_has_value(sym) && visible) | 210 | if (!sym_has_value(sym) && visible) |
211 | prompt += " (NEW)"; | 211 | prompt += _(" (NEW)"); |
212 | set_prompt: | 212 | set_prompt: |
213 | setText(promptColIdx, prompt); | 213 | setText(promptColIdx, prompt); |
214 | } | 214 | } |
@@ -346,7 +346,7 @@ ConfigList::ConfigList(ConfigView* p, const char *name) | |||
346 | 346 | ||
347 | for (i = 0; i < colNr; i++) | 347 | for (i = 0; i < colNr; i++) |
348 | colMap[i] = colRevMap[i] = -1; | 348 | colMap[i] = colRevMap[i] = -1; |
349 | addColumn(promptColIdx, "Option"); | 349 | addColumn(promptColIdx, _("Option")); |
350 | 350 | ||
351 | reinit(); | 351 | reinit(); |
352 | } | 352 | } |
@@ -360,14 +360,14 @@ void ConfigList::reinit(void) | |||
360 | removeColumn(nameColIdx); | 360 | removeColumn(nameColIdx); |
361 | 361 | ||
362 | if (showName) | 362 | if (showName) |
363 | addColumn(nameColIdx, "Name"); | 363 | addColumn(nameColIdx, _("Name")); |
364 | if (showRange) { | 364 | if (showRange) { |
365 | addColumn(noColIdx, "N"); | 365 | addColumn(noColIdx, "N"); |
366 | addColumn(modColIdx, "M"); | 366 | addColumn(modColIdx, "M"); |
367 | addColumn(yesColIdx, "Y"); | 367 | addColumn(yesColIdx, "Y"); |
368 | } | 368 | } |
369 | if (showData) | 369 | if (showData) |
370 | addColumn(dataColIdx, "Value"); | 370 | addColumn(dataColIdx, _("Value")); |
371 | 371 | ||
372 | updateListAll(); | 372 | updateListAll(); |
373 | } | 373 | } |
@@ -803,7 +803,7 @@ void ConfigList::contextMenuEvent(QContextMenuEvent *e) | |||
803 | QAction *action; | 803 | QAction *action; |
804 | 804 | ||
805 | headerPopup = new QPopupMenu(this); | 805 | headerPopup = new QPopupMenu(this); |
806 | action = new QAction(NULL, "Show Name", 0, this); | 806 | action = new QAction(NULL, _("Show Name"), 0, this); |
807 | action->setToggleAction(TRUE); | 807 | action->setToggleAction(TRUE); |
808 | connect(action, SIGNAL(toggled(bool)), | 808 | connect(action, SIGNAL(toggled(bool)), |
809 | parent(), SLOT(setShowName(bool))); | 809 | parent(), SLOT(setShowName(bool))); |
@@ -811,7 +811,7 @@ void ConfigList::contextMenuEvent(QContextMenuEvent *e) | |||
811 | action, SLOT(setOn(bool))); | 811 | action, SLOT(setOn(bool))); |
812 | action->setOn(showName); | 812 | action->setOn(showName); |
813 | action->addTo(headerPopup); | 813 | action->addTo(headerPopup); |
814 | action = new QAction(NULL, "Show Range", 0, this); | 814 | action = new QAction(NULL, _("Show Range"), 0, this); |
815 | action->setToggleAction(TRUE); | 815 | action->setToggleAction(TRUE); |
816 | connect(action, SIGNAL(toggled(bool)), | 816 | connect(action, SIGNAL(toggled(bool)), |
817 | parent(), SLOT(setShowRange(bool))); | 817 | parent(), SLOT(setShowRange(bool))); |
@@ -819,7 +819,7 @@ void ConfigList::contextMenuEvent(QContextMenuEvent *e) | |||
819 | action, SLOT(setOn(bool))); | 819 | action, SLOT(setOn(bool))); |
820 | action->setOn(showRange); | 820 | action->setOn(showRange); |
821 | action->addTo(headerPopup); | 821 | action->addTo(headerPopup); |
822 | action = new QAction(NULL, "Show Data", 0, this); | 822 | action = new QAction(NULL, _("Show Data"), 0, this); |
823 | action->setToggleAction(TRUE); | 823 | action->setToggleAction(TRUE); |
824 | connect(action, SIGNAL(toggled(bool)), | 824 | connect(action, SIGNAL(toggled(bool)), |
825 | parent(), SLOT(setShowData(bool))); | 825 | parent(), SLOT(setShowData(bool))); |
@@ -1041,7 +1041,12 @@ void ConfigInfoView::menuInfo(void) | |||
1041 | if (showDebug()) | 1041 | if (showDebug()) |
1042 | debug = debug_info(sym); | 1042 | debug = debug_info(sym); |
1043 | 1043 | ||
1044 | help = print_filter(_(menu_get_help(menu))); | 1044 | help = menu_get_help(menu); |
1045 | /* Gettextize if the help text not empty */ | ||
1046 | if (help.isEmpty()) | ||
1047 | help = print_filter(menu_get_help(menu)); | ||
1048 | else | ||
1049 | help = print_filter(_(menu_get_help(menu))); | ||
1045 | } else if (menu->prompt) { | 1050 | } else if (menu->prompt) { |
1046 | head += "<big><b>"; | 1051 | head += "<big><b>"; |
1047 | head += print_filter(_(menu->prompt->text)); | 1052 | head += print_filter(_(menu->prompt->text)); |
@@ -1083,7 +1088,11 @@ QString ConfigInfoView::debug_info(struct symbol *sym) | |||
1083 | debug += "</a><br>"; | 1088 | debug += "</a><br>"; |
1084 | break; | 1089 | break; |
1085 | case P_DEFAULT: | 1090 | case P_DEFAULT: |
1086 | debug += "default: "; | 1091 | case P_SELECT: |
1092 | case P_RANGE: | ||
1093 | case P_ENV: | ||
1094 | debug += prop_get_type_name(prop->type); | ||
1095 | debug += ": "; | ||
1087 | expr_print(prop->expr, expr_print_help, &debug, E_NONE); | 1096 | expr_print(prop->expr, expr_print_help, &debug, E_NONE); |
1088 | debug += "<br>"; | 1097 | debug += "<br>"; |
1089 | break; | 1098 | break; |
@@ -1094,16 +1103,6 @@ QString ConfigInfoView::debug_info(struct symbol *sym) | |||
1094 | debug += "<br>"; | 1103 | debug += "<br>"; |
1095 | } | 1104 | } |
1096 | break; | 1105 | break; |
1097 | case P_SELECT: | ||
1098 | debug += "select: "; | ||
1099 | expr_print(prop->expr, expr_print_help, &debug, E_NONE); | ||
1100 | debug += "<br>"; | ||
1101 | break; | ||
1102 | case P_RANGE: | ||
1103 | debug += "range: "; | ||
1104 | expr_print(prop->expr, expr_print_help, &debug, E_NONE); | ||
1105 | debug += "<br>"; | ||
1106 | break; | ||
1107 | default: | 1106 | default: |
1108 | debug += "unknown property: "; | 1107 | debug += "unknown property: "; |
1109 | debug += prop_get_type_name(prop->type); | 1108 | debug += prop_get_type_name(prop->type); |
@@ -1167,7 +1166,7 @@ void ConfigInfoView::expr_print_help(void *data, struct symbol *sym, const char | |||
1167 | QPopupMenu* ConfigInfoView::createPopupMenu(const QPoint& pos) | 1166 | QPopupMenu* ConfigInfoView::createPopupMenu(const QPoint& pos) |
1168 | { | 1167 | { |
1169 | QPopupMenu* popup = Parent::createPopupMenu(pos); | 1168 | QPopupMenu* popup = Parent::createPopupMenu(pos); |
1170 | QAction* action = new QAction(NULL,"Show Debug Info", 0, popup); | 1169 | QAction* action = new QAction(NULL, _("Show Debug Info"), 0, popup); |
1171 | action->setToggleAction(TRUE); | 1170 | action->setToggleAction(TRUE); |
1172 | connect(action, SIGNAL(toggled(bool)), SLOT(setShowDebug(bool))); | 1171 | connect(action, SIGNAL(toggled(bool)), SLOT(setShowDebug(bool))); |
1173 | connect(this, SIGNAL(showDebugChanged(bool)), action, SLOT(setOn(bool))); | 1172 | connect(this, SIGNAL(showDebugChanged(bool)), action, SLOT(setOn(bool))); |
@@ -1189,11 +1188,11 @@ ConfigSearchWindow::ConfigSearchWindow(ConfigMainWindow* parent, const char *nam | |||
1189 | 1188 | ||
1190 | QVBoxLayout* layout1 = new QVBoxLayout(this, 11, 6); | 1189 | QVBoxLayout* layout1 = new QVBoxLayout(this, 11, 6); |
1191 | QHBoxLayout* layout2 = new QHBoxLayout(0, 0, 6); | 1190 | QHBoxLayout* layout2 = new QHBoxLayout(0, 0, 6); |
1192 | layout2->addWidget(new QLabel("Find:", this)); | 1191 | layout2->addWidget(new QLabel(_("Find:"), this)); |
1193 | editField = new QLineEdit(this); | 1192 | editField = new QLineEdit(this); |
1194 | connect(editField, SIGNAL(returnPressed()), SLOT(search())); | 1193 | connect(editField, SIGNAL(returnPressed()), SLOT(search())); |
1195 | layout2->addWidget(editField); | 1194 | layout2->addWidget(editField); |
1196 | searchButton = new QPushButton("Search", this); | 1195 | searchButton = new QPushButton(_("Search"), this); |
1197 | searchButton->setAutoDefault(FALSE); | 1196 | searchButton->setAutoDefault(FALSE); |
1198 | connect(searchButton, SIGNAL(clicked()), SLOT(search())); | 1197 | connect(searchButton, SIGNAL(clicked()), SLOT(search())); |
1199 | layout2->addWidget(searchButton); | 1198 | layout2->addWidget(searchButton); |
@@ -1313,58 +1312,58 @@ ConfigMainWindow::ConfigMainWindow(void) | |||
1313 | menu = menuBar(); | 1312 | menu = menuBar(); |
1314 | toolBar = new QToolBar("Tools", this); | 1313 | toolBar = new QToolBar("Tools", this); |
1315 | 1314 | ||
1316 | backAction = new QAction("Back", QPixmap(xpm_back), "Back", 0, this); | 1315 | backAction = new QAction("Back", QPixmap(xpm_back), _("Back"), 0, this); |
1317 | connect(backAction, SIGNAL(activated()), SLOT(goBack())); | 1316 | connect(backAction, SIGNAL(activated()), SLOT(goBack())); |
1318 | backAction->setEnabled(FALSE); | 1317 | backAction->setEnabled(FALSE); |
1319 | QAction *quitAction = new QAction("Quit", "&Quit", CTRL+Key_Q, this); | 1318 | QAction *quitAction = new QAction("Quit", _("&Quit"), CTRL+Key_Q, this); |
1320 | connect(quitAction, SIGNAL(activated()), SLOT(close())); | 1319 | connect(quitAction, SIGNAL(activated()), SLOT(close())); |
1321 | QAction *loadAction = new QAction("Load", QPixmap(xpm_load), "&Load", CTRL+Key_L, this); | 1320 | QAction *loadAction = new QAction("Load", QPixmap(xpm_load), _("&Load"), CTRL+Key_L, this); |
1322 | connect(loadAction, SIGNAL(activated()), SLOT(loadConfig())); | 1321 | connect(loadAction, SIGNAL(activated()), SLOT(loadConfig())); |
1323 | saveAction = new QAction("Save", QPixmap(xpm_save), "&Save", CTRL+Key_S, this); | 1322 | saveAction = new QAction("Save", QPixmap(xpm_save), _("&Save"), CTRL+Key_S, this); |
1324 | connect(saveAction, SIGNAL(activated()), SLOT(saveConfig())); | 1323 | connect(saveAction, SIGNAL(activated()), SLOT(saveConfig())); |
1325 | conf_set_changed_callback(conf_changed); | 1324 | conf_set_changed_callback(conf_changed); |
1326 | // Set saveAction's initial state | 1325 | // Set saveAction's initial state |
1327 | conf_changed(); | 1326 | conf_changed(); |
1328 | QAction *saveAsAction = new QAction("Save As...", "Save &As...", 0, this); | 1327 | QAction *saveAsAction = new QAction("Save As...", _("Save &As..."), 0, this); |
1329 | connect(saveAsAction, SIGNAL(activated()), SLOT(saveConfigAs())); | 1328 | connect(saveAsAction, SIGNAL(activated()), SLOT(saveConfigAs())); |
1330 | QAction *searchAction = new QAction("Find", "&Find", CTRL+Key_F, this); | 1329 | QAction *searchAction = new QAction("Find", _("&Find"), CTRL+Key_F, this); |
1331 | connect(searchAction, SIGNAL(activated()), SLOT(searchConfig())); | 1330 | connect(searchAction, SIGNAL(activated()), SLOT(searchConfig())); |
1332 | QAction *singleViewAction = new QAction("Single View", QPixmap(xpm_single_view), "Split View", 0, this); | 1331 | QAction *singleViewAction = new QAction("Single View", QPixmap(xpm_single_view), _("Single View"), 0, this); |
1333 | connect(singleViewAction, SIGNAL(activated()), SLOT(showSingleView())); | 1332 | connect(singleViewAction, SIGNAL(activated()), SLOT(showSingleView())); |
1334 | QAction *splitViewAction = new QAction("Split View", QPixmap(xpm_split_view), "Split View", 0, this); | 1333 | QAction *splitViewAction = new QAction("Split View", QPixmap(xpm_split_view), _("Split View"), 0, this); |
1335 | connect(splitViewAction, SIGNAL(activated()), SLOT(showSplitView())); | 1334 | connect(splitViewAction, SIGNAL(activated()), SLOT(showSplitView())); |
1336 | QAction *fullViewAction = new QAction("Full View", QPixmap(xpm_tree_view), "Full View", 0, this); | 1335 | QAction *fullViewAction = new QAction("Full View", QPixmap(xpm_tree_view), _("Full View"), 0, this); |
1337 | connect(fullViewAction, SIGNAL(activated()), SLOT(showFullView())); | 1336 | connect(fullViewAction, SIGNAL(activated()), SLOT(showFullView())); |
1338 | 1337 | ||
1339 | QAction *showNameAction = new QAction(NULL, "Show Name", 0, this); | 1338 | QAction *showNameAction = new QAction(NULL, _("Show Name"), 0, this); |
1340 | showNameAction->setToggleAction(TRUE); | 1339 | showNameAction->setToggleAction(TRUE); |
1341 | connect(showNameAction, SIGNAL(toggled(bool)), configView, SLOT(setShowName(bool))); | 1340 | connect(showNameAction, SIGNAL(toggled(bool)), configView, SLOT(setShowName(bool))); |
1342 | connect(configView, SIGNAL(showNameChanged(bool)), showNameAction, SLOT(setOn(bool))); | 1341 | connect(configView, SIGNAL(showNameChanged(bool)), showNameAction, SLOT(setOn(bool))); |
1343 | showNameAction->setOn(configView->showName()); | 1342 | showNameAction->setOn(configView->showName()); |
1344 | QAction *showRangeAction = new QAction(NULL, "Show Range", 0, this); | 1343 | QAction *showRangeAction = new QAction(NULL, _("Show Range"), 0, this); |
1345 | showRangeAction->setToggleAction(TRUE); | 1344 | showRangeAction->setToggleAction(TRUE); |
1346 | connect(showRangeAction, SIGNAL(toggled(bool)), configView, SLOT(setShowRange(bool))); | 1345 | connect(showRangeAction, SIGNAL(toggled(bool)), configView, SLOT(setShowRange(bool))); |
1347 | connect(configView, SIGNAL(showRangeChanged(bool)), showRangeAction, SLOT(setOn(bool))); | 1346 | connect(configView, SIGNAL(showRangeChanged(bool)), showRangeAction, SLOT(setOn(bool))); |
1348 | showRangeAction->setOn(configList->showRange); | 1347 | showRangeAction->setOn(configList->showRange); |
1349 | QAction *showDataAction = new QAction(NULL, "Show Data", 0, this); | 1348 | QAction *showDataAction = new QAction(NULL, _("Show Data"), 0, this); |
1350 | showDataAction->setToggleAction(TRUE); | 1349 | showDataAction->setToggleAction(TRUE); |
1351 | connect(showDataAction, SIGNAL(toggled(bool)), configView, SLOT(setShowData(bool))); | 1350 | connect(showDataAction, SIGNAL(toggled(bool)), configView, SLOT(setShowData(bool))); |
1352 | connect(configView, SIGNAL(showDataChanged(bool)), showDataAction, SLOT(setOn(bool))); | 1351 | connect(configView, SIGNAL(showDataChanged(bool)), showDataAction, SLOT(setOn(bool))); |
1353 | showDataAction->setOn(configList->showData); | 1352 | showDataAction->setOn(configList->showData); |
1354 | QAction *showAllAction = new QAction(NULL, "Show All Options", 0, this); | 1353 | QAction *showAllAction = new QAction(NULL, _("Show All Options"), 0, this); |
1355 | showAllAction->setToggleAction(TRUE); | 1354 | showAllAction->setToggleAction(TRUE); |
1356 | connect(showAllAction, SIGNAL(toggled(bool)), configView, SLOT(setShowAll(bool))); | 1355 | connect(showAllAction, SIGNAL(toggled(bool)), configView, SLOT(setShowAll(bool))); |
1357 | connect(showAllAction, SIGNAL(toggled(bool)), menuView, SLOT(setShowAll(bool))); | 1356 | connect(showAllAction, SIGNAL(toggled(bool)), menuView, SLOT(setShowAll(bool))); |
1358 | showAllAction->setOn(configList->showAll); | 1357 | showAllAction->setOn(configList->showAll); |
1359 | QAction *showDebugAction = new QAction(NULL, "Show Debug Info", 0, this); | 1358 | QAction *showDebugAction = new QAction(NULL, _("Show Debug Info"), 0, this); |
1360 | showDebugAction->setToggleAction(TRUE); | 1359 | showDebugAction->setToggleAction(TRUE); |
1361 | connect(showDebugAction, SIGNAL(toggled(bool)), helpText, SLOT(setShowDebug(bool))); | 1360 | connect(showDebugAction, SIGNAL(toggled(bool)), helpText, SLOT(setShowDebug(bool))); |
1362 | connect(helpText, SIGNAL(showDebugChanged(bool)), showDebugAction, SLOT(setOn(bool))); | 1361 | connect(helpText, SIGNAL(showDebugChanged(bool)), showDebugAction, SLOT(setOn(bool))); |
1363 | showDebugAction->setOn(helpText->showDebug()); | 1362 | showDebugAction->setOn(helpText->showDebug()); |
1364 | 1363 | ||
1365 | QAction *showIntroAction = new QAction(NULL, "Introduction", 0, this); | 1364 | QAction *showIntroAction = new QAction(NULL, _("Introduction"), 0, this); |
1366 | connect(showIntroAction, SIGNAL(activated()), SLOT(showIntro())); | 1365 | connect(showIntroAction, SIGNAL(activated()), SLOT(showIntro())); |
1367 | QAction *showAboutAction = new QAction(NULL, "About", 0, this); | 1366 | QAction *showAboutAction = new QAction(NULL, _("About"), 0, this); |
1368 | connect(showAboutAction, SIGNAL(activated()), SLOT(showAbout())); | 1367 | connect(showAboutAction, SIGNAL(activated()), SLOT(showAbout())); |
1369 | 1368 | ||
1370 | // init tool bar | 1369 | // init tool bar |
@@ -1379,7 +1378,7 @@ ConfigMainWindow::ConfigMainWindow(void) | |||
1379 | 1378 | ||
1380 | // create config menu | 1379 | // create config menu |
1381 | QPopupMenu* config = new QPopupMenu(this); | 1380 | QPopupMenu* config = new QPopupMenu(this); |
1382 | menu->insertItem("&File", config); | 1381 | menu->insertItem(_("&File"), config); |
1383 | loadAction->addTo(config); | 1382 | loadAction->addTo(config); |
1384 | saveAction->addTo(config); | 1383 | saveAction->addTo(config); |
1385 | saveAsAction->addTo(config); | 1384 | saveAsAction->addTo(config); |
@@ -1388,12 +1387,12 @@ ConfigMainWindow::ConfigMainWindow(void) | |||
1388 | 1387 | ||
1389 | // create edit menu | 1388 | // create edit menu |
1390 | QPopupMenu* editMenu = new QPopupMenu(this); | 1389 | QPopupMenu* editMenu = new QPopupMenu(this); |
1391 | menu->insertItem("&Edit", editMenu); | 1390 | menu->insertItem(_("&Edit"), editMenu); |
1392 | searchAction->addTo(editMenu); | 1391 | searchAction->addTo(editMenu); |
1393 | 1392 | ||
1394 | // create options menu | 1393 | // create options menu |
1395 | QPopupMenu* optionMenu = new QPopupMenu(this); | 1394 | QPopupMenu* optionMenu = new QPopupMenu(this); |
1396 | menu->insertItem("&Option", optionMenu); | 1395 | menu->insertItem(_("&Option"), optionMenu); |
1397 | showNameAction->addTo(optionMenu); | 1396 | showNameAction->addTo(optionMenu); |
1398 | showRangeAction->addTo(optionMenu); | 1397 | showRangeAction->addTo(optionMenu); |
1399 | showDataAction->addTo(optionMenu); | 1398 | showDataAction->addTo(optionMenu); |
@@ -1404,7 +1403,7 @@ ConfigMainWindow::ConfigMainWindow(void) | |||
1404 | // create help menu | 1403 | // create help menu |
1405 | QPopupMenu* helpMenu = new QPopupMenu(this); | 1404 | QPopupMenu* helpMenu = new QPopupMenu(this); |
1406 | menu->insertSeparator(); | 1405 | menu->insertSeparator(); |
1407 | menu->insertItem("&Help", helpMenu); | 1406 | menu->insertItem(_("&Help"), helpMenu); |
1408 | showIntroAction->addTo(helpMenu); | 1407 | showIntroAction->addTo(helpMenu); |
1409 | showAboutAction->addTo(helpMenu); | 1408 | showAboutAction->addTo(helpMenu); |
1410 | 1409 | ||
@@ -1452,14 +1451,14 @@ void ConfigMainWindow::loadConfig(void) | |||
1452 | if (s.isNull()) | 1451 | if (s.isNull()) |
1453 | return; | 1452 | return; |
1454 | if (conf_read(QFile::encodeName(s))) | 1453 | if (conf_read(QFile::encodeName(s))) |
1455 | QMessageBox::information(this, "qconf", "Unable to load configuration!"); | 1454 | QMessageBox::information(this, "qconf", _("Unable to load configuration!")); |
1456 | ConfigView::updateListAll(); | 1455 | ConfigView::updateListAll(); |
1457 | } | 1456 | } |
1458 | 1457 | ||
1459 | void ConfigMainWindow::saveConfig(void) | 1458 | void ConfigMainWindow::saveConfig(void) |
1460 | { | 1459 | { |
1461 | if (conf_write(NULL)) | 1460 | if (conf_write(NULL)) |
1462 | QMessageBox::information(this, "qconf", "Unable to save configuration!"); | 1461 | QMessageBox::information(this, "qconf", _("Unable to save configuration!")); |
1463 | } | 1462 | } |
1464 | 1463 | ||
1465 | void ConfigMainWindow::saveConfigAs(void) | 1464 | void ConfigMainWindow::saveConfigAs(void) |
@@ -1468,7 +1467,7 @@ void ConfigMainWindow::saveConfigAs(void) | |||
1468 | if (s.isNull()) | 1467 | if (s.isNull()) |
1469 | return; | 1468 | return; |
1470 | if (conf_write(QFile::encodeName(s))) | 1469 | if (conf_write(QFile::encodeName(s))) |
1471 | QMessageBox::information(this, "qconf", "Unable to save configuration!"); | 1470 | QMessageBox::information(this, "qconf", _("Unable to save configuration!")); |
1472 | } | 1471 | } |
1473 | 1472 | ||
1474 | void ConfigMainWindow::searchConfig(void) | 1473 | void ConfigMainWindow::searchConfig(void) |
@@ -1612,11 +1611,11 @@ void ConfigMainWindow::closeEvent(QCloseEvent* e) | |||
1612 | e->accept(); | 1611 | e->accept(); |
1613 | return; | 1612 | return; |
1614 | } | 1613 | } |
1615 | QMessageBox mb("qconf", "Save configuration?", QMessageBox::Warning, | 1614 | QMessageBox mb("qconf", _("Save configuration?"), QMessageBox::Warning, |
1616 | QMessageBox::Yes | QMessageBox::Default, QMessageBox::No, QMessageBox::Cancel | QMessageBox::Escape); | 1615 | QMessageBox::Yes | QMessageBox::Default, QMessageBox::No, QMessageBox::Cancel | QMessageBox::Escape); |
1617 | mb.setButtonText(QMessageBox::Yes, "&Save Changes"); | 1616 | mb.setButtonText(QMessageBox::Yes, _("&Save Changes")); |
1618 | mb.setButtonText(QMessageBox::No, "&Discard Changes"); | 1617 | mb.setButtonText(QMessageBox::No, _("&Discard Changes")); |
1619 | mb.setButtonText(QMessageBox::Cancel, "Cancel Exit"); | 1618 | mb.setButtonText(QMessageBox::Cancel, _("Cancel Exit")); |
1620 | switch (mb.exec()) { | 1619 | switch (mb.exec()) { |
1621 | case QMessageBox::Yes: | 1620 | case QMessageBox::Yes: |
1622 | conf_write(NULL); | 1621 | conf_write(NULL); |
@@ -1631,7 +1630,7 @@ void ConfigMainWindow::closeEvent(QCloseEvent* e) | |||
1631 | 1630 | ||
1632 | void ConfigMainWindow::showIntro(void) | 1631 | void ConfigMainWindow::showIntro(void) |
1633 | { | 1632 | { |
1634 | static char str[] = "Welcome to the qconf graphical kernel configuration tool for Linux.\n\n" | 1633 | static const QString str = _("Welcome to the qconf graphical kernel configuration tool for Linux.\n\n" |
1635 | "For each option, a blank box indicates the feature is disabled, a check\n" | 1634 | "For each option, a blank box indicates the feature is disabled, a check\n" |
1636 | "indicates it is enabled, and a dot indicates that it is to be compiled\n" | 1635 | "indicates it is enabled, and a dot indicates that it is to be compiled\n" |
1637 | "as a module. Clicking on the box will cycle through the three states.\n\n" | 1636 | "as a module. Clicking on the box will cycle through the three states.\n\n" |
@@ -1641,15 +1640,15 @@ void ConfigMainWindow::showIntro(void) | |||
1641 | "options must be enabled to support the option you are interested in, you can\n" | 1640 | "options must be enabled to support the option you are interested in, you can\n" |
1642 | "still view the help of a grayed-out option.\n\n" | 1641 | "still view the help of a grayed-out option.\n\n" |
1643 | "Toggling Show Debug Info under the Options menu will show the dependencies,\n" | 1642 | "Toggling Show Debug Info under the Options menu will show the dependencies,\n" |
1644 | "which you can then match by examining other options.\n\n"; | 1643 | "which you can then match by examining other options.\n\n"); |
1645 | 1644 | ||
1646 | QMessageBox::information(this, "qconf", str); | 1645 | QMessageBox::information(this, "qconf", str); |
1647 | } | 1646 | } |
1648 | 1647 | ||
1649 | void ConfigMainWindow::showAbout(void) | 1648 | void ConfigMainWindow::showAbout(void) |
1650 | { | 1649 | { |
1651 | static char str[] = "qconf is Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>.\n\n" | 1650 | static const QString str = _("qconf is Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>.\n\n" |
1652 | "Bug reports and feature request can also be entered at http://bugzilla.kernel.org/\n"; | 1651 | "Bug reports and feature request can also be entered at http://bugzilla.kernel.org/\n"); |
1653 | 1652 | ||
1654 | QMessageBox::information(this, "qconf", str); | 1653 | QMessageBox::information(this, "qconf", str); |
1655 | } | 1654 | } |
@@ -1707,7 +1706,7 @@ static const char *progname; | |||
1707 | 1706 | ||
1708 | static void usage(void) | 1707 | static void usage(void) |
1709 | { | 1708 | { |
1710 | printf("%s <config>\n", progname); | 1709 | printf(_("%s <config>\n"), progname); |
1711 | exit(0); | 1710 | exit(0); |
1712 | } | 1711 | } |
1713 | 1712 | ||
diff --git a/scripts/kconfig/symbol.c b/scripts/kconfig/symbol.c index c35dcc5d6189..3929e5b35e79 100644 --- a/scripts/kconfig/symbol.c +++ b/scripts/kconfig/symbol.c | |||
@@ -34,6 +34,8 @@ struct symbol *sym_defconfig_list; | |||
34 | struct symbol *modules_sym; | 34 | struct symbol *modules_sym; |
35 | tristate modules_val; | 35 | tristate modules_val; |
36 | 36 | ||
37 | struct expr *sym_env_list; | ||
38 | |||
37 | void sym_add_default(struct symbol *sym, const char *def) | 39 | void sym_add_default(struct symbol *sym, const char *def) |
38 | { | 40 | { |
39 | struct property *prop = prop_alloc(P_DEFAULT, sym); | 41 | struct property *prop = prop_alloc(P_DEFAULT, sym); |
@@ -45,7 +47,6 @@ void sym_init(void) | |||
45 | { | 47 | { |
46 | struct symbol *sym; | 48 | struct symbol *sym; |
47 | struct utsname uts; | 49 | struct utsname uts; |
48 | char *p; | ||
49 | static bool inited = false; | 50 | static bool inited = false; |
50 | 51 | ||
51 | if (inited) | 52 | if (inited) |
@@ -54,20 +55,6 @@ void sym_init(void) | |||
54 | 55 | ||
55 | uname(&uts); | 56 | uname(&uts); |
56 | 57 | ||
57 | sym = sym_lookup("ARCH", 0); | ||
58 | sym->type = S_STRING; | ||
59 | sym->flags |= SYMBOL_AUTO; | ||
60 | p = getenv("ARCH"); | ||
61 | if (p) | ||
62 | sym_add_default(sym, p); | ||
63 | |||
64 | sym = sym_lookup("KERNELVERSION", 0); | ||
65 | sym->type = S_STRING; | ||
66 | sym->flags |= SYMBOL_AUTO; | ||
67 | p = getenv("KERNELVERSION"); | ||
68 | if (p) | ||
69 | sym_add_default(sym, p); | ||
70 | |||
71 | sym = sym_lookup("UNAME_RELEASE", 0); | 58 | sym = sym_lookup("UNAME_RELEASE", 0); |
72 | sym->type = S_STRING; | 59 | sym->type = S_STRING; |
73 | sym->flags |= SYMBOL_AUTO; | 60 | sym->flags |= SYMBOL_AUTO; |
@@ -117,6 +104,15 @@ struct property *sym_get_choice_prop(struct symbol *sym) | |||
117 | return NULL; | 104 | return NULL; |
118 | } | 105 | } |
119 | 106 | ||
107 | struct property *sym_get_env_prop(struct symbol *sym) | ||
108 | { | ||
109 | struct property *prop; | ||
110 | |||
111 | for_all_properties(sym, prop, P_ENV) | ||
112 | return prop; | ||
113 | return NULL; | ||
114 | } | ||
115 | |||
120 | struct property *sym_get_default_prop(struct symbol *sym) | 116 | struct property *sym_get_default_prop(struct symbol *sym) |
121 | { | 117 | { |
122 | struct property *prop; | 118 | struct property *prop; |
@@ -199,7 +195,7 @@ static void sym_calc_visibility(struct symbol *sym) | |||
199 | tri = no; | 195 | tri = no; |
200 | for_all_prompts(sym, prop) { | 196 | for_all_prompts(sym, prop) { |
201 | prop->visible.tri = expr_calc_value(prop->visible.expr); | 197 | prop->visible.tri = expr_calc_value(prop->visible.expr); |
202 | tri = E_OR(tri, prop->visible.tri); | 198 | tri = EXPR_OR(tri, prop->visible.tri); |
203 | } | 199 | } |
204 | if (tri == mod && (sym->type != S_TRISTATE || modules_val == no)) | 200 | if (tri == mod && (sym->type != S_TRISTATE || modules_val == no)) |
205 | tri = yes; | 201 | tri = yes; |
@@ -247,8 +243,7 @@ static struct symbol *sym_calc_choice(struct symbol *sym) | |||
247 | 243 | ||
248 | /* just get the first visible value */ | 244 | /* just get the first visible value */ |
249 | prop = sym_get_choice_prop(sym); | 245 | prop = sym_get_choice_prop(sym); |
250 | for (e = prop->expr; e; e = e->left.expr) { | 246 | expr_list_for_each_sym(prop->expr, e, def_sym) { |
251 | def_sym = e->right.sym; | ||
252 | sym_calc_visibility(def_sym); | 247 | sym_calc_visibility(def_sym); |
253 | if (def_sym->visible != no) | 248 | if (def_sym->visible != no) |
254 | return def_sym; | 249 | return def_sym; |
@@ -303,7 +298,7 @@ void sym_calc_value(struct symbol *sym) | |||
303 | if (sym_is_choice_value(sym) && sym->visible == yes) { | 298 | if (sym_is_choice_value(sym) && sym->visible == yes) { |
304 | prop = sym_get_choice_prop(sym); | 299 | prop = sym_get_choice_prop(sym); |
305 | newval.tri = (prop_get_symbol(prop)->curr.val == sym) ? yes : no; | 300 | newval.tri = (prop_get_symbol(prop)->curr.val == sym) ? yes : no; |
306 | } else if (E_OR(sym->visible, sym->rev_dep.tri) != no) { | 301 | } else if (EXPR_OR(sym->visible, sym->rev_dep.tri) != no) { |
307 | sym->flags |= SYMBOL_WRITE; | 302 | sym->flags |= SYMBOL_WRITE; |
308 | if (sym_has_value(sym)) | 303 | if (sym_has_value(sym)) |
309 | newval.tri = sym->def[S_DEF_USER].tri; | 304 | newval.tri = sym->def[S_DEF_USER].tri; |
@@ -312,7 +307,7 @@ void sym_calc_value(struct symbol *sym) | |||
312 | if (prop) | 307 | if (prop) |
313 | newval.tri = expr_calc_value(prop->expr); | 308 | newval.tri = expr_calc_value(prop->expr); |
314 | } | 309 | } |
315 | newval.tri = E_OR(E_AND(newval.tri, sym->visible), sym->rev_dep.tri); | 310 | newval.tri = EXPR_OR(EXPR_AND(newval.tri, sym->visible), sym->rev_dep.tri); |
316 | } else if (!sym_is_choice(sym)) { | 311 | } else if (!sym_is_choice(sym)) { |
317 | prop = sym_get_default_prop(sym); | 312 | prop = sym_get_default_prop(sym); |
318 | if (prop) { | 313 | if (prop) { |
@@ -347,6 +342,9 @@ void sym_calc_value(struct symbol *sym) | |||
347 | ; | 342 | ; |
348 | } | 343 | } |
349 | 344 | ||
345 | if (sym->flags & SYMBOL_AUTO) | ||
346 | sym->flags &= ~SYMBOL_WRITE; | ||
347 | |||
350 | sym->curr = newval; | 348 | sym->curr = newval; |
351 | if (sym_is_choice(sym) && newval.tri == yes) | 349 | if (sym_is_choice(sym) && newval.tri == yes) |
352 | sym->curr.val = sym_calc_choice(sym); | 350 | sym->curr.val = sym_calc_choice(sym); |
@@ -361,12 +359,14 @@ void sym_calc_value(struct symbol *sym) | |||
361 | } | 359 | } |
362 | 360 | ||
363 | if (sym_is_choice(sym)) { | 361 | if (sym_is_choice(sym)) { |
362 | struct symbol *choice_sym; | ||
364 | int flags = sym->flags & (SYMBOL_CHANGED | SYMBOL_WRITE); | 363 | int flags = sym->flags & (SYMBOL_CHANGED | SYMBOL_WRITE); |
364 | |||
365 | prop = sym_get_choice_prop(sym); | 365 | prop = sym_get_choice_prop(sym); |
366 | for (e = prop->expr; e; e = e->left.expr) { | 366 | expr_list_for_each_sym(prop->expr, e, choice_sym) { |
367 | e->right.sym->flags |= flags; | 367 | choice_sym->flags |= flags; |
368 | if (flags & SYMBOL_CHANGED) | 368 | if (flags & SYMBOL_CHANGED) |
369 | sym_set_changed(e->right.sym); | 369 | sym_set_changed(choice_sym); |
370 | } | 370 | } |
371 | } | 371 | } |
372 | } | 372 | } |
@@ -849,7 +849,7 @@ struct property *prop_alloc(enum prop_type type, struct symbol *sym) | |||
849 | struct symbol *prop_get_symbol(struct property *prop) | 849 | struct symbol *prop_get_symbol(struct property *prop) |
850 | { | 850 | { |
851 | if (prop->expr && (prop->expr->type == E_SYMBOL || | 851 | if (prop->expr && (prop->expr->type == E_SYMBOL || |
852 | prop->expr->type == E_CHOICE)) | 852 | prop->expr->type == E_LIST)) |
853 | return prop->expr->left.sym; | 853 | return prop->expr->left.sym; |
854 | return NULL; | 854 | return NULL; |
855 | } | 855 | } |
@@ -859,6 +859,8 @@ const char *prop_get_type_name(enum prop_type type) | |||
859 | switch (type) { | 859 | switch (type) { |
860 | case P_PROMPT: | 860 | case P_PROMPT: |
861 | return "prompt"; | 861 | return "prompt"; |
862 | case P_ENV: | ||
863 | return "env"; | ||
862 | case P_COMMENT: | 864 | case P_COMMENT: |
863 | return "comment"; | 865 | return "comment"; |
864 | case P_MENU: | 866 | case P_MENU: |
@@ -876,3 +878,32 @@ const char *prop_get_type_name(enum prop_type type) | |||
876 | } | 878 | } |
877 | return "unknown"; | 879 | return "unknown"; |
878 | } | 880 | } |
881 | |||
882 | void prop_add_env(const char *env) | ||
883 | { | ||
884 | struct symbol *sym, *sym2; | ||
885 | struct property *prop; | ||
886 | char *p; | ||
887 | |||
888 | sym = current_entry->sym; | ||
889 | sym->flags |= SYMBOL_AUTO; | ||
890 | for_all_properties(sym, prop, P_ENV) { | ||
891 | sym2 = prop_get_symbol(prop); | ||
892 | if (strcmp(sym2->name, env)) | ||
893 | menu_warn(current_entry, "redefining environment symbol from %s", | ||
894 | sym2->name); | ||
895 | return; | ||
896 | } | ||
897 | |||
898 | prop = prop_alloc(P_ENV, sym); | ||
899 | prop->expr = expr_alloc_symbol(sym_lookup(env, 1)); | ||
900 | |||
901 | sym_env_list = expr_alloc_one(E_LIST, sym_env_list); | ||
902 | sym_env_list->right.sym = sym; | ||
903 | |||
904 | p = getenv(env); | ||
905 | if (p) | ||
906 | sym_add_default(sym, p); | ||
907 | else | ||
908 | menu_warn(current_entry, "environment variable %s undefined", env); | ||
909 | } | ||
diff --git a/scripts/kconfig/util.c b/scripts/kconfig/util.c index e1cad924c0a4..f8e73c039dc8 100644 --- a/scripts/kconfig/util.c +++ b/scripts/kconfig/util.c | |||
@@ -29,6 +29,8 @@ struct file *file_lookup(const char *name) | |||
29 | /* write a dependency file as used by kbuild to track dependencies */ | 29 | /* write a dependency file as used by kbuild to track dependencies */ |
30 | int file_write_dep(const char *name) | 30 | int file_write_dep(const char *name) |
31 | { | 31 | { |
32 | struct symbol *sym, *env_sym; | ||
33 | struct expr *e; | ||
32 | struct file *file; | 34 | struct file *file; |
33 | FILE *out; | 35 | FILE *out; |
34 | 36 | ||
@@ -45,8 +47,25 @@ int file_write_dep(const char *name) | |||
45 | fprintf(out, "\t%s\n", file->name); | 47 | fprintf(out, "\t%s\n", file->name); |
46 | } | 48 | } |
47 | fprintf(out, "\ninclude/config/auto.conf: \\\n" | 49 | fprintf(out, "\ninclude/config/auto.conf: \\\n" |
48 | "\t$(deps_config)\n\n" | 50 | "\t$(deps_config)\n\n"); |
49 | "$(deps_config): ;\n"); | 51 | |
52 | expr_list_for_each_sym(sym_env_list, e, sym) { | ||
53 | struct property *prop; | ||
54 | const char *value; | ||
55 | |||
56 | prop = sym_get_env_prop(sym); | ||
57 | env_sym = prop_get_symbol(prop); | ||
58 | if (!env_sym) | ||
59 | continue; | ||
60 | value = getenv(env_sym->name); | ||
61 | if (!value) | ||
62 | value = ""; | ||
63 | fprintf(out, "ifneq \"$(%s)\" \"%s\"\n", env_sym->name, value); | ||
64 | fprintf(out, "include/config/auto.conf: FORCE\n"); | ||
65 | fprintf(out, "endif\n"); | ||
66 | } | ||
67 | |||
68 | fprintf(out, "\n$(deps_config): ;\n"); | ||
50 | fclose(out); | 69 | fclose(out); |
51 | rename("..config.tmp", name); | 70 | rename("..config.tmp", name); |
52 | return 0; | 71 | return 0; |
diff --git a/scripts/kconfig/zconf.gperf b/scripts/kconfig/zconf.gperf index 93538e567bda..25ef5d01c0af 100644 --- a/scripts/kconfig/zconf.gperf +++ b/scripts/kconfig/zconf.gperf | |||
@@ -35,10 +35,10 @@ int, T_TYPE, TF_COMMAND, S_INT | |||
35 | hex, T_TYPE, TF_COMMAND, S_HEX | 35 | hex, T_TYPE, TF_COMMAND, S_HEX |
36 | string, T_TYPE, TF_COMMAND, S_STRING | 36 | string, T_TYPE, TF_COMMAND, S_STRING |
37 | select, T_SELECT, TF_COMMAND | 37 | select, T_SELECT, TF_COMMAND |
38 | enable, T_SELECT, TF_COMMAND | ||
39 | range, T_RANGE, TF_COMMAND | 38 | range, T_RANGE, TF_COMMAND |
40 | option, T_OPTION, TF_COMMAND | 39 | option, T_OPTION, TF_COMMAND |
41 | on, T_ON, TF_PARAM | 40 | on, T_ON, TF_PARAM |
42 | modules, T_OPT_MODULES, TF_OPTION | 41 | modules, T_OPT_MODULES, TF_OPTION |
43 | defconfig_list, T_OPT_DEFCONFIG_LIST,TF_OPTION | 42 | defconfig_list, T_OPT_DEFCONFIG_LIST,TF_OPTION |
43 | env, T_OPT_ENV, TF_OPTION | ||
44 | %% | 44 | %% |
diff --git a/scripts/kconfig/zconf.hash.c_shipped b/scripts/kconfig/zconf.hash.c_shipped index ab28b18153a7..5c73d51339d8 100644 --- a/scripts/kconfig/zconf.hash.c_shipped +++ b/scripts/kconfig/zconf.hash.c_shipped | |||
@@ -1,4 +1,4 @@ | |||
1 | /* ANSI-C code produced by gperf version 3.0.2 */ | 1 | /* ANSI-C code produced by gperf version 3.0.3 */ |
2 | /* Command-line: gperf */ | 2 | /* Command-line: gperf */ |
3 | /* Computed positions: -k'1,3' */ | 3 | /* Computed positions: -k'1,3' */ |
4 | 4 | ||
@@ -53,9 +53,9 @@ kconf_id_hash (register const char *str, register unsigned int len) | |||
53 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, | 53 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, |
54 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, | 54 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, |
55 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, | 55 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, |
56 | 49, 49, 49, 49, 49, 49, 49, 18, 11, 5, | 56 | 49, 49, 49, 49, 49, 49, 49, 49, 11, 5, |
57 | 0, 0, 5, 49, 5, 20, 49, 49, 5, 20, | 57 | 0, 0, 5, 49, 5, 20, 49, 49, 5, 20, |
58 | 5, 0, 30, 49, 0, 15, 0, 10, 49, 49, | 58 | 5, 0, 30, 49, 0, 15, 0, 10, 0, 49, |
59 | 25, 49, 49, 49, 49, 49, 49, 49, 49, 49, | 59 | 25, 49, 49, 49, 49, 49, 49, 49, 49, 49, |
60 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, | 60 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, |
61 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, | 61 | 49, 49, 49, 49, 49, 49, 49, 49, 49, 49, |
@@ -89,6 +89,7 @@ kconf_id_hash (register const char *str, register unsigned int len) | |||
89 | struct kconf_id_strings_t | 89 | struct kconf_id_strings_t |
90 | { | 90 | { |
91 | char kconf_id_strings_str2[sizeof("on")]; | 91 | char kconf_id_strings_str2[sizeof("on")]; |
92 | char kconf_id_strings_str3[sizeof("env")]; | ||
92 | char kconf_id_strings_str5[sizeof("endif")]; | 93 | char kconf_id_strings_str5[sizeof("endif")]; |
93 | char kconf_id_strings_str6[sizeof("option")]; | 94 | char kconf_id_strings_str6[sizeof("option")]; |
94 | char kconf_id_strings_str7[sizeof("endmenu")]; | 95 | char kconf_id_strings_str7[sizeof("endmenu")]; |
@@ -107,7 +108,6 @@ struct kconf_id_strings_t | |||
107 | char kconf_id_strings_str21[sizeof("string")]; | 108 | char kconf_id_strings_str21[sizeof("string")]; |
108 | char kconf_id_strings_str22[sizeof("if")]; | 109 | char kconf_id_strings_str22[sizeof("if")]; |
109 | char kconf_id_strings_str23[sizeof("int")]; | 110 | char kconf_id_strings_str23[sizeof("int")]; |
110 | char kconf_id_strings_str24[sizeof("enable")]; | ||
111 | char kconf_id_strings_str26[sizeof("select")]; | 111 | char kconf_id_strings_str26[sizeof("select")]; |
112 | char kconf_id_strings_str27[sizeof("modules")]; | 112 | char kconf_id_strings_str27[sizeof("modules")]; |
113 | char kconf_id_strings_str28[sizeof("tristate")]; | 113 | char kconf_id_strings_str28[sizeof("tristate")]; |
@@ -123,6 +123,7 @@ struct kconf_id_strings_t | |||
123 | static struct kconf_id_strings_t kconf_id_strings_contents = | 123 | static struct kconf_id_strings_t kconf_id_strings_contents = |
124 | { | 124 | { |
125 | "on", | 125 | "on", |
126 | "env", | ||
126 | "endif", | 127 | "endif", |
127 | "option", | 128 | "option", |
128 | "endmenu", | 129 | "endmenu", |
@@ -141,7 +142,6 @@ static struct kconf_id_strings_t kconf_id_strings_contents = | |||
141 | "string", | 142 | "string", |
142 | "if", | 143 | "if", |
143 | "int", | 144 | "int", |
144 | "enable", | ||
145 | "select", | 145 | "select", |
146 | "modules", | 146 | "modules", |
147 | "tristate", | 147 | "tristate", |
@@ -157,6 +157,9 @@ static struct kconf_id_strings_t kconf_id_strings_contents = | |||
157 | #define kconf_id_strings ((const char *) &kconf_id_strings_contents) | 157 | #define kconf_id_strings ((const char *) &kconf_id_strings_contents) |
158 | #ifdef __GNUC__ | 158 | #ifdef __GNUC__ |
159 | __inline | 159 | __inline |
160 | #ifdef __GNUC_STDC_INLINE__ | ||
161 | __attribute__ ((__gnu_inline__)) | ||
162 | #endif | ||
160 | #endif | 163 | #endif |
161 | struct kconf_id * | 164 | struct kconf_id * |
162 | kconf_id_lookup (register const char *str, register unsigned int len) | 165 | kconf_id_lookup (register const char *str, register unsigned int len) |
@@ -174,7 +177,8 @@ kconf_id_lookup (register const char *str, register unsigned int len) | |||
174 | { | 177 | { |
175 | {-1}, {-1}, | 178 | {-1}, {-1}, |
176 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str2, T_ON, TF_PARAM}, | 179 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str2, T_ON, TF_PARAM}, |
177 | {-1}, {-1}, | 180 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str3, T_OPT_ENV, TF_OPTION}, |
181 | {-1}, | ||
178 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str5, T_ENDIF, TF_COMMAND}, | 182 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str5, T_ENDIF, TF_COMMAND}, |
179 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str6, T_OPTION, TF_COMMAND}, | 183 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str6, T_OPTION, TF_COMMAND}, |
180 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str7, T_ENDMENU, TF_COMMAND}, | 184 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str7, T_ENDMENU, TF_COMMAND}, |
@@ -194,8 +198,7 @@ kconf_id_lookup (register const char *str, register unsigned int len) | |||
194 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str21, T_TYPE, TF_COMMAND, S_STRING}, | 198 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str21, T_TYPE, TF_COMMAND, S_STRING}, |
195 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str22, T_IF, TF_COMMAND|TF_PARAM}, | 199 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str22, T_IF, TF_COMMAND|TF_PARAM}, |
196 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str23, T_TYPE, TF_COMMAND, S_INT}, | 200 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str23, T_TYPE, TF_COMMAND, S_INT}, |
197 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str24, T_SELECT, TF_COMMAND}, | 201 | {-1}, {-1}, |
198 | {-1}, | ||
199 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str26, T_SELECT, TF_COMMAND}, | 202 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str26, T_SELECT, TF_COMMAND}, |
200 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str27, T_OPT_MODULES, TF_OPTION}, | 203 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str27, T_OPT_MODULES, TF_OPTION}, |
201 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str28, T_TYPE, TF_COMMAND, S_TRISTATE}, | 204 | {(int)(long)&((struct kconf_id_strings_t *)0)->kconf_id_strings_str28, T_TYPE, TF_COMMAND, S_TRISTATE}, |
diff --git a/scripts/kconfig/zconf.l b/scripts/kconfig/zconf.l index 187d38ccadd5..4cea5c85cd0a 100644 --- a/scripts/kconfig/zconf.l +++ b/scripts/kconfig/zconf.l | |||
@@ -217,6 +217,11 @@ n [A-Za-z0-9_] | |||
217 | append_string("\n", 1); | 217 | append_string("\n", 1); |
218 | } | 218 | } |
219 | [^ \t\n].* { | 219 | [^ \t\n].* { |
220 | while (yyleng) { | ||
221 | if ((yytext[yyleng-1] != ' ') && (yytext[yyleng-1] != '\t')) | ||
222 | break; | ||
223 | yyleng--; | ||
224 | } | ||
220 | append_string(yytext, yyleng); | 225 | append_string(yytext, yyleng); |
221 | if (!first_ts) | 226 | if (!first_ts) |
222 | first_ts = last_ts; | 227 | first_ts = last_ts; |
diff --git a/scripts/kernel-doc b/scripts/kernel-doc index 1d1401807e95..ec54f12f57b0 100755 --- a/scripts/kernel-doc +++ b/scripts/kernel-doc | |||
@@ -46,21 +46,24 @@ use strict; | |||
46 | # Note: This only supports 'c'. | 46 | # Note: This only supports 'c'. |
47 | 47 | ||
48 | # usage: | 48 | # usage: |
49 | # kernel-doc [ -docbook | -html | -text | -man ] | 49 | # kernel-doc [ -docbook | -html | -text | -man ] [ -no-doc-sections ] |
50 | # [ -function funcname [ -function funcname ...] ] c file(s)s > outputfile | 50 | # [ -function funcname [ -function funcname ...] ] c file(s)s > outputfile |
51 | # or | 51 | # or |
52 | # [ -nofunction funcname [ -function funcname ...] ] c file(s)s > outputfile | 52 | # [ -nofunction funcname [ -function funcname ...] ] c file(s)s > outputfile |
53 | # | 53 | # |
54 | # Set output format using one of -docbook -html -text or -man. Default is man. | 54 | # Set output format using one of -docbook -html -text or -man. Default is man. |
55 | # | 55 | # |
56 | # -no-doc-sections | ||
57 | # Do not output DOC: sections | ||
58 | # | ||
56 | # -function funcname | 59 | # -function funcname |
57 | # If set, then only generate documentation for the given function(s). All | 60 | # If set, then only generate documentation for the given function(s) or |
58 | # other functions are ignored. | 61 | # DOC: section titles. All other functions and DOC: sections are ignored. |
59 | # | 62 | # |
60 | # -nofunction funcname | 63 | # -nofunction funcname |
61 | # If set, then only generate documentation for the other function(s). | 64 | # If set, then only generate documentation for the other function(s)/DOC: |
62 | # Cannot be used together with -function | 65 | # sections. Cannot be used together with -function (yes, that's a bug -- |
63 | # (yes, that's a bug -- perl hackers can fix it 8)) | 66 | # perl hackers can fix it 8)) |
64 | # | 67 | # |
65 | # c files - list of 'c' files to process | 68 | # c files - list of 'c' files to process |
66 | # | 69 | # |
@@ -182,10 +185,10 @@ my $blankline_html = $local_lt . "p" . $local_gt; # was "<p>" | |||
182 | my %highlights_xml = ( "([^=])\\\"([^\\\"<]+)\\\"", "\$1<quote>\$2</quote>", | 185 | my %highlights_xml = ( "([^=])\\\"([^\\\"<]+)\\\"", "\$1<quote>\$2</quote>", |
183 | $type_constant, "<constant>\$1</constant>", | 186 | $type_constant, "<constant>\$1</constant>", |
184 | $type_func, "<function>\$1</function>", | 187 | $type_func, "<function>\$1</function>", |
185 | $type_struct, "<structname>\$1</structname>", | 188 | $type_struct_xml, "<structname>\$1</structname>", |
186 | $type_env, "<envar>\$1</envar>", | 189 | $type_env, "<envar>\$1</envar>", |
187 | $type_param, "<parameter>\$1</parameter>" ); | 190 | $type_param, "<parameter>\$1</parameter>" ); |
188 | my $blankline_xml = "</para><para>\n"; | 191 | my $blankline_xml = $local_lt . "/para" . $local_gt . $local_lt . "para" . $local_gt . "\n"; |
189 | 192 | ||
190 | # gnome, docbook format | 193 | # gnome, docbook format |
191 | my %highlights_gnome = ( $type_constant, "<replaceable class=\"option\">\$1</replaceable>", | 194 | my %highlights_gnome = ( $type_constant, "<replaceable class=\"option\">\$1</replaceable>", |
@@ -211,7 +214,7 @@ my $blankline_text = ""; | |||
211 | 214 | ||
212 | 215 | ||
213 | sub usage { | 216 | sub usage { |
214 | print "Usage: $0 [ -v ] [ -docbook | -html | -text | -man ]\n"; | 217 | print "Usage: $0 [ -v ] [ -docbook | -html | -text | -man ] [ -no-doc-sections ]\n"; |
215 | print " [ -function funcname [ -function funcname ...] ]\n"; | 218 | print " [ -function funcname [ -function funcname ...] ]\n"; |
216 | print " [ -nofunction funcname [ -nofunction funcname ...] ]\n"; | 219 | print " [ -nofunction funcname [ -nofunction funcname ...] ]\n"; |
217 | print " c source file(s) > outputfile\n"; | 220 | print " c source file(s) > outputfile\n"; |
@@ -225,6 +228,7 @@ if ($#ARGV==-1) { | |||
225 | 228 | ||
226 | my $verbose = 0; | 229 | my $verbose = 0; |
227 | my $output_mode = "man"; | 230 | my $output_mode = "man"; |
231 | my $no_doc_sections = 0; | ||
228 | my %highlights = %highlights_man; | 232 | my %highlights = %highlights_man; |
229 | my $blankline = $blankline_man; | 233 | my $blankline = $blankline_man; |
230 | my $modulename = "Kernel API"; | 234 | my $modulename = "Kernel API"; |
@@ -329,12 +333,14 @@ while ($ARGV[0] =~ m/^-(.*)/) { | |||
329 | usage(); | 333 | usage(); |
330 | } elsif ($cmd eq '-filelist') { | 334 | } elsif ($cmd eq '-filelist') { |
331 | $filelist = shift @ARGV; | 335 | $filelist = shift @ARGV; |
336 | } elsif ($cmd eq '-no-doc-sections') { | ||
337 | $no_doc_sections = 1; | ||
332 | } | 338 | } |
333 | } | 339 | } |
334 | 340 | ||
335 | # get kernel version from env | 341 | # get kernel version from env |
336 | sub get_kernel_version() { | 342 | sub get_kernel_version() { |
337 | my $version; | 343 | my $version = 'unknown kernel version'; |
338 | 344 | ||
339 | if (defined($ENV{'KERNELVERSION'})) { | 345 | if (defined($ENV{'KERNELVERSION'})) { |
340 | $version = $ENV{'KERNELVERSION'}; | 346 | $version = $ENV{'KERNELVERSION'}; |
@@ -374,6 +380,29 @@ sub dump_section { | |||
374 | } | 380 | } |
375 | 381 | ||
376 | ## | 382 | ## |
383 | # dump DOC: section after checking that it should go out | ||
384 | # | ||
385 | sub dump_doc_section { | ||
386 | my $name = shift; | ||
387 | my $contents = join "\n", @_; | ||
388 | |||
389 | if ($no_doc_sections) { | ||
390 | return; | ||
391 | } | ||
392 | |||
393 | if (($function_only == 0) || | ||
394 | ( $function_only == 1 && defined($function_table{$name})) || | ||
395 | ( $function_only == 2 && !defined($function_table{$name}))) | ||
396 | { | ||
397 | dump_section $name, $contents; | ||
398 | output_blockhead({'sectionlist' => \@sectionlist, | ||
399 | 'sections' => \%sections, | ||
400 | 'module' => $modulename, | ||
401 | 'content-only' => ($function_only != 0), }); | ||
402 | } | ||
403 | } | ||
404 | |||
405 | ## | ||
377 | # output function | 406 | # output function |
378 | # | 407 | # |
379 | # parameterdescs, a hash. | 408 | # parameterdescs, a hash. |
@@ -394,7 +423,7 @@ sub output_highlight { | |||
394 | # confess "output_highlight got called with no args?\n"; | 423 | # confess "output_highlight got called with no args?\n"; |
395 | # } | 424 | # } |
396 | 425 | ||
397 | if ($output_mode eq "html") { | 426 | if ($output_mode eq "html" || $output_mode eq "xml") { |
398 | $contents = local_unescape($contents); | 427 | $contents = local_unescape($contents); |
399 | # convert data read & converted thru xml_escape() into &xyz; format: | 428 | # convert data read & converted thru xml_escape() into &xyz; format: |
400 | $contents =~ s/\\\\\\/&/g; | 429 | $contents =~ s/\\\\\\/&/g; |
@@ -564,8 +593,8 @@ sub output_function_html(%) { | |||
564 | print "<hr>\n"; | 593 | print "<hr>\n"; |
565 | } | 594 | } |
566 | 595 | ||
567 | # output intro in html | 596 | # output DOC: block header in html |
568 | sub output_intro_html(%) { | 597 | sub output_blockhead_html(%) { |
569 | my %args = %{$_[0]}; | 598 | my %args = %{$_[0]}; |
570 | my ($parameter, $section); | 599 | my ($parameter, $section); |
571 | my $count; | 600 | my $count; |
@@ -871,7 +900,7 @@ sub output_typedef_xml(%) { | |||
871 | } | 900 | } |
872 | 901 | ||
873 | # output in XML DocBook | 902 | # output in XML DocBook |
874 | sub output_intro_xml(%) { | 903 | sub output_blockhead_xml(%) { |
875 | my %args = %{$_[0]}; | 904 | my %args = %{$_[0]}; |
876 | my ($parameter, $section); | 905 | my ($parameter, $section); |
877 | my $count; | 906 | my $count; |
@@ -882,15 +911,23 @@ sub output_intro_xml(%) { | |||
882 | # print out each section | 911 | # print out each section |
883 | $lineprefix=" "; | 912 | $lineprefix=" "; |
884 | foreach $section (@{$args{'sectionlist'}}) { | 913 | foreach $section (@{$args{'sectionlist'}}) { |
885 | print "<refsect1>\n <title>$section</title>\n <para>\n"; | 914 | if (!$args{'content-only'}) { |
915 | print "<refsect1>\n <title>$section</title>\n"; | ||
916 | } | ||
886 | if ($section =~ m/EXAMPLE/i) { | 917 | if ($section =~ m/EXAMPLE/i) { |
887 | print "<example><para>\n"; | 918 | print "<example><para>\n"; |
919 | } else { | ||
920 | print "<para>\n"; | ||
888 | } | 921 | } |
889 | output_highlight($args{'sections'}{$section}); | 922 | output_highlight($args{'sections'}{$section}); |
890 | if ($section =~ m/EXAMPLE/i) { | 923 | if ($section =~ m/EXAMPLE/i) { |
891 | print "</para></example>\n"; | 924 | print "</para></example>\n"; |
925 | } else { | ||
926 | print "</para>"; | ||
927 | } | ||
928 | if (!$args{'content-only'}) { | ||
929 | print "\n</refsect1>\n"; | ||
892 | } | 930 | } |
893 | print " </para>\n</refsect1>\n"; | ||
894 | } | 931 | } |
895 | 932 | ||
896 | print "\n\n"; | 933 | print "\n\n"; |
@@ -1137,7 +1174,7 @@ sub output_typedef_man(%) { | |||
1137 | } | 1174 | } |
1138 | } | 1175 | } |
1139 | 1176 | ||
1140 | sub output_intro_man(%) { | 1177 | sub output_blockhead_man(%) { |
1141 | my %args = %{$_[0]}; | 1178 | my %args = %{$_[0]}; |
1142 | my ($parameter, $section); | 1179 | my ($parameter, $section); |
1143 | my $count; | 1180 | my $count; |
@@ -1294,7 +1331,7 @@ sub output_struct_text(%) { | |||
1294 | output_section_text(@_); | 1331 | output_section_text(@_); |
1295 | } | 1332 | } |
1296 | 1333 | ||
1297 | sub output_intro_text(%) { | 1334 | sub output_blockhead_text(%) { |
1298 | my %args = %{$_[0]}; | 1335 | my %args = %{$_[0]}; |
1299 | my ($parameter, $section); | 1336 | my ($parameter, $section); |
1300 | 1337 | ||
@@ -1325,9 +1362,9 @@ sub output_declaration { | |||
1325 | 1362 | ||
1326 | ## | 1363 | ## |
1327 | # generic output function - calls the right one based on current output mode. | 1364 | # generic output function - calls the right one based on current output mode. |
1328 | sub output_intro { | 1365 | sub output_blockhead { |
1329 | no strict 'refs'; | 1366 | no strict 'refs'; |
1330 | my $func = "output_intro_".$output_mode; | 1367 | my $func = "output_blockhead_".$output_mode; |
1331 | &$func(@_); | 1368 | &$func(@_); |
1332 | $section_counter++; | 1369 | $section_counter++; |
1333 | } | 1370 | } |
@@ -1926,9 +1963,7 @@ sub process_file($) { | |||
1926 | } elsif ($state == 4) { | 1963 | } elsif ($state == 4) { |
1927 | # Documentation block | 1964 | # Documentation block |
1928 | if (/$doc_block/) { | 1965 | if (/$doc_block/) { |
1929 | dump_section($section, xml_escape($contents)); | 1966 | dump_doc_section($section, xml_escape($contents)); |
1930 | output_intro({'sectionlist' => \@sectionlist, | ||
1931 | 'sections' => \%sections }); | ||
1932 | $contents = ""; | 1967 | $contents = ""; |
1933 | $function = ""; | 1968 | $function = ""; |
1934 | %constants = (); | 1969 | %constants = (); |
@@ -1946,9 +1981,7 @@ sub process_file($) { | |||
1946 | } | 1981 | } |
1947 | elsif (/$doc_end/) | 1982 | elsif (/$doc_end/) |
1948 | { | 1983 | { |
1949 | dump_section($section, xml_escape($contents)); | 1984 | dump_doc_section($section, xml_escape($contents)); |
1950 | output_intro({'sectionlist' => \@sectionlist, | ||
1951 | 'sections' => \%sections }); | ||
1952 | $contents = ""; | 1985 | $contents = ""; |
1953 | $function = ""; | 1986 | $function = ""; |
1954 | %constants = (); | 1987 | %constants = (); |
diff --git a/scripts/mkmakefile b/scripts/mkmakefile index e0f54b9d8fec..e65d8b33faa4 100644 --- a/scripts/mkmakefile +++ b/scripts/mkmakefile | |||
@@ -25,8 +25,11 @@ cat << EOF > $2/Makefile | |||
25 | VERSION = $3 | 25 | VERSION = $3 |
26 | PATCHLEVEL = $4 | 26 | PATCHLEVEL = $4 |
27 | 27 | ||
28 | KERNELSRC := $1 | 28 | lastword = \$(word \$(words \$(1)),\$(1)) |
29 | KERNELOUTPUT := $2 | 29 | makedir := \$(dir \$(call lastword,\$(MAKEFILE_LIST))) |
30 | |||
31 | MAKEARGS := -C $1 | ||
32 | MAKEARGS += O=\$(if \$(patsubst /%,,\$(makedir)),\$(CURDIR)/)\$(patsubst %/,%,\$(makedir)) | ||
30 | 33 | ||
31 | MAKEFLAGS += --no-print-directory | 34 | MAKEFLAGS += --no-print-directory |
32 | 35 | ||
@@ -35,10 +38,11 @@ MAKEFLAGS += --no-print-directory | |||
35 | all := \$(filter-out all Makefile,\$(MAKECMDGOALS)) | 38 | all := \$(filter-out all Makefile,\$(MAKECMDGOALS)) |
36 | 39 | ||
37 | all: | 40 | all: |
38 | \$(MAKE) -C \$(KERNELSRC) O=\$(KERNELOUTPUT) \$(all) | 41 | \$(MAKE) \$(MAKEARGS) \$(all) |
39 | 42 | ||
40 | Makefile:; | 43 | Makefile:; |
41 | 44 | ||
42 | \$(all) %/: all | 45 | \$(all) %/: all |
43 | @: | 46 | @: |
47 | |||
44 | EOF | 48 | EOF |
diff --git a/scripts/mod/modpost.c b/scripts/mod/modpost.c index 93ac52adb498..f8efc93eb700 100644 --- a/scripts/mod/modpost.c +++ b/scripts/mod/modpost.c | |||
@@ -2,7 +2,7 @@ | |||
2 | * | 2 | * |
3 | * Copyright 2003 Kai Germaschewski | 3 | * Copyright 2003 Kai Germaschewski |
4 | * Copyright 2002-2004 Rusty Russell, IBM Corporation | 4 | * Copyright 2002-2004 Rusty Russell, IBM Corporation |
5 | * Copyright 2006 Sam Ravnborg | 5 | * Copyright 2006-2008 Sam Ravnborg |
6 | * Based in part on module-init-tools/depmod.c,file2alias | 6 | * Based in part on module-init-tools/depmod.c,file2alias |
7 | * | 7 | * |
8 | * This software may be used and distributed according to the terms | 8 | * This software may be used and distributed according to the terms |
@@ -28,12 +28,17 @@ static int vmlinux_section_warnings = 1; | |||
28 | /* Only warn about unresolved symbols */ | 28 | /* Only warn about unresolved symbols */ |
29 | static int warn_unresolved = 0; | 29 | static int warn_unresolved = 0; |
30 | /* How a symbol is exported */ | 30 | /* How a symbol is exported */ |
31 | static int sec_mismatch_count = 0; | ||
32 | static int sec_mismatch_verbose = 1; | ||
33 | |||
31 | enum export { | 34 | enum export { |
32 | export_plain, export_unused, export_gpl, | 35 | export_plain, export_unused, export_gpl, |
33 | export_unused_gpl, export_gpl_future, export_unknown | 36 | export_unused_gpl, export_gpl_future, export_unknown |
34 | }; | 37 | }; |
35 | 38 | ||
36 | void fatal(const char *fmt, ...) | 39 | #define PRINTF __attribute__ ((format (printf, 1, 2))) |
40 | |||
41 | PRINTF void fatal(const char *fmt, ...) | ||
37 | { | 42 | { |
38 | va_list arglist; | 43 | va_list arglist; |
39 | 44 | ||
@@ -46,7 +51,7 @@ void fatal(const char *fmt, ...) | |||
46 | exit(1); | 51 | exit(1); |
47 | } | 52 | } |
48 | 53 | ||
49 | void warn(const char *fmt, ...) | 54 | PRINTF void warn(const char *fmt, ...) |
50 | { | 55 | { |
51 | va_list arglist; | 56 | va_list arglist; |
52 | 57 | ||
@@ -57,7 +62,7 @@ void warn(const char *fmt, ...) | |||
57 | va_end(arglist); | 62 | va_end(arglist); |
58 | } | 63 | } |
59 | 64 | ||
60 | void merror(const char *fmt, ...) | 65 | PRINTF void merror(const char *fmt, ...) |
61 | { | 66 | { |
62 | va_list arglist; | 67 | va_list arglist; |
63 | 68 | ||
@@ -72,7 +77,8 @@ static int is_vmlinux(const char *modname) | |||
72 | { | 77 | { |
73 | const char *myname; | 78 | const char *myname; |
74 | 79 | ||
75 | if ((myname = strrchr(modname, '/'))) | 80 | myname = strrchr(modname, '/'); |
81 | if (myname) | ||
76 | myname++; | 82 | myname++; |
77 | else | 83 | else |
78 | myname = modname; | 84 | myname = modname; |
@@ -83,14 +89,13 @@ static int is_vmlinux(const char *modname) | |||
83 | 89 | ||
84 | void *do_nofail(void *ptr, const char *expr) | 90 | void *do_nofail(void *ptr, const char *expr) |
85 | { | 91 | { |
86 | if (!ptr) { | 92 | if (!ptr) |
87 | fatal("modpost: Memory allocation failure: %s.\n", expr); | 93 | fatal("modpost: Memory allocation failure: %s.\n", expr); |
88 | } | 94 | |
89 | return ptr; | 95 | return ptr; |
90 | } | 96 | } |
91 | 97 | ||
92 | /* A list of all modules we processed */ | 98 | /* A list of all modules we processed */ |
93 | |||
94 | static struct module *modules; | 99 | static struct module *modules; |
95 | 100 | ||
96 | static struct module *find_module(char *modname) | 101 | static struct module *find_module(char *modname) |
@@ -113,7 +118,8 @@ static struct module *new_module(char *modname) | |||
113 | p = NOFAIL(strdup(modname)); | 118 | p = NOFAIL(strdup(modname)); |
114 | 119 | ||
115 | /* strip trailing .o */ | 120 | /* strip trailing .o */ |
116 | if ((s = strrchr(p, '.')) != NULL) | 121 | s = strrchr(p, '.'); |
122 | if (s != NULL) | ||
117 | if (strcmp(s, ".o") == 0) | 123 | if (strcmp(s, ".o") == 0) |
118 | *s = '\0'; | 124 | *s = '\0'; |
119 | 125 | ||
@@ -154,7 +160,7 @@ static inline unsigned int tdb_hash(const char *name) | |||
154 | unsigned i; /* Used to cycle through random values. */ | 160 | unsigned i; /* Used to cycle through random values. */ |
155 | 161 | ||
156 | /* Set the initial value from the key size. */ | 162 | /* Set the initial value from the key size. */ |
157 | for (value = 0x238F13AF * strlen(name), i=0; name[i]; i++) | 163 | for (value = 0x238F13AF * strlen(name), i = 0; name[i]; i++) |
158 | value = (value + (((unsigned char *)name)[i] << (i*5 % 24))); | 164 | value = (value + (((unsigned char *)name)[i] << (i*5 % 24))); |
159 | 165 | ||
160 | return (1103515243 * value + 12345); | 166 | return (1103515243 * value + 12345); |
@@ -198,7 +204,7 @@ static struct symbol *find_symbol(const char *name) | |||
198 | if (name[0] == '.') | 204 | if (name[0] == '.') |
199 | name++; | 205 | name++; |
200 | 206 | ||
201 | for (s = symbolhash[tdb_hash(name) % SYMBOL_HASH_SIZE]; s; s=s->next) { | 207 | for (s = symbolhash[tdb_hash(name) % SYMBOL_HASH_SIZE]; s; s = s->next) { |
202 | if (strcmp(s->name, name) == 0) | 208 | if (strcmp(s->name, name) == 0) |
203 | return s; | 209 | return s; |
204 | } | 210 | } |
@@ -223,9 +229,10 @@ static const char *export_str(enum export ex) | |||
223 | return export_list[ex].str; | 229 | return export_list[ex].str; |
224 | } | 230 | } |
225 | 231 | ||
226 | static enum export export_no(const char * s) | 232 | static enum export export_no(const char *s) |
227 | { | 233 | { |
228 | int i; | 234 | int i; |
235 | |||
229 | if (!s) | 236 | if (!s) |
230 | return export_unknown; | 237 | return export_unknown; |
231 | for (i = 0; export_list[i].export != export_unknown; i++) { | 238 | for (i = 0; export_list[i].export != export_unknown; i++) { |
@@ -315,7 +322,7 @@ void *grab_file(const char *filename, unsigned long *size) | |||
315 | * spaces in the beginning of the line is trimmed away. | 322 | * spaces in the beginning of the line is trimmed away. |
316 | * Return a pointer to a static buffer. | 323 | * Return a pointer to a static buffer. |
317 | **/ | 324 | **/ |
318 | char* get_next_line(unsigned long *pos, void *file, unsigned long size) | 325 | char *get_next_line(unsigned long *pos, void *file, unsigned long size) |
319 | { | 326 | { |
320 | static char line[4096]; | 327 | static char line[4096]; |
321 | int skip = 1; | 328 | int skip = 1; |
@@ -323,8 +330,7 @@ char* get_next_line(unsigned long *pos, void *file, unsigned long size) | |||
323 | signed char *p = (signed char *)file + *pos; | 330 | signed char *p = (signed char *)file + *pos; |
324 | char *s = line; | 331 | char *s = line; |
325 | 332 | ||
326 | for (; *pos < size ; (*pos)++) | 333 | for (; *pos < size ; (*pos)++) { |
327 | { | ||
328 | if (skip && isspace(*p)) { | 334 | if (skip && isspace(*p)) { |
329 | p++; | 335 | p++; |
330 | continue; | 336 | continue; |
@@ -386,7 +392,9 @@ static int parse_elf(struct elf_info *info, const char *filename) | |||
386 | 392 | ||
387 | /* Check if file offset is correct */ | 393 | /* Check if file offset is correct */ |
388 | if (hdr->e_shoff > info->size) { | 394 | if (hdr->e_shoff > info->size) { |
389 | fatal("section header offset=%u in file '%s' is bigger then filesize=%lu\n", hdr->e_shoff, filename, info->size); | 395 | fatal("section header offset=%lu in file '%s' is bigger than " |
396 | "filesize=%lu\n", (unsigned long)hdr->e_shoff, | ||
397 | filename, info->size); | ||
390 | return 0; | 398 | return 0; |
391 | } | 399 | } |
392 | 400 | ||
@@ -407,7 +415,10 @@ static int parse_elf(struct elf_info *info, const char *filename) | |||
407 | const char *secname; | 415 | const char *secname; |
408 | 416 | ||
409 | if (sechdrs[i].sh_offset > info->size) { | 417 | if (sechdrs[i].sh_offset > info->size) { |
410 | fatal("%s is truncated. sechdrs[i].sh_offset=%u > sizeof(*hrd)=%ul\n", filename, (unsigned int)sechdrs[i].sh_offset, sizeof(*hdr)); | 418 | fatal("%s is truncated. sechdrs[i].sh_offset=%lu > " |
419 | "sizeof(*hrd)=%zu\n", filename, | ||
420 | (unsigned long)sechdrs[i].sh_offset, | ||
421 | sizeof(*hdr)); | ||
411 | return 0; | 422 | return 0; |
412 | } | 423 | } |
413 | secname = secstrings + sechdrs[i].sh_name; | 424 | secname = secstrings + sechdrs[i].sh_name; |
@@ -434,9 +445,9 @@ static int parse_elf(struct elf_info *info, const char *filename) | |||
434 | info->strtab = (void *)hdr + | 445 | info->strtab = (void *)hdr + |
435 | sechdrs[sechdrs[i].sh_link].sh_offset; | 446 | sechdrs[sechdrs[i].sh_link].sh_offset; |
436 | } | 447 | } |
437 | if (!info->symtab_start) { | 448 | if (!info->symtab_start) |
438 | fatal("%s has no symtab?\n", filename); | 449 | fatal("%s has no symtab?\n", filename); |
439 | } | 450 | |
440 | /* Fix endianness in symbols */ | 451 | /* Fix endianness in symbols */ |
441 | for (sym = info->symtab_start; sym < info->symtab_stop; sym++) { | 452 | for (sym = info->symtab_start; sym < info->symtab_stop; sym++) { |
442 | sym->st_shndx = TO_NATIVE(sym->st_shndx); | 453 | sym->st_shndx = TO_NATIVE(sym->st_shndx); |
@@ -505,11 +516,13 @@ static void handle_modversions(struct module *mod, struct elf_info *info, | |||
505 | #endif | 516 | #endif |
506 | 517 | ||
507 | if (memcmp(symname, MODULE_SYMBOL_PREFIX, | 518 | if (memcmp(symname, MODULE_SYMBOL_PREFIX, |
508 | strlen(MODULE_SYMBOL_PREFIX)) == 0) | 519 | strlen(MODULE_SYMBOL_PREFIX)) == 0) { |
509 | mod->unres = alloc_symbol(symname + | 520 | mod->unres = |
510 | strlen(MODULE_SYMBOL_PREFIX), | 521 | alloc_symbol(symname + |
511 | ELF_ST_BIND(sym->st_info) == STB_WEAK, | 522 | strlen(MODULE_SYMBOL_PREFIX), |
512 | mod->unres); | 523 | ELF_ST_BIND(sym->st_info) == STB_WEAK, |
524 | mod->unres); | ||
525 | } | ||
513 | break; | 526 | break; |
514 | default: | 527 | default: |
515 | /* All exported symbols */ | 528 | /* All exported symbols */ |
@@ -578,69 +591,303 @@ static char *get_modinfo(void *modinfo, unsigned long modinfo_len, | |||
578 | **/ | 591 | **/ |
579 | static int strrcmp(const char *s, const char *sub) | 592 | static int strrcmp(const char *s, const char *sub) |
580 | { | 593 | { |
581 | int slen, sublen; | 594 | int slen, sublen; |
582 | 595 | ||
583 | if (!s || !sub) | 596 | if (!s || !sub) |
584 | return 1; | 597 | return 1; |
585 | 598 | ||
586 | slen = strlen(s); | 599 | slen = strlen(s); |
587 | sublen = strlen(sub); | 600 | sublen = strlen(sub); |
588 | 601 | ||
589 | if ((slen == 0) || (sublen == 0)) | 602 | if ((slen == 0) || (sublen == 0)) |
590 | return 1; | 603 | return 1; |
591 | 604 | ||
592 | if (sublen > slen) | 605 | if (sublen > slen) |
593 | return 1; | 606 | return 1; |
594 | 607 | ||
595 | return memcmp(s + slen - sublen, sub, sublen); | 608 | return memcmp(s + slen - sublen, sub, sublen); |
596 | } | 609 | } |
597 | 610 | ||
598 | /* | 611 | static const char *sym_name(struct elf_info *elf, Elf_Sym *sym) |
599 | * Functions used only during module init is marked __init and is stored in | 612 | { |
600 | * a .init.text section. Likewise data is marked __initdata and stored in | 613 | if (sym) |
601 | * a .init.data section. | 614 | return elf->strtab + sym->st_name; |
602 | * If this section is one of these sections return 1 | 615 | else |
603 | * See include/linux/init.h for the details | 616 | return ""; |
617 | } | ||
618 | |||
619 | static const char *sec_name(struct elf_info *elf, int shndx) | ||
620 | { | ||
621 | Elf_Shdr *sechdrs = elf->sechdrs; | ||
622 | return (void *)elf->hdr + | ||
623 | elf->sechdrs[elf->hdr->e_shstrndx].sh_offset + | ||
624 | sechdrs[shndx].sh_name; | ||
625 | } | ||
626 | |||
627 | static const char *sech_name(struct elf_info *elf, Elf_Shdr *sechdr) | ||
628 | { | ||
629 | return (void *)elf->hdr + | ||
630 | elf->sechdrs[elf->hdr->e_shstrndx].sh_offset + | ||
631 | sechdr->sh_name; | ||
632 | } | ||
633 | |||
634 | /* if sym is empty or point to a string | ||
635 | * like ".[0-9]+" then return 1. | ||
636 | * This is the optional prefix added by ld to some sections | ||
604 | */ | 637 | */ |
605 | static int init_section(const char *name) | 638 | static int number_prefix(const char *sym) |
606 | { | 639 | { |
607 | if (strcmp(name, ".init") == 0) | 640 | if (*sym++ == '\0') |
608 | return 1; | ||
609 | if (strncmp(name, ".init.", strlen(".init.")) == 0) | ||
610 | return 1; | 641 | return 1; |
642 | if (*sym != '.') | ||
643 | return 0; | ||
644 | do { | ||
645 | char c = *sym++; | ||
646 | if (c < '0' || c > '9') | ||
647 | return 0; | ||
648 | } while (*sym); | ||
649 | return 1; | ||
650 | } | ||
651 | |||
652 | /* The pattern is an array of simple patterns. | ||
653 | * "foo" will match an exact string equal to "foo" | ||
654 | * "*foo" will match a string that ends with "foo" | ||
655 | * "foo*" will match a string that begins with "foo" | ||
656 | * "foo$" will match a string equal to "foo" or "foo.1" | ||
657 | * where the '1' can be any number including several digits. | ||
658 | * The $ syntax is for sections where ld append a dot number | ||
659 | * to make section name unique. | ||
660 | */ | ||
661 | int match(const char *sym, const char * const pat[]) | ||
662 | { | ||
663 | const char *p; | ||
664 | while (*pat) { | ||
665 | p = *pat++; | ||
666 | const char *endp = p + strlen(p) - 1; | ||
667 | |||
668 | /* "*foo" */ | ||
669 | if (*p == '*') { | ||
670 | if (strrcmp(sym, p + 1) == 0) | ||
671 | return 1; | ||
672 | } | ||
673 | /* "foo*" */ | ||
674 | else if (*endp == '*') { | ||
675 | if (strncmp(sym, p, strlen(p) - 1) == 0) | ||
676 | return 1; | ||
677 | } | ||
678 | /* "foo$" */ | ||
679 | else if (*endp == '$') { | ||
680 | if (strncmp(sym, p, strlen(p) - 1) == 0) { | ||
681 | if (number_prefix(sym + strlen(p) - 1)) | ||
682 | return 1; | ||
683 | } | ||
684 | } | ||
685 | /* no wildcards */ | ||
686 | else { | ||
687 | if (strcmp(p, sym) == 0) | ||
688 | return 1; | ||
689 | } | ||
690 | } | ||
691 | /* no match */ | ||
611 | return 0; | 692 | return 0; |
612 | } | 693 | } |
613 | 694 | ||
695 | /* sections that we do not want to do full section mismatch check on */ | ||
696 | static const char *section_white_list[] = | ||
697 | { ".debug*", ".stab*", ".note*", ".got*", ".toc*", NULL }; | ||
698 | |||
614 | /* | 699 | /* |
615 | * Functions used only during module exit is marked __exit and is stored in | 700 | * Is this section one we do not want to check? |
616 | * a .exit.text section. Likewise data is marked __exitdata and stored in | 701 | * This is often debug sections. |
617 | * a .exit.data section. | 702 | * If we are going to check this section then |
618 | * If this section is one of these sections return 1 | 703 | * test if section name ends with a dot and a number. |
619 | * See include/linux/init.h for the details | 704 | * This is used to find sections where the linker have |
620 | **/ | 705 | * appended a dot-number to make the name unique. |
621 | static int exit_section(const char *name) | 706 | * The cause of this is often a section specified in assembler |
707 | * without "ax" / "aw" and the same section used in .c | ||
708 | * code where gcc add these. | ||
709 | */ | ||
710 | static int check_section(const char *modname, const char *sec) | ||
622 | { | 711 | { |
623 | if (strcmp(name, ".exit.text") == 0) | 712 | const char *e = sec + strlen(sec) - 1; |
624 | return 1; | 713 | if (match(sec, section_white_list)) |
625 | if (strcmp(name, ".exit.data") == 0) | ||
626 | return 1; | 714 | return 1; |
627 | return 0; | ||
628 | 715 | ||
716 | if (*e && isdigit(*e)) { | ||
717 | /* consume all digits */ | ||
718 | while (*e && e != sec && isdigit(*e)) | ||
719 | e--; | ||
720 | if (*e == '.') { | ||
721 | warn("%s (%s): unexpected section name.\n" | ||
722 | "The (.[number]+) following section name are " | ||
723 | "ld generated and not expected.\n" | ||
724 | "Did you forget to use \"ax\"/\"aw\" " | ||
725 | "in a .S file?\n" | ||
726 | "Note that for example <linux/init.h> contains\n" | ||
727 | "section definitions for use in .S files.\n\n", | ||
728 | modname, sec); | ||
729 | } | ||
730 | } | ||
731 | return 0; | ||
629 | } | 732 | } |
630 | 733 | ||
631 | /* | 734 | |
632 | * Data sections are named like this: | 735 | |
633 | * .data | .data.rel | .data.rel.* | 736 | #define ALL_INIT_DATA_SECTIONS \ |
634 | * Return 1 if the specified section is a data section | 737 | ".init.data$", ".devinit.data$", ".cpuinit.data$", ".meminit.data$" |
738 | #define ALL_EXIT_DATA_SECTIONS \ | ||
739 | ".exit.data$", ".devexit.data$", ".cpuexit.data$", ".memexit.data$" | ||
740 | |||
741 | #define ALL_INIT_TEXT_SECTIONS \ | ||
742 | ".init.text$", ".devinit.text$", ".cpuinit.text$", ".meminit.text$" | ||
743 | #define ALL_EXIT_TEXT_SECTIONS \ | ||
744 | ".exit.text$", ".devexit.text$", ".cpuexit.text$", ".memexit.text$" | ||
745 | |||
746 | #define ALL_INIT_SECTIONS ALL_INIT_DATA_SECTIONS, ALL_INIT_TEXT_SECTIONS | ||
747 | #define ALL_EXIT_SECTIONS ALL_EXIT_DATA_SECTIONS, ALL_EXIT_TEXT_SECTIONS | ||
748 | |||
749 | #define DATA_SECTIONS ".data$", ".data.rel$" | ||
750 | #define TEXT_SECTIONS ".text$" | ||
751 | |||
752 | #define INIT_SECTIONS ".init.data$", ".init.text$" | ||
753 | #define DEV_INIT_SECTIONS ".devinit.data$", ".devinit.text$" | ||
754 | #define CPU_INIT_SECTIONS ".cpuinit.data$", ".cpuinit.text$" | ||
755 | #define MEM_INIT_SECTIONS ".meminit.data$", ".meminit.text$" | ||
756 | |||
757 | #define EXIT_SECTIONS ".exit.data$", ".exit.text$" | ||
758 | #define DEV_EXIT_SECTIONS ".devexit.data$", ".devexit.text$" | ||
759 | #define CPU_EXIT_SECTIONS ".cpuexit.data$", ".cpuexit.text$" | ||
760 | #define MEM_EXIT_SECTIONS ".memexit.data$", ".memexit.text$" | ||
761 | |||
762 | /* init data sections */ | ||
763 | static const char *init_data_sections[] = { ALL_INIT_DATA_SECTIONS, NULL }; | ||
764 | |||
765 | /* all init sections */ | ||
766 | static const char *init_sections[] = { ALL_INIT_SECTIONS, NULL }; | ||
767 | |||
768 | /* All init and exit sections (code + data) */ | ||
769 | static const char *init_exit_sections[] = | ||
770 | {ALL_INIT_SECTIONS, ALL_EXIT_SECTIONS, NULL }; | ||
771 | |||
772 | /* data section */ | ||
773 | static const char *data_sections[] = { DATA_SECTIONS, NULL }; | ||
774 | |||
775 | /* sections that may refer to an init/exit section with no warning */ | ||
776 | static const char *initref_sections[] = | ||
777 | { | ||
778 | ".text.init.refok*", | ||
779 | ".exit.text.refok*", | ||
780 | ".data.init.refok*", | ||
781 | NULL | ||
782 | }; | ||
783 | |||
784 | |||
785 | /* symbols in .data that may refer to init/exit sections */ | ||
786 | static const char *symbol_white_list[] = | ||
787 | { | ||
788 | "*driver", | ||
789 | "*_template", /* scsi uses *_template a lot */ | ||
790 | "*_timer", /* arm uses ops structures named _timer a lot */ | ||
791 | "*_sht", /* scsi also used *_sht to some extent */ | ||
792 | "*_ops", | ||
793 | "*_probe", | ||
794 | "*_probe_one", | ||
795 | "*_console", | ||
796 | NULL | ||
797 | }; | ||
798 | |||
799 | static const char *head_sections[] = { ".head.text*", NULL }; | ||
800 | static const char *linker_symbols[] = | ||
801 | { "__init_begin", "_sinittext", "_einittext", NULL }; | ||
802 | |||
803 | enum mismatch { | ||
804 | NO_MISMATCH, | ||
805 | TEXT_TO_INIT, | ||
806 | DATA_TO_INIT, | ||
807 | TEXT_TO_EXIT, | ||
808 | DATA_TO_EXIT, | ||
809 | XXXINIT_TO_INIT, | ||
810 | XXXEXIT_TO_EXIT, | ||
811 | INIT_TO_EXIT, | ||
812 | EXIT_TO_INIT, | ||
813 | EXPORT_TO_INIT_EXIT, | ||
814 | }; | ||
815 | |||
816 | struct sectioncheck { | ||
817 | const char *fromsec[20]; | ||
818 | const char *tosec[20]; | ||
819 | enum mismatch mismatch; | ||
820 | }; | ||
821 | |||
822 | const struct sectioncheck sectioncheck[] = { | ||
823 | /* Do not reference init/exit code/data from | ||
824 | * normal code and data | ||
635 | */ | 825 | */ |
636 | static int data_section(const char *name) | ||
637 | { | 826 | { |
638 | if ((strcmp(name, ".data") == 0) || | 827 | .fromsec = { TEXT_SECTIONS, NULL }, |
639 | (strcmp(name, ".data.rel") == 0) || | 828 | .tosec = { ALL_INIT_SECTIONS, NULL }, |
640 | (strncmp(name, ".data.rel.", strlen(".data.rel.")) == 0)) | 829 | .mismatch = TEXT_TO_INIT, |
641 | return 1; | 830 | }, |
642 | else | 831 | { |
643 | return 0; | 832 | .fromsec = { DATA_SECTIONS, NULL }, |
833 | .tosec = { ALL_INIT_SECTIONS, NULL }, | ||
834 | .mismatch = DATA_TO_INIT, | ||
835 | }, | ||
836 | { | ||
837 | .fromsec = { TEXT_SECTIONS, NULL }, | ||
838 | .tosec = { ALL_EXIT_SECTIONS, NULL }, | ||
839 | .mismatch = TEXT_TO_EXIT, | ||
840 | }, | ||
841 | { | ||
842 | .fromsec = { DATA_SECTIONS, NULL }, | ||
843 | .tosec = { ALL_EXIT_SECTIONS, NULL }, | ||
844 | .mismatch = DATA_TO_EXIT, | ||
845 | }, | ||
846 | /* Do not reference init code/data from devinit/cpuinit/meminit code/data */ | ||
847 | { | ||
848 | .fromsec = { DEV_INIT_SECTIONS, CPU_INIT_SECTIONS, MEM_INIT_SECTIONS, NULL }, | ||
849 | .tosec = { INIT_SECTIONS, NULL }, | ||
850 | .mismatch = XXXINIT_TO_INIT, | ||
851 | }, | ||
852 | /* Do not reference exit code/data from devexit/cpuexit/memexit code/data */ | ||
853 | { | ||
854 | .fromsec = { DEV_EXIT_SECTIONS, CPU_EXIT_SECTIONS, MEM_EXIT_SECTIONS, NULL }, | ||
855 | .tosec = { EXIT_SECTIONS, NULL }, | ||
856 | .mismatch = XXXEXIT_TO_EXIT, | ||
857 | }, | ||
858 | /* Do not use exit code/data from init code */ | ||
859 | { | ||
860 | .fromsec = { ALL_INIT_SECTIONS, NULL }, | ||
861 | .tosec = { ALL_EXIT_SECTIONS, NULL }, | ||
862 | .mismatch = INIT_TO_EXIT, | ||
863 | }, | ||
864 | /* Do not use init code/data from exit code */ | ||
865 | { | ||
866 | .fromsec = { ALL_EXIT_SECTIONS, NULL }, | ||
867 | .tosec = { ALL_INIT_SECTIONS, NULL }, | ||
868 | .mismatch = EXIT_TO_INIT, | ||
869 | }, | ||
870 | /* Do not export init/exit functions or data */ | ||
871 | { | ||
872 | .fromsec = { "__ksymtab*", NULL }, | ||
873 | .tosec = { ALL_INIT_SECTIONS, ALL_EXIT_SECTIONS, NULL }, | ||
874 | .mismatch = EXPORT_TO_INIT_EXIT | ||
875 | } | ||
876 | }; | ||
877 | |||
878 | static int section_mismatch(const char *fromsec, const char *tosec) | ||
879 | { | ||
880 | int i; | ||
881 | int elems = sizeof(sectioncheck) / sizeof(struct sectioncheck); | ||
882 | const struct sectioncheck *check = §ioncheck[0]; | ||
883 | |||
884 | for (i = 0; i < elems; i++) { | ||
885 | if (match(fromsec, check->fromsec) && | ||
886 | match(tosec, check->tosec)) | ||
887 | return check->mismatch; | ||
888 | check++; | ||
889 | } | ||
890 | return NO_MISMATCH; | ||
644 | } | 891 | } |
645 | 892 | ||
646 | /** | 893 | /** |
@@ -669,7 +916,8 @@ static int data_section(const char *name) | |||
669 | * the pattern is identified by: | 916 | * the pattern is identified by: |
670 | * tosec = init or exit section | 917 | * tosec = init or exit section |
671 | * fromsec = data section | 918 | * fromsec = data section |
672 | * atsym = *driver, *_template, *_sht, *_ops, *_probe, *probe_one, *_console, *_timer | 919 | * atsym = *driver, *_template, *_sht, *_ops, *_probe, |
920 | * *probe_one, *_console, *_timer | ||
673 | * | 921 | * |
674 | * Pattern 3: | 922 | * Pattern 3: |
675 | * Whitelist all refereces from .text.head to .init.data | 923 | * Whitelist all refereces from .text.head to .init.data |
@@ -684,77 +932,36 @@ static int data_section(const char *name) | |||
684 | * This pattern is identified by | 932 | * This pattern is identified by |
685 | * refsymname = __init_begin, _sinittext, _einittext | 933 | * refsymname = __init_begin, _sinittext, _einittext |
686 | * | 934 | * |
687 | * Pattern 5: | ||
688 | * Xtensa uses literal sections for constants that are accessed PC-relative. | ||
689 | * Literal sections may safely reference their text sections. | ||
690 | * (Note that the name for the literal section omits any trailing '.text') | ||
691 | * tosec = <section>[.text] | ||
692 | * fromsec = <section>.literal | ||
693 | **/ | 935 | **/ |
694 | static int secref_whitelist(const char *modname, const char *tosec, | 936 | static int secref_whitelist(const char *fromsec, const char *fromsym, |
695 | const char *fromsec, const char *atsym, | 937 | const char *tosec, const char *tosym) |
696 | const char *refsymname) | ||
697 | { | 938 | { |
698 | int len; | ||
699 | const char **s; | ||
700 | const char *pat2sym[] = { | ||
701 | "driver", | ||
702 | "_template", /* scsi uses *_template a lot */ | ||
703 | "_timer", /* arm uses ops structures named _timer a lot */ | ||
704 | "_sht", /* scsi also used *_sht to some extent */ | ||
705 | "_ops", | ||
706 | "_probe", | ||
707 | "_probe_one", | ||
708 | "_console", | ||
709 | NULL | ||
710 | }; | ||
711 | |||
712 | const char *pat3refsym[] = { | ||
713 | "__init_begin", | ||
714 | "_sinittext", | ||
715 | "_einittext", | ||
716 | NULL | ||
717 | }; | ||
718 | |||
719 | /* Check for pattern 0 */ | 939 | /* Check for pattern 0 */ |
720 | if ((strncmp(fromsec, ".text.init.refok", strlen(".text.init.refok")) == 0) || | 940 | if (match(fromsec, initref_sections)) |
721 | (strncmp(fromsec, ".exit.text.refok", strlen(".exit.text.refok")) == 0) || | 941 | return 0; |
722 | (strncmp(fromsec, ".data.init.refok", strlen(".data.init.refok")) == 0)) | ||
723 | return 1; | ||
724 | 942 | ||
725 | /* Check for pattern 1 */ | 943 | /* Check for pattern 1 */ |
726 | if ((strcmp(tosec, ".init.data") == 0) && | 944 | if (match(tosec, init_data_sections) && |
727 | (strncmp(fromsec, ".data", strlen(".data")) == 0) && | 945 | match(fromsec, data_sections) && |
728 | (strncmp(atsym, "__param", strlen("__param")) == 0)) | 946 | (strncmp(fromsym, "__param", strlen("__param")) == 0)) |
729 | return 1; | 947 | return 0; |
730 | 948 | ||
731 | /* Check for pattern 2 */ | 949 | /* Check for pattern 2 */ |
732 | if ((init_section(tosec) || exit_section(tosec)) && data_section(fromsec)) | 950 | if (match(tosec, init_exit_sections) && |
733 | for (s = pat2sym; *s; s++) | 951 | match(fromsec, data_sections) && |
734 | if (strrcmp(atsym, *s) == 0) | 952 | match(fromsym, symbol_white_list)) |
735 | return 1; | 953 | return 0; |
736 | 954 | ||
737 | /* Check for pattern 3 */ | 955 | /* Check for pattern 3 */ |
738 | if ((strcmp(fromsec, ".text.head") == 0) && | 956 | if (match(fromsec, head_sections) && |
739 | ((strcmp(tosec, ".init.data") == 0) || | 957 | match(tosec, init_sections)) |
740 | (strcmp(tosec, ".init.text") == 0))) | 958 | return 0; |
741 | return 1; | ||
742 | 959 | ||
743 | /* Check for pattern 4 */ | 960 | /* Check for pattern 4 */ |
744 | for (s = pat3refsym; *s; s++) | 961 | if (match(tosym, linker_symbols)) |
745 | if (strcmp(refsymname, *s) == 0) | 962 | return 0; |
746 | return 1; | ||
747 | |||
748 | /* Check for pattern 5 */ | ||
749 | if (strrcmp(tosec, ".text") == 0) | ||
750 | len = strlen(tosec) - strlen(".text"); | ||
751 | else | ||
752 | len = strlen(tosec); | ||
753 | if ((strncmp(tosec, fromsec, len) == 0) && (strlen(fromsec) > len) && | ||
754 | (strcmp(fromsec + len, ".literal") == 0)) | ||
755 | return 1; | ||
756 | 963 | ||
757 | return 0; | 964 | return 1; |
758 | } | 965 | } |
759 | 966 | ||
760 | /** | 967 | /** |
@@ -764,10 +971,13 @@ static int secref_whitelist(const char *modname, const char *tosec, | |||
764 | * In other cases the symbol needs to be looked up in the symbol table | 971 | * In other cases the symbol needs to be looked up in the symbol table |
765 | * based on section and address. | 972 | * based on section and address. |
766 | * **/ | 973 | * **/ |
767 | static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf_Addr addr, | 974 | static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf64_Sword addr, |
768 | Elf_Sym *relsym) | 975 | Elf_Sym *relsym) |
769 | { | 976 | { |
770 | Elf_Sym *sym; | 977 | Elf_Sym *sym; |
978 | Elf_Sym *near = NULL; | ||
979 | Elf64_Sword distance = 20; | ||
980 | Elf64_Sword d; | ||
771 | 981 | ||
772 | if (relsym->st_name != 0) | 982 | if (relsym->st_name != 0) |
773 | return relsym; | 983 | return relsym; |
@@ -778,8 +988,20 @@ static Elf_Sym *find_elf_symbol(struct elf_info *elf, Elf_Addr addr, | |||
778 | continue; | 988 | continue; |
779 | if (sym->st_value == addr) | 989 | if (sym->st_value == addr) |
780 | return sym; | 990 | return sym; |
991 | /* Find a symbol nearby - addr are maybe negative */ | ||
992 | d = sym->st_value - addr; | ||
993 | if (d < 0) | ||
994 | d = addr - sym->st_value; | ||
995 | if (d < distance) { | ||
996 | distance = d; | ||
997 | near = sym; | ||
998 | } | ||
781 | } | 999 | } |
782 | return NULL; | 1000 | /* We need a close match */ |
1001 | if (distance < 20) | ||
1002 | return near; | ||
1003 | else | ||
1004 | return NULL; | ||
783 | } | 1005 | } |
784 | 1006 | ||
785 | static inline int is_arm_mapping_symbol(const char *str) | 1007 | static inline int is_arm_mapping_symbol(const char *str) |
@@ -812,121 +1034,245 @@ static inline int is_valid_name(struct elf_info *elf, Elf_Sym *sym) | |||
812 | * The ELF format may have a better way to detect what type of symbol | 1034 | * The ELF format may have a better way to detect what type of symbol |
813 | * it is, but this works for now. | 1035 | * it is, but this works for now. |
814 | **/ | 1036 | **/ |
815 | static void find_symbols_between(struct elf_info *elf, Elf_Addr addr, | 1037 | static Elf_Sym *find_elf_symbol2(struct elf_info *elf, Elf_Addr addr, |
816 | const char *sec, | 1038 | const char *sec) |
817 | Elf_Sym **before, Elf_Sym **after) | ||
818 | { | 1039 | { |
819 | Elf_Sym *sym; | 1040 | Elf_Sym *sym; |
820 | Elf_Ehdr *hdr = elf->hdr; | 1041 | Elf_Sym *near = NULL; |
821 | Elf_Addr beforediff = ~0; | 1042 | Elf_Addr distance = ~0; |
822 | Elf_Addr afterdiff = ~0; | ||
823 | const char *secstrings = (void *)hdr + | ||
824 | elf->sechdrs[hdr->e_shstrndx].sh_offset; | ||
825 | |||
826 | *before = NULL; | ||
827 | *after = NULL; | ||
828 | 1043 | ||
829 | for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) { | 1044 | for (sym = elf->symtab_start; sym < elf->symtab_stop; sym++) { |
830 | const char *symsec; | 1045 | const char *symsec; |
831 | 1046 | ||
832 | if (sym->st_shndx >= SHN_LORESERVE) | 1047 | if (sym->st_shndx >= SHN_LORESERVE) |
833 | continue; | 1048 | continue; |
834 | symsec = secstrings + elf->sechdrs[sym->st_shndx].sh_name; | 1049 | symsec = sec_name(elf, sym->st_shndx); |
835 | if (strcmp(symsec, sec) != 0) | 1050 | if (strcmp(symsec, sec) != 0) |
836 | continue; | 1051 | continue; |
837 | if (!is_valid_name(elf, sym)) | 1052 | if (!is_valid_name(elf, sym)) |
838 | continue; | 1053 | continue; |
839 | if (sym->st_value <= addr) { | 1054 | if (sym->st_value <= addr) { |
840 | if ((addr - sym->st_value) < beforediff) { | 1055 | if ((addr - sym->st_value) < distance) { |
841 | beforediff = addr - sym->st_value; | 1056 | distance = addr - sym->st_value; |
842 | *before = sym; | 1057 | near = sym; |
843 | } | 1058 | } else if ((addr - sym->st_value) == distance) { |
844 | else if ((addr - sym->st_value) == beforediff) { | 1059 | near = sym; |
845 | *before = sym; | ||
846 | } | 1060 | } |
847 | } | 1061 | } |
1062 | } | ||
1063 | return near; | ||
1064 | } | ||
1065 | |||
1066 | /* | ||
1067 | * Convert a section name to the function/data attribute | ||
1068 | * .init.text => __init | ||
1069 | * .cpuinit.data => __cpudata | ||
1070 | * .memexitconst => __memconst | ||
1071 | * etc. | ||
1072 | */ | ||
1073 | static char *sec2annotation(const char *s) | ||
1074 | { | ||
1075 | if (match(s, init_exit_sections)) { | ||
1076 | char *p = malloc(20); | ||
1077 | char *r = p; | ||
1078 | |||
1079 | *p++ = '_'; | ||
1080 | *p++ = '_'; | ||
1081 | if (*s == '.') | ||
1082 | s++; | ||
1083 | while (*s && *s != '.') | ||
1084 | *p++ = *s++; | ||
1085 | *p = '\0'; | ||
1086 | if (*s == '.') | ||
1087 | s++; | ||
1088 | if (strstr(s, "rodata") != NULL) | ||
1089 | strcat(p, "const "); | ||
1090 | else if (strstr(s, "data") != NULL) | ||
1091 | strcat(p, "data "); | ||
848 | else | 1092 | else |
849 | { | 1093 | strcat(p, " "); |
850 | if ((sym->st_value - addr) < afterdiff) { | 1094 | return r; /* we leak her but we do not care */ |
851 | afterdiff = sym->st_value - addr; | 1095 | } else { |
852 | *after = sym; | 1096 | return ""; |
853 | } | ||
854 | else if ((sym->st_value - addr) == afterdiff) { | ||
855 | *after = sym; | ||
856 | } | ||
857 | } | ||
858 | } | 1097 | } |
859 | } | 1098 | } |
860 | 1099 | ||
861 | /** | 1100 | static int is_function(Elf_Sym *sym) |
1101 | { | ||
1102 | if (sym) | ||
1103 | return ELF_ST_TYPE(sym->st_info) == STT_FUNC; | ||
1104 | else | ||
1105 | return 0; | ||
1106 | } | ||
1107 | |||
1108 | /* | ||
862 | * Print a warning about a section mismatch. | 1109 | * Print a warning about a section mismatch. |
863 | * Try to find symbols near it so user can find it. | 1110 | * Try to find symbols near it so user can find it. |
864 | * Check whitelist before warning - it may be a false positive. | 1111 | * Check whitelist before warning - it may be a false positive. |
865 | **/ | 1112 | */ |
866 | static void warn_sec_mismatch(const char *modname, const char *fromsec, | 1113 | static void report_sec_mismatch(const char *modname, enum mismatch mismatch, |
867 | struct elf_info *elf, Elf_Sym *sym, Elf_Rela r) | 1114 | const char *fromsec, |
1115 | unsigned long long fromaddr, | ||
1116 | const char *fromsym, | ||
1117 | int from_is_func, | ||
1118 | const char *tosec, const char *tosym, | ||
1119 | int to_is_func) | ||
868 | { | 1120 | { |
869 | const char *refsymname = ""; | 1121 | const char *from, *from_p; |
870 | Elf_Sym *before, *after; | 1122 | const char *to, *to_p; |
871 | Elf_Sym *refsym; | 1123 | from = from_is_func ? "function" : "variable"; |
872 | Elf_Ehdr *hdr = elf->hdr; | 1124 | from_p = from_is_func ? "()" : ""; |
873 | Elf_Shdr *sechdrs = elf->sechdrs; | 1125 | to = to_is_func ? "function" : "variable"; |
874 | const char *secstrings = (void *)hdr + | 1126 | to_p = to_is_func ? "()" : ""; |
875 | sechdrs[hdr->e_shstrndx].sh_offset; | 1127 | |
876 | const char *secname = secstrings + sechdrs[sym->st_shndx].sh_name; | 1128 | fprintf(stderr, "WARNING: %s(%s+0x%llx): Section mismatch in" |
877 | 1129 | " reference from the %s %s%s to the %s %s:%s%s\n", | |
878 | find_symbols_between(elf, r.r_offset, fromsec, &before, &after); | 1130 | modname, fromsec, fromaddr, from, fromsym, from_p, |
879 | 1131 | to, tosec, tosym, to_p); | |
880 | refsym = find_elf_symbol(elf, r.r_addend, sym); | 1132 | |
881 | if (refsym && strlen(elf->strtab + refsym->st_name)) | 1133 | sec_mismatch_count++; |
882 | refsymname = elf->strtab + refsym->st_name; | 1134 | if (!sec_mismatch_verbose) |
883 | |||
884 | /* check whitelist - we may ignore it */ | ||
885 | if (secref_whitelist(modname, secname, fromsec, | ||
886 | before ? elf->strtab + before->st_name : "", | ||
887 | refsymname)) | ||
888 | return; | 1135 | return; |
889 | 1136 | ||
890 | if (before && after) { | 1137 | switch (mismatch) { |
891 | warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s " | 1138 | case TEXT_TO_INIT: |
892 | "(between '%s' and '%s')\n", | 1139 | fprintf(stderr, |
893 | modname, fromsec, (unsigned long long)r.r_offset, | 1140 | "The function %s %s() references\n" |
894 | secname, refsymname, | 1141 | "the %s %s%s%s.\n" |
895 | elf->strtab + before->st_name, | 1142 | "This is often because %s lacks a %s\n" |
896 | elf->strtab + after->st_name); | 1143 | "annotation or the annotation of %s is wrong.\n", |
897 | } else if (before) { | 1144 | sec2annotation(fromsec), fromsym, |
898 | warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s " | 1145 | to, sec2annotation(tosec), tosym, to_p, |
899 | "(after '%s')\n", | 1146 | fromsym, sec2annotation(tosec), tosym); |
900 | modname, fromsec, (unsigned long long)r.r_offset, | 1147 | break; |
901 | secname, refsymname, | 1148 | case DATA_TO_INIT: { |
902 | elf->strtab + before->st_name); | 1149 | const char **s = symbol_white_list; |
903 | } else if (after) { | 1150 | fprintf(stderr, |
904 | warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s " | 1151 | "The variable %s references\n" |
905 | "before '%s' (at offset -0x%llx)\n", | 1152 | "the %s %s%s%s\n" |
906 | modname, fromsec, (unsigned long long)r.r_offset, | 1153 | "If the reference is valid then annotate the\n" |
907 | secname, refsymname, | 1154 | "variable with __init* (see linux/init.h) " |
908 | elf->strtab + after->st_name); | 1155 | "or name the variable:\n", |
909 | } else { | 1156 | fromsym, to, sec2annotation(tosec), tosym, to_p); |
910 | warn("%s(%s+0x%llx): Section mismatch: reference to %s:%s\n", | 1157 | while (*s) |
911 | modname, fromsec, (unsigned long long)r.r_offset, | 1158 | fprintf(stderr, "%s, ", *s++); |
912 | secname, refsymname); | 1159 | fprintf(stderr, "\n"); |
1160 | break; | ||
1161 | } | ||
1162 | case TEXT_TO_EXIT: | ||
1163 | fprintf(stderr, | ||
1164 | "The function %s() references a %s in an exit section.\n" | ||
1165 | "Often the %s %s%s has valid usage outside the exit section\n" | ||
1166 | "and the fix is to remove the %sannotation of %s.\n", | ||
1167 | fromsym, to, to, tosym, to_p, sec2annotation(tosec), tosym); | ||
1168 | break; | ||
1169 | case DATA_TO_EXIT: { | ||
1170 | const char **s = symbol_white_list; | ||
1171 | fprintf(stderr, | ||
1172 | "The variable %s references\n" | ||
1173 | "the %s %s%s%s\n" | ||
1174 | "If the reference is valid then annotate the\n" | ||
1175 | "variable with __exit* (see linux/init.h) or " | ||
1176 | "name the variable:\n", | ||
1177 | fromsym, to, sec2annotation(tosec), tosym, to_p); | ||
1178 | while (*s) | ||
1179 | fprintf(stderr, "%s, ", *s++); | ||
1180 | fprintf(stderr, "\n"); | ||
1181 | break; | ||
1182 | } | ||
1183 | case XXXINIT_TO_INIT: | ||
1184 | case XXXEXIT_TO_EXIT: | ||
1185 | fprintf(stderr, | ||
1186 | "The %s %s%s%s references\n" | ||
1187 | "a %s %s%s%s.\n" | ||
1188 | "If %s is only used by %s then\n" | ||
1189 | "annotate %s with a matching annotation.\n", | ||
1190 | from, sec2annotation(fromsec), fromsym, from_p, | ||
1191 | to, sec2annotation(tosec), tosym, to_p, | ||
1192 | fromsym, tosym, fromsym); | ||
1193 | break; | ||
1194 | case INIT_TO_EXIT: | ||
1195 | fprintf(stderr, | ||
1196 | "The %s %s%s%s references\n" | ||
1197 | "a %s %s%s%s.\n" | ||
1198 | "This is often seen when error handling " | ||
1199 | "in the init function\n" | ||
1200 | "uses functionality in the exit path.\n" | ||
1201 | "The fix is often to remove the %sannotation of\n" | ||
1202 | "%s%s so it may be used outside an exit section.\n", | ||
1203 | from, sec2annotation(fromsec), fromsym, from_p, | ||
1204 | to, sec2annotation(tosec), tosym, to_p, | ||
1205 | sec2annotation(tosec), tosym, to_p); | ||
1206 | break; | ||
1207 | case EXIT_TO_INIT: | ||
1208 | fprintf(stderr, | ||
1209 | "The %s %s%s%s references\n" | ||
1210 | "a %s %s%s%s.\n" | ||
1211 | "This is often seen when error handling " | ||
1212 | "in the exit function\n" | ||
1213 | "uses functionality in the init path.\n" | ||
1214 | "The fix is often to remove the %sannotation of\n" | ||
1215 | "%s%s so it may be used outside an init section.\n", | ||
1216 | from, sec2annotation(fromsec), fromsym, from_p, | ||
1217 | to, sec2annotation(tosec), tosym, to_p, | ||
1218 | sec2annotation(tosec), tosym, to_p); | ||
1219 | break; | ||
1220 | case EXPORT_TO_INIT_EXIT: | ||
1221 | fprintf(stderr, | ||
1222 | "The symbol %s is exported and annotated %s\n" | ||
1223 | "Fix this by removing the %sannotation of %s " | ||
1224 | "or drop the export.\n", | ||
1225 | tosym, sec2annotation(tosec), sec2annotation(tosec), tosym); | ||
1226 | case NO_MISMATCH: | ||
1227 | /* To get warnings on missing members */ | ||
1228 | break; | ||
1229 | } | ||
1230 | fprintf(stderr, "\n"); | ||
1231 | } | ||
1232 | |||
1233 | static void check_section_mismatch(const char *modname, struct elf_info *elf, | ||
1234 | Elf_Rela *r, Elf_Sym *sym, const char *fromsec) | ||
1235 | { | ||
1236 | const char *tosec; | ||
1237 | enum mismatch mismatch; | ||
1238 | |||
1239 | tosec = sec_name(elf, sym->st_shndx); | ||
1240 | mismatch = section_mismatch(fromsec, tosec); | ||
1241 | if (mismatch != NO_MISMATCH) { | ||
1242 | Elf_Sym *to; | ||
1243 | Elf_Sym *from; | ||
1244 | const char *tosym; | ||
1245 | const char *fromsym; | ||
1246 | |||
1247 | from = find_elf_symbol2(elf, r->r_offset, fromsec); | ||
1248 | fromsym = sym_name(elf, from); | ||
1249 | to = find_elf_symbol(elf, r->r_addend, sym); | ||
1250 | tosym = sym_name(elf, to); | ||
1251 | |||
1252 | /* check whitelist - we may ignore it */ | ||
1253 | if (secref_whitelist(fromsec, fromsym, tosec, tosym)) { | ||
1254 | report_sec_mismatch(modname, mismatch, | ||
1255 | fromsec, r->r_offset, fromsym, | ||
1256 | is_function(from), tosec, tosym, | ||
1257 | is_function(to)); | ||
1258 | } | ||
913 | } | 1259 | } |
914 | } | 1260 | } |
915 | 1261 | ||
916 | static unsigned int *reloc_location(struct elf_info *elf, | 1262 | static unsigned int *reloc_location(struct elf_info *elf, |
917 | int rsection, Elf_Rela *r) | 1263 | Elf_Shdr *sechdr, Elf_Rela *r) |
918 | { | 1264 | { |
919 | Elf_Shdr *sechdrs = elf->sechdrs; | 1265 | Elf_Shdr *sechdrs = elf->sechdrs; |
920 | int section = sechdrs[rsection].sh_info; | 1266 | int section = sechdr->sh_info; |
921 | 1267 | ||
922 | return (void *)elf->hdr + sechdrs[section].sh_offset + | 1268 | return (void *)elf->hdr + sechdrs[section].sh_offset + |
923 | (r->r_offset - sechdrs[section].sh_addr); | 1269 | (r->r_offset - sechdrs[section].sh_addr); |
924 | } | 1270 | } |
925 | 1271 | ||
926 | static int addend_386_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | 1272 | static int addend_386_rel(struct elf_info *elf, Elf_Shdr *sechdr, Elf_Rela *r) |
927 | { | 1273 | { |
928 | unsigned int r_typ = ELF_R_TYPE(r->r_info); | 1274 | unsigned int r_typ = ELF_R_TYPE(r->r_info); |
929 | unsigned int *location = reloc_location(elf, rsection, r); | 1275 | unsigned int *location = reloc_location(elf, sechdr, r); |
930 | 1276 | ||
931 | switch (r_typ) { | 1277 | switch (r_typ) { |
932 | case R_386_32: | 1278 | case R_386_32: |
@@ -942,19 +1288,21 @@ static int addend_386_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | |||
942 | return 0; | 1288 | return 0; |
943 | } | 1289 | } |
944 | 1290 | ||
945 | static int addend_arm_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | 1291 | static int addend_arm_rel(struct elf_info *elf, Elf_Shdr *sechdr, Elf_Rela *r) |
946 | { | 1292 | { |
947 | unsigned int r_typ = ELF_R_TYPE(r->r_info); | 1293 | unsigned int r_typ = ELF_R_TYPE(r->r_info); |
948 | 1294 | ||
949 | switch (r_typ) { | 1295 | switch (r_typ) { |
950 | case R_ARM_ABS32: | 1296 | case R_ARM_ABS32: |
951 | /* From ARM ABI: (S + A) | T */ | 1297 | /* From ARM ABI: (S + A) | T */ |
952 | r->r_addend = (int)(long)(elf->symtab_start + ELF_R_SYM(r->r_info)); | 1298 | r->r_addend = (int)(long) |
1299 | (elf->symtab_start + ELF_R_SYM(r->r_info)); | ||
953 | break; | 1300 | break; |
954 | case R_ARM_PC24: | 1301 | case R_ARM_PC24: |
955 | /* From ARM ABI: ((S + A) | T) - P */ | 1302 | /* From ARM ABI: ((S + A) | T) - P */ |
956 | r->r_addend = (int)(long)(elf->hdr + elf->sechdrs[rsection].sh_offset + | 1303 | r->r_addend = (int)(long)(elf->hdr + |
957 | (r->r_offset - elf->sechdrs[rsection].sh_addr)); | 1304 | sechdr->sh_offset + |
1305 | (r->r_offset - sechdr->sh_addr)); | ||
958 | break; | 1306 | break; |
959 | default: | 1307 | default: |
960 | return 1; | 1308 | return 1; |
@@ -962,10 +1310,10 @@ static int addend_arm_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | |||
962 | return 0; | 1310 | return 0; |
963 | } | 1311 | } |
964 | 1312 | ||
965 | static int addend_mips_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | 1313 | static int addend_mips_rel(struct elf_info *elf, Elf_Shdr *sechdr, Elf_Rela *r) |
966 | { | 1314 | { |
967 | unsigned int r_typ = ELF_R_TYPE(r->r_info); | 1315 | unsigned int r_typ = ELF_R_TYPE(r->r_info); |
968 | unsigned int *location = reloc_location(elf, rsection, r); | 1316 | unsigned int *location = reloc_location(elf, sechdr, r); |
969 | unsigned int inst; | 1317 | unsigned int inst; |
970 | 1318 | ||
971 | if (r_typ == R_MIPS_HI16) | 1319 | if (r_typ == R_MIPS_HI16) |
@@ -985,6 +1333,108 @@ static int addend_mips_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | |||
985 | return 0; | 1333 | return 0; |
986 | } | 1334 | } |
987 | 1335 | ||
1336 | static void section_rela(const char *modname, struct elf_info *elf, | ||
1337 | Elf_Shdr *sechdr) | ||
1338 | { | ||
1339 | Elf_Sym *sym; | ||
1340 | Elf_Rela *rela; | ||
1341 | Elf_Rela r; | ||
1342 | unsigned int r_sym; | ||
1343 | const char *fromsec; | ||
1344 | |||
1345 | Elf_Rela *start = (void *)elf->hdr + sechdr->sh_offset; | ||
1346 | Elf_Rela *stop = (void *)start + sechdr->sh_size; | ||
1347 | |||
1348 | fromsec = sech_name(elf, sechdr); | ||
1349 | fromsec += strlen(".rela"); | ||
1350 | /* if from section (name) is know good then skip it */ | ||
1351 | if (check_section(modname, fromsec)) | ||
1352 | return; | ||
1353 | |||
1354 | for (rela = start; rela < stop; rela++) { | ||
1355 | r.r_offset = TO_NATIVE(rela->r_offset); | ||
1356 | #if KERNEL_ELFCLASS == ELFCLASS64 | ||
1357 | if (elf->hdr->e_machine == EM_MIPS) { | ||
1358 | unsigned int r_typ; | ||
1359 | r_sym = ELF64_MIPS_R_SYM(rela->r_info); | ||
1360 | r_sym = TO_NATIVE(r_sym); | ||
1361 | r_typ = ELF64_MIPS_R_TYPE(rela->r_info); | ||
1362 | r.r_info = ELF64_R_INFO(r_sym, r_typ); | ||
1363 | } else { | ||
1364 | r.r_info = TO_NATIVE(rela->r_info); | ||
1365 | r_sym = ELF_R_SYM(r.r_info); | ||
1366 | } | ||
1367 | #else | ||
1368 | r.r_info = TO_NATIVE(rela->r_info); | ||
1369 | r_sym = ELF_R_SYM(r.r_info); | ||
1370 | #endif | ||
1371 | r.r_addend = TO_NATIVE(rela->r_addend); | ||
1372 | sym = elf->symtab_start + r_sym; | ||
1373 | /* Skip special sections */ | ||
1374 | if (sym->st_shndx >= SHN_LORESERVE) | ||
1375 | continue; | ||
1376 | check_section_mismatch(modname, elf, &r, sym, fromsec); | ||
1377 | } | ||
1378 | } | ||
1379 | |||
1380 | static void section_rel(const char *modname, struct elf_info *elf, | ||
1381 | Elf_Shdr *sechdr) | ||
1382 | { | ||
1383 | Elf_Sym *sym; | ||
1384 | Elf_Rel *rel; | ||
1385 | Elf_Rela r; | ||
1386 | unsigned int r_sym; | ||
1387 | const char *fromsec; | ||
1388 | |||
1389 | Elf_Rel *start = (void *)elf->hdr + sechdr->sh_offset; | ||
1390 | Elf_Rel *stop = (void *)start + sechdr->sh_size; | ||
1391 | |||
1392 | fromsec = sech_name(elf, sechdr); | ||
1393 | fromsec += strlen(".rel"); | ||
1394 | /* if from section (name) is know good then skip it */ | ||
1395 | if (check_section(modname, fromsec)) | ||
1396 | return; | ||
1397 | |||
1398 | for (rel = start; rel < stop; rel++) { | ||
1399 | r.r_offset = TO_NATIVE(rel->r_offset); | ||
1400 | #if KERNEL_ELFCLASS == ELFCLASS64 | ||
1401 | if (elf->hdr->e_machine == EM_MIPS) { | ||
1402 | unsigned int r_typ; | ||
1403 | r_sym = ELF64_MIPS_R_SYM(rel->r_info); | ||
1404 | r_sym = TO_NATIVE(r_sym); | ||
1405 | r_typ = ELF64_MIPS_R_TYPE(rel->r_info); | ||
1406 | r.r_info = ELF64_R_INFO(r_sym, r_typ); | ||
1407 | } else { | ||
1408 | r.r_info = TO_NATIVE(rel->r_info); | ||
1409 | r_sym = ELF_R_SYM(r.r_info); | ||
1410 | } | ||
1411 | #else | ||
1412 | r.r_info = TO_NATIVE(rel->r_info); | ||
1413 | r_sym = ELF_R_SYM(r.r_info); | ||
1414 | #endif | ||
1415 | r.r_addend = 0; | ||
1416 | switch (elf->hdr->e_machine) { | ||
1417 | case EM_386: | ||
1418 | if (addend_386_rel(elf, sechdr, &r)) | ||
1419 | continue; | ||
1420 | break; | ||
1421 | case EM_ARM: | ||
1422 | if (addend_arm_rel(elf, sechdr, &r)) | ||
1423 | continue; | ||
1424 | break; | ||
1425 | case EM_MIPS: | ||
1426 | if (addend_mips_rel(elf, sechdr, &r)) | ||
1427 | continue; | ||
1428 | break; | ||
1429 | } | ||
1430 | sym = elf->symtab_start + r_sym; | ||
1431 | /* Skip special sections */ | ||
1432 | if (sym->st_shndx >= SHN_LORESERVE) | ||
1433 | continue; | ||
1434 | check_section_mismatch(modname, elf, &r, sym, fromsec); | ||
1435 | } | ||
1436 | } | ||
1437 | |||
988 | /** | 1438 | /** |
989 | * A module includes a number of sections that are discarded | 1439 | * A module includes a number of sections that are discarded |
990 | * either when loaded or when used as built-in. | 1440 | * either when loaded or when used as built-in. |
@@ -998,257 +1448,21 @@ static int addend_mips_rel(struct elf_info *elf, int rsection, Elf_Rela *r) | |||
998 | * be discarded and warns about it. | 1448 | * be discarded and warns about it. |
999 | **/ | 1449 | **/ |
1000 | static void check_sec_ref(struct module *mod, const char *modname, | 1450 | static void check_sec_ref(struct module *mod, const char *modname, |
1001 | struct elf_info *elf, | 1451 | struct elf_info *elf) |
1002 | int section(const char*), | ||
1003 | int section_ref_ok(const char *)) | ||
1004 | { | 1452 | { |
1005 | int i; | 1453 | int i; |
1006 | Elf_Sym *sym; | ||
1007 | Elf_Ehdr *hdr = elf->hdr; | ||
1008 | Elf_Shdr *sechdrs = elf->sechdrs; | 1454 | Elf_Shdr *sechdrs = elf->sechdrs; |
1009 | const char *secstrings = (void *)hdr + | ||
1010 | sechdrs[hdr->e_shstrndx].sh_offset; | ||
1011 | 1455 | ||
1012 | /* Walk through all sections */ | 1456 | /* Walk through all sections */ |
1013 | for (i = 0; i < hdr->e_shnum; i++) { | 1457 | for (i = 0; i < elf->hdr->e_shnum; i++) { |
1014 | const char *name = secstrings + sechdrs[i].sh_name; | ||
1015 | const char *secname; | ||
1016 | Elf_Rela r; | ||
1017 | unsigned int r_sym; | ||
1018 | /* We want to process only relocation sections and not .init */ | 1458 | /* We want to process only relocation sections and not .init */ |
1019 | if (sechdrs[i].sh_type == SHT_RELA) { | 1459 | if (sechdrs[i].sh_type == SHT_RELA) |
1020 | Elf_Rela *rela; | 1460 | section_rela(modname, elf, &elf->sechdrs[i]); |
1021 | Elf_Rela *start = (void *)hdr + sechdrs[i].sh_offset; | 1461 | else if (sechdrs[i].sh_type == SHT_REL) |
1022 | Elf_Rela *stop = (void*)start + sechdrs[i].sh_size; | 1462 | section_rel(modname, elf, &elf->sechdrs[i]); |
1023 | name += strlen(".rela"); | ||
1024 | if (section_ref_ok(name)) | ||
1025 | continue; | ||
1026 | |||
1027 | for (rela = start; rela < stop; rela++) { | ||
1028 | r.r_offset = TO_NATIVE(rela->r_offset); | ||
1029 | #if KERNEL_ELFCLASS == ELFCLASS64 | ||
1030 | if (hdr->e_machine == EM_MIPS) { | ||
1031 | unsigned int r_typ; | ||
1032 | r_sym = ELF64_MIPS_R_SYM(rela->r_info); | ||
1033 | r_sym = TO_NATIVE(r_sym); | ||
1034 | r_typ = ELF64_MIPS_R_TYPE(rela->r_info); | ||
1035 | r.r_info = ELF64_R_INFO(r_sym, r_typ); | ||
1036 | } else { | ||
1037 | r.r_info = TO_NATIVE(rela->r_info); | ||
1038 | r_sym = ELF_R_SYM(r.r_info); | ||
1039 | } | ||
1040 | #else | ||
1041 | r.r_info = TO_NATIVE(rela->r_info); | ||
1042 | r_sym = ELF_R_SYM(r.r_info); | ||
1043 | #endif | ||
1044 | r.r_addend = TO_NATIVE(rela->r_addend); | ||
1045 | sym = elf->symtab_start + r_sym; | ||
1046 | /* Skip special sections */ | ||
1047 | if (sym->st_shndx >= SHN_LORESERVE) | ||
1048 | continue; | ||
1049 | |||
1050 | secname = secstrings + | ||
1051 | sechdrs[sym->st_shndx].sh_name; | ||
1052 | if (section(secname)) | ||
1053 | warn_sec_mismatch(modname, name, | ||
1054 | elf, sym, r); | ||
1055 | } | ||
1056 | } else if (sechdrs[i].sh_type == SHT_REL) { | ||
1057 | Elf_Rel *rel; | ||
1058 | Elf_Rel *start = (void *)hdr + sechdrs[i].sh_offset; | ||
1059 | Elf_Rel *stop = (void*)start + sechdrs[i].sh_size; | ||
1060 | name += strlen(".rel"); | ||
1061 | if (section_ref_ok(name)) | ||
1062 | continue; | ||
1063 | |||
1064 | for (rel = start; rel < stop; rel++) { | ||
1065 | r.r_offset = TO_NATIVE(rel->r_offset); | ||
1066 | #if KERNEL_ELFCLASS == ELFCLASS64 | ||
1067 | if (hdr->e_machine == EM_MIPS) { | ||
1068 | unsigned int r_typ; | ||
1069 | r_sym = ELF64_MIPS_R_SYM(rel->r_info); | ||
1070 | r_sym = TO_NATIVE(r_sym); | ||
1071 | r_typ = ELF64_MIPS_R_TYPE(rel->r_info); | ||
1072 | r.r_info = ELF64_R_INFO(r_sym, r_typ); | ||
1073 | } else { | ||
1074 | r.r_info = TO_NATIVE(rel->r_info); | ||
1075 | r_sym = ELF_R_SYM(r.r_info); | ||
1076 | } | ||
1077 | #else | ||
1078 | r.r_info = TO_NATIVE(rel->r_info); | ||
1079 | r_sym = ELF_R_SYM(r.r_info); | ||
1080 | #endif | ||
1081 | r.r_addend = 0; | ||
1082 | switch (hdr->e_machine) { | ||
1083 | case EM_386: | ||
1084 | if (addend_386_rel(elf, i, &r)) | ||
1085 | continue; | ||
1086 | break; | ||
1087 | case EM_ARM: | ||
1088 | if(addend_arm_rel(elf, i, &r)) | ||
1089 | continue; | ||
1090 | break; | ||
1091 | case EM_MIPS: | ||
1092 | if (addend_mips_rel(elf, i, &r)) | ||
1093 | continue; | ||
1094 | break; | ||
1095 | } | ||
1096 | sym = elf->symtab_start + r_sym; | ||
1097 | /* Skip special sections */ | ||
1098 | if (sym->st_shndx >= SHN_LORESERVE) | ||
1099 | continue; | ||
1100 | |||
1101 | secname = secstrings + | ||
1102 | sechdrs[sym->st_shndx].sh_name; | ||
1103 | if (section(secname)) | ||
1104 | warn_sec_mismatch(modname, name, | ||
1105 | elf, sym, r); | ||
1106 | } | ||
1107 | } | ||
1108 | } | 1463 | } |
1109 | } | 1464 | } |
1110 | 1465 | ||
1111 | /* | ||
1112 | * Identify sections from which references to either a | ||
1113 | * .init or a .exit section is OK. | ||
1114 | * | ||
1115 | * [OPD] Keith Ownes <kaos@sgi.com> commented: | ||
1116 | * For our future {in}sanity, add a comment that this is the ppc .opd | ||
1117 | * section, not the ia64 .opd section. | ||
1118 | * ia64 .opd should not point to discarded sections. | ||
1119 | * [.rodata] like for .init.text we ignore .rodata references -same reason | ||
1120 | */ | ||
1121 | static int initexit_section_ref_ok(const char *name) | ||
1122 | { | ||
1123 | const char **s; | ||
1124 | /* Absolute section names */ | ||
1125 | const char *namelist1[] = { | ||
1126 | "__bug_table", /* used by powerpc for BUG() */ | ||
1127 | "__ex_table", | ||
1128 | ".altinstructions", | ||
1129 | ".cranges", /* used by sh64 */ | ||
1130 | ".fixup", | ||
1131 | ".machvec", /* ia64 + powerpc uses these */ | ||
1132 | ".machine.desc", | ||
1133 | ".opd", /* See comment [OPD] */ | ||
1134 | "__dbe_table", | ||
1135 | ".parainstructions", | ||
1136 | ".pdr", | ||
1137 | ".plt", /* seen on ARCH=um build on x86_64. Harmless */ | ||
1138 | ".smp_locks", | ||
1139 | ".stab", | ||
1140 | ".m68k_fixup", | ||
1141 | ".xt.prop", /* xtensa informational section */ | ||
1142 | ".xt.lit", /* xtensa informational section */ | ||
1143 | NULL | ||
1144 | }; | ||
1145 | /* Start of section names */ | ||
1146 | const char *namelist2[] = { | ||
1147 | ".debug", | ||
1148 | ".eh_frame", | ||
1149 | ".note", /* ignore ELF notes - may contain anything */ | ||
1150 | ".got", /* powerpc - global offset table */ | ||
1151 | ".toc", /* powerpc - table of contents */ | ||
1152 | NULL | ||
1153 | }; | ||
1154 | /* part of section name */ | ||
1155 | const char *namelist3 [] = { | ||
1156 | ".unwind", /* Sample: IA_64.unwind.exit.text */ | ||
1157 | NULL | ||
1158 | }; | ||
1159 | |||
1160 | for (s = namelist1; *s; s++) | ||
1161 | if (strcmp(*s, name) == 0) | ||
1162 | return 1; | ||
1163 | for (s = namelist2; *s; s++) | ||
1164 | if (strncmp(*s, name, strlen(*s)) == 0) | ||
1165 | return 1; | ||
1166 | for (s = namelist3; *s; s++) | ||
1167 | if (strstr(name, *s) != NULL) | ||
1168 | return 1; | ||
1169 | return 0; | ||
1170 | } | ||
1171 | |||
1172 | |||
1173 | /* | ||
1174 | * Identify sections from which references to a .init section is OK. | ||
1175 | * | ||
1176 | * Unfortunately references to read only data that referenced .init | ||
1177 | * sections had to be excluded. Almost all of these are false | ||
1178 | * positives, they are created by gcc. The downside of excluding rodata | ||
1179 | * is that there really are some user references from rodata to | ||
1180 | * init code, e.g. drivers/video/vgacon.c: | ||
1181 | * | ||
1182 | * const struct consw vga_con = { | ||
1183 | * con_startup: vgacon_startup, | ||
1184 | * | ||
1185 | * where vgacon_startup is __init. If you want to wade through the false | ||
1186 | * positives, take out the check for rodata. | ||
1187 | */ | ||
1188 | static int init_section_ref_ok(const char *name) | ||
1189 | { | ||
1190 | const char **s; | ||
1191 | /* Absolute section names */ | ||
1192 | const char *namelist1[] = { | ||
1193 | "__dbe_table", /* MIPS generate these */ | ||
1194 | "__ftr_fixup", /* powerpc cpu feature fixup */ | ||
1195 | "__fw_ftr_fixup", /* powerpc firmware feature fixup */ | ||
1196 | "__param", | ||
1197 | ".data.rel.ro", /* used by parisc64 */ | ||
1198 | ".init", | ||
1199 | ".text.lock", | ||
1200 | NULL | ||
1201 | }; | ||
1202 | /* Start of section names */ | ||
1203 | const char *namelist2[] = { | ||
1204 | ".init.", | ||
1205 | ".pci_fixup", | ||
1206 | ".rodata", | ||
1207 | NULL | ||
1208 | }; | ||
1209 | |||
1210 | if (initexit_section_ref_ok(name)) | ||
1211 | return 1; | ||
1212 | |||
1213 | for (s = namelist1; *s; s++) | ||
1214 | if (strcmp(*s, name) == 0) | ||
1215 | return 1; | ||
1216 | for (s = namelist2; *s; s++) | ||
1217 | if (strncmp(*s, name, strlen(*s)) == 0) | ||
1218 | return 1; | ||
1219 | |||
1220 | /* If section name ends with ".init" we allow references | ||
1221 | * as is the case with .initcallN.init, .early_param.init, .taglist.init etc | ||
1222 | */ | ||
1223 | if (strrcmp(name, ".init") == 0) | ||
1224 | return 1; | ||
1225 | return 0; | ||
1226 | } | ||
1227 | |||
1228 | /* | ||
1229 | * Identify sections from which references to a .exit section is OK. | ||
1230 | */ | ||
1231 | static int exit_section_ref_ok(const char *name) | ||
1232 | { | ||
1233 | const char **s; | ||
1234 | /* Absolute section names */ | ||
1235 | const char *namelist1[] = { | ||
1236 | ".exit.data", | ||
1237 | ".exit.text", | ||
1238 | ".exitcall.exit", | ||
1239 | ".rodata", | ||
1240 | NULL | ||
1241 | }; | ||
1242 | |||
1243 | if (initexit_section_ref_ok(name)) | ||
1244 | return 1; | ||
1245 | |||
1246 | for (s = namelist1; *s; s++) | ||
1247 | if (strcmp(*s, name) == 0) | ||
1248 | return 1; | ||
1249 | return 0; | ||
1250 | } | ||
1251 | |||
1252 | static void read_symbols(char *modname) | 1466 | static void read_symbols(char *modname) |
1253 | { | 1467 | { |
1254 | const char *symname; | 1468 | const char *symname; |
@@ -1288,10 +1502,9 @@ static void read_symbols(char *modname) | |||
1288 | handle_modversions(mod, &info, sym, symname); | 1502 | handle_modversions(mod, &info, sym, symname); |
1289 | handle_moddevtable(mod, &info, sym, symname); | 1503 | handle_moddevtable(mod, &info, sym, symname); |
1290 | } | 1504 | } |
1291 | if (is_vmlinux(modname) && vmlinux_section_warnings) { | 1505 | if (!is_vmlinux(modname) || |
1292 | check_sec_ref(mod, modname, &info, init_section, init_section_ref_ok); | 1506 | (is_vmlinux(modname) && vmlinux_section_warnings)) |
1293 | check_sec_ref(mod, modname, &info, exit_section, exit_section_ref_ok); | 1507 | check_sec_ref(mod, modname, &info); |
1294 | } | ||
1295 | 1508 | ||
1296 | version = get_modinfo(info.modinfo, info.modinfo_len, "version"); | 1509 | version = get_modinfo(info.modinfo, info.modinfo_len, "version"); |
1297 | if (version) | 1510 | if (version) |
@@ -1365,7 +1578,7 @@ static void check_for_gpl_usage(enum export exp, const char *m, const char *s) | |||
1365 | } | 1578 | } |
1366 | } | 1579 | } |
1367 | 1580 | ||
1368 | static void check_for_unused(enum export exp, const char* m, const char* s) | 1581 | static void check_for_unused(enum export exp, const char *m, const char *s) |
1369 | { | 1582 | { |
1370 | const char *e = is_vmlinux(m) ?"":".ko"; | 1583 | const char *e = is_vmlinux(m) ?"":".ko"; |
1371 | 1584 | ||
@@ -1398,7 +1611,7 @@ static void check_exports(struct module *mod) | |||
1398 | if (!mod->gpl_compatible) | 1611 | if (!mod->gpl_compatible) |
1399 | check_for_gpl_usage(exp->export, basename, exp->name); | 1612 | check_for_gpl_usage(exp->export, basename, exp->name); |
1400 | check_for_unused(exp->export, basename, exp->name); | 1613 | check_for_unused(exp->export, basename, exp->name); |
1401 | } | 1614 | } |
1402 | } | 1615 | } |
1403 | 1616 | ||
1404 | /** | 1617 | /** |
@@ -1458,13 +1671,12 @@ static int add_versions(struct buffer *b, struct module *mod) | |||
1458 | 1671 | ||
1459 | buf_printf(b, "\n"); | 1672 | buf_printf(b, "\n"); |
1460 | buf_printf(b, "static const struct modversion_info ____versions[]\n"); | 1673 | buf_printf(b, "static const struct modversion_info ____versions[]\n"); |
1461 | buf_printf(b, "__attribute_used__\n"); | 1674 | buf_printf(b, "__used\n"); |
1462 | buf_printf(b, "__attribute__((section(\"__versions\"))) = {\n"); | 1675 | buf_printf(b, "__attribute__((section(\"__versions\"))) = {\n"); |
1463 | 1676 | ||
1464 | for (s = mod->unres; s; s = s->next) { | 1677 | for (s = mod->unres; s; s = s->next) { |
1465 | if (!s->module) { | 1678 | if (!s->module) |
1466 | continue; | 1679 | continue; |
1467 | } | ||
1468 | if (!s->crc_valid) { | 1680 | if (!s->crc_valid) { |
1469 | warn("\"%s\" [%s.ko] has no CRC!\n", | 1681 | warn("\"%s\" [%s.ko] has no CRC!\n", |
1470 | s->name, mod->name); | 1682 | s->name, mod->name); |
@@ -1485,13 +1697,12 @@ static void add_depends(struct buffer *b, struct module *mod, | |||
1485 | struct module *m; | 1697 | struct module *m; |
1486 | int first = 1; | 1698 | int first = 1; |
1487 | 1699 | ||
1488 | for (m = modules; m; m = m->next) { | 1700 | for (m = modules; m; m = m->next) |
1489 | m->seen = is_vmlinux(m->name); | 1701 | m->seen = is_vmlinux(m->name); |
1490 | } | ||
1491 | 1702 | ||
1492 | buf_printf(b, "\n"); | 1703 | buf_printf(b, "\n"); |
1493 | buf_printf(b, "static const char __module_depends[]\n"); | 1704 | buf_printf(b, "static const char __module_depends[]\n"); |
1494 | buf_printf(b, "__attribute_used__\n"); | 1705 | buf_printf(b, "__used\n"); |
1495 | buf_printf(b, "__attribute__((section(\".modinfo\"))) =\n"); | 1706 | buf_printf(b, "__attribute__((section(\".modinfo\"))) =\n"); |
1496 | buf_printf(b, "\"depends="); | 1707 | buf_printf(b, "\"depends="); |
1497 | for (s = mod->unres; s; s = s->next) { | 1708 | for (s = mod->unres; s; s = s->next) { |
@@ -1503,7 +1714,8 @@ static void add_depends(struct buffer *b, struct module *mod, | |||
1503 | continue; | 1714 | continue; |
1504 | 1715 | ||
1505 | s->module->seen = 1; | 1716 | s->module->seen = 1; |
1506 | if ((p = strrchr(s->module->name, '/')) != NULL) | 1717 | p = strrchr(s->module->name, '/'); |
1718 | if (p) | ||
1507 | p++; | 1719 | p++; |
1508 | else | 1720 | else |
1509 | p = s->module->name; | 1721 | p = s->module->name; |
@@ -1575,7 +1787,7 @@ static void read_dump(const char *fname, unsigned int kernel) | |||
1575 | void *file = grab_file(fname, &size); | 1787 | void *file = grab_file(fname, &size); |
1576 | char *line; | 1788 | char *line; |
1577 | 1789 | ||
1578 | if (!file) | 1790 | if (!file) |
1579 | /* No symbol versions, silently ignore */ | 1791 | /* No symbol versions, silently ignore */ |
1580 | return; | 1792 | return; |
1581 | 1793 | ||
@@ -1598,11 +1810,10 @@ static void read_dump(const char *fname, unsigned int kernel) | |||
1598 | crc = strtoul(line, &d, 16); | 1810 | crc = strtoul(line, &d, 16); |
1599 | if (*symname == '\0' || *modname == '\0' || *d != '\0') | 1811 | if (*symname == '\0' || *modname == '\0' || *d != '\0') |
1600 | goto fail; | 1812 | goto fail; |
1601 | 1813 | mod = find_module(modname); | |
1602 | if (!(mod = find_module(modname))) { | 1814 | if (!mod) { |
1603 | if (is_vmlinux(modname)) { | 1815 | if (is_vmlinux(modname)) |
1604 | have_vmlinux = 1; | 1816 | have_vmlinux = 1; |
1605 | } | ||
1606 | mod = new_module(NOFAIL(strdup(modname))); | 1817 | mod = new_module(NOFAIL(strdup(modname))); |
1607 | mod->skip = 1; | 1818 | mod->skip = 1; |
1608 | } | 1819 | } |
@@ -1653,38 +1864,40 @@ int main(int argc, char **argv) | |||
1653 | { | 1864 | { |
1654 | struct module *mod; | 1865 | struct module *mod; |
1655 | struct buffer buf = { }; | 1866 | struct buffer buf = { }; |
1656 | char fname[SZ]; | ||
1657 | char *kernel_read = NULL, *module_read = NULL; | 1867 | char *kernel_read = NULL, *module_read = NULL; |
1658 | char *dump_write = NULL; | 1868 | char *dump_write = NULL; |
1659 | int opt; | 1869 | int opt; |
1660 | int err; | 1870 | int err; |
1661 | 1871 | ||
1662 | while ((opt = getopt(argc, argv, "i:I:mso:aw")) != -1) { | 1872 | while ((opt = getopt(argc, argv, "i:I:msSo:aw")) != -1) { |
1663 | switch(opt) { | 1873 | switch (opt) { |
1664 | case 'i': | 1874 | case 'i': |
1665 | kernel_read = optarg; | 1875 | kernel_read = optarg; |
1666 | break; | 1876 | break; |
1667 | case 'I': | 1877 | case 'I': |
1668 | module_read = optarg; | 1878 | module_read = optarg; |
1669 | external_module = 1; | 1879 | external_module = 1; |
1670 | break; | 1880 | break; |
1671 | case 'm': | 1881 | case 'm': |
1672 | modversions = 1; | 1882 | modversions = 1; |
1673 | break; | 1883 | break; |
1674 | case 'o': | 1884 | case 'o': |
1675 | dump_write = optarg; | 1885 | dump_write = optarg; |
1676 | break; | 1886 | break; |
1677 | case 'a': | 1887 | case 'a': |
1678 | all_versions = 1; | 1888 | all_versions = 1; |
1679 | break; | 1889 | break; |
1680 | case 's': | 1890 | case 's': |
1681 | vmlinux_section_warnings = 0; | 1891 | vmlinux_section_warnings = 0; |
1682 | break; | 1892 | break; |
1683 | case 'w': | 1893 | case 'S': |
1684 | warn_unresolved = 1; | 1894 | sec_mismatch_verbose = 0; |
1685 | break; | 1895 | break; |
1686 | default: | 1896 | case 'w': |
1687 | exit(1); | 1897 | warn_unresolved = 1; |
1898 | break; | ||
1899 | default: | ||
1900 | exit(1); | ||
1688 | } | 1901 | } |
1689 | } | 1902 | } |
1690 | 1903 | ||
@@ -1693,9 +1906,8 @@ int main(int argc, char **argv) | |||
1693 | if (module_read) | 1906 | if (module_read) |
1694 | read_dump(module_read, 0); | 1907 | read_dump(module_read, 0); |
1695 | 1908 | ||
1696 | while (optind < argc) { | 1909 | while (optind < argc) |
1697 | read_symbols(argv[optind++]); | 1910 | read_symbols(argv[optind++]); |
1698 | } | ||
1699 | 1911 | ||
1700 | for (mod = modules; mod; mod = mod->next) { | 1912 | for (mod = modules; mod; mod = mod->next) { |
1701 | if (mod->skip) | 1913 | if (mod->skip) |
@@ -1706,6 +1918,8 @@ int main(int argc, char **argv) | |||
1706 | err = 0; | 1918 | err = 0; |
1707 | 1919 | ||
1708 | for (mod = modules; mod; mod = mod->next) { | 1920 | for (mod = modules; mod; mod = mod->next) { |
1921 | char fname[strlen(mod->name) + 10]; | ||
1922 | |||
1709 | if (mod->skip) | 1923 | if (mod->skip) |
1710 | continue; | 1924 | continue; |
1711 | 1925 | ||
@@ -1723,6 +1937,12 @@ int main(int argc, char **argv) | |||
1723 | 1937 | ||
1724 | if (dump_write) | 1938 | if (dump_write) |
1725 | write_dump(dump_write); | 1939 | write_dump(dump_write); |
1940 | if (sec_mismatch_count && !sec_mismatch_verbose) | ||
1941 | fprintf(stderr, "modpost: Found %d section mismatch(es).\n" | ||
1942 | "To see additional details select \"Enable full " | ||
1943 | "Section mismatch analysis\"\n" | ||
1944 | "in the Kernel Hacking menu " | ||
1945 | "(CONFIG_SECTION_MISMATCH).\n", sec_mismatch_count); | ||
1726 | 1946 | ||
1727 | return err; | 1947 | return err; |
1728 | } | 1948 | } |
diff --git a/scripts/mod/modpost.h b/scripts/mod/modpost.h index 0ffed17ec20c..999f15e0e008 100644 --- a/scripts/mod/modpost.h +++ b/scripts/mod/modpost.h | |||
@@ -17,6 +17,7 @@ | |||
17 | #define Elf_Shdr Elf32_Shdr | 17 | #define Elf_Shdr Elf32_Shdr |
18 | #define Elf_Sym Elf32_Sym | 18 | #define Elf_Sym Elf32_Sym |
19 | #define Elf_Addr Elf32_Addr | 19 | #define Elf_Addr Elf32_Addr |
20 | #define Elf_Sword Elf64_Sword | ||
20 | #define Elf_Section Elf32_Half | 21 | #define Elf_Section Elf32_Half |
21 | #define ELF_ST_BIND ELF32_ST_BIND | 22 | #define ELF_ST_BIND ELF32_ST_BIND |
22 | #define ELF_ST_TYPE ELF32_ST_TYPE | 23 | #define ELF_ST_TYPE ELF32_ST_TYPE |
@@ -31,6 +32,7 @@ | |||
31 | #define Elf_Shdr Elf64_Shdr | 32 | #define Elf_Shdr Elf64_Shdr |
32 | #define Elf_Sym Elf64_Sym | 33 | #define Elf_Sym Elf64_Sym |
33 | #define Elf_Addr Elf64_Addr | 34 | #define Elf_Addr Elf64_Addr |
35 | #define Elf_Sword Elf64_Sxword | ||
34 | #define Elf_Section Elf64_Half | 36 | #define Elf_Section Elf64_Half |
35 | #define ELF_ST_BIND ELF64_ST_BIND | 37 | #define ELF_ST_BIND ELF64_ST_BIND |
36 | #define ELF_ST_TYPE ELF64_ST_TYPE | 38 | #define ELF_ST_TYPE ELF64_ST_TYPE |
diff --git a/scripts/package/Makefile b/scripts/package/Makefile index 7c434e037e7f..5e326078a4a2 100644 --- a/scripts/package/Makefile +++ b/scripts/package/Makefile | |||
@@ -89,9 +89,8 @@ clean-dirs += $(objtree)/tar-install/ | |||
89 | # Help text displayed when executing 'make help' | 89 | # Help text displayed when executing 'make help' |
90 | # --------------------------------------------------------------------------- | 90 | # --------------------------------------------------------------------------- |
91 | help: FORCE | 91 | help: FORCE |
92 | @echo ' rpm-pkg - Build the kernel as an RPM package' | 92 | @echo ' rpm-pkg - Build both source and binary RPM kernel packages' |
93 | @echo ' binrpm-pkg - Build an rpm package containing the compiled kernel' | 93 | @echo ' binrpm-pkg - Build only the binary kernel package' |
94 | @echo ' and modules' | ||
95 | @echo ' deb-pkg - Build the kernel as an deb package' | 94 | @echo ' deb-pkg - Build the kernel as an deb package' |
96 | @echo ' tar-pkg - Build the kernel as an uncompressed tarball' | 95 | @echo ' tar-pkg - Build the kernel as an uncompressed tarball' |
97 | @echo ' targz-pkg - Build the kernel as a gzip compressed tarball' | 96 | @echo ' targz-pkg - Build the kernel as a gzip compressed tarball' |
diff --git a/scripts/package/buildtar b/scripts/package/buildtar index aa0ccdbd1f47..28574ae55170 100644 --- a/scripts/package/buildtar +++ b/scripts/package/buildtar | |||
@@ -69,8 +69,8 @@ cp -v -- "${objtree}/vmlinux" "${tmpdir}/boot/vmlinux-${KERNELRELEASE}" | |||
69 | # Install arch-specific kernel image(s) | 69 | # Install arch-specific kernel image(s) |
70 | # | 70 | # |
71 | case "${ARCH}" in | 71 | case "${ARCH}" in |
72 | i386|x86_64) | 72 | x86|i386|x86_64) |
73 | [ -f "${objtree}/arch/$ARCH/boot/bzImage" ] && cp -v -- "${objtree}/arch/$ARCH/boot/bzImage" "${tmpdir}/boot/vmlinuz-${KERNELRELEASE}" | 73 | [ -f "${objtree}/arch/x86/boot/bzImage" ] && cp -v -- "${objtree}/arch/x86/boot/bzImage" "${tmpdir}/boot/vmlinuz-${KERNELRELEASE}" |
74 | ;; | 74 | ;; |
75 | alpha) | 75 | alpha) |
76 | [ -f "${objtree}/arch/alpha/boot/vmlinux.gz" ] && cp -v -- "${objtree}/arch/alpha/boot/vmlinux.gz" "${tmpdir}/boot/vmlinuz-${KERNELRELEASE}" | 76 | [ -f "${objtree}/arch/alpha/boot/vmlinux.gz" ] && cp -v -- "${objtree}/arch/alpha/boot/vmlinux.gz" "${tmpdir}/boot/vmlinuz-${KERNELRELEASE}" |
diff --git a/scripts/patch-kernel b/scripts/patch-kernel index 67e4b1868e50..ece46ef0ba54 100755 --- a/scripts/patch-kernel +++ b/scripts/patch-kernel | |||
@@ -65,7 +65,7 @@ sourcedir=${1-/usr/src/linux} | |||
65 | patchdir=${2-.} | 65 | patchdir=${2-.} |
66 | stopvers=${3-default} | 66 | stopvers=${3-default} |
67 | 67 | ||
68 | if [ "$1" == -h -o "$1" == --help -o ! -r "$sourcedir/Makefile" ]; then | 68 | if [ "$1" = -h -o "$1" = --help -o ! -r "$sourcedir/Makefile" ]; then |
69 | cat << USAGE | 69 | cat << USAGE |
70 | usage: $PNAME [-h] [ sourcedir [ patchdir [ stopversion ] [ -acxx ] ] ] | 70 | usage: $PNAME [-h] [ sourcedir [ patchdir [ stopversion ] [ -acxx ] ] ] |
71 | source directory defaults to /usr/src/linux, | 71 | source directory defaults to /usr/src/linux, |
@@ -182,10 +182,12 @@ reversePatch () { | |||
182 | } | 182 | } |
183 | 183 | ||
184 | # set current VERSION, PATCHLEVEL, SUBLEVEL, EXTRAVERSION | 184 | # set current VERSION, PATCHLEVEL, SUBLEVEL, EXTRAVERSION |
185 | TMPFILE=`mktemp .tmpver.XXXXXX` || { echo "cannot make temp file" ; exit 1; } | 185 | # force $TMPFILEs below to be in local directory: a slash character prevents |
186 | # the dot command from using the search path. | ||
187 | TMPFILE=`mktemp ./.tmpver.XXXXXX` || { echo "cannot make temp file" ; exit 1; } | ||
186 | grep -E "^(VERSION|PATCHLEVEL|SUBLEVEL|EXTRAVERSION)" $sourcedir/Makefile > $TMPFILE | 188 | grep -E "^(VERSION|PATCHLEVEL|SUBLEVEL|EXTRAVERSION)" $sourcedir/Makefile > $TMPFILE |
187 | tr -d [:blank:] < $TMPFILE > $TMPFILE.1 | 189 | tr -d [:blank:] < $TMPFILE > $TMPFILE.1 |
188 | source $TMPFILE.1 | 190 | . $TMPFILE.1 |
189 | rm -f $TMPFILE* | 191 | rm -f $TMPFILE* |
190 | if [ -z "$VERSION" -o -z "$PATCHLEVEL" -o -z "$SUBLEVEL" ] | 192 | if [ -z "$VERSION" -o -z "$PATCHLEVEL" -o -z "$SUBLEVEL" ] |
191 | then | 193 | then |
@@ -202,11 +204,7 @@ echo "Current kernel version is $VERSION.$PATCHLEVEL.$SUBLEVEL${EXTRAVERSION} ($ | |||
202 | EXTRAVER= | 204 | EXTRAVER= |
203 | if [ x$EXTRAVERSION != "x" ] | 205 | if [ x$EXTRAVERSION != "x" ] |
204 | then | 206 | then |
205 | if [ ${EXTRAVERSION:0:1} == "." ]; then | 207 | EXTRAVER=${EXTRAVERSION#.} |
206 | EXTRAVER=${EXTRAVERSION:1} | ||
207 | else | ||
208 | EXTRAVER=$EXTRAVERSION | ||
209 | fi | ||
210 | EXTRAVER=${EXTRAVER%%[[:punct:]]*} | 208 | EXTRAVER=${EXTRAVER%%[[:punct:]]*} |
211 | #echo "$PNAME: changing EXTRAVERSION from $EXTRAVERSION to $EXTRAVER" | 209 | #echo "$PNAME: changing EXTRAVERSION from $EXTRAVERSION to $EXTRAVER" |
212 | fi | 210 | fi |
@@ -251,16 +249,16 @@ while : # incrementing SUBLEVEL (s in v.p.s) | |||
251 | do | 249 | do |
252 | CURRENTFULLVERSION="$VERSION.$PATCHLEVEL.$SUBLEVEL" | 250 | CURRENTFULLVERSION="$VERSION.$PATCHLEVEL.$SUBLEVEL" |
253 | EXTRAVER= | 251 | EXTRAVER= |
254 | if [ $stopvers == $CURRENTFULLVERSION ]; then | 252 | if [ $stopvers = $CURRENTFULLVERSION ]; then |
255 | echo "Stopping at $CURRENTFULLVERSION base as requested." | 253 | echo "Stopping at $CURRENTFULLVERSION base as requested." |
256 | break | 254 | break |
257 | fi | 255 | fi |
258 | 256 | ||
259 | SUBLEVEL=$((SUBLEVEL + 1)) | 257 | SUBLEVEL=$(($SUBLEVEL + 1)) |
260 | FULLVERSION="$VERSION.$PATCHLEVEL.$SUBLEVEL" | 258 | FULLVERSION="$VERSION.$PATCHLEVEL.$SUBLEVEL" |
261 | #echo "#___ trying $FULLVERSION ___" | 259 | #echo "#___ trying $FULLVERSION ___" |
262 | 260 | ||
263 | if [ $((SUBLEVEL)) -gt $((STOPSUBLEVEL)) ]; then | 261 | if [ $(($SUBLEVEL)) -gt $(($STOPSUBLEVEL)) ]; then |
264 | echo "Stopping since sublevel ($SUBLEVEL) is beyond stop-sublevel ($STOPSUBLEVEL)" | 262 | echo "Stopping since sublevel ($SUBLEVEL) is beyond stop-sublevel ($STOPSUBLEVEL)" |
265 | exit 1 | 263 | exit 1 |
266 | fi | 264 | fi |
@@ -297,7 +295,7 @@ fi | |||
297 | if [ x$gotac != x ]; then | 295 | if [ x$gotac != x ]; then |
298 | # Out great user wants the -ac patches | 296 | # Out great user wants the -ac patches |
299 | # They could have done -ac (get latest) or -acxx where xx=version they want | 297 | # They could have done -ac (get latest) or -acxx where xx=version they want |
300 | if [ $gotac == "-ac" ]; then | 298 | if [ $gotac = "-ac" ]; then |
301 | # They want the latest version | 299 | # They want the latest version |
302 | HIGHESTPATCH=0 | 300 | HIGHESTPATCH=0 |
303 | for PATCHNAMES in $patchdir/patch-${CURRENTFULLVERSION}-ac*\.* | 301 | for PATCHNAMES in $patchdir/patch-${CURRENTFULLVERSION}-ac*\.* |
diff --git a/scripts/setlocalversion b/scripts/setlocalversion index 82e4993f0a73..52f032e409a3 100644 --- a/scripts/setlocalversion +++ b/scripts/setlocalversion | |||
@@ -12,11 +12,36 @@ cd "${1:-.}" || usage | |||
12 | if head=`git rev-parse --verify HEAD 2>/dev/null`; then | 12 | if head=`git rev-parse --verify HEAD 2>/dev/null`; then |
13 | # Do we have an untagged version? | 13 | # Do we have an untagged version? |
14 | if git name-rev --tags HEAD | grep -E '^HEAD[[:space:]]+(.*~[0-9]*|undefined)$' > /dev/null; then | 14 | if git name-rev --tags HEAD | grep -E '^HEAD[[:space:]]+(.*~[0-9]*|undefined)$' > /dev/null; then |
15 | printf '%s%s' -g `echo "$head" | cut -c1-8` | 15 | git describe | awk -F- '{printf("-%05d-%s", $(NF-1),$(NF))}' |
16 | fi | 16 | fi |
17 | 17 | ||
18 | # Are there uncommitted changes? | 18 | # Are there uncommitted changes? |
19 | if git diff-index HEAD | read dummy; then | 19 | git update-index --refresh --unmerged > /dev/null |
20 | if git diff-index --name-only HEAD | grep -v "^scripts/package" \ | ||
21 | | read dummy; then | ||
20 | printf '%s' -dirty | 22 | printf '%s' -dirty |
21 | fi | 23 | fi |
24 | |||
25 | # All done with git | ||
26 | exit | ||
27 | fi | ||
28 | |||
29 | # Check for mercurial and a mercurial repo. | ||
30 | if hgid=`hg id 2>/dev/null`; then | ||
31 | tag=`printf '%s' "$hgid" | cut -d' ' -f2` | ||
32 | |||
33 | # Do we have an untagged version? | ||
34 | if [ -z "$tag" -o "$tag" = tip ]; then | ||
35 | id=`printf '%s' "$hgid" | sed 's/[+ ].*//'` | ||
36 | printf '%s%s' -hg "$id" | ||
37 | fi | ||
38 | |||
39 | # Are there uncommitted changes? | ||
40 | # These are represented by + after the changeset id. | ||
41 | case "$hgid" in | ||
42 | *+|*+\ *) printf '%s' -dirty ;; | ||
43 | esac | ||
44 | |||
45 | # All done with mercurial | ||
46 | exit | ||
22 | fi | 47 | fi |