1*10465441SEvalZero /*
2*10465441SEvalZero * Copyright (C) 2002 Roman Zippel <[email protected]>
3*10465441SEvalZero * Released under the terms of the GNU GPL v2.0.
4*10465441SEvalZero */
5*10465441SEvalZero
6*10465441SEvalZero #include <sys/stat.h>
7*10465441SEvalZero #include <ctype.h>
8*10465441SEvalZero #include <errno.h>
9*10465441SEvalZero #include <fcntl.h>
10*10465441SEvalZero #include <stdarg.h>
11*10465441SEvalZero #include <stdio.h>
12*10465441SEvalZero #include <stdlib.h>
13*10465441SEvalZero #include <string.h>
14*10465441SEvalZero #include <time.h>
15*10465441SEvalZero #include <unistd.h>
16*10465441SEvalZero
17*10465441SEvalZero #include "lkc.h"
18*10465441SEvalZero
19*10465441SEvalZero struct conf_printer {
20*10465441SEvalZero void (*print_symbol)(FILE *, struct symbol *, const char *, void *);
21*10465441SEvalZero void (*print_comment)(FILE *, const char *, void *);
22*10465441SEvalZero };
23*10465441SEvalZero
24*10465441SEvalZero static void conf_warning(const char *fmt, ...)
25*10465441SEvalZero __attribute__ ((format (printf, 1, 2)));
26*10465441SEvalZero
27*10465441SEvalZero static void conf_message(const char *fmt, ...)
28*10465441SEvalZero __attribute__ ((format (printf, 1, 2)));
29*10465441SEvalZero
30*10465441SEvalZero static const char *conf_filename;
31*10465441SEvalZero static int conf_lineno, conf_warnings, conf_unsaved;
32*10465441SEvalZero
33*10465441SEvalZero const char conf_defname[] = "arch/$ARCH/defconfig";
34*10465441SEvalZero
conf_warning(const char * fmt,...)35*10465441SEvalZero static void conf_warning(const char *fmt, ...)
36*10465441SEvalZero {
37*10465441SEvalZero va_list ap;
38*10465441SEvalZero va_start(ap, fmt);
39*10465441SEvalZero fprintf(stderr, "%s:%d:warning: ", conf_filename, conf_lineno);
40*10465441SEvalZero vfprintf(stderr, fmt, ap);
41*10465441SEvalZero fprintf(stderr, "\n");
42*10465441SEvalZero va_end(ap);
43*10465441SEvalZero conf_warnings++;
44*10465441SEvalZero }
45*10465441SEvalZero
conf_default_message_callback(const char * fmt,va_list ap)46*10465441SEvalZero static void conf_default_message_callback(const char *fmt, va_list ap)
47*10465441SEvalZero {
48*10465441SEvalZero printf("#\n# ");
49*10465441SEvalZero vprintf(fmt, ap);
50*10465441SEvalZero printf("\n#\n");
51*10465441SEvalZero }
52*10465441SEvalZero
53*10465441SEvalZero static void (*conf_message_callback) (const char *fmt, va_list ap) =
54*10465441SEvalZero conf_default_message_callback;
conf_set_message_callback(void (* fn)(const char * fmt,va_list ap))55*10465441SEvalZero void conf_set_message_callback(void (*fn) (const char *fmt, va_list ap))
56*10465441SEvalZero {
57*10465441SEvalZero conf_message_callback = fn;
58*10465441SEvalZero }
59*10465441SEvalZero
conf_message(const char * fmt,...)60*10465441SEvalZero static void conf_message(const char *fmt, ...)
61*10465441SEvalZero {
62*10465441SEvalZero va_list ap;
63*10465441SEvalZero
64*10465441SEvalZero va_start(ap, fmt);
65*10465441SEvalZero if (conf_message_callback)
66*10465441SEvalZero conf_message_callback(fmt, ap);
67*10465441SEvalZero va_end(ap);
68*10465441SEvalZero }
69*10465441SEvalZero
conf_get_configname(void)70*10465441SEvalZero const char *conf_get_configname(void)
71*10465441SEvalZero {
72*10465441SEvalZero char *name = getenv("KCONFIG_CONFIG");
73*10465441SEvalZero
74*10465441SEvalZero return name ? name : ".config";
75*10465441SEvalZero }
76*10465441SEvalZero
conf_get_autoconfig_name(void)77*10465441SEvalZero const char *conf_get_autoconfig_name(void)
78*10465441SEvalZero {
79*10465441SEvalZero char *name = getenv("KCONFIG_AUTOCONFIG");
80*10465441SEvalZero
81*10465441SEvalZero return name ? name : "include/config/auto.conf";
82*10465441SEvalZero }
83*10465441SEvalZero
conf_expand_value(const char * in)84*10465441SEvalZero static char *conf_expand_value(const char *in)
85*10465441SEvalZero {
86*10465441SEvalZero struct symbol *sym;
87*10465441SEvalZero const char *src;
88*10465441SEvalZero static char res_value[SYMBOL_MAXLENGTH];
89*10465441SEvalZero char *dst, name[SYMBOL_MAXLENGTH];
90*10465441SEvalZero
91*10465441SEvalZero res_value[0] = 0;
92*10465441SEvalZero dst = name;
93*10465441SEvalZero while ((src = strchr(in, '$'))) {
94*10465441SEvalZero strncat(res_value, in, src - in);
95*10465441SEvalZero src++;
96*10465441SEvalZero dst = name;
97*10465441SEvalZero while (isalnum(*src) || *src == '_')
98*10465441SEvalZero *dst++ = *src++;
99*10465441SEvalZero *dst = 0;
100*10465441SEvalZero sym = sym_lookup(name, 0);
101*10465441SEvalZero sym_calc_value(sym);
102*10465441SEvalZero strcat(res_value, sym_get_string_value(sym));
103*10465441SEvalZero in = src;
104*10465441SEvalZero }
105*10465441SEvalZero strcat(res_value, in);
106*10465441SEvalZero
107*10465441SEvalZero return res_value;
108*10465441SEvalZero }
109*10465441SEvalZero
conf_get_default_confname(void)110*10465441SEvalZero char *conf_get_default_confname(void)
111*10465441SEvalZero {
112*10465441SEvalZero struct stat buf;
113*10465441SEvalZero static char fullname[PATH_MAX+1];
114*10465441SEvalZero char *env, *name;
115*10465441SEvalZero
116*10465441SEvalZero name = conf_expand_value(conf_defname);
117*10465441SEvalZero env = getenv(SRCTREE);
118*10465441SEvalZero if (env) {
119*10465441SEvalZero sprintf(fullname, "%s/%s", env, name);
120*10465441SEvalZero if (!stat(fullname, &buf))
121*10465441SEvalZero return fullname;
122*10465441SEvalZero }
123*10465441SEvalZero return name;
124*10465441SEvalZero }
125*10465441SEvalZero
conf_set_sym_val(struct symbol * sym,int def,int def_flags,char * p)126*10465441SEvalZero static int conf_set_sym_val(struct symbol *sym, int def, int def_flags, char *p)
127*10465441SEvalZero {
128*10465441SEvalZero char *p2;
129*10465441SEvalZero
130*10465441SEvalZero switch (sym->type) {
131*10465441SEvalZero case S_TRISTATE:
132*10465441SEvalZero if (p[0] == 'm') {
133*10465441SEvalZero sym->def[def].tri = mod;
134*10465441SEvalZero sym->flags |= def_flags;
135*10465441SEvalZero break;
136*10465441SEvalZero }
137*10465441SEvalZero /* fall through */
138*10465441SEvalZero case S_BOOLEAN:
139*10465441SEvalZero if (p[0] == 'y') {
140*10465441SEvalZero sym->def[def].tri = yes;
141*10465441SEvalZero sym->flags |= def_flags;
142*10465441SEvalZero break;
143*10465441SEvalZero }
144*10465441SEvalZero if (p[0] == 'n') {
145*10465441SEvalZero sym->def[def].tri = no;
146*10465441SEvalZero sym->flags |= def_flags;
147*10465441SEvalZero break;
148*10465441SEvalZero }
149*10465441SEvalZero if (def != S_DEF_AUTO)
150*10465441SEvalZero conf_warning("symbol value '%s' invalid for %s",
151*10465441SEvalZero p, sym->name);
152*10465441SEvalZero return 1;
153*10465441SEvalZero case S_OTHER:
154*10465441SEvalZero if (*p != '"') {
155*10465441SEvalZero for (p2 = p; *p2 && !isspace(*p2); p2++)
156*10465441SEvalZero ;
157*10465441SEvalZero sym->type = S_STRING;
158*10465441SEvalZero goto done;
159*10465441SEvalZero }
160*10465441SEvalZero /* fall through */
161*10465441SEvalZero case S_STRING:
162*10465441SEvalZero if (*p++ != '"')
163*10465441SEvalZero break;
164*10465441SEvalZero for (p2 = p; (p2 = strpbrk(p2, "\"\\")); p2++) {
165*10465441SEvalZero if (*p2 == '"') {
166*10465441SEvalZero *p2 = 0;
167*10465441SEvalZero break;
168*10465441SEvalZero }
169*10465441SEvalZero memmove(p2, p2 + 1, strlen(p2));
170*10465441SEvalZero }
171*10465441SEvalZero if (!p2) {
172*10465441SEvalZero if (def != S_DEF_AUTO)
173*10465441SEvalZero conf_warning("invalid string found");
174*10465441SEvalZero return 1;
175*10465441SEvalZero }
176*10465441SEvalZero /* fall through */
177*10465441SEvalZero case S_INT:
178*10465441SEvalZero case S_HEX:
179*10465441SEvalZero done:
180*10465441SEvalZero if (sym_string_valid(sym, p)) {
181*10465441SEvalZero sym->def[def].val = strdup(p);
182*10465441SEvalZero sym->flags |= def_flags;
183*10465441SEvalZero } else {
184*10465441SEvalZero if (def != S_DEF_AUTO)
185*10465441SEvalZero conf_warning("symbol value '%s' invalid for %s",
186*10465441SEvalZero p, sym->name);
187*10465441SEvalZero return 1;
188*10465441SEvalZero }
189*10465441SEvalZero break;
190*10465441SEvalZero default:
191*10465441SEvalZero ;
192*10465441SEvalZero }
193*10465441SEvalZero return 0;
194*10465441SEvalZero }
195*10465441SEvalZero
196*10465441SEvalZero #define LINE_GROWTH 16
add_byte(int c,char ** lineptr,size_t slen,size_t * n)197*10465441SEvalZero static int add_byte(int c, char **lineptr, size_t slen, size_t *n)
198*10465441SEvalZero {
199*10465441SEvalZero char *nline;
200*10465441SEvalZero size_t new_size = slen + 1;
201*10465441SEvalZero if (new_size > *n) {
202*10465441SEvalZero new_size += LINE_GROWTH - 1;
203*10465441SEvalZero new_size *= 2;
204*10465441SEvalZero nline = realloc(*lineptr, new_size);
205*10465441SEvalZero if (!nline)
206*10465441SEvalZero return -1;
207*10465441SEvalZero
208*10465441SEvalZero *lineptr = nline;
209*10465441SEvalZero *n = new_size;
210*10465441SEvalZero }
211*10465441SEvalZero
212*10465441SEvalZero (*lineptr)[slen] = c;
213*10465441SEvalZero
214*10465441SEvalZero return 0;
215*10465441SEvalZero }
216*10465441SEvalZero
compat_getline(char ** lineptr,size_t * n,FILE * stream)217*10465441SEvalZero static ssize_t compat_getline(char **lineptr, size_t *n, FILE *stream)
218*10465441SEvalZero {
219*10465441SEvalZero char *line = *lineptr;
220*10465441SEvalZero size_t slen = 0;
221*10465441SEvalZero
222*10465441SEvalZero for (;;) {
223*10465441SEvalZero int c = getc(stream);
224*10465441SEvalZero
225*10465441SEvalZero switch (c) {
226*10465441SEvalZero case '\n':
227*10465441SEvalZero if (add_byte(c, &line, slen, n) < 0)
228*10465441SEvalZero goto e_out;
229*10465441SEvalZero slen++;
230*10465441SEvalZero /* fall through */
231*10465441SEvalZero case EOF:
232*10465441SEvalZero if (add_byte('\0', &line, slen, n) < 0)
233*10465441SEvalZero goto e_out;
234*10465441SEvalZero *lineptr = line;
235*10465441SEvalZero if (slen == 0)
236*10465441SEvalZero return -1;
237*10465441SEvalZero return slen;
238*10465441SEvalZero default:
239*10465441SEvalZero if (add_byte(c, &line, slen, n) < 0)
240*10465441SEvalZero goto e_out;
241*10465441SEvalZero slen++;
242*10465441SEvalZero }
243*10465441SEvalZero }
244*10465441SEvalZero
245*10465441SEvalZero e_out:
246*10465441SEvalZero line[slen-1] = '\0';
247*10465441SEvalZero *lineptr = line;
248*10465441SEvalZero return -1;
249*10465441SEvalZero }
250*10465441SEvalZero
conf_read_simple(const char * name,int def)251*10465441SEvalZero int conf_read_simple(const char *name, int def)
252*10465441SEvalZero {
253*10465441SEvalZero FILE *in = NULL;
254*10465441SEvalZero char *line = NULL;
255*10465441SEvalZero size_t line_asize = 0;
256*10465441SEvalZero char *p, *p2;
257*10465441SEvalZero struct symbol *sym;
258*10465441SEvalZero int i, def_flags;
259*10465441SEvalZero
260*10465441SEvalZero if (name) {
261*10465441SEvalZero in = zconf_fopen(name);
262*10465441SEvalZero } else {
263*10465441SEvalZero struct property *prop;
264*10465441SEvalZero
265*10465441SEvalZero name = conf_get_configname();
266*10465441SEvalZero in = zconf_fopen(name);
267*10465441SEvalZero if (in)
268*10465441SEvalZero goto load;
269*10465441SEvalZero sym_add_change_count(1);
270*10465441SEvalZero if (!sym_defconfig_list)
271*10465441SEvalZero return 1;
272*10465441SEvalZero
273*10465441SEvalZero for_all_defaults(sym_defconfig_list, prop) {
274*10465441SEvalZero if (expr_calc_value(prop->visible.expr) == no ||
275*10465441SEvalZero prop->expr->type != E_SYMBOL)
276*10465441SEvalZero continue;
277*10465441SEvalZero name = conf_expand_value(prop->expr->left.sym->name);
278*10465441SEvalZero in = zconf_fopen(name);
279*10465441SEvalZero if (in) {
280*10465441SEvalZero conf_message(_("using defaults found in %s"),
281*10465441SEvalZero name);
282*10465441SEvalZero goto load;
283*10465441SEvalZero }
284*10465441SEvalZero }
285*10465441SEvalZero }
286*10465441SEvalZero if (!in)
287*10465441SEvalZero return 1;
288*10465441SEvalZero
289*10465441SEvalZero load:
290*10465441SEvalZero conf_filename = name;
291*10465441SEvalZero conf_lineno = 0;
292*10465441SEvalZero conf_warnings = 0;
293*10465441SEvalZero conf_unsaved = 0;
294*10465441SEvalZero
295*10465441SEvalZero def_flags = SYMBOL_DEF << def;
296*10465441SEvalZero for_all_symbols(i, sym) {
297*10465441SEvalZero sym->flags |= SYMBOL_CHANGED;
298*10465441SEvalZero sym->flags &= ~(def_flags|SYMBOL_VALID);
299*10465441SEvalZero if (sym_is_choice(sym))
300*10465441SEvalZero sym->flags |= def_flags;
301*10465441SEvalZero switch (sym->type) {
302*10465441SEvalZero case S_INT:
303*10465441SEvalZero case S_HEX:
304*10465441SEvalZero case S_STRING:
305*10465441SEvalZero if (sym->def[def].val)
306*10465441SEvalZero free(sym->def[def].val);
307*10465441SEvalZero /* fall through */
308*10465441SEvalZero default:
309*10465441SEvalZero sym->def[def].val = NULL;
310*10465441SEvalZero sym->def[def].tri = no;
311*10465441SEvalZero }
312*10465441SEvalZero }
313*10465441SEvalZero
314*10465441SEvalZero while (compat_getline(&line, &line_asize, in) != -1) {
315*10465441SEvalZero conf_lineno++;
316*10465441SEvalZero sym = NULL;
317*10465441SEvalZero if (line[0] == '#') {
318*10465441SEvalZero if (memcmp(line + 2, CONFIG_, strlen(CONFIG_)))
319*10465441SEvalZero continue;
320*10465441SEvalZero p = strchr(line + 2 + strlen(CONFIG_), ' ');
321*10465441SEvalZero if (!p)
322*10465441SEvalZero continue;
323*10465441SEvalZero *p++ = 0;
324*10465441SEvalZero if (strncmp(p, "is not set", 10))
325*10465441SEvalZero continue;
326*10465441SEvalZero if (def == S_DEF_USER) {
327*10465441SEvalZero sym = sym_find(line + 2 + strlen(CONFIG_));
328*10465441SEvalZero if (!sym) {
329*10465441SEvalZero sym_add_change_count(1);
330*10465441SEvalZero goto setsym;
331*10465441SEvalZero }
332*10465441SEvalZero } else {
333*10465441SEvalZero sym = sym_lookup(line + 2 + strlen(CONFIG_), 0);
334*10465441SEvalZero if (sym->type == S_UNKNOWN)
335*10465441SEvalZero sym->type = S_BOOLEAN;
336*10465441SEvalZero }
337*10465441SEvalZero if (sym->flags & def_flags) {
338*10465441SEvalZero conf_warning("override: reassigning to symbol %s", sym->name);
339*10465441SEvalZero }
340*10465441SEvalZero switch (sym->type) {
341*10465441SEvalZero case S_BOOLEAN:
342*10465441SEvalZero case S_TRISTATE:
343*10465441SEvalZero sym->def[def].tri = no;
344*10465441SEvalZero sym->flags |= def_flags;
345*10465441SEvalZero break;
346*10465441SEvalZero default:
347*10465441SEvalZero ;
348*10465441SEvalZero }
349*10465441SEvalZero } else if (memcmp(line, CONFIG_, strlen(CONFIG_)) == 0) {
350*10465441SEvalZero p = strchr(line + strlen(CONFIG_), '=');
351*10465441SEvalZero if (!p)
352*10465441SEvalZero continue;
353*10465441SEvalZero *p++ = 0;
354*10465441SEvalZero p2 = strchr(p, '\n');
355*10465441SEvalZero if (p2) {
356*10465441SEvalZero *p2-- = 0;
357*10465441SEvalZero if (*p2 == '\r')
358*10465441SEvalZero *p2 = 0;
359*10465441SEvalZero }
360*10465441SEvalZero if (def == S_DEF_USER) {
361*10465441SEvalZero sym = sym_find(line + strlen(CONFIG_));
362*10465441SEvalZero if (!sym) {
363*10465441SEvalZero sym_add_change_count(1);
364*10465441SEvalZero goto setsym;
365*10465441SEvalZero }
366*10465441SEvalZero } else {
367*10465441SEvalZero sym = sym_lookup(line + strlen(CONFIG_), 0);
368*10465441SEvalZero if (sym->type == S_UNKNOWN)
369*10465441SEvalZero sym->type = S_OTHER;
370*10465441SEvalZero }
371*10465441SEvalZero if (sym->flags & def_flags) {
372*10465441SEvalZero conf_warning("override: reassigning to symbol %s", sym->name);
373*10465441SEvalZero }
374*10465441SEvalZero if (conf_set_sym_val(sym, def, def_flags, p))
375*10465441SEvalZero continue;
376*10465441SEvalZero } else {
377*10465441SEvalZero if (line[0] != '\r' && line[0] != '\n')
378*10465441SEvalZero conf_warning("unexpected data: %.*s",
379*10465441SEvalZero (int)strcspn(line, "\r\n"), line);
380*10465441SEvalZero
381*10465441SEvalZero continue;
382*10465441SEvalZero }
383*10465441SEvalZero setsym:
384*10465441SEvalZero if (sym && sym_is_choice_value(sym)) {
385*10465441SEvalZero struct symbol *cs = prop_get_symbol(sym_get_choice_prop(sym));
386*10465441SEvalZero switch (sym->def[def].tri) {
387*10465441SEvalZero case no:
388*10465441SEvalZero break;
389*10465441SEvalZero case mod:
390*10465441SEvalZero if (cs->def[def].tri == yes) {
391*10465441SEvalZero conf_warning("%s creates inconsistent choice state", sym->name);
392*10465441SEvalZero cs->flags &= ~def_flags;
393*10465441SEvalZero }
394*10465441SEvalZero break;
395*10465441SEvalZero case yes:
396*10465441SEvalZero if (cs->def[def].tri != no)
397*10465441SEvalZero conf_warning("override: %s changes choice state", sym->name);
398*10465441SEvalZero cs->def[def].val = sym;
399*10465441SEvalZero break;
400*10465441SEvalZero }
401*10465441SEvalZero cs->def[def].tri = EXPR_OR(cs->def[def].tri, sym->def[def].tri);
402*10465441SEvalZero }
403*10465441SEvalZero }
404*10465441SEvalZero free(line);
405*10465441SEvalZero fclose(in);
406*10465441SEvalZero return 0;
407*10465441SEvalZero }
408*10465441SEvalZero
conf_read(const char * name)409*10465441SEvalZero int conf_read(const char *name)
410*10465441SEvalZero {
411*10465441SEvalZero struct symbol *sym;
412*10465441SEvalZero int i;
413*10465441SEvalZero
414*10465441SEvalZero sym_set_change_count(0);
415*10465441SEvalZero
416*10465441SEvalZero if (conf_read_simple(name, S_DEF_USER)) {
417*10465441SEvalZero sym_calc_value(modules_sym);
418*10465441SEvalZero return 1;
419*10465441SEvalZero }
420*10465441SEvalZero
421*10465441SEvalZero sym_calc_value(modules_sym);
422*10465441SEvalZero
423*10465441SEvalZero for_all_symbols(i, sym) {
424*10465441SEvalZero sym_calc_value(sym);
425*10465441SEvalZero if (sym_is_choice(sym) || (sym->flags & SYMBOL_AUTO))
426*10465441SEvalZero continue;
427*10465441SEvalZero if (sym_has_value(sym) && (sym->flags & SYMBOL_WRITE)) {
428*10465441SEvalZero /* check that calculated value agrees with saved value */
429*10465441SEvalZero switch (sym->type) {
430*10465441SEvalZero case S_BOOLEAN:
431*10465441SEvalZero case S_TRISTATE:
432*10465441SEvalZero if (sym->def[S_DEF_USER].tri != sym_get_tristate_value(sym))
433*10465441SEvalZero break;
434*10465441SEvalZero if (!sym_is_choice(sym))
435*10465441SEvalZero continue;
436*10465441SEvalZero /* fall through */
437*10465441SEvalZero default:
438*10465441SEvalZero if (!strcmp(sym->curr.val, sym->def[S_DEF_USER].val))
439*10465441SEvalZero continue;
440*10465441SEvalZero break;
441*10465441SEvalZero }
442*10465441SEvalZero } else if (!sym_has_value(sym) && !(sym->flags & SYMBOL_WRITE))
443*10465441SEvalZero /* no previous value and not saved */
444*10465441SEvalZero continue;
445*10465441SEvalZero conf_unsaved++;
446*10465441SEvalZero /* maybe print value in verbose mode... */
447*10465441SEvalZero }
448*10465441SEvalZero
449*10465441SEvalZero for_all_symbols(i, sym) {
450*10465441SEvalZero if (sym_has_value(sym) && !sym_is_choice_value(sym)) {
451*10465441SEvalZero /* Reset values of generates values, so they'll appear
452*10465441SEvalZero * as new, if they should become visible, but that
453*10465441SEvalZero * doesn't quite work if the Kconfig and the saved
454*10465441SEvalZero * configuration disagree.
455*10465441SEvalZero */
456*10465441SEvalZero if (sym->visible == no && !conf_unsaved)
457*10465441SEvalZero sym->flags &= ~SYMBOL_DEF_USER;
458*10465441SEvalZero switch (sym->type) {
459*10465441SEvalZero case S_STRING:
460*10465441SEvalZero case S_INT:
461*10465441SEvalZero case S_HEX:
462*10465441SEvalZero /* Reset a string value if it's out of range */
463*10465441SEvalZero if (sym_string_within_range(sym, sym->def[S_DEF_USER].val))
464*10465441SEvalZero break;
465*10465441SEvalZero sym->flags &= ~(SYMBOL_VALID|SYMBOL_DEF_USER);
466*10465441SEvalZero conf_unsaved++;
467*10465441SEvalZero break;
468*10465441SEvalZero default:
469*10465441SEvalZero break;
470*10465441SEvalZero }
471*10465441SEvalZero }
472*10465441SEvalZero }
473*10465441SEvalZero
474*10465441SEvalZero sym_add_change_count(conf_warnings || conf_unsaved);
475*10465441SEvalZero
476*10465441SEvalZero return 0;
477*10465441SEvalZero }
478*10465441SEvalZero
479*10465441SEvalZero /*
480*10465441SEvalZero * Kconfig configuration printer
481*10465441SEvalZero *
482*10465441SEvalZero * This printer is used when generating the resulting configuration after
483*10465441SEvalZero * kconfig invocation and `defconfig' files. Unset symbol might be omitted by
484*10465441SEvalZero * passing a non-NULL argument to the printer.
485*10465441SEvalZero *
486*10465441SEvalZero */
487*10465441SEvalZero static void
kconfig_print_symbol(FILE * fp,struct symbol * sym,const char * value,void * arg)488*10465441SEvalZero kconfig_print_symbol(FILE *fp, struct symbol *sym, const char *value, void *arg)
489*10465441SEvalZero {
490*10465441SEvalZero
491*10465441SEvalZero switch (sym->type) {
492*10465441SEvalZero case S_BOOLEAN:
493*10465441SEvalZero case S_TRISTATE:
494*10465441SEvalZero if (*value == 'n') {
495*10465441SEvalZero bool skip_unset = (arg != NULL);
496*10465441SEvalZero
497*10465441SEvalZero if (!skip_unset)
498*10465441SEvalZero fprintf(fp, "# %s%s is not set\n",
499*10465441SEvalZero CONFIG_, sym->name);
500*10465441SEvalZero return;
501*10465441SEvalZero }
502*10465441SEvalZero break;
503*10465441SEvalZero default:
504*10465441SEvalZero break;
505*10465441SEvalZero }
506*10465441SEvalZero
507*10465441SEvalZero fprintf(fp, "%s%s=%s\n", CONFIG_, sym->name, value);
508*10465441SEvalZero }
509*10465441SEvalZero
510*10465441SEvalZero static void
kconfig_print_comment(FILE * fp,const char * value,void * arg)511*10465441SEvalZero kconfig_print_comment(FILE *fp, const char *value, void *arg)
512*10465441SEvalZero {
513*10465441SEvalZero const char *p = value;
514*10465441SEvalZero size_t l;
515*10465441SEvalZero
516*10465441SEvalZero for (;;) {
517*10465441SEvalZero l = strcspn(p, "\n");
518*10465441SEvalZero fprintf(fp, "#");
519*10465441SEvalZero if (l) {
520*10465441SEvalZero fprintf(fp, " ");
521*10465441SEvalZero xfwrite(p, l, 1, fp);
522*10465441SEvalZero p += l;
523*10465441SEvalZero }
524*10465441SEvalZero fprintf(fp, "\n");
525*10465441SEvalZero if (*p++ == '\0')
526*10465441SEvalZero break;
527*10465441SEvalZero }
528*10465441SEvalZero }
529*10465441SEvalZero
530*10465441SEvalZero static struct conf_printer kconfig_printer_cb =
531*10465441SEvalZero {
532*10465441SEvalZero .print_symbol = kconfig_print_symbol,
533*10465441SEvalZero .print_comment = kconfig_print_comment,
534*10465441SEvalZero };
535*10465441SEvalZero
536*10465441SEvalZero /*
537*10465441SEvalZero * Header printer
538*10465441SEvalZero *
539*10465441SEvalZero * This printer is used when generating the `include/generated/autoconf.h' file.
540*10465441SEvalZero */
541*10465441SEvalZero static void
header_print_symbol(FILE * fp,struct symbol * sym,const char * value,void * arg)542*10465441SEvalZero header_print_symbol(FILE *fp, struct symbol *sym, const char *value, void *arg)
543*10465441SEvalZero {
544*10465441SEvalZero
545*10465441SEvalZero switch (sym->type) {
546*10465441SEvalZero case S_BOOLEAN:
547*10465441SEvalZero case S_TRISTATE: {
548*10465441SEvalZero const char *suffix = "";
549*10465441SEvalZero
550*10465441SEvalZero switch (*value) {
551*10465441SEvalZero case 'n':
552*10465441SEvalZero break;
553*10465441SEvalZero case 'm':
554*10465441SEvalZero suffix = "_MODULE";
555*10465441SEvalZero /* fall through */
556*10465441SEvalZero default:
557*10465441SEvalZero fprintf(fp, "#define %s%s%s 1\n",
558*10465441SEvalZero CONFIG_, sym->name, suffix);
559*10465441SEvalZero }
560*10465441SEvalZero break;
561*10465441SEvalZero }
562*10465441SEvalZero case S_HEX: {
563*10465441SEvalZero const char *prefix = "";
564*10465441SEvalZero
565*10465441SEvalZero if (value[0] != '0' || (value[1] != 'x' && value[1] != 'X'))
566*10465441SEvalZero prefix = "0x";
567*10465441SEvalZero fprintf(fp, "#define %s%s %s%s\n",
568*10465441SEvalZero CONFIG_, sym->name, prefix, value);
569*10465441SEvalZero break;
570*10465441SEvalZero }
571*10465441SEvalZero case S_STRING:
572*10465441SEvalZero case S_INT:
573*10465441SEvalZero fprintf(fp, "#define %s%s %s\n",
574*10465441SEvalZero CONFIG_, sym->name, value);
575*10465441SEvalZero break;
576*10465441SEvalZero default:
577*10465441SEvalZero break;
578*10465441SEvalZero }
579*10465441SEvalZero
580*10465441SEvalZero }
581*10465441SEvalZero
582*10465441SEvalZero static void
header_print_comment(FILE * fp,const char * value,void * arg)583*10465441SEvalZero header_print_comment(FILE *fp, const char *value, void *arg)
584*10465441SEvalZero {
585*10465441SEvalZero const char *p = value;
586*10465441SEvalZero size_t l;
587*10465441SEvalZero
588*10465441SEvalZero fprintf(fp, "/*\n");
589*10465441SEvalZero for (;;) {
590*10465441SEvalZero l = strcspn(p, "\n");
591*10465441SEvalZero fprintf(fp, " *");
592*10465441SEvalZero if (l) {
593*10465441SEvalZero fprintf(fp, " ");
594*10465441SEvalZero xfwrite(p, l, 1, fp);
595*10465441SEvalZero p += l;
596*10465441SEvalZero }
597*10465441SEvalZero fprintf(fp, "\n");
598*10465441SEvalZero if (*p++ == '\0')
599*10465441SEvalZero break;
600*10465441SEvalZero }
601*10465441SEvalZero fprintf(fp, " */\n");
602*10465441SEvalZero }
603*10465441SEvalZero
604*10465441SEvalZero static struct conf_printer header_printer_cb =
605*10465441SEvalZero {
606*10465441SEvalZero .print_symbol = header_print_symbol,
607*10465441SEvalZero .print_comment = header_print_comment,
608*10465441SEvalZero };
609*10465441SEvalZero
610*10465441SEvalZero /*
611*10465441SEvalZero * Tristate printer
612*10465441SEvalZero *
613*10465441SEvalZero * This printer is used when generating the `include/config/tristate.conf' file.
614*10465441SEvalZero */
615*10465441SEvalZero static void
tristate_print_symbol(FILE * fp,struct symbol * sym,const char * value,void * arg)616*10465441SEvalZero tristate_print_symbol(FILE *fp, struct symbol *sym, const char *value, void *arg)
617*10465441SEvalZero {
618*10465441SEvalZero
619*10465441SEvalZero if (sym->type == S_TRISTATE && *value != 'n')
620*10465441SEvalZero fprintf(fp, "%s%s=%c\n", CONFIG_, sym->name, (char)toupper(*value));
621*10465441SEvalZero }
622*10465441SEvalZero
623*10465441SEvalZero static struct conf_printer tristate_printer_cb =
624*10465441SEvalZero {
625*10465441SEvalZero .print_symbol = tristate_print_symbol,
626*10465441SEvalZero .print_comment = kconfig_print_comment,
627*10465441SEvalZero };
628*10465441SEvalZero
conf_write_symbol(FILE * fp,struct symbol * sym,struct conf_printer * printer,void * printer_arg)629*10465441SEvalZero static void conf_write_symbol(FILE *fp, struct symbol *sym,
630*10465441SEvalZero struct conf_printer *printer, void *printer_arg)
631*10465441SEvalZero {
632*10465441SEvalZero const char *str;
633*10465441SEvalZero
634*10465441SEvalZero switch (sym->type) {
635*10465441SEvalZero case S_OTHER:
636*10465441SEvalZero case S_UNKNOWN:
637*10465441SEvalZero break;
638*10465441SEvalZero case S_STRING:
639*10465441SEvalZero str = sym_get_string_value(sym);
640*10465441SEvalZero str = sym_escape_string_value(str);
641*10465441SEvalZero printer->print_symbol(fp, sym, str, printer_arg);
642*10465441SEvalZero free((void *)str);
643*10465441SEvalZero break;
644*10465441SEvalZero default:
645*10465441SEvalZero str = sym_get_string_value(sym);
646*10465441SEvalZero printer->print_symbol(fp, sym, str, printer_arg);
647*10465441SEvalZero }
648*10465441SEvalZero }
649*10465441SEvalZero
650*10465441SEvalZero static void
conf_write_heading(FILE * fp,struct conf_printer * printer,void * printer_arg)651*10465441SEvalZero conf_write_heading(FILE *fp, struct conf_printer *printer, void *printer_arg)
652*10465441SEvalZero {
653*10465441SEvalZero char buf[256];
654*10465441SEvalZero
655*10465441SEvalZero snprintf(buf, sizeof(buf),
656*10465441SEvalZero "\n"
657*10465441SEvalZero "Automatically generated file; DO NOT EDIT.\n"
658*10465441SEvalZero "%s\n",
659*10465441SEvalZero rootmenu.prompt->text);
660*10465441SEvalZero
661*10465441SEvalZero printer->print_comment(fp, buf, printer_arg);
662*10465441SEvalZero }
663*10465441SEvalZero
664*10465441SEvalZero /*
665*10465441SEvalZero * Write out a minimal config.
666*10465441SEvalZero * All values that has default values are skipped as this is redundant.
667*10465441SEvalZero */
conf_write_defconfig(const char * filename)668*10465441SEvalZero int conf_write_defconfig(const char *filename)
669*10465441SEvalZero {
670*10465441SEvalZero struct symbol *sym;
671*10465441SEvalZero struct menu *menu;
672*10465441SEvalZero FILE *out;
673*10465441SEvalZero
674*10465441SEvalZero out = fopen(filename, "w");
675*10465441SEvalZero if (!out)
676*10465441SEvalZero return 1;
677*10465441SEvalZero
678*10465441SEvalZero sym_clear_all_valid();
679*10465441SEvalZero
680*10465441SEvalZero /* Traverse all menus to find all relevant symbols */
681*10465441SEvalZero menu = rootmenu.list;
682*10465441SEvalZero
683*10465441SEvalZero while (menu != NULL)
684*10465441SEvalZero {
685*10465441SEvalZero sym = menu->sym;
686*10465441SEvalZero if (sym == NULL) {
687*10465441SEvalZero if (!menu_is_visible(menu))
688*10465441SEvalZero goto next_menu;
689*10465441SEvalZero } else if (!sym_is_choice(sym)) {
690*10465441SEvalZero sym_calc_value(sym);
691*10465441SEvalZero if (!(sym->flags & SYMBOL_WRITE))
692*10465441SEvalZero goto next_menu;
693*10465441SEvalZero sym->flags &= ~SYMBOL_WRITE;
694*10465441SEvalZero /* If we cannot change the symbol - skip */
695*10465441SEvalZero if (!sym_is_changable(sym))
696*10465441SEvalZero goto next_menu;
697*10465441SEvalZero /* If symbol equals to default value - skip */
698*10465441SEvalZero if (strcmp(sym_get_string_value(sym), sym_get_string_default(sym)) == 0)
699*10465441SEvalZero goto next_menu;
700*10465441SEvalZero
701*10465441SEvalZero /*
702*10465441SEvalZero * If symbol is a choice value and equals to the
703*10465441SEvalZero * default for a choice - skip.
704*10465441SEvalZero * But only if value is bool and equal to "y" and
705*10465441SEvalZero * choice is not "optional".
706*10465441SEvalZero * (If choice is "optional" then all values can be "n")
707*10465441SEvalZero */
708*10465441SEvalZero if (sym_is_choice_value(sym)) {
709*10465441SEvalZero struct symbol *cs;
710*10465441SEvalZero struct symbol *ds;
711*10465441SEvalZero
712*10465441SEvalZero cs = prop_get_symbol(sym_get_choice_prop(sym));
713*10465441SEvalZero ds = sym_choice_default(cs);
714*10465441SEvalZero if (!sym_is_optional(cs) && sym == ds) {
715*10465441SEvalZero if ((sym->type == S_BOOLEAN) &&
716*10465441SEvalZero sym_get_tristate_value(sym) == yes)
717*10465441SEvalZero goto next_menu;
718*10465441SEvalZero }
719*10465441SEvalZero }
720*10465441SEvalZero conf_write_symbol(out, sym, &kconfig_printer_cb, NULL);
721*10465441SEvalZero }
722*10465441SEvalZero next_menu:
723*10465441SEvalZero if (menu->list != NULL) {
724*10465441SEvalZero menu = menu->list;
725*10465441SEvalZero }
726*10465441SEvalZero else if (menu->next != NULL) {
727*10465441SEvalZero menu = menu->next;
728*10465441SEvalZero } else {
729*10465441SEvalZero while ((menu = menu->parent)) {
730*10465441SEvalZero if (menu->next != NULL) {
731*10465441SEvalZero menu = menu->next;
732*10465441SEvalZero break;
733*10465441SEvalZero }
734*10465441SEvalZero }
735*10465441SEvalZero }
736*10465441SEvalZero }
737*10465441SEvalZero fclose(out);
738*10465441SEvalZero return 0;
739*10465441SEvalZero }
740*10465441SEvalZero
conf_write(const char * name)741*10465441SEvalZero int conf_write(const char *name)
742*10465441SEvalZero {
743*10465441SEvalZero FILE *out;
744*10465441SEvalZero struct symbol *sym;
745*10465441SEvalZero struct menu *menu;
746*10465441SEvalZero const char *basename;
747*10465441SEvalZero const char *str;
748*10465441SEvalZero char dirname[PATH_MAX+1], tmpname[PATH_MAX+1], newname[PATH_MAX+1];
749*10465441SEvalZero char *env;
750*10465441SEvalZero
751*10465441SEvalZero dirname[0] = 0;
752*10465441SEvalZero if (name && name[0]) {
753*10465441SEvalZero struct stat st;
754*10465441SEvalZero char *slash;
755*10465441SEvalZero
756*10465441SEvalZero if (!stat(name, &st) && S_ISDIR(st.st_mode)) {
757*10465441SEvalZero strcpy(dirname, name);
758*10465441SEvalZero strcat(dirname, "/");
759*10465441SEvalZero basename = conf_get_configname();
760*10465441SEvalZero } else if ((slash = strrchr(name, '/'))) {
761*10465441SEvalZero int size = slash - name + 1;
762*10465441SEvalZero memcpy(dirname, name, size);
763*10465441SEvalZero dirname[size] = 0;
764*10465441SEvalZero if (slash[1])
765*10465441SEvalZero basename = slash + 1;
766*10465441SEvalZero else
767*10465441SEvalZero basename = conf_get_configname();
768*10465441SEvalZero } else
769*10465441SEvalZero basename = name;
770*10465441SEvalZero } else
771*10465441SEvalZero basename = conf_get_configname();
772*10465441SEvalZero
773*10465441SEvalZero sprintf(newname, "%s%s", dirname, basename);
774*10465441SEvalZero env = getenv("KCONFIG_OVERWRITECONFIG");
775*10465441SEvalZero if (!env || !*env) {
776*10465441SEvalZero sprintf(tmpname, "%s.tmpconfig.%d", dirname, (int)getpid());
777*10465441SEvalZero out = fopen(tmpname, "w");
778*10465441SEvalZero } else {
779*10465441SEvalZero *tmpname = 0;
780*10465441SEvalZero out = fopen(newname, "w");
781*10465441SEvalZero }
782*10465441SEvalZero if (!out)
783*10465441SEvalZero return 1;
784*10465441SEvalZero
785*10465441SEvalZero conf_write_heading(out, &kconfig_printer_cb, NULL);
786*10465441SEvalZero
787*10465441SEvalZero if (!conf_get_changed())
788*10465441SEvalZero sym_clear_all_valid();
789*10465441SEvalZero
790*10465441SEvalZero menu = rootmenu.list;
791*10465441SEvalZero while (menu) {
792*10465441SEvalZero sym = menu->sym;
793*10465441SEvalZero if (!sym) {
794*10465441SEvalZero if (!menu_is_visible(menu))
795*10465441SEvalZero goto next;
796*10465441SEvalZero str = menu_get_prompt(menu);
797*10465441SEvalZero fprintf(out, "\n"
798*10465441SEvalZero "#\n"
799*10465441SEvalZero "# %s\n"
800*10465441SEvalZero "#\n", str);
801*10465441SEvalZero } else if (!(sym->flags & SYMBOL_CHOICE)) {
802*10465441SEvalZero sym_calc_value(sym);
803*10465441SEvalZero if (!(sym->flags & SYMBOL_WRITE))
804*10465441SEvalZero goto next;
805*10465441SEvalZero sym->flags &= ~SYMBOL_WRITE;
806*10465441SEvalZero
807*10465441SEvalZero conf_write_symbol(out, sym, &kconfig_printer_cb, NULL);
808*10465441SEvalZero }
809*10465441SEvalZero
810*10465441SEvalZero next:
811*10465441SEvalZero if (menu->list) {
812*10465441SEvalZero menu = menu->list;
813*10465441SEvalZero continue;
814*10465441SEvalZero }
815*10465441SEvalZero if (menu->next)
816*10465441SEvalZero menu = menu->next;
817*10465441SEvalZero else while ((menu = menu->parent)) {
818*10465441SEvalZero if (menu->next) {
819*10465441SEvalZero menu = menu->next;
820*10465441SEvalZero break;
821*10465441SEvalZero }
822*10465441SEvalZero }
823*10465441SEvalZero }
824*10465441SEvalZero fclose(out);
825*10465441SEvalZero
826*10465441SEvalZero if (*tmpname) {
827*10465441SEvalZero strcat(dirname, basename);
828*10465441SEvalZero strcat(dirname, ".old");
829*10465441SEvalZero rename(newname, dirname);
830*10465441SEvalZero if (rename(tmpname, newname))
831*10465441SEvalZero return 1;
832*10465441SEvalZero }
833*10465441SEvalZero
834*10465441SEvalZero conf_message(_("configuration written to %s"), newname);
835*10465441SEvalZero
836*10465441SEvalZero sym_set_change_count(0);
837*10465441SEvalZero
838*10465441SEvalZero return 0;
839*10465441SEvalZero }
840*10465441SEvalZero
conf_split_config(void)841*10465441SEvalZero static int conf_split_config(void)
842*10465441SEvalZero {
843*10465441SEvalZero const char *name;
844*10465441SEvalZero char path[PATH_MAX+1];
845*10465441SEvalZero char *s, *d, c;
846*10465441SEvalZero struct symbol *sym;
847*10465441SEvalZero struct stat sb;
848*10465441SEvalZero int res, i, fd;
849*10465441SEvalZero
850*10465441SEvalZero name = conf_get_autoconfig_name();
851*10465441SEvalZero conf_read_simple(name, S_DEF_AUTO);
852*10465441SEvalZero sym_calc_value(modules_sym);
853*10465441SEvalZero
854*10465441SEvalZero if (chdir("include/config"))
855*10465441SEvalZero return 1;
856*10465441SEvalZero
857*10465441SEvalZero res = 0;
858*10465441SEvalZero for_all_symbols(i, sym) {
859*10465441SEvalZero sym_calc_value(sym);
860*10465441SEvalZero if ((sym->flags & SYMBOL_AUTO) || !sym->name)
861*10465441SEvalZero continue;
862*10465441SEvalZero if (sym->flags & SYMBOL_WRITE) {
863*10465441SEvalZero if (sym->flags & SYMBOL_DEF_AUTO) {
864*10465441SEvalZero /*
865*10465441SEvalZero * symbol has old and new value,
866*10465441SEvalZero * so compare them...
867*10465441SEvalZero */
868*10465441SEvalZero switch (sym->type) {
869*10465441SEvalZero case S_BOOLEAN:
870*10465441SEvalZero case S_TRISTATE:
871*10465441SEvalZero if (sym_get_tristate_value(sym) ==
872*10465441SEvalZero sym->def[S_DEF_AUTO].tri)
873*10465441SEvalZero continue;
874*10465441SEvalZero break;
875*10465441SEvalZero case S_STRING:
876*10465441SEvalZero case S_HEX:
877*10465441SEvalZero case S_INT:
878*10465441SEvalZero if (!strcmp(sym_get_string_value(sym),
879*10465441SEvalZero sym->def[S_DEF_AUTO].val))
880*10465441SEvalZero continue;
881*10465441SEvalZero break;
882*10465441SEvalZero default:
883*10465441SEvalZero break;
884*10465441SEvalZero }
885*10465441SEvalZero } else {
886*10465441SEvalZero /*
887*10465441SEvalZero * If there is no old value, only 'no' (unset)
888*10465441SEvalZero * is allowed as new value.
889*10465441SEvalZero */
890*10465441SEvalZero switch (sym->type) {
891*10465441SEvalZero case S_BOOLEAN:
892*10465441SEvalZero case S_TRISTATE:
893*10465441SEvalZero if (sym_get_tristate_value(sym) == no)
894*10465441SEvalZero continue;
895*10465441SEvalZero break;
896*10465441SEvalZero default:
897*10465441SEvalZero break;
898*10465441SEvalZero }
899*10465441SEvalZero }
900*10465441SEvalZero } else if (!(sym->flags & SYMBOL_DEF_AUTO))
901*10465441SEvalZero /* There is neither an old nor a new value. */
902*10465441SEvalZero continue;
903*10465441SEvalZero /* else
904*10465441SEvalZero * There is an old value, but no new value ('no' (unset)
905*10465441SEvalZero * isn't saved in auto.conf, so the old value is always
906*10465441SEvalZero * different from 'no').
907*10465441SEvalZero */
908*10465441SEvalZero
909*10465441SEvalZero /* Replace all '_' and append ".h" */
910*10465441SEvalZero s = sym->name;
911*10465441SEvalZero d = path;
912*10465441SEvalZero while ((c = *s++)) {
913*10465441SEvalZero c = tolower(c);
914*10465441SEvalZero *d++ = (c == '_') ? '/' : c;
915*10465441SEvalZero }
916*10465441SEvalZero strcpy(d, ".h");
917*10465441SEvalZero
918*10465441SEvalZero /* Assume directory path already exists. */
919*10465441SEvalZero fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, 0644);
920*10465441SEvalZero if (fd == -1) {
921*10465441SEvalZero if (errno != ENOENT) {
922*10465441SEvalZero res = 1;
923*10465441SEvalZero break;
924*10465441SEvalZero }
925*10465441SEvalZero /*
926*10465441SEvalZero * Create directory components,
927*10465441SEvalZero * unless they exist already.
928*10465441SEvalZero */
929*10465441SEvalZero d = path;
930*10465441SEvalZero while ((d = strchr(d, '/'))) {
931*10465441SEvalZero *d = 0;
932*10465441SEvalZero if (stat(path, &sb) && mkdir(path, 0755)) {
933*10465441SEvalZero res = 1;
934*10465441SEvalZero goto out;
935*10465441SEvalZero }
936*10465441SEvalZero *d++ = '/';
937*10465441SEvalZero }
938*10465441SEvalZero /* Try it again. */
939*10465441SEvalZero fd = open(path, O_WRONLY | O_CREAT | O_TRUNC, 0644);
940*10465441SEvalZero if (fd == -1) {
941*10465441SEvalZero res = 1;
942*10465441SEvalZero break;
943*10465441SEvalZero }
944*10465441SEvalZero }
945*10465441SEvalZero close(fd);
946*10465441SEvalZero }
947*10465441SEvalZero out:
948*10465441SEvalZero if (chdir("../.."))
949*10465441SEvalZero return 1;
950*10465441SEvalZero
951*10465441SEvalZero return res;
952*10465441SEvalZero }
953*10465441SEvalZero
conf_write_autoconf(void)954*10465441SEvalZero int conf_write_autoconf(void)
955*10465441SEvalZero {
956*10465441SEvalZero struct symbol *sym;
957*10465441SEvalZero const char *name;
958*10465441SEvalZero FILE *out, *tristate, *out_h;
959*10465441SEvalZero int i;
960*10465441SEvalZero
961*10465441SEvalZero sym_clear_all_valid();
962*10465441SEvalZero
963*10465441SEvalZero file_write_dep("include/config/auto.conf.cmd");
964*10465441SEvalZero
965*10465441SEvalZero if (conf_split_config())
966*10465441SEvalZero return 1;
967*10465441SEvalZero
968*10465441SEvalZero out = fopen(".tmpconfig", "w");
969*10465441SEvalZero if (!out)
970*10465441SEvalZero return 1;
971*10465441SEvalZero
972*10465441SEvalZero tristate = fopen(".tmpconfig_tristate", "w");
973*10465441SEvalZero if (!tristate) {
974*10465441SEvalZero fclose(out);
975*10465441SEvalZero return 1;
976*10465441SEvalZero }
977*10465441SEvalZero
978*10465441SEvalZero out_h = fopen(".tmpconfig.h", "w");
979*10465441SEvalZero if (!out_h) {
980*10465441SEvalZero fclose(out);
981*10465441SEvalZero fclose(tristate);
982*10465441SEvalZero return 1;
983*10465441SEvalZero }
984*10465441SEvalZero
985*10465441SEvalZero conf_write_heading(out, &kconfig_printer_cb, NULL);
986*10465441SEvalZero
987*10465441SEvalZero conf_write_heading(tristate, &tristate_printer_cb, NULL);
988*10465441SEvalZero
989*10465441SEvalZero conf_write_heading(out_h, &header_printer_cb, NULL);
990*10465441SEvalZero
991*10465441SEvalZero for_all_symbols(i, sym) {
992*10465441SEvalZero sym_calc_value(sym);
993*10465441SEvalZero if (!(sym->flags & SYMBOL_WRITE) || !sym->name)
994*10465441SEvalZero continue;
995*10465441SEvalZero
996*10465441SEvalZero /* write symbol to auto.conf, tristate and header files */
997*10465441SEvalZero conf_write_symbol(out, sym, &kconfig_printer_cb, (void *)1);
998*10465441SEvalZero
999*10465441SEvalZero conf_write_symbol(tristate, sym, &tristate_printer_cb, (void *)1);
1000*10465441SEvalZero
1001*10465441SEvalZero conf_write_symbol(out_h, sym, &header_printer_cb, NULL);
1002*10465441SEvalZero }
1003*10465441SEvalZero fclose(out);
1004*10465441SEvalZero fclose(tristate);
1005*10465441SEvalZero fclose(out_h);
1006*10465441SEvalZero
1007*10465441SEvalZero name = getenv("KCONFIG_AUTOHEADER");
1008*10465441SEvalZero if (!name)
1009*10465441SEvalZero name = "include/generated/autoconf.h";
1010*10465441SEvalZero if (rename(".tmpconfig.h", name))
1011*10465441SEvalZero return 1;
1012*10465441SEvalZero name = getenv("KCONFIG_TRISTATE");
1013*10465441SEvalZero if (!name)
1014*10465441SEvalZero name = "include/config/tristate.conf";
1015*10465441SEvalZero if (rename(".tmpconfig_tristate", name))
1016*10465441SEvalZero return 1;
1017*10465441SEvalZero name = conf_get_autoconfig_name();
1018*10465441SEvalZero /*
1019*10465441SEvalZero * This must be the last step, kbuild has a dependency on auto.conf
1020*10465441SEvalZero * and this marks the successful completion of the previous steps.
1021*10465441SEvalZero */
1022*10465441SEvalZero if (rename(".tmpconfig", name))
1023*10465441SEvalZero return 1;
1024*10465441SEvalZero
1025*10465441SEvalZero return 0;
1026*10465441SEvalZero }
1027*10465441SEvalZero
1028*10465441SEvalZero static int sym_change_count;
1029*10465441SEvalZero static void (*conf_changed_callback)(void);
1030*10465441SEvalZero
sym_set_change_count(int count)1031*10465441SEvalZero void sym_set_change_count(int count)
1032*10465441SEvalZero {
1033*10465441SEvalZero int _sym_change_count = sym_change_count;
1034*10465441SEvalZero sym_change_count = count;
1035*10465441SEvalZero if (conf_changed_callback &&
1036*10465441SEvalZero (bool)_sym_change_count != (bool)count)
1037*10465441SEvalZero conf_changed_callback();
1038*10465441SEvalZero }
1039*10465441SEvalZero
sym_add_change_count(int count)1040*10465441SEvalZero void sym_add_change_count(int count)
1041*10465441SEvalZero {
1042*10465441SEvalZero sym_set_change_count(count + sym_change_count);
1043*10465441SEvalZero }
1044*10465441SEvalZero
conf_get_changed(void)1045*10465441SEvalZero bool conf_get_changed(void)
1046*10465441SEvalZero {
1047*10465441SEvalZero return sym_change_count;
1048*10465441SEvalZero }
1049*10465441SEvalZero
conf_set_changed_callback(void (* fn)(void))1050*10465441SEvalZero void conf_set_changed_callback(void (*fn)(void))
1051*10465441SEvalZero {
1052*10465441SEvalZero conf_changed_callback = fn;
1053*10465441SEvalZero }
1054*10465441SEvalZero
randomize_choice_values(struct symbol * csym)1055*10465441SEvalZero static bool randomize_choice_values(struct symbol *csym)
1056*10465441SEvalZero {
1057*10465441SEvalZero struct property *prop;
1058*10465441SEvalZero struct symbol *sym;
1059*10465441SEvalZero struct expr *e;
1060*10465441SEvalZero int cnt, def;
1061*10465441SEvalZero
1062*10465441SEvalZero /*
1063*10465441SEvalZero * If choice is mod then we may have more items selected
1064*10465441SEvalZero * and if no then no-one.
1065*10465441SEvalZero * In both cases stop.
1066*10465441SEvalZero */
1067*10465441SEvalZero if (csym->curr.tri != yes)
1068*10465441SEvalZero return false;
1069*10465441SEvalZero
1070*10465441SEvalZero prop = sym_get_choice_prop(csym);
1071*10465441SEvalZero
1072*10465441SEvalZero /* count entries in choice block */
1073*10465441SEvalZero cnt = 0;
1074*10465441SEvalZero expr_list_for_each_sym(prop->expr, e, sym)
1075*10465441SEvalZero cnt++;
1076*10465441SEvalZero
1077*10465441SEvalZero /*
1078*10465441SEvalZero * find a random value and set it to yes,
1079*10465441SEvalZero * set the rest to no so we have only one set
1080*10465441SEvalZero */
1081*10465441SEvalZero def = (rand() % cnt);
1082*10465441SEvalZero
1083*10465441SEvalZero cnt = 0;
1084*10465441SEvalZero expr_list_for_each_sym(prop->expr, e, sym) {
1085*10465441SEvalZero if (def == cnt++) {
1086*10465441SEvalZero sym->def[S_DEF_USER].tri = yes;
1087*10465441SEvalZero csym->def[S_DEF_USER].val = sym;
1088*10465441SEvalZero }
1089*10465441SEvalZero else {
1090*10465441SEvalZero sym->def[S_DEF_USER].tri = no;
1091*10465441SEvalZero }
1092*10465441SEvalZero sym->flags |= SYMBOL_DEF_USER;
1093*10465441SEvalZero /* clear VALID to get value calculated */
1094*10465441SEvalZero sym->flags &= ~SYMBOL_VALID;
1095*10465441SEvalZero }
1096*10465441SEvalZero csym->flags |= SYMBOL_DEF_USER;
1097*10465441SEvalZero /* clear VALID to get value calculated */
1098*10465441SEvalZero csym->flags &= ~(SYMBOL_VALID);
1099*10465441SEvalZero
1100*10465441SEvalZero return true;
1101*10465441SEvalZero }
1102*10465441SEvalZero
set_all_choice_values(struct symbol * csym)1103*10465441SEvalZero void set_all_choice_values(struct symbol *csym)
1104*10465441SEvalZero {
1105*10465441SEvalZero struct property *prop;
1106*10465441SEvalZero struct symbol *sym;
1107*10465441SEvalZero struct expr *e;
1108*10465441SEvalZero
1109*10465441SEvalZero prop = sym_get_choice_prop(csym);
1110*10465441SEvalZero
1111*10465441SEvalZero /*
1112*10465441SEvalZero * Set all non-assinged choice values to no
1113*10465441SEvalZero */
1114*10465441SEvalZero expr_list_for_each_sym(prop->expr, e, sym) {
1115*10465441SEvalZero if (!sym_has_value(sym))
1116*10465441SEvalZero sym->def[S_DEF_USER].tri = no;
1117*10465441SEvalZero }
1118*10465441SEvalZero csym->flags |= SYMBOL_DEF_USER;
1119*10465441SEvalZero /* clear VALID to get value calculated */
1120*10465441SEvalZero csym->flags &= ~(SYMBOL_VALID | SYMBOL_NEED_SET_CHOICE_VALUES);
1121*10465441SEvalZero }
1122*10465441SEvalZero
conf_set_all_new_symbols(enum conf_def_mode mode)1123*10465441SEvalZero bool conf_set_all_new_symbols(enum conf_def_mode mode)
1124*10465441SEvalZero {
1125*10465441SEvalZero struct symbol *sym, *csym;
1126*10465441SEvalZero int i, cnt, pby, pty, ptm; /* pby: probability of boolean = y
1127*10465441SEvalZero * pty: probability of tristate = y
1128*10465441SEvalZero * ptm: probability of tristate = m
1129*10465441SEvalZero */
1130*10465441SEvalZero
1131*10465441SEvalZero pby = 50; pty = ptm = 33; /* can't go as the default in switch-case
1132*10465441SEvalZero * below, otherwise gcc whines about
1133*10465441SEvalZero * -Wmaybe-uninitialized */
1134*10465441SEvalZero if (mode == def_random) {
1135*10465441SEvalZero int n, p[3];
1136*10465441SEvalZero char *env = getenv("KCONFIG_PROBABILITY");
1137*10465441SEvalZero n = 0;
1138*10465441SEvalZero while( env && *env ) {
1139*10465441SEvalZero char *endp;
1140*10465441SEvalZero int tmp = strtol( env, &endp, 10 );
1141*10465441SEvalZero if( tmp >= 0 && tmp <= 100 ) {
1142*10465441SEvalZero p[n++] = tmp;
1143*10465441SEvalZero } else {
1144*10465441SEvalZero errno = ERANGE;
1145*10465441SEvalZero perror( "KCONFIG_PROBABILITY" );
1146*10465441SEvalZero exit( 1 );
1147*10465441SEvalZero }
1148*10465441SEvalZero env = (*endp == ':') ? endp+1 : endp;
1149*10465441SEvalZero if( n >=3 ) {
1150*10465441SEvalZero break;
1151*10465441SEvalZero }
1152*10465441SEvalZero }
1153*10465441SEvalZero switch( n ) {
1154*10465441SEvalZero case 1:
1155*10465441SEvalZero pby = p[0]; ptm = pby/2; pty = pby-ptm;
1156*10465441SEvalZero break;
1157*10465441SEvalZero case 2:
1158*10465441SEvalZero pty = p[0]; ptm = p[1]; pby = pty + ptm;
1159*10465441SEvalZero break;
1160*10465441SEvalZero case 3:
1161*10465441SEvalZero pby = p[0]; pty = p[1]; ptm = p[2];
1162*10465441SEvalZero break;
1163*10465441SEvalZero }
1164*10465441SEvalZero
1165*10465441SEvalZero if( pty+ptm > 100 ) {
1166*10465441SEvalZero errno = ERANGE;
1167*10465441SEvalZero perror( "KCONFIG_PROBABILITY" );
1168*10465441SEvalZero exit( 1 );
1169*10465441SEvalZero }
1170*10465441SEvalZero }
1171*10465441SEvalZero bool has_changed = false;
1172*10465441SEvalZero
1173*10465441SEvalZero for_all_symbols(i, sym) {
1174*10465441SEvalZero if (sym_has_value(sym) || (sym->flags & SYMBOL_VALID))
1175*10465441SEvalZero continue;
1176*10465441SEvalZero switch (sym_get_type(sym)) {
1177*10465441SEvalZero case S_BOOLEAN:
1178*10465441SEvalZero case S_TRISTATE:
1179*10465441SEvalZero has_changed = true;
1180*10465441SEvalZero switch (mode) {
1181*10465441SEvalZero case def_yes:
1182*10465441SEvalZero sym->def[S_DEF_USER].tri = yes;
1183*10465441SEvalZero break;
1184*10465441SEvalZero case def_mod:
1185*10465441SEvalZero sym->def[S_DEF_USER].tri = mod;
1186*10465441SEvalZero break;
1187*10465441SEvalZero case def_no:
1188*10465441SEvalZero if (sym->flags & SYMBOL_ALLNOCONFIG_Y)
1189*10465441SEvalZero sym->def[S_DEF_USER].tri = yes;
1190*10465441SEvalZero else
1191*10465441SEvalZero sym->def[S_DEF_USER].tri = no;
1192*10465441SEvalZero break;
1193*10465441SEvalZero case def_random:
1194*10465441SEvalZero sym->def[S_DEF_USER].tri = no;
1195*10465441SEvalZero cnt = rand() % 100;
1196*10465441SEvalZero if (sym->type == S_TRISTATE) {
1197*10465441SEvalZero if (cnt < pty)
1198*10465441SEvalZero sym->def[S_DEF_USER].tri = yes;
1199*10465441SEvalZero else if (cnt < (pty+ptm))
1200*10465441SEvalZero sym->def[S_DEF_USER].tri = mod;
1201*10465441SEvalZero } else if (cnt < pby)
1202*10465441SEvalZero sym->def[S_DEF_USER].tri = yes;
1203*10465441SEvalZero break;
1204*10465441SEvalZero default:
1205*10465441SEvalZero continue;
1206*10465441SEvalZero }
1207*10465441SEvalZero if (!(sym_is_choice(sym) && mode == def_random))
1208*10465441SEvalZero sym->flags |= SYMBOL_DEF_USER;
1209*10465441SEvalZero break;
1210*10465441SEvalZero default:
1211*10465441SEvalZero break;
1212*10465441SEvalZero }
1213*10465441SEvalZero
1214*10465441SEvalZero }
1215*10465441SEvalZero
1216*10465441SEvalZero sym_clear_all_valid();
1217*10465441SEvalZero
1218*10465441SEvalZero /*
1219*10465441SEvalZero * We have different type of choice blocks.
1220*10465441SEvalZero * If curr.tri equals to mod then we can select several
1221*10465441SEvalZero * choice symbols in one block.
1222*10465441SEvalZero * In this case we do nothing.
1223*10465441SEvalZero * If curr.tri equals yes then only one symbol can be
1224*10465441SEvalZero * selected in a choice block and we set it to yes,
1225*10465441SEvalZero * and the rest to no.
1226*10465441SEvalZero */
1227*10465441SEvalZero if (mode != def_random) {
1228*10465441SEvalZero for_all_symbols(i, csym) {
1229*10465441SEvalZero if ((sym_is_choice(csym) && !sym_has_value(csym)) ||
1230*10465441SEvalZero sym_is_choice_value(csym))
1231*10465441SEvalZero csym->flags |= SYMBOL_NEED_SET_CHOICE_VALUES;
1232*10465441SEvalZero }
1233*10465441SEvalZero }
1234*10465441SEvalZero
1235*10465441SEvalZero for_all_symbols(i, csym) {
1236*10465441SEvalZero if (sym_has_value(csym) || !sym_is_choice(csym))
1237*10465441SEvalZero continue;
1238*10465441SEvalZero
1239*10465441SEvalZero sym_calc_value(csym);
1240*10465441SEvalZero if (mode == def_random)
1241*10465441SEvalZero has_changed = randomize_choice_values(csym);
1242*10465441SEvalZero else {
1243*10465441SEvalZero set_all_choice_values(csym);
1244*10465441SEvalZero has_changed = true;
1245*10465441SEvalZero }
1246*10465441SEvalZero }
1247*10465441SEvalZero
1248*10465441SEvalZero return has_changed;
1249*10465441SEvalZero }
1250