blob: 240cef02e0e1bc139aacc151e322112ae93a532e [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
Dave Barach961e3c82020-06-18 17:04:18 -04002 * macros.c - a simple macro expander
3 *
4 * Copyright (c) 2010-2020 Cisco and/or its affiliates.
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
Ed Warnickecb9cada2015-12-08 15:45:58 -07007 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at:
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17*/
18
19#include <vppinfra/macros.h>
20
Dave Barachc3799992016-08-15 11:12:27 -040021static inline int
22macro_isalnum (i8 c)
Ed Warnickecb9cada2015-12-08 15:45:58 -070023{
24 if ((c >= 'A' && c <= 'Z')
Dave Barachc3799992016-08-15 11:12:27 -040025 || (c >= 'a' && c <= 'z') || (c >= '0' && c <= '9') || (c == '_'))
Ed Warnickecb9cada2015-12-08 15:45:58 -070026 return 1;
27 return 0;
28}
29
Dave Barachc3799992016-08-15 11:12:27 -040030static i8 *
Dave Barach961e3c82020-06-18 17:04:18 -040031builtin_eval (clib_macro_main_t * mm, i8 * varname, i32 complain)
Ed Warnickecb9cada2015-12-08 15:45:58 -070032{
Dave Barachc3799992016-08-15 11:12:27 -040033 uword *p;
Dave Barach961e3c82020-06-18 17:04:18 -040034 i8 *(*fp) (clib_macro_main_t *, i32);
Ed Warnickecb9cada2015-12-08 15:45:58 -070035
Dave Barachc3799992016-08-15 11:12:27 -040036 p = hash_get_mem (mm->the_builtin_eval_hash, varname);
37 if (p == 0)
38 return 0;
39 fp = (void *) (p[0]);
40 return (*fp) (mm, complain);
Ed Warnickecb9cada2015-12-08 15:45:58 -070041}
42
Dave Barachc3799992016-08-15 11:12:27 -040043int
Dave Barach961e3c82020-06-18 17:04:18 -040044clib_macro_unset (clib_macro_main_t * mm, char *name)
Ed Warnickecb9cada2015-12-08 15:45:58 -070045{
Dave Barachc3799992016-08-15 11:12:27 -040046 hash_pair_t *p;
47 u8 *key, *value;
Ed Warnickecb9cada2015-12-08 15:45:58 -070048
49 p = hash_get_pair (mm->the_value_table_hash, name);
Dave Barachc3799992016-08-15 11:12:27 -040050
Ed Warnickecb9cada2015-12-08 15:45:58 -070051 if (p == 0)
52 return 1;
53
Dave Barachc3799992016-08-15 11:12:27 -040054 key = (u8 *) (p->key);
Ed Warnickecb9cada2015-12-08 15:45:58 -070055 value = (u8 *) (p->value[0]);
56 hash_unset_mem (mm->the_value_table_hash, name);
57
58 vec_free (value);
59 vec_free (key);
60 return 0;
61}
62
Dave Barachc3799992016-08-15 11:12:27 -040063int
Dave Barach961e3c82020-06-18 17:04:18 -040064clib_macro_set_value (clib_macro_main_t * mm, char *name, char *value)
Ed Warnickecb9cada2015-12-08 15:45:58 -070065{
Dave Barachc3799992016-08-15 11:12:27 -040066 u8 *key_copy, *value_copy;
Ed Warnickecb9cada2015-12-08 15:45:58 -070067 int rv;
68
69 rv = clib_macro_unset (mm, name);
70
71 key_copy = format (0, "%s%c", name, 0);
72 value_copy = format (0, "%s%c", value, 0);
73
74 hash_set_mem (mm->the_value_table_hash, key_copy, value_copy);
75 return rv;
76}
77
Dave Barachc3799992016-08-15 11:12:27 -040078i8 *
Dave Barach961e3c82020-06-18 17:04:18 -040079clib_macro_get_value (clib_macro_main_t * mm, char *name)
Ed Warnickecb9cada2015-12-08 15:45:58 -070080{
Dave Barachc3799992016-08-15 11:12:27 -040081 uword *p;
Ed Warnickecb9cada2015-12-08 15:45:58 -070082
83 p = hash_get_mem (mm->the_value_table_hash, name);
84 if (p)
Dave Barachc3799992016-08-15 11:12:27 -040085 return (i8 *) (p[0]);
Ed Warnickecb9cada2015-12-08 15:45:58 -070086 else
87 return 0;
88}
89
90/*
91 * eval: takes a string, returns a vector.
92 * looks up $foobar in the variable table.
93 */
Dave Barachc3799992016-08-15 11:12:27 -040094i8 *
Dave Barachb30b9542020-06-22 10:02:25 -040095clib_macro_eval (clib_macro_main_t * mm, i8 * s, i32 complain, u16 level,
96 u16 max_level)
Ed Warnickecb9cada2015-12-08 15:45:58 -070097{
Dave Barachc3799992016-08-15 11:12:27 -040098 i8 *rv = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -070099 i8 *varname, *varvalue;
100 i8 *ts;
101
Dave Barachb30b9542020-06-22 10:02:25 -0400102 if (level >= max_level)
103 {
104 if (complain)
105 clib_warning ("circular definition, level %d", level);
106 return (i8 *) format (0, " CIRCULAR ");
107 }
108
Dave Barachc3799992016-08-15 11:12:27 -0400109 while (*s)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700110 {
Dave Barachc3799992016-08-15 11:12:27 -0400111 switch (*s)
112 {
113 case '\\':
114 s++;
115 /* fallthrough */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700116
Dave Barachc3799992016-08-15 11:12:27 -0400117 default:
118 vec_add1 (rv, *s);
119 s++;
120 break;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700121
Dave Barachc3799992016-08-15 11:12:27 -0400122 case '$':
123 s++;
124 varname = 0;
125 /*
126 * Make vector with variable name in it.
127 */
128 while (*s && (macro_isalnum (*s) || (*s == '_') || (*s == '(')))
129 {
Ed Warnickecb9cada2015-12-08 15:45:58 -0700130
Dave Barachc3799992016-08-15 11:12:27 -0400131 /* handle $(foo) */
132 if (*s == '(')
133 {
134 s++; /* skip '(' */
135 while (*s && *s != ')')
136 {
137 vec_add1 (varname, *s);
138 s++;
139 }
140 if (*s)
141 s++; /* skip ')' */
142 break;
143 }
144 vec_add1 (varname, *s);
145 s++;
146 }
147 /* null terminate */
148 vec_add1 (varname, 0);
149 /* Look for a builtin, e.g. $my_hostname */
150 if (!(varvalue = builtin_eval (mm, varname, complain)))
151 {
152 /* Look in value table */
153 if (!varvalue)
154 {
Damjan Marionffe9d212018-05-30 09:26:11 +0200155 i8 *tmp = clib_macro_get_value (mm, (char *) varname);
Dave Barachc3799992016-08-15 11:12:27 -0400156 if (tmp)
157 varvalue = (i8 *) format (0, "%s%c", tmp, 0);
158 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700159#ifdef CLIB_UNIX
Dave Barachc3799992016-08-15 11:12:27 -0400160 /* Look in environment. */
161 if (!varvalue)
162 {
Damjan Marionffe9d212018-05-30 09:26:11 +0200163 char *tmp = getenv ((char *) varname);
Dave Barachc3799992016-08-15 11:12:27 -0400164 if (tmp)
165 varvalue = (i8 *) format (0, "%s%c", tmp, 0);
166 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700167#endif /* CLIB_UNIX */
Dave Barachc3799992016-08-15 11:12:27 -0400168 }
169 if (varvalue)
170 {
171 /* recursively evaluate */
Dave Barachb30b9542020-06-22 10:02:25 -0400172 ts = clib_macro_eval (mm, varvalue, complain, level + 1,
173 max_level);
Dave Barachc3799992016-08-15 11:12:27 -0400174 vec_free (varvalue);
175 /* add results to answer */
176 vec_append (rv, ts);
177 /* Remove NULL termination or the results are sad */
178 _vec_len (rv) = vec_len (rv) - 1;
179 vec_free (ts);
180 }
181 else
182 {
183 if (complain)
184 clib_warning ("Undefined Variable Reference: %s\n", varname);
185 vec_append (rv, format (0, "UNSET "));
186 _vec_len (rv) = vec_len (rv) - 1;
187
188 }
189 vec_free (varname);
190 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700191 }
Dave Barachc3799992016-08-15 11:12:27 -0400192 vec_add1 (rv, 0);
193 return (rv);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700194}
195
196/*
197 * eval: takes a string, returns a vector.
198 * looks up $foobar in the variable table.
199 */
Dave Barachc3799992016-08-15 11:12:27 -0400200i8 *
Dave Barach961e3c82020-06-18 17:04:18 -0400201clib_macro_eval_dollar (clib_macro_main_t * mm, i8 * s, i32 complain)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700202{
203 i8 *s2;
204 i8 *rv;
205
Dave Barachc3799992016-08-15 11:12:27 -0400206 s2 = (i8 *) format (0, "$(%s)%c", s, 0);
Dave Barachb30b9542020-06-22 10:02:25 -0400207 rv = clib_macro_eval (mm, s2, complain, 0 /* level */ , 8 /* max_level */ );
Ed Warnickecb9cada2015-12-08 15:45:58 -0700208 vec_free (s2);
209 return (rv);
210}
211
Dave Barachc3799992016-08-15 11:12:27 -0400212void
Dave Barach961e3c82020-06-18 17:04:18 -0400213clib_macro_add_builtin (clib_macro_main_t * mm, char *name, void *eval_fn)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700214{
Dave Barachc3799992016-08-15 11:12:27 -0400215 hash_set_mem (mm->the_builtin_eval_hash, name, (uword) eval_fn);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700216}
217
218#ifdef CLIB_UNIX
Dave Barachc3799992016-08-15 11:12:27 -0400219static i8 *
Dave Barach961e3c82020-06-18 17:04:18 -0400220eval_hostname (clib_macro_main_t * mm, i32 complain)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700221{
222 char tmp[128];
Dave Barachc3799992016-08-15 11:12:27 -0400223 if (gethostname (tmp, sizeof (tmp)))
Ed Warnickecb9cada2015-12-08 15:45:58 -0700224 return ((i8 *) format (0, "gethostname-error%c", 0));
225 return ((i8 *) format (0, "%s%c", tmp, 0));
226}
227#endif
228
Dave Barachc3799992016-08-15 11:12:27 -0400229void
Dave Barach961e3c82020-06-18 17:04:18 -0400230clib_macro_init (clib_macro_main_t * mm)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700231{
232 if (mm->the_builtin_eval_hash != 0)
233 {
Christophe Fontainefef15b42016-04-09 12:38:49 +0900234 clib_warning ("mm %p already initialized", mm);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700235 return;
236 }
237
Dave Barachc3799992016-08-15 11:12:27 -0400238 mm->the_builtin_eval_hash = hash_create_string (0, sizeof (uword));
239 mm->the_value_table_hash = hash_create_string (0, sizeof (uword));
240
Ed Warnickecb9cada2015-12-08 15:45:58 -0700241#ifdef CLIB_UNIX
Dave Barachc3799992016-08-15 11:12:27 -0400242 hash_set_mem (mm->the_builtin_eval_hash, "hostname", (uword) eval_hostname);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700243#endif
244}
245
Dave Barachc3799992016-08-15 11:12:27 -0400246void
Dave Barach961e3c82020-06-18 17:04:18 -0400247clib_macro_free (clib_macro_main_t * mm)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700248{
Dave Barachc3799992016-08-15 11:12:27 -0400249 hash_pair_t *p;
250 u8 **strings_to_free = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700251 int i;
252
253 hash_free (mm->the_builtin_eval_hash);
254
Dave Barachc3799992016-08-15 11:12:27 -0400255 /* *INDENT-OFF* */
256 hash_foreach_pair (p, mm->the_value_table_hash,
Ed Warnickecb9cada2015-12-08 15:45:58 -0700257 ({
258 vec_add1 (strings_to_free, (u8 *) (p->key));
259 vec_add1 (strings_to_free, (u8 *) (p->value[0]));
260 }));
Dave Barachc3799992016-08-15 11:12:27 -0400261 /* *INDENT-ON* */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700262
263 for (i = 0; i < vec_len (strings_to_free); i++)
264 vec_free (strings_to_free[i]);
265 vec_free (strings_to_free);
266 hash_free (mm->the_value_table_hash);
267}
Dave Barachc3799992016-08-15 11:12:27 -0400268
Dave Barach961e3c82020-06-18 17:04:18 -0400269typedef struct
270{
271 u8 *name;
272 u8 *value;
273} name_sort_t;
274
275static int
276name_compare (void *a1, void *a2)
277{
278 name_sort_t *ns1 = a1;
279 name_sort_t *ns2 = a2;
280
281 return strcmp ((char *) ns1->name, (char *) ns2->name);
282}
283
284
285u8 *
286format_clib_macro_main (u8 * s, va_list * args)
287{
288 clib_macro_main_t *mm = va_arg (*args, clib_macro_main_t *);
289 int evaluate = va_arg (*args, int);
290 hash_pair_t *p;
291 name_sort_t *nses = 0, *ns;
292 int i;
293
294 /* *INDENT-OFF* */
295 hash_foreach_pair (p, mm->the_value_table_hash,
296 ({
297 vec_add2 (nses, ns, 1);
298 ns->name = (u8 *)(p->key);
299 ns->value = (u8 *)(p->value[0]);
300 }));
301 /* *INDENT-ON* */
302
303 if (vec_len (nses) == 0)
304 return s;
305
306 vec_sort_with_function (nses, name_compare);
307
308 for (i = 0; i < vec_len (nses); i++)
309 {
310 s = format (s, "%-20s", nses[i].name);
311 if (evaluate == 0)
312 s = format (s, "%s\n", nses[i].value);
313 else
314 {
315 u8 *rv = (u8 *) clib_macro_eval_dollar (mm, (i8 *) nses[i].name,
316 0 /* no complain */ );
317 s = format (s, "%s\n", rv);
318 vec_free (rv);
319 }
320 }
321 return s;
322}
323
324
Dave Barachc3799992016-08-15 11:12:27 -0400325/*
326 * fd.io coding-style-patch-verification: ON
327 *
328 * Local Variables:
329 * eval: (c-set-style "gnu")
330 * End:
331 */