xref: /freebsd/usr.bin/systat/zarc.c (revision 4bde63536c9817a41cc555805532d02ecf7afc92)
127aa4769SMichael Reifenberger /*-
2*4bde6353SMichael Reifenberger  * Copyright (c) 2014 - 2017, 2019 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 
35970bdbf5SMichael Tuexen #include <inttypes.h>
3627aa4769SMichael Reifenberger #include <string.h>
3745518845SMichael Reifenberger #include <err.h>
38*4bde6353SMichael Reifenberger #include <libutil.h>
3927aa4769SMichael Reifenberger 
4027aa4769SMichael Reifenberger #include "systat.h"
4127aa4769SMichael Reifenberger #include "extern.h"
4245518845SMichael 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 
60*4bde6353SMichael Reifenberger struct zarcrates {
61*4bde6353SMichael Reifenberger 	struct zfield current;
62*4bde6353SMichael Reifenberger 	struct zfield total;
63*4bde6353SMichael Reifenberger };
64*4bde6353SMichael Reifenberger 
6527aa4769SMichael Reifenberger static void
6627aa4769SMichael Reifenberger getinfo(struct zarcstats *ls);
6727aa4769SMichael Reifenberger 
6827aa4769SMichael Reifenberger WINDOW *
6927aa4769SMichael Reifenberger openzarc(void)
7027aa4769SMichael Reifenberger {
71*4bde6353SMichael Reifenberger 
7227aa4769SMichael Reifenberger 	return (subwin(stdscr, LINES - 3 - 1, 0, MAINWIN_ROW, 0));
7327aa4769SMichael Reifenberger }
7427aa4769SMichael Reifenberger 
7527aa4769SMichael Reifenberger void
7627aa4769SMichael Reifenberger closezarc(WINDOW *w)
7727aa4769SMichael Reifenberger {
78*4bde6353SMichael Reifenberger 
7927aa4769SMichael Reifenberger 	if (w == NULL)
8027aa4769SMichael Reifenberger 		return;
8127aa4769SMichael Reifenberger 	wclear(w);
8227aa4769SMichael Reifenberger 	wrefresh(w);
8327aa4769SMichael Reifenberger 	delwin(w);
8427aa4769SMichael Reifenberger }
8527aa4769SMichael Reifenberger 
8627aa4769SMichael Reifenberger void
8727aa4769SMichael Reifenberger labelzarc(void)
8827aa4769SMichael Reifenberger {
8945518845SMichael Reifenberger 	int row = 1;
90*4bde6353SMichael Reifenberger 
9127aa4769SMichael Reifenberger 	wmove(wnd, 0, 0); wclrtoeol(wnd);
92*4bde6353SMichael Reifenberger 	mvwprintw(wnd, 0, 31+1, "%4.4s %6.6s %6.6s | Total %4.4s %6.6s %6.6s",
93*4bde6353SMichael Reifenberger 		"Rate", "Hits", "Misses", "Rate", "Hits", "Misses");
94*4bde6353SMichael Reifenberger #define L(str) mvwprintw(wnd, row++, 5, \
95*4bde6353SMichael Reifenberger 		"%-26.26s:   %%               |          %%", #str)
9645518845SMichael Reifenberger 	L(arcstats);
9745518845SMichael Reifenberger 	L(arcstats.demand_data);
9845518845SMichael Reifenberger 	L(arcstats.demand_metadata);
9945518845SMichael Reifenberger 	L(arcstats.prefetch_data);
10045518845SMichael Reifenberger 	L(arcstats.prefetch_metadata);
10145518845SMichael Reifenberger 	L(zfetchstats);
10245518845SMichael Reifenberger 	L(arcstats.l2);
10345518845SMichael Reifenberger 	L(vdev_cache_stats);
10427aa4769SMichael Reifenberger #undef L
10545518845SMichael Reifenberger 	dslabel(12, 0, 18);
10627aa4769SMichael Reifenberger }
10727aa4769SMichael Reifenberger 
108*4bde6353SMichael Reifenberger static int
109*4bde6353SMichael Reifenberger calc_rate(uint64_t hits, uint64_t misses)
11027aa4769SMichael Reifenberger {
11127aa4769SMichael Reifenberger     if(hits)
11227aa4769SMichael Reifenberger 	return 100 * hits / (hits + misses);
11327aa4769SMichael Reifenberger     else
11427aa4769SMichael Reifenberger 	return 0;
11527aa4769SMichael Reifenberger }
11627aa4769SMichael Reifenberger 
11727aa4769SMichael Reifenberger static void
118*4bde6353SMichael Reifenberger domode(struct zarcstats *delta, struct zarcrates *rate)
11927aa4769SMichael Reifenberger {
12027aa4769SMichael Reifenberger #define DO(stat) \
12127aa4769SMichael Reifenberger 	delta->hits.stat = (curstat.hits.stat - oldstat.hits.stat); \
12227aa4769SMichael Reifenberger 	delta->misses.stat = (curstat.misses.stat - oldstat.misses.stat); \
123*4bde6353SMichael Reifenberger 	rate->current.stat = calc_rate(delta->hits.stat, delta->misses.stat); \
124*4bde6353SMichael Reifenberger 	rate->total.stat = calc_rate(curstat.hits.stat, curstat.misses.stat)
12527aa4769SMichael Reifenberger 	DO(arcstats);
12627aa4769SMichael Reifenberger 	DO(arcstats_demand_data);
12727aa4769SMichael Reifenberger 	DO(arcstats_demand_metadata);
12827aa4769SMichael Reifenberger 	DO(arcstats_prefetch_data);
12927aa4769SMichael Reifenberger 	DO(arcstats_prefetch_metadata);
13027aa4769SMichael Reifenberger 	DO(zfetchstats);
13127aa4769SMichael Reifenberger 	DO(arcstats_l2);
13227aa4769SMichael Reifenberger 	DO(vdev_cache_stats);
13327aa4769SMichael Reifenberger 	DO(arcstats);
13427aa4769SMichael Reifenberger 	DO(arcstats_demand_data);
13527aa4769SMichael Reifenberger 	DO(arcstats_demand_metadata);
13627aa4769SMichael Reifenberger 	DO(arcstats_prefetch_data);
13727aa4769SMichael Reifenberger 	DO(arcstats_prefetch_metadata);
13827aa4769SMichael Reifenberger 	DO(zfetchstats);
13927aa4769SMichael Reifenberger 	DO(arcstats_l2);
14027aa4769SMichael Reifenberger 	DO(vdev_cache_stats);
14127aa4769SMichael Reifenberger #undef DO
14227aa4769SMichael Reifenberger }
14327aa4769SMichael Reifenberger 
14427aa4769SMichael Reifenberger void
14527aa4769SMichael Reifenberger showzarc(void)
14627aa4769SMichael Reifenberger {
14745518845SMichael Reifenberger 	int row = 1;
148*4bde6353SMichael Reifenberger 	struct zarcstats delta = {};
149*4bde6353SMichael Reifenberger 	struct zarcrates rate = {};
15027aa4769SMichael Reifenberger 
15127aa4769SMichael Reifenberger 	domode(&delta, &rate);
15227aa4769SMichael Reifenberger 
153*4bde6353SMichael Reifenberger #define DO(stat, col, width) \
154*4bde6353SMichael Reifenberger 	sysputuint64(wnd, row, col, width, stat, HN_DIVISOR_1000)
155*4bde6353SMichael Reifenberger #define	RATES(stat) mvwprintw(wnd, row, 31+1, "%3"PRIu64, rate.current.stat);\
156*4bde6353SMichael Reifenberger 	mvwprintw(wnd, row, 31+1+5+7+7+8, "%3"PRIu64, rate.total.stat)
157*4bde6353SMichael Reifenberger #define	HITS(stat) DO(delta.hits.stat, 31+1+5, 6); \
158*4bde6353SMichael Reifenberger 	DO(curstat.hits.stat, 31+1+5+7+7+8+5, 6)
159*4bde6353SMichael Reifenberger #define	MISSES(stat) DO(delta.misses.stat, 31+1+5+7, 6); \
160*4bde6353SMichael Reifenberger 	DO(curstat.misses.stat, 31+1+5+7+7+8+5+7, 6)
161*4bde6353SMichael Reifenberger #define	E(stat) RATES(stat); HITS(stat); MISSES(stat); ++row
16245518845SMichael Reifenberger 	E(arcstats);
16345518845SMichael Reifenberger 	E(arcstats_demand_data);
16445518845SMichael Reifenberger 	E(arcstats_demand_metadata);
16545518845SMichael Reifenberger 	E(arcstats_prefetch_data);
16645518845SMichael Reifenberger 	E(arcstats_prefetch_metadata);
16745518845SMichael Reifenberger 	E(zfetchstats);
16845518845SMichael Reifenberger 	E(arcstats_l2);
16945518845SMichael Reifenberger 	E(vdev_cache_stats);
17027aa4769SMichael Reifenberger #undef DO
17127aa4769SMichael Reifenberger #undef E
172*4bde6353SMichael Reifenberger #undef MISSES
173*4bde6353SMichael Reifenberger #undef HITS
174*4bde6353SMichael Reifenberger #undef RATES
17545518845SMichael Reifenberger 	dsshow(12, 0, 18, &cur_dev, &last_dev);
17627aa4769SMichael Reifenberger }
17727aa4769SMichael Reifenberger 
17827aa4769SMichael Reifenberger int
17927aa4769SMichael Reifenberger initzarc(void)
18027aa4769SMichael Reifenberger {
18145518845SMichael Reifenberger 	dsinit(12);
18227aa4769SMichael Reifenberger 	getinfo(&initstat);
18327aa4769SMichael Reifenberger 	curstat = oldstat = initstat;
18445518845SMichael Reifenberger 
18527aa4769SMichael Reifenberger 	return 1;
18627aa4769SMichael Reifenberger }
18727aa4769SMichael Reifenberger 
18827aa4769SMichael Reifenberger void
18927aa4769SMichael Reifenberger resetzarc(void)
19027aa4769SMichael Reifenberger {
191*4bde6353SMichael Reifenberger 
19227aa4769SMichael Reifenberger 	initzarc();
19327aa4769SMichael Reifenberger }
19427aa4769SMichael Reifenberger 
19527aa4769SMichael Reifenberger static void
19627aa4769SMichael Reifenberger getinfo(struct zarcstats *ls)
19727aa4769SMichael Reifenberger {
19845518845SMichael Reifenberger 	struct devinfo *tmp_dinfo;
19945518845SMichael Reifenberger 
20045518845SMichael Reifenberger 	tmp_dinfo = last_dev.dinfo;
20145518845SMichael Reifenberger 	last_dev.dinfo = cur_dev.dinfo;
20245518845SMichael Reifenberger 	cur_dev.dinfo = tmp_dinfo;
20345518845SMichael Reifenberger 
20445518845SMichael Reifenberger 	last_dev.snap_time = cur_dev.snap_time;
20545518845SMichael Reifenberger 	dsgetinfo(&cur_dev);
20645518845SMichael Reifenberger 
20727aa4769SMichael Reifenberger 	size_t size = sizeof(ls->hits.arcstats);
20827aa4769SMichael Reifenberger 	if (sysctlbyname("kstat.zfs.misc.arcstats.hits",
20927aa4769SMichael Reifenberger 		&ls->hits.arcstats, &size, NULL, 0) != 0)
21027aa4769SMichael Reifenberger 		return;
21127aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.misses",
21227aa4769SMichael Reifenberger 		ls->misses.arcstats);
21327aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_data_hits",
21427aa4769SMichael Reifenberger 		ls->hits.arcstats_demand_data);
21527aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_data_misses",
21627aa4769SMichael Reifenberger 		ls->misses.arcstats_demand_data);
21727aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_metadata_hits",
21827aa4769SMichael Reifenberger 		ls->hits.arcstats_demand_metadata);
21927aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.demand_metadata_misses",
22027aa4769SMichael Reifenberger 		ls->misses.arcstats_demand_metadata);
22127aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_data_hits",
22227aa4769SMichael Reifenberger 		ls->hits.arcstats_prefetch_data);
22327aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_data_misses",
22427aa4769SMichael Reifenberger 		ls->misses.arcstats_prefetch_data);
22527aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_metadata_hits",
22627aa4769SMichael Reifenberger 		ls->hits.arcstats_prefetch_metadata);
22727aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.prefetch_metadata_misses",
22827aa4769SMichael Reifenberger 		ls->misses.arcstats_prefetch_metadata);
22927aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.zfetchstats.hits",
23027aa4769SMichael Reifenberger 		ls->hits.zfetchstats);
23127aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.zfetchstats.misses",
23227aa4769SMichael Reifenberger 		ls->misses.zfetchstats);
23327aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.l2_hits",
23427aa4769SMichael Reifenberger 		ls->hits.arcstats_l2);
23527aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.arcstats.l2_misses",
23627aa4769SMichael Reifenberger 		ls->misses.arcstats_l2);
23727aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.vdev_cache_stats.hits",
23827aa4769SMichael Reifenberger 		ls->hits.vdev_cache_stats);
23927aa4769SMichael Reifenberger 	GETSYSCTL("kstat.zfs.misc.vdev_cache_stats.misses",
24027aa4769SMichael Reifenberger 		ls->misses.vdev_cache_stats);
24127aa4769SMichael Reifenberger }
24227aa4769SMichael Reifenberger 
24327aa4769SMichael Reifenberger void
24427aa4769SMichael Reifenberger fetchzarc(void)
24527aa4769SMichael Reifenberger {
246*4bde6353SMichael Reifenberger 
24727aa4769SMichael Reifenberger 	oldstat = curstat;
24827aa4769SMichael Reifenberger 	getinfo(&curstat);
24927aa4769SMichael Reifenberger }
250