diff --git a/lib/vdsm/api/vdsm-api.yml b/lib/vdsm/api/vdsm-api.yml index 2382ecfb7a..3055e5d938 100644 --- a/lib/vdsm/api/vdsm-api.yml +++ b/lib/vdsm/api/vdsm-api.yml @@ -557,6 +557,46 @@ types: 1.1: qemu versions starting with 1.1 which are supported by QCOW2 version 3. + Qcow2BitmapInfoFlags: &Qcow2BitmapInfoFlags + added: '4.5.5' + description: An enumeration of qcow compat version. + name: Qcow2BitmapInfoFlags + type: enum + values: + 'in-use': This flag is set by any process actively modifying the + qcow2 file, and cleared when the updated bitmap is flushed + to the qcow2 image. The presence of this flag in an + offline image means that the bitmap was not saved correctly + after its last usage, and may contain inconsistent data. + 'auto': The bitmap must reflect all changes of the virtual disk by + any application that would write to this qcow2 file. + + Qcow2BitmapInfo: &Qcow2BitmapInfo + added: '4.5.5' + description: Information about a Bitmap + name: Qcow2BitmapInfo + properties: + - description: The UUID of bitmap + name: name + type: *UUID + + - description: Granularity of the bitmap + name: granularity + type: uint + + - description: Bitmap flags + name: flags + defaultvalue: '[]' + type: *Qcow2BitmapInfoFlags + type: object + + Qcow2Bitmaps: &Qcow2Bitmaps + added: '4.5.5' + description: A list of qcow2 bitmaps + name: Qcow2Bitmaps + defaultvalue: '[]' + type: *Qcow2BitmapInfo + Qcow2Attributes: &Qcow2Attributes added: '4.1' description: Possible QCOW2 attributes which are allowed @@ -8131,6 +8171,9 @@ types: - description: The compat version. name: compat type: *Qcow2Compat + - description: Volume bitmaps + name: bitmaps + type: *Qcow2Bitmaps type: object VolumeSizeInfo: &VolumeSizeInfo diff --git a/lib/vdsm/host/caps.py b/lib/vdsm/host/caps.py index c6ae827bc8..87b87d5d3c 100644 --- a/lib/vdsm/host/caps.py +++ b/lib/vdsm/host/caps.py @@ -175,6 +175,7 @@ def get(): caps['measure_active'] = True caps['mailbox_events'] = config.getboolean("mailbox", "events_enable") caps['zerocopy_migrations'] = hasattr(libvirt, 'VIR_MIGRATE_ZEROCOPY') + caps['qemu_image_info_bitmaps'] = True return caps diff --git a/lib/vdsm/storage/volume.py b/lib/vdsm/storage/volume.py index 066fc69eb9..88551b9d16 100644 --- a/lib/vdsm/storage/volume.py +++ b/lib/vdsm/storage/volume.py @@ -5,6 +5,7 @@ import os.path import logging +from collections import namedtuple from contextlib import contextmanager from vdsm import utils @@ -27,6 +28,8 @@ log = logging.getLogger('storage.volume') +Qcow2BitmapInfo = namedtuple("Qcow2BitmapInfo", "name, granularity, flags") + def getBackingVolumePath(imgUUID, volUUID): # We used to return a relative path ..// but this caused @@ -289,6 +292,13 @@ def getQemuImageInfo(self): result["backingfile"] = info["backing-filename"] if "format-specific" in info: result["compat"] = info["format-specific"]["data"]["compat"] + if "bitmaps" in info["format-specific"]["data"]: + result["bitmaps"] = [ + Qcow2BitmapInfo(bitmap["name"], + bitmap["granularity"], + bitmap["flags"]) + for bitmap in info["format-specific"]["data"]["bitmaps"] + ] return result diff --git a/tests/storage/bitmaps_test.py b/tests/storage/bitmaps_test.py index 02046253fe..553c6d3918 100644 --- a/tests/storage/bitmaps_test.py +++ b/tests/storage/bitmaps_test.py @@ -82,6 +82,11 @@ def test_add_only_valid_bitmaps(vol_chain): }, ] + qemuInfo = vol_chain.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmap, 65536), + ] + def test_no_bitmaps_to_add(vol_chain): # Add bitmaps from base volume to top volume @@ -90,6 +95,9 @@ def test_no_bitmaps_to_add(vol_chain): info = qemuimg.info(vol_chain.top_vol) assert 'bitmaps' not in info['format-specific']['data'] + qemuInfo = vol_chain.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + def test_add_bitmap_failed(monkeypatch, vol_chain): # Add new bitmap to base volume @@ -134,6 +142,11 @@ def test_merge_only_valid_bitmaps(vol_chain): }, ] + qemuInfo = vol_chain.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmap, 65536), + ] + def test_no_bitmaps_to_merge(vol_chain): # Merge bitmaps from base volume to top volume @@ -142,6 +155,9 @@ def test_no_bitmaps_to_merge(vol_chain): info = qemuimg.info(vol_chain.base_vol) assert 'bitmaps' not in info['format-specific']['data'] + qemuInfo = vol_chain.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + def test_merge_bitmaps_failed(monkeypatch, vol_chain): bitmap = 'bitmap' @@ -164,6 +180,11 @@ def test_merge_bitmaps_failed(monkeypatch, vol_chain): }, ] + qemuInfo = vol_chain.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmap, 65536), + ] + def test_merge_bitmaps_failed_to_add_bitmap( monkeypatch, vol_chain): @@ -180,6 +201,9 @@ def test_merge_bitmaps_failed_to_add_bitmap( info = qemuimg.info(vol_chain.base_vol) assert 'bitmaps' not in info['format-specific']['data'] + qemuInfo = vol_chain.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + def test_skip_holes_during_merge_bitmaps(tmp_mount, vol_chain): virtual_size = MiB @@ -219,6 +243,9 @@ def test_skip_holes_during_merge_bitmaps(tmp_mount, vol_chain): info = qemuimg.info(vol_chain.base_vol) assert 'bitmaps' not in info['format-specific']['data'] + qemuInfo = vol_chain.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + def test_remove_bitmap_failed(monkeypatch, tmp_mount, vol_chain): bitmap = 'bitmap' @@ -257,6 +284,11 @@ def test_prune_stale_bitmaps(tmp_mount, vol_chain): }, ] + qemuInfo = vol_chain.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmap, 65536), + ] + def test_prune_disabled_bitmaps(tmp_mount, vol_chain): # Add valid bitmap to volumes @@ -282,6 +314,11 @@ def test_prune_disabled_bitmaps(tmp_mount, vol_chain): }, ] + qemuInfo = vol_chain.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmap, 65536), + ] + def test_prune_in_use_bitmaps(tmp_mount, vol_chain): # Add inconsistent "in-use" bitmaps to volumes @@ -307,6 +344,11 @@ def test_prune_in_use_bitmaps(tmp_mount, vol_chain): }, ] + qemuInfo = vol_chain.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmap, 65536), + ] + def test_clear_bitmaps(tmp_mount, vol_chain): bitmap_1 = 'bitmap_1' @@ -324,6 +366,9 @@ def test_clear_bitmaps(tmp_mount, vol_chain): vol_bitmaps = info["format-specific"]["data"].get("bitmaps", []) assert not vol_bitmaps + qemuInfo = vol_chain.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + def test_clear_bitmaps_failed(monkeypatch, tmp_mount, vol_chain): # Add new bitmap to top volume diff --git a/tests/storage/blocksd_test.py b/tests/storage/blocksd_test.py index 63dc6055c9..3929bc8cf8 100644 --- a/tests/storage/blocksd_test.py +++ b/tests/storage/blocksd_test.py @@ -29,6 +29,7 @@ from vdsm.storage import sanlock_direct from vdsm.storage import sd from vdsm.storage import sp +from vdsm.storage.volume import Qcow2BitmapInfo from vdsm.storage.sdc import sdCache from vdsm.storage.sdm.api import merge as api_merge from vdsm.storage.spbackends import StoragePoolDiskBackend @@ -49,7 +50,6 @@ Chain = namedtuple("Chain", ["base", "internal", "top"]) - class TestMetadataValidity: MIN_MD_SIZE = blockSD.VG_METADATASIZE * MiB // 2 @@ -1109,6 +1109,12 @@ def test_create_snapshot_cloning_bitmaps( }, ] + qemuInfo = top_vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo(bitmap_names[0], 65536, ["auto"]), + Qcow2BitmapInfo(bitmap_names[1], 65536, ["auto"]), + ] + @requires_root @pytest.mark.root @@ -1193,6 +1199,12 @@ def test_create_snapshot_with_new_bitmap( }, ] + qemuInfo = top.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo("old-bitmap", 65536, ["auto"]), + Qcow2BitmapInfo("new-bitmap", 65536, ["auto"]), + ] + @requires_root @pytest.mark.root @@ -1250,6 +1262,11 @@ def test_create_volume_with_new_bitmap( }, ] + qemuInfo = vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo("new-bitmap", 65536, ["auto"]), + ] + @requires_root @pytest.mark.root diff --git a/tests/storage/localfssd_test.py b/tests/storage/localfssd_test.py index 5d969fab1f..107f8ba4fb 100644 --- a/tests/storage/localfssd_test.py +++ b/tests/storage/localfssd_test.py @@ -20,6 +20,7 @@ from vdsm.storage import localFsSD from vdsm.storage import qemuimg from vdsm.storage import sd +from vdsm.storage.volume import Qcow2BitmapInfo from . import qemuio from . marks import requires_unprivileged_user @@ -1316,6 +1317,12 @@ def test_create_snapshot_cloning_bitmaps(user_domain, local_fallocate): }, ] + qemuInfo = vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo(bitmap_names[0], 65536, ["auto"]), + Qcow2BitmapInfo(bitmap_names[1], 65536, ["auto"]), + ] + def test_create_snapshot_with_new_bitmap(user_domain, local_fallocate): if user_domain.getVersion() == 3: @@ -1374,6 +1381,12 @@ def test_create_snapshot_with_new_bitmap(user_domain, local_fallocate): }, ] + qemuInfo = top.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo("old-bitmap", 65536, ["auto"]), + Qcow2BitmapInfo("new-bitmap", 65536, ["auto"]), + ] + def test_create_volume_with_new_bitmap(user_domain, local_fallocate): if user_domain.getVersion() == 3: @@ -1408,6 +1421,11 @@ def test_create_volume_with_new_bitmap(user_domain, local_fallocate): }, ] + qemuInfo = vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo("new-bitmap", 65536, ["auto"]), + ] + def test_fail_add_bitmaps_to_v3_domain(user_domain, local_fallocate): if user_domain.getVersion() != 3: diff --git a/tests/storage/qemuimg_test.py b/tests/storage/qemuimg_test.py index 171f6f8dd5..e02834e9c4 100644 --- a/tests/storage/qemuimg_test.py +++ b/tests/storage/qemuimg_test.py @@ -774,6 +774,12 @@ def test_copy_bitmaps(self, tmp_mount): }, ] + qemuInfo = vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + (["auto"], bitmaps[0], 65536), + (["auto"], bitmaps[1], 65536), + ] + def test_convert_without_copy_bitmaps(self, tmp_mount): virtual_size = MiB @@ -814,6 +820,7 @@ def test_convert_without_copy_bitmaps(self, tmp_mount): info = qemuimg.info(dst) assert 'bitmaps' not in info['format-specific']['data'] + def test_copy_with_disabled_bitmaps(self, tmp_mount): virtual_size = MiB bitmaps = [("a", True), ("b", False)] diff --git a/tests/storage/sdm_add_bitmap_test.py b/tests/storage/sdm_add_bitmap_test.py index 5c7714a812..0bb9db3c7e 100644 --- a/tests/storage/sdm_add_bitmap_test.py +++ b/tests/storage/sdm_add_bitmap_test.py @@ -28,6 +28,7 @@ from vdsm.storage import exception as se from vdsm.storage import guarded from vdsm.storage import qemuimg +from vdsm.storage.volume import Qcow2BitmapInfo from vdsm.storage.sdm import volume_info from vdsm.storage.sdm.api import add_bitmap @@ -76,6 +77,11 @@ def test_add_bitmap(fake_scheduler, env_type): assert qcow2_data["bitmaps"][0]["name"] == bitmap assert env_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = env_vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo(bitmap, 65536, ["auto"]), + ] + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_vol_type_not_qcow(fake_scheduler, env_type): diff --git a/tests/storage/sdm_clear_bitmaps_test.py b/tests/storage/sdm_clear_bitmaps_test.py index e8ff2ec82b..7de23cfb77 100644 --- a/tests/storage/sdm_clear_bitmaps_test.py +++ b/tests/storage/sdm_clear_bitmaps_test.py @@ -79,6 +79,9 @@ def test_clear_bitmaps(fake_scheduler, env_type): assert "bitmaps" not in vol_info["format-specific"]["data"] assert top_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = top_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_clear_invalid_bitmaps(fake_scheduler, env_type): @@ -110,6 +113,9 @@ def test_clear_invalid_bitmaps(fake_scheduler, env_type): assert "bitmaps" not in vol_info["format-specific"]["data"] assert top_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = top_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_vol_type_not_qcow(fake_scheduler, env_type): @@ -179,6 +185,9 @@ def test_clear_missing_bitmaps(fake_scheduler, env_type): assert not bitmaps assert top_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = top_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_clear_bitmaps_from_vol_chain(fake_scheduler, env_type): @@ -214,6 +223,9 @@ def test_clear_bitmaps_from_vol_chain(fake_scheduler, env_type): assert not bitmaps assert leaf_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = leaf_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + # Clear all the bitmaps from an internal volume internal_vol = env.chain[1] generation = internal_vol.getMetaParam(sc.GENERATION) @@ -234,3 +246,6 @@ def test_clear_bitmaps_from_vol_chain(fake_scheduler, env_type): bitmaps = vol_info["format-specific"]["data"].get("bitmaps", []) assert not bitmaps assert internal_vol.getMetaParam(sc.GENERATION) == generation + 1 + + qemuInfo = internal_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo diff --git a/tests/storage/sdm_copy_data_test.py b/tests/storage/sdm_copy_data_test.py index a590f81c1d..9e2ce701f0 100644 --- a/tests/storage/sdm_copy_data_test.py +++ b/tests/storage/sdm_copy_data_test.py @@ -47,6 +47,7 @@ from vdsm.storage import resourceManager as rm from vdsm.storage import volume from vdsm.storage import workarounds +from vdsm.storage.volume import Qcow2BitmapInfo from vdsm.storage.sdm.api import copy_data BACKENDS = userstorage.load_config("storage.py").BACKENDS @@ -302,6 +303,12 @@ def test_volume_chain_copy_with_bitmaps( }, ] + qemuInfo = dst_vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo(bitmaps[0], 65536, ["auto"]), + Qcow2BitmapInfo(bitmaps[1], 65536, ["auto"]), + ] + @pytest.mark.parametrize( "env_type, dst_fmt", [ diff --git a/tests/storage/sdm_merge_test.py b/tests/storage/sdm_merge_test.py index 2338a37acb..319c3c0ddb 100644 --- a/tests/storage/sdm_merge_test.py +++ b/tests/storage/sdm_merge_test.py @@ -3,6 +3,7 @@ from __future__ import absolute_import from __future__ import division +from collections import namedtuple from contextlib import contextmanager @@ -37,6 +38,7 @@ from vdsm.storage import qemuimg from vdsm.storage import resourceManager as rm from vdsm.storage import volume +from vdsm.storage.volume import Qcow2BitmapInfo from vdsm.storage.sdm.api import merge as api_merge @@ -275,3 +277,9 @@ def test_merge_subchain_with_bitmaps( "granularity": 65536 }, ] + + qemuInfo = base_vol.getQemuImageInfo() + assert qemuInfo["bitmaps"] == [ + Qcow2BitmapInfo(bitmap1_name, 65536, ["auto"]), + Qcow2BitmapInfo(bitmap2_name, 65536, ["auto"]), + ] diff --git a/tests/storage/sdm_remove_bitmap_test.py b/tests/storage/sdm_remove_bitmap_test.py index db7e04a979..8eea0fce09 100644 --- a/tests/storage/sdm_remove_bitmap_test.py +++ b/tests/storage/sdm_remove_bitmap_test.py @@ -87,6 +87,11 @@ def test_add_remove_bitmap(fake_scheduler, env_type): assert bitmap1 not in bitmaps and bitmap2 in bitmaps assert top_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = top_vol.getQemuImageInfo() + assert not any(bitmap[0] == bitmap1 for bitmap in qemuInfo["bitmaps"]) + assert any(bitmap[0] == bitmap2 for bitmap in qemuInfo["bitmaps"]) + + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_vol_type_not_qcow(fake_scheduler, env_type): @@ -143,6 +148,10 @@ def test_remove_bitmap_non_leaf_vol(fake_scheduler, env_type): assert bitmap1 not in bitmaps and bitmap2 in bitmaps assert base_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = base_vol.getQemuImageInfo() + assert not any(bitmap[0] == bitmap1 for bitmap in qemuInfo["bitmaps"]) + assert any(bitmap[0] == bitmap2 for bitmap in qemuInfo["bitmaps"]) + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_remove_missing_bitmap(fake_scheduler, env_type): @@ -165,6 +174,9 @@ def test_remove_missing_bitmap(fake_scheduler, env_type): assert not bitmaps assert top_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = top_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_remove_inactive_bitmap(fake_scheduler, env_type): @@ -198,6 +210,9 @@ def test_remove_inactive_bitmap(fake_scheduler, env_type): assert not bitmaps assert base_vol.getMetaParam(sc.GENERATION) == generation + 1 + qemuInfo = base_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo + @pytest.mark.parametrize("env_type", ["file", "block"]) def test_remove_invalid_bitmap(fake_scheduler, env_type): @@ -233,3 +248,6 @@ def test_remove_invalid_bitmap(fake_scheduler, env_type): bitmaps = vol_info["format-specific"]["data"].get("bitmaps", []) assert not bitmaps assert base_vol.getMetaParam(sc.GENERATION) == generation + 1 + + qemuInfo = base_vol.getQemuImageInfo() + assert 'bitmaps' not in qemuInfo