| | varnish-cache/lib/libvarnishapi/vut.c |
0 |
|
/*- |
1 |
|
* Copyright (c) 2006 Verdens Gang AS |
2 |
|
* Copyright (c) 2006-2015 Varnish Software AS |
3 |
|
* All rights reserved. |
4 |
|
* |
5 |
|
* Author: Martin Blix Grydeland <martin@varnish-software.com> |
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 |
|
* Common functions for the utilities |
31 |
|
*/ |
32 |
|
|
33 |
|
#include "config.h" |
34 |
|
|
35 |
|
#include <ctype.h> |
36 |
|
#include <math.h> |
37 |
|
#include <stdint.h> |
38 |
|
#include <stdarg.h> |
39 |
|
#include <stdlib.h> |
40 |
|
#include <unistd.h> |
41 |
|
#include <stdio.h> |
42 |
|
#include <string.h> |
43 |
|
#include <sys/stat.h> /* for MUSL */ |
44 |
|
|
45 |
|
#include "compat/daemon.h" |
46 |
|
#include "vdef.h" |
47 |
|
#include "vpf.h" |
48 |
|
#include "vapi/vsm.h" |
49 |
|
#include "vapi/vsl.h" |
50 |
|
#include "vtim.h" |
51 |
|
#include "vas.h" |
52 |
|
#include "miniobj.h" |
53 |
|
#include "vcs.h" |
54 |
|
#include "vsb.h" |
55 |
|
#include "vfil.h" |
56 |
|
|
57 |
|
#include "vapi/voptget.h" |
58 |
|
#include "vapi/vsig.h" |
59 |
|
|
60 |
|
#include "vut.h" |
61 |
|
|
62 |
|
|
63 |
|
static int vut_synopsis(const struct vopt_spec *); |
64 |
|
static int vut_options(const struct vopt_spec *); |
65 |
|
|
66 |
|
static struct vpf_fh *pfh; |
67 |
|
static unsigned daemonized; |
68 |
|
|
69 |
|
static struct VUT pfh_vut; |
70 |
|
|
71 |
|
static int |
72 |
2 |
vut_daemon(struct VUT *vut) |
73 |
|
{ |
74 |
2 |
if (daemonized) |
75 |
0 |
VUT_Error(vut, 1, "Already running as a daemon"); |
76 |
2 |
daemonized = 1; |
77 |
2 |
return (varnish_daemon(0, 0)); |
78 |
|
} |
79 |
|
|
80 |
|
static void |
81 |
4 |
vut_vpf_remove(void) |
82 |
|
{ |
83 |
|
|
84 |
4 |
CHECK_OBJ(&pfh_vut, VUT_MAGIC); |
85 |
4 |
AN(pfh); |
86 |
4 |
AN(pfh_vut.P_arg); |
87 |
|
|
88 |
4 |
VPF_Remove(pfh); |
89 |
4 |
free(pfh_vut.P_arg); |
90 |
4 |
ZERO_OBJ(&pfh_vut, sizeof pfh_vut); |
91 |
4 |
pfh = NULL; |
92 |
4 |
} |
93 |
|
|
94 |
|
static int v_matchproto_(VSLQ_dispatch_f) |
95 |
972 |
vut_dispatch(struct VSL_data *vsl, struct VSL_transaction * const trans[], |
96 |
|
void *priv) |
97 |
|
{ |
98 |
|
struct VUT *vut; |
99 |
|
int i; |
100 |
|
|
101 |
972 |
CAST_OBJ_NOTNULL(vut, priv, VUT_MAGIC); |
102 |
|
|
103 |
972 |
if (vut->k_arg == 0) |
104 |
0 |
return (-1); /* End of file */ |
105 |
972 |
AN(vut->dispatch_f); |
106 |
972 |
i = vut->dispatch_f(vsl, trans, vut->dispatch_priv); |
107 |
972 |
if (vut->k_arg > 0) |
108 |
3 |
vut->k_arg--; |
109 |
972 |
if (i >= 0 && vut->k_arg == 0) |
110 |
3 |
return (-1); /* End of file */ |
111 |
969 |
return (i); |
112 |
972 |
} |
113 |
|
|
114 |
|
void |
115 |
97 |
VUT_Error(struct VUT *vut, int status, const char *fmt, ...) |
116 |
|
{ |
117 |
|
va_list ap; |
118 |
|
|
119 |
97 |
CHECK_OBJ_NOTNULL(vut, VUT_MAGIC); |
120 |
97 |
AN(status); |
121 |
|
|
122 |
97 |
va_start(ap, fmt); |
123 |
97 |
if (vut->error_f != NULL) { |
124 |
0 |
vut->error_f(vut, status, fmt, ap); |
125 |
0 |
} else { |
126 |
97 |
vfprintf(stderr, fmt, ap); |
127 |
97 |
fprintf(stderr, "\n"); |
128 |
|
} |
129 |
97 |
va_end(ap); |
130 |
97 |
exit(status); |
131 |
|
} |
132 |
|
|
133 |
|
static void |
134 |
59 |
vut_arg_q(struct VUT *vut, const char *arg) |
135 |
|
{ |
136 |
|
struct vsb *vsb; |
137 |
|
|
138 |
59 |
AN(arg); |
139 |
59 |
if (vut->q_arg == NULL) { |
140 |
56 |
REPLACE(vut->q_arg, arg); |
141 |
56 |
return; |
142 |
|
} |
143 |
|
|
144 |
3 |
vsb = VSB_new_auto(); |
145 |
3 |
AN(vsb); |
146 |
3 |
AZ(VSB_printf(vsb, "%s\n%s", vut->q_arg, arg)); |
147 |
3 |
AZ(VSB_finish(vsb)); |
148 |
|
|
149 |
3 |
REPLACE(vut->q_arg, VSB_data(vsb)); |
150 |
|
|
151 |
3 |
VSB_destroy(&vsb); |
152 |
59 |
} |
153 |
|
|
154 |
|
int |
155 |
402 |
VUT_Arg(struct VUT *vut, int opt, const char *arg) |
156 |
|
{ |
157 |
|
int i; |
158 |
|
char *p; |
159 |
|
|
160 |
402 |
CHECK_OBJ_NOTNULL(vut, VUT_MAGIC); |
161 |
402 |
AN(opt); |
162 |
|
|
163 |
402 |
switch (opt) { |
164 |
|
case 'd': |
165 |
|
/* Head */ |
166 |
75 |
vut->d_opt = 1; |
167 |
75 |
return (1); |
168 |
|
case 'D': |
169 |
|
/* Daemon mode */ |
170 |
6 |
vut->D_opt = 1; |
171 |
6 |
return (1); |
172 |
|
case 'g': |
173 |
|
/* Grouping */ |
174 |
14 |
AN(arg); |
175 |
14 |
vut->g_arg = VSLQ_Name2Grouping(arg, -1); |
176 |
14 |
if (vut->g_arg == -2) |
177 |
1 |
VUT_Error(vut, 1, "Ambiguous grouping type: %s", arg); |
178 |
13 |
else if (vut->g_arg < 0) |
179 |
1 |
VUT_Error(vut, 1, "Unknown grouping type: %s", arg); |
180 |
12 |
return (1); |
181 |
|
case 'k': |
182 |
|
/* Log transaction limit */ |
183 |
4 |
AN(arg); |
184 |
4 |
vut->k_arg = (int)strtol(arg, &p, 10); |
185 |
4 |
if (*p != '\0' || vut->k_arg <= 0) |
186 |
1 |
VUT_Error(vut, 1, "-k: Invalid number '%s'", arg); |
187 |
3 |
return (1); |
188 |
|
case 'n': |
189 |
|
/* Varnish instance name */ |
190 |
87 |
AN(arg); |
191 |
87 |
REPLACE(vut->n_arg, arg); |
192 |
87 |
return (1); |
193 |
|
case 'P': |
194 |
|
/* PID file */ |
195 |
4 |
AN(arg); |
196 |
4 |
REPLACE(vut->P_arg, arg); |
197 |
4 |
return (1); |
198 |
|
case 'Q': |
199 |
8 |
AN(arg); |
200 |
8 |
p = VFIL_readfile(NULL, arg, NULL); |
201 |
8 |
if (p == NULL) |
202 |
1 |
VUT_Error(vut, 1, "-Q %s: %s", arg, strerror(errno)); |
203 |
7 |
vut_arg_q(vut, p); |
204 |
7 |
free(p); |
205 |
7 |
return (1); |
206 |
|
case 'q': |
207 |
|
/* Query to use */ |
208 |
52 |
AN(arg); |
209 |
52 |
vut_arg_q(vut, arg); |
210 |
52 |
return (1); |
211 |
|
case 'r': |
212 |
|
/* Binary file input */ |
213 |
8 |
AN(arg); |
214 |
8 |
REPLACE(vut->r_arg, arg); |
215 |
8 |
return (1); |
216 |
|
case 't': |
217 |
|
/* VSM connect timeout */ |
218 |
4 |
REPLACE(vut->t_arg, arg); |
219 |
4 |
return (1); |
220 |
|
case 'V': |
221 |
|
/* Print version number and exit */ |
222 |
5 |
VCS_Message(vut->progname); |
223 |
5 |
exit(0); |
224 |
|
default: |
225 |
135 |
AN(vut->vsl); |
226 |
135 |
i = VSL_Arg(vut->vsl, opt, arg); |
227 |
135 |
if (i < 0) |
228 |
18 |
VUT_Error(vut, 1, "%s", VSL_Error(vut->vsl)); |
229 |
117 |
return (i); |
230 |
|
} |
231 |
375 |
} |
232 |
|
|
233 |
|
struct VUT * |
234 |
231 |
VUT_Init(const char *progname, int argc, char * const *argv, |
235 |
|
const struct vopt_spec *voc) |
236 |
|
{ |
237 |
|
struct VUT *vut; |
238 |
|
|
239 |
231 |
AN(progname); |
240 |
231 |
AN(argv); |
241 |
231 |
AN(voc); |
242 |
|
|
243 |
231 |
VSIG_Arm_hup(); |
244 |
231 |
VSIG_Arm_int(); |
245 |
231 |
VSIG_Arm_term(); |
246 |
231 |
VSIG_Arm_usr1(); |
247 |
|
|
248 |
231 |
if (argc == 2 && !strcmp(argv[1], "--synopsis")) |
249 |
5 |
exit(vut_synopsis(voc)); |
250 |
226 |
if (argc == 2 && !strcmp(argv[1], "--options")) |
251 |
5 |
exit(vut_options(voc)); |
252 |
221 |
if (argc == 2 && !strcmp(argv[1], "--optstring")) { |
253 |
0 |
(void)printf("%s\n", voc->vopt_optstring); |
254 |
0 |
exit(0); |
255 |
|
} |
256 |
|
|
257 |
221 |
ALLOC_OBJ(vut, VUT_MAGIC); |
258 |
221 |
AN(vut); |
259 |
221 |
vut->progname = progname; |
260 |
221 |
vut->g_arg = VSL_g_vxid; |
261 |
221 |
vut->k_arg = -1; |
262 |
221 |
AZ(vut->vsl); |
263 |
221 |
vut->vsl = VSL_New(); |
264 |
221 |
AN(vut->vsl); |
265 |
221 |
return (vut); |
266 |
|
} |
267 |
|
|
268 |
|
void |
269 |
0 |
VUT_Signal(VUT_sighandler_f sig_cb) |
270 |
|
{ |
271 |
|
|
272 |
0 |
AN(sig_cb); |
273 |
0 |
(void)signal(SIGHUP, sig_cb); |
274 |
0 |
(void)signal(SIGINT, sig_cb); |
275 |
0 |
(void)signal(SIGTERM, sig_cb); |
276 |
0 |
(void)signal(SIGUSR1, sig_cb); |
277 |
0 |
} |
278 |
|
|
279 |
|
void |
280 |
0 |
VUT_Signaled(struct VUT *vut, int sig) |
281 |
|
{ |
282 |
|
|
283 |
0 |
CHECK_OBJ_NOTNULL(vut, VUT_MAGIC); |
284 |
|
#define VSIG_SIGNAL(UPPER, lower) \ |
285 |
|
VSIG_##lower += (int)(sig == SIG##UPPER); |
286 |
|
#include "tbl/vsig_list.h" |
287 |
|
} |
288 |
|
|
289 |
|
void |
290 |
118 |
VUT_Setup(struct VUT *vut) |
291 |
|
{ |
292 |
|
struct VSL_cursor *c; |
293 |
|
|
294 |
118 |
CHECK_OBJ_NOTNULL(vut, VUT_MAGIC); |
295 |
118 |
AN(vut->vsl); |
296 |
118 |
AZ(vut->vsm); |
297 |
118 |
AZ(vut->vslq); |
298 |
|
|
299 |
|
/* Check input arguments (2 used for bug in FlexeLint) */ |
300 |
354 |
if ((vut->n_arg == NULL ? 0 : 2) + |
301 |
236 |
(vut->r_arg == NULL ? 0 : 2) > 2) |
302 |
1 |
VUT_Error(vut, 1, "Only one of -n and -r options may be used"); |
303 |
|
|
304 |
117 |
if (vut->r_arg != NULL && !strcmp(vut->r_arg, "-") && vut->D_opt) |
305 |
0 |
VUT_Error(vut, 1, "Daemon cannot read from stdin"); |
306 |
|
|
307 |
|
/* Create and validate the query expression */ |
308 |
234 |
vut->vslq = VSLQ_New(vut->vsl, NULL, |
309 |
117 |
(enum VSL_grouping_e)vut->g_arg, vut->q_arg); |
310 |
117 |
if (vut->vslq == NULL) |
311 |
64 |
VUT_Error(vut, 1, "Query expression error:\n%s", |
312 |
32 |
VSL_Error(vut->vsl)); |
313 |
|
|
314 |
|
/* Setup input */ |
315 |
85 |
if (vut->r_arg) { |
316 |
7 |
c = VSL_CursorFile(vut->vsl, vut->r_arg, 0); |
317 |
7 |
if (c == NULL) |
318 |
2 |
VUT_Error(vut, 1, "%s", VSL_Error(vut->vsl)); |
319 |
5 |
VSLQ_SetCursor(vut->vslq, &c); |
320 |
5 |
AZ(c); |
321 |
5 |
} else { |
322 |
78 |
vut->vsm = VSM_New(); |
323 |
78 |
AN(vut->vsm); |
324 |
78 |
if (vut->n_arg && VSM_Arg(vut->vsm, 'n', vut->n_arg) <= 0) |
325 |
0 |
VUT_Error(vut, 1, "%s", VSM_Error(vut->vsm)); |
326 |
78 |
if (vut->t_arg && VSM_Arg(vut->vsm, 't', vut->t_arg) <= 0) |
327 |
3 |
VUT_Error(vut, 1, "%s", VSM_Error(vut->vsm)); |
328 |
75 |
if (VSM_Attach(vut->vsm, STDERR_FILENO)) |
329 |
1 |
VUT_Error(vut, 1, "VSM: %s", VSM_Error(vut->vsm)); |
330 |
|
// Cursor is handled in VUT_Main() |
331 |
|
} |
332 |
|
|
333 |
|
/* Open PID file */ |
334 |
79 |
if (vut->P_arg) { |
335 |
4 |
if (pfh != NULL) |
336 |
0 |
VUT_Error(vut, 1, "PID file already created"); |
337 |
4 |
pfh = VPF_Open(vut->P_arg, 0644, NULL); |
338 |
4 |
if (pfh == NULL) |
339 |
0 |
VUT_Error(vut, 1, |
340 |
0 |
"%s: %s", vut->P_arg, strerror(errno)); |
341 |
4 |
} |
342 |
|
|
343 |
|
/* Daemon mode */ |
344 |
79 |
if (vut->D_opt && vut_daemon(vut) == -1) |
345 |
0 |
VUT_Error(vut, 1, "Daemon mode: %s", strerror(errno)); |
346 |
|
|
347 |
|
/* Write PID and setup exit handler */ |
348 |
79 |
if (vut->P_arg) { |
349 |
4 |
AN(pfh); |
350 |
4 |
VPF_Write(pfh); |
351 |
|
|
352 |
|
/* NB: move ownership to a global pseudo-VUT. */ |
353 |
4 |
INIT_OBJ(&pfh_vut, VUT_MAGIC); |
354 |
4 |
pfh_vut.P_arg = vut->P_arg; |
355 |
4 |
pfh_vut.error_f = vut->error_f; |
356 |
4 |
vut->P_arg = NULL; |
357 |
|
|
358 |
4 |
AZ(atexit(vut_vpf_remove)); |
359 |
4 |
} |
360 |
79 |
} |
361 |
|
|
362 |
|
void |
363 |
79 |
VUT_Fini(struct VUT **vutp) |
364 |
|
{ |
365 |
|
struct VUT *vut; |
366 |
|
|
367 |
79 |
TAKE_OBJ_NOTNULL(vut, vutp, VUT_MAGIC); |
368 |
79 |
AN(vut->progname); |
369 |
|
|
370 |
79 |
free(vut->n_arg); |
371 |
79 |
free(vut->q_arg); |
372 |
79 |
free(vut->r_arg); |
373 |
79 |
free(vut->t_arg); |
374 |
79 |
AZ(vut->P_arg); |
375 |
|
|
376 |
79 |
if (vut->vslq) |
377 |
79 |
VSLQ_Delete(&vut->vslq); |
378 |
79 |
if (vut->vsl) |
379 |
79 |
VSL_Delete(vut->vsl); |
380 |
79 |
if (vut->vsm) |
381 |
74 |
VSM_Destroy(&vut->vsm); |
382 |
|
|
383 |
79 |
FREE_OBJ(vut); |
384 |
79 |
} |
385 |
|
|
386 |
|
static void |
387 |
0 |
vut_CursorError(struct VUT *vut, vtim_mono *last) |
388 |
|
{ |
389 |
|
const char *diag; |
390 |
|
vtim_mono now; |
391 |
|
|
392 |
0 |
CHECK_OBJ_NOTNULL(vut, VUT_MAGIC); |
393 |
0 |
AN(vut->vsl); |
394 |
0 |
AN(last); |
395 |
|
|
396 |
0 |
diag = VSL_Error(vut->vsl); |
397 |
0 |
if (diag == NULL) |
398 |
0 |
diag = "Missing diagnostic"; |
399 |
|
|
400 |
0 |
now = VTIM_mono(); |
401 |
0 |
if (isnan(*last) || *last + 1 < now) { |
402 |
0 |
fprintf(stderr, "Failed to acquire log: %s\n", diag); |
403 |
0 |
*last = now; |
404 |
0 |
} |
405 |
0 |
} |
406 |
|
|
407 |
|
int |
408 |
79 |
VUT_Main(struct VUT *vut) |
409 |
|
{ |
410 |
|
struct VSL_cursor *c; |
411 |
79 |
int i = -1; |
412 |
79 |
int hascursor = -1; |
413 |
79 |
vtim_mono t_failcursor = NAN; |
414 |
|
|
415 |
79 |
CHECK_OBJ_NOTNULL(vut, VUT_MAGIC); |
416 |
79 |
AN(vut->vslq); |
417 |
|
|
418 |
7157 |
while (!VSIG_int && !VSIG_term) { |
419 |
7145 |
if (VSIG_hup != vut->last_sighup) { |
420 |
|
/* sighup callback */ |
421 |
4 |
vut->last_sighup = VSIG_hup; |
422 |
4 |
if (vut->sighup_f != NULL) |
423 |
2 |
i = vut->sighup_f(vut); |
424 |
|
else |
425 |
2 |
i = 1; |
426 |
4 |
if (i) |
427 |
2 |
break; |
428 |
2 |
} |
429 |
|
|
430 |
7143 |
if (VSIG_usr1 != vut->last_sigusr1) { |
431 |
|
/* Flush and report any incomplete records */ |
432 |
0 |
vut->last_sigusr1 = VSIG_usr1; |
433 |
0 |
(void)VSLQ_Flush(vut->vslq, vut_dispatch, vut); |
434 |
0 |
} |
435 |
|
|
436 |
|
/* We must repeatedly call VSM_Status() when !hascursor |
437 |
|
* to make VSM discover our segment. |
438 |
|
* |
439 |
|
* XXX consider moving the error handling to VSLQ_Dispatch. |
440 |
|
* or some other VSL utility function |
441 |
|
* Reasons: |
442 |
|
* |
443 |
|
* - it does not seem to make much sense to call VSM_StillValid |
444 |
|
* in vsl if that can only detect invalid segments after |
445 |
|
* VSM_Status has run, so it appears both should be |
446 |
|
* consolidated |
447 |
|
* |
448 |
|
* - not all VSL Clients will use VUT, yet the log abandoned/ |
449 |
|
* overrun situation will be occur for all of them. |
450 |
|
*/ |
451 |
|
|
452 |
7143 |
if (vut->vsm != NULL && |
453 |
7138 |
(VSM_Status(vut->vsm) & VSM_WRK_RESTARTED)) { |
454 |
0 |
if (hascursor < 1) { |
455 |
0 |
fprintf(stderr, "Log abandoned (vsm)\n"); |
456 |
0 |
VSLQ_SetCursor(vut->vslq, NULL); |
457 |
0 |
hascursor = 0; |
458 |
0 |
} |
459 |
0 |
} |
460 |
7143 |
if (vut->vsm != NULL && hascursor < 1) { |
461 |
|
/* Reconnect VSM */ |
462 |
74 |
AZ(vut->r_arg); |
463 |
74 |
VTIM_sleep(0.1); |
464 |
148 |
c = VSL_CursorVSM(vut->vsl, vut->vsm, |
465 |
74 |
(vut->d_opt ? VSL_COPT_TAILSTOP : VSL_COPT_TAIL) |
466 |
74 |
| VSL_COPT_BATCH); |
467 |
74 |
if (c == NULL) { |
468 |
0 |
vut_CursorError(vut, &t_failcursor); |
469 |
0 |
VSL_ResetError(vut->vsl); |
470 |
0 |
continue; |
471 |
|
} |
472 |
74 |
if (hascursor >= 0) |
473 |
0 |
fprintf(stderr, "Log reacquired\n"); |
474 |
74 |
hascursor = 1; |
475 |
74 |
VSLQ_SetCursor(vut->vslq, &c); |
476 |
74 |
AZ(c); |
477 |
74 |
} |
478 |
|
|
479 |
7143 |
do |
480 |
18108 |
i = VSLQ_Dispatch(vut->vslq, vut_dispatch, vut); |
481 |
10965 |
while (i == vsl_more && |
482 |
3822 |
VSIG_usr1 == vut->last_sigusr1 && |
483 |
3822 |
VSIG_hup == vut->last_sighup); |
484 |
|
|
485 |
7143 |
if (i == vsl_more) |
486 |
0 |
continue; |
487 |
7143 |
else if (i == vsl_end) { |
488 |
7078 |
if (vut->idle_f) { |
489 |
3174 |
i = vut->idle_f(vut); |
490 |
3174 |
if (i) |
491 |
0 |
break; |
492 |
3174 |
} |
493 |
7078 |
VTIM_sleep(0.01); |
494 |
7078 |
continue; |
495 |
65 |
} else if (i == vsl_e_eof) |
496 |
65 |
break; |
497 |
|
|
498 |
0 |
if (vut->vsm == NULL) |
499 |
0 |
break; |
500 |
|
|
501 |
|
/* XXX: Make continuation optional */ |
502 |
|
|
503 |
0 |
(void)VSLQ_Flush(vut->vslq, vut_dispatch, vut); |
504 |
|
|
505 |
0 |
if (i == vsl_e_abandon) { |
506 |
0 |
fprintf(stderr, "Log abandoned (vsl)\n"); |
507 |
0 |
VSLQ_SetCursor(vut->vslq, NULL); |
508 |
0 |
hascursor = 0; |
509 |
0 |
} else if (i == vsl_e_overrun) { |
510 |
0 |
fprintf(stderr, "Log overrun\n"); |
511 |
0 |
VSLQ_SetCursor(vut->vslq, NULL); |
512 |
0 |
hascursor = 0; |
513 |
0 |
} else |
514 |
0 |
fprintf(stderr, "Error %d from VSLQ_Dispatch()", i); |
515 |
|
} |
516 |
|
|
517 |
79 |
return (i); |
518 |
|
} |
519 |
|
|
520 |
|
/**********************************************************************/ |
521 |
|
|
522 |
|
void v_noreturn_ |
523 |
10 |
VUT_Usage(const struct VUT *vut, const struct vopt_spec *voc, int status) |
524 |
|
{ |
525 |
|
const char **opt; |
526 |
|
|
527 |
10 |
fprintf(stderr, "Usage: %s <options>\n\n", vut->progname); |
528 |
10 |
fprintf(stderr, "Options:\n"); |
529 |
244 |
for (opt = voc->vopt_usage; *opt != NULL; opt += 2) |
530 |
234 |
fprintf(stderr, " %-25s %s\n", *opt, *(opt + 1)); |
531 |
10 |
exit(status); |
532 |
|
} |
533 |
|
|
534 |
|
/**********************************************************************/ |
535 |
|
|
536 |
|
|
537 |
|
static void |
538 |
107 |
print_nobrackets(const char *s) |
539 |
|
{ |
540 |
|
const char *e; |
541 |
|
|
542 |
|
/* Remove whitespace */ |
543 |
107 |
while (isspace(*s)) |
544 |
0 |
s++; |
545 |
107 |
e = s + strlen(s); |
546 |
107 |
while (e > s && isspace(e[-1])) |
547 |
0 |
e--; |
548 |
|
|
549 |
|
/* Remove outer layer brackets if present */ |
550 |
107 |
if (e > s && *s == '[' && e[-1] == ']') { |
551 |
107 |
s++; |
552 |
107 |
e--; |
553 |
107 |
} |
554 |
|
|
555 |
107 |
printf("%.*s", (int)(e - s), s); |
556 |
107 |
} |
557 |
|
|
558 |
|
static void |
559 |
107 |
print_tabbed(const char *string, int tabs) |
560 |
|
{ |
561 |
|
int i; |
562 |
|
const char *c; |
563 |
|
|
564 |
16737 |
for (c = string; *c; c++) { |
565 |
16630 |
if (c == string || *(c - 1) == '\n') |
566 |
230 |
for (i = 0; i < tabs; i++) |
567 |
230 |
printf("\t"); |
568 |
16630 |
printf("%c", *c); |
569 |
16630 |
} |
570 |
107 |
} |
571 |
|
|
572 |
|
static void |
573 |
107 |
print_opt(const struct vopt_list *opt) |
574 |
|
{ |
575 |
107 |
print_nobrackets(opt->synopsis); |
576 |
107 |
printf("\n\n"); |
577 |
107 |
print_tabbed(opt->ldesc, 1); |
578 |
107 |
printf("\n\n"); |
579 |
107 |
} |
580 |
|
|
581 |
|
static int |
582 |
5 |
vut_synopsis(const struct vopt_spec *voc) |
583 |
|
{ |
584 |
5 |
printf(".. |synopsis| replace:: %s\n", voc->vopt_synopsis); |
585 |
5 |
return (0); |
586 |
|
} |
587 |
|
|
588 |
|
static int |
589 |
5 |
vut_options(const struct vopt_spec *voc) |
590 |
|
{ |
591 |
|
int i; |
592 |
|
|
593 |
112 |
for (i = 0; i < voc->vopt_list_n; i++) |
594 |
107 |
print_opt(&voc->vopt_list[i]); |
595 |
5 |
printf("--optstring\n" |
596 |
|
"\tPrint the optstring parameter to ``getopt(3)`` to help" |
597 |
|
" writing wrapper scripts.\n\n"); |
598 |
5 |
return (0); |
599 |
|
} |