我希望从由环回文件托管的 LVM 分区引导我的系统,其中引导位于 USB 驱动器上。
使用非 LVM 设置,带有 ext4 环回文件,它启动没有问题(网上有几个指南),但我需要更多......我需要 LVM!
我的分区布局是:
- USB 驱动器 (MBR):Grub 2
- USB驱动器(hd0,msdos1):ext4 512Mb /boot
- 笔记本内部磁盘:hosts .linux-loops/system0.lvm文件,我在其中安装了操作系统并将其映射为/dev/mapper/vg_system-lv_root。
这是我的 Grub 2 菜单项:
set BOOT_PART=(hd0,msdos1)
set HOST_PART=(hd1,gpt5)
set LOOP_FILE=.linux-loops/system0.lvm
set LOOP_DEV=loop0
set LVM_VG=vg_system
set LVM_LV=lv_root
set ROOT_DEV=/dev/mapper/${LVM_VG}-${LVM_LV}
menuentry 'Loopback / 4.10.0-38-generic' --class ubuntu --class gnu-linux --class gnu --class os {
echo "Initializing environment..."
set KERN_VER=4.10.0-38-generic
recordfail load_video gfxmode $linux_gfx_mode
echo "Loading partition drivers..."
insmod ext2
insmod gzio
insmod part_msdos
insmod part_gpt
echo Loopback(s) setup...
loopback ${LOOP_DEV} ${HOST_PART}/${LOOP_FILE}
insmod lvm
echo Debug LVM...
ls
ls (lvm/${LVM_VG}/${LVM_LV)/
echo "Loading kernel..."
set root=${BOOT_PART}
linux /vmlinuz-${KERN_VER} root=${ROOT_DEV} rw verbose nosplash debug
initrd /initrd.img-${KERN_VER}
}
两个ls调试命令显示 grub 已经看到了 LVM 分区,但是在initrd运行期间,系统显示未找到组vg_system并且lvmetad尚未运行。
我认为 grub 脚本中的环回设置无法传播到 init 脚本,因此它无法再次初始化 LVM 映射器。
在非 LVM 场景(上面提到)中,vmlinuz ... loop=...解决了这个问题。
问题是:如何设置环回文件以便通过 grub 将它们提供给 LVM 映射器?
太感谢了!
您的想法是正确的:GRUB 不能代表内核进行设置。
GRUB 依赖系统固件(BIOS 或 UEFI)进行磁盘访问,因此它所做的任何事情都基于这些固件例程。一旦内核接管,这些固件例程就不再使用了……在它们之上构建的任何东西也变得不可用,直到操作系统为相同的工作提供自己的驱动程序。
(在 BIOS 的情况下,随着内核将处理器转换到完整的 32/64 位模式,16 位 BIOS 磁盘访问例程变得不可用。在 UEFI 的情况下,我认为磁盘访问服务是 UEFI 功能之一当执行标准的“终止启动服务”UEFI 调用时变得不可用,从而完成了将硬件控制权移交给操作系统的 UEFI 端。)
相反,您需要在很早的阶段设置一个要在 initrd 中执行的脚本:它应该在 LVM 启动之前激活必要的回送设备,或者如果 LVM 是在设置回送设备之后执行
vgscan --mknodes
和/或vgchange -ay <name of the VG containing the root filesystem>
已经初始化。在不知道您使用的 Linux 发行版的名称和版本的情况下,很难给出更具体的建议。
我在 GitHub 上发布了我的解决方案:
本指南将为您提供有关如何在从 USB 驱动器(grub 和引导分区)引导的 LVM 环回磁盘上安装 linux 的详细说明,而无需更改 PC 内部磁盘引导扇区。
相关部分已由https://github.com/antonio-petricca/buddy-linux/blob/master/assets/initramfs/lvm-loops-setup脚本实现: