17a59208dSJustin T. Gibbs /* 22a888f93SKenneth D. Merry * Copyright (c) 1997, 1998, 1999 Kenneth D. Merry. 37a59208dSJustin T. Gibbs * All rights reserved. 47a59208dSJustin T. Gibbs * 57a59208dSJustin T. Gibbs * Redistribution and use in source and binary forms, with or without 67a59208dSJustin T. Gibbs * modification, are permitted provided that the following conditions 77a59208dSJustin T. Gibbs * are met: 87a59208dSJustin T. Gibbs * 1. Redistributions of source code must retain the above copyright 97a59208dSJustin T. Gibbs * notice, this list of conditions and the following disclaimer. 107a59208dSJustin T. Gibbs * 2. Redistributions in binary form must reproduce the above copyright 117a59208dSJustin T. Gibbs * notice, this list of conditions and the following disclaimer in the 127a59208dSJustin T. Gibbs * documentation and/or other materials provided with the distribution. 137a59208dSJustin T. Gibbs * 3. The name of the author may not be used to endorse or promote products 147a59208dSJustin T. Gibbs * derived from this software without specific prior written permission. 157a59208dSJustin T. Gibbs * 167a59208dSJustin T. Gibbs * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 177a59208dSJustin T. Gibbs * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 187a59208dSJustin T. Gibbs * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 197a59208dSJustin T. Gibbs * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 207a59208dSJustin T. Gibbs * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 217a59208dSJustin T. Gibbs * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 227a59208dSJustin T. Gibbs * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 237a59208dSJustin T. Gibbs * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 247a59208dSJustin T. Gibbs * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 257a59208dSJustin T. Gibbs * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 267a59208dSJustin T. Gibbs * SUCH DAMAGE. 277a59208dSJustin T. Gibbs * 28c3aac50fSPeter Wemm * $FreeBSD$ 297a59208dSJustin T. Gibbs */ 307a59208dSJustin T. Gibbs 317a59208dSJustin T. Gibbs #include <sys/param.h> 327a59208dSJustin T. Gibbs #include <sys/kernel.h> 337a59208dSJustin T. Gibbs #include <sys/systm.h> 349626b608SPoul-Henning Kamp #include <sys/bio.h> 357a59208dSJustin T. Gibbs #include <sys/sysctl.h> 36c7e73d59SPoul-Henning Kamp #include <sys/malloc.h> 37c7e73d59SPoul-Henning Kamp #include <sys/conf.h> 38c7e73d59SPoul-Henning Kamp #include <vm/vm.h> 39c7e73d59SPoul-Henning Kamp #include <vm/pmap.h> 407a59208dSJustin T. Gibbs 417a59208dSJustin T. Gibbs #include <sys/devicestat.h> 427a59208dSJustin T. Gibbs 437a59208dSJustin T. Gibbs static int devstat_num_devs; 44bcc6a3daSKenneth D. Merry static long devstat_generation; 457a59208dSJustin T. Gibbs static int devstat_version = DEVSTAT_VERSION; 467a59208dSJustin T. Gibbs static int devstat_current_devnumber; 477a59208dSJustin T. Gibbs 48938a4e5cSThomas Moestl static struct devstatlist device_statq; 49c7e73d59SPoul-Henning Kamp static struct devstat *devstat_alloc(void); 50c7e73d59SPoul-Henning Kamp static void devstat_free(struct devstat *); 51f37de122SPoul-Henning Kamp static void devstat_add_entry(struct devstat *ds, const char *dev_name, 52f37de122SPoul-Henning Kamp int unit_number, u_int32_t block_size, 53f37de122SPoul-Henning Kamp devstat_support_flags flags, 54f37de122SPoul-Henning Kamp devstat_type_flags device_type, 55f37de122SPoul-Henning Kamp devstat_priority priority); 56c7e73d59SPoul-Henning Kamp 57c7e73d59SPoul-Henning Kamp /* 58c7e73d59SPoul-Henning Kamp * Allocate a devstat and initialize it 59c7e73d59SPoul-Henning Kamp */ 60c7e73d59SPoul-Henning Kamp struct devstat * 61c7e73d59SPoul-Henning Kamp devstat_new_entry(const char *dev_name, 62c7e73d59SPoul-Henning Kamp int unit_number, u_int32_t block_size, 63c7e73d59SPoul-Henning Kamp devstat_support_flags flags, 64c7e73d59SPoul-Henning Kamp devstat_type_flags device_type, 65c7e73d59SPoul-Henning Kamp devstat_priority priority) 66c7e73d59SPoul-Henning Kamp { 67c7e73d59SPoul-Henning Kamp struct devstat *ds; 68c7e73d59SPoul-Henning Kamp 69c7e73d59SPoul-Henning Kamp ds = devstat_alloc(); 70c7e73d59SPoul-Henning Kamp devstat_add_entry(ds, dev_name, unit_number, block_size, 71c7e73d59SPoul-Henning Kamp flags, device_type, priority); 72c7e73d59SPoul-Henning Kamp return (ds); 73c7e73d59SPoul-Henning Kamp } 747a59208dSJustin T. Gibbs 757a59208dSJustin T. Gibbs /* 767a59208dSJustin T. Gibbs * Take a malloced and zeroed devstat structure given to us, fill it in 777a59208dSJustin T. Gibbs * and add it to the queue of devices. 787a59208dSJustin T. Gibbs */ 79f37de122SPoul-Henning Kamp static void 8014177d72SGarrett Wollman devstat_add_entry(struct devstat *ds, const char *dev_name, 817a59208dSJustin T. Gibbs int unit_number, u_int32_t block_size, 827a59208dSJustin T. Gibbs devstat_support_flags flags, 832a888f93SKenneth D. Merry devstat_type_flags device_type, 842a888f93SKenneth D. Merry devstat_priority priority) 857a59208dSJustin T. Gibbs { 867a59208dSJustin T. Gibbs struct devstatlist *devstat_head; 872a888f93SKenneth D. Merry struct devstat *ds_tmp; 887a59208dSJustin T. Gibbs 897a59208dSJustin T. Gibbs if (ds == NULL) 907a59208dSJustin T. Gibbs return; 917a59208dSJustin T. Gibbs 927a59208dSJustin T. Gibbs if (devstat_num_devs == 0) 937a59208dSJustin T. Gibbs STAILQ_INIT(&device_statq); 947a59208dSJustin T. Gibbs 957a59208dSJustin T. Gibbs devstat_generation++; 967a59208dSJustin T. Gibbs devstat_num_devs++; 977a59208dSJustin T. Gibbs 987a59208dSJustin T. Gibbs devstat_head = &device_statq; 997a59208dSJustin T. Gibbs 1002a888f93SKenneth D. Merry /* 1012a888f93SKenneth D. Merry * Priority sort. Each driver passes in its priority when it adds 1022a888f93SKenneth D. Merry * its devstat entry. Drivers are sorted first by priority, and 1032a888f93SKenneth D. Merry * then by probe order. 1042a888f93SKenneth D. Merry * 1052a888f93SKenneth D. Merry * For the first device, we just insert it, since the priority 1062a888f93SKenneth D. Merry * doesn't really matter yet. Subsequent devices are inserted into 1072a888f93SKenneth D. Merry * the list using the order outlined above. 1082a888f93SKenneth D. Merry */ 1092a888f93SKenneth D. Merry if (devstat_num_devs == 1) 1107a59208dSJustin T. Gibbs STAILQ_INSERT_TAIL(devstat_head, ds, dev_links); 1112a888f93SKenneth D. Merry else { 11237d40066SPoul-Henning Kamp STAILQ_FOREACH(ds_tmp, devstat_head, dev_links) { 1132a888f93SKenneth D. Merry struct devstat *ds_next; 1142a888f93SKenneth D. Merry 1152a888f93SKenneth D. Merry ds_next = STAILQ_NEXT(ds_tmp, dev_links); 1162a888f93SKenneth D. Merry 1172a888f93SKenneth D. Merry /* 1182a888f93SKenneth D. Merry * If we find a break between higher and lower 1192a888f93SKenneth D. Merry * priority items, and if this item fits in the 1202a888f93SKenneth D. Merry * break, insert it. This also applies if the 1212a888f93SKenneth D. Merry * "lower priority item" is the end of the list. 1222a888f93SKenneth D. Merry */ 1232a888f93SKenneth D. Merry if ((priority <= ds_tmp->priority) 1242a888f93SKenneth D. Merry && ((ds_next == NULL) 1252a888f93SKenneth D. Merry || (priority > ds_next->priority))) { 1262a888f93SKenneth D. Merry STAILQ_INSERT_AFTER(devstat_head, ds_tmp, ds, 1272a888f93SKenneth D. Merry dev_links); 1282a888f93SKenneth D. Merry break; 1292a888f93SKenneth D. Merry } else if (priority > ds_tmp->priority) { 1302a888f93SKenneth D. Merry /* 1312a888f93SKenneth D. Merry * If this is the case, we should be able 1322a888f93SKenneth D. Merry * to insert ourselves at the head of the 1332a888f93SKenneth D. Merry * list. If we can't, something is wrong. 1342a888f93SKenneth D. Merry */ 1352a888f93SKenneth D. Merry if (ds_tmp == STAILQ_FIRST(devstat_head)) { 1362a888f93SKenneth D. Merry STAILQ_INSERT_HEAD(devstat_head, 1372a888f93SKenneth D. Merry ds, dev_links); 1382a888f93SKenneth D. Merry break; 1392a888f93SKenneth D. Merry } else { 1402a888f93SKenneth D. Merry STAILQ_INSERT_TAIL(devstat_head, 1412a888f93SKenneth D. Merry ds, dev_links); 1422a888f93SKenneth D. Merry printf("devstat_add_entry: HELP! " 1432a888f93SKenneth D. Merry "sorting problem detected " 1442a888f93SKenneth D. Merry "for %s%d\n", dev_name, 1452a888f93SKenneth D. Merry unit_number); 1462a888f93SKenneth D. Merry break; 1472a888f93SKenneth D. Merry } 1482a888f93SKenneth D. Merry } 1492a888f93SKenneth D. Merry } 1502a888f93SKenneth D. Merry } 1517a59208dSJustin T. Gibbs 1527a59208dSJustin T. Gibbs ds->device_number = devstat_current_devnumber++; 1537a59208dSJustin T. Gibbs ds->unit_number = unit_number; 154e80fb434SRobert Drehmel strlcpy(ds->device_name, dev_name, DEVSTAT_NAME_LEN); 1557a59208dSJustin T. Gibbs ds->block_size = block_size; 1567a59208dSJustin T. Gibbs ds->flags = flags; 1577a59208dSJustin T. Gibbs ds->device_type = device_type; 1582a888f93SKenneth D. Merry ds->priority = priority; 1597194d335SPoul-Henning Kamp binuptime(&ds->creation_time); 1607a59208dSJustin T. Gibbs } 1617a59208dSJustin T. Gibbs 1627a59208dSJustin T. Gibbs /* 1637a59208dSJustin T. Gibbs * Remove a devstat structure from the list of devices. 1647a59208dSJustin T. Gibbs */ 1657a59208dSJustin T. Gibbs void 1667a59208dSJustin T. Gibbs devstat_remove_entry(struct devstat *ds) 1677a59208dSJustin T. Gibbs { 1687a59208dSJustin T. Gibbs struct devstatlist *devstat_head; 1697a59208dSJustin T. Gibbs 1707a59208dSJustin T. Gibbs if (ds == NULL) 1717a59208dSJustin T. Gibbs return; 1727a59208dSJustin T. Gibbs 1737a59208dSJustin T. Gibbs devstat_generation++; 1747a59208dSJustin T. Gibbs devstat_num_devs--; 1757a59208dSJustin T. Gibbs 1767a59208dSJustin T. Gibbs devstat_head = &device_statq; 1777a59208dSJustin T. Gibbs 1787a59208dSJustin T. Gibbs /* Remove this entry from the devstat queue */ 179e3975643SJake Burkholder STAILQ_REMOVE(devstat_head, ds, devstat, dev_links); 180c7e73d59SPoul-Henning Kamp if (ds->allocated) 181c7e73d59SPoul-Henning Kamp devstat_free(ds); 1827a59208dSJustin T. Gibbs } 1837a59208dSJustin T. Gibbs 1847a59208dSJustin T. Gibbs /* 1857a59208dSJustin T. Gibbs * Record a transaction start. 1867194d335SPoul-Henning Kamp * 1877194d335SPoul-Henning Kamp * See comments for devstat_end_transaction(). Ordering is very important 1887194d335SPoul-Henning Kamp * here. 1897a59208dSJustin T. Gibbs */ 1907a59208dSJustin T. Gibbs void 1917194d335SPoul-Henning Kamp devstat_start_transaction(struct devstat *ds, struct bintime *now) 1927a59208dSJustin T. Gibbs { 1937a59208dSJustin T. Gibbs /* sanity check */ 1947a59208dSJustin T. Gibbs if (ds == NULL) 1957a59208dSJustin T. Gibbs return; 1967a59208dSJustin T. Gibbs 1977a59208dSJustin T. Gibbs /* 1987a59208dSJustin T. Gibbs * We only want to set the start time when we are going from idle 1997a59208dSJustin T. Gibbs * to busy. The start time is really the start of the latest busy 2007a59208dSJustin T. Gibbs * period. 2017a59208dSJustin T. Gibbs */ 2027194d335SPoul-Henning Kamp if (ds->start_count == ds->end_count) { 2037194d335SPoul-Henning Kamp if (now != NULL) 2047194d335SPoul-Henning Kamp ds->busy_from = *now; 2057194d335SPoul-Henning Kamp else 2067194d335SPoul-Henning Kamp binuptime(&ds->busy_from); 2077194d335SPoul-Henning Kamp } 2087194d335SPoul-Henning Kamp ds->start_count++; 2097194d335SPoul-Henning Kamp } 2107194d335SPoul-Henning Kamp 2117194d335SPoul-Henning Kamp void 2127194d335SPoul-Henning Kamp devstat_start_transaction_bio(struct devstat *ds, struct bio *bp) 2137194d335SPoul-Henning Kamp { 2147194d335SPoul-Henning Kamp 2157194d335SPoul-Henning Kamp binuptime(&bp->bio_t0); 2167194d335SPoul-Henning Kamp devstat_start_transaction(ds, &bp->bio_t0); 2177a59208dSJustin T. Gibbs } 2187a59208dSJustin T. Gibbs 2199fa85de2SPoul-Henning Kamp void 2209fa85de2SPoul-Henning Kamp devstat_start_transaction_bio(struct devstat *ds, struct bio *bp) 2219fa85de2SPoul-Henning Kamp { 2229fa85de2SPoul-Henning Kamp 2239fa85de2SPoul-Henning Kamp devstat_start_transaction(ds); 2249fa85de2SPoul-Henning Kamp } 2259fa85de2SPoul-Henning Kamp 2267a59208dSJustin T. Gibbs /* 2277a59208dSJustin T. Gibbs * Record the ending of a transaction, and incrment the various counters. 2287194d335SPoul-Henning Kamp * 2297194d335SPoul-Henning Kamp * Ordering in this function, and in devstat_start_transaction() is VERY 2307194d335SPoul-Henning Kamp * important. The idea here is to run without locks, so we are very 2317194d335SPoul-Henning Kamp * careful to only modify some fields on the way "down" (i.e. at 2327194d335SPoul-Henning Kamp * transaction start) and some fields on the way "up" (i.e. at transaction 2337194d335SPoul-Henning Kamp * completion). One exception is busy_from, which we only modify in 2347194d335SPoul-Henning Kamp * devstat_start_transaction() when there are no outstanding transactions, 2357194d335SPoul-Henning Kamp * and thus it can't be modified in devstat_end_transaction() 2367194d335SPoul-Henning Kamp * simultaneously. 2377a59208dSJustin T. Gibbs */ 2387a59208dSJustin T. Gibbs void 2397a59208dSJustin T. Gibbs devstat_end_transaction(struct devstat *ds, u_int32_t bytes, 2407194d335SPoul-Henning Kamp devstat_tag_type tag_type, devstat_trans_flags flags, 2417194d335SPoul-Henning Kamp struct bintime *now, struct bintime *then) 2427a59208dSJustin T. Gibbs { 2437194d335SPoul-Henning Kamp struct bintime dt, lnow; 2447a59208dSJustin T. Gibbs 2457a59208dSJustin T. Gibbs /* sanity check */ 2467a59208dSJustin T. Gibbs if (ds == NULL) 2477a59208dSJustin T. Gibbs return; 2487a59208dSJustin T. Gibbs 2497194d335SPoul-Henning Kamp if (now == NULL) { 2507194d335SPoul-Henning Kamp now = &lnow; 2517194d335SPoul-Henning Kamp binuptime(now); 2527194d335SPoul-Henning Kamp } 2537a59208dSJustin T. Gibbs 2547194d335SPoul-Henning Kamp /* Update byte and operations counts */ 2557194d335SPoul-Henning Kamp ds->bytes[flags] += bytes; 2567194d335SPoul-Henning Kamp ds->operations[flags]++; 2577a59208dSJustin T. Gibbs 2587a59208dSJustin T. Gibbs /* 2597a59208dSJustin T. Gibbs * Keep a count of the various tag types sent. 2607a59208dSJustin T. Gibbs */ 2618db3b947SPoul-Henning Kamp if ((ds->flags & DEVSTAT_NO_ORDERED_TAGS) == 0 && 262f80d57eeSPoul-Henning Kamp tag_type != DEVSTAT_TAG_NONE) 2637a59208dSJustin T. Gibbs ds->tag_types[tag_type]++; 2647a59208dSJustin T. Gibbs 2657194d335SPoul-Henning Kamp if (then != NULL) { 2667194d335SPoul-Henning Kamp /* Update duration of operations */ 2677194d335SPoul-Henning Kamp dt = *now; 2687194d335SPoul-Henning Kamp bintime_sub(&dt, then); 2697194d335SPoul-Henning Kamp bintime_add(&ds->duration[flags], &dt); 2707194d335SPoul-Henning Kamp } 2717a59208dSJustin T. Gibbs 2727194d335SPoul-Henning Kamp /* Accumulate busy time */ 2737194d335SPoul-Henning Kamp dt = *now; 2747194d335SPoul-Henning Kamp bintime_sub(&dt, &ds->busy_from); 2757194d335SPoul-Henning Kamp bintime_add(&ds->busy_time, &dt); 2767194d335SPoul-Henning Kamp ds->busy_from = *now; 2777194d335SPoul-Henning Kamp 2787194d335SPoul-Henning Kamp ds->end_count++; 2797a59208dSJustin T. Gibbs } 2807a59208dSJustin T. Gibbs 281f80d57eeSPoul-Henning Kamp void 282282ac69eSPoul-Henning Kamp devstat_end_transaction_bio(struct devstat *ds, struct bio *bp) 283282ac69eSPoul-Henning Kamp { 284282ac69eSPoul-Henning Kamp devstat_trans_flags flg; 285282ac69eSPoul-Henning Kamp 286282ac69eSPoul-Henning Kamp if (bp->bio_cmd == BIO_DELETE) 287282ac69eSPoul-Henning Kamp flg = DEVSTAT_FREE; 288282ac69eSPoul-Henning Kamp else if (bp->bio_cmd == BIO_READ) 289282ac69eSPoul-Henning Kamp flg = DEVSTAT_READ; 290282ac69eSPoul-Henning Kamp else 291282ac69eSPoul-Henning Kamp flg = DEVSTAT_WRITE; 292282ac69eSPoul-Henning Kamp 293282ac69eSPoul-Henning Kamp devstat_end_transaction(ds, bp->bio_bcount - bp->bio_resid, 2947194d335SPoul-Henning Kamp DEVSTAT_TAG_SIMPLE, flg, NULL, &bp->bio_t0); 295282ac69eSPoul-Henning Kamp } 296282ac69eSPoul-Henning Kamp 2977a59208dSJustin T. Gibbs /* 2987a59208dSJustin T. Gibbs * This is the sysctl handler for the devstat package. The data pushed out 2997a59208dSJustin T. Gibbs * on the kern.devstat.all sysctl variable consists of the current devstat 3007a59208dSJustin T. Gibbs * generation number, and then an array of devstat structures, one for each 3017a59208dSJustin T. Gibbs * device in the system. 3027a59208dSJustin T. Gibbs * 3037a59208dSJustin T. Gibbs * I'm really not too fond of this method of doing things, but there really 3047a59208dSJustin T. Gibbs * aren't that many alternatives. We must have some method of making sure 3057a59208dSJustin T. Gibbs * that the generation number the user gets corresponds with the data the 3067a59208dSJustin T. Gibbs * user gets. If the user makes a separate sysctl call to get the 3077a59208dSJustin T. Gibbs * generation, and then a sysctl call to get the device statistics, the 3087a59208dSJustin T. Gibbs * device list could have changed in that brief period of time. By 3097a59208dSJustin T. Gibbs * supplying the generation number along with the statistics output, we can 3107a59208dSJustin T. Gibbs * guarantee that the generation number and the statistics match up. 3117a59208dSJustin T. Gibbs */ 3127a59208dSJustin T. Gibbs static int 31382d9ae4eSPoul-Henning Kamp sysctl_devstat(SYSCTL_HANDLER_ARGS) 3147a59208dSJustin T. Gibbs { 3157a59208dSJustin T. Gibbs int error, i; 3167a59208dSJustin T. Gibbs struct devstat *nds; 3177a59208dSJustin T. Gibbs struct devstatlist *devstat_head; 3187a59208dSJustin T. Gibbs 3197a59208dSJustin T. Gibbs if (devstat_num_devs == 0) 3207a59208dSJustin T. Gibbs return(EINVAL); 3217a59208dSJustin T. Gibbs 3227a59208dSJustin T. Gibbs error = 0; 3237a59208dSJustin T. Gibbs devstat_head = &device_statq; 3247a59208dSJustin T. Gibbs 3257a59208dSJustin T. Gibbs /* 3267a59208dSJustin T. Gibbs * First push out the generation number. 3277a59208dSJustin T. Gibbs */ 328bcc6a3daSKenneth D. Merry error = SYSCTL_OUT(req, &devstat_generation, sizeof(long)); 3297a59208dSJustin T. Gibbs 3307a59208dSJustin T. Gibbs /* 3317a59208dSJustin T. Gibbs * Now push out all the devices. 3327a59208dSJustin T. Gibbs */ 3332e3c8fcbSPoul-Henning Kamp for (i = 0, nds = STAILQ_FIRST(devstat_head); 3347a59208dSJustin T. Gibbs (nds != NULL) && (i < devstat_num_devs) && (error == 0); 3352e3c8fcbSPoul-Henning Kamp nds = STAILQ_NEXT(nds, dev_links), i++) 3367a59208dSJustin T. Gibbs error = SYSCTL_OUT(req, nds, sizeof(struct devstat)); 3377a59208dSJustin T. Gibbs 3387a59208dSJustin T. Gibbs return(error); 3397a59208dSJustin T. Gibbs } 3407a59208dSJustin T. Gibbs 3417a59208dSJustin T. Gibbs /* 3427a59208dSJustin T. Gibbs * Sysctl entries for devstat. The first one is a node that all the rest 3437a59208dSJustin T. Gibbs * hang off of. 3447a59208dSJustin T. Gibbs */ 3457a59208dSJustin T. Gibbs SYSCTL_NODE(_kern, OID_AUTO, devstat, CTLFLAG_RD, 0, "Device Statistics"); 3467a59208dSJustin T. Gibbs 3477a59208dSJustin T. Gibbs SYSCTL_PROC(_kern_devstat, OID_AUTO, all, CTLFLAG_RD|CTLTYPE_OPAQUE, 3483d177f46SBill Fumerola 0, 0, sysctl_devstat, "S,devstat", "All devices in the devstat list"); 3497a59208dSJustin T. Gibbs /* 3507a59208dSJustin T. Gibbs * Export the number of devices in the system so that userland utilities 3517a59208dSJustin T. Gibbs * can determine how much memory to allocate to hold all the devices. 3527a59208dSJustin T. Gibbs */ 3533d177f46SBill Fumerola SYSCTL_INT(_kern_devstat, OID_AUTO, numdevs, CTLFLAG_RD, 3543d177f46SBill Fumerola &devstat_num_devs, 0, "Number of devices in the devstat list"); 355bcc6a3daSKenneth D. Merry SYSCTL_LONG(_kern_devstat, OID_AUTO, generation, CTLFLAG_RD, 3569701cd40SJohn Baldwin &devstat_generation, 0, "Devstat list generation"); 3573d177f46SBill Fumerola SYSCTL_INT(_kern_devstat, OID_AUTO, version, CTLFLAG_RD, 3583d177f46SBill Fumerola &devstat_version, 0, "Devstat list version number"); 359c7e73d59SPoul-Henning Kamp 360c7e73d59SPoul-Henning Kamp #define statsperpage (PAGE_SIZE / sizeof(struct devstat)) 361c7e73d59SPoul-Henning Kamp 362c7e73d59SPoul-Henning Kamp static d_mmap_t devstat_mmap; 363c7e73d59SPoul-Henning Kamp 364c7e73d59SPoul-Henning Kamp static struct cdevsw devstat_cdevsw = { 365c7e73d59SPoul-Henning Kamp .d_open = nullopen, 366c7e73d59SPoul-Henning Kamp .d_close = nullclose, 367c7e73d59SPoul-Henning Kamp .d_mmap = devstat_mmap, 368c7e73d59SPoul-Henning Kamp .d_name = "devstat", 369c7e73d59SPoul-Henning Kamp }; 370c7e73d59SPoul-Henning Kamp 371c7e73d59SPoul-Henning Kamp struct statspage { 372c7e73d59SPoul-Henning Kamp TAILQ_ENTRY(statspage) list; 373c7e73d59SPoul-Henning Kamp struct devstat *stat; 374c7e73d59SPoul-Henning Kamp u_int nfree; 375c7e73d59SPoul-Henning Kamp }; 376c7e73d59SPoul-Henning Kamp 377c7e73d59SPoul-Henning Kamp static TAILQ_HEAD(, statspage) pagelist = TAILQ_HEAD_INITIALIZER(pagelist); 378c7e73d59SPoul-Henning Kamp static MALLOC_DEFINE(M_DEVSTAT, "devstat", "Device statistics"); 379c7e73d59SPoul-Henning Kamp 380c7e73d59SPoul-Henning Kamp static int 381c7e73d59SPoul-Henning Kamp devstat_mmap(dev_t dev, vm_offset_t offset, vm_offset_t *paddr, int nprot) 382c7e73d59SPoul-Henning Kamp { 383c7e73d59SPoul-Henning Kamp struct statspage *spp; 384c7e73d59SPoul-Henning Kamp 385c7e73d59SPoul-Henning Kamp if (nprot != VM_PROT_READ) 386c7e73d59SPoul-Henning Kamp return (-1); 387c7e73d59SPoul-Henning Kamp TAILQ_FOREACH(spp, &pagelist, list) { 388c7e73d59SPoul-Henning Kamp if (offset == 0) { 389c7e73d59SPoul-Henning Kamp *paddr = vtophys(spp->stat); 390c7e73d59SPoul-Henning Kamp return (0); 391c7e73d59SPoul-Henning Kamp } 392c7e73d59SPoul-Henning Kamp offset -= PAGE_SIZE; 393c7e73d59SPoul-Henning Kamp } 394c7e73d59SPoul-Henning Kamp return (-1); 395c7e73d59SPoul-Henning Kamp } 396c7e73d59SPoul-Henning Kamp 397c7e73d59SPoul-Henning Kamp static struct devstat * 398c7e73d59SPoul-Henning Kamp devstat_alloc(void) 399c7e73d59SPoul-Henning Kamp { 400c7e73d59SPoul-Henning Kamp struct devstat *dsp; 401c7e73d59SPoul-Henning Kamp struct statspage *spp; 402c7e73d59SPoul-Henning Kamp u_int u; 403c7e73d59SPoul-Henning Kamp static int once; 404c7e73d59SPoul-Henning Kamp 405c7e73d59SPoul-Henning Kamp if (!once) { 406c7e73d59SPoul-Henning Kamp make_dev(&devstat_cdevsw, 0, 407c7e73d59SPoul-Henning Kamp UID_ROOT, GID_WHEEL, 0400, "devstat"); 408c7e73d59SPoul-Henning Kamp once++; 409c7e73d59SPoul-Henning Kamp } 410c7e73d59SPoul-Henning Kamp TAILQ_FOREACH(spp, &pagelist, list) { 411c7e73d59SPoul-Henning Kamp if (spp->nfree > 0) 412c7e73d59SPoul-Henning Kamp break; 413c7e73d59SPoul-Henning Kamp } 414c7e73d59SPoul-Henning Kamp if (spp == NULL) { 415c7e73d59SPoul-Henning Kamp spp = malloc(sizeof *spp, M_DEVSTAT, M_ZERO | M_WAITOK); 416c7e73d59SPoul-Henning Kamp TAILQ_INSERT_TAIL(&pagelist, spp, list); 417c7e73d59SPoul-Henning Kamp spp->stat = malloc(PAGE_SIZE, M_DEVSTAT, M_ZERO | M_WAITOK); 418c7e73d59SPoul-Henning Kamp spp->nfree = statsperpage; 419c7e73d59SPoul-Henning Kamp } 420c7e73d59SPoul-Henning Kamp dsp = spp->stat; 421c7e73d59SPoul-Henning Kamp for (u = 0; u < statsperpage; u++) { 422c7e73d59SPoul-Henning Kamp if (dsp->allocated == 0) 423c7e73d59SPoul-Henning Kamp break; 424c7e73d59SPoul-Henning Kamp dsp++; 425c7e73d59SPoul-Henning Kamp } 426c7e73d59SPoul-Henning Kamp spp->nfree--; 427c7e73d59SPoul-Henning Kamp dsp->allocated = 1; 428c7e73d59SPoul-Henning Kamp return (dsp); 429c7e73d59SPoul-Henning Kamp } 430c7e73d59SPoul-Henning Kamp 431c7e73d59SPoul-Henning Kamp static void 432c7e73d59SPoul-Henning Kamp devstat_free(struct devstat *dsp) 433c7e73d59SPoul-Henning Kamp { 434c7e73d59SPoul-Henning Kamp struct statspage *spp; 435c7e73d59SPoul-Henning Kamp 436c7e73d59SPoul-Henning Kamp bzero(dsp, sizeof *dsp); 437c7e73d59SPoul-Henning Kamp TAILQ_FOREACH(spp, &pagelist, list) { 438c7e73d59SPoul-Henning Kamp if (dsp >= spp->stat && dsp < (spp->stat + statsperpage)) { 439c7e73d59SPoul-Henning Kamp spp->nfree++; 440c7e73d59SPoul-Henning Kamp return; 441c7e73d59SPoul-Henning Kamp } 442c7e73d59SPoul-Henning Kamp } 443c7e73d59SPoul-Henning Kamp } 4447194d335SPoul-Henning Kamp 4457194d335SPoul-Henning Kamp SYSCTL_INT(_debug_sizeof, OID_AUTO, devstat, CTLFLAG_RD, 4467194d335SPoul-Henning Kamp 0, sizeof(struct devstat), "sizeof(struct devstat)"); 447