Merge tag 'v9.0.0-rc3'
[qemu/ar7.git] / block / monitor / bitmap-qmp-cmds.c
bloba738e7bbf7a132fcb66a016c0e568993b5108932
1 /*
2 * QEMU block dirty bitmap QMP commands
4 * Copyright (c) 2003-2008 Fabrice Bellard
6 * This work is licensed under the terms of the GNU GPL, version 2 or
7 * later. See the COPYING file in the top-level directory.
9 * This file incorporates work covered by the following copyright and
10 * permission notice:
12 * Copyright (c) 2003-2008 Fabrice Bellard
14 * Permission is hereby granted, free of charge, to any person obtaining a copy
15 * of this software and associated documentation files (the "Software"), to deal
16 * in the Software without restriction, including without limitation the rights
17 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
18 * copies of the Software, and to permit persons to whom the Software is
19 * furnished to do so, subject to the following conditions:
21 * The above copyright notice and this permission notice shall be included in
22 * all copies or substantial portions of the Software.
24 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
25 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
26 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
27 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
28 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
29 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
30 * THE SOFTWARE.
33 #include "qemu/osdep.h"
35 #include "block/block-io.h"
36 #include "block/block_int.h"
37 #include "block/dirty-bitmap.h"
38 #include "qapi/qapi-commands-block.h"
39 #include "qapi/error.h"
41 /**
42 * block_dirty_bitmap_lookup:
43 * Return a dirty bitmap (if present), after validating
44 * the node reference and bitmap names.
46 * @node: The name of the BDS node to search for bitmaps
47 * @name: The name of the bitmap to search for
48 * @pbs: Output pointer for BDS lookup, if desired. Can be NULL.
49 * @errp: Output pointer for error information. Can be NULL.
51 * @return: A bitmap object on success, or NULL on failure.
53 BdrvDirtyBitmap *block_dirty_bitmap_lookup(const char *node,
54 const char *name,
55 BlockDriverState **pbs,
56 Error **errp)
58 BlockDriverState *bs;
59 BdrvDirtyBitmap *bitmap;
61 GLOBAL_STATE_CODE();
63 if (!node) {
64 error_setg(errp, "Node cannot be NULL");
65 return NULL;
67 if (!name) {
68 error_setg(errp, "Bitmap name cannot be NULL");
69 return NULL;
71 bs = bdrv_lookup_bs(node, node, NULL);
72 if (!bs) {
73 error_setg(errp, "Node '%s' not found", node);
74 return NULL;
77 bitmap = bdrv_find_dirty_bitmap(bs, name);
78 if (!bitmap) {
79 error_setg(errp, "Dirty bitmap '%s' not found", name);
80 return NULL;
83 if (pbs) {
84 *pbs = bs;
87 return bitmap;
90 void qmp_block_dirty_bitmap_add(const char *node, const char *name,
91 bool has_granularity, uint32_t granularity,
92 bool has_persistent, bool persistent,
93 bool has_disabled, bool disabled,
94 Error **errp)
96 BlockDriverState *bs;
97 BdrvDirtyBitmap *bitmap;
99 if (!name || name[0] == '\0') {
100 error_setg(errp, "Bitmap name cannot be empty");
101 return;
104 bs = bdrv_lookup_bs(node, node, errp);
105 if (!bs) {
106 return;
109 if (has_granularity) {
110 if (granularity < 512 || !is_power_of_2(granularity)) {
111 error_setg(errp, "Granularity must be power of 2 "
112 "and at least 512");
113 return;
115 } else {
116 /* Default to cluster size, if available: */
117 granularity = bdrv_get_default_bitmap_granularity(bs);
120 if (!has_persistent) {
121 persistent = false;
124 if (!has_disabled) {
125 disabled = false;
128 if (persistent &&
129 !bdrv_can_store_new_dirty_bitmap(bs, name, granularity, errp))
131 return;
134 bitmap = bdrv_create_dirty_bitmap(bs, granularity, name, errp);
135 if (bitmap == NULL) {
136 return;
139 if (disabled) {
140 bdrv_disable_dirty_bitmap(bitmap);
143 bdrv_dirty_bitmap_set_persistence(bitmap, persistent);
146 BdrvDirtyBitmap *block_dirty_bitmap_remove(const char *node, const char *name,
147 bool release,
148 BlockDriverState **bitmap_bs,
149 Error **errp)
151 BlockDriverState *bs;
152 BdrvDirtyBitmap *bitmap;
154 GLOBAL_STATE_CODE();
156 bitmap = block_dirty_bitmap_lookup(node, name, &bs, errp);
157 if (!bitmap || !bs) {
158 return NULL;
161 if (bdrv_dirty_bitmap_check(bitmap, BDRV_BITMAP_BUSY | BDRV_BITMAP_RO,
162 errp)) {
163 return NULL;
166 if (bdrv_dirty_bitmap_get_persistence(bitmap) &&
167 bdrv_remove_persistent_dirty_bitmap(bs, name, errp) < 0)
169 return NULL;
172 if (release) {
173 bdrv_release_dirty_bitmap(bitmap);
176 if (bitmap_bs) {
177 *bitmap_bs = bs;
180 return release ? NULL : bitmap;
183 void qmp_block_dirty_bitmap_remove(const char *node, const char *name,
184 Error **errp)
186 block_dirty_bitmap_remove(node, name, true, NULL, errp);
190 * Completely clear a bitmap, for the purposes of synchronizing a bitmap
191 * immediately after a full backup operation.
193 void qmp_block_dirty_bitmap_clear(const char *node, const char *name,
194 Error **errp)
196 BdrvDirtyBitmap *bitmap;
197 BlockDriverState *bs;
199 bitmap = block_dirty_bitmap_lookup(node, name, &bs, errp);
200 if (!bitmap || !bs) {
201 return;
204 if (bdrv_dirty_bitmap_check(bitmap, BDRV_BITMAP_DEFAULT, errp)) {
205 return;
208 bdrv_clear_dirty_bitmap(bitmap, NULL);
211 void qmp_block_dirty_bitmap_enable(const char *node, const char *name,
212 Error **errp)
214 BlockDriverState *bs;
215 BdrvDirtyBitmap *bitmap;
217 bitmap = block_dirty_bitmap_lookup(node, name, &bs, errp);
218 if (!bitmap) {
219 return;
222 if (bdrv_dirty_bitmap_check(bitmap, BDRV_BITMAP_ALLOW_RO, errp)) {
223 return;
226 bdrv_enable_dirty_bitmap(bitmap);
229 void qmp_block_dirty_bitmap_disable(const char *node, const char *name,
230 Error **errp)
232 BlockDriverState *bs;
233 BdrvDirtyBitmap *bitmap;
235 bitmap = block_dirty_bitmap_lookup(node, name, &bs, errp);
236 if (!bitmap) {
237 return;
240 if (bdrv_dirty_bitmap_check(bitmap, BDRV_BITMAP_ALLOW_RO, errp)) {
241 return;
244 bdrv_disable_dirty_bitmap(bitmap);
247 BdrvDirtyBitmap *block_dirty_bitmap_merge(const char *dst_node,
248 const char *dst_bitmap,
249 BlockDirtyBitmapOrStrList *bms,
250 HBitmap **backup, Error **errp)
252 BlockDriverState *bs;
253 BdrvDirtyBitmap *dst, *src;
254 BlockDirtyBitmapOrStrList *lst;
255 const char *src_node, *src_bitmap;
256 HBitmap *local_backup = NULL;
258 GLOBAL_STATE_CODE();
260 dst = block_dirty_bitmap_lookup(dst_node, dst_bitmap, &bs, errp);
261 if (!dst) {
262 return NULL;
265 for (lst = bms; lst; lst = lst->next) {
266 switch (lst->value->type) {
267 case QTYPE_QSTRING:
268 src_bitmap = lst->value->u.local;
269 src = bdrv_find_dirty_bitmap(bs, src_bitmap);
270 if (!src) {
271 error_setg(errp, "Dirty bitmap '%s' not found", src_bitmap);
272 goto fail;
274 break;
275 case QTYPE_QDICT:
276 src_node = lst->value->u.external.node;
277 src_bitmap = lst->value->u.external.name;
278 src = block_dirty_bitmap_lookup(src_node, src_bitmap, NULL, errp);
279 if (!src) {
280 goto fail;
282 break;
283 default:
284 abort();
287 /* We do backup only for first merge operation */
288 if (!bdrv_merge_dirty_bitmap(dst, src,
289 local_backup ? NULL : &local_backup,
290 errp))
292 goto fail;
296 if (backup) {
297 *backup = local_backup;
298 } else {
299 hbitmap_free(local_backup);
302 return dst;
304 fail:
305 if (local_backup) {
306 bdrv_restore_dirty_bitmap(dst, local_backup);
309 return NULL;
312 void qmp_block_dirty_bitmap_merge(const char *node, const char *target,
313 BlockDirtyBitmapOrStrList *bitmaps,
314 Error **errp)
316 block_dirty_bitmap_merge(node, target, bitmaps, NULL, errp);