Index: sys/kern/subr_blist.c =================================================================== --- sys/kern/subr_blist.c +++ sys/kern/subr_blist.c @@ -90,6 +90,7 @@ #include #include #include +#include #include #include @@ -101,6 +102,7 @@ #include #include +#include #include #include #include @@ -144,6 +146,7 @@ #endif CTASSERT(BLIST_BMAP_RADIX % BLIST_META_RADIX == 0); +#define BLIST_META_MASK (BLIST_META_RADIX - 1) /* * For a subtree that can represent the state of up to 'radix' blocks, the @@ -166,7 +169,36 @@ { return (radix / - ((BLIST_BMAP_RADIX / BLIST_META_RADIX) * (BLIST_META_RADIX - 1))); + ((BLIST_BMAP_RADIX / BLIST_META_RADIX) * BLIST_META_MASK)); +} + +/* + * Use binary search to find the 1 bit in a u_daddr_t. + * Assumes that the argument has only one bit set. ffs(x) + * could be implemented as return (x==0? 0: 1+bitpos(x&-x)). + */ +static inline int +bitpos(u_daddr_t mask) +{ + int hi, lo, mid; + + switch (sizeof(u_daddr_t)) { +#ifdef HAVE_INLINE_FFSLL + case sizeof(long long): + return (ffsll(mask) - 1); +#endif + default: + lo = 0; + hi = BLIST_BMAP_RADIX; + while (lo + 1 < hi) { + mid = (lo + hi) >> 1; + if ((mask >> mid) != 0) + lo = mid; + else + hi = mid; + } + return (lo); + } } /* @@ -340,6 +372,173 @@ #endif +struct gap_stats { + daddr_t start; /* current gap start, or SWAPBLK_NONE */ + daddr_t num; /* number of gaps observed */ + daddr_t max; /* largest gap size */ + daddr_t avg; /* average gap size */ + daddr_t err; /* sum - num * avg */ + daddr_t histo[2 * BLIST_BMAP_RADIX - 1]; /* # gaps in each size range */ + int max_bucket; /* last histo elt with nonzero val */ +}; + +/* + * gap_stats_counting() - is the state 'counting 1 bits'? + * or 'skipping 0 bits'? + */ +static inline int +gap_stats_counting(struct gap_stats *stats) +{ + + return (stats->start != SWAPBLK_NONE); +} + +/* + * init_gap_stats() - initialize stats on gap sizes + */ +static inline void +init_gap_stats(struct gap_stats *stats) +{ + + bzero(stats, sizeof(*stats)); + stats->start = SWAPBLK_NONE; +} + +/* + * update_gap_stats() - update stats on gap sizes + */ +static void +update_gap_stats(struct gap_stats *stats, u_daddr_t posn) +{ + u_daddr_t size; + int id, logsize; + + if (!gap_stats_counting(stats)) { + stats->start = posn; + return; + } + size = posn - stats->start; + stats->start = SWAPBLK_NONE; + if (size > stats->max) + stats->max = size; + if (size == 1) { + id = 0; + } else { + logsize = 1; + while ((size >> logsize) > 1) + logsize++; + id = 2 * logsize - 1; + if (size >> (logsize - 1) >= 3) + id++; + } + stats->histo[id]++; + if (id > stats->max_bucket) + stats->max_bucket = id; + stats->err += (daddr_t)size - stats->avg; + stats->num++; + stats->avg += stats->err / stats->num; + stats->err %= stats->num; +} + +/* + * dump_stats() - print stats on gap sizes + */ +static inline void +dump_gap_stats(struct sbuf *s, struct gap_stats *stats) +{ + int i, val; + + sbuf_printf(s, "number of gaps: %jd\n", (intmax_t)stats->num); + sbuf_printf(s, "largest gap: %jd\n", (intmax_t)stats->max); + sbuf_printf(s, "average gap size: %jd\n", (intmax_t)stats->avg); + sbuf_printf(s, "histogram:\n"); + for (i = 0; i <= stats->max_bucket; i++) { + val = (i == 0) ? 1 : ((3 - i % 2) << ((i - 1) / 2)); + sbuf_printf(s, "%d\t%jd\n", val, (intmax_t)stats->histo[i]); + } +} + +/* + * blist_stats_create() - allocate stats struct + */ +struct gap_stats * +blist_stats_create(void) +{ + + return (malloc(sizeof(struct gap_stats), M_SWAP, M_WAITOK)); +} + +/* + * blist_stats_destroy() - free stats struct + */ +void +blist_stats_destroy(struct gap_stats *stats) +{ + + free(stats, M_SWAP); +} + +/* + * blist_stats_compute() - dump radix tree stats + */ +void +blist_stats_compute(struct sbuf *s, blist_t bl, struct gap_stats *stats) +{ + daddr_t bit, diff, i, nodes, radix; + u_daddr_t mask; + + init_gap_stats(stats); + nodes = 0; + i = bl->bl_radix; + while (i < bl->bl_radix + bl->bl_blocks) { + /* + * Find max size subtree starting at i + */ + radix = BLIST_BMAP_RADIX; + while (((i / radix) & BLIST_META_MASK) == 0) + radix *= BLIST_META_RADIX; + + /* + * Check for skippable subtrees starting at i + */ + while (radix > BLIST_BMAP_RADIX) { + if (bl->bl_root[nodes].u.bmu_avail == 0) { + if (gap_stats_counting(stats)) + update_gap_stats(stats, i); + break; + } + if (bl->bl_root[nodes].u.bmu_avail == radix) { + if (!gap_stats_counting(stats)) + update_gap_stats(stats, i); + break; + } + + /* + * Skip subtree root. + */ + nodes++; + radix /= BLIST_META_RADIX; + } + if (radix == BLIST_BMAP_RADIX) { + /* + * Scan leaf. + */ + mask = bl->bl_root[nodes].u.bmu_bitmap; + diff = mask ^ (mask << 1); + diff ^= gap_stats_counting(stats); + while (diff != 0) { + bit = diff & -diff; + update_gap_stats(stats, i + bitpos(bit)); + diff ^= bit; + } + } + nodes += radix_to_skip(radix); + i += radix; + } + update_gap_stats(stats, i); + dump_gap_stats(s, stats); +} + /************************************************************************ * ALLOCATION SUPPORT FUNCTIONS * ************************************************************************ @@ -355,13 +554,13 @@ * * This is the core of the allocator and is optimized for the * BLIST_BMAP_RADIX block allocation case. Otherwise, execution - * time is proportional to log2(count) + log2(BLIST_BMAP_RADIX). + * time is proportional to log2(count) + bitpos time. */ static daddr_t blst_leaf_alloc(blmeta_t *scan, daddr_t blk, int count, daddr_t cursor) { u_daddr_t mask; - int count1, hi, lo, mid, num_shifts, range1, range_ext; + int count1, lo, num_shifts, range1, range_ext; if (count == BLIST_BMAP_RADIX) { /* @@ -419,17 +618,10 @@ /* * The least significant set bit in mask marks the start of the first * available range of sufficient size. Clear all the bits but that one, - * and then perform a binary search to find its position. + * and then find its position. */ mask &= -mask; - hi = BLIST_BMAP_RADIX - count1; - while (lo + 1 < hi) { - mid = (lo + hi) >> 1; - if ((mask >> mid) != 0) - lo = mid; - else - hi = mid; - } + lo = bitpos(mask); /* * Set in mask exactly the bits being allocated, and clear them from @@ -980,6 +1172,8 @@ int size = 1024; int i; blist_t bl; + struct sbuf *s; + struct gap_stats *stats; for (i = 1; i < ac; ++i) { const char *ptr = av[i]; @@ -1014,6 +1208,15 @@ case 'p': blist_print(bl); break; + case 's': + s = sbuf_new_auto(); + stats = blist_stats_create(); + blist_stats_compute(s, bl, stats); + blist_stats_destroy(stats); + sbuf_finish(s); + printf("%s", sbuf_data(s)); + sbuf_delete(s); + break; case 'a': if (sscanf(buf + 1, "%lld", &count) == 1) { daddr_t blk = blist_alloc(bl, count); @@ -1041,6 +1244,7 @@ case 'h': puts( "p -print\n" + "s -stats\n" "a %d -allocate\n" "f %x %d -free\n" "l %x %d -fill\n" Index: sys/sys/blist.h =================================================================== --- sys/sys/blist.h +++ sys/sys/blist.h @@ -56,6 +56,8 @@ #ifndef _SYS_BLIST_H_ #define _SYS_BLIST_H_ +struct gap_stats; +struct sbuf; typedef uint64_t u_daddr_t; /* unsigned disk address */ /* @@ -98,6 +100,9 @@ void blist_free(blist_t blist, daddr_t blkno, daddr_t count); void blist_print(blist_t blist); void blist_resize(blist_t *pblist, daddr_t count, int freenew, int flags); +struct gap_stats *blist_stats_create(void); +void blist_stats_compute(struct sbuf *s, blist_t bl, struct gap_stats *); +void blist_stats_destroy(struct gap_stats *); #endif /* _SYS_BLIST_H_ */ Index: sys/vm/swap_pager.c =================================================================== --- sys/vm/swap_pager.c +++ sys/vm/swap_pager.c @@ -92,6 +92,7 @@ #include #include #include +#include #include #include #include @@ -323,6 +324,9 @@ SYSCTL_PROC(_vm, OID_AUTO, swap_async_max, CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_MPSAFE, NULL, 0, sysctl_swap_async_max, "I", "Maximum running async swap ops"); +static int sysctl_swap_pager_dumpstats(SYSCTL_HANDLER_ARGS); +SYSCTL_OID(_vm, OID_AUTO, swap_pager_dumpstats, CTLTYPE_STRING | CTLFLAG_RD, + NULL, 0, sysctl_swap_pager_dumpstats, "A", "Swap Pager Info"); static struct sx sw_alloc_sx; @@ -780,6 +784,33 @@ } /* + * SYSCTL_SWAP_PAGER_DUMPSTATS() - produce raw swap space stats + */ +static int +sysctl_swap_pager_dumpstats(SYSCTL_HANDLER_ARGS) +{ + struct sbuf sbuf; + struct gap_stats *stats; + struct swdevt *sp; + int error; + + error = sysctl_wire_old_buffer(req, 0); + if (error != 0) + return (error); + sbuf_new_for_sysctl(&sbuf, NULL, 128, req); + stats = blist_stats_create(); + + mtx_lock(&sw_dev_mtx); + TAILQ_FOREACH(sp, &swtailq, sw_list) + blist_stats_compute(&sbuf, sp->sw_blist, stats); + mtx_unlock(&sw_dev_mtx); + blist_stats_destroy(stats); + error = sbuf_finish(&sbuf); + sbuf_delete(&sbuf); + return (error); +} + +/* * SWAP_PAGER_FREESPACE() - frees swap blocks associated with a page * range within an object. *