varnish-cache/lib/libvcc/vcc_compile.c
0
/*-
1
 * Copyright (c) 2006 Verdens Gang AS
2
 * Copyright (c) 2006-2015 Varnish Software AS
3
 * All rights reserved.
4
 *
5
 * Author: Poul-Henning Kamp <phk@phk.freebsd.dk>
6
 *
7
 * SPDX-License-Identifier: BSD-2-Clause
8
 *
9
 * Redistribution and use in source and binary forms, with or without
10
 * modification, are permitted provided that the following conditions
11
 * are met:
12
 * 1. Redistributions of source code must retain the above copyright
13
 *    notice, this list of conditions and the following disclaimer.
14
 * 2. Redistributions in binary form must reproduce the above copyright
15
 *    notice, this list of conditions and the following disclaimer in the
16
 *    documentation and/or other materials provided with the distribution.
17
 *
18
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
19
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21
 * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
22
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28
 * SUCH DAMAGE.
29
 */
30
31
/*
32
 * XXX:
33
 *      Better error messages, throughout.
34
 *      >It also occurred to me that we could link the errors to the error
35
 *      >documentation.
36
 *      >
37
 *      >Unreferenced  function 'request_policy', first mention is
38
 *      >         Line 8 Pos 4
39
 *      >         sub request_policy {
40
 *      >         ----##############--
41
 *      >Read more about this type of error:
42
 *      >http://varnish/doc/error.html#Unreferenced%20function
43
 *      >
44
 *      >
45
 *      >         Unknown variable 'obj.bandwidth'
46
 *      >         At: Line 88 Pos 12
47
 *      >                 if (obj.bandwidth < 1 kb/h) {
48
 *      >         ------------#############------------
49
 *      >Read more about this type of error:
50
 *      >http://varnish/doc/error.html#Unknown%20variable
51
 *
52
 */
53
54
#include "config.h"
55
56
#include <fcntl.h>
57
#include <stdarg.h>
58
#include <stdlib.h>
59
#include <string.h>
60
#include <unistd.h>
61
62
#include "vcc_compile.h"
63
64
#include "libvcc.h"
65
#include "vfil.h"
66
#include "vct.h"
67
68
static const struct method method_tab[] = {
69
        { "none", 0U, 0},
70
#define VCL_MET_MAC(l,U,t,b)    { "vcl_"#l, b, VCL_MET_##U },
71
#include "tbl/vcl_returns.h"
72
        { NULL, 0U, 0}
73
};
74
75
struct vcc *vcc_builtin;
76
77
/*--------------------------------------------------------------------*/
78
79
static void
80 414744
vcc_vcl_met2c(struct vsb *vsb, unsigned method)
81
{
82 414744
        int d = 0;
83
84
        //lint -e{774} Boolean within 'if' always evaluates to False
85
#define VCL_MET_MAC(l,U,t,b)                            \
86
        do {                                            \
87
                if (method & VCL_MET_##U) {             \
88
                        if (d)                          \
89
                                VSB_putc(vsb, '|');     \
90
                        VSB_cat(vsb, "VCL_MET_" #U);    \
91
                        d = 1;                          \
92
                }                                       \
93
        } while (0);
94
#include "tbl/vcl_returns.h"
95 414744
        AN(d);
96
}
97
98
99
/*--------------------------------------------------------------------*/
100
101
void * v_matchproto_(TlAlloc)
102 33634368
TlAlloc(struct vcc *tl, unsigned len)
103
{
104
        void *p;
105
106 33634368
        (void)tl;
107 33634368
        p = calloc(1, len);
108 33634368
        assert(p != NULL);
109 33634368
        return (p);
110
}
111
112
char *
113 1738144
TlDup(struct vcc *tl, const char *s)
114
{
115
        char *p;
116
117 1738144
        p = TlAlloc(tl, strlen(s) + 1);
118 1738144
        AN(p);
119 1738144
        strcpy(p, s);
120 1738144
        return (p);
121
}
122
123
static int
124 20208
TLWriteVSB(struct vcc *tl, const char *fn, const struct vsb *vsb,
125
    const char *what)
126
{
127
        int fo;
128 414744
        int i;
129
130 20208
        fo = open(fn, O_WRONLY|O_TRUNC|O_CREAT, 0600);
131 20208
        if (fo < 0) {
132 0
                VSB_printf(tl->sb,
133
                    "Could not open %s file %s: %s\n",
134 0
                    what, fn, strerror(errno));
135 414744
                return (-1);
136
        }
137 20208
        i = VSB_tofile(vsb, fo);
138 20208
        if (i) {
139 0
                VSB_printf(tl->sb,
140
                    "Could not write %s to %s: %s\n",
141 0
                    what, fn, strerror(errno));
142 414744
        }
143 20208
        closefd(&fo);
144 20208
        return (i);
145 20208
}
146
147
/*--------------------------------------------------------------------*/
148
149
static void
150 104
vcc_refUnusedProbes(struct vcc *tl, const struct symbol *sym)
151
{
152 414848
        if (sym->ndef != 0 && sym->nref == 0)
153 8
                Fc(tl, 0, "\t(void)%s;\n", sym->rname);
154 104
}
155
156
struct proc *
157 974064
vcc_NewProc(struct vcc *tl, struct symbol *sym)
158
{
159
        struct proc *p;
160
161 1388808
        ALLOC_OBJ(p, PROC_MAGIC);
162 974064
        AN(p);
163 974064
        VTAILQ_INIT(&p->calls);
164 974064
        VTAILQ_INIT(&p->uses);
165 974064
        VTAILQ_INSERT_TAIL(&tl->procs, p, list);
166 974064
        p->prologue = VSB_new_auto();
167 974064
        AN(p->prologue);
168 1388808
        p->body = VSB_new_auto();
169 974064
        AN(p->body);
170 974064
        p->cname = VSB_new_auto();
171 974064
        AN(p->cname);
172 1388808
        p->okmask = VCL_MET_TASK_ALL;
173 974064
        sym->proc = p;
174 974064
        p->sym = sym;
175 974064
        return (p);
176
}
177 414744
178
static void
179 414744
vcc_EmitProc(struct vcc *tl, struct proc *p)
180
{
181
        struct vsb *vsbm;
182
        unsigned mask, nsub;
183
        const char *maskcmp;
184
        const char *cc_adv;
185 829488
        int dyn = (p->sym->nref > p->called);
186
187 414744
        AN(p->okmask);
188 414744
        AZ(VSB_finish(p->cname));
189 414744
        AZ(VSB_finish(p->prologue));
190 829488
        AZ(VSB_finish(p->body));
191 414744
        AN(p->sym);
192
193 414744
        if (p->method) {
194 151560
                mask = p->method->bitval;
195 151560
                maskcmp = "==";
196 151560
        } else {
197 263184
                mask = p->okmask;
198 677928
                maskcmp = "&";
199
        }
200
201 414744
        if (dyn == 0 && (p->calledfrom & VCL_MET_TASK_H) == p->calledfrom)
202 20304
                cc_adv = "v_dont_optimize ";
203
        else
204 394440
                cc_adv = "";
205 414744
206 414744
        nsub = tl->nsub++;
207
208 414744
        Fh(tl, 1, "vcl_func_f %s;\n", VSB_data(p->cname));
209 829488
        Fh(tl, 1, "extern const struct vcl_sub sub_%s[1];\n",
210 414744
            VSB_data(p->cname));
211 829488
        Fh(tl, 1, "const struct vcl_sub sub_%s[1] = {{\n", VSB_data(p->cname));
212 414744
        Fh(tl, 1, "\t.magic\t\t= VCL_SUB_MAGIC,\n");
213 414744
        Fh(tl, 1, "\t.methods\t= 0x%x,\n", p->okmask);
214 414744
        Fh(tl, 1, "\t.name\t\t= \"%.*s\",\n", PF(p->name));
215 414744
        Fh(tl, 1, "\t.vcl_conf\t= &VCL_conf,\n");
216 414744
        Fh(tl, 1, "\t.func\t\t= %s,\n", VSB_data(p->cname));
217 829488
        Fh(tl, 1, "\t.n\t\t= %d,\n", nsub);
218 414744
        Fh(tl, 1, "\t.nref\t\t= %d,\n", p->sym->nref);
219 414744
        Fh(tl, 1, "\t.called\t\t= %d\n", p->called);
220 414744
        Fh(tl, 1, "\t// calledfrom\t  0x%x\n", p->calledfrom);
221 414744
        Fh(tl, 1, "}};\n");
222
223 414744
        if (dyn) {
224 248
                Fc(tl, 1, "\nstatic inline void %s\n", cc_adv);
225 248
                Fc(tl, 1, "%s_checked(VRT_CTX)\n{\n", VSB_data(p->cname));
226 248
        } else {
227 414496
                Fc(tl, 1, "\nvoid %sv_matchproto_(vcl_func_f)\n", cc_adv);
228 1243736
                Fc(tl, 1, "%s(VRT_CTX, enum vcl_func_call_e call,\n",
229 414496
                    VSB_data(p->cname));
230 414496
                Fc(tl, 1, "    enum vcl_func_fail_e *failp)\n{\n");
231 414496
                Fc(tl, 1, "  assert(call == VSUB_STATIC);\n");
232 414496
                Fc(tl, 1, "  assert(failp == NULL);\n");
233
        }
234
235 414744
        vsbm = VSB_new_auto();
236 414744
        AN(vsbm);
237 414744
        vcc_vcl_met2c(vsbm, mask);
238 414744
        AZ(VSB_finish(vsbm));
239 414744
        Fc(tl, 1, "  assert(ctx->method %s (%s));\n", maskcmp, VSB_data(vsbm));
240 414744
        VSB_destroy(&vsbm);
241 414744
        Fc(tl, 1, "%s\n%s}\n", VSB_data(p->prologue), VSB_data(p->body));
242 414744
        VSB_destroy(&p->body);
243 414744
        VSB_destroy(&p->prologue);
244
245 414744
        if (! dyn) {
246 414496
                VSB_destroy(&p->cname);
247 414496
                return;
248
        }
249
250
        /* wrapper to call the actual (_checked) function */
251 248
        Fc(tl, 1, "\nvoid v_matchproto_(vcl_func_f)\n");
252 496
        Fc(tl, 1, "%s(VRT_CTX, enum vcl_func_call_e call,\n",
253 248
            VSB_data(p->cname));
254 248
        Fc(tl, 1, "    enum vcl_func_fail_e *failp)\n{\n");
255 248
        Fc(tl, 1, "  enum vcl_func_fail_e fail;\n\n");
256 496
        Fc(tl, 1, "  fail = VPI_Call_Check(ctx, &VCL_conf, 0x%x, %d);\n",
257 248
            mask, nsub);
258 248
        Fc(tl, 1, "  if (failp)\n");
259 248
        Fc(tl, 1, "    *failp = fail;\n");
260 248
        Fc(tl, 1, "  else if (fail == VSUB_E_METHOD)\n");
261 496
        Fc(tl, 1, "    VRT_fail(ctx, \"call to \\\"sub %.*s{}\\\""
262 248
            " not allowed from here\");\n", PF(p->name));
263 248
        Fc(tl, 1, "  else if (fail == VSUB_E_RECURSE)\n");
264 496
        Fc(tl, 1, "    VRT_fail(ctx, \"Recursive call to "
265 248
            "\\\"sub %.*s{}\\\"\");\n", PF(p->name));
266 248
        Fc(tl, 1, "  else\n");
267 248
        Fc(tl, 1, "    assert (fail == VSUB_E_OK);\n");
268 248
        Fc(tl, 1, "  if (fail != VSUB_E_OK || call == VSUB_CHECK)\n");
269 248
        Fc(tl, 1, "    return;\n");
270 248
        Fc(tl, 1, "  VPI_Call_Begin(ctx, %d);\n", nsub);
271 248
        Fc(tl, 1, "  %s_checked(ctx);\n", VSB_data(p->cname));
272 248
        Fc(tl, 1, "  VPI_Call_End(ctx, %d);\n", nsub);
273 248
        Fc(tl, 1, "}\n");
274 248
        VSB_destroy(&p->cname);
275 414744
}
276
277
/*--------------------------------------------------------------------*/
278
279
struct inifin *
280 126200
New_IniFin(struct vcc *tl)
281
{
282
        struct inifin *p;
283
284 126200
        ALLOC_OBJ(p, INIFIN_MAGIC);
285 126200
        AN(p);
286 126200
        p->ini = VSB_new_auto();
287 126200
        AN(p->ini);
288 126200
        p->fin = VSB_new_auto();
289 126200
        AN(p->fin);
290 126200
        p->final = VSB_new_auto();
291 126200
        AN(p->final);
292 126200
        p->event = VSB_new_auto();
293 126200
        AN(p->event);
294 126200
        p->n = ++tl->ninifin;
295 126200
        VTAILQ_INSERT_TAIL(&tl->inifin, p, list);
296 126200
        return (p);
297
}
298
299
/*--------------------------------------------------------------------
300
 * Printf output to the vsbs, possibly indented
301
 */
302
303
void
304 7380872
Fh(const struct vcc *tl, int indent, const char *fmt, ...)
305
{
306
        va_list ap;
307
308 7380872
        if (indent)
309 5401808
                VSB_printf(tl->fh, "%*.*s", tl->hindent, tl->hindent, "");
310 7380872
        va_start(ap, fmt);
311 7380872
        VSB_vprintf(tl->fh, fmt, ap);
312 7380872
        va_end(ap);
313 7380872
}
314
315
void
316 18790336
Fb(const struct vcc *tl, int indent, const char *fmt, ...)
317
{
318
        va_list ap;
319
320 18790336
        assert(tl->fb != NULL);
321 18790336
        if (indent)
322 16567984
                VSB_printf(tl->fb, "%*.*s", tl->indent, tl->indent, "");
323 18790336
        va_start(ap, fmt);
324 18790336
        VSB_vprintf(tl->fb, fmt, ap);
325 18790336
        va_end(ap);
326 18790336
}
327
328
void
329 15092872
Fc(const struct vcc *tl, int indent, const char *fmt, ...)
330
{
331
        va_list ap;
332
333 15092872
        if (indent)
334 2907176
                VSB_printf(tl->fc, "%*.*s", tl->indent, tl->indent, "");
335 15092872
        va_start(ap, fmt);
336 15092872
        VSB_vprintf(tl->fc, fmt, ap);
337 15092872
        va_end(ap);
338 15092872
}
339
340
/*--------------------------------------------------------------------*/
341
342
void
343 858224
EncToken(struct vsb *sb, const struct token *t)
344
{
345
346 858224
        assert(t->tok == CSTR);
347 858224
        VSB_quote(sb, t->dec, -1, VSB_QUOTE_CSTR);
348 858224
}
349
350
/*--------------------------------------------------------------------
351
 * Output the location/profiling table.  For each counted token, we
352
 * record source+line+charpos for the first character in the token.
353
 */
354
355
static void
356 10104
EmitCoordinates(const struct vcc *tl, struct vsb *vsb)
357
{
358
        struct token *t;
359
        unsigned lin, pos;
360
        const struct source *sp;
361
        const char *p;
362
363 10104
        VSB_cat(vsb, "/* ---===### Source Code ###===---*/\n");
364
365 10104
        VSB_printf(vsb, "\n#define VGC_NSRCS %u\n", tl->nsources);
366
367 10104
        VSB_cat(vsb, "\nstatic const char *srcname[VGC_NSRCS] = {\n");
368 30560
        VTAILQ_FOREACH(sp, &tl->sources, list) {
369 20456
                VSB_cat(vsb, "\t");
370 20456
                VSB_quote(vsb, sp->name, -1, VSB_QUOTE_CSTR);
371 20456
                VSB_cat(vsb, ",\n");
372 20456
        }
373 10104
        VSB_cat(vsb, "};\n");
374
375 10104
        VSB_printf(vsb, "\nstatic const char *srcbody[%u] = {\n", tl->nsources);
376 30560
        VTAILQ_FOREACH(sp, &tl->sources, list) {
377 20456
                VSB_cat(vsb, "    /* ");
378 20456
                VSB_quote(vsb, sp->name, -1, VSB_QUOTE_CSTR);
379 20456
                VSB_cat(vsb, " */\n");
380 20456
                VSB_quote_pfx(vsb, "\t", sp->b, sp->e - sp->b, VSB_QUOTE_CSTR);
381 20456
                VSB_cat(vsb, ",\n");
382 20456
        }
383 10104
        VSB_cat(vsb, "};\n\n");
384
385 10104
        VSB_cat(vsb, "/* ---===### Location Counters ###===---*/\n");
386
387 10104
        VSB_printf(vsb, "\n#define VGC_NREFS %u\n\n", tl->cnt + 1);
388
389 10104
        VSB_cat(vsb, "static const struct vpi_ref VGC_ref[VGC_NREFS] = {\n");
390 10104
        lin = 1;
391 10104
        pos = 0;
392 10104
        sp = 0;
393 10104
        p = NULL;
394 9613576
        VTAILQ_FOREACH(t, &tl->tokens, list) {
395 9603472
                if (t->cnt == 0)
396 8820528
                        continue;
397 782944
                assert(t->src != NULL);
398 782944
                if (t->src != sp) {
399 17160
                        lin = 1;
400 17160
                        pos = 0;
401 17160
                        sp = t->src;
402 17160
                        p = sp->b;
403 17160
                }
404 782944
                assert(sp != NULL);
405 782944
                assert(p != NULL);
406 76758026
                for (;p < t->b; p++) {
407 75975082
                        if (*p == '\n') {
408 3410072
                                lin++;
409 3410072
                                pos = 0;
410 75975082
                        } else if (*p == '\t') {
411 1736016
                                pos &= ~7;
412 1736016
                                pos += 8;
413 1736016
                        } else
414 70828994
                                pos++;
415
416 75975082
                }
417 1565888
                VSB_printf(vsb, "  [%3u] = { VPI_REF_MAGIC, %u, %8tu, %4u, %3u, ",
418 782944
                    t->cnt, sp->idx, t->b - sp->b, lin, pos + 1);
419 782944
                if (t->tok == CSRC)
420 56
                        VSB_cat(vsb, " \"C{\"},\n");
421
                else
422 782888
                        VSB_printf(vsb, " \"%.*s\" },\n", PF(t));
423 782944
        }
424 10104
        VSB_cat(vsb, "};\n\n");
425 10104
}
426
427
/*--------------------------------------------------------------------
428
 * Init/Fini/Event
429
 *
430
 * We call DISCARD and COLD events in the opposite order of LOAD and
431
 * WARM.
432
 */
433
434
static void
435 10104
EmitInitFini(struct vcc *tl)
436
{
437 10104
        struct inifin *p, *q = NULL;
438 10104
        unsigned has_event = 0;
439
        struct symbol *sy;
440
441 10104
        Fh(tl, 0, "\n");
442 10104
        Fh(tl, 0, "static unsigned vgc_inistep;\n");
443 10104
        Fh(tl, 0, "static unsigned vgc_warmupstep;\n");
444
445
        /*
446
         * LOAD
447
         */
448 10104
        Fc(tl, 0, "\nstatic int\nVGC_Load(VRT_CTX)\n{\n\n");
449 10104
        Fc(tl, 0, "\tvgc_inistep = 0;\n");
450 10104
        Fc(tl, 0, "\tsize_t ndirector = %dUL;\n", tl->ndirector);
451 10104
        Fc(tl, 0, "\n");
452 80800
        VTAILQ_FOREACH(p, &tl->inifin, list) {
453 70696
                AZ(VSB_finish(p->ini));
454 70696
                assert(p->n > 0);
455 70696
                if (VSB_len(p->ini))
456 68968
                        Fc(tl, 0, "\t/* %u */\n%s\n", p->n, VSB_data(p->ini));
457 70696
                if (p->ignore_errors == 0) {
458 60592
                        Fc(tl, 0, "\tif (ctx->vpi->handling == VCL_RET_FAIL)\n");
459 60592
                        Fc(tl, 0, "\t\treturn(1);\n");
460 60592
                }
461 70696
                Fc(tl, 0, "\tvgc_inistep = %u;\n\n", p->n);
462 70696
                VSB_destroy(&p->ini);
463
464 70696
                AZ(VSB_finish(p->event));
465 70696
                if (VSB_len(p->event))
466 872
                        has_event = 1;
467 70696
        }
468
469
        /* Handle failures from vcl_init */
470 10104
        Fc(tl, 0, "\n");
471 10104
        Fc(tl, 0, "\tif (ctx->vpi->handling != VCL_RET_OK)\n");
472 10104
        Fc(tl, 0, "\t\treturn(1);\n");
473 10104
        Fc(tl, 0, "\tctx->vpi->handling = 0;\n");
474
475 11624
        VTAILQ_FOREACH(sy, &tl->sym_objects, sideways) {
476 1520
                Fc(tl, 0, "\tif (!%s) {\n", sy->rname);
477 3040
                Fc(tl, 0, "\t\tVRT_fail(ctx, "
478 1520
                    "\"Object %s not initialized\");\n" , sy->name);
479 1520
                Fc(tl, 0, "\t\treturn(1);\n");
480 1520
                Fc(tl, 0, "\t}\n");
481 1520
        }
482
483 10104
        Fc(tl, 0, "\treturn(0);\n");
484 10104
        Fc(tl, 0, "}\n");
485
486
        /*
487
         * DISCARD
488
         */
489 10104
        Fc(tl, 0, "\nstatic int\nVGC_Discard(VRT_CTX)\n{\n\n");
490
491 10104
        Fc(tl, 0, "\tswitch (vgc_inistep) {\n");
492 80800
        VTAILQ_FOREACH_REVERSE(p, &tl->inifin, inifinhead, list) {
493 70696
                AZ(VSB_finish(p->fin));
494 70696
                if (q)
495 60592
                        assert(q->n > p->n);
496 70696
                q = p;
497 70696
                Fc(tl, 0, "\t\tcase %u:\n", p->n);
498 70696
                if (VSB_len(p->fin))
499 70680
                        Fc(tl, 0, "\t%s\n", VSB_data(p->fin));
500 70696
                Fc(tl, 0, "\t\t\t/* FALLTHROUGH */\n");
501 70696
                VSB_destroy(&p->fin);
502 70696
        }
503 10104
        Fc(tl, 0, "\t\tdefault:\n\t\t\tbreak;\n");
504 10104
        Fc(tl, 0, "\t}\n\n");
505 10104
        Fc(tl, 0, "\tswitch (vgc_inistep) {\n");
506 80800
        VTAILQ_FOREACH_REVERSE(p, &tl->inifin, inifinhead, list) {
507 70696
                AZ(VSB_finish(p->final));
508 70696
                Fc(tl, 0, "\t\tcase %u:\n", p->n);
509 70696
                if (VSB_len(p->final))
510 4112
                        Fc(tl, 0, "\t%s\n", VSB_data(p->final));
511 70696
                Fc(tl, 0, "\t\t\t/* FALLTHROUGH */\n");
512 70696
                VSB_destroy(&p->final);
513 70696
        }
514 10104
        Fc(tl, 0, "\t\tdefault:\n\t\t\tbreak;\n");
515 10104
        Fc(tl, 0, "\t}\n\n");
516
517 10104
        Fc(tl, 0, "\treturn (0);\n");
518 10104
        Fc(tl, 0, "}\n");
519
520 10104
        if (has_event) {
521
                /*
522
                 * WARM
523
                 */
524 872
                Fc(tl, 0, "\nstatic int\n");
525 872
                Fc(tl, 0, "VGC_Warmup(VRT_CTX, enum vcl_event_e ev)\n{\n\n");
526
527 872
                Fc(tl, 0, "\tvgc_warmupstep = 0;\n\n");
528 8824
                VTAILQ_FOREACH(p, &tl->inifin, list) {
529 7952
                        assert(p->n > 0);
530 7952
                        if (VSB_len(p->event)) {
531 872
                                Fc(tl, 0, "\t/* %u */\n", p->n);
532 872
                                Fc(tl, 0, "\tif (%s)\n", VSB_data(p->event));
533 872
                                Fc(tl, 0, "\t\treturn (1);\n");
534 872
                                Fc(tl, 0, "\tvgc_warmupstep = %u;\n\n", p->n);
535 872
                        }
536 7952
                }
537
538 872
                Fc(tl, 0, "\treturn (0);\n");
539 872
                Fc(tl, 0, "}\n");
540
541
                /*
542
                 * COLD
543
                 */
544 872
                Fc(tl, 0, "\nstatic int\n");
545 872
                Fc(tl, 0, "VGC_Cooldown(VRT_CTX, enum vcl_event_e ev)\n{\n");
546 872
                Fc(tl, 0, "\tint retval = 0;\n\n");
547
548 8824
                VTAILQ_FOREACH_REVERSE(p, &tl->inifin, inifinhead, list) {
549 7952
                        if (VSB_len(p->event)) {
550 872
                                Fc(tl, 0, "\t/* %u */\n", p->n);
551 1744
                                Fc(tl, 0,
552 872
                                    "\tif (vgc_warmupstep >= %u &&\n", p->n);
553 1744
                                Fc(tl, 0,
554 872
                                    "\t    %s != 0)\n", VSB_data(p->event));
555 872
                                Fc(tl, 0, "\t\tretval = 1;\n\n");
556 872
                        }
557 7952
                        VSB_destroy(&p->event);
558 7952
                }
559
560 872
                Fc(tl, 0, "\treturn (retval);\n");
561 872
                Fc(tl, 0, "}\n");
562 872
        }
563
564
        /*
565
         * EVENTS
566
         */
567 10104
        Fc(tl, 0, "\nstatic int\n");
568 10104
        Fc(tl, 0, "VGC_Event(VRT_CTX, enum vcl_event_e ev)\n");
569 10104
        Fc(tl, 0, "{\n");
570 10104
        Fc(tl, 0, "\tif (ev == VCL_EVENT_LOAD)\n");
571 10104
        Fc(tl, 0, "\t\treturn (VGC_Load(ctx));\n");
572 10104
        if (has_event) {
573 872
                Fc(tl, 0, "\tif (ev == VCL_EVENT_WARM)\n");
574 872
                Fc(tl, 0, "\t\treturn (VGC_Warmup(ctx, ev));\n");
575 872
                Fc(tl, 0, "\tif (ev == VCL_EVENT_COLD)\n");
576 872
                Fc(tl, 0, "\t\treturn (VGC_Cooldown(ctx, ev));\n");
577 872
        }
578 10104
        Fc(tl, 0, "\tif (ev == VCL_EVENT_DISCARD)\n");
579 10104
        Fc(tl, 0, "\t\treturn (VGC_Discard(ctx));\n");
580 10104
        Fc(tl, 0, "\n");
581 10104
        if (!has_event)
582 9232
                Fc(tl, 0, "\t(void)vgc_warmupstep;\n");
583 10104
        VCC_WalkSymbols(tl, vcc_refUnusedProbes, SYM_MAIN, SYM_PROBE);
584 10104
        Fc(tl, 0, "\treturn (%d);\n", has_event ? 1 : 0);
585 10104
        Fc(tl, 0, "}\n");
586 10104
}
587
588
/*--------------------------------------------------------------------*/
589
590
static void
591 10104
EmitStruct(const struct vcc *tl)
592
{
593 10104
        Fc(tl, 0, "\nconst struct VCL_conf VCL_conf = {\n");
594 10104
        Fc(tl, 0, "\t.magic = VCL_CONF_MAGIC,\n");
595 10104
        Fc(tl, 0, "\t.syntax = %u,\n", tl->syntax);
596 10104
        Fc(tl, 0, "\t.event_vcl = VGC_Event,\n");
597 10104
        Fc(tl, 0, "\t.default_director = &%s,\n", tl->default_director);
598 10104
        if (tl->default_probe != NULL)
599 80
                Fc(tl, 0, "\t.default_probe = %s,\n", tl->default_probe);
600 10104
        Fc(tl, 0, "\t.ref = VGC_ref,\n");
601 10104
        Fc(tl, 0, "\t.nref = VGC_NREFS,\n");
602 10104
        Fc(tl, 0, "\t.nsrc = VGC_NSRCS,\n");
603 10104
        Fc(tl, 0, "\t.nsub = %d,\n", tl->subref > 0 ? tl->nsub : 0);
604 10104
        Fc(tl, 0, "\t.srcname = srcname,\n");
605 10104
        Fc(tl, 0, "\t.srcbody = srcbody,\n");
606 10104
        Fc(tl, 0, "\t.nvmod = %u,\n", tl->vmod_count);
607
#define VCL_MET_MAC(l,u,t,b) \
608
        Fc(tl, 0, "\t." #l "_func = VGC_function_vcl_" #l ",\n");
609
#include "tbl/vcl_returns.h"
610
        Fc(tl, 0, "\t.instance_info = VGC_instance_info\n");
611
        Fc(tl, 0, "};\n");
612
}
613
614
/*--------------------------------------------------------------------
615
 * Compile the VCL code from the given source and return the C-source
616
 */
617
618
static struct vsb *
619 24952
vcc_CompileSource(struct vcc *tl, struct source *sp, const char *jfile)
620
{
621
        struct proc *p;
622
        struct vsb *vsb;
623
        struct inifin *ifp;
624
625 24952
        Fh(tl, 0, "/* ---===### VCC generated .h code ###===---*/\n");
626 24952
        Fc(tl, 0, "\n/* ---===### VCC generated .c code ###===---*/\n");
627
628 24952
        Fc(tl, 0, "\n#define END_ if (ctx->vpi->handling) return\n");
629
630 24952
        vcc_Parse_Init(tl);
631
632 24952
        vcc_Expr_Init(tl);
633
634 24952
        vcc_Action_Init(tl);
635
636 24952
        vcc_Backend_Init(tl);
637
638 24952
        vcc_Var_Init(tl);
639
640 24952
        vcc_Type_Init(tl);
641
642 24952
        Fh(tl, 0, "\nextern const struct VCL_conf VCL_conf;\n");
643
644
        /* Register and lex the main source */
645 24952
        if (sp != NULL) {
646 12472
                AN(vcc_builtin);
647 12472
                vcc_lex_source(tl, sp, 0);
648 12472
                if (tl->err)
649 272
                        return (NULL);
650 12200
        }
651
652
        /* Register and lex the builtin VCL */
653 24680
        sp = vcc_new_source(tl->builtin_vcl, "builtin", "<builtin>");
654 24680
        assert(sp != NULL);
655 24680
        vcc_lex_source(tl, sp, 1);
656 24680
        if (tl->err)
657 0
                return (NULL);
658
659
        /* Expand and lex any includes in the token string */
660 24680
        if (tl->err)
661 0
                return (NULL);
662
663
        /* Parse the token string */
664 24680
        tl->t = VTAILQ_FIRST(&tl->tokens);
665 24680
        vcc_Parse(tl);
666 24680
        if (tl->err)
667 1640
                return (NULL);
668
669
        /* Check for orphans */
670 23040
        if (vcc_CheckReferences(tl))
671 80
                return (NULL);
672
673
        /* Check that all action returns are legal */
674 22960
        if (vcc_CheckAction(tl) || tl->err)
675 24
                return (NULL);
676
677
        /* Check that all variable uses are legal */
678 22936
        if (vcc_CheckUses(tl) || tl->err)
679 336
                return (NULL);
680
681 22600
        if (vcc_builtin == NULL)
682 12480
                return (NULL);
683
684
        /* Check if we have any backends at all */
685 10120
        if (tl->default_director == NULL) {
686 16
                VSB_cat(tl->sb,
687
                    "No backends or directors found in VCL program, "
688
                    "at least one is necessary.\n");
689 16
                tl->err = 1;
690 16
                return (NULL);
691
        }
692
693
        /* Tie vcl_init/fini in */
694 10104
        ifp = New_IniFin(tl);
695 10104
        VSB_cat(ifp->ini, "\tVGC_function_vcl_init(ctx, VSUB_STATIC, NULL);\n");
696
        /*
697
         * Because the failure could be half way into vcl_init{} so vcl_fini{}
698
         * must always be called, also on failure.
699
         */
700 10104
        ifp->ignore_errors = 1;
701 10104
        VSB_cat(ifp->fin, "\t\tVGC_function_vcl_fini(ctx, VSUB_STATIC, NULL);\n");
702 10104
        VSB_cat(ifp->fin, "\t\t\tVPI_vcl_fini(ctx);");
703
704
        /* Emit method functions */
705 10104
        Fh(tl, 1, "\n");
706 424848
        VTAILQ_FOREACH(p, &tl->procs, list)
707 677928
                if (p->method == NULL)
708 263184
                        vcc_EmitProc(tl, p);
709 424848
        VTAILQ_FOREACH(p, &tl->procs, list)
710 566304
                if (p->method != NULL)
711 151560
                        vcc_EmitProc(tl, p);
712
713 10104
        EmitInitFini(tl);
714
715 10104
        VCC_InstanceInfo(tl);
716
717 10104
        EmitStruct(tl);
718
719 10104
        VCC_XrefTable(tl);
720
721 10104
        VSB_cat(tl->symtab, "\n]\n");
722 10104
        AZ(VSB_finish(tl->symtab));
723 10104
        if (TLWriteVSB(tl, jfile, tl->symtab, "Symbol table"))
724 0
                return (NULL);
725
726
        /* Combine it all */
727
728 10104
        vsb = VSB_new_auto();
729 10104
        AN(vsb);
730
731 10104
        vcl_output_lang_h(vsb);
732
733 10104
        EmitCoordinates(tl, vsb);
734
735 10104
        AZ(VSB_finish(tl->fh));
736 10104
        VSB_cat(vsb, VSB_data(tl->fh));
737
738 10104
        AZ(VSB_finish(tl->fc));
739 10104
        VSB_cat(vsb, VSB_data(tl->fc));
740
741 10104
        AZ(VSB_finish(vsb));
742 10104
        return (vsb);
743 24952
}
744
745
static struct vcc *
746 12480
vcc_ParseBuiltin(struct vcc *tl)
747
{
748
        struct vcc *tl_builtin;
749
750 12480
        CHECK_OBJ_NOTNULL(tl, VCC_MAGIC);
751 12480
        tl_builtin = VCC_New();
752 12480
        AN(tl_builtin);
753 12480
        VCC_Builtin_VCL(tl_builtin, tl->builtin_vcl);
754 12480
        AZ(vcc_CompileSource(tl_builtin, NULL, NULL));
755 12480
        return (tl_builtin);
756
}
757
758
/*--------------------------------------------------------------------
759
 * Report the range of VCL language we support
760
 */
761
void
762 8464
VCC_VCL_Range(unsigned *lo, unsigned *hi)
763
{
764
765 8464
        AN(lo);
766 8464
        *lo = VCL_LOW;
767 8464
        AN(hi);
768 8464
        *hi = VCL_HIGH;
769 8464
}
770
771
/*--------------------------------------------------------------------
772
 * Compile the VCL code in the argument.  Error messages, if any are
773
 * formatted into the vsb.
774
 */
775
776
int
777 12480
VCC_Compile(struct vcc *tl, struct vsb **sb,
778
    const char *vclsrc, const char *vclsrcfile,
779
    const char *ofile, const char *jfile)
780
{
781
        struct source *sp;
782 12480
        struct vsb *r = NULL;
783 12480
        int retval = 0;
784
785 12480
        CHECK_OBJ_NOTNULL(tl, VCC_MAGIC);
786 12480
        AN(sb);
787 12480
        AN(vclsrcfile);
788 12480
        AN(ofile);
789 12480
        AN(jfile);
790
791 12480
        AZ(vcc_builtin);
792 12480
        vcc_builtin = vcc_ParseBuiltin(tl);
793 12480
        AN(vcc_builtin);
794 12480
        if (vcc_builtin->err) {
795 0
                AZ(VSB_finish(vcc_builtin->sb));
796 0
                *sb = vcc_builtin->sb;
797 0
                return (-1);
798
        }
799
800 12480
        if (vclsrc != NULL)
801 12376
                sp = vcc_new_source(vclsrc, "vcl.inline", vclsrcfile);
802
        else
803 104
                sp = vcc_file_source(tl, vclsrcfile);
804
805 12480
        if (sp != NULL)
806 12472
                r = vcc_CompileSource(tl, sp, jfile);
807
808 12480
        if (r != NULL) {
809 10104
                retval = TLWriteVSB(tl, ofile, r, "C-source");
810 10104
                VSB_destroy(&r);
811 10104
        } else {
812 2376
                retval = -1;
813
        }
814 12480
        AZ(VSB_finish(tl->sb));
815 12480
        *sb = tl->sb;
816 12480
        return (retval);
817 12480
}
818
819
/*--------------------------------------------------------------------
820
 * Allocate a compiler instance
821
 */
822
823
struct vcc *
824 24960
VCC_New(void)
825
{
826
        struct vcc *tl;
827
        struct symbol *sym;
828
        struct proc *p;
829
        struct vsb *vsb;
830
        int i;
831
832 24960
        ALLOC_OBJ(tl, VCC_MAGIC);
833 24960
        AN(tl);
834 24960
        VTAILQ_INIT(&tl->inifin);
835 24960
        VTAILQ_INIT(&tl->tokens);
836 24960
        VTAILQ_INIT(&tl->sources);
837 24960
        VTAILQ_INIT(&tl->procs);
838 24960
        VTAILQ_INIT(&tl->sym_objects);
839 24960
        VTAILQ_INIT(&tl->sym_vmods);
840 24960
        VTAILQ_INIT(&tl->vmod_objects);
841
842 24960
        tl->nsources = 0;
843
844 24960
        tl->symtab = VSB_new_auto();
845 24960
        assert(tl->symtab != NULL);
846 24960
        VSB_cat(tl->symtab, "[\n    {\"version\": 0}");
847
848 24960
        tl->fc = VSB_new_auto();
849 24960
        assert(tl->fc != NULL);
850
851 24960
        tl->fh = VSB_new_auto();
852 24960
        assert(tl->fh != NULL);
853
854 399360
        for (i = 1; i < VCL_MET_MAX; i++) {
855 748800
                sym = VCC_MkSym(tl, method_tab[i].name,
856 374400
                    SYM_MAIN, SYM_SUB, VCL_LOW, VCL_HIGH);
857 374400
                p = vcc_NewProc(tl, sym);
858 374400
                p->method = &method_tab[i];
859
860
                // see also VCC_GlobalSymbol()
861 374400
                vsb = VSB_new_auto();
862 374400
                AN(vsb);
863 374400
                VSB_printf(vsb, "%s_%s", SUB->global_pfx, p->method->name);
864 374400
                AZ(VSB_finish(vsb));
865
866 374400
                AZ(VSB_bcat(p->cname, VSB_data(vsb), VSB_len(vsb)));
867
868 374400
                sym->lname = strdup(VSB_data(vsb));
869 374400
                AN(sym->lname);
870
871 374400
                VSB_clear(vsb);
872 374400
                VSB_printf(vsb, "sub_%s", sym->lname);
873 374400
                AZ(VSB_finish(vsb));
874
875 374400
                sym->rname = strdup(VSB_data(vsb));
876 374400
                AN(sym->rname);
877 374400
                VSB_destroy(&vsb);
878
879 374400
                sym->type = SUB;
880 374400
                sym->kind = VCC_HandleKind(SUB);
881 374400
                AZ(VCT_invalid_name(sym->rname, NULL));
882 374400
                sym->eval = vcc_Eval_Sub;
883 374400
        }
884 24960
        tl->sb = VSB_new_auto();
885 24960
        AN(tl->sb);
886 24960
        return (tl);
887
}
888
889
/*--------------------------------------------------------------------
890
 * Configure builtin VCL source code
891
 */
892
893
void
894 24960
VCC_Builtin_VCL(struct vcc *vcc, const char *str)
895
{
896
897 24960
        CHECK_OBJ_NOTNULL(vcc, VCC_MAGIC);
898 24960
        REPLACE(vcc->builtin_vcl, str);
899 24960
}
900
901
/*--------------------------------------------------------------------
902
 * Configure default VCL source path
903
 */
904
905
void
906 12480
VCC_VCL_path(struct vcc *vcc, const char *str)
907
{
908
909 12480
        CHECK_OBJ_NOTNULL(vcc, VCC_MAGIC);
910 12480
        VFIL_setpath(&vcc->vcl_path, str);
911 12480
}
912
913
/*--------------------------------------------------------------------
914
 * Configure default VMOD path
915
 */
916
917
void
918 12480
VCC_VMOD_path(struct vcc *vcc, const char *str)
919
{
920
921 12480
        CHECK_OBJ_NOTNULL(vcc, VCC_MAGIC);
922 12480
        VFIL_setpath(&vcc->vmod_path, str);
923 12480
}
924
925
/*--------------------------------------------------------------------
926
 * Configure settings
927
 */
928
929
#define VCC_FEATURE_BIT(U, l, d)                                \
930
        void VCC_Opt_ ## l(struct vcc *vcc, unsigned val)       \
931
        {                                                       \
932
                CHECK_OBJ_NOTNULL(vcc, VCC_MAGIC);              \
933
                vcc->l = val;                                   \
934
        }
935
#include "tbl/vcc_feature_bits.h"
936
937
/*--------------------------------------------------------------------
938
 * Configure settings
939
 */
940
941
static void
942 336
vcc_predef_vcl(struct vcc *vcc, const char *name)
943
{
944
        struct symbol *sym;
945
946 336
        sym = VCC_MkSym(vcc, name, SYM_MAIN, SYM_VCL, VCL_LOW, VCL_HIGH);
947 336
        AN(sym);
948 336
        sym->type = VCL;
949 336
        sym->r_methods = VCL_MET_RECV;
950 336
}
951
952
void
953 25336
VCC_Predef(struct vcc *vcc, const char *type, const char *name)
954
{
955
956 25336
        CHECK_OBJ_NOTNULL(vcc, VCC_MAGIC);
957 25336
        if (!strcmp(type, "VCL_STEVEDORE"))
958 25000
                vcc_stevedore(vcc, name);
959 336
        else if (!strcmp(type, "VCL_VCL"))
960 336
                vcc_predef_vcl(vcc, name);
961
        else
962 0
                WRONG("Unknown VCC predef type");
963 25336
}
964
965
void
966 8
VCC_VEXT(struct vcc *vcc, const char *filename)
967
{
968 8
        CHECK_OBJ_NOTNULL(vcc, VCC_MAGIC);
969 8
        vcc_ImportVext(vcc, filename);
970 8
}