4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or https://opensource.org/licenses/CDDL-1.0.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
22 * Copyright (c) 2020, 2021, 2022 by Pawel Jakub Dawidek
25 #ifndef _SYS_BRT_IMPL_H
26 #define _SYS_BRT_IMPL_H
33 * BRT - Block Reference Table.
35 #define BRT_OBJECT_VDEV_PREFIX "com.fudosecurity:brt:vdev:"
38 * We divide each VDEV into 16MB chunks. Each chunk is represented in memory
39 * by a 16bit counter, thus 1TB VDEV requires 128kB of memory: (1TB / 16MB) * 2B
40 * Each element in this array represents how many BRT entries do we have in this
41 * chunk of storage. We always load this entire array into memory and update as
42 * needed. By having it in memory we can quickly tell (during zio_free()) if
43 * there are any BRT entries that we might need to update.
45 * This value cannot be larger than 16MB, at least as long as we support
46 * 512 byte block sizes. With 512 byte block size we can have exactly
47 * 32768 blocks in 16MB. In 32MB we could have 65536 blocks, which is one too
48 * many for a 16bit counter.
50 #define BRT_RANGESIZE (16 * 1024 * 1024)
51 _Static_assert(BRT_RANGESIZE
/ SPA_MINBLOCKSIZE
<= UINT16_MAX
,
52 "BRT_RANGESIZE is too large.");
54 * We don't want to update the whole structure every time. Maintain bitmap
55 * of dirty blocks within the regions, so that a single bit represents a
56 * block size of entcounts. For example if we have a 1PB vdev then all
57 * entcounts take 128MB of memory ((64TB / 16MB) * 2B). We can divide this
58 * 128MB array of entcounts into 32kB disk blocks, as we don't want to update
59 * the whole 128MB on disk when we have updated only a single entcount.
60 * We maintain a bitmap where each 32kB disk block within 128MB entcounts array
61 * is represented by a single bit. This gives us 4096 bits. A set bit in the
62 * bitmap means that we had a change in at least one of the 16384 entcounts
63 * that reside on a 32kB disk block (32kB / sizeof (uint16_t)).
65 #define BRT_BLOCKSIZE (32 * 1024)
66 #define BRT_RANGESIZE_TO_NBLOCKS(size) \
67 (((size) - 1) / BRT_BLOCKSIZE / sizeof (uint16_t) + 1)
69 #define BRT_LITTLE_ENDIAN 0
70 #define BRT_BIG_ENDIAN 1
71 #ifdef _ZFS_LITTLE_ENDIAN
72 #define BRT_NATIVE_BYTEORDER BRT_LITTLE_ENDIAN
73 #define BRT_NON_NATIVE_BYTEORDER BRT_BIG_ENDIAN
75 #define BRT_NATIVE_BYTEORDER BRT_BIG_ENDIAN
76 #define BRT_NON_NATIVE_BYTEORDER BRT_LITTLE_ENDIAN
79 typedef struct brt_vdev_phys
{
80 uint64_t bvp_mos_entries
;
82 uint64_t bvp_byteorder
;
83 uint64_t bvp_totalcount
;
84 uint64_t bvp_rangesize
;
85 uint64_t bvp_usedspace
;
86 uint64_t bvp_savedspace
;
89 typedef struct brt_vdev
{
95 * Is the structure initiated?
96 * (bv_entcount and bv_bitmap are allocated?)
98 boolean_t bv_initiated
;
100 * Object number in the MOS for the entcount array and brt_vdev_phys.
102 uint64_t bv_mos_brtvdev
;
104 * Object number in the MOS for the entries table.
106 uint64_t bv_mos_entries
;
112 * Does the bv_entcount[] array needs byte swapping?
114 boolean_t bv_need_byteswap
;
116 * Number of entries in the bv_entcount[] array.
120 * This is the array with BRT entry count per BRT_RANGESIZE.
122 uint16_t *bv_entcount
;
124 * Sum of all bv_entcount[]s.
126 uint64_t bv_totalcount
;
128 * Space on disk occupied by cloned blocks (without compression).
130 uint64_t bv_usedspace
;
132 * How much additional space would be occupied without block cloning.
134 uint64_t bv_savedspace
;
136 * brt_vdev_phys needs updating on disk.
138 boolean_t bv_meta_dirty
;
140 * bv_entcount[] needs updating on disk.
142 boolean_t bv_entcount_dirty
;
144 * bv_entcount[] potentially can be a bit too big to sychronize it all
145 * when we just changed few entcounts. The fields below allow us to
146 * track updates to bv_entcount[] array since the last sync.
147 * A single bit in the bv_bitmap represents as many entcounts as can
148 * fit into a single BRT_BLOCKSIZE.
149 * For example we have 65536 entcounts in the bv_entcount array
150 * (so the whole array is 128kB). We updated bv_entcount[2] and
151 * bv_entcount[5]. In that case only first bit in the bv_bitmap will
152 * be set and we will write only first BRT_BLOCKSIZE out of 128kB.
164 #define brt_mos brt_spa->spa_meta_objset
165 uint64_t brt_rangesize
;
166 uint64_t brt_usedspace
;
167 uint64_t brt_savedspace
;
168 avl_tree_t brt_pending_tree
[TXG_SIZE
];
169 kmutex_t brt_pending_lock
[TXG_SIZE
];
170 /* Sum of all entries across all bv_trees. */
171 uint64_t brt_nentries
;
172 brt_vdev_t
*brt_vdevs
;
176 /* Size of bre_offset / sizeof (uint64_t). */
177 #define BRT_KEY_WORDS (1)
181 * On-disk we use bre_offset as the key and bre_refcount as the value.
183 typedef struct brt_entry
{
185 uint64_t bre_refcount
;
189 typedef struct brt_pending_entry
{
193 } brt_pending_entry_t
;
199 #endif /* _SYS_BRT_IMPL_H */