blob: b8644b2738edc93fbc183865cb555e440d6f744c [file] [log] [blame]
/*
* macros.c - a simple macro expander
*
* Copyright (c) 2010-2020 Cisco and/or its affiliates.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <vppinfra/macros.h>
static inline int
macro_isalnum (i8 c)
{
if ((c >= 'A' && c <= 'Z')
|| (c >= 'a' && c <= 'z') || (c >= '0' && c <= '9') || (c == '_'))
return 1;
return 0;
}
static i8 *
builtin_eval (clib_macro_main_t * mm, i8 * varname, i32 complain)
{
uword *p;
i8 *(*fp) (clib_macro_main_t *, i32);
p = hash_get_mem (mm->the_builtin_eval_hash, varname);
if (p == 0)
return 0;
fp = (void *) (p[0]);
return (*fp) (mm, complain);
}
__clib_export int
clib_macro_unset (clib_macro_main_t * mm, char *name)
{
hash_pair_t *p;
u8 *key, *value;
p = hash_get_pair (mm->the_value_table_hash, name);
if (p == 0)
return 1;
key = (u8 *) (p->key);
value = (u8 *) (p->value[0]);
hash_unset_mem (mm->the_value_table_hash, name);
vec_free (value);
vec_free (key);
return 0;
}
__clib_export int
clib_macro_set_value (clib_macro_main_t * mm, char *name, char *value)
{
u8 *key_copy, *value_copy;
int rv;
rv = clib_macro_unset (mm, name);
key_copy = format (0, "%s%c", name, 0);
value_copy = format (0, "%s%c", value, 0);
hash_set_mem (mm->the_value_table_hash, key_copy, value_copy);
return rv;
}
i8 *
clib_macro_get_value (clib_macro_main_t * mm, char *name)
{
uword *p;
p = hash_get_mem (mm->the_value_table_hash, name);
if (p)
return (i8 *) (p[0]);
else
return 0;
}
/*
* eval: takes a string, returns a vector.
* looks up $foobar in the variable table.
*/
__clib_export i8 *
clib_macro_eval (clib_macro_main_t * mm, i8 * s, i32 complain, u16 level,
u16 max_level)
{
i8 *rv = 0;
i8 *varname, *varvalue;
i8 *ts;
if (level >= max_level)
{
if (complain)
clib_warning ("circular definition, level %d", level);
return (i8 *) format (0, " CIRCULAR ");
}
while (*s)
{
switch (*s)
{
case '\\':
s++;
/* fallthrough */
default:
vec_add1 (rv, *s);
s++;
break;
case '$':
s++;
varname = 0;
/*
* Make vector with variable name in it.
*/
while (*s && (macro_isalnum (*s) || (*s == '_') || (*s == '(')))
{
/* handle $(foo) */
if (*s == '(')
{
s++; /* skip '(' */
while (*s && *s != ')')
{
vec_add1 (varname, *s);
s++;
}
if (*s)
s++; /* skip ')' */
break;
}
vec_add1 (varname, *s);
s++;
}
/* null terminate */
vec_add1 (varname, 0);
/* Look for a builtin, e.g. $my_hostname */
if (!(varvalue = builtin_eval (mm, varname, complain)))
{
/* Look in value table */
if (!varvalue)
{
i8 *tmp = clib_macro_get_value (mm, (char *) varname);
if (tmp)
varvalue = (i8 *) format (0, "%s%c", tmp, 0);
}
#ifdef CLIB_UNIX
/* Look in environment. */
if (!varvalue)
{
char *tmp = getenv ((char *) varname);
if (tmp)
varvalue = (i8 *) format (0, "%s%c", tmp, 0);
}
#endif /* CLIB_UNIX */
}
if (varvalue)
{
/* recursively evaluate */
ts = clib_macro_eval (mm, varvalue, complain, level + 1,
max_level);
vec_free (varvalue);
/* add results to answer */
vec_append (rv, ts);
/* Remove NULL termination or the results are sad */
vec_set_len (rv, vec_len (rv) - 1);
vec_free (ts);
}
else
{
if (complain)
clib_warning ("Undefined Variable Reference: %s\n", varname);
vec_append (rv, format (0, "UNSET "));
vec_set_len (rv, vec_len (rv) - 1);
}
vec_free (varname);
}
}
vec_add1 (rv, 0);
return (rv);
}
/*
* eval: takes a string, returns a vector.
* looks up $foobar in the variable table.
*/
__clib_export i8 *
clib_macro_eval_dollar (clib_macro_main_t *mm, i8 *s, i32 complain)
{
i8 *s2;
i8 *rv;
s2 = (i8 *) format (0, "$(%s)%c", s, 0);
rv = clib_macro_eval (mm, s2, complain, 0 /* level */ , 8 /* max_level */ );
vec_free (s2);
return (rv);
}
__clib_export void
clib_macro_add_builtin (clib_macro_main_t * mm, char *name, void *eval_fn)
{
hash_set_mem (mm->the_builtin_eval_hash, name, (uword) eval_fn);
}
#ifdef CLIB_UNIX
static i8 *
eval_hostname (clib_macro_main_t * mm, i32 complain)
{
char tmp[128];
if (gethostname (tmp, sizeof (tmp)))
return ((i8 *) format (0, "gethostname-error%c", 0));
return ((i8 *) format (0, "%s%c", tmp, 0));
}
#endif
__clib_export void
clib_macro_init (clib_macro_main_t * mm)
{
if (mm->the_builtin_eval_hash != 0)
{
clib_warning ("mm %p already initialized", mm);
return;
}
mm->the_builtin_eval_hash = hash_create_string (0, sizeof (uword));
mm->the_value_table_hash = hash_create_string (0, sizeof (uword));
#ifdef CLIB_UNIX
hash_set_mem (mm->the_builtin_eval_hash, "hostname", (uword) eval_hostname);
#endif
}
__clib_export void
clib_macro_free (clib_macro_main_t * mm)
{
hash_pair_t *p;
u8 **strings_to_free = 0;
int i;
hash_free (mm->the_builtin_eval_hash);
hash_foreach_pair (p, mm->the_value_table_hash,
({
vec_add1 (strings_to_free, (u8 *) (p->key));
vec_add1 (strings_to_free, (u8 *) (p->value[0]));
}));
for (i = 0; i < vec_len (strings_to_free); i++)
vec_free (strings_to_free[i]);
vec_free (strings_to_free);
hash_free (mm->the_value_table_hash);
}
typedef struct
{
u8 *name;
u8 *value;
} name_sort_t;
static int
name_compare (void *a1, void *a2)
{
name_sort_t *ns1 = a1;
name_sort_t *ns2 = a2;
return strcmp ((char *) ns1->name, (char *) ns2->name);
}
__clib_export u8 *
format_clib_macro_main (u8 * s, va_list * args)
{
clib_macro_main_t *mm = va_arg (*args, clib_macro_main_t *);
int evaluate = va_arg (*args, int);
hash_pair_t *p;
name_sort_t *nses = 0, *ns;
int i;
hash_foreach_pair (p, mm->the_value_table_hash,
({
vec_add2 (nses, ns, 1);
ns->name = (u8 *)(p->key);
ns->value = (u8 *)(p->value[0]);
}));
if (vec_len (nses) == 0)
return s;
vec_sort_with_function (nses, name_compare);
for (i = 0; i < vec_len (nses); i++)
{
s = format (s, "%-20s", nses[i].name);
if (evaluate == 0)
s = format (s, "%s\n", nses[i].value);
else
{
u8 *rv = (u8 *) clib_macro_eval_dollar (mm, (i8 *) nses[i].name,
0 /* no complain */ );
s = format (s, "%s\n", rv);
vec_free (rv);
}
}
return s;
}
/*
* fd.io coding-style-patch-verification: ON
*
* Local Variables:
* eval: (c-set-style "gnu")
* End:
*/