​ 在之前的文章:《QEMU/KVM启动物理分区的Windows并调优》中笔者使用mdadm创建线性阵列,使VM启动物理硬盘分区上的Windows系统。这个做法思路清晰且具有实操性,但根据这个issue,Linux内核上游已将CONFIG_MD_LINEAR编译参数弃置了,这意味着在6.8及以后的内核中将无法使用mdadm创建线性阵列,相关模块已不再被内核包含,modprobe linear命令将失效。

​ 好在创建线性阵列的方法不止这一种,使用device mapper也可以实现同样的目的。参考这个帖子,这里提供一个新的libvirt hook脚本,使用device mapper动态创建和销毁线性阵列:

#!/usr/bin/env bash
#
# Author: yjzzjy4 (https://github.com/yjzzjy4)
#
# This script creates and distroys /dev/mapper/win10-kvm for booting physical windows drive.
#

WIN_PART=/dev/disk/by-uuid/7CEA3A30EA39E6D4
EFI_DIR=/etc/libvirt/hooks/qemu.d/win10/vdisk

VM_ACTION="$2/$3"

if [[ "$VM_ACTION" == "prepare/begin" ]]; then
	if [[ -e /dev/mapper/win10-kvm ]]; then
		echo "/dev/mapper/win10-kvm already exists" > /dev/kmsg 2>&1
		exit 1
	fi

	if mountpoint -q -- "${WIN_PART}"; then
		echo "Unmounting ${WIN_PART}..." > /dev/kmsg 2>&1
		umount ${WIN_PART}
	fi

	modprobe loop
	table=""
	cur_size=0

	LOOP0=$(losetup -f "${EFI_DIR}/win10-vdisk-loop0" --show)
	sector_size=$(blockdev --getsz $LOOP0)
	table+="$cur_size $sector_size linear $LOOP0 0\n"
	cur_size=$((cur_size+sector_size))

	sector_size=$(blockdev --getsz $WIN_PART)
	table+="$cur_size $sector_size linear $WIN_PART 0\n"
	cur_size=$((cur_size+sector_size))

	LOOP1=$(losetup -f "${EFI_DIR}/win10-vdisk-loop1" --show)
	sector_size=$(blockdev --getsz $LOOP1)
    table+="$cur_size $sector_size linear $LOOP1 0"
    cur_size=$((cur_size+sector_size))

	echo -e "$table" | dmsetup create win10-kvm
elif [[ "$VM_ACTION" == "release/end" ]]; then
	dmsetup remove win10-kvm
	losetup | grep "win10-vdisk" | awk '{print $1}' | xargs sudo losetup -d
fi

​ 使用这个hook脚本替代原文中manage-vdisk.sh即可。相较于之前的脚本,做了几个小优化:

  • 使用device mapper创建线性阵列;
  • 使用UUID标识分区(笔者遇到过在某次重启后分区名称发生改变的情况,使用UUID更准确);
  • 规范化脚本中使用的一些目录和文件命名。

​ 接下来对脚本进行测试,首先是创建线性阵列:

创建阵列

​ 然后是销毁阵列:

销毁阵列

​ 至此,脚本可以作为libvirt hook正常使用,当然,不要忘记修改VM对应的配置,将启动盘设置为/dev/mapper/win10-kvm,如下图所示:

VM配置