varnish-cache/bin/varnishtest/vtc_sess.c
0
/*-
1
 * Copyright (c) 2020 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
 */
30
31
#include "config.h"
32
33
#include <sys/types.h>
34
35
#include <stdlib.h>
36
#include <string.h>
37
38
#include "vtc.h"
39
#include "vtc_http.h"
40
41
struct thread_arg {
42
        unsigned                magic;
43
#define THREAD_ARG_MAGIC        0xd5dc5f1c
44
        void                    *priv;
45
        sess_conn_f             *conn_f;
46
        sess_disc_f             *disc_f;
47
        const char              *listen_addr;
48
        struct vtc_sess         *vsp;
49
        int                     *asocket;
50
        const char              *spec;
51
};
52
53
struct vtc_sess *
54 17470
Sess_New(struct vtclog *vl, const char *name)
55
{
56
        struct vtc_sess *vsp;
57
58 17470
        ALLOC_OBJ(vsp, VTC_SESS_MAGIC);
59 17470
        AN(vsp);
60 17470
        vsp->vl = vl;
61 17470
        REPLACE(vsp->name, name);
62 17470
        vsp->repeat = 1;
63 17470
        return (vsp);
64
}
65
66
void
67 17470
Sess_Destroy(struct vtc_sess **vspp)
68
{
69
        struct vtc_sess *vsp;
70
71 17470
        TAKE_OBJ_NOTNULL(vsp, vspp, VTC_SESS_MAGIC);
72 17470
        REPLACE(vsp->name, NULL);
73 17470
        FREE_OBJ(vsp);
74 17470
}
75
76
int
77 47168
Sess_GetOpt(struct vtc_sess *vsp, char * const **avp)
78
{
79
        char * const *av;
80 47168
        int rv = 0;
81
82 47168
        CHECK_OBJ_NOTNULL(vsp, VTC_SESS_MAGIC);
83 47168
        AN(avp);
84 47168
        av = *avp;
85 47168
        AN(*av);
86 47168
        if (!strcmp(*av, "-rcvbuf")) {
87 72
                AN(av[1]);
88 72
                vsp->rcvbuf = atoi(av[1]);
89 72
                av += 1;
90 72
                rv = 1;
91 47168
        } else if (!strcmp(*av, "-repeat")) {
92 704
                AN(av[1]);
93 704
                vsp->repeat = atoi(av[1]);
94 704
                av += 1;
95 704
                rv = 1;
96 47096
        } else if (!strcmp(*av, "-keepalive")) {
97 120
                vsp->keepalive = 1;
98 120
                rv = 1;
99 120
        }
100 47168
        *avp = av;
101 47168
        return (rv);
102
}
103
104
int
105 30920
sess_process(struct vtclog *vl, struct vtc_sess *vsp,
106
    const char *spec, int sock, int *sfd, const char *addr)
107
{
108
        int rv;
109
110 30920
        CHECK_OBJ_NOTNULL(vsp, VTC_SESS_MAGIC);
111
112 30920
        rv = http_process(vl, vsp, spec, sock, sfd, addr, vsp->rcvbuf);
113 30920
        return (rv);
114
}
115
116
static void *
117 22432
sess_thread(void *priv)
118
{
119
        struct vtclog *vl;
120
        struct vtc_sess *vsp;
121
        struct thread_arg ta, *tap;
122 22432
        int i, fd = -1;
123
124 22432
        CAST_OBJ_NOTNULL(tap, priv, THREAD_ARG_MAGIC);
125 22432
        ta = *tap;
126 22432
        FREE_OBJ(tap);
127
128 22432
        vsp = ta.vsp;
129 22432
        CHECK_OBJ_NOTNULL(vsp, VTC_SESS_MAGIC);
130 22432
        vl = vtc_logopen("%s", vsp->name);
131 22432
        pthread_cleanup_push(vtc_logclose, vl);
132
133 22432
        assert(vsp->repeat > 0);
134 44864
        vtc_log(vl, 2, "Started on %s (%u iterations%s)", ta.listen_addr,
135 22432
                vsp->repeat, vsp->keepalive ? " using keepalive" : "");
136 52635
        for (i = 0; i < vsp->repeat; i++) {
137 30203
                if (fd < 0)
138 29076
                        fd = ta.conn_f(ta.priv, vl);
139 60406
                fd = sess_process(vl, ta.vsp, ta.spec, fd,
140 30203
                    ta.asocket, ta.listen_addr);
141 30203
                if (! vsp->keepalive)
142 27945
                        ta.disc_f(ta.priv, vl, &fd);
143 30203
        }
144 22432
        if (vsp->keepalive)
145 144
                ta.disc_f(ta.priv, vl, &fd);
146 22432
        vtc_log(vl, 2, "Ending");
147 22432
        pthread_cleanup_pop(0);
148 22432
        vtc_logclose(vl);
149 22432
        return (NULL);
150
}
151
152
pthread_t
153 22432
Sess_Start_Thread(
154
    void *priv,
155
    struct vtc_sess *vsp,
156
    sess_conn_f *conn,
157
    sess_disc_f *disc,
158
    const char *listen_addr,
159
    int *asocket,
160
    const char *spec
161
)
162
{
163
        struct thread_arg *ta;
164
        pthread_t pt;
165
166 22432
        AN(priv);
167 22432
        CHECK_OBJ_NOTNULL(vsp, VTC_SESS_MAGIC);
168 22432
        AN(conn);
169 22432
        AN(disc);
170 22432
        AN(listen_addr);
171 22432
        ALLOC_OBJ(ta, THREAD_ARG_MAGIC);
172 22432
        AN(ta);
173 22432
        ta->priv = priv;
174 22432
        ta->vsp = vsp;
175
176 22432
        ta->conn_f = conn;
177 22432
        ta->disc_f = disc;
178 22432
        ta->listen_addr = listen_addr;
179 22432
        ta->asocket = asocket;
180 22432
        ta->spec = spec;
181 22432
        PTOK(pthread_create(&pt, NULL, sess_thread, ta));
182 22432
        return (pt);
183
}