xref: /freebsd/usr.bin/systat/zarc.c (revision 45518845960293799455f49c03d126fb0724f819)
127aa4769SMichael Reifenberger /*-
2*45518845SMichael Reifenberger  * Copyright (c) 2014 - 2017 Yoshihiro Ota
327aa4769SMichael Reifenberger  *
427aa4769SMichael Reifenberger  * Redistribution and use in source and binary forms, with or without
527aa4769SMichael Reifenberger  * modification, are permitted provided that the following conditions
627aa4769SMichael Reifenberger  * are met:
727aa4769SMichael Reifenberger  * 1. Redistributions of source code must retain the above copyright
827aa4769SMichael Reifenberger  *    notice, this list of conditions and the following disclaimer.
927aa4769SMichael Reifenberger  * 2. Redistributions in binary form must reproduce the above copyright
1027aa4769SMichael Reifenberger  *    notice, this list of conditions and the following disclaimer in the
1127aa4769SMichael Reifenberger  *    documentation and/or other materials provided with the distribution.
12fbbd9655SWarner Losh  * 3. Neither the name of the University nor the names of its contributors
1327aa4769SMichael Reifenberger  *    may be used to endorse or promote products derived from this software
1427aa4769SMichael Reifenberger  *    without specific prior written permission.
1527aa4769SMichael Reifenberger  *
1627aa4769SMichael Reifenberger  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
1727aa4769SMichael Reifenberger  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1827aa4769SMichael Reifenberger  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1927aa4769SMichael Reifenberger  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2027aa4769SMichael Reifenberger  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2127aa4769SMichael Reifenberger  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2227aa4769SMichael Reifenberger  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2327aa4769SMichael Reifenberger  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2427aa4769SMichael Reifenberger  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2527aa4769SMichael Reifenberger  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2627aa4769SMichael Reifenberger  * SUCH DAMAGE.
2727aa4769SMichael Reifenberger  */
2827aa4769SMichael Reifenberger 
2927aa4769SMichael Reifenberger #include <sys/cdefs.h>
3027aa4769SMichael Reifenberger __FBSDID("$FreeBSD$");
3127aa4769SMichael Reifenberger 
3227aa4769SMichael Reifenberger #include <sys/types.h>
3327aa4769SMichael Reifenberger #include <sys/sysctl.h>
3427aa4769SMichael Reifenberger 
35*45518845SMichael Reifenberger /* #include <stdlib.h> */
36970bdbf5SMichael Tuexen #include <inttypes.h>
3727aa4769SMichael Reifenberger #include <string.h>
38*45518845SMichael Reifenberger #include <err.h>
3927aa4769SMichael Reifenberger 
4027aa4769SMichael Reifenberger #include "systat.h"
4127aa4769SMichael Reifenberger #include "extern.h"
42*45518845SMichael Reifenberger #include "devs.h"
4327aa4769SMichael Reifenberger 
4427aa4769SMichael Reifenberger struct zfield{
4527aa4769SMichael Reifenberger 	uint64_t arcstats;
4627aa4769SMichael Reifenberger 	uint64_t arcstats_demand_data;
4727aa4769SMichael Reifenberger 	uint64_t arcstats_demand_metadata;
4827aa4769SMichael Reifenberger 	uint64_t arcstats_prefetch_data;
4927aa4769SMichael Reifenberger 	uint64_t arcstats_prefetch_metadata;
5027aa4769SMichael Reifenberger 	uint64_t zfetchstats;
5127aa4769SMichael Reifenberger 	uint64_t arcstats_l2;
5227aa4769SMichael Reifenberger 	uint64_t vdev_cache_stats;
5327aa4769SMichael Reifenberger };
5427aa4769SMichael Reifenberger 
5527aa4769SMichael Reifenberger static struct zarcstats {
5627aa4769SMichael Reifenberger 	struct zfield hits;
5727aa4769SMichael Reifenberger 	struct zfield misses;
5827aa4769SMichael Reifenberger } curstat, initstat, oldstat;
5927aa4769SMichael Reifenberger 
6027aa4769SMichael Reifenberger static void
6127aa4769SMichael Reifenberger getinfo(struct zarcstats *ls);
6227aa4769SMichael Reifenberger 
6327aa4769SMichael Reifenberger WINDOW *
6427aa4769SMichael Reifenberger openzarc(void)
6527aa4769SMichael Reifenberger {
6627aa4769SMichael Reifenberger 	return (subwin(stdscr, LINES-3-1, 0, MAINWIN_ROW, 0));
6727aa4769SMichael Reifenberger }
6827aa4769SMichael Reifenberger 
6927aa4769SMichael Reifenberger void
7027aa4769SMichael Reifenberger closezarc(WINDOW *w)
7127aa4769SMichael Reifenberger {
7227aa4769SMichael Reifenberger 	if (w == NULL)
7327aa4769SMichael Reifenberger 		return;
7427aa4769SMichael Reifenberger 	wclear(w);
7527aa4769SMichael Reifenberger 	wrefresh(w);
7627aa4769SMichael Reifenberger 	delwin(w);
7727aa4769SMichael Reifenberger }
7827aa4769SMichael Reifenberger 
7927aa4769SMichael Reifenberger void
8027aa4769SMichael Reifenberger labelzarc(void)
8127aa4769SMichael Reifenberger {
82*45518845SMichael Reifenberger 	int row = 1;
8327aa4769SMichael Reifenberger 	wmove(wnd, 0, 0); wclrtoeol(wnd);
8427aa4769SMichael Reifenberger 	mvwprintw(wnd, 0, 31+1, "%4.4s %7.7s %7.7s %12.12s %12.12s",
8527aa4769SMichael Reifenberger 		"rate", "hits", "misses", "total hits", "total misses");
86*45518845SMichael Reifenberger #define L(str) mvwprintw(wnd, row, 5, #str); \
8727aa4769SMichael Reifenberger 	mvwprintw(wnd, row, 31, ":"); \
88*45518845SMichael Reifenberger 	mvwprintw(wnd, row, 31+4, "%%"); ++row
89*45518845SMichael Reifenberger 	L(arcstats);
90*45518845SMichael Reifenberger 	L(arcstats.demand_data);
91*45518845SMichael Reifenberger 	L(arcstats.demand_metadata);
92*45518845SMichael Reifenberger 	L(arcstats.prefetch_data);
93*45518845SMichael Reifenberger 	L(arcstats.prefetch_metadata);
94*45518845SMichael Reifenberger 	L(zfetchstats);
95*45518845SMichael Reifenberger 	L(arcstats.l2);
96*45518845SMichael Reifenberger 	L(vdev_cache_stats);
9727aa4769SMichael Reifenberger #undef L
98*45518845SMichael Reifenberger 	dslabel(12, 0, 18);
9927aa4769SMichael Reifenberger }
10027aa4769SMichael Reifenberger 
10127aa4769SMichael Reifenberger static int calc(uint64_t hits, uint64_t misses)
10227aa4769SMichael Reifenberger {
10327aa4769SMichael Reifenberger     if( hits )
10427aa4769SMichael Reifenberger 	return 100 * hits / ( hits + misses );
10527aa4769SMichael Reifenberger     else
10627aa4769SMichael Reifenberger 	return 0;
10727aa4769SMichael Reifenberger }
10827aa4769SMichael Reifenberger 
10927aa4769SMichael Reifenberger static void
11027aa4769SMichael Reifenberger domode(struct zarcstats *delta, struct zarcstats *rate)
11127aa4769SMichael Reifenberger {
11227aa4769SMichael Reifenberger #define DO(stat) \
11327aa4769SMichael Reifenberger 	delta->hits.stat = (curstat.hits.stat - oldstat.hits.stat); \
11427aa4769SMichael Reifenberger 	delta->misses.stat = (curstat.misses.stat - oldstat.misses.stat); \
11527aa4769SMichael Reifenberger 	rate->hits.stat = calc(delta->hits.stat, delta->misses.stat)
11627aa4769SMichael Reifenberger 	DO(arcstats);
11727aa4769SMichael Reifenberger 	DO(arcstats_demand_data);
11827aa4769SMichael Reifenberger 	DO(arcstats_demand_metadata);
11927aa4769SMichael Reifenberger 	DO(arcstats_prefetch_data);
12027aa4769SMichael Reifenberger 	DO(arcstats_prefetch_metadata);
12127aa4769SMichael Reifenberger 	DO(zfetchstats);
12227aa4769SMichael Reifenberger 	DO(arcstats_l2);
12327aa4769SMichael Reifenberger 	DO(vdev_cache_stats);
12427aa4769SMichael Reifenberger 	DO(arcstats);
12527aa4769SMichael Reifenberger 	DO(arcstats_demand_data);
12627aa4769SMichael Reifenberger 	DO(arcstats_demand_metadata);
12727aa4769SMichael Reifenberger 	DO(arcstats_prefetch_data);
12827aa4769SMichael Reifenberger 	DO(arcstats_prefetch_metadata);
12927aa4769SMichael Reifenberger 	DO(zfetchstats);
13027aa4769SMichael Reifenberger 	DO(arcstats_l2);
13127aa4769SMichael Reifenberger 	DO(vdev_cache_stats);
13227aa4769SMichael Reifenberger #undef DO
13327aa4769SMichael Reifenberger }
13427aa4769SMichael Reifenberger 
13527aa4769SMichael Reifenberger void
13627aa4769SMichael Reifenberger showzarc(void)
13727aa4769SMichael Reifenberger {
138*45518845SMichael Reifenberger 	int row = 1;
13927aa4769SMichael Reifenberger 	struct zarcstats delta, rate;
14027aa4769SMichael Reifenberger 
14127aa4769SMichael Reifenberger 	memset(&delta, 0, sizeof delta);
14227aa4769SMichael Reifenberger 	memset(&rate, 0, sizeof rate);
14327aa4769SMichael Reifenberger 
14427aa4769SMichael Reifenberger 	domode(&delta, &rate);
14527aa4769SMichael Reifenberger 
146*45518845SMichael Reifenberger #define DO(stat, col, fmt) \
14727aa4769SMichael Reifenberger 	mvwprintw(wnd, row, col, fmt, stat)
148*45518845SMichael Reifenberger #define	R(stat) DO(rate.hits.stat, 31+1, "%3"PRIu64)
149*45518845SMichael Reifenberger #define	H(stat) DO(delta.hits.stat, 31+1+5, "%7"PRIu64); \
150*45518845SMichael Reifenberger 	DO(curstat.hits.stat, 31+1+5+8+8, "%12"PRIu64)
151*45518845SMichael Reifenberger #define	M(stat) DO(delta.misses.stat, 31+1+5+8, "%7"PRIu64); \
152*45518845SMichael Reifenberger 	DO(curstat.misses.stat, 31+1+5+8+8+13, "%12"PRIu64)
153*45518845SMichael Reifenberger #define	E(stat) R(stat); H(stat); M(stat); ++row
154*45518845SMichael Reifenberger 	E(arcstats);
155*45518845SMichael Reifenberger 	E(arcstats_demand_data);
156*45518845SMichael Reifenberger 	E(arcstats_demand_metadata);
157*45518845SMichael Reifenberger 	E(arcstats_prefetch_data);
158*45518845SMichael Reifenberger 	E(arcstats_prefetch_metadata);
159*45518845SMichael Reifenberger 	E(zfetchstats);
160*45518845SMichael Reifenberger 	E(arcstats_l2);
161*45518845SMichael Reifenberger 	E(vdev_cache_stats);
16227aa4769SMichael Reifenberger #undef DO
16327aa4769SMichael Reifenberger #undef E
16427aa4769SMichael Reifenberger #undef M
16527aa4769SMichael Reifenberger #undef H
16627aa4769SMichael Reifenberger #undef R
167*45518845SMichael Reifenberger 	dsshow(12, 0, 18, &cur_dev, &last_dev);
16827aa4769SMichael Reifenberger }
16927aa4769SMichael Reifenberger 
17027aa4769SMichael Reifenberger int
17127aa4769SMichael Reifenberger initzarc(void)
17227aa4769SMichael Reifenberger {
173*45518845SMichael Reifenberger 	dsinit(12);
17427aa4769SMichael Reifenberger 	getinfo(&initstat);
17527aa4769SMichael Reifenberger 	curstat = oldstat = initstat;
176*45518845SMichael Reifenberger 
17727aa4769SMichael Reifenberger 	return 1;
17827aa4769SMichael Reifenberger }
17927aa4769SMichael Reifenberger 
18027aa4769SMichael Reifenberger void
18127aa4769SMichael Reifenberger resetzarc(void)
18227aa4769SMichael Reifenberger {
18327aa4769SMichael Reifenberger 	initzarc();
18427aa4769SMichael Reifenberger }
18527aa4769SMichael Reifenberger 
18627aa4769SMichael Reifenberger static void
18727aa4769SMichael Reifenberger getinfo(struct zarcstats *ls)
18827aa4769SMichael Reifenberger {
189*45518845SMichael Reifenberger 	struct devinfo *tmp_dinfo;
190*45518845SMichael Reifenberger 
191*45518845SMichael Reifenberger 	tmp_dinfo = last_dev.dinfo;
192*45518845SMichael Reifenberger 	last_dev.dinfo = cur_dev.dinfo;
193*45518845SMichael Reifenberger 	cur_dev.dinfo = tmp_dinfo;
194*45518845SMichael Reifenberger 
195*45518845SMichael Reifenberger 	last_dev.snap_time = cur_dev.snap_time;
196*45518845SMichael Reifenberger 	dsgetinfo( &cur_dev );
197*45518845SMichael Reifenberger 
19827aa4769SMichael Reifenberger 	size_t size = sizeof( ls->hits.arcstats );
19927aa4769SMichael Reifenberger 	if ( sysctlbyname("kstat.zfs.misc.arcstats.hits",
20027aa4769SMichael Reifenberger 		&ls->hits.arcstats, &size, NULL, 0 ) != 0 )
20127aa4769SMichael Reifenberger 		return;
20227aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.misses",
20327aa4769SMichael Reifenberger 		ls->misses.arcstats);
20427aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_data_hits",
20527aa4769SMichael Reifenberger 		ls->hits.arcstats_demand_data);
20627aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_data_misses",
20727aa4769SMichael Reifenberger 		ls->misses.arcstats_demand_data);
20827aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_metadata_hits",
20927aa4769SMichael Reifenberger 		ls->hits.arcstats_demand_metadata);
21027aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_metadata_misses",
21127aa4769SMichael Reifenberger 		ls->misses.arcstats_demand_metadata);
21227aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_data_hits",
21327aa4769SMichael Reifenberger 		ls->hits.arcstats_prefetch_data);
21427aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_data_misses",
21527aa4769SMichael Reifenberger 		ls->misses.arcstats_prefetch_data);
21627aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_metadata_hits",
21727aa4769SMichael Reifenberger 		ls->hits.arcstats_prefetch_metadata);
21827aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_metadata_misses",
21927aa4769SMichael Reifenberger 		ls->misses.arcstats_prefetch_metadata);
22027aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.zfetchstats.hits",
22127aa4769SMichael Reifenberger 		ls->hits.zfetchstats);
22227aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.zfetchstats.misses",
22327aa4769SMichael Reifenberger 		ls->misses.zfetchstats);
22427aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.l2_hits",
22527aa4769SMichael Reifenberger 		ls->hits.arcstats_l2);
22627aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.l2_misses",
22727aa4769SMichael Reifenberger 		ls->misses.arcstats_l2);
22827aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.vdev_cache_stats.hits",
22927aa4769SMichael Reifenberger 		ls->hits.vdev_cache_stats);
23027aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.vdev_cache_stats.misses",
23127aa4769SMichael Reifenberger 		ls->misses.vdev_cache_stats);
23227aa4769SMichael Reifenberger }
23327aa4769SMichael Reifenberger 
23427aa4769SMichael Reifenberger void
23527aa4769SMichael Reifenberger fetchzarc(void)
23627aa4769SMichael Reifenberger {
23727aa4769SMichael Reifenberger 	oldstat = curstat;
23827aa4769SMichael Reifenberger 	getinfo(&curstat);
23927aa4769SMichael Reifenberger }
240