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 91
init_hitrate(void)
143
{
144 91
        memset(&hitrate, 0, sizeof (struct hitrate));
145 91
        if (main_cache_hit != NULL) {
146 91
                hitrate.lhit = *main_cache_hit;
147 91
                hitrate.lmiss = *main_cache_miss;
148 91
        }
149 91
        hitrate.hr_10.nmax = 10;
150 91
        hitrate.hr_100.nmax = 100;
151 91
        hitrate.hr_1000.nmax = 1000;
152 91
}
153
154
static void
155 88743
update_ma(struct ma *ma, double val)
156
{
157 88743
        AN(ma);
158 88743
        AN(ma->nmax);
159 88743
        if (ma->n < ma->nmax)
160 88743
                ma->n++;
161 88743
        ma->acc += (val - ma->acc) / (double)ma->n;
162 88743
}
163
164
static void
165 751
update_position(void)
166
{
167
        int old_current, old_page_start;
168
169 751
        old_current = current;
170 751
        old_page_start = page_start;
171
172 751
        if (n_ptarray == 0) {
173 343
                current = 0;
174 343
                page_start = 0;
175 343
        } else {
176 408
                current = vlimit_t(int, current, 0, n_ptarray - 1);
177 408
                page_start = vmin(page_start, current);
178 408
                if (current > page_start + (l_points - 1))
179 26
                        page_start = current - (l_points - 1);
180 408
                page_start = vlimit_t(int, page_start, 0, n_ptarray - 1);
181
        }
182
183 751
        if (current != old_current || page_start != old_page_start)
184 148
                redraw = 1;
185 751
}
186
187
static void
188 252
delete_pt_array(void)
189
{
190 252
        if (ptarray != NULL)
191 252
                free(ptarray);
192 252
        ptarray = NULL;
193 252
        n_ptarray = 0;
194
195 252
        update_position();
196 252
}
197
198
static void
199 343
build_pt_array(void)
200
{
201
        int i;
202
        struct pt *pt;
203 343
        struct pt *pt_current = NULL;
204 343
        int current_line = 0;
205
206 343
        if (current < n_ptarray) {
207 161
                pt_current = ptarray[current];
208 161
                current_line = current - page_start;
209 161
        }
210
211 343
        if (ptarray != NULL)
212 252
                delete_pt_array();
213 343
        AZ(n_ptarray);
214 343
        ptarray = calloc(n_ptlist, sizeof *ptarray);
215 343
        AN(ptarray);
216
217 115538
        VTAILQ_FOREACH(pt, &ptlist, list) {
218 115195
                CHECK_OBJ_NOTNULL(pt, PT_MAGIC);
219 115195
                if (pt->vpt->level > verbosity) {
220 68173
                        if (has_f && (rebuild & REBUILD_FIRST))
221 78
                                verbosity = VSC_ChangeLevel(verbosity,
222 39
                                    pt->vpt->level - verbosity);
223
                        else
224 68134
                                continue;
225 39
                }
226 47061
                if (!pt->seen && hide_unseen)
227 37948
                        continue;
228 9113
                assert(n_ptarray < n_ptlist);
229 9113
                ptarray[n_ptarray++] = pt;
230 9113
        }
231 343
        assert(n_ptarray <= n_ptlist);
232
233 5659
        for (i = 0; pt_current != NULL && i < n_ptarray; i++)
234 5477
                if (ptarray[i] == pt_current)
235 161
                        break;
236 343
        current = i;
237 343
        page_start = current - current_line;
238 343
        update_position();
239
240 343
        rebuild = 0;
241 343
        redraw = 1;
242 343
}
243
244
static void
245 400
sample_points(void)
246
{
247
        struct pt *pt;
248
        uint64_t v;
249
250 137255
        VTAILQ_FOREACH(pt, &ptlist, list) {
251 136855
                AN(pt->vpt);
252 136855
                AN(pt->vpt->ptr);
253 136855
                v = VSC_Value(pt->vpt);
254 136855
                if (v == 0 && !pt->seen)
255 101481
                        continue;
256 35374
                if (!pt->seen) {
257 8045
                        pt->seen = 1;
258 8045
                        rebuild = REBUILD_NEXT;
259 8045
                }
260 35374
                pt->last = pt->cur;
261 35374
                pt->cur = v;
262 35374
                pt->t_last = pt->t_cur;
263 35374
                pt->t_cur = VTIM_mono();
264
265 35374
                if (reset_averages) {
266 1576
                        pt->chg = 0;
267 1576
                        pt->ma_10.n = 0;
268 1576
                        pt->ma_100.n = 0;
269 1576
                        pt->ma_1000.n = 0;
270 1576
                }
271 35374
                if (pt->t_last)
272 54658
                        pt->chg = ((int64_t)pt->cur - (int64_t)pt->last) /
273 27329
                            (pt->t_cur - pt->t_last);
274
275 35374
                if (pt->vpt->semantics == 'g') {
276 9650
                        pt->avg = 0.;
277 9650
                        update_ma(&pt->ma_10, (int64_t)pt->cur);
278 9650
                        update_ma(&pt->ma_100, (int64_t)pt->cur);
279 9650
                        update_ma(&pt->ma_1000, (int64_t)pt->cur);
280 35374
                } else if (pt->vpt->semantics == 'c') {
281 25389
                        if (main_uptime != NULL && *main_uptime)
282 21918
                                pt->avg = pt->cur / (double)*main_uptime;
283
                        else
284 3471
                                pt->avg = 0.;
285 25389
                        if (pt->t_last) {
286 19531
                                update_ma(&pt->ma_10, pt->chg);
287 19531
                                update_ma(&pt->ma_100, pt->chg);
288 19531
                                update_ma(&pt->ma_1000, pt->chg);
289 19531
                        }
290 25389
                }
291 35374
        }
292 400
}
293
294
static void
295 400
sample_hitrate(void)
296
{
297
        double hr, mr, ratio;
298
        uint64_t hit, miss;
299
300 400
        if (main_cache_hit == NULL)
301 0
                return;
302
303 400
        hit = *main_cache_hit;
304 400
        miss = *main_cache_miss;
305 400
        hr = hit - hitrate.lhit;
306 400
        mr = miss - hitrate.lmiss;
307 400
        hitrate.lhit = hit;
308 400
        hitrate.lmiss = miss;
309
310 400
        if (hr + mr != 0)
311 53
                ratio = hr / (hr + mr);
312
        else
313 347
                ratio = 0;
314 400
        if (reset_averages) {
315 13
                hitrate.hr_10.n = 0;
316 13
                hitrate.hr_100.n = 0;
317 13
                hitrate.hr_1000.n = 0;
318 13
        }
319 400
        update_ma(&hitrate.hr_10, ratio);
320 400
        update_ma(&hitrate.hr_100, ratio);
321 400
        update_ma(&hitrate.hr_1000, ratio);
322 400
}
323
324
static void
325 400
sample_data(void)
326
{
327 400
        t_sample = VTIM_mono();
328 400
        sample = 0;
329 400
        redraw = 1;
330 400
        sample_points();
331 400
        sample_hitrate();
332 400
        reset_averages = 0;
333 400
}
334
335
static void
336 520
destroy_window(WINDOW **w)
337
{
338
339 520
        AN(w);
340 520
        if (*w == NULL)
341 455
                return;
342 65
        AC(delwin(*w));
343 65
        *w = NULL;
344 520
}
345
346
static void
347 104
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 104
        destroy_window(&w_status);
354 104
        destroy_window(&w_bar_t);
355 104
        destroy_window(&w_points);
356 104
        destroy_window(&w_bar_b);
357 104
        destroy_window(&w_info);
358
359 104
        Y = LINES;
360 104
        X = COLS;
361
362 104
        l_status = LINES_STATUS;
363 104
        l_bar_t = LINES_BAR_T;
364 104
        l_bar_b = LINES_BAR_B;
365 104
        l_info = LINES_INFO;
366 104
        l_points = Y - (l_status + l_bar_t + l_bar_b + l_info);
367 104
        if (l_points < LINES_POINTS_MIN) {
368 0
                l_points += l_info;
369 0
                l_info = 0;
370 0
        }
371 104
        l_points = vmax(l_points, LINES_POINTS_MIN);
372
373 104
        y = 0;
374 104
        y_status = y;
375 104
        y += l_status;
376 104
        y_bar_t = y;
377 104
        y += l_bar_t;
378 104
        y_points = y;
379 104
        y += l_points;
380 104
        y_bar_b = y;
381 104
        y += l_bar_b;
382 104
        y_info = y;
383 104
        y += l_info;
384 104
        assert(y >= Y);
385
386 104
        w_status = newwin(l_status, X, y_status, 0);
387 104
        AN(w_status);
388 104
        AC(nodelay(w_status, 1));
389 104
        AC(keypad(w_status, 1));
390 104
        AC(wnoutrefresh(w_status));
391
392 104
        w_bar_t = newwin(l_bar_t, X, y_bar_t, 0);
393 104
        AN(w_bar_t);
394 104
        wbkgd(w_bar_t, A_REVERSE);
395 104
        AC(wnoutrefresh(w_bar_t));
396
397 104
        w_points = newwin(l_points, X, y_points, 0);
398 104
        AN(w_points);
399 104
        AC(wnoutrefresh(w_points));
400
401 104
        w_bar_b = newwin(l_bar_b, X, y_bar_b, 0);
402 104
        AN(w_bar_b);
403 104
        wbkgd(w_bar_b, A_REVERSE);
404 104
        AC(wnoutrefresh(w_bar_b));
405
406 104
        if (l_info) {
407 104
                w_info = newwin(l_info, X, y_info, 0);
408 104
                AN(w_info);
409 104
                AC(wnoutrefresh(w_info));
410 104
        }
411
412 104
        if (X - COLW_NAME_MIN > N_COL * COLW)
413 13
                colw_name = X - (N_COL * COLW);
414
        else
415 91
                colw_name = COLW_NAME_MIN;
416
417 104
        redraw = 1;
418 104
}
419
420
static void
421 1019
print_duration(WINDOW *w, uint64_t t)
422
{
423
424 1019
        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 1019
}
428
429
static void
430 1242
running(WINDOW *w, uint64_t up, int flg)
431
{
432 1242
        if (vsm_status & flg) {
433 1006
                print_duration(w_status, up);
434 1006
        } else {
435 236
                wattron(w, A_STANDOUT);
436 236
                IC(wprintw(w, "  Not Running"));
437 236
                wattroff(w, A_STANDOUT);
438
        }
439 1242
}
440
441
static void
442 621
draw_status(void)
443
{
444 621
        uint64_t up_mgt = 0;
445 621
        uint64_t up_chld = 0;
446
447 621
        AN(w_status);
448
449 621
        AC(werase(w_status));
450
451 621
        if (mgt_uptime != NULL)
452 621
                up_mgt = *mgt_uptime;
453 621
        if (main_uptime != NULL)
454 621
                up_chld = *main_uptime;
455
456 621
        IC(mvwprintw(w_status, 0, 0, "Uptime mgt:   "));
457 621
        running(w_status, up_mgt, VSM_MGT_RUNNING);
458 621
        IC(mvwprintw(w_status, 1, 0, "Uptime child: "));
459 621
        running(w_status, up_chld, VSM_WRK_RUNNING);
460 621
        IC(mvwprintw(w_status, 2, 0, "Press <h> to toggle help screen"));
461
462 621
        if (VTIM_mono() < notification_eol)
463 89
                mvwaddstr(w_status, 2, 0, notification_message);
464
465 621
        if (COLS > 70) {
466 621
                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 621
                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 621
        }
473
474 621
        AC(wnoutrefresh(w_status));
475 621
}
476
477
static void
478 582
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 582
        AN(w_bar_t);
492
493 582
        X = getmaxx(w_bar_t);
494 582
        x = 0;
495 582
        AC(werase(w_bar_t));
496 582
        if (page_start > 0)
497 377
                IC(mvwprintw(w_bar_t, 0, x, "^^^"));
498 582
        x += 4;
499 582
        IC(mvwprintw(w_bar_t, 0, x, "%.*s", colw_name - 4, "NAME"));
500 582
        x += colw_name - 4;
501 582
        col = COL_CUR;
502 3040
        while (col < COL_LAST) {
503 2975
                if (X - x < COLW)
504 517
                        break;
505 2458
                switch (col) {
506
                case COL_CUR:
507 582
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "CURRENT"));
508 582
                        break;
509
                case COL_CHG:
510 582
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "CHANGE"));
511 582
                        break;
512
                case COL_AVG:
513 582
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVERAGE"));
514 582
                        break;
515
                case COL_MA10:
516 582
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVG_10"));
517 582
                        break;
518
                case COL_MA100:
519 65
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVG_100"));
520 65
                        break;
521
                case COL_MA1000:
522 65
                        IC(mvwprintw(w_bar_t, 0, x, " %12.12s", "AVG_1000"));
523 65
                        break;
524
                default:
525 0
                        break;
526
                }
527 2458
                x += COLW;
528 2458
                col++;
529
        }
530
531 582
        AC(wnoutrefresh(w_bar_t));
532 582
}
533
534
static void
535 4237
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 4237
        AN(w);
548 4237
        AN(pt);
549
550 4237
        col = COL_CUR;
551 22381
        while (col < COL_LAST) {
552 21783
                if (X - x < COLW)
553 3639
                        break;
554 18144
                switch (col) {
555
                case COL_CUR:
556 4237
                        IC(mvwprintw(w, y, x, " %12ju", (uintmax_t)pt->cur));
557 4237
                        break;
558
                case COL_CHG:
559 4237
                        if (pt->t_last)
560 3509
                                IC(mvwprintw(w, y, x, " %12.2f", pt->chg));
561
                        else
562 728
                                IC(mvwprintw(w, y, x, " %12s", ".  "));
563 4237
                        break;
564
                case COL_AVG:
565 4237
                        if (pt->avg)
566 1395
                                IC(mvwprintw(w, y, x, " %12.2f", pt->avg));
567
                        else
568 2842
                                IC(mvwprintw(w, y, x, " %12s", ".  "));
569 4237
                        break;
570
                case COL_MA10:
571 4237
                        IC(mvwprintw(w, y, x, " %12.2f", pt->ma_10.acc));
572 4237
                        break;
573
                case COL_MA100:
574 598
                        IC(mvwprintw(w, y, x, " %12.2f", pt->ma_100.acc));
575 598
                        break;
576
                case COL_MA1000:
577 598
                        IC(mvwprintw(w, y, x, " %12.2f", pt->ma_1000.acc));
578 598
                        break;
579
                default:
580 0
                        break;
581
                }
582 18144
                x += COLW;
583 18144
                col++;
584
        }
585 4237
}
586
587
static double
588 2745
scale_bytes(double val, char *q)
589
{
590
        const char *p;
591
592 3479
        for (p = " KMGTPEZY"; *p; p++) {
593 3479
                if (fabs(val) < 1024.)
594 2745
                        break;
595 734
                val /= 1024.;
596 734
        }
597 2745
        *q = *p;
598 2745
        return (val);
599
}
600
601
static void
602 4708
print_bytes(WINDOW *w, double val)
603
{
604 4708
        char q = ' ';
605
606 4708
        if (scale)
607 2745
                val = scale_bytes(val, &q);
608 4708
        IC(wprintw(w, " %12.2f%c", val, q));
609 4708
}
610
611
static void
612 1706
print_trunc(WINDOW *w, uintmax_t val)
613
{
614 1706
        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 1706
                IC(wprintw(w, " %12ju", val));
620 1706
}
621
622
static void
623 1925
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 1925
        AN(w);
636 1925
        AN(pt);
637
638 1925
        col = COL_CUR;
639 10327
        while (col < COL_LAST) {
640 9976
                if (X - x < COLW)
641 1574
                        break;
642 8402
                wmove(w, y, x);
643 8402
                switch (col) {
644
                case COL_CUR:
645 1925
                        if (scale && pt->cur > 1024)
646 219
                                print_bytes(w, (double)pt->cur);
647
                        else
648 1706
                                print_trunc(w, (uintmax_t)pt->cur);
649 1925
                        break;
650
                case COL_CHG:
651 1925
                        if (pt->t_last)
652 1054
                                print_bytes(w, pt->chg);
653
                        else
654 871
                                IC(wprintw(w, " %12s", ".  "));
655 1925
                        break;
656
                case COL_AVG:
657 1925
                        if (pt->avg)
658 808
                                print_bytes(w, pt->avg);
659
                        else
660 1117
                                IC(wprintw(w, " %12s", ".  "));
661 1925
                        break;
662
                case COL_MA10:
663 1925
                        print_bytes(w, pt->ma_10.acc);
664 1925
                        break;
665
                case COL_MA100:
666 351
                        print_bytes(w, pt->ma_100.acc);
667 351
                        break;
668
                case COL_MA1000:
669 351
                        print_bytes(w, pt->ma_1000.acc);
670 351
                        break;
671
                default:
672 0
                        break;
673
                }
674 8402
                x += COLW;
675 8402
                col++;
676
        }
677 1925
}
678
679
static void
680 237
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 237
        AN(w);
690 237
        AN(pt);
691 237
        assert(pt->vpt->format == 'b');
692
693 237
        col = COL_VAL;
694 711
        while (col < COL_LAST) {
695 474
                switch (col) {
696
                case COL_VAL:
697 237
                        if (X - x < COLW)
698 0
                                return;
699 237
                        IC(mvwprintw(w, y, x, "   %10.10jx",
700
                            (uintmax_t)((pt->cur >> 24) & 0xffffffffffLL)));
701 237
                        x += COLW;
702 237
                        break;
703
                case COL_MAP:
704 237
                        if (X - x < 2 * COLW)
705 0
                                return;
706 237
                        x += (2 * COLW) - 24;
707 5925
                        for (ch = 0x800000; ch; ch >>= 1) {
708 5688
                                if (pt->cur & ch)
709 1362
                                        mvwaddch(w, y, x, 'V');
710
                                else
711 4326
                                        mvwaddch(w, y, x, '_');
712 5688
                                x++;
713 5688
                        }
714 237
                        break;
715
                default:
716 0
                        break;
717
                }
718 474
                col++;
719
        }
720 237
}
721
722
static void
723 39
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 39
        AN(w);
731 39
        AN(pt);
732
733 39
        col = COL_DUR;
734 78
        while (col < COL_LAST) {
735 39
                if (X - x < COLW)
736 0
                        break;
737 39
                switch (col) {
738
                case COL_DUR:
739 39
                        wmove(w, y, x);
740 39
                        if (scale)
741 13
                                print_duration(w, pt->cur);
742
                        else
743 26
                                IC(wprintw(w, " %12ju", (uintmax_t)pt->cur));
744 39
                        break;
745
                default:
746 0
                        break;
747
                }
748 39
                x += COLW;
749 39
                col++;
750
        }
751 39
}
752
753
static void
754 6438
draw_line(WINDOW *w, int y, const struct pt *pt)
755
{
756
        int x, X;
757
758 6438
        assert(colw_name >= COLW_NAME_MIN);
759 6438
        X = getmaxx(w);
760 6438
        x = 0;
761 6438
        if (strlen(pt->vpt->name) > colw_name)
762 449
                IC(mvwprintw(w, y, x, "%.*s...", colw_name - 3, pt->vpt->name));
763
        else
764 5989
                IC(mvwprintw(w, y, x, "%.*s", colw_name, pt->vpt->name));
765 6438
        x += colw_name;
766
767 6438
        switch (pt->vpt->format) {
768
        case 'b':
769 237
                draw_line_bitmap(w, y, x, X, pt);
770 237
                break;
771
        case 'B':
772 1925
                draw_line_bytes(w, y, x, X, pt);
773 1925
                break;
774
        case 'd':
775 39
                draw_line_duration(w, y, x, X, pt);
776 39
                break;
777
        default:
778 4237
                draw_line_default(w, y, x, X, pt);
779 4237
                break;
780
        }
781 6438
}
782
783
static void
784 582
draw_points(void)
785
{
786
        int line;
787
        int n;
788
789 582
        AN(w_points);
790
791 582
        AC(werase(w_points));
792 582
        if (n_ptarray == 0) {
793 91
                AC(wnoutrefresh(w_points));
794 91
                return;
795
        }
796
797 491
        assert(current >= 0);
798 491
        assert(current < n_ptarray);
799 491
        assert(page_start >= 0);
800 491
        assert(page_start < n_ptarray);
801 491
        assert(current >= page_start);
802 491
        assert(current - page_start < l_points);
803
804 6929
        for (line = 0; line < l_points; line++) {
805 6599
                n = line + page_start;
806 6599
                if (n >= n_ptarray)
807 161
                        break;
808 6438
                if (n == current)
809 491
                        wattron(w_points, A_BOLD);
810 6438
                draw_line(w_points, line, ptarray[n]);
811 6438
                if (n == current)
812 491
                        wattroff(w_points, A_BOLD);
813 6438
        }
814 491
        AC(wnoutrefresh(w_points));
815 582
}
816
817
static void
818 39
draw_help(void)
819
{
820
        const char *const *p;
821
        int l, y, X;
822
823 39
        if (l_points >= bindings_help_len) {
824 0
                assert(help_line == 0);
825 0
                l = bindings_help_len;
826 0
        } else {
827 39
                assert(help_line >= 0);
828 39
                assert(help_line <= bindings_help_len - l_points);
829 39
                l = l_points;
830
        }
831
832 39
        X = getmaxx(w_points);
833 39
        AC(werase(w_points));
834
835 663
        for (y = 0, p = bindings_help + help_line; y < l; y++, p++) {
836 624
                if (**p == '\t') {
837 325
                        IC(mvwprintw(w_points, y, 0, "    %.*s", X - 4, *p + 1));
838 325
                } else {
839 299
                        wattron(w_points, A_BOLD);
840 299
                        IC(mvwprintw(w_points, y, 0, "%.*s", X, *p));
841 299
                        wattroff(w_points, A_BOLD);
842
                }
843 624
        }
844
845 39
        AC(wnoutrefresh(w_points));
846 39
}
847
848
static void
849 582
draw_bar_b(void)
850
{
851
        int x, X;
852
        char buf[64];
853
854 582
        AN(w_bar_b);
855
856 582
        x = 0;
857 582
        X = getmaxx(w_bar_b);
858 582
        AC(werase(w_bar_b));
859 582
        if (page_start + l_points < n_ptarray)
860 280
                IC(mvwprintw(w_bar_b, 0, x, "vvv"));
861 582
        x += 4;
862 582
        if (current < n_ptarray)
863 491
                IC(mvwprintw(w_bar_b, 0, x, "%s", ptarray[current]->vpt->name));
864
865 582
        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 582
        IC(mvwprintw(w_bar_b, 0, X - strlen(buf), "%s", buf));
870 582
        X -= strlen(buf) + 2;
871
872 582
        if (verbosity != NULL) {
873 582
                IC(mvwprintw(w_bar_b, 0, X - strlen(verbosity->label), "%s",
874
                    verbosity->label));
875 582
                X -= strlen(verbosity->label) + 2;
876 582
        }
877 582
        if (!hide_unseen) {
878 130
                IC(mvwprintw(w_bar_b, 0, X - 6, "%s", "UNSEEN"));
879 130
                X -= 8;
880 130
        }
881 582
        if (raw_vsc)
882 0
                IC(mvwprintw(w_bar_b, 0, X - 3, "%s", "RAW"));
883
884 582
        AC(wnoutrefresh(w_bar_b));
885 582
}
886
887
static void
888 582
draw_info(void)
889
{
890
891 582
        if (w_info == NULL)
892 0
                return;
893
894 582
        AC(werase(w_info));
895 582
        if (current < n_ptarray) {
896
                /* XXX: Word wrapping, and overflow handling? */
897 491
                IC(mvwprintw(w_info, 0, 0, "%s:",
898
                    ptarray[current]->vpt->sdesc));
899 491
                IC(mvwprintw(w_info, 1, 0, "%s",
900
                    ptarray[current]->vpt->ldesc));
901 491
        }
902 582
        AC(wnoutrefresh(w_info));
903 582
}
904
905
static void
906 621
draw_screen(void)
907
{
908 621
        draw_status();
909 621
        if (show_help) {
910 39
                AC(werase(w_bar_t));
911 39
                AC(werase(w_bar_b));
912 39
                AC(werase(w_info));
913 39
                AC(wnoutrefresh(w_bar_t));
914 39
                AC(wnoutrefresh(w_bar_b));
915 39
                AC(wnoutrefresh(w_info));
916 39
                draw_help();
917 39
        } else {
918 582
                draw_bar_t();
919 582
                draw_points();
920 582
                draw_bar_b();
921 582
                draw_info();
922
        }
923 621
        AC(doupdate());
924 621
        redraw = 0;
925 621
}
926
927
static void
928 26
handle_common_keypress(enum kb_e kb)
929
{
930
931 26
        switch (kb) {
932
        case KB_QUIT:
933 26
                keep_running = 0;
934 26
                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 26
}
945
946
static void
947 208
handle_points_keypress(struct vsc *vsc, enum kb_e kb)
948
{
949
950 208
        switch (kb) {
951
        case KB_HELP:
952 26
                show_help = 1;
953 26
                help_line = 0;
954 26
                redraw = 1;
955 26
                return;
956
        case KB_UP:
957 13
                if (current == 0)
958 0
                        return;
959 13
                current--;
960 13
                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 13
                current -= l_points;
968 13
                page_start -= l_points;
969 13
                break;
970
        case KB_PAGEDOWN:
971 13
                current += l_points;
972 13
                if (page_start + l_points < n_ptarray - 1)
973 13
                        page_start += l_points;
974 13
                break;
975
        case KB_TOP:
976 13
                current = 0;
977 13
                break;
978
        case KB_BOTTOM:
979 26
                current = n_ptarray - 1;
980 26
                break;
981
        case KB_UNSEEN:
982 13
                hide_unseen = 1 - hide_unseen;
983 13
                rebuild = REBUILD_NEXT;
984 13
                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 13
                scale = 1 - scale;
992 13
                rebuild = REBUILD_NEXT;
993 13
                break;
994
        case KB_ACCEL:
995 13
                interval += 0.1;
996 13
                (void)snprintf(notification_message, NOTIF_MAXLEN,
997 13
                    "Refresh interval set to %.1f seconds.", interval);
998
999 13
                notification_eol = VTIM_mono() + 1.25;
1000 13
                break;
1001
        case KB_DECEL:
1002 13
                interval -= 0.1;
1003 13
                if (interval < 0.1)
1004 0
                        interval = 0.1;
1005 13
                (void)snprintf(notification_message, NOTIF_MAXLEN,
1006 13
                    "Refresh interval set to %.1f seconds.", interval);
1007
1008 13
                notification_eol = VTIM_mono() + 1.25;
1009 13
                break;
1010
        case KB_VERBOSE:
1011 13
                verbosity = VSC_ChangeLevel(verbosity, 1);
1012 13
                rebuild = REBUILD_NEXT;
1013 13
                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 13
                reset_averages = 1;
1023 13
                return;
1024
        case KB_QUIT:
1025
        case KB_SIG_INT:
1026
        case KB_SIG_TSTP:
1027 26
                handle_common_keypress(kb);
1028 26
                return;
1029
        default:
1030 0
                WRONG("unhandled key binding");
1031 0
        }
1032
1033 143
        update_position();
1034 143
        redraw = 1;
1035 208
}
1036
1037
static void
1038 39
handle_help_keypress(enum kb_e kb)
1039
{
1040 39
        int hl = help_line;
1041
1042 39
        switch (kb) {
1043
        case KB_HELP:
1044 26
                show_help = 0;
1045 26
                redraw = 1;
1046 26
                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 13
                help_line = bindings_help_len;
1064 13
                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 13
        help_line = vlimit_t(int, help_line, 0, bindings_help_len - l_points);
1085 13
        redraw = (help_line != hl);
1086 39
}
1087
1088
static void
1089 247
handle_keypress(struct vsc *vsc, int ch)
1090
{
1091
        enum kb_e kb;
1092
1093 247
        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 247
        if (show_help)
1106 39
                handle_help_keypress(kb);
1107
        else
1108 208
                handle_points_keypress(vsc, kb);
1109 247
}
1110
1111
static void * v_matchproto_(VSC_new_f)
1112 29445
newpt(void *priv, const struct VSC_point *const vpt)
1113
{
1114
        struct pt *pt;
1115
1116 29445
        AZ(priv);
1117 29445
        ALLOC_OBJ(pt, PT_MAGIC);
1118 29445
        rebuild |= REBUILD_NEXT;
1119 29445
        AN(pt);
1120 29445
        pt->vpt = vpt;
1121 29445
        pt->last = VSC_Value(vpt);
1122 29445
        pt->ma_10.nmax = 10;
1123 29445
        pt->ma_100.nmax = 100;
1124 29445
        pt->ma_1000.nmax = 1000;
1125
1126 29445
        VTAILQ_INSERT_TAIL(&ptlist, pt, list);
1127 29445
        n_ptlist++;
1128
1129 29445
        AZ(strcmp(vpt->ctype, "uint64_t"));
1130
1131 29445
        if (!strcmp(vpt->name, "MGT.uptime"))
1132 91
                mgt_uptime = vpt->ptr;
1133 29445
        if (!strcmp(vpt->name, "MAIN.uptime"))
1134 91
                main_uptime = vpt->ptr;
1135 29445
        if (!strcmp(vpt->name, "MAIN.cache_hit"))
1136 91
                main_cache_hit = vpt->ptr;
1137 29445
        if (!strcmp(vpt->name, "MAIN.cache_miss"))
1138 91
                main_cache_miss = vpt->ptr;
1139 29445
        return (pt);
1140
}
1141
1142
static void v_matchproto_(VSC_destroy_f)
1143 29445
delpt(void *priv, const struct VSC_point *const vpt)
1144
{
1145
        struct pt *pt;
1146
1147 29445
        AZ(priv);
1148 29445
        CAST_OBJ_NOTNULL(pt, vpt->priv, PT_MAGIC);
1149 29445
        rebuild |= REBUILD_NEXT;
1150 29445
        VTAILQ_REMOVE(&ptlist, pt, list);
1151 29445
        n_ptlist--;
1152 29445
        FREE_OBJ(pt);
1153 29445
        if (vpt->ptr == mgt_uptime)
1154 91
                mgt_uptime = NULL;
1155 29445
        if (vpt->ptr == main_uptime)
1156 91
                main_uptime = NULL;
1157 29445
        if (vpt->ptr == main_cache_hit)
1158 91
                main_cache_hit = NULL;
1159 29445
        if (vpt->ptr == main_cache_miss)
1160 91
                main_cache_miss = NULL;
1161 29445
}
1162
1163
void
1164 91
do_curses(struct vsm *vsm, struct vsc *vsc)
1165
{
1166
        long t;
1167
        int ch;
1168
        double now;
1169
1170 91
        verbosity = VSC_ChangeLevel(NULL, 0);
1171
1172 91
        (void)initscr();
1173 91
        AC(raw());
1174 91
        AC(noecho());
1175 91
        AC(nonl());
1176 91
        IC(curs_set(0));
1177
1178 91
        make_windows();
1179 91
        AC(doupdate());
1180
1181 91
        VSC_State(vsc, newpt, delpt, NULL);
1182
1183 91
        raw_vsc = VSC_IsRaw(vsc);
1184 91
        rebuild |= REBUILD_FIRST;
1185 91
        (void)VSC_Iter(vsc, vsm, NULL, NULL);
1186 91
        build_pt_array();
1187 91
        init_hitrate();
1188
1189 770
        while (keep_running && !VSIG_int && !VSIG_term && !VSIG_hup) {
1190 679
                (void)VSC_Iter(vsc, vsm, NULL, NULL);
1191 679
                vsm_status = VSM_Status(vsm);
1192 679
                if (vsm_status & (VSM_MGT_RESTARTED|VSM_WRK_RESTARTED))
1193 0
                        init_hitrate();
1194 679
                if (rebuild)
1195 252
                        build_pt_array();
1196
1197 679
                now = VTIM_mono();
1198 679
                if (now - t_sample > interval)
1199 400
                        sample = 1;
1200 679
                if (sample)
1201 400
                        sample_data();
1202 679
                if (redraw)
1203 621
                        draw_screen();
1204
1205 679
                t = (long)((t_sample + interval - now) * 1000);
1206 679
                wtimeout(w_status, t);
1207
1208 679
                ch = wgetch(w_status);
1209 679
                switch (ch) {
1210
                case ERR:
1211 419
                        break;
1212
#ifdef KEY_RESIZE /* sigh, Solaris lacks this.. */
1213
                case KEY_RESIZE:
1214 13
                        make_windows();
1215 13
                        update_position();
1216 13
                        break;
1217
#endif
1218
                default:
1219 247
                        handle_keypress(vsc, ch);
1220 247
                        break;
1221
                }
1222
        }
1223 91
        VSC_Destroy(&vsc, vsm);
1224 91
        AN(VTAILQ_EMPTY(&ptlist));
1225 91
        VSM_Destroy(&vsm);
1226 91
        AZ(endwin());
1227 91
}