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