qapi: support external bitmaps in block-dirty-bitmap-merge

Add new optional parameter making possible to merge bitmaps from
different nodes. It is needed to maintain external snapshots during
incremental backup chain history.

Signed-off-by: Vladimir Sementsov-Ogievskiy <vsementsov@virtuozzo.com>
Reviewed-by: John Snow <jsnow@redhat.com>
Message-id: 20190517152111.206494-2-vsementsov@virtuozzo.com
Signed-off-by: John Snow <jsnow@redhat.com>
This commit is contained in:
Vladimir Sementsov-Ogievskiy 2019-05-28 19:33:31 -04:00 committed by John Snow
parent 592203e7cf
commit eff0829b07
3 changed files with 58 additions and 23 deletions

View File

@ -816,10 +816,10 @@ void bdrv_merge_dirty_bitmap(BdrvDirtyBitmap *dest, const BdrvDirtyBitmap *src,
{ {
bool ret; bool ret;
/* only bitmaps from one bds are supported */
assert(dest->mutex == src->mutex);
qemu_mutex_lock(dest->mutex); qemu_mutex_lock(dest->mutex);
if (src->mutex != dest->mutex) {
qemu_mutex_lock(src->mutex);
}
if (bdrv_dirty_bitmap_check(dest, BDRV_BITMAP_DEFAULT, errp)) { if (bdrv_dirty_bitmap_check(dest, BDRV_BITMAP_DEFAULT, errp)) {
goto out; goto out;
@ -845,4 +845,7 @@ void bdrv_merge_dirty_bitmap(BdrvDirtyBitmap *dest, const BdrvDirtyBitmap *src,
out: out:
qemu_mutex_unlock(dest->mutex); qemu_mutex_unlock(dest->mutex);
if (src->mutex != dest->mutex) {
qemu_mutex_unlock(src->mutex);
}
} }

View File

@ -2112,11 +2112,10 @@ static void block_dirty_bitmap_disable_abort(BlkActionState *common)
} }
} }
static BdrvDirtyBitmap *do_block_dirty_bitmap_merge(const char *node, static BdrvDirtyBitmap *do_block_dirty_bitmap_merge(
const char *target, const char *node, const char *target,
strList *bitmaps, BlockDirtyBitmapMergeSourceList *bitmaps,
HBitmap **backup, HBitmap **backup, Error **errp);
Error **errp);
static void block_dirty_bitmap_merge_prepare(BlkActionState *common, static void block_dirty_bitmap_merge_prepare(BlkActionState *common,
Error **errp) Error **errp)
@ -2965,15 +2964,14 @@ void qmp_block_dirty_bitmap_disable(const char *node, const char *name,
bdrv_disable_dirty_bitmap(bitmap); bdrv_disable_dirty_bitmap(bitmap);
} }
static BdrvDirtyBitmap *do_block_dirty_bitmap_merge(const char *node, static BdrvDirtyBitmap *do_block_dirty_bitmap_merge(
const char *target, const char *node, const char *target,
strList *bitmaps, BlockDirtyBitmapMergeSourceList *bitmaps,
HBitmap **backup, HBitmap **backup, Error **errp)
Error **errp)
{ {
BlockDriverState *bs; BlockDriverState *bs;
BdrvDirtyBitmap *dst, *src, *anon; BdrvDirtyBitmap *dst, *src, *anon;
strList *lst; BlockDirtyBitmapMergeSourceList *lst;
Error *local_err = NULL; Error *local_err = NULL;
dst = block_dirty_bitmap_lookup(node, target, &bs, errp); dst = block_dirty_bitmap_lookup(node, target, &bs, errp);
@ -2988,11 +2986,28 @@ static BdrvDirtyBitmap *do_block_dirty_bitmap_merge(const char *node,
} }
for (lst = bitmaps; lst; lst = lst->next) { for (lst = bitmaps; lst; lst = lst->next) {
src = bdrv_find_dirty_bitmap(bs, lst->value); switch (lst->value->type) {
if (!src) { const char *name, *node;
error_setg(errp, "Dirty bitmap '%s' not found", lst->value); case QTYPE_QSTRING:
dst = NULL; name = lst->value->u.local;
goto out; src = bdrv_find_dirty_bitmap(bs, name);
if (!src) {
error_setg(errp, "Dirty bitmap '%s' not found", name);
dst = NULL;
goto out;
}
break;
case QTYPE_QDICT:
node = lst->value->u.external.node;
name = lst->value->u.external.name;
src = block_dirty_bitmap_lookup(node, name, NULL, errp);
if (!src) {
dst = NULL;
goto out;
}
break;
default:
abort();
} }
bdrv_merge_dirty_bitmap(anon, src, NULL, &local_err); bdrv_merge_dirty_bitmap(anon, src, NULL, &local_err);
@ -3012,7 +3027,8 @@ static BdrvDirtyBitmap *do_block_dirty_bitmap_merge(const char *node,
} }
void qmp_block_dirty_bitmap_merge(const char *node, const char *target, void qmp_block_dirty_bitmap_merge(const char *node, const char *target,
strList *bitmaps, Error **errp) BlockDirtyBitmapMergeSourceList *bitmaps,
Error **errp)
{ {
do_block_dirty_bitmap_merge(node, target, bitmaps, NULL, errp); do_block_dirty_bitmap_merge(node, target, bitmaps, NULL, errp);
} }

View File

@ -2003,19 +2003,35 @@
'data': { 'node': 'str', 'name': 'str', '*granularity': 'uint32', 'data': { 'node': 'str', 'name': 'str', '*granularity': 'uint32',
'*persistent': 'bool', '*autoload': 'bool', '*disabled': 'bool' } } '*persistent': 'bool', '*autoload': 'bool', '*disabled': 'bool' } }
##
# @BlockDirtyBitmapMergeSource:
#
# @local: name of the bitmap, attached to the same node as target bitmap.
#
# @external: bitmap with specified node
#
# Since: 4.1
##
{ 'alternate': 'BlockDirtyBitmapMergeSource',
'data': { 'local': 'str',
'external': 'BlockDirtyBitmap' } }
## ##
# @BlockDirtyBitmapMerge: # @BlockDirtyBitmapMerge:
# #
# @node: name of device/node which the bitmap is tracking # @node: name of device/node which the @target bitmap is tracking
# #
# @target: name of the destination dirty bitmap # @target: name of the destination dirty bitmap
# #
# @bitmaps: name(s) of the source dirty bitmap(s) # @bitmaps: name(s) of the source dirty bitmap(s) at @node and/or fully
# specifed BlockDirtyBitmap elements. The latter are supported
# since 4.1.
# #
# Since: 4.0 # Since: 4.0
## ##
{ 'struct': 'BlockDirtyBitmapMerge', { 'struct': 'BlockDirtyBitmapMerge',
'data': { 'node': 'str', 'target': 'str', 'bitmaps': ['str'] } } 'data': { 'node': 'str', 'target': 'str',
'bitmaps': ['BlockDirtyBitmapMergeSource'] } }
## ##
# @block-dirty-bitmap-add: # @block-dirty-bitmap-add: