From af5208a21a9360aa6706153647daf93cb4a829e5 Mon Sep 17 00:00:00 2001 From: chunfuwen Date: Fri, 25 Aug 2023 02:36:18 -0400 Subject: [PATCH] Automate disk io_uring attribute case xx-298346: Start or hot-plug a disk with io=io_uring BZ1946936 Signed-off-by: chunfuwen --- .../virtual_disks/virtual_disks_io_tuning.cfg | 17 ++ .../virtual_disks/virtual_disks_io_tuning.py | 194 ++++++++++++++++++ 2 files changed, 211 insertions(+) create mode 100755 libvirt/tests/cfg/virtual_disks/virtual_disks_io_tuning.cfg create mode 100644 libvirt/tests/src/virtual_disks/virtual_disks_io_tuning.py diff --git a/libvirt/tests/cfg/virtual_disks/virtual_disks_io_tuning.cfg b/libvirt/tests/cfg/virtual_disks/virtual_disks_io_tuning.cfg new file mode 100755 index 0000000000..73deead6f3 --- /dev/null +++ b/libvirt/tests/cfg/virtual_disks/virtual_disks_io_tuning.cfg @@ -0,0 +1,17 @@ +- virtual_disks.io_tuning: + type = virtual_disks_io_tuning + status_error = "no" + start_vm = "no" + variants tuning_type: + - io_uring: + func_supported_since_libvirt_ver = (9, 3, 0) + variants test_scenario: + - normal_start: + driver_attribute = {'name': "qemu", 'type': "qcow2", 'queue_size': '1024', 'io': 'io_uring'} + source_file_path = "/var/lib/libvirt/images/io_uring.qcow2" + target_device = "sdb" + variants plug_mode: + - coldplug: + coldplug = "yes" + - hotplug: + coldplug = "no" diff --git a/libvirt/tests/src/virtual_disks/virtual_disks_io_tuning.py b/libvirt/tests/src/virtual_disks/virtual_disks_io_tuning.py new file mode 100644 index 0000000000..414c54b319 --- /dev/null +++ b/libvirt/tests/src/virtual_disks/virtual_disks_io_tuning.py @@ -0,0 +1,194 @@ +# +# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ +# +# Copyright Red Hat +# +# SPDX-License-Identifier: GPL-2.0 + +# Author: Chunfu Wen +# +# ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +import logging +import os + +from avocado.utils import process + +from virttest import libvirt_version +from virttest import virt_vm +from virttest import virsh + +from virttest.libvirt_xml import vm_xml, xcepts + +from virttest.utils_test import libvirt +from virttest.utils_libvirt import libvirt_disk + +LOG = logging.getLogger('avocado.' + __name__) +cleanup_files = [] + + +def create_one_disk(params): + """ + create one disk xml + + :param params: dict wrapped with params + """ + # Prepare disk source xml + source_file_path = params.get("source_file_path") + libvirt.create_local_disk("file", source_file_path, 1, + disk_format="qcow2") + + cleanup_files.append(source_file_path) + + disk_src_dict = {"attrs": {"file": source_file_path}} + target_device = params.get("target_device") + + customized_disk = libvirt_disk.create_primitive_disk_xml( + "file", "disk", + target_device, 'scsi', + 'qcow2', disk_src_dict, None) + + driver_dict = params.get("driver_attribute") + customized_disk.driver = eval(driver_dict) + + LOG.info("disk xml is: %s", customized_disk) + return customized_disk + + +def pre_check_host_condition(test): + """ + Check host whether it meets basic configuration + + :param test: one test object instance + """ + # Enable io_uring + enable_cmd_sysctl = "sysctl kernel.io_uring_disabled=0" + process.run(enable_cmd_sysctl, shell=True) + # check whether system control output contains kernel.io_uring_disabled = 0, otherwise skip test + cmd_sysctl = "sysctl kernel.io_uring_disabled" + sysctl_output = process.run(cmd_sysctl, shell=True).stdout_text.strip() + if "kernel.io_uring_disabled = 0" not in sysctl_output: + test.cancel("please enable io_uring by sysctl kernel.io_uring_disabled=0 considering previous output: %s" % sysctl_output) + + +def test_coldplug_io_uring_normal_start(test, params, env): + """ + Test coldplug scsi device with io_uring in VM + + :param test: one test object instance + :param params: dict wrapped with params + :param env: environment instance + """ + vm_name = params.get("main_vm") + vm = env.get_vm(vm_name) + + device_obj = create_one_disk(params) + params.update({'device_obj': device_obj}) + virsh.attach_device(vm_name, device_obj.xml, flagstr="--config", + ignore_status=False) + vm.start() + vm.wait_for_login().close() + + +def test_hotplug_io_uring_normal_start(test, params, env): + """ + Test hotplug scsi device with io_uring in VM + + :param test: one test object instance + :param params: dict wrapped with params + :param env: environment instance + """ + vm_name = params.get("main_vm") + vm = env.get_vm(vm_name) + vm.start() + vm.wait_for_login().close() + device_obj = create_one_disk(params) + params.update({'device_obj': device_obj}) + virsh.attach_device(vm_name, device_obj.xml, flagstr="--live", + ignore_status=False) + + +def check_io_uring_device(test, params, env): + """ + Check io_uring device in VM xml + + :param test: one test object instance + :param params: dict wrapped with params + :param env: environment instance + """ + vm_name = params.get("main_vm") + + io_uring_xml = vm_xml.VMXML.new_from_dumpxml(vm_name) + disk_list = io_uring_xml.devices.by_device_tag("disk") + + # Get the disks driver attribute's value. + for disk in disk_list: + if disk.target['dev'] != params.get('target_device'): + continue + driver_dict = disk.driver + LOG.debug("driver value:%s", driver_dict) + if driver_dict.get("io") is None: + test.fail("failed to get io attribute from: %s" % driver_dict) + elif driver_dict.get("io") != "io_uring": + test.fail("get wrong io attribute from : %s" % driver_dict) + + virsh.detach_device(vm_name, params.get('device_obj').xml, flagstr="--live", + ignore_status=False) + + +def run(test, params, env): + """ + Test disk io tuning related scenarios + + 1.Prepare test environment + 2.Prepare disk image with io tuning attribute + 3.Start the domain. + 4.Perform test operation and check result + 5.Recover test environment. + """ + # Skip test if version not match expected one + libvirt_version.is_libvirt_feature_supported(params) + vm_name = params.get("main_vm") + vm = env.get_vm(vm_name) + virsh_dargs = {'debug': True, 'ignore_status': True} + + # Destroy VM first. + if vm.is_alive(): + vm.destroy(gracefully=False) + + # Back up xml file. + vmxml_backup = vm_xml.VMXML.new_from_inactive_dumpxml(vm_name) + + coldplug = "yes" == params.get("coldplug") + define_error = "yes" == params.get("define_error", "no") + + plug_mode = params.get("plug_mode") + tuning_type = params.get("tuning_type") + test_scenario = params.get("test_scenario") + run_test_case = eval("test_%s_%s_%s" % (plug_mode, tuning_type, test_scenario)) + + try: + pre_check_host_condition(test) + run_test_case(test, params, env) + except virt_vm.VMStartError as e: + test.fail("VM failed to start." + "Error: %s" % str(e)) + except xcepts.LibvirtXMLError as xml_error: + if not define_error: + test.fail("Failed to define VM:\n%s" % str(xml_error)) + else: + if params.get('error_msg') not in str(xml_error): + test.fail("Get unexpected error message from: %s" % str(xml_error)) + else: + if test_scenario == "normal_start": + check_io_uring_device(test, params, env) + finally: + # Recover VM. + if vm.is_alive(): + vm.destroy(gracefully=False) + LOG.info("Restoring vm...") + vmxml_backup.sync() + # Clean up images + for file_path in cleanup_files: + if os.path.exists(file_path): + os.remove(file_path)