varnish-cache/bin/varnishstat/varnishstat_curses.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
 * Author: Dag-Erling Smørgrav <des@des.no>
7
 * Author: Martin Blix Grydeland <martin@varnish-software.com>
8
 *
9
 * SPDX-License-Identifier: BSD-2-Clause
10
 *
11
 * Redistribution and use in source and binary forms, with or without
12
 * modification, are permitted provided that the following conditions
13
 * are met:
14
 * 1. Redistributions of source code must retain the above copyright
15
 *    notice, this list of conditions and the following disclaimer.
16
 * 2. Redistributions in binary form must reproduce the above copyright
17
 *    notice, this list of conditions and the following disclaimer in the
18
 *    documentation and/or other materials provided with the distribution.
19
 *
20
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23
 * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
24
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30
 * SUCH DAMAGE.
31
 *
32
 * Statistics output program
33
 */
34
35
#include "config.h"
36
37
#include <stdlib.h>
38
#include <unistd.h>
39
#include <string.h>
40
#include <stdint.h>
41
#include <math.h>
42
43
#include "vdef.h"
44
#include "vas.h"
45
#include "miniobj.h"
46
#include "vqueue.h"
47
#include "vtim.h"
48
#include "vapi/vsig.h"
49
50
#include "varnishstat.h"
51
#include "vcurses.h"
52
53
#define LINES_STATUS            3
54
#define LINES_BAR_T             1
55
#define LINES_BAR_B             1
56
#define LINES_INFO              3
57
#define LINES_POINTS_MIN        3
58
59
#define N_COL                   6
60
#define COLW                    14
61
#define COLW_NAME_MIN           24
62
63
#define VALUE_MAX               999999999999
64
65
#define REBUILD_NEXT            (1u << 0)
66
#define REBUILD_FIRST           (1u << 1)
67
68
enum kb_e {
69
#define BINDING(name, desc) KB_ ## name,
70
#define BINDING_SIG
71
#include "varnishstat_bindings.h"
72
};
73
74
struct ma {
75
        unsigned n, nmax;
76
        double acc;
77
};
78
79
struct pt {
80
        unsigned                magic;
81
#define PT_MAGIC                0x41698E4F
82
        VTAILQ_ENTRY(pt)        list;
83
84
        const struct VSC_point  *vpt;
85
86
        char                    seen;
87
88
        uint64_t                cur, last;
89
        double                  t_cur, t_last;
90
        double                  chg, avg;
91
92
        struct ma               ma_10, ma_100, ma_1000;
93
};
94
95
struct hitrate {
96
        uint64_t lhit, lmiss;
97
        struct ma hr_10;
98
        struct ma hr_100;
99
        struct ma hr_1000;
100
};
101
static struct hitrate hitrate;
102
103
static VTAILQ_HEAD(, pt) ptlist = VTAILQ_HEAD_INITIALIZER(ptlist);
104
static int n_ptlist = 0;
105
static int n_ptarray = 0;
106
static struct pt **ptarray = NULL;
107
static const volatile uint64_t *mgt_uptime;
108
static const volatile uint64_t *main_uptime;
109
static const volatile uint64_t *main_cache_hit;
110
static const volatile uint64_t *main_cache_miss;
111
112
static int l_status, l_bar_t, l_points, l_bar_b, l_info;
113
static unsigned colw_name = COLW_NAME_MIN;
114
static WINDOW *w_status = NULL;
115
static WINDOW *w_bar_t = NULL;
116
static WINDOW *w_points = NULL;
117
static WINDOW *w_bar_b = NULL;
118
static WINDOW *w_info = NULL;
119
120
static const struct VSC_level_desc *verbosity;
121
static int show_help = 0;
122
static int help_line = 0;
123
static int keep_running = 1;
124
static int hide_unseen = 1;
125
static int raw_vsc = 0;
126
static int page_start = 0;
127
static int current = 0;
128
static int rebuild = 0;
129
static int redraw = 0;
130
static int sample = 0;
131
static int reset_averages = 0;
132
static int scale = 1;
133
static double t_sample = 0.;
134
static double interval = 1.;
135
static unsigned vsm_status = 0;
136
137
#define NOTIF_MAXLEN 256
138
static char notification_message[NOTIF_MAXLEN] = "";
139
static vtim_mono notification_eol = 0.0;
140
141
static void
142 252
init_hitrate(void)
143
{
144 252
        memset(&hitrate, 0, sizeof (struct hitrate));
145 252
        if (main_cache_hit != NULL) {
146 252
                hitrate.lhit = *main_cache_hit;
147 252
                hitrate.lmiss = *main_cache_miss;
148 252
        }
149 252
        hitrate.hr_10.nmax = 10;
150 252
        hitrate.hr_100.nmax = 100;
151 252
        hitrate.hr_1000.nmax = 1000;
152 252
}
153
154
static void
155 239379
update_ma(struct ma *ma, double val)
156
{
157 239379
        AN(ma);
158 239379
        AN(ma->nmax);
159 239379
        if (ma->n < ma->nmax)
160 239379
                ma->n++;
161 239379
        ma->acc += (val - ma->acc) / (double)ma->n;
162 239379
}
163
164
static void
165 2078
update_position(void)
166
{
167
        int old_current, old_page_start;
168
169 2078
        old_current = current;
170 2078
        old_page_start = page_start;
171
172 2078
        if (n_ptarray == 0) {
173 949
                current = 0;
174 949
                page_start = 0;
175 949
        } else {
176 1129
                current = vlimit_t(int, current, 0, n_ptarray - 1);
177 1129
                page_start = vmin(page_start, current);
178 1129
                if (current > page_start + (l_points - 1))
179 72
                        page_start = current - (l_points - 1);
180 1129
                page_start = vlimit_t(int, page_start, 0, n_ptarray - 1);
181
        }
182
183 2078
        if (current != old_current || page_start != old_page_start)
184 409
                redraw = 1;
185 2078
}
186
187
static void
188 697
delete_pt_array(void)
189
{
190 697
        if (ptarray != NULL)
191 697
                free(ptarray);
192 697
        ptarray = NULL;
193 697
        n_ptarray = 0;
194
195 697
        update_position();
196 697
}
197
198
static void
199 949
build_pt_array(void)
200
{
201
        int i;
202
        struct pt *pt;
203 949
        struct pt *pt_current = NULL;
204 949
        int current_line = 0;
205
206 949
        if (current < n_ptarray) {
207 445
                pt_current = ptarray[current];
208 445
                current_line = current - page_start;
209 445
        }
210
211 949
        if (ptarray != NULL)
212 697
                delete_pt_array();
213 949
        AZ(n_ptarray);
214 949
        ptarray = calloc(n_ptlist, sizeof *ptarray);
215 949
        AN(ptarray);
216
217 319629
        VTAILQ_FOREACH(pt, &ptlist, list) {
218 318680
                CHECK_OBJ_NOTNULL(pt, PT_MAGIC);
219 318680
                if (pt->vpt->level > verbosity) {
220 188571
                        if (has_f && (rebuild & REBUILD_FIRST))
221 216
                                verbosity = VSC_ChangeLevel(verbosity,
222 108
                                    pt->vpt->level - verbosity);
223
                        else
224 188463
                                continue;
225 108
                }
226 130217
                if (!pt->seen && hide_unseen)
227 105004
                        continue;
228 25213
                assert(n_ptarray < n_ptlist);
229 25213
                ptarray[n_ptarray++] = pt;
230 25213
        }
231 949
        assert(n_ptarray <= n_ptlist);
232
233 15642
        for (i = 0; pt_current != NULL && i < n_ptarray; i++)
234 15138
                if (ptarray[i] == pt_current)
235 445
                        break;
236 949
        current = i;
237 949
        page_start = current - current_line;
238 949
        update_position();
239
240 949
        rebuild = 0;
241 949
        redraw = 1;
242 949
}
243
244
static void
245 1090
sample_points(void)
246
{
247
        struct pt *pt;
248
        uint64_t v;
249
250 373350
        VTAILQ_FOREACH(pt, &ptlist, list) {
251 372260
                AN(pt->vpt);
252 372260
                AN(pt->vpt->ptr);
253 372260
                v = VSC_Value(pt->vpt);
254 372260
                if (v == 0 && !pt->seen)
255 276431
                        continue;
256 95829
                if (!pt->seen) {
257 22278
                        pt->seen = 1;
258 22278
                        rebuild = REBUILD_NEXT;
259 22278
                }
260 95829
                pt->last = pt->cur;
261 95829
                pt->cur = v;
262 95829
                pt->t_last = pt->t_cur;
263 95829
                pt->t_cur = VTIM_mono();
264
265 95829
                if (reset_averages) {
266 4370
                        pt->chg = 0;
267 4370
                        pt->ma_10.n = 0;
268 4370
                        pt->ma_100.n = 0;
269 4370
                        pt->ma_1000.n = 0;
270 4370
                }
271 95829
                if (pt->t_last)
272 147102
                        pt->chg = ((int64_t)pt->cur - (int64_t)pt->last) /
273 73551
                            (pt->t_cur - pt->t_last);
274
275 95829
                if (pt->vpt->semantics == 'g') {
276 26230
                        pt->avg = 0.;
277 26230
                        update_ma(&pt->ma_10, (int64_t)pt->cur);
278 26230
                        update_ma(&pt->ma_100, (int64_t)pt->cur);
279 26230
                        update_ma(&pt->ma_1000, (int64_t)pt->cur);
280 95829
                } else if (pt->vpt->semantics == 'c') {
281 68689
                        if (main_uptime != NULL && *main_uptime)
282 59077
                                pt->avg = pt->cur / (double)*main_uptime;
283
                        else
284 9612
                                pt->avg = 0.;
285 68689
                        if (pt->t_last) {
286 52473
                                update_ma(&pt->ma_10, pt->chg);
287 52473
                                update_ma(&pt->ma_100, pt->chg);
288 52473
                                update_ma(&pt->ma_1000, pt->chg);
289 52473
                        }
290 68689
                }
291 95829
        }
292 1090
}
293
294
static void
295 1090
sample_hitrate(void)
296
{
297
        double hr, mr, ratio;
298
        uint64_t hit, miss;
299
300 1090
        if (main_cache_hit == NULL)
301 0
                return;
302
303 1090
        hit = *main_cache_hit;
304 1090
        miss = *main_cache_miss;
305 1090
        hr = hit - hitrate.lhit;
306 1090
        mr = miss - hitrate.lmiss;
307 1090
        hitrate.lhit = hit;
308 1090
        hitrate.lmiss = miss;
309
310 1090
        if (hr + mr != 0)
311 149
                ratio = hr / (hr + mr);
312
        else
313 941
                ratio = 0;
314 1090
        if (reset_averages) {
315 36
                hitrate.hr_10.n = 0;
316 36
                hitrate.hr_100.n = 0;
317 36
                hitrate.hr_1000.n = 0;
318 36
        }
319 1090
        update_ma(&hitrate.hr_10, ratio);
320 1090
        update_ma(&hitrate.hr_100, ratio);
321 1090
        update_ma(&hitrate.hr_1000, ratio);
322 1090
}
323
324
static void
325 1090
sample_data(void)
326
{
327 1090
        t_sample = VTIM_mono();
328 1090
        sample = 0;
329 1090
        redraw = 1;
330 1090
        sample_points();
331 1090
        sample_hitrate();
332 1090
        reset_averages = 0;
333 1090
}
334
335
static void
336 1440
destroy_window(WINDOW **w)
337
{
338
339 1440
        AN(w);
340 1440
        if (*w == NULL)
341 1260
                return;
342 180
        AC(delwin(*w));
343 180
        *w = NULL;
344 1440
}
345
346
static void
347 288
make_windows(void)
348
{
349
        int Y, X;
350
        int y;
351
        int y_status, y_bar_t, y_points, y_bar_b, y_info;
352
353 288
        destroy_window(&w_status);
354 288
        destroy_window(&w_bar_t);
355 288
        destroy_window(&w_points);
356 288
        destroy_window(&w_bar_b);
357 288
        destroy_window(&w_info);
358
359 288
        Y = LINES;
360 288
        X = COLS;
361
362 288
        l_status = LINES_STATUS;
363 288
        l_bar_t = LINES_BAR_T;
364 288
        l_bar_b = LINES_BAR_B;
365 288
        l_info = LINES_INFO;
366 288
        l_points = Y - (l_status + l_bar_t + l_bar_b + l_info);
367 288
        if (l_points < LINES_POINTS_MIN) {
368 0
                l_points += l_info;
369 0
                l_info = 0;
370 0
        }
371 288
        l_points = vmax(l_points, LINES_POINTS_MIN);
372
373 288
        y = 0;
374 288
        y_status = y;
375 288
        y += l_status;
376 288
        y_bar_t = y;
377 288
        y += l_bar_t;
378 288
        y_points = y;
379 288
        y += l_points;
380 288
        y_bar_b = y;
381 288
        y += l_bar_b;
382 288
        y_info = y;
383 288
        y += l_info;
384 288
        assert(y >= Y);
385
386 288
        w_status = newwin(l_status, X, y_status, 0);
387 288
        AN(w_status);
388 288
        AC(nodelay(w_status, 1));
389 288
        AC(keypad(w_status, 1));
390 288
        AC(wnoutrefresh(w_status));
391
392 288
        w_bar_t = newwin(l_bar_t, X, y_bar_t, 0);
393 288
        AN(w_bar_t);
394 288
        wbkgd(w_bar_t, A_REVERSE);
395 288
        AC(wnoutrefresh(w_bar_t));
396
397 288
        w_points = newwin(l_points, X, y_points, 0);
398 288
        AN(w_points);
399 288
        AC(wnoutrefresh(w_points));
400
401 288
        w_bar_b = newwin(l_bar_b, X, y_bar_b, 0);
402 288
        AN(w_bar_b);
403 288
        wbkgd(w_bar_b, A_REVERSE);
404 288
        AC(wnoutrefresh(w_bar_b));
405
406 288
        if (l_info) {
407 288
                w_info = newwin(l_info, X, y_info, 0);
408 288
                AN(w_info);
409 288
                AC(wnoutrefresh(w_info));
410 288
        }
411
412 288
        if (X - COLW_NAME_MIN > N_COL * COLW)
413 36
                colw_name = X - (N_COL * COLW);
414
        else
415 252
                colw_name = COLW_NAME_MIN;
416
417 288
        redraw = 1;
418 288
}
419
420
static void
421 2858
print_duration(WINDOW *w, uint64_t t)
422
{
423
424 2858
        IC(wprintw(w, "%4ju+%02ju:%02ju:%02ju",
425
            (uintmax_t)t / 86400, (uintmax_t)(t % 86400) / 3600,
426
            (uintmax_t)(t % 3600) / 60, (uintmax_t)t % 60));
427 2858
}
428
429
static void
430 3408
running(WINDOW *w, uint64_t up, int flg)
431
{
432 3408
        if (vsm_status & flg) {
433 2822
                print_duration(w_status, up);
434 2822
        } else {
435 586
                wattron(w, A_STANDOUT);
436 586
                IC(wprintw(w, "  Not Running"));
437 586
                wattroff(w, A_STANDOUT);
438
        }
439 3408
}
440
441
static void
442 1704
draw_status(void)
443
{
444 1704
        uint64_t up_mgt = 0;
445 1704
        uint64_t up_chld = 0;
446
447 1704
        AN(w_status);
448
449 1704
        AC(werase(w_status));
450
451 1704
        if (mgt_uptime != NULL)
452 1704
                up_mgt = *mgt_uptime;
453 1704
        if (main_uptime != NULL)
454 1704
                up_chld = *main_uptime;
455
456 1704
        IC(mvwprintw(w_status, 0, 0, "Uptime mgt:   "));
457 1704
        running(w_status, up_mgt, VSM_MGT_RUNNING);
458 1704
        IC(mvwprintw(w_status, 1, 0, "Uptime child: "));
459 1704
        running(w_status, up_chld, VSM_WRK_RUNNING);
460 1704
        IC(mvwprintw(w_status, 2, 0, "Press <h> to toggle help screen"));
461
462 1704
        if (VTIM_mono() < notification_eol)
463 276
                mvwaddstr(w_status, 2, 0, notification_message);
464
465 1704
        if (COLS > 70) {
466 1704
                IC(mvwprintw(w_status, 0, getmaxx(w_status) - 37,
467
                    "Hitrate n: %8u %8u %8u", hitrate.hr_10.n, hitrate.hr_100.n,
468
                    hitrate.hr_1000.n));
469 1704
                IC(mvwprintw(w_status, 1, getmaxx(w_status) - 37,
470
                    "   avg(n): %8.4f %8.4f %8.4f", hitrate.hr_10.acc,
471
                    hitrate.hr_100.acc, hitrate.hr_1000.acc));
472 1704
        }
473
474 1704
        AC(wnoutrefresh(w_status));
475 1704
}
476
477
static void
478 1596
draw_bar_t(void)
479
{
480
        int X, x;
481
        enum {
482
                COL_CUR,
483
                COL_CHG,
484
                COL_AVG,
485
                COL_MA10,
486
                COL_MA100,
487
                COL_MA1000,
488
                COL_LAST
489
        } col;
490
491 1596
        AN(w_bar_t);
492
493 1596
        X = getmaxx(w_bar_t);
494 1596
        x = 0;
495 1596
        AC(werase(w_bar_t));
496 1596
        if (page_start > 0)
497 1026
                IC(mvwprintw(w_bar_t, 0, x, "^^^"));
498 1596
        x += 4;
499 1596
        IC(mvwprintw(w_bar_t, 0, x, "%.*s", colw_name - 4, "NAME"));
500 1596
        x += colw_name - 4;
501 1596
        col = COL_CUR;
502 8340
        while (col < COL_LAST) {
503 8160
                if (X - x < COLW)
504 1416
                        break;
505 6744
                switch (col) {
506
                case COL_CUR:
507 1596
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "CURRENT"));
508 1596
                        break;
509
                case COL_CHG:
510 1596
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "CHANGE"));
511 1596
                        break;
512
                case COL_AVG:
513 1596
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVERAGE"));
514 1596
                        break;
515
                case COL_MA10:
516 1596
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVG_10"));
517 1596
                        break;
518
                case COL_MA100:
519 180
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVG_100"));
520 180
                        break;
521
                case COL_MA1000:
522 180
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVG_1000"));
523 180
                        break;
524
                default:
525 0
                        break;
526
                }
527 6744
                x += COLW;
528 6744
                col++;
529
        }
530
531 1596
        AC(wnoutrefresh(w_bar_t));
532 1596
}
533
534
static void
535 11534
draw_line_default(WINDOW *w, int y, int x, int X, const struct pt *pt)
536
{
537
        enum {
538
                COL_CUR,
539
                COL_CHG,
540
                COL_AVG,
541
                COL_MA10,
542
                COL_MA100,
543
                COL_MA1000,
544
                COL_LAST
545
        } col;
546
547 11534
        AN(w);
548 11534
        AN(pt);
549
550 11534
        col = COL_CUR;
551 60982
        while (col < COL_LAST) {
552 59326
                if (X - x < COLW)
553 9878
                        break;
554 49448
                switch (col) {
555
                case COL_CUR:
556 11534
                        IC(mvwprintw(w, y, x, " %12ju", (uintmax_t)pt->cur));
557 11534
                        break;
558
                case COL_CHG:
559 11534
                        if (pt->t_last)
560 9499
                                IC(mvwprintw(w, y, x, " %12.2f", pt->chg));
561
                        else
562 2035
                                IC(mvwprintw(w, y, x, " %12s", ".  "));
563 11534
                        break;
564
                case COL_AVG:
565 11534
                        if (pt->avg)
566 3789
                                IC(mvwprintw(w, y, x, " %12.2f", pt->avg));
567
                        else
568 7745
                                IC(mvwprintw(w, y, x, " %12s", ".  "));
569 11534
                        break;
570
                case COL_MA10:
571 11534
                        IC(mvwprintw(w, y, x, " %12.2f", pt->ma_10.acc));
572 11534
                        break;
573
                case COL_MA100:
574 1656
                        IC(mvwprintw(w, y, x, " %12.2f", pt->ma_100.acc));
575 1656
                        break;
576
                case COL_MA1000:
577 1656
                        IC(mvwprintw(w, y, x, " %12.2f", pt->ma_1000.acc));
578 1656
                        break;
579
                default:
580 0
                        break;
581
                }
582 49448
                x += COLW;
583 49448
                col++;
584
        }
585 11534
}
586
587
static double
588 7449
scale_bytes(double val, char *q)
589
{
590
        const char *p;
591
592 9475
        for (p = " KMGTPEZY"; *p; p++) {
593 9475
                if (fabs(val) < 1024.)
594 7449
                        break;
595 2026
                val /= 1024.;
596 2026
        }
597 7449
        *q = *p;
598 7449
        return (val);
599
}
600
601
static void
602 12885
print_bytes(WINDOW *w, double val)
603
{
604 12885
        char q = ' ';
605
606 12885
        if (scale)
607 7449
                val = scale_bytes(val, &q);
608 12885
        IC(wprintw(w, " %12.2f%c", val, q));
609 12885
}
610
611
static void
612 4691
print_trunc(WINDOW *w, uintmax_t val)
613
{
614 4691
        if (val > VALUE_MAX) {
615 0
                while (val > VALUE_MAX)
616 0
                        val /= 1000;
617 0
                IC(wprintw(w, " %9ju...", val));
618 0
        } else
619 4691
                IC(wprintw(w, " %12ju", val));
620 4691
}
621
622
static void
623 5287
draw_line_bytes(WINDOW *w, int y, int x, int X, const struct pt *pt)
624
{
625
        enum {
626
                COL_CUR,
627
                COL_CHG,
628
                COL_AVG,
629
                COL_MA10,
630
                COL_MA100,
631
                COL_MA1000,
632
                COL_LAST
633
        } col;
634
635 5287
        AN(w);
636 5287
        AN(pt);
637
638 5287
        col = COL_CUR;
639 28379
        while (col < COL_LAST) {
640 27407
                if (X - x < COLW)
641 4315
                        break;
642 23092
                wmove(w, y, x);
643 23092
                switch (col) {
644
                case COL_CUR:
645 5287
                        if (scale && pt->cur > 1024)
646 596
                                print_bytes(w, (double)pt->cur);
647
                        else
648 4691
                                print_trunc(w, (uintmax_t)pt->cur);
649 5287
                        break;
650
                case COL_CHG:
651 5287
                        if (pt->t_last)
652 2867
                                print_bytes(w, pt->chg);
653
                        else
654 2420
                                IC(wprintw(w, " %12s", ".  "));
655 5287
                        break;
656
                case COL_AVG:
657 5287
                        if (pt->avg)
658 2191
                                print_bytes(w, pt->avg);
659
                        else
660 3096
                                IC(wprintw(w, " %12s", ".  "));
661 5287
                        break;
662
                case COL_MA10:
663 5287
                        print_bytes(w, pt->ma_10.acc);
664 5287
                        break;
665
                case COL_MA100:
666 972
                        print_bytes(w, pt->ma_100.acc);
667 972
                        break;
668
                case COL_MA1000:
669 972
                        print_bytes(w, pt->ma_1000.acc);
670 972
                        break;
671
                default:
672 0
                        break;
673
                }
674 23092
                x += COLW;
675 23092
                col++;
676
        }
677 5287
}
678
679
static void
680 657
draw_line_bitmap(WINDOW *w, int y, int x, int X, const struct pt *pt)
681
{
682
        unsigned ch;
683
        enum {
684
                COL_VAL,
685
                COL_MAP,
686
                COL_LAST
687
        } col;
688
689 657
        AN(w);
690 657
        AN(pt);
691 657
        assert(pt->vpt->format == 'b');
692
693 657
        col = COL_VAL;
694 1971
        while (col < COL_LAST) {
695 1314
                switch (col) {
696
                case COL_VAL:
697 657
                        if (X - x < COLW)
698 0
                                return;
699 657
                        IC(mvwprintw(w, y, x, "   %10.10jx",
700
                            (uintmax_t)((pt->cur >> 24) & 0xffffffffffLL)));
701 657
                        x += COLW;
702 657
                        break;
703
                case COL_MAP:
704 657
                        if (X - x < 2 * COLW)
705 0
                                return;
706 657
                        x += (2 * COLW) - 24;
707 16425
                        for (ch = 0x800000; ch; ch >>= 1) {
708 15768
                                if (pt->cur & ch)
709 3777
                                        mvwaddch(w, y, x, 'V');
710
                                else
711 11991
                                        mvwaddch(w, y, x, '_');
712 15768
                                x++;
713 15768
                        }
714 657
                        break;
715
                default:
716 0
                        break;
717
                }
718 1314
                col++;
719
        }
720 657
}
721
722
static void
723 108
draw_line_duration(WINDOW *w, int y, int x, int X, const struct pt *pt)
724
{
725
        enum {
726
                COL_DUR,
727
                COL_LAST
728
        } col;
729
730 108
        AN(w);
731 108
        AN(pt);
732
733 108
        col = COL_DUR;
734 216
        while (col < COL_LAST) {
735 108
                if (X - x < COLW)
736 0
                        break;
737 108
                switch (col) {
738
                case COL_DUR:
739 108
                        wmove(w, y, x);
740 108
                        if (scale)
741 36
                                print_duration(w, pt->cur);
742
                        else
743 72
                                IC(wprintw(w, " %12ju", (uintmax_t)pt->cur));
744 108
                        break;
745
                default:
746 0
                        break;
747
                }
748 108
                x += COLW;
749 108
                col++;
750
        }
751 108
}
752
753
static void
754 17586
draw_line(WINDOW *w, int y, const struct pt *pt)
755
{
756
        int x, X;
757
758 17586
        assert(colw_name >= COLW_NAME_MIN);
759 17586
        X = getmaxx(w);
760 17586
        x = 0;
761 17586
        if (strlen(pt->vpt->name) > colw_name)
762 1245
                IC(mvwprintw(w, y, x, "%.*s...", colw_name - 3, pt->vpt->name));
763
        else
764 16341
                IC(mvwprintw(w, y, x, "%.*s", colw_name, pt->vpt->name));
765 17586
        x += colw_name;
766
767 17586
        switch (pt->vpt->format) {
768
        case 'b':
769 657
                draw_line_bitmap(w, y, x, X, pt);
770 657
                break;
771
        case 'B':
772 5287
                draw_line_bytes(w, y, x, X, pt);
773 5287
                break;
774
        case 'd':
775 108
                draw_line_duration(w, y, x, X, pt);
776 108
                break;
777
        default:
778 11534
                draw_line_default(w, y, x, X, pt);
779 11534
                break;
780
        }
781 17586
}
782
783
static void
784 1596
draw_points(void)
785
{
786
        int line;
787
        int n;
788
789 1596
        AN(w_points);
790
791 1596
        AC(werase(w_points));
792 1596
        if (n_ptarray == 0) {
793 252
                AC(wnoutrefresh(w_points));
794 252
                return;
795
        }
796
797 1344
        assert(current >= 0);
798 1344
        assert(current < n_ptarray);
799 1344
        assert(page_start >= 0);
800 1344
        assert(page_start < n_ptarray);
801 1344
        assert(current >= page_start);
802 1344
        assert(current - page_start < l_points);
803
804 18930
        for (line = 0; line < l_points; line++) {
805 18032
                n = line + page_start;
806 18032
                if (n >= n_ptarray)
807 446
                        break;
808 17586
                if (n == current)
809 1344
                        wattron(w_points, A_BOLD);
810 17586
                draw_line(w_points, line, ptarray[n]);
811 17586
                if (n == current)
812 1344
                        wattroff(w_points, A_BOLD);
813 17586
        }
814 1344
        AC(wnoutrefresh(w_points));
815 1596
}
816
817
static void
818 108
draw_help(void)
819
{
820
        const char *const *p;
821
        int l, y, X;
822
823 108
        if (l_points >= bindings_help_len) {
824 0
                assert(help_line == 0);
825 0
                l = bindings_help_len;
826 0
        } else {
827 108
                assert(help_line >= 0);
828 108
                assert(help_line <= bindings_help_len - l_points);
829 108
                l = l_points;
830
        }
831
832 108
        X = getmaxx(w_points);
833 108
        AC(werase(w_points));
834
835 1836
        for (y = 0, p = bindings_help + help_line; y < l; y++, p++) {
836 1728
                if (**p == '\t') {
837 900
                        IC(mvwprintw(w_points, y, 0, "    %.*s", X - 4, *p + 1));
838 900
                } else {
839 828
                        wattron(w_points, A_BOLD);
840 828
                        IC(mvwprintw(w_points, y, 0, "%.*s", X, *p));
841 828
                        wattroff(w_points, A_BOLD);
842
                }
843 1728
        }
844
845 108
        AC(wnoutrefresh(w_points));
846 108
}
847
848
static void
849 1596
draw_bar_b(void)
850
{
851
        int x, X;
852
        char buf[64];
853
854 1596
        AN(w_bar_b);
855
856 1596
        x = 0;
857 1596
        X = getmaxx(w_bar_b);
858 1596
        AC(werase(w_bar_b));
859 1596
        if (page_start + l_points < n_ptarray)
860 759
                IC(mvwprintw(w_bar_b, 0, x, "vvv"));
861 1596
        x += 4;
862 1596
        if (current < n_ptarray)
863 1344
                IC(mvwprintw(w_bar_b, 0, x, "%s", ptarray[current]->vpt->name));
864
865 1596
        bprintf(buf, "%d-%d/%d", page_start + 1,
866
            page_start + l_points < n_ptarray ?
867
                page_start + l_points : n_ptarray,
868
            n_ptarray);
869 1596
        IC(mvwprintw(w_bar_b, 0, X - strlen(buf), "%s", buf));
870 1596
        X -= strlen(buf) + 2;
871
872 1596
        if (verbosity != NULL) {
873 1596
                IC(mvwprintw(w_bar_b, 0, X - strlen(verbosity->label), "%s",
874
                    verbosity->label));
875 1596
                X -= strlen(verbosity->label) + 2;
876 1596
        }
877 1596
        if (!hide_unseen) {
878 360
                IC(mvwprintw(w_bar_b, 0, X - 6, "%s", "UNSEEN"));
879 360
                X -= 8;
880 360
        }
881 1596
        if (raw_vsc)
882 0
                IC(mvwprintw(w_bar_b, 0, X - 3, "%s", "RAW"));
883
884 1596
        AC(wnoutrefresh(w_bar_b));
885 1596
}
886
887
static void
888 1596
draw_info(void)
889
{
890
891 1596
        if (w_info == NULL)
892 0
                return;
893
894 1596
        AC(werase(w_info));
895 1596
        if (current < n_ptarray) {
896
                /* XXX: Word wrapping, and overflow handling? */
897 1344
                IC(mvwprintw(w_info, 0, 0, "%s:",
898
                    ptarray[current]->vpt->sdesc));
899 1344
                IC(mvwprintw(w_info, 1, 0, "%s",
900
                    ptarray[current]->vpt->ldesc));
901 1344
        }
902 1596
        AC(wnoutrefresh(w_info));
903 1596
}
904
905
static void
906 1704
draw_screen(void)
907
{
908 1704
        draw_status();
909 1704
        if (show_help) {
910 108
                AC(werase(w_bar_t));
911 108
                AC(werase(w_bar_b));
912 108
                AC(werase(w_info));
913 108
                AC(wnoutrefresh(w_bar_t));
914 108
                AC(wnoutrefresh(w_bar_b));
915 108
                AC(wnoutrefresh(w_info));
916 108
                draw_help();
917 108
        } else {
918 1596
                draw_bar_t();
919 1596
                draw_points();
920 1596
                draw_bar_b();
921 1596
                draw_info();
922
        }
923 1704
        AC(doupdate());
924 1704
        redraw = 0;
925 1704
}
926
927
static void
928 72
handle_common_keypress(enum kb_e kb)
929
{
930
931 72
        switch (kb) {
932
        case KB_QUIT:
933 72
                keep_running = 0;
934 72
                return;
935
        case KB_SIG_INT:
936 0
                AZ(raise(SIGINT));
937 0
                return;
938
        case KB_SIG_TSTP:
939 0
                AZ(raise(SIGTSTP));
940 0
                return;
941
        default:
942 0
                WRONG("unexpected key binding");
943 0
        }
944 72
}
945
946
static void
947 576
handle_points_keypress(struct vsc *vsc, enum kb_e kb)
948
{
949
950 576
        switch (kb) {
951
        case KB_HELP:
952 72
                show_help = 1;
953 72
                help_line = 0;
954 72
                redraw = 1;
955 72
                return;
956
        case KB_UP:
957 36
                if (current == 0)
958 0
                        return;
959 36
                current--;
960 36
                break;
961
        case KB_DOWN:
962 0
                if (current == n_ptarray - 1)
963 0
                        return;
964 0
                current++;
965 0
                break;
966
        case KB_PAGEUP:
967 36
                current -= l_points;
968 36
                page_start -= l_points;
969 36
                break;
970
        case KB_PAGEDOWN:
971 36
                current += l_points;
972 36
                if (page_start + l_points < n_ptarray - 1)
973 36
                        page_start += l_points;
974 36
                break;
975
        case KB_TOP:
976 36
                current = 0;
977 36
                break;
978
        case KB_BOTTOM:
979 72
                current = n_ptarray - 1;
980 72
                break;
981
        case KB_UNSEEN:
982 36
                hide_unseen = 1 - hide_unseen;
983 36
                rebuild = REBUILD_NEXT;
984 36
                break;
985
        case KB_RAW:
986 0
                AN(VSC_Arg(vsc, 'r', NULL));
987 0
                raw_vsc = VSC_IsRaw(vsc);
988 0
                rebuild = REBUILD_NEXT;
989 0
                break;
990
        case KB_SCALE:
991 36
                scale = 1 - scale;
992 36
                rebuild = REBUILD_NEXT;
993 36
                break;
994
        case KB_ACCEL:
995 36
                interval += 0.1;
996 36
                (void)snprintf(notification_message, NOTIF_MAXLEN,
997 36
                    "Refresh interval set to %.1f seconds.", interval);
998
999 36
                notification_eol = VTIM_mono() + 1.25;
1000 36
                break;
1001
        case KB_DECEL:
1002 36
                interval -= 0.1;
1003 36
                if (interval < 0.1)
1004 0
                        interval = 0.1;
1005 36
                (void)snprintf(notification_message, NOTIF_MAXLEN,
1006 36
                    "Refresh interval set to %.1f seconds.", interval);
1007
1008 36
                notification_eol = VTIM_mono() + 1.25;
1009 36
                break;
1010
        case KB_VERBOSE:
1011 36
                verbosity = VSC_ChangeLevel(verbosity, 1);
1012 36
                rebuild = REBUILD_NEXT;
1013 36
                break;
1014
        case KB_QUIET:
1015 0
                verbosity = VSC_ChangeLevel(verbosity, -1);
1016 0
                rebuild = REBUILD_NEXT;
1017 0
                break;
1018
        case KB_SAMPLE:
1019 0
                sample = 1;
1020 0
                return;
1021
        case KB_RESET_AVERAGES:
1022 36
                reset_averages = 1;
1023 36
                return;
1024
        case KB_QUIT:
1025
        case KB_SIG_INT:
1026
        case KB_SIG_TSTP:
1027 72
                handle_common_keypress(kb);
1028 72
                return;
1029
        default:
1030 0
                WRONG("unhandled key binding");
1031 0
        }
1032
1033 396
        update_position();
1034 396
        redraw = 1;
1035 576
}
1036
1037
static void
1038 108
handle_help_keypress(enum kb_e kb)
1039
{
1040 108
        int hl = help_line;
1041
1042 108
        switch (kb) {
1043
        case KB_HELP:
1044 72
                show_help = 0;
1045 72
                redraw = 1;
1046 72
                return;
1047
        case KB_UP:
1048 0
                help_line--;
1049 0
                break;
1050
        case KB_DOWN:
1051 0
                help_line++;
1052 0
                break;
1053
        case KB_PAGEUP:
1054 0
                help_line -= l_points;
1055 0
                break;
1056
        case KB_PAGEDOWN:
1057 0
                help_line += l_points;
1058 0
                break;
1059
        case KB_TOP:
1060 0
                help_line = 0;
1061 0
                break;
1062
        case KB_BOTTOM:
1063 36
                help_line = bindings_help_len;
1064 36
                break;
1065
        case KB_UNSEEN:
1066
        case KB_RAW:
1067
        case KB_SCALE:
1068
        case KB_ACCEL:
1069
        case KB_DECEL:
1070
        case KB_VERBOSE:
1071
        case KB_QUIET:
1072
        case KB_SAMPLE:
1073
        case KB_RESET_AVERAGES:
1074 0
                break;
1075
        case KB_QUIT:
1076
        case KB_SIG_INT:
1077
        case KB_SIG_TSTP:
1078 0
                handle_common_keypress(kb);
1079 0
                return;
1080
        default:
1081 0
                WRONG("unhandled key binding");
1082 0
        }
1083
1084 36
        help_line = vlimit_t(int, help_line, 0, bindings_help_len - l_points);
1085 36
        redraw = (help_line != hl);
1086 108
}
1087
1088
static void
1089 684
handle_keypress(struct vsc *vsc, int ch)
1090
{
1091
        enum kb_e kb;
1092
1093 684
        switch (ch) {
1094
#define BINDING_KEY(chr, name, or)      \
1095
        case chr:
1096
#define BINDING(name, desc)             \
1097
                kb = KB_ ## name;       \
1098
                break;
1099
#define BINDING_SIG
1100
#include "varnishstat_bindings.h"
1101
        default:
1102
                return;
1103
        }
1104
1105 684
        if (show_help)
1106 108
                handle_help_keypress(kb);
1107
        else
1108 576
                handle_points_keypress(vsc, kb);
1109 684
}
1110
1111
static void * v_matchproto_(VSC_new_f)
1112 81540
newpt(void *priv, const struct VSC_point *const vpt)
1113
{
1114
        struct pt *pt;
1115
1116 81540
        AZ(priv);
1117 81540
        ALLOC_OBJ(pt, PT_MAGIC);
1118 81540
        rebuild |= REBUILD_NEXT;
1119 81540
        AN(pt);
1120 81540
        pt->vpt = vpt;
1121 81540
        pt->last = VSC_Value(vpt);
1122 81540
        pt->ma_10.nmax = 10;
1123 81540
        pt->ma_100.nmax = 100;
1124 81540
        pt->ma_1000.nmax = 1000;
1125
1126 81540
        VTAILQ_INSERT_TAIL(&ptlist, pt, list);
1127 81540
        n_ptlist++;
1128
1129 81540
        AZ(strcmp(vpt->ctype, "uint64_t"));
1130
1131 81540
        if (!strcmp(vpt->name, "MGT.uptime"))
1132 252
                mgt_uptime = vpt->ptr;
1133 81540
        if (!strcmp(vpt->name, "MAIN.uptime"))
1134 252
                main_uptime = vpt->ptr;
1135 81540
        if (!strcmp(vpt->name, "MAIN.cache_hit"))
1136 252
                main_cache_hit = vpt->ptr;
1137 81540
        if (!strcmp(vpt->name, "MAIN.cache_miss"))
1138 252
                main_cache_miss = vpt->ptr;
1139 81540
        return (pt);
1140
}
1141
1142
static void v_matchproto_(VSC_destroy_f)
1143 81540
delpt(void *priv, const struct VSC_point *const vpt)
1144
{
1145
        struct pt *pt;
1146
1147 81540
        AZ(priv);
1148 81540
        CAST_OBJ_NOTNULL(pt, vpt->priv, PT_MAGIC);
1149 81540
        rebuild |= REBUILD_NEXT;
1150 81540
        VTAILQ_REMOVE(&ptlist, pt, list);
1151 81540
        n_ptlist--;
1152 81540
        FREE_OBJ(pt);
1153 81540
        if (vpt->ptr == mgt_uptime)
1154 252
                mgt_uptime = NULL;
1155 81540
        if (vpt->ptr == main_uptime)
1156 252
                main_uptime = NULL;
1157 81540
        if (vpt->ptr == main_cache_hit)
1158 252
                main_cache_hit = NULL;
1159 81540
        if (vpt->ptr == main_cache_miss)
1160 252
                main_cache_miss = NULL;
1161 81540
}
1162
1163
void
1164 252
do_curses(struct vsm *vsm, struct vsc *vsc)
1165
{
1166
        long t;
1167
        int ch;
1168
        double now;
1169
1170 252
        verbosity = VSC_ChangeLevel(NULL, 0);
1171
1172 252
        (void)initscr();
1173 252
        AC(raw());
1174 252
        AC(noecho());
1175 252
        AC(nonl());
1176 252
        IC(curs_set(0));
1177
1178 252
        make_windows();
1179 252
        AC(doupdate());
1180
1181 252
        VSC_State(vsc, newpt, delpt, NULL);
1182
1183 252
        raw_vsc = VSC_IsRaw(vsc);
1184 252
        rebuild |= REBUILD_FIRST;
1185 252
        (void)VSC_Iter(vsc, vsm, NULL, NULL);
1186 252
        build_pt_array();
1187 252
        init_hitrate();
1188
1189 1989
        while (keep_running && !VSIG_int && !VSIG_term && !VSIG_hup) {
1190 1737
                (void)VSC_Iter(vsc, vsm, NULL, NULL);
1191 1737
                vsm_status = VSM_Status(vsm);
1192 1737
                if (vsm_status & (VSM_MGT_RESTARTED|VSM_WRK_RESTARTED))
1193 0
                        init_hitrate();
1194 1737
                if (rebuild)
1195 697
                        build_pt_array();
1196
1197 1737
                now = VTIM_mono();
1198 1737
                if (now - t_sample > interval)
1199 1090
                        sample = 1;
1200 1737
                if (sample)
1201 1090
                        sample_data();
1202 1737
                if (redraw)
1203 1704
                        draw_screen();
1204
1205 1737
                t = (long)((t_sample + interval - now) * 1000);
1206 1737
                wtimeout(w_status, t);
1207
1208 1737
                ch = wgetch(w_status);
1209 1737
                switch (ch) {
1210
                case ERR:
1211 1017
                        break;
1212
#ifdef KEY_RESIZE /* sigh, Solaris lacks this.. */
1213
                case KEY_RESIZE:
1214 36
                        make_windows();
1215 36
                        update_position();
1216 36
                        break;
1217
#endif
1218
                default:
1219 684
                        handle_keypress(vsc, ch);
1220 684
                        break;
1221
                }
1222
        }
1223 252
        VSC_Destroy(&vsc, vsm);
1224 252
        AN(VTAILQ_EMPTY(&ptlist));
1225 252
        VSM_Destroy(&vsm);
1226 252
        AZ(endwin());
1227 252
}