varnish-cache/bin/varnishd/storage/stevedore.c
0
/*-
1
 * Copyright (c) 2007-2015 Varnish Software AS
2
 * All rights reserved.
3
 *
4
 * Author: Dag-Erling Smørgav <des@des.no>
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
 * STEVEDORE: one who works at or is responsible for loading and
30
 * unloading ships in port.  Example: "on the wharves, stevedores were
31
 * unloading cargo from the far corners of the world." Origin: Spanish
32
 * estibador, from estibar to pack.  First Known Use: 1788
33
 */
34
35
#include "config.h"
36
37
#include "cache/cache_varnishd.h"
38
39
#include <stdio.h>
40
#include <stdlib.h>
41
42
#include "storage/storage.h"
43
#include "vrt_obj.h"
44
45
extern const char *mgt_stv_h2_rxbuf;
46
struct stevedore *stv_h2_rxbuf = NULL;
47
48
static pthread_mutex_t stv_mtx;
49
50
/*--------------------------------------------------------------------
51
 * XXX: trust pointer writes to be atomic
52
 */
53
54
const struct stevedore *
55 61640
STV_next(void)
56
{
57
        static struct stevedore *stv;
58
        struct stevedore *r;
59
60 61640
        PTOK(pthread_mutex_lock(&stv_mtx));
61 61640
        if (!STV__iter(&stv))
62 200
                AN(STV__iter(&stv));
63 61640
        if (stv == stv_transient) {
64 35160
                stv = NULL;
65 35160
                AN(STV__iter(&stv));
66 35160
        }
67 61640
        r = stv;
68 61640
        PTOK(pthread_mutex_unlock(&stv_mtx));
69 61640
        AN(r);
70 61640
        return (r);
71
}
72
73
/*-------------------------------------------------------------------
74
 * Allocate storage for an object, based on the header information.
75
 * XXX: If we know (a hint of) the length, we could allocate space
76
 * XXX: for the body in the same allocation while we are at it.
77
 */
78
79
int
80 115759
STV_NewObject(struct worker *wrk, struct objcore *oc,
81
    const struct stevedore *stv, unsigned wsl)
82
{
83 115759
        CHECK_OBJ_NOTNULL(oc, OBJCORE_MAGIC);
84 115759
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
85 115759
        CHECK_OBJ_NOTNULL(stv, STEVEDORE_MAGIC);
86
87 115759
        wrk->strangelove = cache_param->nuke_limit;
88 115759
        AN(stv->allocobj);
89 115759
        if (stv->allocobj(wrk, stv, oc, wsl) == 0) {
90 880
                VSLb(wrk->vsl, SLT_Error,
91
                    "Failed to create object from %s %s",
92 440
                    stv->name, stv->ident);
93 440
                return (0);
94
        }
95 115319
        oc->oa_present = 0;
96 115319
        wrk->stats->n_object++;
97 230638
        VSLb(wrk->vsl, SLT_Storage, "%s %s",
98 115319
            oc->stobj->stevedore->name, oc->stobj->stevedore->ident);
99 115319
        return (1);
100 115759
}
101
102
/*-------------------------------------------------------------------*/
103
104
struct stv_buffer {
105
        unsigned                magic;
106
#define STV_BUFFER_MAGIC        0xf39cb6c2
107
        const struct stevedore  *stv;
108
        size_t                  size;
109
        uintptr_t               priv;
110
};
111
112
struct stv_buffer *
113 11760
STV_AllocBuf(struct worker *wrk, const struct stevedore *stv, size_t size)
114
{
115
        struct stv_buffer *stvbuf;
116
        uint8_t *buf;
117 11760
        uintptr_t priv = 0;
118
119 11760
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
120 11760
        CHECK_OBJ_NOTNULL(stv, STEVEDORE_MAGIC);
121
122 11760
        if (size == 0)
123 0
                return (NULL);
124 11760
        if (stv->allocbuf == NULL)
125 0
                return (NULL);
126
127 11760
        wrk->strangelove = cache_param->nuke_limit;
128 11760
        buf = stv->allocbuf(wrk, stv, size + PRNDUP(sizeof *stvbuf), &priv);
129 11760
        if (buf == NULL)
130 0
                return (NULL);
131
132 11760
        assert(PAOK(buf));
133 11760
        stvbuf = (void *)buf;
134 11760
        INIT_OBJ(stvbuf, STV_BUFFER_MAGIC);
135 11760
        stvbuf->stv = stv;
136 11760
        stvbuf->priv = priv;
137 11760
        stvbuf->size = size;
138
139 11760
        return (stvbuf);
140 11760
}
141
142
void
143 11760
STV_FreeBuf(struct worker *wrk, struct stv_buffer **pstvbuf)
144
{
145
        struct stv_buffer *stvbuf;
146
        const struct stevedore *stv;
147
        uintptr_t priv;
148
149 11760
        CHECK_OBJ_NOTNULL(wrk, WORKER_MAGIC);
150 11760
        TAKE_OBJ_NOTNULL(stvbuf, pstvbuf, STV_BUFFER_MAGIC);
151 11760
        CHECK_OBJ_NOTNULL(stvbuf->stv, STEVEDORE_MAGIC);
152
153 11760
        stv = stvbuf->stv;
154 11760
        priv = stvbuf->priv;
155 11760
        CHECK_OBJ_NOTNULL(stv, STEVEDORE_MAGIC);
156 11760
        ZERO_OBJ(stvbuf, sizeof *stvbuf);
157
158 11760
        AN(stv->freebuf);
159 11760
        stv->freebuf(wrk, stv, priv);
160 11760
}
161
162
void *
163 11760
STV_GetBufPtr(struct stv_buffer *stvbuf, size_t *psize)
164
{
165 11760
        CHECK_OBJ_NOTNULL(stvbuf, STV_BUFFER_MAGIC);
166 11760
        if (psize)
167 11760
                *psize = stvbuf->size;
168 11760
        return (&stvbuf[1]);
169
}
170
171
/*-------------------------------------------------------------------*/
172
173
void
174 37947
STV_open(void)
175
{
176
        struct stevedore *stv;
177
        char buf[1024];
178
179 37947
        ASSERT_CLI();
180 37947
        PTOK(pthread_mutex_init(&stv_mtx, &mtxattr_errorcheck));
181
182
        /* This string was prepared for us before the fork, and should
183
         * point to a configured stevedore. */
184 37947
        AN(mgt_stv_h2_rxbuf);
185
186 37947
        stv_h2_rxbuf = NULL;
187 114281
        STV_Foreach(stv) {
188 76334
                CHECK_OBJ_NOTNULL(stv, STEVEDORE_MAGIC);
189 76334
                bprintf(buf, "storage.%s", stv->ident);
190 76334
                stv->vclname = strdup(buf);
191 76334
                AN(stv->vclname);
192 76334
                if (stv->open != NULL)
193 76334
                        stv->open(stv);
194 76334
                if (!strcmp(stv->ident, mgt_stv_h2_rxbuf))
195 37947
                        stv_h2_rxbuf = stv;
196
        }
197 37947
        AN(stv_h2_rxbuf);
198 37947
}
199
200
void
201 37520
STV_warn(void)
202
{
203
        struct stevedore *stv;
204
205 37520
        ASSERT_CLI();
206 112920
        STV_Foreach(stv)
207 75400
                if (stv->close != NULL)
208 1480
                        stv->close(stv, 1);
209 37520
}
210
211
void
212 37520
STV_close(void)
213
{
214
        struct stevedore *stv;
215
216 37520
        ASSERT_CLI();
217 112920
        STV_Foreach(stv)
218 75400
                if (stv->close != NULL)
219 1480
                        stv->close(stv, 0);
220 37520
}
221
222
/*-------------------------------------------------------------------
223
 * Notify the stevedores of BAN related events. A non-zero return
224
 * value indicates that the stevedore is unable to persist the
225
 * event.
226
 */
227
228
int
229 2884
STV_BanInfoDrop(const uint8_t *ban, unsigned len)
230
{
231
        struct stevedore *stv;
232 2884
        int r = 0;
233
234 8852
        STV_Foreach(stv)
235 5968
                if (stv->baninfo != NULL)
236 840
                        r |= stv->baninfo(stv, BI_DROP, ban, len);
237
238 2884
        return (r);
239
}
240
241
int
242 41947
STV_BanInfoNew(const uint8_t *ban, unsigned len)
243
{
244
        struct stevedore *stv;
245 41947
        int r = 0;
246
247 126321
        STV_Foreach(stv)
248 84374
                if (stv->baninfo != NULL)
249 1920
                        r |= stv->baninfo(stv, BI_NEW, ban, len);
250
251 41947
        return (r);
252
}
253
254
/*-------------------------------------------------------------------
255
 * Export a complete ban list to the stevedores for persistence.
256
 * The stevedores should clear any previous ban lists and replace
257
 * them with this list.
258
 */
259
260
void
261 75467
STV_BanExport(const uint8_t *bans, unsigned len)
262
{
263
        struct stevedore *stv;
264
265 227121
        STV_Foreach(stv)
266 151654
                if (stv->banexport != NULL)
267 3000
                        stv->banexport(stv, bans, len);
268 75467
}
269
270
/*--------------------------------------------------------------------
271
 * VRT functions for stevedores
272
 */
273
274
static const struct stevedore *
275 1760
stv_find(const char *nm)
276
{
277
        struct stevedore *stv;
278
279 2360
        STV_Foreach(stv)
280 2360
                if (!strcmp(stv->ident, nm))
281 1760
                        return (stv);
282 0
        return (NULL);
283 1760
}
284
285
int
286 0
VRT_Stv(const char *nm)
287
{
288
289 0
        if (stv_find(nm) != NULL)
290 0
                return (1);
291 0
        return (0);
292 0
}
293
294
const char * v_matchproto_()
295 720
VRT_STEVEDORE_string(VCL_STEVEDORE s)
296
{
297 720
        if (s == NULL)
298 40
                return (NULL);
299 680
        CHECK_OBJ_NOTNULL(s, STEVEDORE_MAGIC);
300 680
        return (s->vclname);
301 720
}
302
303
VCL_STEVEDORE
304 1760
VRT_stevedore(const char *nm)
305
{
306 1760
        return (stv_find(nm));
307
}
308
309
#define VRTSTVVAR(nm, vtype, ctype, dval)               \
310
ctype                                                   \
311
VRT_stevedore_##nm(VCL_STEVEDORE stv)                   \
312
{                                                       \
313
        if (stv == NULL)                                \
314
                return (0);                             \
315
        if (stv->var_##nm == NULL)                      \
316
                return (dval);                          \
317
        CHECK_OBJ_NOTNULL(stv, STEVEDORE_MAGIC);        \
318
        return (stv->var_##nm(stv));                    \
319
}
320
#include "tbl/vrt_stv_var.h"