| | varnish-cache/lib/libvcc/vcc_symb.c |
| 0 |
|
/*- |
| 1 |
|
* Copyright (c) 2010 Varnish Software AS |
| 2 |
|
* All rights reserved. |
| 3 |
|
* |
| 4 |
|
* Author: Poul-Henning Kamp <phk@phk.freebsd.dk> |
| 5 |
|
* |
| 6 |
|
* SPDX-License-Identifier: BSD-2-Clause |
| 7 |
|
* |
| 8 |
|
* Redistribution and use in source and binary forms, with or without |
| 9 |
|
* modification, are permitted provided that the following conditions |
| 10 |
|
* are met: |
| 11 |
|
* 1. Redistributions of source code must retain the above copyright |
| 12 |
|
* notice, this list of conditions and the following disclaimer. |
| 13 |
|
* 2. Redistributions in binary form must reproduce the above copyright |
| 14 |
|
* notice, this list of conditions and the following disclaimer in the |
| 15 |
|
* documentation and/or other materials provided with the distribution. |
| 16 |
|
* |
| 17 |
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND |
| 18 |
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| 19 |
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| 20 |
|
* ARE DISCLAIMED. IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE |
| 21 |
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| 22 |
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS |
| 23 |
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) |
| 24 |
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
| 25 |
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY |
| 26 |
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF |
| 27 |
|
* SUCH DAMAGE. |
| 28 |
|
*/ |
| 29 |
|
/*lint -save -esym(759, SYMTAB_NOERR) -esym(765, SYMTAB_NOERR)*/ |
| 30 |
|
|
| 31 |
|
#include "config.h" |
| 32 |
|
|
| 33 |
|
#include <ctype.h> |
| 34 |
|
#include <stdlib.h> |
| 35 |
|
#include <string.h> |
| 36 |
|
|
| 37 |
|
#include "vcc_compile.h" |
| 38 |
|
|
| 39 |
|
#include "vct.h" |
| 40 |
|
|
| 41 |
|
/*--------------------------------------------------------------------*/ |
| 42 |
|
|
| 43 |
|
#define VCC_KIND(U,l) const struct kind SYM_##U[1] = {{ KIND_MAGIC, #l}}; |
| 44 |
|
#include "tbl/symbol_kind.h" |
| 45 |
|
|
| 46 |
|
/*--------------------------------------------------------------------*/ |
| 47 |
|
|
| 48 |
|
struct vcc_namespace { |
| 49 |
|
unsigned magic; |
| 50 |
|
#define VCC_NAMESPACE_MAGIC 0x27b842f4 |
| 51 |
|
const char *name; |
| 52 |
|
enum vcc_namespace_e id; |
| 53 |
|
}; |
| 54 |
|
|
| 55 |
|
#define VCC_NAMESPACE(U, l) \ |
| 56 |
|
static const struct vcc_namespace sym_##l = { \ |
| 57 |
|
VCC_NAMESPACE_MAGIC, \ |
| 58 |
|
#l, \ |
| 59 |
|
VCC_NAMESPACE_##U \ |
| 60 |
|
}; \ |
| 61 |
|
vcc_ns_t SYM_##U = &sym_##l; |
| 62 |
|
#include "vcc_namespace.h" |
| 63 |
|
|
| 64 |
|
/*--------------------------------------------------------------------*/ |
| 65 |
|
|
| 66 |
|
struct symtab { |
| 67 |
|
unsigned magic; |
| 68 |
|
#define SYMTAB_MAGIC 0x084d9c8a |
| 69 |
|
unsigned nlen; |
| 70 |
|
const char *name; |
| 71 |
|
const struct symtab *parent; |
| 72 |
|
VTAILQ_ENTRY(symtab) list; |
| 73 |
|
VTAILQ_HEAD(,symtab) children; |
| 74 |
|
VTAILQ_HEAD(,symbol) symbols; |
| 75 |
|
}; |
| 76 |
|
|
| 77 |
|
vcc_kind_t |
| 78 |
7959788 |
VCC_HandleKind(vcc_type_t fmt) |
| 79 |
|
{ |
| 80 |
7959788 |
if (fmt == ACL) return (SYM_ACL); |
| 81 |
7956228 |
if (fmt == BACKEND) return (SYM_BACKEND); |
| 82 |
7827754 |
if (fmt == PROBE) return (SYM_PROBE); |
| 83 |
7826834 |
if (fmt == STEVEDORE) return (SYM_STEVEDORE); |
| 84 |
7826834 |
if (fmt == SUB) return (SYM_SUB); |
| 85 |
16400 |
if (fmt == INSTANCE) return (SYM_INSTANCE); |
| 86 |
0 |
AZ(fmt->global_pfx); |
| 87 |
0 |
return (SYM_NONE); |
| 88 |
7959788 |
} |
| 89 |
|
|
| 90 |
|
void |
| 91 |
5844266 |
VCC_PrintCName(struct vsb *vsb, const char *b, const char *e) |
| 92 |
|
{ |
| 93 |
|
|
| 94 |
5844266 |
AN(vsb); |
| 95 |
5844266 |
AN(b); |
| 96 |
|
|
| 97 |
5844266 |
if (e == NULL) |
| 98 |
5844266 |
e = strchr(b, '\0'); |
| 99 |
5844266 |
assert(b < e); |
| 100 |
|
|
| 101 |
94917768 |
for (; b < e; b++) |
| 102 |
178147004 |
if (vct_isalnum(*b)) |
| 103 |
80275274 |
VSB_putc(vsb, *b); |
| 104 |
|
else |
| 105 |
8798228 |
VSB_printf(vsb, "_%02x_", *b); |
| 106 |
5844266 |
} |
| 107 |
|
|
| 108 |
|
static void |
| 109 |
25661600 |
vcc_symtabname(struct vsb *vsb, const struct symtab *st) |
| 110 |
|
{ |
| 111 |
25661600 |
if (st->parent != NULL && st->parent->parent != NULL) { |
| 112 |
11443400 |
vcc_symtabname(vsb, st->parent); |
| 113 |
11443400 |
VSB_putc(vsb, '.'); |
| 114 |
11443400 |
} |
| 115 |
25661600 |
VSB_cat(vsb, st->name); |
| 116 |
25661600 |
} |
| 117 |
|
|
| 118 |
|
void |
| 119 |
14218200 |
VCC_SymName(struct vsb *vsb, const struct symbol *sym) |
| 120 |
|
{ |
| 121 |
14218200 |
AN(vsb); |
| 122 |
14218200 |
CHECK_OBJ_NOTNULL(sym, SYMBOL_MAGIC); |
| 123 |
14218200 |
CHECK_OBJ_NOTNULL(sym->symtab, SYMTAB_MAGIC); |
| 124 |
14218200 |
vcc_symtabname(vsb, sym->symtab); |
| 125 |
14218200 |
} |
| 126 |
|
|
| 127 |
|
static struct symtab * |
| 128 |
32104682 |
vcc_symtab_new(const char *name) |
| 129 |
|
{ |
| 130 |
|
struct symtab *st; |
| 131 |
|
|
| 132 |
32104682 |
ALLOC_OBJ(st, SYMTAB_MAGIC); |
| 133 |
32104682 |
AN(st); |
| 134 |
32104682 |
st->name = name; |
| 135 |
32104682 |
st->nlen = strlen(st->name); |
| 136 |
32104682 |
VTAILQ_INIT(&st->children); |
| 137 |
32104682 |
VTAILQ_INIT(&st->symbols); |
| 138 |
32104682 |
return (st); |
| 139 |
|
} |
| 140 |
|
|
| 141 |
|
static struct symtab * |
| 142 |
66975857 |
vcc_symtab_str(struct symtab *st, const char *b, const char *e, unsigned tok) |
| 143 |
|
{ |
| 144 |
|
struct symtab *st2, *st3; |
| 145 |
|
size_t l; |
| 146 |
|
int i; |
| 147 |
|
const char *q; |
| 148 |
|
|
| 149 |
66975857 |
assert(tok == ID || tok == CSTR); |
| 150 |
66975857 |
if (e == NULL) |
| 151 |
26173336 |
e = strchr(b, '\0'); |
| 152 |
66975857 |
q = e; |
| 153 |
|
|
| 154 |
156407805 |
while (b < e) { |
| 155 |
89431948 |
if (tok == ID) { |
| 156 |
773382540 |
for (q = b; q < e && *q != '.'; q++) |
| 157 |
683950992 |
continue; |
| 158 |
89431548 |
} |
| 159 |
89431948 |
l = pdiff(b, q); |
| 160 |
1506750775 |
VTAILQ_FOREACH(st2, &st->children, list) { |
| 161 |
1485098887 |
i = strncasecmp(st2->name, b, l); |
| 162 |
1485098887 |
if (i < 0) |
| 163 |
1417318827 |
continue; |
| 164 |
67780060 |
if (i == 0 && l == st2->nlen) |
| 165 |
57575054 |
break; |
| 166 |
10205006 |
st3 = vcc_symtab_new(vcc_Dup_be(b, q)); |
| 167 |
10205006 |
st3->parent = st; |
| 168 |
10205006 |
VTAILQ_INSERT_BEFORE(st2, st3, list); |
| 169 |
10205006 |
st2 = st3; |
| 170 |
10205006 |
break; |
| 171 |
|
} |
| 172 |
89431948 |
if (st2 == NULL) { |
| 173 |
21651888 |
st2 = vcc_symtab_new(vcc_Dup_be(b, q)); |
| 174 |
21651888 |
st2->parent = st; |
| 175 |
21651888 |
VTAILQ_INSERT_TAIL(&st->children, st2, list); |
| 176 |
21651888 |
} |
| 177 |
89431948 |
st = st2; |
| 178 |
89431948 |
b = q + 1; |
| 179 |
|
} |
| 180 |
66975857 |
return (st); |
| 181 |
|
} |
| 182 |
|
|
| 183 |
|
static struct symbol * |
| 184 |
31044130 |
vcc_new_symbol(struct vcc *tl, struct symtab *st, |
| 185 |
|
vcc_kind_t kind, int vlo, int vhi) |
| 186 |
|
{ |
| 187 |
|
struct symbol *sym; |
| 188 |
|
|
| 189 |
31044130 |
sym = TlAlloc(tl, sizeof *sym); |
| 190 |
31044130 |
INIT_OBJ(sym, SYMBOL_MAGIC); |
| 191 |
31044130 |
AN(sym); |
| 192 |
31044130 |
sym->name = st->name; |
| 193 |
31044130 |
sym->symtab = st; |
| 194 |
31044130 |
sym->kind = kind; |
| 195 |
31044130 |
sym->type = VOID; |
| 196 |
31044130 |
sym->lorev = vlo; |
| 197 |
31044130 |
sym->hirev = vhi; |
| 198 |
31044130 |
VTAILQ_INSERT_TAIL(&st->symbols, sym, list); |
| 199 |
31044130 |
return (sym); |
| 200 |
|
} |
| 201 |
|
|
| 202 |
|
static struct symbol * |
| 203 |
66976897 |
vcc_sym_in_tab(struct vcc *tl, struct symtab *st, |
| 204 |
|
vcc_kind_t kind, int vlo, int vhi) |
| 205 |
|
{ |
| 206 |
|
const struct symtab *pst; |
| 207 |
|
struct symbol *sym, *psym; |
| 208 |
|
|
| 209 |
67524513 |
VTAILQ_FOREACH(sym, &st->symbols, list) { |
| 210 |
32795882 |
if (sym->lorev > vhi || sym->hirev < vlo) |
| 211 |
547296 |
continue; |
| 212 |
32248586 |
if (kind == SYM_NONE && kind == sym->kind && |
| 213 |
1644558 |
sym->wildcard == NULL) |
| 214 |
0 |
continue; |
| 215 |
32696605 |
if (tl->syntax < VCL_41 && strcmp(sym->name, "default") && |
| 216 |
18381475 |
kind != SYM_NONE && kind != sym->kind && |
| 217 |
448019 |
sym->wildcard == NULL) |
| 218 |
320 |
continue; |
| 219 |
32248266 |
return (sym); |
| 220 |
|
} |
| 221 |
34728631 |
pst = st->parent; |
| 222 |
34728631 |
if (pst == NULL) |
| 223 |
0 |
return (sym); |
| 224 |
34728631 |
psym = VTAILQ_FIRST(&pst->symbols); |
| 225 |
34728631 |
if (psym == NULL) |
| 226 |
20229683 |
return (sym); |
| 227 |
14498948 |
if (psym->wildcard == NULL) |
| 228 |
11707841 |
return (sym); |
| 229 |
|
|
| 230 |
2791107 |
sym = vcc_new_symbol(tl, st, kind, vlo, vhi); |
| 231 |
2791107 |
psym->wildcard(tl, psym, sym); |
| 232 |
2791107 |
if (tl->err) |
| 233 |
80 |
return (NULL); |
| 234 |
2791027 |
return (sym); |
| 235 |
66976897 |
} |
| 236 |
|
|
| 237 |
|
|
| 238 |
|
const struct symxref XREF_NONE[1] = {{"xref_none"}}; |
| 239 |
|
const struct symxref XREF_DEF[1] = {{"xref_def"}}; |
| 240 |
|
const struct symxref XREF_REF[1] = {{"xref_ref"}}; |
| 241 |
|
|
| 242 |
|
const struct symmode SYMTAB_NOERR[1] = {{ |
| 243 |
|
.name = "sym_noerror", |
| 244 |
|
.noerr = 1 |
| 245 |
|
}}; |
| 246 |
|
|
| 247 |
|
const struct symmode SYMTAB_CREATE[1] = {{ |
| 248 |
|
.name = "sym_create" |
| 249 |
|
}}; |
| 250 |
|
|
| 251 |
|
const struct symmode SYMTAB_EXISTING[1] = {{ |
| 252 |
|
.name = "Symbol not found" |
| 253 |
|
}}; |
| 254 |
|
|
| 255 |
|
const struct symmode SYMTAB_PARTIAL[1] = {{ |
| 256 |
|
.name = "Symbol not found", |
| 257 |
|
.partial = 1 |
| 258 |
|
}}; |
| 259 |
|
|
| 260 |
|
const struct symmode SYMTAB_PARTIAL_NOERR[1] = {{ |
| 261 |
|
.name = "Symbol not found", |
| 262 |
|
.partial = 1, |
| 263 |
|
.noerr = 1 |
| 264 |
|
}}; |
| 265 |
|
|
| 266 |
|
struct symbol * |
| 267 |
30966170 |
VCC_SymbolGet(struct vcc *tl, vcc_ns_t ns, vcc_kind_t kind, |
| 268 |
|
const struct symmode *e, const struct symxref *x) |
| 269 |
|
{ |
| 270 |
30966170 |
struct symtab *st, *st2 = NULL; |
| 271 |
30966170 |
struct symbol *sym = NULL, *sym2 = NULL; |
| 272 |
30966170 |
struct token *t0, *tn, *tn1, *tn2 = NULL; |
| 273 |
|
|
| 274 |
30966170 |
AN(tl); |
| 275 |
30966170 |
CHECK_OBJ_NOTNULL(ns, VCC_NAMESPACE_MAGIC); |
| 276 |
30966170 |
AN(ns->name); |
| 277 |
30966170 |
CHECK_OBJ_NOTNULL(kind, KIND_MAGIC); |
| 278 |
30966170 |
AN(e); |
| 279 |
30966170 |
AN(x); |
| 280 |
30966170 |
AN(x->name); |
| 281 |
30966170 |
if (tl->syntax >= VCL_41 && e == SYMTAB_CREATE && kind != SYM_SUB && |
| 282 |
96597 |
vcc_Has_vcl_prefix(tl->t->b)) { |
| 283 |
80 |
VSB_cat(tl->sb, "Symbols named 'vcl_*' are reserved.\nAt:"); |
| 284 |
80 |
vcc_ErrWhere(tl, tl->t); |
| 285 |
80 |
return (NULL); |
| 286 |
|
} |
| 287 |
|
|
| 288 |
30966090 |
st = tl->syms[ns->id]; |
| 289 |
30966090 |
t0 = tl->t; |
| 290 |
30966090 |
tn = tl->t; |
| 291 |
30966090 |
assert(tn->tok == ID); |
| 292 |
40802961 |
while (1) { |
| 293 |
40802961 |
assert(tn->tok == ID || tn->tok == CSTR); |
| 294 |
40802961 |
if (tn->tok == CSTR && tl->syntax < VCL_41) { |
| 295 |
40 |
VSB_cat(tl->sb, |
| 296 |
|
"Quoted headers are available for VCL >= 4.1.\n" |
| 297 |
|
"At:"); |
| 298 |
40 |
vcc_ErrWhere(tl, tn); |
| 299 |
40 |
return (NULL); |
| 300 |
|
} |
| 301 |
40802921 |
if (tn->tok == ID) |
| 302 |
40802521 |
st = vcc_symtab_str(st, tn->b, tn->e, tn->tok); |
| 303 |
|
else |
| 304 |
400 |
st = vcc_symtab_str(st, tn->dec, NULL, tn->tok); |
| 305 |
40802921 |
sym2 = vcc_sym_in_tab(tl, st, kind, tl->syntax, tl->syntax); |
| 306 |
40802921 |
if (sym2 != NULL) { |
| 307 |
34042781 |
sym = sym2; |
| 308 |
34042781 |
st2 = st; |
| 309 |
34042781 |
tn2 = tn; |
| 310 |
34042781 |
} |
| 311 |
40802921 |
tn1 = vcc_PeekTokenFrom(tl, tn); |
| 312 |
40802921 |
if (tn1->tok != '.') |
| 313 |
30965970 |
break; |
| 314 |
9836951 |
tn1 = vcc_PeekTokenFrom(tl, tn1); |
| 315 |
9836951 |
if (tn1->tok == CSTR && (sym == NULL || sym->wildcard == NULL)) |
| 316 |
40 |
break; |
| 317 |
9836911 |
if (tn1->tok != CSTR && tn1->tok != ID) |
| 318 |
40 |
break; |
| 319 |
9836871 |
tn = tn1; |
| 320 |
|
} |
| 321 |
30966050 |
if (sym != NULL && sym->kind == SYM_ALIAS) { |
| 322 |
40 |
assert(ns == SYM_MAIN); |
| 323 |
40 |
st = vcc_symtab_str(tl->syms[ns->id], sym->eval_priv, NULL, ID); |
| 324 |
40 |
AN(st); |
| 325 |
40 |
st2 = st; |
| 326 |
40 |
sym = vcc_sym_in_tab(tl, st, SYM_NONE, sym->lorev, sym->hirev); |
| 327 |
40 |
AN(sym); |
| 328 |
40 |
} |
| 329 |
30966050 |
if (sym != NULL && sym->kind == SYM_VMOD && e->partial) |
| 330 |
40 |
e = SYMTAB_EXISTING; |
| 331 |
30966050 |
if (sym != NULL && e->partial) { |
| 332 |
16216700 |
st = st2; |
| 333 |
16216700 |
tn = tn2; |
| 334 |
30966050 |
} else if (st != st2) { |
| 335 |
6130078 |
sym = NULL; |
| 336 |
6130078 |
} |
| 337 |
30966050 |
if (tl->err || (sym == NULL && e->noerr)) |
| 338 |
3052839 |
return (sym); |
| 339 |
27913211 |
AN(st); |
| 340 |
27913211 |
AN(tn); |
| 341 |
27913211 |
if (sym == NULL && e == SYMTAB_CREATE) |
| 342 |
3076199 |
sym = vcc_new_symbol(tl, st, kind, tl->syntax, tl->syntax); |
| 343 |
27913211 |
tl->t = vcc_PeekTokenFrom(tl, tn); |
| 344 |
27913211 |
if (tl->err) |
| 345 |
0 |
return (NULL); |
| 346 |
27913211 |
if (sym == NULL) { |
| 347 |
1040 |
VSB_printf(tl->sb, "%s: '", e->name); |
| 348 |
1040 |
vcc_PrintTokens(tl, t0, tl->t); |
| 349 |
1040 |
VSB_cat(tl->sb, "'"); |
| 350 |
1040 |
sym = vcc_sym_in_tab(tl, st, kind, VCL_LOW, VCL_HIGH); |
| 351 |
1040 |
if (sym != NULL && sym->kind != SYM_OBJECT && |
| 352 |
400 |
sym->kind != SYM_INSTANCE) { /* XXX: too specific */ |
| 353 |
400 |
VSB_cat(tl->sb, " (Only available when"); |
| 354 |
400 |
if (sym->lorev >= VCL_LOW) |
| 355 |
320 |
VSB_printf(tl->sb, " %.1f <=", .1 * sym->lorev); |
| 356 |
400 |
VSB_cat(tl->sb, " VCL syntax"); |
| 357 |
400 |
if (sym->hirev <= VCL_HIGH) |
| 358 |
80 |
VSB_printf(tl->sb, " <= %.1f", .1 * sym->hirev); |
| 359 |
400 |
VSB_cat(tl->sb, ")"); |
| 360 |
400 |
} |
| 361 |
1040 |
VSB_cat(tl->sb, "\nAt: "); |
| 362 |
1040 |
vcc_ErrWhere2(tl, t0, tl->t); |
| 363 |
1040 |
return (NULL); |
| 364 |
|
} |
| 365 |
27912171 |
if (kind != SYM_NONE && kind != sym->kind && sym->type != DEFAULT) { |
| 366 |
120 |
VSB_cat(tl->sb, "Symbol '"); |
| 367 |
120 |
vcc_PrintTokens(tl, t0, tl->t); |
| 368 |
240 |
VSB_printf(tl->sb, "' has wrong type (%s), expected %s:", |
| 369 |
120 |
sym->kind->name, kind->name); |
| 370 |
120 |
VSB_cat(tl->sb, "\nAt: "); |
| 371 |
120 |
vcc_ErrWhere2(tl, t0, tl->t); |
| 372 |
120 |
if (sym->def_b != NULL) { |
| 373 |
80 |
VSB_cat(tl->sb, "Symbol was defined here: "); |
| 374 |
80 |
vcc_ErrWhere(tl, sym->def_b); |
| 375 |
120 |
} else if (sym->ref_b != NULL) { |
| 376 |
40 |
VSB_cat(tl->sb, "Symbol was declared here: "); |
| 377 |
40 |
vcc_ErrWhere(tl, sym->ref_b); |
| 378 |
40 |
} else { |
| 379 |
0 |
VSB_cat(tl->sb, "Symbol was builtin\n"); |
| 380 |
|
} |
| 381 |
120 |
return (NULL); |
| 382 |
|
} |
| 383 |
27912051 |
if (x == XREF_DEF) { |
| 384 |
4759794 |
if (sym->def_b == NULL) |
| 385 |
4698314 |
sym->def_b = t0; |
| 386 |
4759794 |
sym->ndef++; |
| 387 |
27912051 |
} else if (x == XREF_REF) { |
| 388 |
9131803 |
if (sym->ref_b == NULL) |
| 389 |
6140278 |
sym->ref_b = t0; |
| 390 |
9131803 |
sym->nref++; |
| 391 |
9131803 |
} else { |
| 392 |
14020454 |
assert (x == XREF_NONE); |
| 393 |
|
} |
| 394 |
27912051 |
return (sym); |
| 395 |
30966170 |
} |
| 396 |
|
|
| 397 |
|
static struct symbol * |
| 398 |
146237 |
vcc_TypeSymbol(struct vcc *tl, vcc_ns_t ns, vcc_kind_t kind, vcc_type_t type) |
| 399 |
|
{ |
| 400 |
|
struct token t[1], *t0; |
| 401 |
|
struct symbol *sym; |
| 402 |
|
struct vsb *buf; |
| 403 |
|
|
| 404 |
146237 |
buf = VSB_new_auto(); |
| 405 |
146237 |
AN(buf); |
| 406 |
146237 |
VSB_printf(buf, "%s.%.*s", type->name, PF(tl->t)); |
| 407 |
146237 |
AZ(VSB_finish(buf)); |
| 408 |
|
|
| 409 |
|
/* NB: we create a fake token but errors are handled by the caller. */ |
| 410 |
146237 |
memcpy(t, tl->t, sizeof *t); |
| 411 |
146237 |
t->b = VSB_data(buf); |
| 412 |
146237 |
t->e = t->b + VSB_len(buf); |
| 413 |
|
|
| 414 |
146237 |
t0 = tl->t; |
| 415 |
146237 |
tl->t = t; |
| 416 |
146237 |
sym = VCC_SymbolGet(tl, ns, kind, SYMTAB_NOERR, XREF_NONE); |
| 417 |
146237 |
tl->t = t0; |
| 418 |
146237 |
VSB_destroy(&buf); |
| 419 |
|
|
| 420 |
146237 |
return (sym); |
| 421 |
|
} |
| 422 |
|
|
| 423 |
|
struct symbol * |
| 424 |
146237 |
VCC_TypeSymbol(struct vcc *tl, vcc_kind_t kind, vcc_type_t type) |
| 425 |
|
{ |
| 426 |
|
|
| 427 |
146237 |
if (strchr(type->name, '.') == NULL) |
| 428 |
115357 |
return (vcc_TypeSymbol(tl, SYM_TYPE, kind, type)); |
| 429 |
|
|
| 430 |
|
/* NB: type imported from a VMOD */ |
| 431 |
30880 |
return (vcc_TypeSymbol(tl, SYM_MAIN, kind, type)); |
| 432 |
146237 |
} |
| 433 |
|
|
| 434 |
|
struct symbol * |
| 435 |
26167816 |
VCC_MkSym(struct vcc *tl, const char *b, vcc_ns_t ns, vcc_kind_t kind, |
| 436 |
|
int vlo, int vhi) |
| 437 |
|
{ |
| 438 |
|
struct symtab *st; |
| 439 |
|
struct symbol *sym; |
| 440 |
|
const struct symbol *parent; |
| 441 |
|
|
| 442 |
26167816 |
AN(tl); |
| 443 |
26167816 |
AN(b); |
| 444 |
26167816 |
CHECK_OBJ_NOTNULL(ns, VCC_NAMESPACE_MAGIC); |
| 445 |
26167816 |
CHECK_OBJ_NOTNULL(kind, KIND_MAGIC); |
| 446 |
26167816 |
assert(vlo <= vhi); |
| 447 |
|
|
| 448 |
26167816 |
if (tl->syms[ns->id] == NULL) |
| 449 |
247788 |
tl->syms[ns->id] = vcc_symtab_new(""); |
| 450 |
26167816 |
st = vcc_symtab_str(tl->syms[ns->id], b, NULL, ID); |
| 451 |
26167816 |
AN(st); |
| 452 |
26167816 |
sym = vcc_sym_in_tab(tl, st, kind, vlo, vhi); |
| 453 |
26167816 |
if (sym != NULL) { |
| 454 |
990992 |
assert(sym->kind == SYM_VAR); |
| 455 |
990992 |
parent = sym->eval_priv; |
| 456 |
990992 |
AN(parent); |
| 457 |
990992 |
AN(parent->wildcard); |
| 458 |
990992 |
assert(sym->type == parent->type); |
| 459 |
990992 |
return (sym); |
| 460 |
|
} |
| 461 |
25176824 |
AZ(sym); |
| 462 |
25176824 |
sym = vcc_new_symbol(tl, st, kind, vlo, vhi); |
| 463 |
25176824 |
AN(sym); |
| 464 |
25176824 |
sym->noref = 1; |
| 465 |
25176824 |
return (sym); |
| 466 |
26167816 |
} |
| 467 |
|
|
| 468 |
|
struct symbol * |
| 469 |
5080 |
VCC_MkSymAlias(struct vcc *tl, const char *alias, const char *name) |
| 470 |
|
{ |
| 471 |
|
struct symbol *sym_alias, *sym; |
| 472 |
|
struct symtab *st; |
| 473 |
|
|
| 474 |
5080 |
AN(tl); |
| 475 |
5080 |
AN(alias); |
| 476 |
5080 |
AN(name); |
| 477 |
|
|
| 478 |
5080 |
st = vcc_symtab_str(tl->syms[SYM_MAIN->id], name, NULL, ID); |
| 479 |
5080 |
AN(st); |
| 480 |
5080 |
sym = vcc_sym_in_tab(tl, st, SYM_NONE, VCL_LOW, VCL_HIGH); |
| 481 |
5080 |
AN(sym); |
| 482 |
5080 |
assert(sym->kind != SYM_ALIAS); |
| 483 |
10160 |
sym_alias = VCC_MkSym(tl, alias, SYM_MAIN, SYM_ALIAS, sym->lorev, |
| 484 |
5080 |
sym->hirev); |
| 485 |
5080 |
AN(sym_alias); |
| 486 |
5080 |
sym_alias->eval_priv = strdup(name); |
| 487 |
5080 |
AN(sym_alias->eval_priv); |
| 488 |
5080 |
return (sym); |
| 489 |
|
} |
| 490 |
|
|
| 491 |
|
static void |
| 492 |
155360656 |
vcc_walksymbols(struct vcc *tl, const struct symtab *root, |
| 493 |
|
symwalk_f *func, vcc_kind_t kind) |
| 494 |
|
{ |
| 495 |
|
struct symbol *sym; |
| 496 |
155360656 |
struct symtab *st1, *st2 = NULL; |
| 497 |
|
|
| 498 |
305938836 |
VTAILQ_FOREACH(sym, &root->symbols, list) { |
| 499 |
150580340 |
if (kind == SYM_NONE || kind == sym->kind) |
| 500 |
68480766 |
func(tl, sym); |
| 501 |
150580340 |
ERRCHK(tl); |
| 502 |
150578180 |
} |
| 503 |
310011324 |
VTAILQ_FOREACH(st1, &root->children, list) { |
| 504 |
154654988 |
if (st2 != NULL) |
| 505 |
141235988 |
assert(strcasecmp(st1->name, st2->name) >= 0); |
| 506 |
154654988 |
st2 = st1; |
| 507 |
154654988 |
vcc_walksymbols(tl, st1, func, kind); |
| 508 |
154654988 |
ERRCHK(tl); |
| 509 |
154652828 |
} |
| 510 |
155360656 |
} |
| 511 |
|
|
| 512 |
|
void |
| 513 |
705668 |
VCC_WalkSymbols(struct vcc *tl, symwalk_f *func, vcc_ns_t ns, vcc_kind_t kind) |
| 514 |
|
{ |
| 515 |
|
|
| 516 |
705668 |
CHECK_OBJ_NOTNULL(ns, VCC_NAMESPACE_MAGIC); |
| 517 |
705668 |
vcc_walksymbols(tl, tl->syms[ns->id], func, kind); |
| 518 |
705668 |
} |
| 519 |
|
|
| 520 |
|
void |
| 521 |
6026401 |
VCC_GlobalSymbol(struct symbol *sym, vcc_type_t type) |
| 522 |
|
{ |
| 523 |
|
vcc_kind_t kind; |
| 524 |
|
struct vsb *vsb; |
| 525 |
|
|
| 526 |
6026401 |
CHECK_OBJ_NOTNULL(sym, SYMBOL_MAGIC); |
| 527 |
6026401 |
AN(type); |
| 528 |
6026401 |
AN(type->global_pfx); |
| 529 |
|
|
| 530 |
6026401 |
kind = VCC_HandleKind(type); |
| 531 |
|
|
| 532 |
6026401 |
if (sym->lname != NULL) { |
| 533 |
2974322 |
AN(sym->rname); |
| 534 |
2974322 |
assert(sym->type == type); |
| 535 |
2974322 |
assert(sym->kind == kind); |
| 536 |
2974322 |
return; |
| 537 |
|
} |
| 538 |
|
|
| 539 |
3052079 |
vsb = VSB_new_auto(); |
| 540 |
3052079 |
AN(vsb); |
| 541 |
3052079 |
VSB_printf(vsb, "%s_", type->global_pfx); |
| 542 |
3052079 |
VCC_PrintCName(vsb, sym->name, NULL); |
| 543 |
3052079 |
AZ(VSB_finish(vsb)); |
| 544 |
3052079 |
sym->lname = strdup(VSB_data(vsb)); |
| 545 |
3052079 |
AN(sym->lname); |
| 546 |
3052079 |
if (type == SUB) { |
| 547 |
2977482 |
VSB_clear(vsb); |
| 548 |
2977482 |
VSB_printf(vsb, "sub_%s", sym->lname); |
| 549 |
2977482 |
AZ(VSB_finish(vsb)); |
| 550 |
2977482 |
} |
| 551 |
3052079 |
sym->rname = strdup(VSB_data(vsb)); |
| 552 |
3052079 |
AN(sym->rname); |
| 553 |
3052079 |
VSB_destroy(&vsb); |
| 554 |
|
|
| 555 |
3052079 |
sym->type = type; |
| 556 |
3052079 |
sym->kind = kind; |
| 557 |
3052079 |
if (sym->kind != SYM_NONE) { |
| 558 |
3052079 |
AZ(VCT_invalid_name(sym->rname, NULL)); |
| 559 |
3052079 |
if (type == SUB) |
| 560 |
2977482 |
sym->eval = vcc_Eval_Sub; |
| 561 |
|
else |
| 562 |
74597 |
sym->eval = vcc_Eval_Handle; |
| 563 |
3052079 |
} else { |
| 564 |
0 |
WRONG("Wrong kind of global symbol"); |
| 565 |
|
} |
| 566 |
|
|
| 567 |
|
#define VCL_MET_MAC(l,u,t,b) sym->r_methods |= VCL_MET_##u; |
| 568 |
|
#include "tbl/vcl_returns.h" |
| 569 |
|
} |
| 570 |
|
|
| 571 |
|
struct symbol * |
| 572 |
78077 |
VCC_HandleSymbol(struct vcc *tl, vcc_type_t fmt) |
| 573 |
|
{ |
| 574 |
|
struct symbol *sym; |
| 575 |
|
vcc_kind_t kind; |
| 576 |
|
struct token *t; |
| 577 |
|
const char *p; |
| 578 |
|
|
| 579 |
78077 |
if (vcc_IdIs(tl->t, "default") && fmt->default_sym != NULL) { |
| 580 |
3400 |
vcc_NextToken(tl); |
| 581 |
3400 |
return (fmt->default_sym); |
| 582 |
|
} |
| 583 |
|
|
| 584 |
74677 |
kind = VCC_HandleKind(fmt); |
| 585 |
74677 |
assert(kind != SYM_NONE); |
| 586 |
|
|
| 587 |
74677 |
t = tl->t; |
| 588 |
74677 |
sym = VCC_SymbolGet(tl, SYM_MAIN, SYM_NONE, SYMTAB_NOERR, XREF_NONE); |
| 589 |
74677 |
if (sym != NULL && sym->def_b != NULL && kind == sym->kind) { |
| 590 |
120 |
p = sym->kind->name; |
| 591 |
240 |
VSB_printf(tl->sb, "%c%s '%.*s' redefined.\n", |
| 592 |
120 |
toupper(*p), p + 1, PF(t)); |
| 593 |
120 |
vcc_ErrWhere(tl, t); |
| 594 |
120 |
VSB_cat(tl->sb, "First definition:\n"); |
| 595 |
120 |
AN(sym->def_b); |
| 596 |
120 |
vcc_ErrWhere(tl, sym->def_b); |
| 597 |
120 |
return (sym); |
| 598 |
74557 |
} else if (sym != NULL && sym->def_b != NULL) { |
| 599 |
80 |
VSB_printf(tl->sb, "Name '%.*s' already defined.\n", PF(t)); |
| 600 |
80 |
vcc_ErrWhere(tl, t); |
| 601 |
80 |
VSB_cat(tl->sb, "First definition:\n"); |
| 602 |
80 |
AN(sym->def_b); |
| 603 |
80 |
vcc_ErrWhere(tl, sym->def_b); |
| 604 |
80 |
return (sym); |
| 605 |
74477 |
} else if (sym != NULL && sym->kind != kind) { |
| 606 |
80 |
VSB_printf(tl->sb, |
| 607 |
|
"Name %.*s must have type '%s'.\n", |
| 608 |
40 |
PF(t), sym->kind->name); |
| 609 |
40 |
vcc_ErrWhere(tl, t); |
| 610 |
40 |
return (sym); |
| 611 |
|
} |
| 612 |
74437 |
if (sym == NULL) |
| 613 |
74357 |
sym = VCC_SymbolGet(tl, SYM_MAIN, kind, SYMTAB_CREATE, |
| 614 |
|
XREF_NONE); |
| 615 |
74437 |
if (sym == NULL) |
| 616 |
0 |
return (NULL); |
| 617 |
74437 |
AN(sym); |
| 618 |
74437 |
AZ(sym->ndef); |
| 619 |
74437 |
VCC_GlobalSymbol(sym, fmt); |
| 620 |
74437 |
sym->ndef = 1; |
| 621 |
74437 |
if (sym->def_b == NULL) |
| 622 |
74437 |
sym->def_b = t; |
| 623 |
74437 |
return (sym); |
| 624 |
78077 |
} |
| 625 |
|
/*lint -restore */ |