From f118923bfeeb670539786c00ffb2ceb2c2080196 Mon Sep 17 00:00:00 2001 From: Demi Marie Obenour Date: Thu, 24 Oct 2024 21:13:02 -0400 Subject: [PATCH] UEFI test UEFI VMs didn't boot. Add a test to ensure they do. --- qubes/tests/__init__.py | 67 ++++++++++++++++++++++++++------------ qubes/tests/integ/basic.py | 35 +++++++++++++++++--- 2 files changed, 78 insertions(+), 24 deletions(-) diff --git a/qubes/tests/__init__.py b/qubes/tests/__init__.py index c20d0c073..c4dd626fb 100644 --- a/qubes/tests/__init__.py +++ b/qubes/tests/__init__.py @@ -1328,33 +1328,60 @@ def prepare_hvm_system_linux(self, vm, init_script, extra_files=None): def create_bootable_iso(self): """Create simple bootable ISO image. - Type 'poweroff' to it to terminate that VM. + Type 'halt' to it to terminate that VM. """ - isolinux_cfg = ( - 'prompt 1\n' - 'label poweroff\n' - ' kernel poweroff.c32\n' - ) output_fd, output_path = tempfile.mkstemp('.iso') with tempfile.TemporaryDirectory() as tmp_dir: + f = os.open(tmp_dir, os.O_RDONLY|os.O_DIRECTORY) try: - shutil.copy('/usr/share/syslinux/isolinux.bin', tmp_dir) - shutil.copy('/usr/share/syslinux/ldlinux.c32', tmp_dir) - shutil.copy('/usr/share/syslinux/poweroff.c32', tmp_dir) - with open(os.path.join(tmp_dir, 'isolinux.cfg'), 'w') as cfg: - cfg.write(isolinux_cfg) - subprocess.check_call(['genisoimage', '-o', output_path, - '-c', 'boot.cat', - '-b', 'isolinux.bin', + os.mkdir('os', mode=0o755, dir_fd=f) + os.mkdir('iso', mode=0o755, dir_fd=f) + os.mkdir('os/images', mode=0o755, dir_fd=f) + os.mkdir('os/EFI', mode=0o755, dir_fd=f) + os.mkdir('os/EFI/BOOT', mode=0o755, dir_fd=f) + with open(f'/proc/self/fd/{f}/os/images/boot_hybrid.img', 'wb') as v: + subprocess.check_call(['grub2-mkimage', + '-O', 'i386-pc-eltorito', + '-d', '/usr/lib/grub/i386-pc', + '-p', '/boot/grub2', + 'iso9660', 'biosdisk', 'halt'], stdout=v) + with open(f'/proc/self/fd/{f}/os/EFI/BOOT/BOOTX64.EFI', 'wb') as v: + subprocess.check_call(['grub2-mkimage', + '-O', 'x86_64-efi', + '-d', '/usr/lib/grub/x86_64-efi', + '-p', '/boot/grub2', + 'iso9660', 'disk', 'halt'], stdout=v) + graft_path = '.=' + os.path.join(tmp_dir, "os").replace("\\", "\\\\").replace("=", "\\=") + efiboot_img = os.path.join(tmp_dir, 'os/images/efiboot.img') + subprocess.check_call(['sudo', 'mkefiboot', '--label=ANACONDA', + os.path.join(tmp_dir, 'os/EFI/BOOT'), + efiboot_img]) + subprocess.check_call(['sudo', 'chmod', '-R', 'go+rX', tmp_dir]) + subprocess.check_call(['xorrisofs', + '-o', output_path, + '--grub2-mbr', 'os/images/boot_hybrid.img', + '--mbr-force-bootable', + '--gpt-iso-bootable', + '-partition_offset', '16', + '-append_partition', '2', '0xef', efiboot_img, + '-iso_mbr_part_type', 'EBD0A0A2-B9E5-4433-87C0-68B6B72699C7', + '-appended_part_as_gpt', + '-c', 'boot.cat', '--boot-catalog-hide', + '-eltorito-boot', 'images/boot_hybrid.img', '-no-emul-boot', '-boot-load-size', '4', '-boot-info-table', - '-q', - tmp_dir]) - except FileNotFoundError: - self.skipTest('syslinux or genisoimage not installed') - os.close(output_fd) - self.addCleanup(os.unlink, output_path) + '--grub2-boot-info', + '-eltorito-alt-boot', + '-e', '--interval:appended_partition_2:all::', + '-no-emul-boot', + '-graft-points', graft_path, + 'boot/grub2/i386-pc=/usr/lib/grub/i386-pc', + 'boot/grub2/x86_64-efi=/usr/lib/grub/x86_64-efi'], + cwd=tmp_dir) + self.addCleanup(os.unlink, output_path) + finally: + os.close(f) return output_path def create_local_file(self, filename, content, mode='w'): diff --git a/qubes/tests/integ/basic.py b/qubes/tests/integ/basic.py index 1f574dd3e..83f89ad46 100644 --- a/qubes/tests/integ/basic.py +++ b/qubes/tests/integ/basic.py @@ -105,15 +105,42 @@ def test_120_start_standalone_with_cdrom_dom0(self): # check if VM do not crash instantly self.loop.run_until_complete(asyncio.sleep(5)) self.assertTrue(self.vm.is_running()) - # Type 'poweroff' + # Type 'halt' subprocess.check_call(['xdotool', 'search', '--name', self.vm.name, - 'type', '--window', '%1', 'poweroff\r']) + 'type', '--window', '%1', 'halt\r']) for _ in range(10): if not self.vm.is_running(): break self.loop.run_until_complete(asyncio.sleep(1)) self.assertFalse(self.vm.is_running()) + def test_121_start_uefi(self): + vmname = self.make_vm_name('appvm') + self.vm = self.app.add_new_vm('StandaloneVM', label='red', name=vmname) + self.loop.run_until_complete(self.vm.create_on_disk()) + self.vm.kernel = None + self.vm.virt_mode = 'hvm' + self.vm.features["uefi"] = "1" + iso_path = self.create_bootable_iso() + # start the VM using qvm-start tool, to test --cdrom option there + p = self.loop.run_until_complete(asyncio.create_subprocess_exec( + 'qvm-start', '--cdrom=dom0:' + iso_path, self.vm.name, + stdout=subprocess.PIPE, stderr=subprocess.STDOUT)) + (stdout, _) = self.loop.run_until_complete(p.communicate()) + self.assertEqual(p.returncode, 0, stdout) + # check if VM do not crash instantly + self.loop.run_until_complete(asyncio.sleep(5)) + self.assertTrue(self.vm.is_running()) + # Type 'halt' + subprocess.check_call(['xdotool', 'search', '--name', self.vm.name, + 'type', '--window', '%1', 'halt\r']) + for _ in range(10): + if not self.vm.is_running(): + break + self.loop.run_until_complete(asyncio.sleep(1)) + self.assertFalse(self.vm.is_running()) + + def test_130_autostart_disable_on_remove(self): vm = self.app.add_new_vm(qubes.vm.appvm.AppVM, name=self.make_vm_name('vm'), @@ -669,9 +696,9 @@ def test_121_start_standalone_with_cdrom_vm(self): # check if VM do not crash instantly self.loop.run_until_complete(asyncio.sleep(5)) self.assertTrue(self.vm.is_running()) - # Type 'poweroff' + # Type 'halt' subprocess.check_call(['xdotool', 'search', '--name', self.vm.name, - 'type', '--window', '%1', 'poweroff\r']) + 'type', '--window', '%1', 'halt\r']) for _ in range(10): if not self.vm.is_running(): break