diff options
author | Masahiro Yamada <yamada.masahiro@socionext.com> | 2018-05-28 05:21:45 -0400 |
---|---|---|
committer | Masahiro Yamada <yamada.masahiro@socionext.com> | 2018-05-28 14:31:19 -0400 |
commit | e298f3b49def8e67c2466bef8aed355462bfa7f1 (patch) | |
tree | 20a897e3f53ee948054d8025ca98ac6bef18b705 | |
parent | 137c0118a900bc4a3d1673573e22a03fbae3e8fd (diff) |
kconfig: add built-in function support
This commit adds a new concept 'function' to do more text processing
in Kconfig.
A function call looks like this:
$(function,arg1,arg2,arg3,...)
This commit adds the basic infrastructure to expand functions.
Change the text expansion helpers to take arguments.
Signed-off-by: Masahiro Yamada <yamada.masahiro@socionext.com>
-rw-r--r-- | scripts/kconfig/preprocess.c | 142 |
1 files changed, 130 insertions, 12 deletions
diff --git a/scripts/kconfig/preprocess.c b/scripts/kconfig/preprocess.c index a2eb2eb02929..f32a496626da 100644 --- a/scripts/kconfig/preprocess.c +++ b/scripts/kconfig/preprocess.c | |||
@@ -10,6 +10,10 @@ | |||
10 | 10 | ||
11 | #include "list.h" | 11 | #include "list.h" |
12 | 12 | ||
13 | #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0])) | ||
14 | |||
15 | static char *expand_string_with_args(const char *in, int argc, char *argv[]); | ||
16 | |||
13 | static void __attribute__((noreturn)) pperror(const char *format, ...) | 17 | static void __attribute__((noreturn)) pperror(const char *format, ...) |
14 | { | 18 | { |
15 | va_list ap; | 19 | va_list ap; |
@@ -92,20 +96,123 @@ void env_write_dep(FILE *f, const char *autoconfig_name) | |||
92 | } | 96 | } |
93 | } | 97 | } |
94 | 98 | ||
95 | static char *eval_clause(const char *str, size_t len) | 99 | /* |
100 | * Built-in functions | ||
101 | */ | ||
102 | struct function { | ||
103 | const char *name; | ||
104 | unsigned int min_args; | ||
105 | unsigned int max_args; | ||
106 | char *(*func)(int argc, char *argv[]); | ||
107 | }; | ||
108 | |||
109 | static const struct function function_table[] = { | ||
110 | /* Name MIN MAX Function */ | ||
111 | }; | ||
112 | |||
113 | #define FUNCTION_MAX_ARGS 16 | ||
114 | |||
115 | static char *function_expand(const char *name, int argc, char *argv[]) | ||
96 | { | 116 | { |
97 | char *tmp, *name, *res; | 117 | const struct function *f; |
118 | int i; | ||
119 | |||
120 | for (i = 0; i < ARRAY_SIZE(function_table); i++) { | ||
121 | f = &function_table[i]; | ||
122 | if (strcmp(f->name, name)) | ||
123 | continue; | ||
124 | |||
125 | if (argc < f->min_args) | ||
126 | pperror("too few function arguments passed to '%s'", | ||
127 | name); | ||
128 | |||
129 | if (argc > f->max_args) | ||
130 | pperror("too many function arguments passed to '%s'", | ||
131 | name); | ||
132 | |||
133 | return f->func(argc, argv); | ||
134 | } | ||
135 | |||
136 | return NULL; | ||
137 | } | ||
138 | |||
139 | /* | ||
140 | * Evaluate a clause with arguments. argc/argv are arguments from the upper | ||
141 | * function call. | ||
142 | * | ||
143 | * Returned string must be freed when done | ||
144 | */ | ||
145 | static char *eval_clause(const char *str, size_t len, int argc, char *argv[]) | ||
146 | { | ||
147 | char *tmp, *name, *res, *prev, *p; | ||
148 | int new_argc = 0; | ||
149 | char *new_argv[FUNCTION_MAX_ARGS]; | ||
150 | int nest = 0; | ||
151 | int i; | ||
98 | 152 | ||
99 | tmp = xstrndup(str, len); | 153 | tmp = xstrndup(str, len); |
100 | 154 | ||
101 | name = expand_string(tmp); | 155 | prev = p = tmp; |
156 | |||
157 | /* | ||
158 | * Split into tokens | ||
159 | * The function name and arguments are separated by a comma. | ||
160 | * For example, if the function call is like this: | ||
161 | * $(foo,$(x),$(y)) | ||
162 | * | ||
163 | * The input string for this helper should be: | ||
164 | * foo,$(x),$(y) | ||
165 | * | ||
166 | * and split into: | ||
167 | * new_argv[0] = 'foo' | ||
168 | * new_argv[1] = '$(x)' | ||
169 | * new_argv[2] = '$(y)' | ||
170 | */ | ||
171 | while (*p) { | ||
172 | if (nest == 0 && *p == ',') { | ||
173 | *p = 0; | ||
174 | if (new_argc >= FUNCTION_MAX_ARGS) | ||
175 | pperror("too many function arguments"); | ||
176 | new_argv[new_argc++] = prev; | ||
177 | prev = p + 1; | ||
178 | } else if (*p == '(') { | ||
179 | nest++; | ||
180 | } else if (*p == ')') { | ||
181 | nest--; | ||
182 | } | ||
102 | 183 | ||
103 | res = env_expand(name); | 184 | p++; |
185 | } | ||
186 | new_argv[new_argc++] = prev; | ||
187 | |||
188 | /* | ||
189 | * Shift arguments | ||
190 | * new_argv[0] represents a function name or a variable name. Put it | ||
191 | * into 'name', then shift the rest of the arguments. This simplifies | ||
192 | * 'const' handling. | ||
193 | */ | ||
194 | name = expand_string_with_args(new_argv[0], argc, argv); | ||
195 | new_argc--; | ||
196 | for (i = 0; i < new_argc; i++) | ||
197 | new_argv[i] = expand_string_with_args(new_argv[i + 1], | ||
198 | argc, argv); | ||
199 | |||
200 | /* Look for built-in functions */ | ||
201 | res = function_expand(name, new_argc, new_argv); | ||
104 | if (res) | 202 | if (res) |
105 | goto free; | 203 | goto free; |
106 | 204 | ||
205 | /* Last, try environment variable */ | ||
206 | if (new_argc == 0) { | ||
207 | res = env_expand(name); | ||
208 | if (res) | ||
209 | goto free; | ||
210 | } | ||
211 | |||
107 | res = xstrdup(""); | 212 | res = xstrdup(""); |
108 | free: | 213 | free: |
214 | for (i = 0; i < new_argc; i++) | ||
215 | free(new_argv[i]); | ||
109 | free(name); | 216 | free(name); |
110 | free(tmp); | 217 | free(tmp); |
111 | 218 | ||
@@ -124,14 +231,14 @@ free: | |||
124 | * after the corresponding closing parenthesis, in this case, *str will be | 231 | * after the corresponding closing parenthesis, in this case, *str will be |
125 | * $(BAR) | 232 | * $(BAR) |
126 | */ | 233 | */ |
127 | char *expand_dollar(const char **str) | 234 | static char *expand_dollar_with_args(const char **str, int argc, char *argv[]) |
128 | { | 235 | { |
129 | const char *p = *str; | 236 | const char *p = *str; |
130 | const char *q; | 237 | const char *q; |
131 | int nest = 0; | 238 | int nest = 0; |
132 | 239 | ||
133 | /* | 240 | /* |
134 | * In Kconfig, variable references always start with "$(". | 241 | * In Kconfig, variable/function references always start with "$(". |
135 | * Neither single-letter variables as in $A nor curly braces as in ${CC} | 242 | * Neither single-letter variables as in $A nor curly braces as in ${CC} |
136 | * are supported. '$' not followed by '(' loses its special meaning. | 243 | * are supported. '$' not followed by '(' loses its special meaning. |
137 | */ | 244 | */ |
@@ -158,10 +265,16 @@ char *expand_dollar(const char **str) | |||
158 | /* Advance 'str' to after the expanded initial portion of the string */ | 265 | /* Advance 'str' to after the expanded initial portion of the string */ |
159 | *str = q + 1; | 266 | *str = q + 1; |
160 | 267 | ||
161 | return eval_clause(p, q - p); | 268 | return eval_clause(p, q - p, argc, argv); |
269 | } | ||
270 | |||
271 | char *expand_dollar(const char **str) | ||
272 | { | ||
273 | return expand_dollar_with_args(str, 0, NULL); | ||
162 | } | 274 | } |
163 | 275 | ||
164 | static char *__expand_string(const char **str, bool (*is_end)(char c)) | 276 | static char *__expand_string(const char **str, bool (*is_end)(char c), |
277 | int argc, char *argv[]) | ||
165 | { | 278 | { |
166 | const char *in, *p; | 279 | const char *in, *p; |
167 | char *expansion, *out; | 280 | char *expansion, *out; |
@@ -177,7 +290,7 @@ static char *__expand_string(const char **str, bool (*is_end)(char c)) | |||
177 | if (*p == '$') { | 290 | if (*p == '$') { |
178 | in_len = p - in; | 291 | in_len = p - in; |
179 | p++; | 292 | p++; |
180 | expansion = expand_dollar(&p); | 293 | expansion = expand_dollar_with_args(&p, argc, argv); |
181 | out_len += in_len + strlen(expansion); | 294 | out_len += in_len + strlen(expansion); |
182 | out = xrealloc(out, out_len); | 295 | out = xrealloc(out, out_len); |
183 | strncat(out, in, in_len); | 296 | strncat(out, in, in_len); |
@@ -210,13 +323,18 @@ static bool is_end_of_str(char c) | |||
210 | } | 323 | } |
211 | 324 | ||
212 | /* | 325 | /* |
213 | * Expand variables in the given string. Undefined variables | 326 | * Expand variables and functions in the given string. Undefined variables |
214 | * expand to an empty string. | 327 | * expand to an empty string. |
215 | * The returned string must be freed when done. | 328 | * The returned string must be freed when done. |
216 | */ | 329 | */ |
330 | static char *expand_string_with_args(const char *in, int argc, char *argv[]) | ||
331 | { | ||
332 | return __expand_string(&in, is_end_of_str, argc, argv); | ||
333 | } | ||
334 | |||
217 | char *expand_string(const char *in) | 335 | char *expand_string(const char *in) |
218 | { | 336 | { |
219 | return __expand_string(&in, is_end_of_str); | 337 | return expand_string_with_args(in, 0, NULL); |
220 | } | 338 | } |
221 | 339 | ||
222 | static bool is_end_of_token(char c) | 340 | static bool is_end_of_token(char c) |
@@ -234,5 +352,5 @@ static bool is_end_of_token(char c) | |||
234 | */ | 352 | */ |
235 | char *expand_one_token(const char **str) | 353 | char *expand_one_token(const char **str) |
236 | { | 354 | { |
237 | return __expand_string(str, is_end_of_token); | 355 | return __expand_string(str, is_end_of_token, 0, NULL); |
238 | } | 356 | } |