xref: /illumos-gate/usr/src/uts/common/fs/zfs/bplist.c (revision 02f22325adceea5762fbc7f49cee82e407e8f3a1)
1 /*
2  * CDDL HEADER START
3  *
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.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
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]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #pragma ident	"%Z%%M%	%I%	%E% SMI"
27 
28 #include <sys/bplist.h>
29 #include <sys/zfs_context.h>
30 
31 static int
32 bplist_hold(bplist_t *bpl)
33 {
34 	ASSERT(MUTEX_HELD(&bpl->bpl_lock));
35 	if (bpl->bpl_dbuf == NULL) {
36 		int err = dmu_bonus_hold(bpl->bpl_mos,
37 		    bpl->bpl_object, bpl, &bpl->bpl_dbuf);
38 		if (err)
39 			return (err);
40 		bpl->bpl_phys = bpl->bpl_dbuf->db_data;
41 	}
42 	return (0);
43 }
44 
45 uint64_t
46 bplist_create(objset_t *mos, int blocksize, dmu_tx_t *tx)
47 {
48 	int size;
49 
50 	size = spa_version(dmu_objset_spa(mos)) < SPA_VERSION_BPLIST_ACCOUNT ?
51 	    BPLIST_SIZE_V0 : sizeof (bplist_phys_t);
52 
53 	return (dmu_object_alloc(mos, DMU_OT_BPLIST, blocksize,
54 	    DMU_OT_BPLIST_HDR, size, tx));
55 }
56 
57 void
58 bplist_destroy(objset_t *mos, uint64_t object, dmu_tx_t *tx)
59 {
60 	VERIFY(dmu_object_free(mos, object, tx) == 0);
61 }
62 
63 int
64 bplist_open(bplist_t *bpl, objset_t *mos, uint64_t object)
65 {
66 	dmu_object_info_t doi;
67 	int err;
68 
69 	err = dmu_object_info(mos, object, &doi);
70 	if (err)
71 		return (err);
72 
73 	mutex_enter(&bpl->bpl_lock);
74 
75 	ASSERT(bpl->bpl_dbuf == NULL);
76 	ASSERT(bpl->bpl_phys == NULL);
77 	ASSERT(bpl->bpl_cached_dbuf == NULL);
78 	ASSERT(bpl->bpl_queue == NULL);
79 	ASSERT(object != 0);
80 	ASSERT3U(doi.doi_type, ==, DMU_OT_BPLIST);
81 	ASSERT3U(doi.doi_bonus_type, ==, DMU_OT_BPLIST_HDR);
82 
83 	bpl->bpl_mos = mos;
84 	bpl->bpl_object = object;
85 	bpl->bpl_blockshift = highbit(doi.doi_data_block_size - 1);
86 	bpl->bpl_bpshift = bpl->bpl_blockshift - SPA_BLKPTRSHIFT;
87 	bpl->bpl_havecomp = (doi.doi_bonus_size == sizeof (bplist_phys_t));
88 
89 	mutex_exit(&bpl->bpl_lock);
90 	return (0);
91 }
92 
93 void
94 bplist_close(bplist_t *bpl)
95 {
96 	mutex_enter(&bpl->bpl_lock);
97 
98 	ASSERT(bpl->bpl_queue == NULL);
99 
100 	if (bpl->bpl_cached_dbuf) {
101 		dmu_buf_rele(bpl->bpl_cached_dbuf, bpl);
102 		bpl->bpl_cached_dbuf = NULL;
103 	}
104 	if (bpl->bpl_dbuf) {
105 		dmu_buf_rele(bpl->bpl_dbuf, bpl);
106 		bpl->bpl_dbuf = NULL;
107 		bpl->bpl_phys = NULL;
108 	}
109 
110 	mutex_exit(&bpl->bpl_lock);
111 }
112 
113 boolean_t
114 bplist_empty(bplist_t *bpl)
115 {
116 	boolean_t rv;
117 
118 	if (bpl->bpl_object == 0)
119 		return (B_TRUE);
120 
121 	mutex_enter(&bpl->bpl_lock);
122 	VERIFY(0 == bplist_hold(bpl)); /* XXX */
123 	rv = (bpl->bpl_phys->bpl_entries == 0);
124 	mutex_exit(&bpl->bpl_lock);
125 
126 	return (rv);
127 }
128 
129 static int
130 bplist_cache(bplist_t *bpl, uint64_t blkid)
131 {
132 	int err = 0;
133 
134 	if (bpl->bpl_cached_dbuf == NULL ||
135 	    bpl->bpl_cached_dbuf->db_offset != (blkid << bpl->bpl_blockshift)) {
136 		if (bpl->bpl_cached_dbuf != NULL)
137 			dmu_buf_rele(bpl->bpl_cached_dbuf, bpl);
138 		err = dmu_buf_hold(bpl->bpl_mos,
139 		    bpl->bpl_object, blkid << bpl->bpl_blockshift,
140 		    bpl, &bpl->bpl_cached_dbuf);
141 		ASSERT(err || bpl->bpl_cached_dbuf->db_size ==
142 		    1ULL << bpl->bpl_blockshift);
143 	}
144 	return (err);
145 }
146 
147 int
148 bplist_iterate(bplist_t *bpl, uint64_t *itorp, blkptr_t *bp)
149 {
150 	uint64_t blk, off;
151 	blkptr_t *bparray;
152 	int err;
153 
154 	mutex_enter(&bpl->bpl_lock);
155 
156 	err = bplist_hold(bpl);
157 	if (err) {
158 		mutex_exit(&bpl->bpl_lock);
159 		return (err);
160 	}
161 
162 	if (*itorp >= bpl->bpl_phys->bpl_entries) {
163 		mutex_exit(&bpl->bpl_lock);
164 		return (ENOENT);
165 	}
166 
167 	blk = *itorp >> bpl->bpl_bpshift;
168 	off = P2PHASE(*itorp, 1ULL << bpl->bpl_bpshift);
169 
170 	err = bplist_cache(bpl, blk);
171 	if (err) {
172 		mutex_exit(&bpl->bpl_lock);
173 		return (err);
174 	}
175 
176 	bparray = bpl->bpl_cached_dbuf->db_data;
177 	*bp = bparray[off];
178 	(*itorp)++;
179 	mutex_exit(&bpl->bpl_lock);
180 	return (0);
181 }
182 
183 int
184 bplist_enqueue(bplist_t *bpl, const blkptr_t *bp, dmu_tx_t *tx)
185 {
186 	uint64_t blk, off;
187 	blkptr_t *bparray;
188 	int err;
189 
190 	ASSERT(!BP_IS_HOLE(bp));
191 	mutex_enter(&bpl->bpl_lock);
192 	err = bplist_hold(bpl);
193 	if (err)
194 		return (err);
195 
196 	blk = bpl->bpl_phys->bpl_entries >> bpl->bpl_bpshift;
197 	off = P2PHASE(bpl->bpl_phys->bpl_entries, 1ULL << bpl->bpl_bpshift);
198 
199 	err = bplist_cache(bpl, blk);
200 	if (err) {
201 		mutex_exit(&bpl->bpl_lock);
202 		return (err);
203 	}
204 
205 	dmu_buf_will_dirty(bpl->bpl_cached_dbuf, tx);
206 	bparray = bpl->bpl_cached_dbuf->db_data;
207 	bparray[off] = *bp;
208 
209 	/* We never need the fill count. */
210 	bparray[off].blk_fill = 0;
211 
212 	/* The bplist will compress better if we can leave off the checksum */
213 	bzero(&bparray[off].blk_cksum, sizeof (bparray[off].blk_cksum));
214 
215 	dmu_buf_will_dirty(bpl->bpl_dbuf, tx);
216 	bpl->bpl_phys->bpl_entries++;
217 	bpl->bpl_phys->bpl_bytes +=
218 	    bp_get_dasize(dmu_objset_spa(bpl->bpl_mos), bp);
219 	if (bpl->bpl_havecomp) {
220 		bpl->bpl_phys->bpl_comp += BP_GET_PSIZE(bp);
221 		bpl->bpl_phys->bpl_uncomp += BP_GET_UCSIZE(bp);
222 	}
223 	mutex_exit(&bpl->bpl_lock);
224 
225 	return (0);
226 }
227 
228 /*
229  * Deferred entry; will be written later by bplist_sync().
230  */
231 void
232 bplist_enqueue_deferred(bplist_t *bpl, const blkptr_t *bp)
233 {
234 	bplist_q_t *bpq = kmem_alloc(sizeof (*bpq), KM_SLEEP);
235 
236 	ASSERT(!BP_IS_HOLE(bp));
237 	mutex_enter(&bpl->bpl_lock);
238 	bpq->bpq_blk = *bp;
239 	bpq->bpq_next = bpl->bpl_queue;
240 	bpl->bpl_queue = bpq;
241 	mutex_exit(&bpl->bpl_lock);
242 }
243 
244 void
245 bplist_sync(bplist_t *bpl, dmu_tx_t *tx)
246 {
247 	bplist_q_t *bpq;
248 
249 	mutex_enter(&bpl->bpl_lock);
250 	while ((bpq = bpl->bpl_queue) != NULL) {
251 		bpl->bpl_queue = bpq->bpq_next;
252 		mutex_exit(&bpl->bpl_lock);
253 		VERIFY(0 == bplist_enqueue(bpl, &bpq->bpq_blk, tx));
254 		kmem_free(bpq, sizeof (*bpq));
255 		mutex_enter(&bpl->bpl_lock);
256 	}
257 	mutex_exit(&bpl->bpl_lock);
258 }
259 
260 void
261 bplist_vacate(bplist_t *bpl, dmu_tx_t *tx)
262 {
263 	mutex_enter(&bpl->bpl_lock);
264 	ASSERT3P(bpl->bpl_queue, ==, NULL);
265 	VERIFY(0 == bplist_hold(bpl));
266 	dmu_buf_will_dirty(bpl->bpl_dbuf, tx);
267 	VERIFY(0 == dmu_free_range(bpl->bpl_mos,
268 	    bpl->bpl_object, 0, -1ULL, tx));
269 	bpl->bpl_phys->bpl_entries = 0;
270 	bpl->bpl_phys->bpl_bytes = 0;
271 	if (bpl->bpl_havecomp) {
272 		bpl->bpl_phys->bpl_comp = 0;
273 		bpl->bpl_phys->bpl_uncomp = 0;
274 	}
275 	mutex_exit(&bpl->bpl_lock);
276 }
277 
278 int
279 bplist_space(bplist_t *bpl, uint64_t *usedp, uint64_t *compp, uint64_t *uncompp)
280 {
281 	int err;
282 
283 	mutex_enter(&bpl->bpl_lock);
284 
285 	err = bplist_hold(bpl);
286 	if (err) {
287 		mutex_exit(&bpl->bpl_lock);
288 		return (err);
289 	}
290 
291 	*usedp = bpl->bpl_phys->bpl_bytes;
292 	if (bpl->bpl_havecomp) {
293 		*compp = bpl->bpl_phys->bpl_comp;
294 		*uncompp = bpl->bpl_phys->bpl_uncomp;
295 	}
296 	mutex_exit(&bpl->bpl_lock);
297 
298 	if (!bpl->bpl_havecomp) {
299 		uint64_t itor = 0, comp = 0, uncomp = 0;
300 		blkptr_t bp;
301 
302 		while ((err = bplist_iterate(bpl, &itor, &bp)) == 0) {
303 			comp += BP_GET_PSIZE(&bp);
304 			uncomp += BP_GET_UCSIZE(&bp);
305 		}
306 		if (err == ENOENT)
307 			err = 0;
308 		*compp = comp;
309 		*uncompp = uncomp;
310 	}
311 
312 	return (err);
313 }
314