xref: /nrf52832-nimble/rt-thread/tools/kconfig-frontends/libs/parser/confdata.c (revision 104654410c56c573564690304ae786df310c91fc)
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