varnish-cache/bin/varnishd/cache/cache_vpi.c
0
/*-
1
 * Copyright (c) 2006 Verdens Gang AS
2
 * Copyright (c) 2006-2019 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
#include "config.h"
33
34
#include <stdio.h>
35
36
#include "cache_varnishd.h"
37
38
#include "vcl.h"
39
#include "vbm.h"
40
41
#include "vcc_interface.h"
42
43
#include "cache_vcl.h"
44
45
/*--------------------------------------------------------------------
46
 * Private & exclusive interfaces between VCC and varnishd
47
 */
48
49
const size_t vpi_wrk_len = sizeof(struct wrk_vpi);
50
51
void
52 743916
VPI_wrk_init(struct worker *wrk, void *p, size_t spc)
53
{
54 743916
        struct wrk_vpi *vpi = p;
55
56 743916
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
57 743916
        AN(vpi);
58 743916
        assert(spc >= sizeof *vpi);
59 743916
        INIT_OBJ(vpi, WRK_VPI_MAGIC);
60 743916
        wrk->vpi = vpi;
61 743916
}
62
63
void
64 11040
VPI_trace(VRT_CTX, unsigned u)
65
{
66
67 11040
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
68 11040
        CHECK_OBJ_NOTNULL(ctx->vcl, VCL_MAGIC);
69 11040
        CHECK_OBJ_NOTNULL(ctx->vcl->conf, VCL_CONF_MAGIC);
70 11040
        assert(u < ctx->vcl->conf->nref);
71 11040
        if (ctx->vsl != NULL)
72 21600
                VSLb(ctx->vsl, SLT_VCL_trace, "%s %u %u.%u.%u",
73 10800
                    ctx->vcl->loaded_name, u, ctx->vcl->conf->ref[u].source,
74 10800
                    ctx->vcl->conf->ref[u].line, ctx->vcl->conf->ref[u].pos);
75
        else
76 480
                VSL(SLT_VCL_trace, NO_VXID, "%s %u %u.%u.%u",
77 240
                    ctx->vcl->loaded_name, u, ctx->vcl->conf->ref[u].source,
78 240
                    ctx->vcl->conf->ref[u].line, ctx->vcl->conf->ref[u].pos);
79 11040
}
80
81
static void
82 280
vpi_ref_panic(struct vsb *vsb, unsigned n, const struct vcl *vcl)
83
{
84 280
        const struct VCL_conf *conf = NULL;
85
        const struct vpi_ref *ref;
86 280
        const char *p, *src = NULL;
87 280
        const int lim = 40;
88 280
        const char *abbstr = "[...]";
89 280
        char buf[lim + sizeof(abbstr)];
90 280
        int w = 0;
91
92 280
        AN(vsb);
93
94 280
        if (vcl != NULL)
95 280
                conf = vcl->conf;
96 280
        if (conf != NULL && conf->magic != VCL_CONF_MAGIC)
97 0
                conf = NULL;
98
99 280
        if (conf == NULL) {
100 0
                VSB_printf(vsb, "ref = %u, nref = ?,\n", n);
101 0
                return;
102
        }
103 280
        if (n >= conf->nref) {
104 0
                VSB_printf(vsb, "ref = %u *invalid*, nref = %u\n",
105 0
                    n, conf->nref);
106 0
                return;
107
        }
108
109 280
        VSB_printf(vsb, "ref = %u,\n", n);
110
111 280
        ref = &conf->ref[n];
112 280
        if (PAN_dump_struct(vsb, ref, VPI_REF_MAGIC, "vpi_ref"))
113 0
                return;
114
115 280
        if (ref->source < conf->nsrc) {
116 560
                VSB_printf(vsb, "source = %u (\"%s\"),\n", ref->source,
117 280
                    conf->srcname[ref->source]);
118 280
                src = conf->srcbody[ref->source];
119 280
        } else {
120 0
                VSB_printf(vsb, "source = %u *invalid*,\n", ref->source);
121
        }
122
123 280
        if (src != NULL) {
124 280
                w = strlen(src);
125 280
                assert(w > 0);
126 280
                if (ref->offset >= (unsigned)w)
127 0
                        src = NULL;
128 280
        }
129 280
        if (src != NULL) {
130 280
                src += ref->offset;
131 280
                p = strchr(src, '\n');
132 280
                if (p != NULL)
133 280
                        w = p - src;
134
                else
135 0
                        w -= ref->offset;
136 280
                if (w > lim) {
137 80
                        w = snprintf(buf, sizeof buf, "%.*s%s",
138 80
                            lim, src, abbstr);
139 80
                        src = buf;
140 80
                }
141 280
        }
142
143 280
        VSB_printf(vsb, "offset = %u,\n", ref->offset);
144 280
        VSB_printf(vsb, "line = %u,\n", ref->line);
145 280
        VSB_printf(vsb, "pos = %u,\n", ref->pos);
146 280
        if (src != NULL) {
147 280
                VSB_cat(vsb, "src = ");
148 280
                VSB_quote(vsb, src, w, VSB_QUOTE_CSTR);
149 280
                VSB_putc(vsb, '\n');
150 280
        } else {
151 0
                VSB_printf(vsb, "token = \"%s\"\n", ref->token);
152
        }
153 280
        VSB_indent(vsb, -2);
154 280
        VSB_cat(vsb, "},\n");
155
156 280
}
157
void
158 280
VPI_Panic(struct vsb *vsb, const struct wrk_vpi *vpi, const struct vcl *vcl)
159
{
160
        const char *hand;
161
162 280
        AN(vsb);
163 280
        if (PAN_dump_struct(vsb, vpi, WRK_VPI_MAGIC, "vpi"))
164 0
                return;
165
166 280
        hand = VCL_Return_Name(vpi->handling);
167 280
        if (vpi->handling == 0)
168 280
                hand = "none";
169 0
        else if (hand == NULL)
170 0
                hand = "*invalid*";
171
172 560
        VSB_printf(vsb, "handling (VCL::return) = 0x%x (%s),\n",
173 280
            vpi->handling, hand);
174
175 280
        vpi_ref_panic(vsb, vpi->ref, vcl);
176
177 280
        VSB_indent(vsb, -2);
178 280
        VSB_cat(vsb, "},\n");
179 280
}
180
181
/*
182
 * After vcl_fini {} == VGC_function_vcl_fini() is called from VGC_Discard(),
183
 * handling must either be OK from VCL "return (ok)" or FAIL from VRT_fail().
184
 *
185
 * replace OK with 0 for _fini callbacks because that handling has meaning only
186
 * when returning from VCL subs
187
 */
188
189
void
190 3065
VPI_vcl_fini(VRT_CTX)
191
{
192 3065
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
193 3065
        AN(ctx->vpi);
194
195 3065
        if (ctx->vpi->handling == VCL_RET_FAIL)
196 40
                return;
197 3025
        assert(ctx->vpi->handling == VCL_RET_OK);
198 3025
        ctx->vpi->handling = 0;
199 3065
}
200
201
VCL_VCL
202 840
VPI_vcl_get(VRT_CTX, const char *name)
203
{
204
        VCL_VCL vcl;
205
206 840
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
207 840
        vcl = vcl_find(name);
208 840
        AN(vcl);
209 840
        Lck_Lock(&vcl_mtx);
210 840
        vcl->nrefs++;
211 840
        Lck_Unlock(&vcl_mtx);
212 840
        return (vcl);
213
}
214
215
void
216 240
VPI_vcl_rel(VRT_CTX, VCL_VCL vcl)
217
{
218 240
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
219 240
        AN(vcl);
220 240
        Lck_Lock(&vcl_mtx);
221 240
        vcl->nrefs--;
222 240
        Lck_Unlock(&vcl_mtx);
223 240
}
224
225
void
226 1640
VPI_vcl_select(VRT_CTX, VCL_VCL vcl)
227
{
228 1640
        struct req *req = ctx->req;
229
230 1640
        CHECK_OBJ_NOTNULL(vcl, VCL_MAGIC);
231 1640
        CHECK_OBJ_NOTNULL(req, REQ_MAGIC);
232 1640
        CHECK_OBJ_NOTNULL(req->top, REQTOP_MAGIC);
233
234 1640
        if ((IS_TOPREQ(req) && req->top->vcl0 != NULL) || req->restarts > 0)
235 1120
                return;         // Illegal, req-FSM will fail this later.
236
237 520
        if (! IS_TOPREQ(req))
238 40
                assert(req->vcl == req->top->vcl0);
239
240 520
        Req_Rollback(ctx);
241
242 520
        if (IS_TOPREQ(req)) {
243 480
                AN(req->top);
244 480
                AZ(req->top->vcl0);
245 480
                req->top->vcl0 = req->vcl;
246 480
                req->vcl = NULL;
247 480
        }
248 520
        VCL_Update(&req->vcl, vcl);
249 1040
        VSLb(ctx->req->vsl, SLT_VCL_use, "%s via %s",
250 520
            req->vcl->loaded_name, vcl->loaded_name);
251 680
}
252
253
void v_noreturn_
254 0
VPI_Fail(const char *func, const char *file, int line,
255
    const char *cond)
256
{
257 0
        VAS_Fail(func, file, line, cond, VAS_ASSERT);
258
}
259
260
enum vcl_func_fail_e
261 800
VPI_Call_Check(VRT_CTX, const struct VCL_conf *conf,
262
    unsigned methods, unsigned n)
263
{
264
        struct vbitmap *vbm;
265
266 800
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
267 800
        CHECK_OBJ_NOTNULL(ctx->vcl, VCL_MAGIC);
268
269 800
        assert(conf == ctx->vcl->conf);
270
271 760
        vbm = ctx->called;
272 760
        AN(vbm);
273
274 760
        if ((methods & ctx->method) == 0)
275 120
                return (VSUB_E_METHOD);
276
277 640
        if (vbit_test(vbm, n))
278 80
                return (VSUB_E_RECURSE);
279
280 560
        return (VSUB_E_OK);
281 760
}
282
283
void
284 560
VPI_Call_Begin(VRT_CTX, unsigned n)
285
{
286
        struct vbitmap *vbm;
287
288 560
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
289 560
        vbm = ctx->called;
290 560
        AN(vbm);
291 560
        vbit_set(vbm, n);
292 560
}
293
294
void
295 560
VPI_Call_End(VRT_CTX, unsigned n)
296
{
297
        struct vbitmap *vbm;
298
299 560
        CHECK_OBJ_NOTNULL(ctx, VRT_CTX_MAGIC);
300 560
        vbm = ctx->called;
301 560
        AN(vbm);
302 560
        vbit_clr(vbm, n);
303 560
}