xref: /freebsd/sys/geom/cache/g_cache.c (revision 02c62349c93f29b881498630c7a0623540255764)
104c7da70SRuslan Ermilov /*-
204c7da70SRuslan Ermilov  * Copyright (c) 2006 Ruslan Ermilov <ru@FreeBSD.org>
304c7da70SRuslan Ermilov  * All rights reserved.
404c7da70SRuslan Ermilov  *
504c7da70SRuslan Ermilov  * Redistribution and use in source and binary forms, with or without
604c7da70SRuslan Ermilov  * modification, are permitted provided that the following conditions
704c7da70SRuslan Ermilov  * are met:
804c7da70SRuslan Ermilov  * 1. Redistributions of source code must retain the above copyright
904c7da70SRuslan Ermilov  *    notice, this list of conditions and the following disclaimer.
1004c7da70SRuslan Ermilov  * 2. Redistributions in binary form must reproduce the above copyright
1104c7da70SRuslan Ermilov  *    notice, this list of conditions and the following disclaimer in the
1204c7da70SRuslan Ermilov  *    documentation and/or other materials provided with the distribution.
1304c7da70SRuslan Ermilov  *
1404c7da70SRuslan Ermilov  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
1504c7da70SRuslan Ermilov  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1604c7da70SRuslan Ermilov  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1704c7da70SRuslan Ermilov  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
1804c7da70SRuslan Ermilov  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
1904c7da70SRuslan Ermilov  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2004c7da70SRuslan Ermilov  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2104c7da70SRuslan Ermilov  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2204c7da70SRuslan Ermilov  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2304c7da70SRuslan Ermilov  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2404c7da70SRuslan Ermilov  * SUCH DAMAGE.
2504c7da70SRuslan Ermilov  */
2604c7da70SRuslan Ermilov 
2704c7da70SRuslan Ermilov #include <sys/cdefs.h>
2804c7da70SRuslan Ermilov __FBSDID("$FreeBSD$");
2904c7da70SRuslan Ermilov 
3004c7da70SRuslan Ermilov #include <sys/param.h>
3104c7da70SRuslan Ermilov #include <sys/systm.h>
3204c7da70SRuslan Ermilov #include <sys/kernel.h>
3304c7da70SRuslan Ermilov #include <sys/module.h>
3404c7da70SRuslan Ermilov #include <sys/lock.h>
3504c7da70SRuslan Ermilov #include <sys/mutex.h>
3604c7da70SRuslan Ermilov #include <sys/bio.h>
3704c7da70SRuslan Ermilov #include <sys/sysctl.h>
3804c7da70SRuslan Ermilov #include <sys/malloc.h>
3904c7da70SRuslan Ermilov #include <sys/queue.h>
405d807a0eSAndrey V. Elsukov #include <sys/sbuf.h>
4104c7da70SRuslan Ermilov #include <sys/time.h>
4204c7da70SRuslan Ermilov #include <vm/uma.h>
4304c7da70SRuslan Ermilov #include <geom/geom.h>
4404c7da70SRuslan Ermilov #include <geom/cache/g_cache.h>
4504c7da70SRuslan Ermilov 
46cb08c2ccSAlexander Leidinger FEATURE(geom_cache, "GEOM cache module");
47cb08c2ccSAlexander Leidinger 
4804c7da70SRuslan Ermilov static MALLOC_DEFINE(M_GCACHE, "gcache_data", "GEOM_CACHE Data");
4904c7da70SRuslan Ermilov 
5004c7da70SRuslan Ermilov SYSCTL_DECL(_kern_geom);
516472ac3dSEd Schouten static SYSCTL_NODE(_kern_geom, OID_AUTO, cache, CTLFLAG_RW, 0,
526472ac3dSEd Schouten     "GEOM_CACHE stuff");
5304c7da70SRuslan Ermilov static u_int g_cache_debug = 0;
5404c7da70SRuslan Ermilov SYSCTL_UINT(_kern_geom_cache, OID_AUTO, debug, CTLFLAG_RW, &g_cache_debug, 0,
5504c7da70SRuslan Ermilov     "Debug level");
5604c7da70SRuslan Ermilov static u_int g_cache_enable = 1;
5704c7da70SRuslan Ermilov SYSCTL_UINT(_kern_geom_cache, OID_AUTO, enable, CTLFLAG_RW, &g_cache_enable, 0,
5804c7da70SRuslan Ermilov     "");
5904c7da70SRuslan Ermilov static u_int g_cache_timeout = 10;
6004c7da70SRuslan Ermilov SYSCTL_UINT(_kern_geom_cache, OID_AUTO, timeout, CTLFLAG_RW, &g_cache_timeout,
6104c7da70SRuslan Ermilov     0, "");
6204c7da70SRuslan Ermilov static u_int g_cache_idletime = 5;
6304c7da70SRuslan Ermilov SYSCTL_UINT(_kern_geom_cache, OID_AUTO, idletime, CTLFLAG_RW, &g_cache_idletime,
6404c7da70SRuslan Ermilov     0, "");
6504c7da70SRuslan Ermilov static u_int g_cache_used_lo = 5;
6604c7da70SRuslan Ermilov static u_int g_cache_used_hi = 20;
6704c7da70SRuslan Ermilov static int
6804c7da70SRuslan Ermilov sysctl_handle_pct(SYSCTL_HANDLER_ARGS)
6904c7da70SRuslan Ermilov {
7004c7da70SRuslan Ermilov 	u_int val = *(u_int *)arg1;
7104c7da70SRuslan Ermilov 	int error;
7204c7da70SRuslan Ermilov 
73041b706bSDavid Malone 	error = sysctl_handle_int(oidp, &val, 0, req);
7404c7da70SRuslan Ermilov 	if (error || !req->newptr)
7504c7da70SRuslan Ermilov 		return (error);
7604c7da70SRuslan Ermilov 	if (val < 0 || val > 100)
7704c7da70SRuslan Ermilov 		return (EINVAL);
7804c7da70SRuslan Ermilov 	if ((arg1 == &g_cache_used_lo && val > g_cache_used_hi) ||
7904c7da70SRuslan Ermilov 	    (arg1 == &g_cache_used_hi && g_cache_used_lo > val))
8004c7da70SRuslan Ermilov 		return (EINVAL);
8104c7da70SRuslan Ermilov 	*(u_int *)arg1 = val;
8204c7da70SRuslan Ermilov 	return (0);
8304c7da70SRuslan Ermilov }
8404c7da70SRuslan Ermilov SYSCTL_PROC(_kern_geom_cache, OID_AUTO, used_lo, CTLTYPE_UINT|CTLFLAG_RW,
8504c7da70SRuslan Ermilov 	&g_cache_used_lo, 0, sysctl_handle_pct, "IU", "");
8604c7da70SRuslan Ermilov SYSCTL_PROC(_kern_geom_cache, OID_AUTO, used_hi, CTLTYPE_UINT|CTLFLAG_RW,
8704c7da70SRuslan Ermilov 	&g_cache_used_hi, 0, sysctl_handle_pct, "IU", "");
8804c7da70SRuslan Ermilov 
8904c7da70SRuslan Ermilov 
9004c7da70SRuslan Ermilov static int g_cache_destroy(struct g_cache_softc *sc, boolean_t force);
9104c7da70SRuslan Ermilov static g_ctl_destroy_geom_t g_cache_destroy_geom;
9204c7da70SRuslan Ermilov 
9304c7da70SRuslan Ermilov static g_taste_t g_cache_taste;
9404c7da70SRuslan Ermilov static g_ctl_req_t g_cache_config;
9504c7da70SRuslan Ermilov static g_dumpconf_t g_cache_dumpconf;
9604c7da70SRuslan Ermilov 
9704c7da70SRuslan Ermilov struct g_class g_cache_class = {
9804c7da70SRuslan Ermilov 	.name = G_CACHE_CLASS_NAME,
9904c7da70SRuslan Ermilov 	.version = G_VERSION,
10004c7da70SRuslan Ermilov 	.ctlreq = g_cache_config,
10104c7da70SRuslan Ermilov 	.taste = g_cache_taste,
10204c7da70SRuslan Ermilov 	.destroy_geom = g_cache_destroy_geom
10304c7da70SRuslan Ermilov };
10404c7da70SRuslan Ermilov 
10504c7da70SRuslan Ermilov #define	OFF2BNO(off, sc)	((off) >> (sc)->sc_bshift)
10604c7da70SRuslan Ermilov #define	BNO2OFF(bno, sc)	((bno) << (sc)->sc_bshift)
10704c7da70SRuslan Ermilov 
10804c7da70SRuslan Ermilov 
10904c7da70SRuslan Ermilov static struct g_cache_desc *
11004c7da70SRuslan Ermilov g_cache_alloc(struct g_cache_softc *sc)
11104c7da70SRuslan Ermilov {
11204c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
11304c7da70SRuslan Ermilov 
11404c7da70SRuslan Ermilov 	mtx_assert(&sc->sc_mtx, MA_OWNED);
11504c7da70SRuslan Ermilov 
11604c7da70SRuslan Ermilov 	if (!TAILQ_EMPTY(&sc->sc_usedlist)) {
11704c7da70SRuslan Ermilov 		dp = TAILQ_FIRST(&sc->sc_usedlist);
11804c7da70SRuslan Ermilov 		TAILQ_REMOVE(&sc->sc_usedlist, dp, d_used);
11904c7da70SRuslan Ermilov 		sc->sc_nused--;
12004c7da70SRuslan Ermilov 		dp->d_flags = 0;
12104c7da70SRuslan Ermilov 		LIST_REMOVE(dp, d_next);
12204c7da70SRuslan Ermilov 		return (dp);
12304c7da70SRuslan Ermilov 	}
12404c7da70SRuslan Ermilov 	if (sc->sc_nent > sc->sc_maxent) {
12504c7da70SRuslan Ermilov 		sc->sc_cachefull++;
12604c7da70SRuslan Ermilov 		return (NULL);
12704c7da70SRuslan Ermilov 	}
12804c7da70SRuslan Ermilov 	dp = malloc(sizeof(*dp), M_GCACHE, M_NOWAIT | M_ZERO);
12904c7da70SRuslan Ermilov 	if (dp == NULL)
13004c7da70SRuslan Ermilov 		return (NULL);
13104c7da70SRuslan Ermilov 	dp->d_data = uma_zalloc(sc->sc_zone, M_NOWAIT);
13204c7da70SRuslan Ermilov 	if (dp->d_data == NULL) {
13304c7da70SRuslan Ermilov 		free(dp, M_GCACHE);
13404c7da70SRuslan Ermilov 		return (NULL);
13504c7da70SRuslan Ermilov 	}
13604c7da70SRuslan Ermilov 	sc->sc_nent++;
13704c7da70SRuslan Ermilov 	return (dp);
13804c7da70SRuslan Ermilov }
13904c7da70SRuslan Ermilov 
14004c7da70SRuslan Ermilov static void
14104c7da70SRuslan Ermilov g_cache_free(struct g_cache_softc *sc, struct g_cache_desc *dp)
14204c7da70SRuslan Ermilov {
14304c7da70SRuslan Ermilov 
14404c7da70SRuslan Ermilov 	mtx_assert(&sc->sc_mtx, MA_OWNED);
14504c7da70SRuslan Ermilov 
14604c7da70SRuslan Ermilov 	uma_zfree(sc->sc_zone, dp->d_data);
14704c7da70SRuslan Ermilov 	free(dp, M_GCACHE);
14804c7da70SRuslan Ermilov 	sc->sc_nent--;
14904c7da70SRuslan Ermilov }
15004c7da70SRuslan Ermilov 
15104c7da70SRuslan Ermilov static void
15204c7da70SRuslan Ermilov g_cache_free_used(struct g_cache_softc *sc)
15304c7da70SRuslan Ermilov {
15404c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
15504c7da70SRuslan Ermilov 	u_int n;
15604c7da70SRuslan Ermilov 
15704c7da70SRuslan Ermilov 	mtx_assert(&sc->sc_mtx, MA_OWNED);
15804c7da70SRuslan Ermilov 
15904c7da70SRuslan Ermilov 	n = g_cache_used_lo * sc->sc_maxent / 100;
16004c7da70SRuslan Ermilov 	while (sc->sc_nused > n) {
16104c7da70SRuslan Ermilov 		KASSERT(!TAILQ_EMPTY(&sc->sc_usedlist), ("used list empty"));
16204c7da70SRuslan Ermilov 		dp = TAILQ_FIRST(&sc->sc_usedlist);
16304c7da70SRuslan Ermilov 		TAILQ_REMOVE(&sc->sc_usedlist, dp, d_used);
16404c7da70SRuslan Ermilov 		sc->sc_nused--;
16504c7da70SRuslan Ermilov 		LIST_REMOVE(dp, d_next);
16604c7da70SRuslan Ermilov 		g_cache_free(sc, dp);
16704c7da70SRuslan Ermilov 	}
16804c7da70SRuslan Ermilov }
16904c7da70SRuslan Ermilov 
17004c7da70SRuslan Ermilov static void
17104c7da70SRuslan Ermilov g_cache_deliver(struct g_cache_softc *sc, struct bio *bp,
17204c7da70SRuslan Ermilov     struct g_cache_desc *dp, int error)
17304c7da70SRuslan Ermilov {
17404c7da70SRuslan Ermilov 	off_t off1, off, len;
17504c7da70SRuslan Ermilov 
17604c7da70SRuslan Ermilov 	mtx_assert(&sc->sc_mtx, MA_OWNED);
17704c7da70SRuslan Ermilov 	KASSERT(OFF2BNO(bp->bio_offset, sc) <= dp->d_bno, ("wrong entry"));
17804c7da70SRuslan Ermilov 	KASSERT(OFF2BNO(bp->bio_offset + bp->bio_length - 1, sc) >=
17904c7da70SRuslan Ermilov 	    dp->d_bno, ("wrong entry"));
18004c7da70SRuslan Ermilov 
18104c7da70SRuslan Ermilov 	off1 = BNO2OFF(dp->d_bno, sc);
18204c7da70SRuslan Ermilov 	off = MAX(bp->bio_offset, off1);
18304c7da70SRuslan Ermilov 	len = MIN(bp->bio_offset + bp->bio_length, off1 + sc->sc_bsize) - off;
18404c7da70SRuslan Ermilov 
18504c7da70SRuslan Ermilov 	if (bp->bio_error == 0)
18604c7da70SRuslan Ermilov 		bp->bio_error = error;
18704c7da70SRuslan Ermilov 	if (bp->bio_error == 0) {
18804c7da70SRuslan Ermilov 		bcopy(dp->d_data + (off - off1),
18904c7da70SRuslan Ermilov 		    bp->bio_data + (off - bp->bio_offset), len);
19004c7da70SRuslan Ermilov 	}
19104c7da70SRuslan Ermilov 	bp->bio_completed += len;
19204c7da70SRuslan Ermilov 	KASSERT(bp->bio_completed <= bp->bio_length, ("extra data"));
19304c7da70SRuslan Ermilov 	if (bp->bio_completed == bp->bio_length) {
19404c7da70SRuslan Ermilov 		if (bp->bio_error != 0)
19504c7da70SRuslan Ermilov 			bp->bio_completed = 0;
19604c7da70SRuslan Ermilov 		g_io_deliver(bp, bp->bio_error);
19704c7da70SRuslan Ermilov 	}
19804c7da70SRuslan Ermilov 
19904c7da70SRuslan Ermilov 	if (dp->d_flags & D_FLAG_USED) {
20004c7da70SRuslan Ermilov 		TAILQ_REMOVE(&sc->sc_usedlist, dp, d_used);
20104c7da70SRuslan Ermilov 		TAILQ_INSERT_TAIL(&sc->sc_usedlist, dp, d_used);
20204c7da70SRuslan Ermilov 	} else if (OFF2BNO(off + len, sc) > dp->d_bno) {
20304c7da70SRuslan Ermilov 		TAILQ_INSERT_TAIL(&sc->sc_usedlist, dp, d_used);
20404c7da70SRuslan Ermilov 		sc->sc_nused++;
20504c7da70SRuslan Ermilov 		dp->d_flags |= D_FLAG_USED;
20604c7da70SRuslan Ermilov 	}
20704c7da70SRuslan Ermilov 	dp->d_atime = time_uptime;
20804c7da70SRuslan Ermilov }
20904c7da70SRuslan Ermilov 
21004c7da70SRuslan Ermilov static void
21104c7da70SRuslan Ermilov g_cache_done(struct bio *bp)
21204c7da70SRuslan Ermilov {
21304c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
21404c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
21504c7da70SRuslan Ermilov 	struct bio *bp2, *tmpbp;
21604c7da70SRuslan Ermilov 
21704c7da70SRuslan Ermilov 	sc = bp->bio_from->geom->softc;
21804c7da70SRuslan Ermilov 	KASSERT(G_CACHE_DESC1(bp) == sc, ("corrupt bio_caller in g_cache_done()"));
21904c7da70SRuslan Ermilov 	dp = G_CACHE_DESC2(bp);
22004c7da70SRuslan Ermilov 	mtx_lock(&sc->sc_mtx);
22104c7da70SRuslan Ermilov 	bp2 = dp->d_biolist;
22204c7da70SRuslan Ermilov 	while (bp2 != NULL) {
22304c7da70SRuslan Ermilov 		KASSERT(G_CACHE_NEXT_BIO1(bp2) == sc, ("corrupt bio_driver in g_cache_done()"));
22404c7da70SRuslan Ermilov 		tmpbp = G_CACHE_NEXT_BIO2(bp2);
22504c7da70SRuslan Ermilov 		g_cache_deliver(sc, bp2, dp, bp->bio_error);
22604c7da70SRuslan Ermilov 		bp2 = tmpbp;
22704c7da70SRuslan Ermilov 	}
22804c7da70SRuslan Ermilov 	dp->d_biolist = NULL;
22904c7da70SRuslan Ermilov 	if (dp->d_flags & D_FLAG_INVALID) {
23004c7da70SRuslan Ermilov 		sc->sc_invalid--;
23104c7da70SRuslan Ermilov 		g_cache_free(sc, dp);
23204c7da70SRuslan Ermilov 	} else if (bp->bio_error) {
23304c7da70SRuslan Ermilov 		LIST_REMOVE(dp, d_next);
23404c7da70SRuslan Ermilov 		if (dp->d_flags & D_FLAG_USED) {
23504c7da70SRuslan Ermilov 			TAILQ_REMOVE(&sc->sc_usedlist, dp, d_used);
23604c7da70SRuslan Ermilov 			sc->sc_nused--;
23704c7da70SRuslan Ermilov 		}
23804c7da70SRuslan Ermilov 		g_cache_free(sc, dp);
23904c7da70SRuslan Ermilov 	}
24004c7da70SRuslan Ermilov 	mtx_unlock(&sc->sc_mtx);
24104c7da70SRuslan Ermilov 	g_destroy_bio(bp);
24204c7da70SRuslan Ermilov }
24304c7da70SRuslan Ermilov 
24404c7da70SRuslan Ermilov static struct g_cache_desc *
24504c7da70SRuslan Ermilov g_cache_lookup(struct g_cache_softc *sc, off_t bno)
24604c7da70SRuslan Ermilov {
24704c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
24804c7da70SRuslan Ermilov 
24904c7da70SRuslan Ermilov 	mtx_assert(&sc->sc_mtx, MA_OWNED);
25004c7da70SRuslan Ermilov 
25104c7da70SRuslan Ermilov 	LIST_FOREACH(dp, &sc->sc_desclist[G_CACHE_BUCKET(bno)], d_next)
25204c7da70SRuslan Ermilov 		if (dp->d_bno == bno)
25304c7da70SRuslan Ermilov 			return (dp);
25404c7da70SRuslan Ermilov 	return (NULL);
25504c7da70SRuslan Ermilov }
25604c7da70SRuslan Ermilov 
25704c7da70SRuslan Ermilov static int
25804c7da70SRuslan Ermilov g_cache_read(struct g_cache_softc *sc, struct bio *bp)
25904c7da70SRuslan Ermilov {
26004c7da70SRuslan Ermilov 	struct bio *cbp;
26104c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
26204c7da70SRuslan Ermilov 
26304c7da70SRuslan Ermilov 	mtx_lock(&sc->sc_mtx);
26404c7da70SRuslan Ermilov 	dp = g_cache_lookup(sc,
26504c7da70SRuslan Ermilov 	    OFF2BNO(bp->bio_offset + bp->bio_completed, sc));
26604c7da70SRuslan Ermilov 	if (dp != NULL) {
26704c7da70SRuslan Ermilov 		/* Add to waiters list or deliver. */
26804c7da70SRuslan Ermilov 		sc->sc_cachehits++;
26904c7da70SRuslan Ermilov 		if (dp->d_biolist != NULL) {
27004c7da70SRuslan Ermilov 			G_CACHE_NEXT_BIO1(bp) = sc;
27104c7da70SRuslan Ermilov 			G_CACHE_NEXT_BIO2(bp) = dp->d_biolist;
27204c7da70SRuslan Ermilov 			dp->d_biolist = bp;
27304c7da70SRuslan Ermilov 		} else
27404c7da70SRuslan Ermilov 			g_cache_deliver(sc, bp, dp, 0);
27504c7da70SRuslan Ermilov 		mtx_unlock(&sc->sc_mtx);
27604c7da70SRuslan Ermilov 		return (0);
27704c7da70SRuslan Ermilov 	}
27804c7da70SRuslan Ermilov 
27904c7da70SRuslan Ermilov 	/* Cache miss.  Allocate entry and schedule bio.  */
28004c7da70SRuslan Ermilov 	sc->sc_cachemisses++;
28104c7da70SRuslan Ermilov 	dp = g_cache_alloc(sc);
28204c7da70SRuslan Ermilov 	if (dp == NULL) {
28304c7da70SRuslan Ermilov 		mtx_unlock(&sc->sc_mtx);
28404c7da70SRuslan Ermilov 		return (ENOMEM);
28504c7da70SRuslan Ermilov 	}
28604c7da70SRuslan Ermilov 	cbp = g_clone_bio(bp);
28704c7da70SRuslan Ermilov 	if (cbp == NULL) {
28804c7da70SRuslan Ermilov 		g_cache_free(sc, dp);
28904c7da70SRuslan Ermilov 		mtx_unlock(&sc->sc_mtx);
29004c7da70SRuslan Ermilov 		return (ENOMEM);
29104c7da70SRuslan Ermilov 	}
29204c7da70SRuslan Ermilov 
29304c7da70SRuslan Ermilov 	dp->d_bno = OFF2BNO(bp->bio_offset + bp->bio_completed, sc);
29404c7da70SRuslan Ermilov 	G_CACHE_NEXT_BIO1(bp) = sc;
29504c7da70SRuslan Ermilov 	G_CACHE_NEXT_BIO2(bp) = NULL;
29604c7da70SRuslan Ermilov 	dp->d_biolist = bp;
29704c7da70SRuslan Ermilov 	LIST_INSERT_HEAD(&sc->sc_desclist[G_CACHE_BUCKET(dp->d_bno)],
29804c7da70SRuslan Ermilov 	    dp, d_next);
29904c7da70SRuslan Ermilov 	mtx_unlock(&sc->sc_mtx);
30004c7da70SRuslan Ermilov 
30104c7da70SRuslan Ermilov 	G_CACHE_DESC1(cbp) = sc;
30204c7da70SRuslan Ermilov 	G_CACHE_DESC2(cbp) = dp;
30304c7da70SRuslan Ermilov 	cbp->bio_done = g_cache_done;
30404c7da70SRuslan Ermilov 	cbp->bio_offset = BNO2OFF(dp->d_bno, sc);
30504c7da70SRuslan Ermilov 	cbp->bio_data = dp->d_data;
30604c7da70SRuslan Ermilov 	cbp->bio_length = sc->sc_bsize;
30704c7da70SRuslan Ermilov 	g_io_request(cbp, LIST_FIRST(&bp->bio_to->geom->consumer));
30804c7da70SRuslan Ermilov 	return (0);
30904c7da70SRuslan Ermilov }
31004c7da70SRuslan Ermilov 
31104c7da70SRuslan Ermilov static void
31204c7da70SRuslan Ermilov g_cache_invalidate(struct g_cache_softc *sc, struct bio *bp)
31304c7da70SRuslan Ermilov {
31404c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
31504c7da70SRuslan Ermilov 	off_t bno, lim;
31604c7da70SRuslan Ermilov 
31704c7da70SRuslan Ermilov 	mtx_lock(&sc->sc_mtx);
31804c7da70SRuslan Ermilov 	bno = OFF2BNO(bp->bio_offset, sc);
31904c7da70SRuslan Ermilov 	lim = OFF2BNO(bp->bio_offset + bp->bio_length - 1, sc);
32004c7da70SRuslan Ermilov 	do {
32104c7da70SRuslan Ermilov 		if ((dp = g_cache_lookup(sc, bno)) != NULL) {
32204c7da70SRuslan Ermilov 			LIST_REMOVE(dp, d_next);
32304c7da70SRuslan Ermilov 			if (dp->d_flags & D_FLAG_USED) {
32404c7da70SRuslan Ermilov 				TAILQ_REMOVE(&sc->sc_usedlist, dp, d_used);
32504c7da70SRuslan Ermilov 				sc->sc_nused--;
32604c7da70SRuslan Ermilov 			}
32704c7da70SRuslan Ermilov 			if (dp->d_biolist == NULL)
32804c7da70SRuslan Ermilov 				g_cache_free(sc, dp);
32904c7da70SRuslan Ermilov 			else {
33004c7da70SRuslan Ermilov 				dp->d_flags = D_FLAG_INVALID;
33104c7da70SRuslan Ermilov 				sc->sc_invalid++;
33204c7da70SRuslan Ermilov 			}
33304c7da70SRuslan Ermilov 		}
33404c7da70SRuslan Ermilov 		bno++;
33504c7da70SRuslan Ermilov 	} while (bno <= lim);
33604c7da70SRuslan Ermilov 	mtx_unlock(&sc->sc_mtx);
33704c7da70SRuslan Ermilov }
33804c7da70SRuslan Ermilov 
33904c7da70SRuslan Ermilov static void
34004c7da70SRuslan Ermilov g_cache_start(struct bio *bp)
34104c7da70SRuslan Ermilov {
34204c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
34304c7da70SRuslan Ermilov 	struct g_geom *gp;
34404c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
34504c7da70SRuslan Ermilov 	struct bio *cbp;
34604c7da70SRuslan Ermilov 
34704c7da70SRuslan Ermilov 	gp = bp->bio_to->geom;
34804c7da70SRuslan Ermilov 	sc = gp->softc;
34904c7da70SRuslan Ermilov 	G_CACHE_LOGREQ(bp, "Request received.");
35004c7da70SRuslan Ermilov 	switch (bp->bio_cmd) {
35104c7da70SRuslan Ermilov 	case BIO_READ:
35204c7da70SRuslan Ermilov 		sc->sc_reads++;
35304c7da70SRuslan Ermilov 		sc->sc_readbytes += bp->bio_length;
35404c7da70SRuslan Ermilov 		if (!g_cache_enable)
35504c7da70SRuslan Ermilov 			break;
35604c7da70SRuslan Ermilov 		if (bp->bio_offset + bp->bio_length > sc->sc_tail)
35704c7da70SRuslan Ermilov 			break;
35804c7da70SRuslan Ermilov 		if (OFF2BNO(bp->bio_offset, sc) ==
35904c7da70SRuslan Ermilov 		    OFF2BNO(bp->bio_offset + bp->bio_length - 1, sc)) {
36004c7da70SRuslan Ermilov 			sc->sc_cachereads++;
36104c7da70SRuslan Ermilov 			sc->sc_cachereadbytes += bp->bio_length;
36204c7da70SRuslan Ermilov 			if (g_cache_read(sc, bp) == 0)
36304c7da70SRuslan Ermilov 				return;
36404c7da70SRuslan Ermilov 			sc->sc_cachereads--;
36504c7da70SRuslan Ermilov 			sc->sc_cachereadbytes -= bp->bio_length;
36604c7da70SRuslan Ermilov 			break;
36704c7da70SRuslan Ermilov 		} else if (OFF2BNO(bp->bio_offset, sc) + 1 ==
36804c7da70SRuslan Ermilov 		    OFF2BNO(bp->bio_offset + bp->bio_length - 1, sc)) {
36904c7da70SRuslan Ermilov 			mtx_lock(&sc->sc_mtx);
37004c7da70SRuslan Ermilov 			dp = g_cache_lookup(sc, OFF2BNO(bp->bio_offset, sc));
37104c7da70SRuslan Ermilov 			if (dp == NULL || dp->d_biolist != NULL) {
37204c7da70SRuslan Ermilov 				mtx_unlock(&sc->sc_mtx);
37304c7da70SRuslan Ermilov 				break;
37404c7da70SRuslan Ermilov 			}
37504c7da70SRuslan Ermilov 			sc->sc_cachereads++;
37604c7da70SRuslan Ermilov 			sc->sc_cachereadbytes += bp->bio_length;
37704c7da70SRuslan Ermilov 			g_cache_deliver(sc, bp, dp, 0);
37804c7da70SRuslan Ermilov 			mtx_unlock(&sc->sc_mtx);
37904c7da70SRuslan Ermilov 			if (g_cache_read(sc, bp) == 0)
38004c7da70SRuslan Ermilov 				return;
38104c7da70SRuslan Ermilov 			sc->sc_cachereads--;
38204c7da70SRuslan Ermilov 			sc->sc_cachereadbytes -= bp->bio_length;
38304c7da70SRuslan Ermilov 			break;
38404c7da70SRuslan Ermilov 		}
38504c7da70SRuslan Ermilov 		break;
38604c7da70SRuslan Ermilov 	case BIO_WRITE:
38704c7da70SRuslan Ermilov 		sc->sc_writes++;
38804c7da70SRuslan Ermilov 		sc->sc_wrotebytes += bp->bio_length;
38904c7da70SRuslan Ermilov 		g_cache_invalidate(sc, bp);
39004c7da70SRuslan Ermilov 		break;
39104c7da70SRuslan Ermilov 	}
39204c7da70SRuslan Ermilov 	cbp = g_clone_bio(bp);
39304c7da70SRuslan Ermilov 	if (cbp == NULL) {
39404c7da70SRuslan Ermilov 		g_io_deliver(bp, ENOMEM);
39504c7da70SRuslan Ermilov 		return;
39604c7da70SRuslan Ermilov 	}
39704c7da70SRuslan Ermilov 	cbp->bio_done = g_std_done;
39804c7da70SRuslan Ermilov 	G_CACHE_LOGREQ(cbp, "Sending request.");
39904c7da70SRuslan Ermilov 	g_io_request(cbp, LIST_FIRST(&gp->consumer));
40004c7da70SRuslan Ermilov }
40104c7da70SRuslan Ermilov 
40204c7da70SRuslan Ermilov static void
40304c7da70SRuslan Ermilov g_cache_go(void *arg)
40404c7da70SRuslan Ermilov {
40504c7da70SRuslan Ermilov 	struct g_cache_softc *sc = arg;
40604c7da70SRuslan Ermilov 	struct g_cache_desc *dp;
40704c7da70SRuslan Ermilov 	int i;
40804c7da70SRuslan Ermilov 
40904c7da70SRuslan Ermilov 	mtx_assert(&sc->sc_mtx, MA_OWNED);
41004c7da70SRuslan Ermilov 
41104c7da70SRuslan Ermilov 	/* Forcibly mark idle ready entries as used. */
41204c7da70SRuslan Ermilov 	for (i = 0; i < G_CACHE_BUCKETS; i++) {
41304c7da70SRuslan Ermilov 		LIST_FOREACH(dp, &sc->sc_desclist[i], d_next) {
41404c7da70SRuslan Ermilov 			if (dp->d_flags & D_FLAG_USED ||
41504c7da70SRuslan Ermilov 			    dp->d_biolist != NULL ||
41604c7da70SRuslan Ermilov 			    time_uptime - dp->d_atime < g_cache_idletime)
41704c7da70SRuslan Ermilov 				continue;
41804c7da70SRuslan Ermilov 			TAILQ_INSERT_TAIL(&sc->sc_usedlist, dp, d_used);
41904c7da70SRuslan Ermilov 			sc->sc_nused++;
42004c7da70SRuslan Ermilov 			dp->d_flags |= D_FLAG_USED;
42104c7da70SRuslan Ermilov 		}
42204c7da70SRuslan Ermilov 	}
42304c7da70SRuslan Ermilov 
42404c7da70SRuslan Ermilov 	/* Keep the number of used entries low. */
42504c7da70SRuslan Ermilov 	if (sc->sc_nused > g_cache_used_hi * sc->sc_maxent / 100)
42604c7da70SRuslan Ermilov 		g_cache_free_used(sc);
42704c7da70SRuslan Ermilov 
42804c7da70SRuslan Ermilov 	callout_reset(&sc->sc_callout, g_cache_timeout * hz, g_cache_go, sc);
42904c7da70SRuslan Ermilov }
43004c7da70SRuslan Ermilov 
43104c7da70SRuslan Ermilov static int
43204c7da70SRuslan Ermilov g_cache_access(struct g_provider *pp, int dr, int dw, int de)
43304c7da70SRuslan Ermilov {
43404c7da70SRuslan Ermilov 	struct g_geom *gp;
43504c7da70SRuslan Ermilov 	struct g_consumer *cp;
43604c7da70SRuslan Ermilov 	int error;
43704c7da70SRuslan Ermilov 
43804c7da70SRuslan Ermilov 	gp = pp->geom;
43904c7da70SRuslan Ermilov 	cp = LIST_FIRST(&gp->consumer);
44004c7da70SRuslan Ermilov 	error = g_access(cp, dr, dw, de);
44104c7da70SRuslan Ermilov 
44204c7da70SRuslan Ermilov 	return (error);
44304c7da70SRuslan Ermilov }
44404c7da70SRuslan Ermilov 
44504c7da70SRuslan Ermilov static void
44604c7da70SRuslan Ermilov g_cache_orphan(struct g_consumer *cp)
44704c7da70SRuslan Ermilov {
44804c7da70SRuslan Ermilov 
44904c7da70SRuslan Ermilov 	g_topology_assert();
45004c7da70SRuslan Ermilov 	g_cache_destroy(cp->geom->softc, 1);
45104c7da70SRuslan Ermilov }
45204c7da70SRuslan Ermilov 
45304c7da70SRuslan Ermilov static struct g_cache_softc *
45404c7da70SRuslan Ermilov g_cache_find_device(struct g_class *mp, const char *name)
45504c7da70SRuslan Ermilov {
45604c7da70SRuslan Ermilov 	struct g_geom *gp;
45704c7da70SRuslan Ermilov 
45804c7da70SRuslan Ermilov 	LIST_FOREACH(gp, &mp->geom, geom) {
45904c7da70SRuslan Ermilov 		if (strcmp(gp->name, name) == 0)
46004c7da70SRuslan Ermilov 			return (gp->softc);
46104c7da70SRuslan Ermilov 	}
46204c7da70SRuslan Ermilov 	return (NULL);
46304c7da70SRuslan Ermilov }
46404c7da70SRuslan Ermilov 
46504c7da70SRuslan Ermilov static struct g_geom *
46604c7da70SRuslan Ermilov g_cache_create(struct g_class *mp, struct g_provider *pp,
46704c7da70SRuslan Ermilov     const struct g_cache_metadata *md, u_int type)
46804c7da70SRuslan Ermilov {
46904c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
47004c7da70SRuslan Ermilov 	struct g_geom *gp;
47104c7da70SRuslan Ermilov 	struct g_provider *newpp;
47204c7da70SRuslan Ermilov 	struct g_consumer *cp;
47304c7da70SRuslan Ermilov 	u_int bshift;
47404c7da70SRuslan Ermilov 	int i;
47504c7da70SRuslan Ermilov 
47604c7da70SRuslan Ermilov 	g_topology_assert();
47704c7da70SRuslan Ermilov 
47804c7da70SRuslan Ermilov 	gp = NULL;
47904c7da70SRuslan Ermilov 	newpp = NULL;
48004c7da70SRuslan Ermilov 	cp = NULL;
48104c7da70SRuslan Ermilov 
48204c7da70SRuslan Ermilov 	G_CACHE_DEBUG(1, "Creating device %s.", md->md_name);
48304c7da70SRuslan Ermilov 
48404c7da70SRuslan Ermilov 	/* Cache size is minimum 100. */
48504c7da70SRuslan Ermilov 	if (md->md_size < 100) {
48604c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Invalid size for device %s.", md->md_name);
48704c7da70SRuslan Ermilov 		return (NULL);
48804c7da70SRuslan Ermilov 	}
48904c7da70SRuslan Ermilov 
49004c7da70SRuslan Ermilov 	/* Block size restrictions. */
49104c7da70SRuslan Ermilov 	bshift = ffs(md->md_bsize) - 1;
49204c7da70SRuslan Ermilov 	if (md->md_bsize == 0 || md->md_bsize > MAXPHYS ||
49304c7da70SRuslan Ermilov 	    md->md_bsize != 1 << bshift ||
49404c7da70SRuslan Ermilov 	    (md->md_bsize % pp->sectorsize) != 0) {
49504c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Invalid blocksize for provider %s.", pp->name);
49604c7da70SRuslan Ermilov 		return (NULL);
49704c7da70SRuslan Ermilov 	}
49804c7da70SRuslan Ermilov 
49904c7da70SRuslan Ermilov 	/* Check for duplicate unit. */
50004c7da70SRuslan Ermilov 	if (g_cache_find_device(mp, (const char *)&md->md_name) != NULL) {
50104c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Provider %s already exists.", md->md_name);
50204c7da70SRuslan Ermilov 		return (NULL);
50304c7da70SRuslan Ermilov 	}
50404c7da70SRuslan Ermilov 
505*02c62349SJaakko Heinonen 	gp = g_new_geomf(mp, "%s", md->md_name);
50604c7da70SRuslan Ermilov 	sc = g_malloc(sizeof(*sc), M_WAITOK | M_ZERO);
50704c7da70SRuslan Ermilov 	sc->sc_type = type;
50804c7da70SRuslan Ermilov 	sc->sc_bshift = bshift;
50904c7da70SRuslan Ermilov 	sc->sc_bsize = 1 << bshift;
51004c7da70SRuslan Ermilov 	sc->sc_zone = uma_zcreate("gcache", sc->sc_bsize, NULL, NULL, NULL, NULL,
51104c7da70SRuslan Ermilov 	    UMA_ALIGN_PTR, 0);
51204c7da70SRuslan Ermilov 	mtx_init(&sc->sc_mtx, "GEOM CACHE mutex", NULL, MTX_DEF);
51304c7da70SRuslan Ermilov 	for (i = 0; i < G_CACHE_BUCKETS; i++)
51404c7da70SRuslan Ermilov 		LIST_INIT(&sc->sc_desclist[i]);
51504c7da70SRuslan Ermilov 	TAILQ_INIT(&sc->sc_usedlist);
51604c7da70SRuslan Ermilov 	sc->sc_maxent = md->md_size;
51704c7da70SRuslan Ermilov 	callout_init_mtx(&sc->sc_callout, &sc->sc_mtx, 0);
51804c7da70SRuslan Ermilov 	gp->softc = sc;
51904c7da70SRuslan Ermilov 	sc->sc_geom = gp;
52004c7da70SRuslan Ermilov 	gp->start = g_cache_start;
52104c7da70SRuslan Ermilov 	gp->orphan = g_cache_orphan;
52204c7da70SRuslan Ermilov 	gp->access = g_cache_access;
52304c7da70SRuslan Ermilov 	gp->dumpconf = g_cache_dumpconf;
52404c7da70SRuslan Ermilov 
52504c7da70SRuslan Ermilov 	newpp = g_new_providerf(gp, "cache/%s", gp->name);
52604c7da70SRuslan Ermilov 	newpp->sectorsize = pp->sectorsize;
52704c7da70SRuslan Ermilov 	newpp->mediasize = pp->mediasize;
52804c7da70SRuslan Ermilov 	if (type == G_CACHE_TYPE_AUTOMATIC)
52904c7da70SRuslan Ermilov 		newpp->mediasize -= pp->sectorsize;
53004c7da70SRuslan Ermilov 	sc->sc_tail = BNO2OFF(OFF2BNO(newpp->mediasize, sc), sc);
53104c7da70SRuslan Ermilov 
53204c7da70SRuslan Ermilov 	cp = g_new_consumer(gp);
53304c7da70SRuslan Ermilov 	if (g_attach(cp, pp) != 0) {
53404c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Cannot attach to provider %s.", pp->name);
535e62dffbfSAndrey V. Elsukov 		g_destroy_consumer(cp);
536e62dffbfSAndrey V. Elsukov 		g_destroy_provider(newpp);
537e62dffbfSAndrey V. Elsukov 		mtx_destroy(&sc->sc_mtx);
538e62dffbfSAndrey V. Elsukov 		g_free(sc);
539e62dffbfSAndrey V. Elsukov 		g_destroy_geom(gp);
540e62dffbfSAndrey V. Elsukov 		return (NULL);
54104c7da70SRuslan Ermilov 	}
54204c7da70SRuslan Ermilov 
54304c7da70SRuslan Ermilov 	g_error_provider(newpp, 0);
54404c7da70SRuslan Ermilov 	G_CACHE_DEBUG(0, "Device %s created.", gp->name);
54504c7da70SRuslan Ermilov 	callout_reset(&sc->sc_callout, g_cache_timeout * hz, g_cache_go, sc);
54604c7da70SRuslan Ermilov 	return (gp);
54704c7da70SRuslan Ermilov }
54804c7da70SRuslan Ermilov 
54904c7da70SRuslan Ermilov static int
55004c7da70SRuslan Ermilov g_cache_destroy(struct g_cache_softc *sc, boolean_t force)
55104c7da70SRuslan Ermilov {
55204c7da70SRuslan Ermilov 	struct g_geom *gp;
55304c7da70SRuslan Ermilov 	struct g_provider *pp;
55404c7da70SRuslan Ermilov 	struct g_cache_desc *dp, *dp2;
55504c7da70SRuslan Ermilov 	int i;
55604c7da70SRuslan Ermilov 
55704c7da70SRuslan Ermilov 	g_topology_assert();
55804c7da70SRuslan Ermilov 	if (sc == NULL)
55904c7da70SRuslan Ermilov 		return (ENXIO);
56004c7da70SRuslan Ermilov 	gp = sc->sc_geom;
56104c7da70SRuslan Ermilov 	pp = LIST_FIRST(&gp->provider);
56204c7da70SRuslan Ermilov 	if (pp != NULL && (pp->acr != 0 || pp->acw != 0 || pp->ace != 0)) {
56304c7da70SRuslan Ermilov 		if (force) {
56404c7da70SRuslan Ermilov 			G_CACHE_DEBUG(0, "Device %s is still open, so it "
56504c7da70SRuslan Ermilov 			    "can't be definitely removed.", pp->name);
56604c7da70SRuslan Ermilov 		} else {
56704c7da70SRuslan Ermilov 			G_CACHE_DEBUG(1, "Device %s is still open (r%dw%de%d).",
56804c7da70SRuslan Ermilov 			    pp->name, pp->acr, pp->acw, pp->ace);
56904c7da70SRuslan Ermilov 			return (EBUSY);
57004c7da70SRuslan Ermilov 		}
57104c7da70SRuslan Ermilov 	} else {
57204c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Device %s removed.", gp->name);
57304c7da70SRuslan Ermilov 	}
57404c7da70SRuslan Ermilov 	callout_drain(&sc->sc_callout);
57504c7da70SRuslan Ermilov 	mtx_lock(&sc->sc_mtx);
57604c7da70SRuslan Ermilov 	for (i = 0; i < G_CACHE_BUCKETS; i++) {
57704c7da70SRuslan Ermilov 		dp = LIST_FIRST(&sc->sc_desclist[i]);
57804c7da70SRuslan Ermilov 		while (dp != NULL) {
57904c7da70SRuslan Ermilov 			dp2 = LIST_NEXT(dp, d_next);
58004c7da70SRuslan Ermilov 			g_cache_free(sc, dp);
58104c7da70SRuslan Ermilov 			dp = dp2;
58204c7da70SRuslan Ermilov 		}
58304c7da70SRuslan Ermilov 	}
58404c7da70SRuslan Ermilov 	mtx_unlock(&sc->sc_mtx);
58504c7da70SRuslan Ermilov 	mtx_destroy(&sc->sc_mtx);
58604c7da70SRuslan Ermilov 	uma_zdestroy(sc->sc_zone);
58704c7da70SRuslan Ermilov 	g_free(sc);
58804c7da70SRuslan Ermilov 	gp->softc = NULL;
58904c7da70SRuslan Ermilov 	g_wither_geom(gp, ENXIO);
59004c7da70SRuslan Ermilov 
59104c7da70SRuslan Ermilov 	return (0);
59204c7da70SRuslan Ermilov }
59304c7da70SRuslan Ermilov 
59404c7da70SRuslan Ermilov static int
59504c7da70SRuslan Ermilov g_cache_destroy_geom(struct gctl_req *req, struct g_class *mp, struct g_geom *gp)
59604c7da70SRuslan Ermilov {
59704c7da70SRuslan Ermilov 
59804c7da70SRuslan Ermilov 	return (g_cache_destroy(gp->softc, 0));
59904c7da70SRuslan Ermilov }
60004c7da70SRuslan Ermilov 
60104c7da70SRuslan Ermilov static int
60204c7da70SRuslan Ermilov g_cache_read_metadata(struct g_consumer *cp, struct g_cache_metadata *md)
60304c7da70SRuslan Ermilov {
60404c7da70SRuslan Ermilov 	struct g_provider *pp;
60504c7da70SRuslan Ermilov 	u_char *buf;
60604c7da70SRuslan Ermilov 	int error;
60704c7da70SRuslan Ermilov 
60804c7da70SRuslan Ermilov 	g_topology_assert();
60904c7da70SRuslan Ermilov 
61004c7da70SRuslan Ermilov 	error = g_access(cp, 1, 0, 0);
61104c7da70SRuslan Ermilov 	if (error != 0)
61204c7da70SRuslan Ermilov 		return (error);
61304c7da70SRuslan Ermilov 	pp = cp->provider;
61404c7da70SRuslan Ermilov 	g_topology_unlock();
61504c7da70SRuslan Ermilov 	buf = g_read_data(cp, pp->mediasize - pp->sectorsize, pp->sectorsize,
61604c7da70SRuslan Ermilov 	    &error);
61704c7da70SRuslan Ermilov 	g_topology_lock();
61804c7da70SRuslan Ermilov 	g_access(cp, -1, 0, 0);
61904c7da70SRuslan Ermilov 	if (buf == NULL)
62004c7da70SRuslan Ermilov 		return (error);
62104c7da70SRuslan Ermilov 
62204c7da70SRuslan Ermilov 	/* Decode metadata. */
62304c7da70SRuslan Ermilov 	cache_metadata_decode(buf, md);
62404c7da70SRuslan Ermilov 	g_free(buf);
62504c7da70SRuslan Ermilov 
62604c7da70SRuslan Ermilov 	return (0);
62704c7da70SRuslan Ermilov }
62804c7da70SRuslan Ermilov 
62904c7da70SRuslan Ermilov static int
63004c7da70SRuslan Ermilov g_cache_write_metadata(struct g_consumer *cp, struct g_cache_metadata *md)
63104c7da70SRuslan Ermilov {
63204c7da70SRuslan Ermilov 	struct g_provider *pp;
63304c7da70SRuslan Ermilov 	u_char *buf;
63404c7da70SRuslan Ermilov 	int error;
63504c7da70SRuslan Ermilov 
63604c7da70SRuslan Ermilov 	g_topology_assert();
63704c7da70SRuslan Ermilov 
63804c7da70SRuslan Ermilov 	error = g_access(cp, 0, 1, 0);
63904c7da70SRuslan Ermilov 	if (error != 0)
64004c7da70SRuslan Ermilov 		return (error);
64104c7da70SRuslan Ermilov 	pp = cp->provider;
64204c7da70SRuslan Ermilov 	buf = malloc((size_t)pp->sectorsize, M_GCACHE, M_WAITOK | M_ZERO);
64304c7da70SRuslan Ermilov 	cache_metadata_encode(md, buf);
64404c7da70SRuslan Ermilov 	g_topology_unlock();
64504c7da70SRuslan Ermilov 	error = g_write_data(cp, pp->mediasize - pp->sectorsize, buf, pp->sectorsize);
64604c7da70SRuslan Ermilov 	g_topology_lock();
64704c7da70SRuslan Ermilov 	g_access(cp, 0, -1, 0);
64804c7da70SRuslan Ermilov 	free(buf, M_GCACHE);
64904c7da70SRuslan Ermilov 
65004c7da70SRuslan Ermilov 	return (error);
65104c7da70SRuslan Ermilov }
65204c7da70SRuslan Ermilov 
65304c7da70SRuslan Ermilov static struct g_geom *
65404c7da70SRuslan Ermilov g_cache_taste(struct g_class *mp, struct g_provider *pp, int flags __unused)
65504c7da70SRuslan Ermilov {
65604c7da70SRuslan Ermilov 	struct g_cache_metadata md;
65704c7da70SRuslan Ermilov 	struct g_consumer *cp;
65804c7da70SRuslan Ermilov 	struct g_geom *gp;
65904c7da70SRuslan Ermilov 	int error;
66004c7da70SRuslan Ermilov 
66104c7da70SRuslan Ermilov 	g_trace(G_T_TOPOLOGY, "%s(%s, %s)", __func__, mp->name, pp->name);
66204c7da70SRuslan Ermilov 	g_topology_assert();
66304c7da70SRuslan Ermilov 
66404c7da70SRuslan Ermilov 	G_CACHE_DEBUG(3, "Tasting %s.", pp->name);
66504c7da70SRuslan Ermilov 
66604c7da70SRuslan Ermilov 	gp = g_new_geomf(mp, "cache:taste");
66704c7da70SRuslan Ermilov 	gp->start = g_cache_start;
66804c7da70SRuslan Ermilov 	gp->orphan = g_cache_orphan;
66904c7da70SRuslan Ermilov 	gp->access = g_cache_access;
67004c7da70SRuslan Ermilov 	cp = g_new_consumer(gp);
67104c7da70SRuslan Ermilov 	g_attach(cp, pp);
67204c7da70SRuslan Ermilov 	error = g_cache_read_metadata(cp, &md);
67304c7da70SRuslan Ermilov 	g_detach(cp);
67404c7da70SRuslan Ermilov 	g_destroy_consumer(cp);
67504c7da70SRuslan Ermilov 	g_destroy_geom(gp);
67604c7da70SRuslan Ermilov 	if (error != 0)
67704c7da70SRuslan Ermilov 		return (NULL);
67804c7da70SRuslan Ermilov 
67904c7da70SRuslan Ermilov 	if (strcmp(md.md_magic, G_CACHE_MAGIC) != 0)
68004c7da70SRuslan Ermilov 		return (NULL);
68104c7da70SRuslan Ermilov 	if (md.md_version > G_CACHE_VERSION) {
68204c7da70SRuslan Ermilov 		printf("geom_cache.ko module is too old to handle %s.\n",
68304c7da70SRuslan Ermilov 		    pp->name);
68404c7da70SRuslan Ermilov 		return (NULL);
68504c7da70SRuslan Ermilov 	}
68604c7da70SRuslan Ermilov 	if (md.md_provsize != pp->mediasize)
68704c7da70SRuslan Ermilov 		return (NULL);
68804c7da70SRuslan Ermilov 
68904c7da70SRuslan Ermilov 	gp = g_cache_create(mp, pp, &md, G_CACHE_TYPE_AUTOMATIC);
69004c7da70SRuslan Ermilov 	if (gp == NULL) {
69104c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Can't create %s.", md.md_name);
69204c7da70SRuslan Ermilov 		return (NULL);
69304c7da70SRuslan Ermilov 	}
69404c7da70SRuslan Ermilov 	return (gp);
69504c7da70SRuslan Ermilov }
69604c7da70SRuslan Ermilov 
69704c7da70SRuslan Ermilov static void
69804c7da70SRuslan Ermilov g_cache_ctl_create(struct gctl_req *req, struct g_class *mp)
69904c7da70SRuslan Ermilov {
70004c7da70SRuslan Ermilov 	struct g_cache_metadata md;
70104c7da70SRuslan Ermilov 	struct g_provider *pp;
70204c7da70SRuslan Ermilov 	struct g_geom *gp;
70304c7da70SRuslan Ermilov 	intmax_t *bsize, *size;
70404c7da70SRuslan Ermilov 	const char *name;
70504c7da70SRuslan Ermilov 	int *nargs;
70604c7da70SRuslan Ermilov 
70704c7da70SRuslan Ermilov 	g_topology_assert();
70804c7da70SRuslan Ermilov 
70904c7da70SRuslan Ermilov 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
71004c7da70SRuslan Ermilov 	if (nargs == NULL) {
71104c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "nargs");
71204c7da70SRuslan Ermilov 		return;
71304c7da70SRuslan Ermilov 	}
71404c7da70SRuslan Ermilov 	if (*nargs != 2) {
71504c7da70SRuslan Ermilov 		gctl_error(req, "Invalid number of arguments.");
71604c7da70SRuslan Ermilov 		return;
71704c7da70SRuslan Ermilov 	}
71804c7da70SRuslan Ermilov 
71904c7da70SRuslan Ermilov 	strlcpy(md.md_magic, G_CACHE_MAGIC, sizeof(md.md_magic));
72004c7da70SRuslan Ermilov 	md.md_version = G_CACHE_VERSION;
72104c7da70SRuslan Ermilov 	name = gctl_get_asciiparam(req, "arg0");
72204c7da70SRuslan Ermilov 	if (name == NULL) {
72304c7da70SRuslan Ermilov 		gctl_error(req, "No 'arg0' argument");
72404c7da70SRuslan Ermilov 		return;
72504c7da70SRuslan Ermilov 	}
72604c7da70SRuslan Ermilov 	strlcpy(md.md_name, name, sizeof(md.md_name));
72704c7da70SRuslan Ermilov 
72804c7da70SRuslan Ermilov 	size = gctl_get_paraml(req, "size", sizeof(*size));
72904c7da70SRuslan Ermilov 	if (size == NULL) {
73004c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "size");
73104c7da70SRuslan Ermilov 		return;
73204c7da70SRuslan Ermilov 	}
73304c7da70SRuslan Ermilov 	if ((u_int)*size < 100) {
73404c7da70SRuslan Ermilov 		gctl_error(req, "Invalid '%s' argument", "size");
73504c7da70SRuslan Ermilov 		return;
73604c7da70SRuslan Ermilov 	}
73704c7da70SRuslan Ermilov 	md.md_size = (u_int)*size;
73804c7da70SRuslan Ermilov 
73904c7da70SRuslan Ermilov 	bsize = gctl_get_paraml(req, "blocksize", sizeof(*bsize));
74004c7da70SRuslan Ermilov 	if (bsize == NULL) {
74104c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "blocksize");
74204c7da70SRuslan Ermilov 		return;
74304c7da70SRuslan Ermilov 	}
74404c7da70SRuslan Ermilov 	if (*bsize < 0) {
74504c7da70SRuslan Ermilov 		gctl_error(req, "Invalid '%s' argument", "blocksize");
74604c7da70SRuslan Ermilov 		return;
74704c7da70SRuslan Ermilov 	}
74804c7da70SRuslan Ermilov 	md.md_bsize = (u_int)*bsize;
74904c7da70SRuslan Ermilov 
75004c7da70SRuslan Ermilov 	/* This field is not important here. */
75104c7da70SRuslan Ermilov 	md.md_provsize = 0;
75204c7da70SRuslan Ermilov 
75304c7da70SRuslan Ermilov 	name = gctl_get_asciiparam(req, "arg1");
75404c7da70SRuslan Ermilov 	if (name == NULL) {
75504c7da70SRuslan Ermilov 		gctl_error(req, "No 'arg1' argument");
75604c7da70SRuslan Ermilov 		return;
75704c7da70SRuslan Ermilov 	}
75804c7da70SRuslan Ermilov 	if (strncmp(name, "/dev/", strlen("/dev/")) == 0)
75904c7da70SRuslan Ermilov 		name += strlen("/dev/");
76004c7da70SRuslan Ermilov 	pp = g_provider_by_name(name);
76104c7da70SRuslan Ermilov 	if (pp == NULL) {
76204c7da70SRuslan Ermilov 		G_CACHE_DEBUG(1, "Provider %s is invalid.", name);
76304c7da70SRuslan Ermilov 		gctl_error(req, "Provider %s is invalid.", name);
76404c7da70SRuslan Ermilov 		return;
76504c7da70SRuslan Ermilov 	}
76604c7da70SRuslan Ermilov 	gp = g_cache_create(mp, pp, &md, G_CACHE_TYPE_MANUAL);
76704c7da70SRuslan Ermilov 	if (gp == NULL) {
76804c7da70SRuslan Ermilov 		gctl_error(req, "Can't create %s.", md.md_name);
76904c7da70SRuslan Ermilov 		return;
77004c7da70SRuslan Ermilov 	}
77104c7da70SRuslan Ermilov }
77204c7da70SRuslan Ermilov 
77304c7da70SRuslan Ermilov static void
77404c7da70SRuslan Ermilov g_cache_ctl_configure(struct gctl_req *req, struct g_class *mp)
77504c7da70SRuslan Ermilov {
77604c7da70SRuslan Ermilov 	struct g_cache_metadata md;
77704c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
77804c7da70SRuslan Ermilov 	struct g_consumer *cp;
77904c7da70SRuslan Ermilov 	intmax_t *bsize, *size;
78004c7da70SRuslan Ermilov 	const char *name;
78104c7da70SRuslan Ermilov 	int error, *nargs;
78204c7da70SRuslan Ermilov 
78304c7da70SRuslan Ermilov 	g_topology_assert();
78404c7da70SRuslan Ermilov 
78504c7da70SRuslan Ermilov 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
78604c7da70SRuslan Ermilov 	if (nargs == NULL) {
78704c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "nargs");
78804c7da70SRuslan Ermilov 		return;
78904c7da70SRuslan Ermilov 	}
79004c7da70SRuslan Ermilov 	if (*nargs != 1) {
79104c7da70SRuslan Ermilov 		gctl_error(req, "Missing device.");
79204c7da70SRuslan Ermilov 		return;
79304c7da70SRuslan Ermilov 	}
79404c7da70SRuslan Ermilov 
79504c7da70SRuslan Ermilov 	name = gctl_get_asciiparam(req, "arg0");
79604c7da70SRuslan Ermilov 	if (name == NULL) {
79704c7da70SRuslan Ermilov 		gctl_error(req, "No 'arg0' argument");
79804c7da70SRuslan Ermilov 		return;
79904c7da70SRuslan Ermilov 	}
80004c7da70SRuslan Ermilov 	sc = g_cache_find_device(mp, name);
80104c7da70SRuslan Ermilov 	if (sc == NULL) {
80204c7da70SRuslan Ermilov 		G_CACHE_DEBUG(1, "Device %s is invalid.", name);
80304c7da70SRuslan Ermilov 		gctl_error(req, "Device %s is invalid.", name);
80404c7da70SRuslan Ermilov 		return;
80504c7da70SRuslan Ermilov 	}
80604c7da70SRuslan Ermilov 
80704c7da70SRuslan Ermilov 	size = gctl_get_paraml(req, "size", sizeof(*size));
80804c7da70SRuslan Ermilov 	if (size == NULL) {
80904c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "size");
81004c7da70SRuslan Ermilov 		return;
81104c7da70SRuslan Ermilov 	}
81204c7da70SRuslan Ermilov 	if ((u_int)*size != 0 && (u_int)*size < 100) {
81304c7da70SRuslan Ermilov 		gctl_error(req, "Invalid '%s' argument", "size");
81404c7da70SRuslan Ermilov 		return;
81504c7da70SRuslan Ermilov 	}
81604c7da70SRuslan Ermilov 	if ((u_int)*size != 0)
81704c7da70SRuslan Ermilov 		sc->sc_maxent = (u_int)*size;
81804c7da70SRuslan Ermilov 
81904c7da70SRuslan Ermilov 	bsize = gctl_get_paraml(req, "blocksize", sizeof(*bsize));
82004c7da70SRuslan Ermilov 	if (bsize == NULL) {
82104c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "blocksize");
82204c7da70SRuslan Ermilov 		return;
82304c7da70SRuslan Ermilov 	}
82404c7da70SRuslan Ermilov 	if (*bsize < 0) {
82504c7da70SRuslan Ermilov 		gctl_error(req, "Invalid '%s' argument", "blocksize");
82604c7da70SRuslan Ermilov 		return;
82704c7da70SRuslan Ermilov 	}
82804c7da70SRuslan Ermilov 
82904c7da70SRuslan Ermilov 	if (sc->sc_type != G_CACHE_TYPE_AUTOMATIC)
83004c7da70SRuslan Ermilov 		return;
83104c7da70SRuslan Ermilov 
83204c7da70SRuslan Ermilov 	strlcpy(md.md_name, name, sizeof(md.md_name));
83304c7da70SRuslan Ermilov 	strlcpy(md.md_magic, G_CACHE_MAGIC, sizeof(md.md_magic));
83404c7da70SRuslan Ermilov 	md.md_version = G_CACHE_VERSION;
83504c7da70SRuslan Ermilov 	if ((u_int)*size != 0)
83604c7da70SRuslan Ermilov 		md.md_size = (u_int)*size;
83704c7da70SRuslan Ermilov 	else
83804c7da70SRuslan Ermilov 		md.md_size = sc->sc_maxent;
83904c7da70SRuslan Ermilov 	if ((u_int)*bsize != 0)
84004c7da70SRuslan Ermilov 		md.md_bsize = (u_int)*bsize;
84104c7da70SRuslan Ermilov 	else
84204c7da70SRuslan Ermilov 		md.md_bsize = sc->sc_bsize;
84304c7da70SRuslan Ermilov 	cp = LIST_FIRST(&sc->sc_geom->consumer);
84404c7da70SRuslan Ermilov 	md.md_provsize = cp->provider->mediasize;
84504c7da70SRuslan Ermilov 	error = g_cache_write_metadata(cp, &md);
84604c7da70SRuslan Ermilov 	if (error == 0)
84704c7da70SRuslan Ermilov 		G_CACHE_DEBUG(2, "Metadata on %s updated.", cp->provider->name);
84804c7da70SRuslan Ermilov 	else
84904c7da70SRuslan Ermilov 		G_CACHE_DEBUG(0, "Cannot update metadata on %s (error=%d).",
85004c7da70SRuslan Ermilov 		    cp->provider->name, error);
85104c7da70SRuslan Ermilov }
85204c7da70SRuslan Ermilov 
85304c7da70SRuslan Ermilov static void
85404c7da70SRuslan Ermilov g_cache_ctl_destroy(struct gctl_req *req, struct g_class *mp)
85504c7da70SRuslan Ermilov {
85604c7da70SRuslan Ermilov 	int *nargs, *force, error, i;
85704c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
85804c7da70SRuslan Ermilov 	const char *name;
85904c7da70SRuslan Ermilov 	char param[16];
86004c7da70SRuslan Ermilov 
86104c7da70SRuslan Ermilov 	g_topology_assert();
86204c7da70SRuslan Ermilov 
86304c7da70SRuslan Ermilov 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
86404c7da70SRuslan Ermilov 	if (nargs == NULL) {
86504c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "nargs");
86604c7da70SRuslan Ermilov 		return;
86704c7da70SRuslan Ermilov 	}
86804c7da70SRuslan Ermilov 	if (*nargs <= 0) {
86904c7da70SRuslan Ermilov 		gctl_error(req, "Missing device(s).");
87004c7da70SRuslan Ermilov 		return;
87104c7da70SRuslan Ermilov 	}
87204c7da70SRuslan Ermilov 	force = gctl_get_paraml(req, "force", sizeof(*force));
87304c7da70SRuslan Ermilov 	if (force == NULL) {
87404c7da70SRuslan Ermilov 		gctl_error(req, "No 'force' argument");
87504c7da70SRuslan Ermilov 		return;
87604c7da70SRuslan Ermilov 	}
87704c7da70SRuslan Ermilov 
87804c7da70SRuslan Ermilov 	for (i = 0; i < *nargs; i++) {
87904c7da70SRuslan Ermilov 		snprintf(param, sizeof(param), "arg%d", i);
88004c7da70SRuslan Ermilov 		name = gctl_get_asciiparam(req, param);
88104c7da70SRuslan Ermilov 		if (name == NULL) {
88204c7da70SRuslan Ermilov 			gctl_error(req, "No 'arg%d' argument", i);
88304c7da70SRuslan Ermilov 			return;
88404c7da70SRuslan Ermilov 		}
88504c7da70SRuslan Ermilov 		sc = g_cache_find_device(mp, name);
88604c7da70SRuslan Ermilov 		if (sc == NULL) {
88704c7da70SRuslan Ermilov 			G_CACHE_DEBUG(1, "Device %s is invalid.", name);
88804c7da70SRuslan Ermilov 			gctl_error(req, "Device %s is invalid.", name);
88904c7da70SRuslan Ermilov 			return;
89004c7da70SRuslan Ermilov 		}
89104c7da70SRuslan Ermilov 		error = g_cache_destroy(sc, *force);
89204c7da70SRuslan Ermilov 		if (error != 0) {
89304c7da70SRuslan Ermilov 			gctl_error(req, "Cannot destroy device %s (error=%d).",
89404c7da70SRuslan Ermilov 			    sc->sc_name, error);
89504c7da70SRuslan Ermilov 			return;
89604c7da70SRuslan Ermilov 		}
89704c7da70SRuslan Ermilov 	}
89804c7da70SRuslan Ermilov }
89904c7da70SRuslan Ermilov 
90004c7da70SRuslan Ermilov static void
90104c7da70SRuslan Ermilov g_cache_ctl_reset(struct gctl_req *req, struct g_class *mp)
90204c7da70SRuslan Ermilov {
90304c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
90404c7da70SRuslan Ermilov 	const char *name;
90504c7da70SRuslan Ermilov 	char param[16];
90604c7da70SRuslan Ermilov 	int i, *nargs;
90704c7da70SRuslan Ermilov 
90804c7da70SRuslan Ermilov 	g_topology_assert();
90904c7da70SRuslan Ermilov 
91004c7da70SRuslan Ermilov 	nargs = gctl_get_paraml(req, "nargs", sizeof(*nargs));
91104c7da70SRuslan Ermilov 	if (nargs == NULL) {
91204c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument", "nargs");
91304c7da70SRuslan Ermilov 		return;
91404c7da70SRuslan Ermilov 	}
91504c7da70SRuslan Ermilov 	if (*nargs <= 0) {
91604c7da70SRuslan Ermilov 		gctl_error(req, "Missing device(s).");
91704c7da70SRuslan Ermilov 		return;
91804c7da70SRuslan Ermilov 	}
91904c7da70SRuslan Ermilov 
92004c7da70SRuslan Ermilov 	for (i = 0; i < *nargs; i++) {
92104c7da70SRuslan Ermilov 		snprintf(param, sizeof(param), "arg%d", i);
92204c7da70SRuslan Ermilov 		name = gctl_get_asciiparam(req, param);
92304c7da70SRuslan Ermilov 		if (name == NULL) {
92404c7da70SRuslan Ermilov 			gctl_error(req, "No 'arg%d' argument", i);
92504c7da70SRuslan Ermilov 			return;
92604c7da70SRuslan Ermilov 		}
92704c7da70SRuslan Ermilov 		sc = g_cache_find_device(mp, name);
92804c7da70SRuslan Ermilov 		if (sc == NULL) {
92904c7da70SRuslan Ermilov 			G_CACHE_DEBUG(1, "Device %s is invalid.", name);
93004c7da70SRuslan Ermilov 			gctl_error(req, "Device %s is invalid.", name);
93104c7da70SRuslan Ermilov 			return;
93204c7da70SRuslan Ermilov 		}
93304c7da70SRuslan Ermilov 		sc->sc_reads = 0;
93404c7da70SRuslan Ermilov 		sc->sc_readbytes = 0;
93504c7da70SRuslan Ermilov 		sc->sc_cachereads = 0;
93604c7da70SRuslan Ermilov 		sc->sc_cachereadbytes = 0;
93704c7da70SRuslan Ermilov 		sc->sc_cachehits = 0;
93804c7da70SRuslan Ermilov 		sc->sc_cachemisses = 0;
93904c7da70SRuslan Ermilov 		sc->sc_cachefull = 0;
94004c7da70SRuslan Ermilov 		sc->sc_writes = 0;
94104c7da70SRuslan Ermilov 		sc->sc_wrotebytes = 0;
94204c7da70SRuslan Ermilov 	}
94304c7da70SRuslan Ermilov }
94404c7da70SRuslan Ermilov 
94504c7da70SRuslan Ermilov static void
94604c7da70SRuslan Ermilov g_cache_config(struct gctl_req *req, struct g_class *mp, const char *verb)
94704c7da70SRuslan Ermilov {
94804c7da70SRuslan Ermilov 	uint32_t *version;
94904c7da70SRuslan Ermilov 
95004c7da70SRuslan Ermilov 	g_topology_assert();
95104c7da70SRuslan Ermilov 
95204c7da70SRuslan Ermilov 	version = gctl_get_paraml(req, "version", sizeof(*version));
95304c7da70SRuslan Ermilov 	if (version == NULL) {
95404c7da70SRuslan Ermilov 		gctl_error(req, "No '%s' argument.", "version");
95504c7da70SRuslan Ermilov 		return;
95604c7da70SRuslan Ermilov 	}
95704c7da70SRuslan Ermilov 	if (*version != G_CACHE_VERSION) {
95804c7da70SRuslan Ermilov 		gctl_error(req, "Userland and kernel parts are out of sync.");
95904c7da70SRuslan Ermilov 		return;
96004c7da70SRuslan Ermilov 	}
96104c7da70SRuslan Ermilov 
96204c7da70SRuslan Ermilov 	if (strcmp(verb, "create") == 0) {
96304c7da70SRuslan Ermilov 		g_cache_ctl_create(req, mp);
96404c7da70SRuslan Ermilov 		return;
96504c7da70SRuslan Ermilov 	} else if (strcmp(verb, "configure") == 0) {
96604c7da70SRuslan Ermilov 		g_cache_ctl_configure(req, mp);
96704c7da70SRuslan Ermilov 		return;
96804c7da70SRuslan Ermilov 	} else if (strcmp(verb, "destroy") == 0 ||
96904c7da70SRuslan Ermilov 	    strcmp(verb, "stop") == 0) {
97004c7da70SRuslan Ermilov 		g_cache_ctl_destroy(req, mp);
97104c7da70SRuslan Ermilov 		return;
97204c7da70SRuslan Ermilov 	} else if (strcmp(verb, "reset") == 0) {
97304c7da70SRuslan Ermilov 		g_cache_ctl_reset(req, mp);
97404c7da70SRuslan Ermilov 		return;
97504c7da70SRuslan Ermilov 	}
97604c7da70SRuslan Ermilov 
97704c7da70SRuslan Ermilov 	gctl_error(req, "Unknown verb.");
97804c7da70SRuslan Ermilov }
97904c7da70SRuslan Ermilov 
98004c7da70SRuslan Ermilov static void
98104c7da70SRuslan Ermilov g_cache_dumpconf(struct sbuf *sb, const char *indent, struct g_geom *gp,
98204c7da70SRuslan Ermilov     struct g_consumer *cp, struct g_provider *pp)
98304c7da70SRuslan Ermilov {
98404c7da70SRuslan Ermilov 	struct g_cache_softc *sc;
98504c7da70SRuslan Ermilov 
98604c7da70SRuslan Ermilov 	if (pp != NULL || cp != NULL)
98704c7da70SRuslan Ermilov 		return;
98804c7da70SRuslan Ermilov 	sc = gp->softc;
98904c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<Size>%u</Size>\n", indent, sc->sc_maxent);
99004c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<BlockSize>%u</BlockSize>\n", indent, sc->sc_bsize);
99104c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<TailOffset>%ju</TailOffset>\n", indent,
99204c7da70SRuslan Ermilov 	    (uintmax_t)sc->sc_tail);
99304c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<Entries>%u</Entries>\n", indent, sc->sc_nent);
99404c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<UsedEntries>%u</UsedEntries>\n", indent,
99504c7da70SRuslan Ermilov 	    sc->sc_nused);
99604c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<InvalidEntries>%u</InvalidEntries>\n", indent,
99704c7da70SRuslan Ermilov 	    sc->sc_invalid);
99804c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<Reads>%ju</Reads>\n", indent, sc->sc_reads);
99904c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<ReadBytes>%ju</ReadBytes>\n", indent,
100004c7da70SRuslan Ermilov 	    sc->sc_readbytes);
100104c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<CacheReads>%ju</CacheReads>\n", indent,
100204c7da70SRuslan Ermilov 	    sc->sc_cachereads);
100304c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<CacheReadBytes>%ju</CacheReadBytes>\n", indent,
100404c7da70SRuslan Ermilov 	    sc->sc_cachereadbytes);
100504c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<CacheHits>%ju</CacheHits>\n", indent,
100604c7da70SRuslan Ermilov 	    sc->sc_cachehits);
100704c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<CacheMisses>%ju</CacheMisses>\n", indent,
100804c7da70SRuslan Ermilov 	    sc->sc_cachemisses);
100904c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<CacheFull>%ju</CacheFull>\n", indent,
101004c7da70SRuslan Ermilov 	    sc->sc_cachefull);
101104c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<Writes>%ju</Writes>\n", indent, sc->sc_writes);
101204c7da70SRuslan Ermilov 	sbuf_printf(sb, "%s<WroteBytes>%ju</WroteBytes>\n", indent,
101304c7da70SRuslan Ermilov 	    sc->sc_wrotebytes);
101404c7da70SRuslan Ermilov }
101504c7da70SRuslan Ermilov 
101604c7da70SRuslan Ermilov DECLARE_GEOM_CLASS(g_cache_class, g_cache);
1017