1 | /* SPDX-License-Identifier: GPL-2.0 */ |
2 | /* |
3 | * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org> |
4 | */ |
5 | |
6 | #ifndef EXPR_H |
7 | #define EXPR_H |
8 | |
9 | #ifdef __cplusplus |
10 | extern "C" { |
11 | #endif |
12 | |
13 | #include <assert.h> |
14 | #include <stdio.h> |
15 | #ifndef __cplusplus |
16 | #include <stdbool.h> |
17 | #endif |
18 | |
19 | #include <list_types.h> |
20 | |
21 | typedef enum tristate { |
22 | no, mod, yes |
23 | } tristate; |
24 | |
25 | enum expr_type { |
26 | E_NONE, E_OR, E_AND, E_NOT, |
27 | E_EQUAL, E_UNEQUAL, E_LTH, E_LEQ, E_GTH, E_GEQ, |
28 | E_SYMBOL, E_RANGE |
29 | }; |
30 | |
31 | union expr_data { |
32 | struct expr * const expr; |
33 | struct symbol * const sym; |
34 | void *_initdata; |
35 | }; |
36 | |
37 | /** |
38 | * struct expr - expression |
39 | * |
40 | * @node: link node for the hash table |
41 | * @type: expressoin type |
42 | * @val: calculated tristate value |
43 | * @val_is_valid: indicate whether the value is valid |
44 | * @left: left node |
45 | * @right: right node |
46 | */ |
47 | struct expr { |
48 | struct hlist_node node; |
49 | enum expr_type type; |
50 | tristate val; |
51 | bool val_is_valid; |
52 | union expr_data left, right; |
53 | }; |
54 | |
55 | #define EXPR_OR(dep1, dep2) (((dep1)>(dep2))?(dep1):(dep2)) |
56 | #define EXPR_AND(dep1, dep2) (((dep1)<(dep2))?(dep1):(dep2)) |
57 | #define EXPR_NOT(dep) (2-(dep)) |
58 | |
59 | struct expr_value { |
60 | struct expr *expr; |
61 | tristate tri; |
62 | }; |
63 | |
64 | struct symbol_value { |
65 | void *val; |
66 | tristate tri; |
67 | }; |
68 | |
69 | enum symbol_type { |
70 | S_UNKNOWN, S_BOOLEAN, S_TRISTATE, S_INT, S_HEX, S_STRING |
71 | }; |
72 | |
73 | /* enum values are used as index to symbol.def[] */ |
74 | enum { |
75 | S_DEF_USER, /* main user value */ |
76 | S_DEF_AUTO, /* values read from auto.conf */ |
77 | S_DEF_DEF3, /* Reserved for UI usage */ |
78 | S_DEF_DEF4, /* Reserved for UI usage */ |
79 | S_DEF_COUNT |
80 | }; |
81 | |
82 | /* |
83 | * Represents a configuration symbol. |
84 | * |
85 | * Choices are represented as a special kind of symbol with null name. |
86 | * |
87 | * @choice_link: linked to menu::choice_members |
88 | */ |
89 | struct symbol { |
90 | /* link node for the hash table */ |
91 | struct hlist_node node; |
92 | |
93 | /* The name of the symbol, e.g. "FOO" for 'config FOO' */ |
94 | char *name; |
95 | |
96 | /* S_BOOLEAN, S_TRISTATE, ... */ |
97 | enum symbol_type type; |
98 | |
99 | /* |
100 | * The calculated value of the symbol. The SYMBOL_VALID bit is set in |
101 | * 'flags' when this is up to date. Note that this value might differ |
102 | * from the user value set in e.g. a .config file, due to visibility. |
103 | */ |
104 | struct symbol_value curr; |
105 | |
106 | /* |
107 | * Values for the symbol provided from outside. def[S_DEF_USER] holds |
108 | * the .config value. |
109 | */ |
110 | struct symbol_value def[S_DEF_COUNT]; |
111 | |
112 | /* |
113 | * An upper bound on the tristate value the user can set for the symbol |
114 | * if it is a boolean or tristate. Calculated from prompt dependencies, |
115 | * which also inherit dependencies from enclosing menus, choices, and |
116 | * ifs. If 'n', the user value will be ignored. |
117 | * |
118 | * Symbols lacking prompts always have visibility 'n'. |
119 | */ |
120 | tristate visible; |
121 | |
122 | /* config entries associated with this symbol */ |
123 | struct list_head ; |
124 | |
125 | struct list_head choice_link; |
126 | |
127 | /* SYMBOL_* flags */ |
128 | int flags; |
129 | |
130 | /* List of properties. See prop_type. */ |
131 | struct property *prop; |
132 | |
133 | /* Dependencies from enclosing menus, choices, and ifs */ |
134 | struct expr_value dir_dep; |
135 | |
136 | /* Reverse dependencies through being selected by other symbols */ |
137 | struct expr_value rev_dep; |
138 | |
139 | /* |
140 | * "Weak" reverse dependencies through being implied by other symbols |
141 | */ |
142 | struct expr_value implied; |
143 | }; |
144 | |
145 | #define SYMBOL_CONST 0x0001 /* symbol is const */ |
146 | #define SYMBOL_CHECK 0x0008 /* used during dependency checking */ |
147 | #define SYMBOL_VALID 0x0080 /* set when symbol.curr is calculated */ |
148 | #define SYMBOL_WRITE 0x0200 /* write symbol to file (KCONFIG_CONFIG) */ |
149 | #define SYMBOL_WRITTEN 0x0800 /* track info to avoid double-write to .config */ |
150 | #define SYMBOL_CHECKED 0x2000 /* used during dependency checking */ |
151 | #define SYMBOL_WARNED 0x8000 /* warning has been issued */ |
152 | |
153 | /* Set when symbol.def[] is used */ |
154 | #define SYMBOL_DEF 0x10000 /* First bit of SYMBOL_DEF */ |
155 | #define SYMBOL_DEF_USER 0x10000 /* symbol.def[S_DEF_USER] is valid */ |
156 | #define SYMBOL_DEF_AUTO 0x20000 /* symbol.def[S_DEF_AUTO] is valid */ |
157 | #define SYMBOL_DEF3 0x40000 /* symbol.def[S_DEF_3] is valid */ |
158 | #define SYMBOL_DEF4 0x80000 /* symbol.def[S_DEF_4] is valid */ |
159 | |
160 | #define SYMBOL_MAXLENGTH 256 |
161 | |
162 | /* A property represent the config options that can be associated |
163 | * with a config "symbol". |
164 | * Sample: |
165 | * config FOO |
166 | * default y |
167 | * prompt "foo prompt" |
168 | * select BAR |
169 | * config BAZ |
170 | * int "BAZ Value" |
171 | * range 1..255 |
172 | * |
173 | * Please, also check parser.y:print_symbol() when modifying the |
174 | * list of property types! |
175 | */ |
176 | enum prop_type { |
177 | P_UNKNOWN, |
178 | P_PROMPT, /* prompt "foo prompt" or "BAZ Value" */ |
179 | , /* text associated with a comment */ |
180 | , /* prompt associated with a menu or menuconfig symbol */ |
181 | P_DEFAULT, /* default y */ |
182 | P_SELECT, /* select BAR */ |
183 | P_IMPLY, /* imply BAR */ |
184 | P_RANGE, /* range 7..100 (for a symbol) */ |
185 | }; |
186 | |
187 | struct property { |
188 | struct property *next; /* next property - null if last */ |
189 | enum prop_type type; /* type of property */ |
190 | const char *text; /* the prompt value - P_PROMPT, P_MENU, P_COMMENT */ |
191 | struct expr_value visible; |
192 | struct expr *expr; /* the optional conditional part of the property */ |
193 | struct menu *; /* the menu the property are associated with |
194 | * valid for: P_SELECT, P_RANGE, |
195 | * P_PROMPT, P_DEFAULT, P_MENU, P_COMMENT */ |
196 | const char *filename; /* what file was this property defined */ |
197 | int lineno; /* what lineno was this property defined */ |
198 | }; |
199 | |
200 | #define for_all_properties(sym, st, tok) \ |
201 | for (st = sym->prop; st; st = st->next) \ |
202 | if (st->type == (tok)) |
203 | #define for_all_defaults(sym, st) for_all_properties(sym, st, P_DEFAULT) |
204 | #define for_all_prompts(sym, st) \ |
205 | for (st = sym->prop; st; st = st->next) \ |
206 | if (st->text) |
207 | |
208 | enum { |
209 | M_CHOICE, // "choice" |
210 | , // "comment" |
211 | M_IF, // "if" |
212 | , // "mainmenu", "menu", "menuconfig" |
213 | M_NORMAL, // others, i.e., "config" |
214 | }; |
215 | |
216 | /* |
217 | * Represents a node in the menu tree, as seen in e.g. menuconfig (though used |
218 | * for all front ends). Each symbol, menu, etc. defined in the Kconfig files |
219 | * gets a node. A symbol defined in multiple locations gets one node at each |
220 | * location. |
221 | * |
222 | * @type: type of the menu entry |
223 | * @choice_members: list of choice members with priority. |
224 | */ |
225 | struct { |
226 | enum menu_type ; |
227 | |
228 | /* The next menu node at the same level */ |
229 | struct menu *; |
230 | |
231 | /* The parent menu node, corresponding to e.g. a menu or choice */ |
232 | struct menu *; |
233 | |
234 | /* The first child menu node, for e.g. menus and choices */ |
235 | struct menu *; |
236 | |
237 | /* |
238 | * The symbol associated with the menu node. Choices are implemented as |
239 | * a special kind of symbol. NULL for menus, comments, and ifs. |
240 | */ |
241 | struct symbol *; |
242 | |
243 | struct list_head ; /* link to symbol::menus */ |
244 | |
245 | struct list_head ; |
246 | |
247 | /* |
248 | * The prompt associated with the node. This holds the prompt for a |
249 | * symbol as well as the text for a menu or comment, along with the |
250 | * type (P_PROMPT, P_MENU, etc.) |
251 | */ |
252 | struct property *; |
253 | |
254 | /* |
255 | * 'visible if' dependencies. If more than one is given, they will be |
256 | * ANDed together. |
257 | */ |
258 | struct expr *; |
259 | |
260 | /* |
261 | * Ordinary dependencies from e.g. 'depends on' and 'if', ANDed |
262 | * together |
263 | */ |
264 | struct expr *; |
265 | |
266 | /* MENU_* flags */ |
267 | unsigned int ; |
268 | |
269 | /* Any help text associated with the node */ |
270 | char *; |
271 | |
272 | /* The location where the menu node appears in the Kconfig files */ |
273 | const char *; |
274 | int ; |
275 | |
276 | /* For use by front ends that need to store auxiliary data */ |
277 | void *; |
278 | }; |
279 | |
280 | /* |
281 | * Set on a menu node when the corresponding symbol changes state in some way. |
282 | * Can be checked by front ends. |
283 | */ |
284 | #define 0x0001 |
285 | |
286 | #define 0x0002 |
287 | |
288 | struct jump_key { |
289 | struct list_head entries; |
290 | size_t offset; |
291 | struct menu *target; |
292 | }; |
293 | |
294 | extern struct symbol symbol_yes, symbol_no, symbol_mod; |
295 | extern struct symbol *modules_sym; |
296 | extern int cdebug; |
297 | struct expr *expr_alloc_symbol(struct symbol *sym); |
298 | struct expr *expr_alloc_one(enum expr_type type, struct expr *ce); |
299 | struct expr *expr_alloc_two(enum expr_type type, struct expr *e1, struct expr *e2); |
300 | struct expr *expr_alloc_comp(enum expr_type type, struct symbol *s1, struct symbol *s2); |
301 | struct expr *expr_alloc_and(struct expr *e1, struct expr *e2); |
302 | struct expr *expr_alloc_or(struct expr *e1, struct expr *e2); |
303 | void expr_eliminate_eq(struct expr **ep1, struct expr **ep2); |
304 | bool expr_eq(struct expr *e1, struct expr *e2); |
305 | tristate expr_calc_value(struct expr *e); |
306 | struct expr *expr_eliminate_dups(struct expr *e); |
307 | struct expr *expr_transform(struct expr *e); |
308 | bool expr_contains_symbol(struct expr *dep, struct symbol *sym); |
309 | bool expr_depends_symbol(struct expr *dep, struct symbol *sym); |
310 | struct expr *expr_trans_compare(struct expr *e, enum expr_type type, struct symbol *sym); |
311 | |
312 | void expr_fprint(struct expr *e, FILE *out); |
313 | struct gstr; /* forward */ |
314 | void expr_gstr_print(const struct expr *e, struct gstr *gs); |
315 | void expr_gstr_print_revdep(struct expr *e, struct gstr *gs, |
316 | tristate pr_type, const char *title); |
317 | |
318 | static inline bool expr_is_yes(const struct expr *e) |
319 | { |
320 | return !e || (e->type == E_SYMBOL && e->left.sym == &symbol_yes); |
321 | } |
322 | |
323 | #ifdef __cplusplus |
324 | } |
325 | #endif |
326 | |
327 | #endif /* EXPR_H */ |
328 | |