106b4bf3eSWarner Losh /*-
251369649SPedro F. Giffuni * SPDX-License-Identifier: BSD-3-Clause
351369649SPedro F. Giffuni *
406b4bf3eSWarner Losh * Copyright (c) 1998 Matthew Dillon. All Rights Reserved.
506b4bf3eSWarner Losh * Redistribution and use in source and binary forms, with or without
606b4bf3eSWarner Losh * modification, are permitted provided that the following conditions
706b4bf3eSWarner Losh * are met:
806b4bf3eSWarner Losh * 1. Redistributions of source code must retain the above copyright
906b4bf3eSWarner Losh * notice, this list of conditions and the following disclaimer.
1006b4bf3eSWarner Losh * 2. Redistributions in binary form must reproduce the above copyright
1106b4bf3eSWarner Losh * notice, this list of conditions and the following disclaimer in the
1206b4bf3eSWarner Losh * documentation and/or other materials provided with the distribution.
1369a28758SEd Maste * 3. Neither the name of the University nor the names of its contributors
1406b4bf3eSWarner Losh * may be used to endorse or promote products derived from this software
1506b4bf3eSWarner Losh * without specific prior written permission.
1606b4bf3eSWarner Losh *
1706b4bf3eSWarner Losh * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
1806b4bf3eSWarner Losh * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
1906b4bf3eSWarner Losh * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2006b4bf3eSWarner Losh * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
2106b4bf3eSWarner Losh * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2206b4bf3eSWarner Losh * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
2306b4bf3eSWarner Losh * GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
2406b4bf3eSWarner Losh * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
2506b4bf3eSWarner Losh * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
2606b4bf3eSWarner Losh * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
2706b4bf3eSWarner Losh * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2806b4bf3eSWarner Losh */
297090df5aSMatthew Dillon /*
307090df5aSMatthew Dillon * BLIST.C - Bitmap allocator/deallocator, using a radix tree with hinting
317090df5aSMatthew Dillon *
327090df5aSMatthew Dillon * This module implements a general bitmap allocator/deallocator. The
337090df5aSMatthew Dillon * allocator eats around 2 bits per 'block'. The module does not
34f565f395SEitan Adler * try to interpret the meaning of a 'block' other than to return
357090df5aSMatthew Dillon * SWAPBLK_NONE on an allocation failure.
367090df5aSMatthew Dillon *
37ec371b57SAlan Cox * A radix tree controls access to pieces of the bitmap, and includes
38ec371b57SAlan Cox * auxiliary information at each interior node about the availabilty of
3900fd73d2SDoug Moore * contiguous free blocks in the subtree rooted at that node. A radix
4000fd73d2SDoug Moore * constant defines the size of the bitmaps contained in a leaf node
4100fd73d2SDoug Moore * and the number of descendents of each of the meta (interior) nodes.
4200fd73d2SDoug Moore * Each subtree is associated with a range of blocks. The root of any
4300fd73d2SDoug Moore * subtree stores a hint field that defines an upper bound on the size
4400fd73d2SDoug Moore * of the largest allocation that can begin in the associated block
4500fd73d2SDoug Moore * range. A hint is an upper bound on a potential allocation, but not
4600fd73d2SDoug Moore * necessarily a tight upper bound.
477090df5aSMatthew Dillon *
48bb4a27f9SMark Johnston * The bitmap field in each node directs the search for available blocks.
49bb4a27f9SMark Johnston * For a leaf node, a bit is set if the corresponding block is free. For a
50bb4a27f9SMark Johnston * meta node, a bit is set if the corresponding subtree contains a free
51bb4a27f9SMark Johnston * block somewhere within it. The search at a meta node considers only
52bb4a27f9SMark Johnston * children of that node that represent a range that includes a free block.
537090df5aSMatthew Dillon *
547090df5aSMatthew Dillon * The hinting greatly increases code efficiency for allocations while
557090df5aSMatthew Dillon * the general radix structure optimizes both allocations and frees. The
567090df5aSMatthew Dillon * radix tree should be able to operate well no matter how much
577090df5aSMatthew Dillon * fragmentation there is and no matter how large a bitmap is used.
587090df5aSMatthew Dillon *
5951dc4feaSSergey Kandaurov * The blist code wires all necessary memory at creation time. Neither
6051dc4feaSSergey Kandaurov * allocations nor frees require interaction with the memory subsystem.
61bb4a27f9SMark Johnston * The non-blocking nature of allocations and frees is required by swap
62bb4a27f9SMark Johnston * code (vm/swap_pager.c).
637090df5aSMatthew Dillon *
64bb4a27f9SMark Johnston * LAYOUT: The radix tree is laid out recursively using a linear array.
65bb4a27f9SMark Johnston * Each meta node is immediately followed (laid out sequentially in
6600fd73d2SDoug Moore * memory) by BLIST_RADIX lower-level nodes. This is a recursive
67bb4a27f9SMark Johnston * structure but one that can be easily scanned through a very simple
68bb4a27f9SMark Johnston * 'skip' calculation. The memory allocation is only large enough to
69bb4a27f9SMark Johnston * cover the number of blocks requested at creation time. Nodes that
70bb4a27f9SMark Johnston * represent blocks beyond that limit, nodes that would never be read
71bb4a27f9SMark Johnston * or written, are not allocated, so that the last of the
7200fd73d2SDoug Moore * BLIST_RADIX lower-level nodes of a some nodes may not be allocated.
737090df5aSMatthew Dillon *
74f565f395SEitan Adler * NOTE: the allocator cannot currently allocate more than
7500fd73d2SDoug Moore * BLIST_RADIX blocks per call. It will panic with 'allocation too
767090df5aSMatthew Dillon * large' if you try. This is an area that could use improvement. The
777090df5aSMatthew Dillon * radix is large enough that this restriction does not effect the swap
78b411efa4SAlan Cox * system, though. Currently only the allocation code is affected by
797090df5aSMatthew Dillon * this algorithmic unfeature. The freeing code can handle arbitrary
807090df5aSMatthew Dillon * ranges.
817090df5aSMatthew Dillon *
827090df5aSMatthew Dillon * This code can be compiled stand-alone for debugging.
837090df5aSMatthew Dillon */
847090df5aSMatthew Dillon
85677b542eSDavid E. O'Brien #include <sys/cdefs.h>
86c4473420SPeter Wemm #ifdef _KERNEL
877090df5aSMatthew Dillon
887090df5aSMatthew Dillon #include <sys/param.h>
897090df5aSMatthew Dillon #include <sys/systm.h>
907090df5aSMatthew Dillon #include <sys/lock.h>
917090df5aSMatthew Dillon #include <sys/kernel.h>
927090df5aSMatthew Dillon #include <sys/blist.h>
937090df5aSMatthew Dillon #include <sys/malloc.h>
94d027ed2eSAlan Cox #include <sys/sbuf.h>
950cddd8f0SMatthew Dillon #include <sys/proc.h>
9623955314SAlfred Perlstein #include <sys/mutex.h>
977090df5aSMatthew Dillon
987090df5aSMatthew Dillon #else
997090df5aSMatthew Dillon
1007090df5aSMatthew Dillon #ifndef BLIST_NO_DEBUG
1017090df5aSMatthew Dillon #define BLIST_DEBUG
1027090df5aSMatthew Dillon #endif
1037090df5aSMatthew Dillon
104bb4a27f9SMark Johnston #include <sys/errno.h>
1057090df5aSMatthew Dillon #include <sys/types.h>
106d90bf7d8SAlan Cox #include <sys/malloc.h>
107d027ed2eSAlan Cox #include <sys/sbuf.h>
108b1f59c92SDoug Moore #include <assert.h>
1097090df5aSMatthew Dillon #include <stdio.h>
1107090df5aSMatthew Dillon #include <string.h>
1118eefcd40SAlan Cox #include <stddef.h>
1127090df5aSMatthew Dillon #include <stdlib.h>
1137090df5aSMatthew Dillon #include <stdarg.h>
114d3783724SAlan Cox #include <stdbool.h>
1157090df5aSMatthew Dillon
1164be4fd5dSAlan Cox #define bitcount64(x) __bitcount64((uint64_t)(x))
11792da00bbSMatthew Dillon #define malloc(a,b,c) calloc(a, 1)
1187090df5aSMatthew Dillon #define free(a,b) free(a)
119bb4a27f9SMark Johnston #define ummin(a,b) ((a) < (b) ? (a) : (b))
12031c82722SDoug Moore #define imin(a,b) ((a) < (b) ? (a) : (b))
121b1f59c92SDoug Moore #define KASSERT(a,b) assert(a)
1227090df5aSMatthew Dillon
1237090df5aSMatthew Dillon #include <sys/blist.h>
1247090df5aSMatthew Dillon
1257090df5aSMatthew Dillon #endif
1267090df5aSMatthew Dillon
1277090df5aSMatthew Dillon /*
1287090df5aSMatthew Dillon * static support functions
1297090df5aSMatthew Dillon */
13087ae0686SDoug Moore static daddr_t blst_leaf_alloc(blmeta_t *scan, daddr_t blk,
13187ae0686SDoug Moore int *count, int maxcount);
13287ae0686SDoug Moore static daddr_t blst_meta_alloc(blmeta_t *scan, daddr_t cursor, int *count,
13387ae0686SDoug Moore int maxcount, u_daddr_t radix);
1347090df5aSMatthew Dillon static void blst_leaf_free(blmeta_t *scan, daddr_t relblk, int count);
1357090df5aSMatthew Dillon static void blst_meta_free(blmeta_t *scan, daddr_t freeBlk, daddr_t count,
136bee93d3cSAlan Cox u_daddr_t radix);
1377090df5aSMatthew Dillon static void blst_copy(blmeta_t *scan, daddr_t blk, daddr_t radix,
1382ac0c7c3SAlan Cox blist_t dest, daddr_t count);
139a7249a6cSAlan Cox static daddr_t blst_leaf_fill(blmeta_t *scan, daddr_t blk, int count);
140a7249a6cSAlan Cox static daddr_t blst_meta_fill(blmeta_t *scan, daddr_t allocBlk, daddr_t count,
141bee93d3cSAlan Cox u_daddr_t radix);
142c4473420SPeter Wemm #ifndef _KERNEL
143d3783724SAlan Cox static void blst_radix_print(blmeta_t *scan, daddr_t blk, daddr_t radix,
1442ac0c7c3SAlan Cox int tab);
1457090df5aSMatthew Dillon #endif
1467090df5aSMatthew Dillon
147c4473420SPeter Wemm #ifdef _KERNEL
1487090df5aSMatthew Dillon static MALLOC_DEFINE(M_SWAP, "SWAP", "Swap space");
1497090df5aSMatthew Dillon #endif
1507090df5aSMatthew Dillon
15100fd73d2SDoug Moore #define BLIST_MASK (BLIST_RADIX - 1)
152ba98e6a2SAlan Cox
1537090df5aSMatthew Dillon /*
1542ac0c7c3SAlan Cox * For a subtree that can represent the state of up to 'radix' blocks, the
15500fd73d2SDoug Moore * number of leaf nodes of the subtree is L=radix/BLIST_RADIX. If 'm'
15600fd73d2SDoug Moore * is short for BLIST_RADIX, then for a tree of height h with L=m**h
1572ac0c7c3SAlan Cox * leaf nodes, the total number of tree nodes is 1 + m + m**2 + ... + m**h,
1582ac0c7c3SAlan Cox * or, equivalently, (m**(h+1)-1)/(m-1). This quantity is called 'skip'
1592ac0c7c3SAlan Cox * in the 'meta' functions that process subtrees. Since integer division
1602ac0c7c3SAlan Cox * discards remainders, we can express this computation as
1612ac0c7c3SAlan Cox * skip = (m * m**h) / (m - 1)
16200fd73d2SDoug Moore * skip = (m * (radix / m)) / (m - 1)
16300fd73d2SDoug Moore * skip = radix / (m - 1)
164ba98e6a2SAlan Cox * so that simple integer division by a constant can safely be used for the
165ba98e6a2SAlan Cox * calculation.
1662ac0c7c3SAlan Cox */
1672ac0c7c3SAlan Cox static inline daddr_t
radix_to_skip(daddr_t radix)1682ac0c7c3SAlan Cox radix_to_skip(daddr_t radix)
1692ac0c7c3SAlan Cox {
1702ac0c7c3SAlan Cox
17100fd73d2SDoug Moore return (radix / BLIST_MASK);
172d027ed2eSAlan Cox }
173d027ed2eSAlan Cox
174d027ed2eSAlan Cox /*
175bb4a27f9SMark Johnston * Provide a mask with count bits set, starting as position n.
176bb4a27f9SMark Johnston */
177bb4a27f9SMark Johnston static inline u_daddr_t
bitrange(int n,int count)178bb4a27f9SMark Johnston bitrange(int n, int count)
179bb4a27f9SMark Johnston {
180bb4a27f9SMark Johnston
181bb4a27f9SMark Johnston return (((u_daddr_t)-1 << n) &
18200fd73d2SDoug Moore ((u_daddr_t)-1 >> (BLIST_RADIX - (n + count))));
183bb4a27f9SMark Johnston }
184bb4a27f9SMark Johnston
18553519253SDoug Moore static inline int
bitpos(u_daddr_t mask)1864ab18ea2SDoug Moore bitpos(u_daddr_t mask)
1874ab18ea2SDoug Moore {
1884ab18ea2SDoug Moore
189d4e236c7SDoug Moore _Static_assert(sizeof(long long) >= sizeof(mask),
190d4e236c7SDoug Moore "mask too big for ffsll()");
19112cd7dedSDoug Moore return (ffsll(mask) - 1);
1922ac0c7c3SAlan Cox }
1932ac0c7c3SAlan Cox
1942ac0c7c3SAlan Cox /*
1957090df5aSMatthew Dillon * blist_create() - create a blist capable of handling up to the specified
1967090df5aSMatthew Dillon * number of blocks
1977090df5aSMatthew Dillon *
198f565f395SEitan Adler * blocks - must be greater than 0
199c8c7ad92SKip Macy * flags - malloc flags
2007090df5aSMatthew Dillon *
2017090df5aSMatthew Dillon * The smallest blist consists of a single leaf node capable of
20200fd73d2SDoug Moore * managing BLIST_RADIX blocks.
2037090df5aSMatthew Dillon */
2047090df5aSMatthew Dillon blist_t
blist_create(daddr_t blocks,int flags)205c8c7ad92SKip Macy blist_create(daddr_t blocks, int flags)
2067090df5aSMatthew Dillon {
2077090df5aSMatthew Dillon blist_t bl;
208bb4a27f9SMark Johnston u_daddr_t nodes, radix;
2097090df5aSMatthew Dillon
210b1f59c92SDoug Moore KASSERT(blocks > 0, ("invalid block count"));
211ce9eea64SMark Johnston
2127090df5aSMatthew Dillon /*
213ce9eea64SMark Johnston * Calculate the radix and node count used for scanning.
2147090df5aSMatthew Dillon */
215bb4a27f9SMark Johnston nodes = 1;
2162783335cSMark Johnston for (radix = 1; (blocks - 1) / BLIST_RADIX / radix > 0;
2172783335cSMark Johnston radix *= BLIST_RADIX)
2182783335cSMark Johnston nodes += 1 + (blocks - 1) / BLIST_RADIX / radix;
2192783335cSMark Johnston
2202783335cSMark Johnston /*
2212783335cSMark Johnston * Include a sentinel node to ensure that cross-leaf scans stay within
2222783335cSMark Johnston * the bounds of the allocation.
2232783335cSMark Johnston */
2242783335cSMark Johnston if (blocks % BLIST_RADIX == 0)
2252783335cSMark Johnston nodes++;
2267090df5aSMatthew Dillon
22703ca2137SAlan Cox bl = malloc(offsetof(struct blist, bl_root[nodes]), M_SWAP, flags |
22803ca2137SAlan Cox M_ZERO);
229015d7db6SAlan Cox if (bl == NULL)
230015d7db6SAlan Cox return (NULL);
2317090df5aSMatthew Dillon
2327090df5aSMatthew Dillon bl->bl_blocks = blocks;
2337090df5aSMatthew Dillon bl->bl_radix = radix;
2347090df5aSMatthew Dillon
2357090df5aSMatthew Dillon #if defined(BLIST_DEBUG)
2367090df5aSMatthew Dillon printf(
23792da00bbSMatthew Dillon "BLIST representing %lld blocks (%lld MB of swap)"
23892da00bbSMatthew Dillon ", requiring %lldK of ram\n",
23992da00bbSMatthew Dillon (long long)bl->bl_blocks,
24092da00bbSMatthew Dillon (long long)bl->bl_blocks * 4 / 1024,
241015d7db6SAlan Cox (long long)(nodes * sizeof(blmeta_t) + 1023) / 1024
2427090df5aSMatthew Dillon );
24392da00bbSMatthew Dillon printf("BLIST raw radix tree contains %lld records\n",
244015d7db6SAlan Cox (long long)nodes);
2457090df5aSMatthew Dillon #endif
2467090df5aSMatthew Dillon
2477090df5aSMatthew Dillon return (bl);
2487090df5aSMatthew Dillon }
2497090df5aSMatthew Dillon
2507090df5aSMatthew Dillon void
blist_destroy(blist_t bl)2517090df5aSMatthew Dillon blist_destroy(blist_t bl)
2527090df5aSMatthew Dillon {
2538eefcd40SAlan Cox
2547090df5aSMatthew Dillon free(bl, M_SWAP);
2557090df5aSMatthew Dillon }
2567090df5aSMatthew Dillon
2577090df5aSMatthew Dillon /*
2587090df5aSMatthew Dillon * blist_alloc() - reserve space in the block bitmap. Return the base
2597090df5aSMatthew Dillon * of a contiguous region or SWAPBLK_NONE if space could
2607090df5aSMatthew Dillon * not be allocated.
2617090df5aSMatthew Dillon */
2627090df5aSMatthew Dillon daddr_t
blist_alloc(blist_t bl,int * count,int maxcount)26387ae0686SDoug Moore blist_alloc(blist_t bl, int *count, int maxcount)
2647090df5aSMatthew Dillon {
26527d172bbSDoug Moore daddr_t blk, cursor;
2667090df5aSMatthew Dillon
26787ae0686SDoug Moore KASSERT(*count <= maxcount,
26887ae0686SDoug Moore ("invalid parameters %d > %d", *count, maxcount));
26931c82722SDoug Moore KASSERT(*count <= BLIST_MAX_ALLOC,
27031c82722SDoug Moore ("minimum allocation too large: %d", *count));
271bb4a27f9SMark Johnston
272d4e3484bSAlan Cox /*
273d4e3484bSAlan Cox * This loop iterates at most twice. An allocation failure in the
274d4e3484bSAlan Cox * first iteration leads to a second iteration only if the cursor was
275d4e3484bSAlan Cox * non-zero. When the cursor is zero, an allocation failure will
276749cdf6fSAlan Cox * stop further iterations.
277d4e3484bSAlan Cox */
27887ae0686SDoug Moore for (cursor = bl->bl_cursor;; cursor = 0) {
27987ae0686SDoug Moore blk = blst_meta_alloc(bl->bl_root, cursor, count, maxcount,
280bee93d3cSAlan Cox bl->bl_radix);
281d4e3484bSAlan Cox if (blk != SWAPBLK_NONE) {
28287ae0686SDoug Moore bl->bl_avail -= *count;
28387ae0686SDoug Moore bl->bl_cursor = blk + *count;
284a0ae476bSAlan Cox if (bl->bl_cursor == bl->bl_blocks)
285a0ae476bSAlan Cox bl->bl_cursor = 0;
2867090df5aSMatthew Dillon return (blk);
28787ae0686SDoug Moore }
28887ae0686SDoug Moore if (cursor == 0)
289749cdf6fSAlan Cox return (SWAPBLK_NONE);
2907090df5aSMatthew Dillon }
2914be4fd5dSAlan Cox }
2924be4fd5dSAlan Cox
2934be4fd5dSAlan Cox /*
2944be4fd5dSAlan Cox * blist_avail() - return the number of free blocks.
2954be4fd5dSAlan Cox */
2964be4fd5dSAlan Cox daddr_t
blist_avail(blist_t bl)2974be4fd5dSAlan Cox blist_avail(blist_t bl)
2984be4fd5dSAlan Cox {
2994be4fd5dSAlan Cox
300bb4a27f9SMark Johnston return (bl->bl_avail);
3014be4fd5dSAlan Cox }
3027090df5aSMatthew Dillon
3037090df5aSMatthew Dillon /*
3047090df5aSMatthew Dillon * blist_free() - free up space in the block bitmap. Return the base
305b1f59c92SDoug Moore * of a contiguous region.
3067090df5aSMatthew Dillon */
3077090df5aSMatthew Dillon void
blist_free(blist_t bl,daddr_t blkno,daddr_t count)3087090df5aSMatthew Dillon blist_free(blist_t bl, daddr_t blkno, daddr_t count)
3097090df5aSMatthew Dillon {
310a396c83aSAlan Cox
311b1f59c92SDoug Moore KASSERT(blkno >= 0 && blkno + count <= bl->bl_blocks,
312b1f59c92SDoug Moore ("freeing invalid range: blkno %jx, count %d, blocks %jd",
313b1f59c92SDoug Moore (uintmax_t)blkno, (int)count, (uintmax_t)bl->bl_blocks));
314bee93d3cSAlan Cox blst_meta_free(bl->bl_root, blkno, count, bl->bl_radix);
315bb4a27f9SMark Johnston bl->bl_avail += count;
3167090df5aSMatthew Dillon }
3177090df5aSMatthew Dillon
3187090df5aSMatthew Dillon /*
31992da00bbSMatthew Dillon * blist_fill() - mark a region in the block bitmap as off-limits
32092da00bbSMatthew Dillon * to the allocator (i.e. allocate it), ignoring any
32192da00bbSMatthew Dillon * existing allocations. Return the number of blocks
32292da00bbSMatthew Dillon * actually filled that were free before the call.
32392da00bbSMatthew Dillon */
324a7249a6cSAlan Cox daddr_t
blist_fill(blist_t bl,daddr_t blkno,daddr_t count)32592da00bbSMatthew Dillon blist_fill(blist_t bl, daddr_t blkno, daddr_t count)
32692da00bbSMatthew Dillon {
327bb4a27f9SMark Johnston daddr_t filled;
32892da00bbSMatthew Dillon
329b1f59c92SDoug Moore KASSERT(blkno >= 0 && blkno + count <= bl->bl_blocks,
330b1f59c92SDoug Moore ("filling invalid range: blkno %jx, count %d, blocks %jd",
331b1f59c92SDoug Moore (uintmax_t)blkno, (int)count, (uintmax_t)bl->bl_blocks));
332bb4a27f9SMark Johnston filled = blst_meta_fill(bl->bl_root, blkno, count, bl->bl_radix);
333bb4a27f9SMark Johnston bl->bl_avail -= filled;
334bb4a27f9SMark Johnston return (filled);
33592da00bbSMatthew Dillon }
33692da00bbSMatthew Dillon
33792da00bbSMatthew Dillon /*
3387090df5aSMatthew Dillon * blist_resize() - resize an existing radix tree to handle the
3397090df5aSMatthew Dillon * specified number of blocks. This will reallocate
3407090df5aSMatthew Dillon * the tree and transfer the previous bitmap to the new
3417090df5aSMatthew Dillon * one. When extending the tree you can specify whether
3427090df5aSMatthew Dillon * the new blocks are to left allocated or freed.
3437090df5aSMatthew Dillon */
3447090df5aSMatthew Dillon void
blist_resize(blist_t * pbl,daddr_t count,int freenew,int flags)345c8c7ad92SKip Macy blist_resize(blist_t *pbl, daddr_t count, int freenew, int flags)
3467090df5aSMatthew Dillon {
347c8c7ad92SKip Macy blist_t newbl = blist_create(count, flags);
3487090df5aSMatthew Dillon blist_t save = *pbl;
3497090df5aSMatthew Dillon
3507090df5aSMatthew Dillon *pbl = newbl;
3517090df5aSMatthew Dillon if (count > save->bl_blocks)
3527090df5aSMatthew Dillon count = save->bl_blocks;
3532ac0c7c3SAlan Cox blst_copy(save->bl_root, 0, save->bl_radix, newbl, count);
3547090df5aSMatthew Dillon
3557090df5aSMatthew Dillon /*
3567090df5aSMatthew Dillon * If resizing upwards, should we free the new space or not?
3577090df5aSMatthew Dillon */
3587090df5aSMatthew Dillon if (freenew && count < newbl->bl_blocks) {
3597090df5aSMatthew Dillon blist_free(newbl, count, newbl->bl_blocks - count);
3607090df5aSMatthew Dillon }
3617090df5aSMatthew Dillon blist_destroy(save);
3627090df5aSMatthew Dillon }
3637090df5aSMatthew Dillon
3647090df5aSMatthew Dillon #ifdef BLIST_DEBUG
3657090df5aSMatthew Dillon
3667090df5aSMatthew Dillon /*
3677090df5aSMatthew Dillon * blist_print() - dump radix tree
3687090df5aSMatthew Dillon */
3697090df5aSMatthew Dillon void
blist_print(blist_t bl)3707090df5aSMatthew Dillon blist_print(blist_t bl)
3717090df5aSMatthew Dillon {
372bb4a27f9SMark Johnston printf("BLIST avail = %jd, cursor = %08jx {\n",
373bb4a27f9SMark Johnston (uintmax_t)bl->bl_avail, (uintmax_t)bl->bl_cursor);
374bb4a27f9SMark Johnston
375bb4a27f9SMark Johnston if (bl->bl_root->bm_bitmap != 0)
3762ac0c7c3SAlan Cox blst_radix_print(bl->bl_root, 0, bl->bl_radix, 4);
3777090df5aSMatthew Dillon printf("}\n");
3787090df5aSMatthew Dillon }
3797090df5aSMatthew Dillon
3807090df5aSMatthew Dillon #endif
3817090df5aSMatthew Dillon
382d027ed2eSAlan Cox static const u_daddr_t fib[] = {
383d027ed2eSAlan Cox 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233, 377, 610, 987, 1597, 2584,
384d027ed2eSAlan Cox 4181, 6765, 10946, 17711, 28657, 46368, 75025, 121393, 196418, 317811,
385d027ed2eSAlan Cox 514229, 832040, 1346269, 2178309, 3524578,
386d027ed2eSAlan Cox };
387d027ed2eSAlan Cox
388d027ed2eSAlan Cox /*
389d027ed2eSAlan Cox * Use 'gap' to describe a maximal range of unallocated blocks/bits.
390d027ed2eSAlan Cox */
391d027ed2eSAlan Cox struct gap_stats {
392d027ed2eSAlan Cox daddr_t start; /* current gap start, or SWAPBLK_NONE */
393d027ed2eSAlan Cox daddr_t num; /* number of gaps observed */
394d027ed2eSAlan Cox daddr_t max; /* largest gap size */
395d027ed2eSAlan Cox daddr_t avg; /* average gap size */
396d027ed2eSAlan Cox daddr_t err; /* sum - num * avg */
397d027ed2eSAlan Cox daddr_t histo[nitems(fib)]; /* # gaps in each size range */
398d027ed2eSAlan Cox int max_bucket; /* last histo elt with nonzero val */
399d027ed2eSAlan Cox };
400d027ed2eSAlan Cox
401d027ed2eSAlan Cox /*
402d027ed2eSAlan Cox * gap_stats_counting() - is the state 'counting 1 bits'?
403d027ed2eSAlan Cox * or 'skipping 0 bits'?
404d027ed2eSAlan Cox */
405d027ed2eSAlan Cox static inline bool
gap_stats_counting(const struct gap_stats * stats)406d027ed2eSAlan Cox gap_stats_counting(const struct gap_stats *stats)
407d027ed2eSAlan Cox {
408d027ed2eSAlan Cox
409d027ed2eSAlan Cox return (stats->start != SWAPBLK_NONE);
410d027ed2eSAlan Cox }
411d027ed2eSAlan Cox
412d027ed2eSAlan Cox /*
413d027ed2eSAlan Cox * init_gap_stats() - initialize stats on gap sizes
414d027ed2eSAlan Cox */
415d027ed2eSAlan Cox static inline void
init_gap_stats(struct gap_stats * stats)416d027ed2eSAlan Cox init_gap_stats(struct gap_stats *stats)
417d027ed2eSAlan Cox {
418d027ed2eSAlan Cox
419d027ed2eSAlan Cox bzero(stats, sizeof(*stats));
420d027ed2eSAlan Cox stats->start = SWAPBLK_NONE;
421d027ed2eSAlan Cox }
422d027ed2eSAlan Cox
423d027ed2eSAlan Cox /*
424d027ed2eSAlan Cox * update_gap_stats() - update stats on gap sizes
425d027ed2eSAlan Cox */
426d027ed2eSAlan Cox static void
update_gap_stats(struct gap_stats * stats,daddr_t posn)427d027ed2eSAlan Cox update_gap_stats(struct gap_stats *stats, daddr_t posn)
428d027ed2eSAlan Cox {
429d027ed2eSAlan Cox daddr_t size;
430d027ed2eSAlan Cox int hi, lo, mid;
431d027ed2eSAlan Cox
432d027ed2eSAlan Cox if (!gap_stats_counting(stats)) {
433d027ed2eSAlan Cox stats->start = posn;
434d027ed2eSAlan Cox return;
435d027ed2eSAlan Cox }
436d027ed2eSAlan Cox size = posn - stats->start;
437d027ed2eSAlan Cox stats->start = SWAPBLK_NONE;
438d027ed2eSAlan Cox if (size > stats->max)
439d027ed2eSAlan Cox stats->max = size;
440d027ed2eSAlan Cox
441d027ed2eSAlan Cox /*
442d027ed2eSAlan Cox * Find the fibonacci range that contains size,
443d027ed2eSAlan Cox * expecting to find it in an early range.
444d027ed2eSAlan Cox */
445d027ed2eSAlan Cox lo = 0;
446d027ed2eSAlan Cox hi = 1;
447d027ed2eSAlan Cox while (hi < nitems(fib) && fib[hi] <= size) {
448d027ed2eSAlan Cox lo = hi;
449d027ed2eSAlan Cox hi *= 2;
450d027ed2eSAlan Cox }
451d027ed2eSAlan Cox if (hi >= nitems(fib))
452d027ed2eSAlan Cox hi = nitems(fib);
453d027ed2eSAlan Cox while (lo + 1 != hi) {
454d027ed2eSAlan Cox mid = (lo + hi) >> 1;
455d027ed2eSAlan Cox if (fib[mid] <= size)
456d027ed2eSAlan Cox lo = mid;
457d027ed2eSAlan Cox else
458d027ed2eSAlan Cox hi = mid;
459d027ed2eSAlan Cox }
460d027ed2eSAlan Cox stats->histo[lo]++;
461d027ed2eSAlan Cox if (lo > stats->max_bucket)
462d027ed2eSAlan Cox stats->max_bucket = lo;
463d027ed2eSAlan Cox stats->err += size - stats->avg;
464d027ed2eSAlan Cox stats->num++;
465d027ed2eSAlan Cox stats->avg += stats->err / stats->num;
466d027ed2eSAlan Cox stats->err %= stats->num;
467d027ed2eSAlan Cox }
468d027ed2eSAlan Cox
469d027ed2eSAlan Cox /*
470d027ed2eSAlan Cox * dump_gap_stats() - print stats on gap sizes
471d027ed2eSAlan Cox */
472d027ed2eSAlan Cox static inline void
dump_gap_stats(const struct gap_stats * stats,struct sbuf * s)473d027ed2eSAlan Cox dump_gap_stats(const struct gap_stats *stats, struct sbuf *s)
474d027ed2eSAlan Cox {
475d027ed2eSAlan Cox int i;
476d027ed2eSAlan Cox
477d027ed2eSAlan Cox sbuf_printf(s, "number of maximal free ranges: %jd\n",
478d027ed2eSAlan Cox (intmax_t)stats->num);
479d027ed2eSAlan Cox sbuf_printf(s, "largest free range: %jd\n", (intmax_t)stats->max);
480d027ed2eSAlan Cox sbuf_printf(s, "average maximal free range size: %jd\n",
481d027ed2eSAlan Cox (intmax_t)stats->avg);
482*0a713948SAlexander Motin sbuf_cat(s, "number of maximal free ranges of different sizes:\n");
483*0a713948SAlexander Motin sbuf_cat(s, " count | size range\n");
484*0a713948SAlexander Motin sbuf_cat(s, " ----- | ----------\n");
485d027ed2eSAlan Cox for (i = 0; i < stats->max_bucket; i++) {
486d027ed2eSAlan Cox if (stats->histo[i] != 0) {
487d027ed2eSAlan Cox sbuf_printf(s, "%20jd | ",
488d027ed2eSAlan Cox (intmax_t)stats->histo[i]);
489d027ed2eSAlan Cox if (fib[i] != fib[i + 1] - 1)
490d027ed2eSAlan Cox sbuf_printf(s, "%jd to %jd\n", (intmax_t)fib[i],
491d027ed2eSAlan Cox (intmax_t)fib[i + 1] - 1);
492d027ed2eSAlan Cox else
493d027ed2eSAlan Cox sbuf_printf(s, "%jd\n", (intmax_t)fib[i]);
494d027ed2eSAlan Cox }
495d027ed2eSAlan Cox }
496d027ed2eSAlan Cox sbuf_printf(s, "%20jd | ", (intmax_t)stats->histo[i]);
497d027ed2eSAlan Cox if (stats->histo[i] > 1)
498d027ed2eSAlan Cox sbuf_printf(s, "%jd to %jd\n", (intmax_t)fib[i],
499d027ed2eSAlan Cox (intmax_t)stats->max);
500d027ed2eSAlan Cox else
501d027ed2eSAlan Cox sbuf_printf(s, "%jd\n", (intmax_t)stats->max);
502d027ed2eSAlan Cox }
503d027ed2eSAlan Cox
504d027ed2eSAlan Cox /*
505d027ed2eSAlan Cox * blist_stats() - dump radix tree stats
506d027ed2eSAlan Cox */
507d027ed2eSAlan Cox void
blist_stats(blist_t bl,struct sbuf * s)508d027ed2eSAlan Cox blist_stats(blist_t bl, struct sbuf *s)
509d027ed2eSAlan Cox {
510d027ed2eSAlan Cox struct gap_stats gstats;
511d027ed2eSAlan Cox struct gap_stats *stats = &gstats;
512d027ed2eSAlan Cox daddr_t i, nodes, radix;
51353519253SDoug Moore u_daddr_t diff, mask;
51453519253SDoug Moore int digit;
515d027ed2eSAlan Cox
516d027ed2eSAlan Cox init_gap_stats(stats);
517d027ed2eSAlan Cox nodes = 0;
51800fd73d2SDoug Moore radix = bl->bl_radix;
51900fd73d2SDoug Moore for (i = 0; i < bl->bl_blocks; ) {
520d027ed2eSAlan Cox /*
521d027ed2eSAlan Cox * Check for skippable subtrees starting at i.
522d027ed2eSAlan Cox */
52300fd73d2SDoug Moore while (radix != 1) {
524bb4a27f9SMark Johnston if (bl->bl_root[nodes].bm_bitmap == 0) {
525d027ed2eSAlan Cox if (gap_stats_counting(stats))
526d027ed2eSAlan Cox update_gap_stats(stats, i);
527d027ed2eSAlan Cox break;
528d027ed2eSAlan Cox }
529d027ed2eSAlan Cox
530d027ed2eSAlan Cox /*
531d027ed2eSAlan Cox * Skip subtree root.
532d027ed2eSAlan Cox */
533d027ed2eSAlan Cox nodes++;
53400fd73d2SDoug Moore radix /= BLIST_RADIX;
535d027ed2eSAlan Cox }
53600fd73d2SDoug Moore if (radix == 1) {
537d027ed2eSAlan Cox /*
538d027ed2eSAlan Cox * Scan leaf.
539d027ed2eSAlan Cox */
540bb4a27f9SMark Johnston mask = bl->bl_root[nodes].bm_bitmap;
541d027ed2eSAlan Cox diff = mask ^ (mask << 1);
542d027ed2eSAlan Cox if (gap_stats_counting(stats))
543d027ed2eSAlan Cox diff ^= 1;
544d027ed2eSAlan Cox while (diff != 0) {
54553519253SDoug Moore digit = bitpos(diff);
54653519253SDoug Moore update_gap_stats(stats, i + digit);
54753519253SDoug Moore diff ^= bitrange(digit, 1);
548d027ed2eSAlan Cox }
549d027ed2eSAlan Cox }
55000fd73d2SDoug Moore nodes += radix_to_skip(radix * BLIST_RADIX);
55100fd73d2SDoug Moore i += radix * BLIST_RADIX;
55200fd73d2SDoug Moore
55300fd73d2SDoug Moore /*
55400fd73d2SDoug Moore * Find max size subtree starting at i.
55500fd73d2SDoug Moore */
55600fd73d2SDoug Moore for (radix = 1;
55700fd73d2SDoug Moore ((i / BLIST_RADIX / radix) & BLIST_MASK) == 0;
55800fd73d2SDoug Moore radix *= BLIST_RADIX)
55900fd73d2SDoug Moore ;
560d027ed2eSAlan Cox }
561d027ed2eSAlan Cox update_gap_stats(stats, i);
562d027ed2eSAlan Cox dump_gap_stats(stats, s);
563d027ed2eSAlan Cox }
564d027ed2eSAlan Cox
5657090df5aSMatthew Dillon /************************************************************************
5667090df5aSMatthew Dillon * ALLOCATION SUPPORT FUNCTIONS *
5677090df5aSMatthew Dillon ************************************************************************
5687090df5aSMatthew Dillon *
5697090df5aSMatthew Dillon * These support functions do all the actual work. They may seem
5707090df5aSMatthew Dillon * rather longish, but that's because I've commented them up. The
5717090df5aSMatthew Dillon * actual code is straight forward.
5727090df5aSMatthew Dillon *
5737090df5aSMatthew Dillon */
5747090df5aSMatthew Dillon
5757090df5aSMatthew Dillon /*
57631c82722SDoug Moore * BLST_NEXT_LEAF_ALLOC() - allocate the blocks starting with the next leaf.
577bb4a27f9SMark Johnston *
57831c82722SDoug Moore * 'scan' is a leaf node, and its first block is at address 'start'. The
57931c82722SDoug Moore * next leaf node could be adjacent, or several nodes away if the least
58031c82722SDoug Moore * common ancestor of 'scan' and its neighbor is several levels up. Use
58131c82722SDoug Moore * addresses to determine how many meta-nodes lie between the leaves. If
58231c82722SDoug Moore * sequence of leaves starting with the next one has enough initial bits
58331c82722SDoug Moore * set, clear them and clear the bits in the meta nodes on the path up to
58431c82722SDoug Moore * the least common ancestor to mark any subtrees made completely empty.
585bb4a27f9SMark Johnston */
586bb4a27f9SMark Johnston static int
blst_next_leaf_alloc(blmeta_t * scan,daddr_t start,int count,int maxcount)58731c82722SDoug Moore blst_next_leaf_alloc(blmeta_t *scan, daddr_t start, int count, int maxcount)
588bb4a27f9SMark Johnston {
589bb4a27f9SMark Johnston u_daddr_t radix;
59031c82722SDoug Moore daddr_t blk;
59187ae0686SDoug Moore int avail, digit;
592bb4a27f9SMark Johnston
59300fd73d2SDoug Moore start += BLIST_RADIX;
59400fd73d2SDoug Moore for (blk = start; blk - start < maxcount; blk += BLIST_RADIX) {
59531c82722SDoug Moore /* Skip meta-nodes, as long as they promise more free blocks. */
59600fd73d2SDoug Moore radix = BLIST_RADIX;
59731c82722SDoug Moore while (((++scan)->bm_bitmap & 1) == 1 &&
59800fd73d2SDoug Moore ((blk / radix) & BLIST_MASK) == 0)
59900fd73d2SDoug Moore radix *= BLIST_RADIX;
60031c82722SDoug Moore if (~scan->bm_bitmap != 0) {
60131c82722SDoug Moore /*
60231c82722SDoug Moore * Either there is no next leaf with any free blocks,
60331c82722SDoug Moore * or we've reached the next leaf and found that some
60431c82722SDoug Moore * of its blocks are not free. In the first case,
60531c82722SDoug Moore * bitpos() returns zero here.
60631c82722SDoug Moore */
60731c82722SDoug Moore avail = blk - start + bitpos(~scan->bm_bitmap);
6083f3f7c05SDoug Moore if (avail < count || avail == 0) {
609bb4a27f9SMark Johnston /*
61031c82722SDoug Moore * There isn't a next leaf with enough free
6113f3f7c05SDoug Moore * blocks at its beginning to bother
6123f3f7c05SDoug Moore * allocating.
613bb4a27f9SMark Johnston */
61431c82722SDoug Moore return (avail);
615bb4a27f9SMark Johnston }
61631c82722SDoug Moore maxcount = imin(avail, maxcount);
61700fd73d2SDoug Moore if (maxcount % BLIST_RADIX == 0) {
6183f3f7c05SDoug Moore /*
6193f3f7c05SDoug Moore * There was no next leaf. Back scan up to
6203f3f7c05SDoug Moore * last leaf.
6213f3f7c05SDoug Moore */
62200fd73d2SDoug Moore do {
62300fd73d2SDoug Moore radix /= BLIST_RADIX;
6243f3f7c05SDoug Moore --scan;
62500fd73d2SDoug Moore } while (radix != 1);
62600fd73d2SDoug Moore blk -= BLIST_RADIX;
6273f3f7c05SDoug Moore }
62831c82722SDoug Moore }
62931c82722SDoug Moore }
630bb4a27f9SMark Johnston
631bb4a27f9SMark Johnston /*
63231c82722SDoug Moore * 'scan' is the last leaf that provides blocks. Clear from 1 to
63300fd73d2SDoug Moore * BLIST_RADIX bits to represent the allocation of those last blocks.
634bb4a27f9SMark Johnston */
63500fd73d2SDoug Moore if (maxcount % BLIST_RADIX != 0)
63600fd73d2SDoug Moore scan->bm_bitmap &= ~bitrange(0, maxcount % BLIST_RADIX);
63731c82722SDoug Moore else
63831c82722SDoug Moore scan->bm_bitmap = 0;
63931c82722SDoug Moore
64031c82722SDoug Moore for (;;) {
64131c82722SDoug Moore /* Back up over meta-nodes, clearing bits if necessary. */
64200fd73d2SDoug Moore blk -= BLIST_RADIX;
64300fd73d2SDoug Moore for (radix = BLIST_RADIX;
64400fd73d2SDoug Moore (digit = ((blk / radix) & BLIST_MASK)) == 0;
64500fd73d2SDoug Moore radix *= BLIST_RADIX) {
64631c82722SDoug Moore if ((scan--)->bm_bitmap == 0)
64731c82722SDoug Moore scan->bm_bitmap ^= 1;
64831c82722SDoug Moore }
64931c82722SDoug Moore if ((scan--)->bm_bitmap == 0)
650d1c34a6bSDoug Moore scan[-digit * radix_to_skip(radix)].bm_bitmap ^=
651d1c34a6bSDoug Moore (u_daddr_t)1 << digit;
65231c82722SDoug Moore
65331c82722SDoug Moore if (blk == start)
654d1c34a6bSDoug Moore break;
65531c82722SDoug Moore /* Clear all the bits of this leaf. */
65631c82722SDoug Moore scan->bm_bitmap = 0;
657bb4a27f9SMark Johnston }
65831c82722SDoug Moore return (maxcount);
659bb4a27f9SMark Johnston }
660bb4a27f9SMark Johnston
661bb4a27f9SMark Johnston /*
662bb4a27f9SMark Johnston * BLST_LEAF_ALLOC() - allocate at a leaf in the radix tree (a bitmap).
6637090df5aSMatthew Dillon *
6642905d1ceSAlan Cox * This function is the core of the allocator. Its execution time is
6652905d1ceSAlan Cox * proportional to log(count), plus height of the tree if the allocation
6662905d1ceSAlan Cox * crosses a leaf boundary.
6677090df5aSMatthew Dillon */
6687090df5aSMatthew Dillon static daddr_t
blst_leaf_alloc(blmeta_t * scan,daddr_t blk,int * count,int maxcount)66987ae0686SDoug Moore blst_leaf_alloc(blmeta_t *scan, daddr_t blk, int *count, int maxcount)
67054541421SAlan Cox {
6719f70442aSDoug Moore u_daddr_t mask;
6729f70442aSDoug Moore int bighint, count1, hi, lo, num_shifts;
6737090df5aSMatthew Dillon
67487ae0686SDoug Moore count1 = *count - 1;
67554541421SAlan Cox num_shifts = fls(count1);
6769f70442aSDoug Moore mask = ~scan->bm_bitmap;
6779f70442aSDoug Moore while ((mask & (mask + 1)) != 0 && num_shifts > 0) {
67854541421SAlan Cox /*
6799f70442aSDoug Moore * If bit i is 0 in mask, then bits in [i, i + (count1 >>
6809f70442aSDoug Moore * num_shifts)] are 1 in scan->bm_bitmap. Reduce num_shifts to
6819f70442aSDoug Moore * 0, while preserving this invariant. The updates to mask
6829f70442aSDoug Moore * leave fewer bits 0, but each bit that remains 0 represents a
6839f70442aSDoug Moore * longer string of consecutive 1-bits in scan->bm_bitmap. If
6849f70442aSDoug Moore * more updates to mask cannot set more bits, because mask is
6859f70442aSDoug Moore * partitioned with all 1 bits following all 0 bits, the loop
6869f70442aSDoug Moore * terminates immediately.
68754541421SAlan Cox */
68854541421SAlan Cox num_shifts--;
6899f70442aSDoug Moore mask |= mask >> ((count1 >> num_shifts) + 1) / 2;
69054541421SAlan Cox }
6919f70442aSDoug Moore bighint = count1 >> num_shifts;
6929f70442aSDoug Moore if (~mask == 0) {
69354541421SAlan Cox /*
6949f70442aSDoug Moore * Update bighint. There is no allocation bigger than
6959f70442aSDoug Moore * count1 >> num_shifts starting in this leaf.
69654541421SAlan Cox */
6979f70442aSDoug Moore scan->bm_bighint = bighint;
6987090df5aSMatthew Dillon return (SWAPBLK_NONE);
6997090df5aSMatthew Dillon }
70054541421SAlan Cox
701ec371b57SAlan Cox /* Discard any candidates that appear before blk. */
70200fd73d2SDoug Moore if ((blk & BLIST_MASK) != 0) {
70300fd73d2SDoug Moore if ((~mask & bitrange(0, blk & BLIST_MASK)) != 0) {
7049f70442aSDoug Moore /* Grow bighint in case all discarded bits are set. */
70500fd73d2SDoug Moore bighint += blk & BLIST_MASK;
70600fd73d2SDoug Moore mask |= bitrange(0, blk & BLIST_MASK);
7079f70442aSDoug Moore if (~mask == 0) {
7089f70442aSDoug Moore scan->bm_bighint = bighint;
70954541421SAlan Cox return (SWAPBLK_NONE);
7102905d1ceSAlan Cox }
7119f70442aSDoug Moore }
71200fd73d2SDoug Moore blk -= blk & BLIST_MASK;
7132905d1ceSAlan Cox }
7142905d1ceSAlan Cox
7152905d1ceSAlan Cox /*
71654541421SAlan Cox * The least significant set bit in mask marks the start of the first
71787ae0686SDoug Moore * available range of sufficient size. Find its position.
7187090df5aSMatthew Dillon */
7199f70442aSDoug Moore lo = bitpos(~mask);
7207090df5aSMatthew Dillon
72154541421SAlan Cox /*
72287ae0686SDoug Moore * Find how much space is available starting at that position.
72354541421SAlan Cox */
7249f70442aSDoug Moore if ((mask & (mask + 1)) != 0) {
72587ae0686SDoug Moore /* Count the 1 bits starting at position lo. */
7269f70442aSDoug Moore hi = bitpos(mask & (mask + 1)) + count1;
72787ae0686SDoug Moore if (maxcount < hi - lo)
72887ae0686SDoug Moore hi = lo + maxcount;
72987ae0686SDoug Moore *count = hi - lo;
7309f70442aSDoug Moore mask = ~bitrange(lo, *count);
73100fd73d2SDoug Moore } else if (maxcount <= BLIST_RADIX - lo) {
73287ae0686SDoug Moore /* All the blocks we can use are available here. */
73387ae0686SDoug Moore hi = lo + maxcount;
73487ae0686SDoug Moore *count = maxcount;
7359f70442aSDoug Moore mask = ~bitrange(lo, *count);
73600fd73d2SDoug Moore if (hi == BLIST_RADIX)
7379f70442aSDoug Moore scan->bm_bighint = bighint;
73887ae0686SDoug Moore } else {
73987ae0686SDoug Moore /* Check next leaf for some of the blocks we want or need. */
74000fd73d2SDoug Moore count1 = *count - (BLIST_RADIX - lo);
74100fd73d2SDoug Moore maxcount -= BLIST_RADIX - lo;
74287ae0686SDoug Moore hi = blst_next_leaf_alloc(scan, blk, count1, maxcount);
74387ae0686SDoug Moore if (hi < count1)
744ec371b57SAlan Cox /*
74587ae0686SDoug Moore * The next leaf cannot supply enough blocks to reach
74687ae0686SDoug Moore * the minimum required allocation. The hint cannot be
74787ae0686SDoug Moore * updated, because the same allocation request could
74887ae0686SDoug Moore * be satisfied later, by this leaf, if the state of
74987ae0686SDoug Moore * the next leaf changes, and without any changes to
75087ae0686SDoug Moore * this leaf.
751ec371b57SAlan Cox */
752ec371b57SAlan Cox return (SWAPBLK_NONE);
75300fd73d2SDoug Moore *count = BLIST_RADIX - lo + hi;
7549f70442aSDoug Moore scan->bm_bighint = bighint;
755ec371b57SAlan Cox }
756ec371b57SAlan Cox
757ec371b57SAlan Cox /* Clear the allocated bits from this leaf. */
7589f70442aSDoug Moore scan->bm_bitmap &= mask;
7592905d1ceSAlan Cox return (blk + lo);
7607090df5aSMatthew Dillon }
7617090df5aSMatthew Dillon
7627090df5aSMatthew Dillon /*
7637090df5aSMatthew Dillon * blist_meta_alloc() - allocate at a meta in the radix tree.
7647090df5aSMatthew Dillon *
7657090df5aSMatthew Dillon * Attempt to allocate at a meta node. If we can't, we update
7667090df5aSMatthew Dillon * bighint and return a failure. Updating bighint optimize future
7677090df5aSMatthew Dillon * calls that hit this node. We have to check for our collapse cases
7687090df5aSMatthew Dillon * and we have a few optimizations strewn in as well.
7697090df5aSMatthew Dillon */
7707090df5aSMatthew Dillon static daddr_t
blst_meta_alloc(blmeta_t * scan,daddr_t cursor,int * count,int maxcount,u_daddr_t radix)77187ae0686SDoug Moore blst_meta_alloc(blmeta_t *scan, daddr_t cursor, int *count,
77287ae0686SDoug Moore int maxcount, u_daddr_t radix)
773d4e3484bSAlan Cox {
774bb4a27f9SMark Johnston daddr_t blk, i, r, skip;
77553519253SDoug Moore u_daddr_t mask;
776d4e3484bSAlan Cox bool scan_from_start;
777bb4a27f9SMark Johnston int digit;
7787090df5aSMatthew Dillon
77900fd73d2SDoug Moore if (radix == 1)
78087ae0686SDoug Moore return (blst_leaf_alloc(scan, cursor, count, maxcount));
78100fd73d2SDoug Moore blk = cursor & -(radix * BLIST_RADIX);
782bb4a27f9SMark Johnston scan_from_start = (cursor == blk);
7832ac0c7c3SAlan Cox skip = radix_to_skip(radix);
784bb4a27f9SMark Johnston mask = scan->bm_bitmap;
785bb4a27f9SMark Johnston
786bb4a27f9SMark Johnston /* Discard any candidates that appear before cursor. */
78700fd73d2SDoug Moore digit = (cursor / radix) & BLIST_MASK;
788bb4a27f9SMark Johnston mask &= (u_daddr_t)-1 << digit;
789ee73fef9SAlan Cox if (mask == 0)
790ee73fef9SAlan Cox return (SWAPBLK_NONE);
7917090df5aSMatthew Dillon
7924be4fd5dSAlan Cox /*
793bb4a27f9SMark Johnston * If the first try is for a block that includes the cursor, pre-undo
794bb4a27f9SMark Johnston * the digit * radix offset in the first call; otherwise, ignore the
795bb4a27f9SMark Johnston * cursor entirely.
7964be4fd5dSAlan Cox */
797bb4a27f9SMark Johnston if (((mask >> digit) & 1) == 1)
798bb4a27f9SMark Johnston cursor -= digit * radix;
799d4e3484bSAlan Cox else
800bb4a27f9SMark Johnston cursor = blk;
8017090df5aSMatthew Dillon
8024be4fd5dSAlan Cox /*
803bb4a27f9SMark Johnston * Examine the nonempty subtree associated with each bit set in mask.
8044be4fd5dSAlan Cox */
805bb4a27f9SMark Johnston do {
80653519253SDoug Moore digit = bitpos(mask);
807bb4a27f9SMark Johnston i = 1 + digit * skip;
80887ae0686SDoug Moore if (*count <= scan[i].bm_bighint) {
8097090df5aSMatthew Dillon /*
810ec371b57SAlan Cox * The allocation might fit beginning in the i'th subtree.
8117090df5aSMatthew Dillon */
812bb4a27f9SMark Johnston r = blst_meta_alloc(&scan[i], cursor + digit * radix,
81300fd73d2SDoug Moore count, maxcount, radix / BLIST_RADIX);
8147090df5aSMatthew Dillon if (r != SWAPBLK_NONE) {
815bb4a27f9SMark Johnston if (scan[i].bm_bitmap == 0)
81653519253SDoug Moore scan->bm_bitmap ^= bitrange(digit, 1);
8177090df5aSMatthew Dillon return (r);
8187090df5aSMatthew Dillon }
8197090df5aSMatthew Dillon }
820bb4a27f9SMark Johnston cursor = blk;
82153519253SDoug Moore } while ((mask ^= bitrange(digit, 1)) != 0);
8227090df5aSMatthew Dillon
8237090df5aSMatthew Dillon /*
824bb4a27f9SMark Johnston * We couldn't allocate count in this subtree. If the whole tree was
825bb4a27f9SMark Johnston * scanned, and the last tree node is allocated, update bighint.
8267090df5aSMatthew Dillon */
82700fd73d2SDoug Moore if (scan_from_start && !(digit == BLIST_RADIX - 1 &&
828bb4a27f9SMark Johnston scan[i].bm_bighint == BLIST_MAX_ALLOC))
82987ae0686SDoug Moore scan->bm_bighint = *count - 1;
830d4e3484bSAlan Cox
8317090df5aSMatthew Dillon return (SWAPBLK_NONE);
8327090df5aSMatthew Dillon }
8337090df5aSMatthew Dillon
8347090df5aSMatthew Dillon /*
8357090df5aSMatthew Dillon * BLST_LEAF_FREE() - free allocated block from leaf bitmap
8367090df5aSMatthew Dillon *
8377090df5aSMatthew Dillon */
8387090df5aSMatthew Dillon static void
blst_leaf_free(blmeta_t * scan,daddr_t blk,int count)839b411efa4SAlan Cox blst_leaf_free(blmeta_t *scan, daddr_t blk, int count)
840b411efa4SAlan Cox {
841ec371b57SAlan Cox u_daddr_t mask;
842ec371b57SAlan Cox
8437090df5aSMatthew Dillon /*
8447090df5aSMatthew Dillon * free some data in this bitmap
845ec371b57SAlan Cox * mask=0000111111111110000
8467090df5aSMatthew Dillon * \_________/\__/
847ec371b57SAlan Cox * count n
8487090df5aSMatthew Dillon */
84900fd73d2SDoug Moore mask = bitrange(blk & BLIST_MASK, count);
850b1f59c92SDoug Moore KASSERT((scan->bm_bitmap & mask) == 0,
851b1f59c92SDoug Moore ("freeing free block: %jx, size %d, mask %jx",
852b1f59c92SDoug Moore (uintmax_t)blk, count, (uintmax_t)scan->bm_bitmap & mask));
853bb4a27f9SMark Johnston scan->bm_bitmap |= mask;
8547090df5aSMatthew Dillon }
8557090df5aSMatthew Dillon
8567090df5aSMatthew Dillon /*
8577090df5aSMatthew Dillon * BLST_META_FREE() - free allocated blocks from radix tree meta info
8587090df5aSMatthew Dillon *
8597090df5aSMatthew Dillon * This support routine frees a range of blocks from the bitmap.
8607090df5aSMatthew Dillon * The range must be entirely enclosed by this radix node. If a
8617090df5aSMatthew Dillon * meta node, we break the range down recursively to free blocks
8627090df5aSMatthew Dillon * in subnodes (which means that this code can free an arbitrary
8637090df5aSMatthew Dillon * range whereas the allocation code cannot allocate an arbitrary
8647090df5aSMatthew Dillon * range).
8657090df5aSMatthew Dillon */
8667090df5aSMatthew Dillon static void
blst_meta_free(blmeta_t * scan,daddr_t freeBlk,daddr_t count,u_daddr_t radix)867bee93d3cSAlan Cox blst_meta_free(blmeta_t *scan, daddr_t freeBlk, daddr_t count, u_daddr_t radix)
868d3783724SAlan Cox {
869bb4a27f9SMark Johnston daddr_t blk, endBlk, i, skip;
870bb4a27f9SMark Johnston int digit, endDigit;
8717090df5aSMatthew Dillon
872bb4a27f9SMark Johnston /*
873bb4a27f9SMark Johnston * We could probably do a better job here. We are required to make
874bb4a27f9SMark Johnston * bighint at least as large as the biggest allocable block of data.
875bb4a27f9SMark Johnston * If we just shoehorn it, a little extra overhead will be incurred
876bb4a27f9SMark Johnston * on the next allocation (but only that one typically).
877bb4a27f9SMark Johnston */
878bb4a27f9SMark Johnston scan->bm_bighint = BLIST_MAX_ALLOC;
879bb4a27f9SMark Johnston
88000fd73d2SDoug Moore if (radix == 1)
881a396c83aSAlan Cox return (blst_leaf_free(scan, freeBlk, count));
8827090df5aSMatthew Dillon
88300fd73d2SDoug Moore endBlk = freeBlk + count;
88400fd73d2SDoug Moore blk = (freeBlk + radix * BLIST_RADIX) & -(radix * BLIST_RADIX);
88500fd73d2SDoug Moore /*
88600fd73d2SDoug Moore * blk is first block past the end of the range of this meta node,
88700fd73d2SDoug Moore * or 0 in case of overflow.
88800fd73d2SDoug Moore */
88900fd73d2SDoug Moore if (blk != 0)
89000fd73d2SDoug Moore endBlk = ummin(endBlk, blk);
891bb4a27f9SMark Johnston skip = radix_to_skip(radix);
892bb4a27f9SMark Johnston blk = freeBlk & -radix;
89300fd73d2SDoug Moore digit = (blk / radix) & BLIST_MASK;
89400fd73d2SDoug Moore endDigit = 1 + (((endBlk - 1) / radix) & BLIST_MASK);
895bb4a27f9SMark Johnston scan->bm_bitmap |= bitrange(digit, endDigit - digit);
896bb4a27f9SMark Johnston for (i = 1 + digit * skip; blk < endBlk; i += skip) {
8977090df5aSMatthew Dillon blk += radix;
898bb4a27f9SMark Johnston count = ummin(blk, endBlk) - freeBlk;
89900fd73d2SDoug Moore blst_meta_free(&scan[i], freeBlk, count, radix / BLIST_RADIX);
900bb4a27f9SMark Johnston freeBlk = blk;
9017090df5aSMatthew Dillon }
9027090df5aSMatthew Dillon }
9037090df5aSMatthew Dillon
9047090df5aSMatthew Dillon /*
905bb4a27f9SMark Johnston * BLST_COPY() - copy one radix tree to another
9067090df5aSMatthew Dillon *
9077090df5aSMatthew Dillon * Locates free space in the source tree and frees it in the destination
9087090df5aSMatthew Dillon * tree. The space may not already be free in the destination.
9097090df5aSMatthew Dillon */
910b411efa4SAlan Cox static void
blst_copy(blmeta_t * scan,daddr_t blk,daddr_t radix,blist_t dest,daddr_t count)9112ac0c7c3SAlan Cox blst_copy(blmeta_t *scan, daddr_t blk, daddr_t radix, blist_t dest,
9122ac0c7c3SAlan Cox daddr_t count)
913b411efa4SAlan Cox {
914bb4a27f9SMark Johnston daddr_t endBlk, i, skip;
9157090df5aSMatthew Dillon
9167090df5aSMatthew Dillon /*
9177090df5aSMatthew Dillon * Leaf node
9187090df5aSMatthew Dillon */
9197090df5aSMatthew Dillon
92000fd73d2SDoug Moore if (radix == 1) {
921bb4a27f9SMark Johnston u_daddr_t v = scan->bm_bitmap;
9227090df5aSMatthew Dillon
9237090df5aSMatthew Dillon if (v == (u_daddr_t)-1) {
9247090df5aSMatthew Dillon blist_free(dest, blk, count);
9257090df5aSMatthew Dillon } else if (v != 0) {
9267090df5aSMatthew Dillon int i;
9277090df5aSMatthew Dillon
928bb4a27f9SMark Johnston for (i = 0; i < count; ++i) {
929064650c1SAlan Cox if (v & ((u_daddr_t)1 << i))
9307090df5aSMatthew Dillon blist_free(dest, blk + i, 1);
9317090df5aSMatthew Dillon }
9327090df5aSMatthew Dillon }
9337090df5aSMatthew Dillon return;
9347090df5aSMatthew Dillon }
9357090df5aSMatthew Dillon
9367090df5aSMatthew Dillon /*
9377090df5aSMatthew Dillon * Meta node
9387090df5aSMatthew Dillon */
9397090df5aSMatthew Dillon
940bb4a27f9SMark Johnston if (scan->bm_bitmap == 0) {
9417090df5aSMatthew Dillon /*
9427090df5aSMatthew Dillon * Source all allocated, leave dest allocated
9437090df5aSMatthew Dillon */
9447090df5aSMatthew Dillon return;
9457090df5aSMatthew Dillon }
9467090df5aSMatthew Dillon
947bb4a27f9SMark Johnston endBlk = blk + count;
948bb4a27f9SMark Johnston skip = radix_to_skip(radix);
949bb4a27f9SMark Johnston for (i = 1; blk < endBlk; i += skip) {
9507090df5aSMatthew Dillon blk += radix;
951bb4a27f9SMark Johnston count = radix;
952bb4a27f9SMark Johnston if (blk >= endBlk)
953bb4a27f9SMark Johnston count -= blk - endBlk;
95400fd73d2SDoug Moore blst_copy(&scan[i], blk - radix,
95500fd73d2SDoug Moore radix / BLIST_RADIX, dest, count);
9567090df5aSMatthew Dillon }
9577090df5aSMatthew Dillon }
9587090df5aSMatthew Dillon
9597090df5aSMatthew Dillon /*
96092da00bbSMatthew Dillon * BLST_LEAF_FILL() - allocate specific blocks in leaf bitmap
96192da00bbSMatthew Dillon *
96292da00bbSMatthew Dillon * This routine allocates all blocks in the specified range
96392da00bbSMatthew Dillon * regardless of any existing allocations in that range. Returns
96492da00bbSMatthew Dillon * the number of blocks allocated by the call.
96592da00bbSMatthew Dillon */
966a7249a6cSAlan Cox static daddr_t
blst_leaf_fill(blmeta_t * scan,daddr_t blk,int count)96792da00bbSMatthew Dillon blst_leaf_fill(blmeta_t *scan, daddr_t blk, int count)
96892da00bbSMatthew Dillon {
969a7249a6cSAlan Cox daddr_t nblks;
9704be4fd5dSAlan Cox u_daddr_t mask;
97192da00bbSMatthew Dillon
97200fd73d2SDoug Moore mask = bitrange(blk & BLIST_MASK, count);
97392da00bbSMatthew Dillon
9744be4fd5dSAlan Cox /* Count the number of blocks that we are allocating. */
975bb4a27f9SMark Johnston nblks = bitcount64(scan->bm_bitmap & mask);
97692da00bbSMatthew Dillon
977bb4a27f9SMark Johnston scan->bm_bitmap &= ~mask;
9784be4fd5dSAlan Cox return (nblks);
97992da00bbSMatthew Dillon }
98092da00bbSMatthew Dillon
98192da00bbSMatthew Dillon /*
98292da00bbSMatthew Dillon * BLIST_META_FILL() - allocate specific blocks at a meta node
98392da00bbSMatthew Dillon *
98492da00bbSMatthew Dillon * This routine allocates the specified range of blocks,
98592da00bbSMatthew Dillon * regardless of any existing allocations in the range. The
98692da00bbSMatthew Dillon * range must be within the extent of this node. Returns the
98792da00bbSMatthew Dillon * number of blocks allocated by the call.
98892da00bbSMatthew Dillon */
989a7249a6cSAlan Cox static daddr_t
blst_meta_fill(blmeta_t * scan,daddr_t allocBlk,daddr_t count,u_daddr_t radix)990bee93d3cSAlan Cox blst_meta_fill(blmeta_t *scan, daddr_t allocBlk, daddr_t count, u_daddr_t radix)
991d3783724SAlan Cox {
992bb4a27f9SMark Johnston daddr_t blk, endBlk, i, nblks, skip;
993bb4a27f9SMark Johnston int digit;
99492da00bbSMatthew Dillon
99500fd73d2SDoug Moore if (radix == 1)
996a396c83aSAlan Cox return (blst_leaf_fill(scan, allocBlk, count));
997bb4a27f9SMark Johnston
99800fd73d2SDoug Moore endBlk = allocBlk + count;
99900fd73d2SDoug Moore blk = (allocBlk + radix * BLIST_RADIX) & -(radix * BLIST_RADIX);
100000fd73d2SDoug Moore /*
100100fd73d2SDoug Moore * blk is first block past the end of the range of this meta node,
100200fd73d2SDoug Moore * or 0 in case of overflow.
100300fd73d2SDoug Moore */
100400fd73d2SDoug Moore if (blk != 0)
100500fd73d2SDoug Moore endBlk = ummin(endBlk, blk);
10062ac0c7c3SAlan Cox skip = radix_to_skip(radix);
1007bee93d3cSAlan Cox blk = allocBlk & -radix;
1008d3783724SAlan Cox nblks = 0;
1009bb4a27f9SMark Johnston while (blk < endBlk) {
101000fd73d2SDoug Moore digit = (blk / radix) & BLIST_MASK;
1011bb4a27f9SMark Johnston i = 1 + digit * skip;
101292da00bbSMatthew Dillon blk += radix;
1013bb4a27f9SMark Johnston count = ummin(blk, endBlk) - allocBlk;
101400fd73d2SDoug Moore nblks += blst_meta_fill(&scan[i], allocBlk, count,
101500fd73d2SDoug Moore radix / BLIST_RADIX);
1016bb4a27f9SMark Johnston if (scan[i].bm_bitmap == 0)
1017bb4a27f9SMark Johnston scan->bm_bitmap &= ~((u_daddr_t)1 << digit);
1018bb4a27f9SMark Johnston allocBlk = blk;
101992da00bbSMatthew Dillon }
1020a396c83aSAlan Cox return (nblks);
102192da00bbSMatthew Dillon }
102292da00bbSMatthew Dillon
10237090df5aSMatthew Dillon #ifdef BLIST_DEBUG
10247090df5aSMatthew Dillon
10257090df5aSMatthew Dillon static void
blst_radix_print(blmeta_t * scan,daddr_t blk,daddr_t radix,int tab)10262ac0c7c3SAlan Cox blst_radix_print(blmeta_t *scan, daddr_t blk, daddr_t radix, int tab)
10277090df5aSMatthew Dillon {
1028bb4a27f9SMark Johnston daddr_t skip;
102953519253SDoug Moore u_daddr_t mask;
1030bb4a27f9SMark Johnston int digit;
10317090df5aSMatthew Dillon
103200fd73d2SDoug Moore if (radix == 1) {
10337090df5aSMatthew Dillon printf(
1034bb4a27f9SMark Johnston "%*.*s(%08llx,%lld): bitmap %0*llx big=%lld\n",
10357090df5aSMatthew Dillon tab, tab, "",
103600fd73d2SDoug Moore (long long)blk, (long long)BLIST_RADIX,
103700fd73d2SDoug Moore (int)(1 + (BLIST_RADIX - 1) / 4),
1038bb4a27f9SMark Johnston (long long)scan->bm_bitmap,
103992da00bbSMatthew Dillon (long long)scan->bm_bighint
10407090df5aSMatthew Dillon );
10417090df5aSMatthew Dillon return;
10427090df5aSMatthew Dillon }
10437090df5aSMatthew Dillon
10447090df5aSMatthew Dillon printf(
1045bb4a27f9SMark Johnston "%*.*s(%08llx): subtree (%lld/%lld) bitmap %0*llx big=%lld {\n",
10467090df5aSMatthew Dillon tab, tab, "",
104700fd73d2SDoug Moore (long long)blk, (long long)radix * BLIST_RADIX,
104800fd73d2SDoug Moore (long long)radix * BLIST_RADIX,
104900fd73d2SDoug Moore (int)(1 + (BLIST_RADIX - 1) / 4),
1050bb4a27f9SMark Johnston (long long)scan->bm_bitmap,
105192da00bbSMatthew Dillon (long long)scan->bm_bighint
10527090df5aSMatthew Dillon );
10537090df5aSMatthew Dillon
1054bb4a27f9SMark Johnston skip = radix_to_skip(radix);
10557090df5aSMatthew Dillon tab += 4;
10567090df5aSMatthew Dillon
1057bb4a27f9SMark Johnston mask = scan->bm_bitmap;
1058bb4a27f9SMark Johnston /* Examine the nonempty subtree associated with each bit set in mask */
1059bb4a27f9SMark Johnston do {
106053519253SDoug Moore digit = bitpos(mask);
1061bb4a27f9SMark Johnston blst_radix_print(&scan[1 + digit * skip], blk + digit * radix,
106200fd73d2SDoug Moore radix / BLIST_RADIX, tab);
106353519253SDoug Moore } while ((mask ^= bitrange(digit, 1)) != 0);
10647090df5aSMatthew Dillon tab -= 4;
10657090df5aSMatthew Dillon
10667090df5aSMatthew Dillon printf(
10677090df5aSMatthew Dillon "%*.*s}\n",
10687090df5aSMatthew Dillon tab, tab, ""
10697090df5aSMatthew Dillon );
10707090df5aSMatthew Dillon }
10717090df5aSMatthew Dillon
10727090df5aSMatthew Dillon #endif
10737090df5aSMatthew Dillon
10747090df5aSMatthew Dillon #ifdef BLIST_DEBUG
10757090df5aSMatthew Dillon
10767090df5aSMatthew Dillon int
main(int ac,char ** av)10777090df5aSMatthew Dillon main(int ac, char **av)
10787090df5aSMatthew Dillon {
107900fd73d2SDoug Moore daddr_t size = BLIST_RADIX * BLIST_RADIX;
10807090df5aSMatthew Dillon int i;
10817090df5aSMatthew Dillon blist_t bl;
1082d027ed2eSAlan Cox struct sbuf *s;
10837090df5aSMatthew Dillon
10847090df5aSMatthew Dillon for (i = 1; i < ac; ++i) {
10857090df5aSMatthew Dillon const char *ptr = av[i];
10867090df5aSMatthew Dillon if (*ptr != '-') {
108700fd73d2SDoug Moore size = strtoll(ptr, NULL, 0);
10887090df5aSMatthew Dillon continue;
10897090df5aSMatthew Dillon }
10907090df5aSMatthew Dillon ptr += 2;
10917090df5aSMatthew Dillon fprintf(stderr, "Bad option: %s\n", ptr - 2);
10927090df5aSMatthew Dillon exit(1);
10937090df5aSMatthew Dillon }
1094c8c7ad92SKip Macy bl = blist_create(size, M_WAITOK);
109500fd73d2SDoug Moore if (bl == NULL) {
109600fd73d2SDoug Moore fprintf(stderr, "blist_create failed\n");
109700fd73d2SDoug Moore exit(1);
109800fd73d2SDoug Moore }
10997090df5aSMatthew Dillon blist_free(bl, 0, size);
11007090df5aSMatthew Dillon
11017090df5aSMatthew Dillon for (;;) {
11027090df5aSMatthew Dillon char buf[1024];
1103d90bf7d8SAlan Cox long long da = 0;
110487ae0686SDoug Moore int count = 0, maxcount = 0;
11057090df5aSMatthew Dillon
11064be4fd5dSAlan Cox printf("%lld/%lld/%lld> ", (long long)blist_avail(bl),
110700fd73d2SDoug Moore (long long)size, (long long)bl->bl_radix * BLIST_RADIX);
11087090df5aSMatthew Dillon fflush(stdout);
11097090df5aSMatthew Dillon if (fgets(buf, sizeof(buf), stdin) == NULL)
11107090df5aSMatthew Dillon break;
11117090df5aSMatthew Dillon switch(buf[0]) {
11127090df5aSMatthew Dillon case 'r':
111387ae0686SDoug Moore if (sscanf(buf + 1, "%d", &count) == 1) {
1114d90bf7d8SAlan Cox blist_resize(&bl, count, 1, M_WAITOK);
11157090df5aSMatthew Dillon } else {
11167090df5aSMatthew Dillon printf("?\n");
11177090df5aSMatthew Dillon }
11187090df5aSMatthew Dillon case 'p':
11197090df5aSMatthew Dillon blist_print(bl);
11207090df5aSMatthew Dillon break;
1121d027ed2eSAlan Cox case 's':
1122d027ed2eSAlan Cox s = sbuf_new_auto();
1123d027ed2eSAlan Cox blist_stats(bl, s);
1124d027ed2eSAlan Cox sbuf_finish(s);
1125d027ed2eSAlan Cox printf("%s", sbuf_data(s));
1126d027ed2eSAlan Cox sbuf_delete(s);
1127d027ed2eSAlan Cox break;
11287090df5aSMatthew Dillon case 'a':
112987ae0686SDoug Moore if (sscanf(buf + 1, "%d%d", &count, &maxcount) == 2) {
113087ae0686SDoug Moore daddr_t blk = blist_alloc(bl, &count, maxcount);
113187ae0686SDoug Moore printf(" R=%08llx, c=%08d\n",
113287ae0686SDoug Moore (long long)blk, count);
11337090df5aSMatthew Dillon } else {
11347090df5aSMatthew Dillon printf("?\n");
11357090df5aSMatthew Dillon }
11367090df5aSMatthew Dillon break;
11377090df5aSMatthew Dillon case 'f':
113887ae0686SDoug Moore if (sscanf(buf + 1, "%llx %d", &da, &count) == 2) {
11397090df5aSMatthew Dillon blist_free(bl, da, count);
11407090df5aSMatthew Dillon } else {
11417090df5aSMatthew Dillon printf("?\n");
11427090df5aSMatthew Dillon }
11437090df5aSMatthew Dillon break;
114492da00bbSMatthew Dillon case 'l':
114587ae0686SDoug Moore if (sscanf(buf + 1, "%llx %d", &da, &count) == 2) {
1146a7249a6cSAlan Cox printf(" n=%jd\n",
1147a7249a6cSAlan Cox (intmax_t)blist_fill(bl, da, count));
114892da00bbSMatthew Dillon } else {
114992da00bbSMatthew Dillon printf("?\n");
115092da00bbSMatthew Dillon }
115192da00bbSMatthew Dillon break;
11527090df5aSMatthew Dillon case '?':
11537090df5aSMatthew Dillon case 'h':
11547090df5aSMatthew Dillon puts(
11557090df5aSMatthew Dillon "p -print\n"
1156d027ed2eSAlan Cox "s -stats\n"
115787ae0686SDoug Moore "a %d %d -allocate\n"
11587090df5aSMatthew Dillon "f %x %d -free\n"
115992da00bbSMatthew Dillon "l %x %d -fill\n"
11607090df5aSMatthew Dillon "r %d -resize\n"
1161d4808c44SDoug Moore "h/? -help\n"
1162d4808c44SDoug Moore "q -quit"
11637090df5aSMatthew Dillon );
11647090df5aSMatthew Dillon break;
1165d4808c44SDoug Moore case 'q':
1166d4808c44SDoug Moore break;
11677090df5aSMatthew Dillon default:
11687090df5aSMatthew Dillon printf("?\n");
11697090df5aSMatthew Dillon break;
11707090df5aSMatthew Dillon }
1171d4808c44SDoug Moore if (buf[0] == 'q')
1172d4808c44SDoug Moore break;
11737090df5aSMatthew Dillon }
11747090df5aSMatthew Dillon return (0);
11757090df5aSMatthew Dillon }
11767090df5aSMatthew Dillon
11777090df5aSMatthew Dillon #endif
1178