本文作者是一位开源理念的坚定支持者,所以本文虽然不是软件,但是遵照开源的精神发布。
本文作者十分愿意与他人分享劳动成果,如果你对我的其他翻译作品或者技术文章有兴趣,可以在如下位置查看现有的作品集:
由于作者水平有限,因此不能保证作品内容准确无误。如果你发现了作品中的错误(哪怕是错别字也好),请来信指出,任何提高作品质量的建议我都将虚心接纳。
对于GRUB-2.0.2版本来说,官方提供的模块一共有200多个,这些模块大致可以分为以下几类(模块间的依赖关系位于"moddep.lst"文件中):
GRUB2在BIOS平台上的常规启动步骤是这样的:BIOS --> boot.img[MBR] --> core.img[MBR gap/embedding area/BIOS Boot Partition] --> 设置"prefix root cmdpath"环境变量 --> 加载"normal.mod"模块[同时还包括它所依赖的 terminal crypto extcmd boot gettext 模块] --> 执行"normal $prefix/grub.cfg"命令
GRUB2在UEFI平台上的常规启动步骤是这样的:UEFI --> core.img[BOOTX64.EFI/BOOTX86.EFI] --> 设置"prefix root cmdpath"环境变量 --> 加载"normal.mod"模块[同时还包括它所依赖的 terminal crypto extcmd boot gettext 模块] --> 执行"normal $prefix/grub.cfg"命令
如果上述步骤全部成功,那么你将进入'普通模式',一般是显示一个菜单(找到了'$prefix/grub.cfg'),或者直接进入GRUB SHELL(没找到'$prefix/grub.cfg')。在普通模式中,命令模块[command.lst]与加密模块[crypto.lst]会被自动按需载入(无需使用"insmod"命令),并且可使用完整的GRUB脚本功能。但是其他模块则可能需要明确使用"insmod"命令来载入。
如果在加载"normal.mod"模块这一步出现故障,那么你将进入GRUB2的'救援模式',而不是常规的'普通模式'。在救援模式中,GRUB只自动设置了"cmdpath prefix root"三个环境变量,并且只能使用"insmod ls set unset"四个命令。只有当额外的模块被加载之后,才可以使用一些其它的命令,变量,解析器,驱动程序。通常来说,进入救援模式可能意味着你的GRUB2没有正确安装。请认真阅读'grub-install --help'的输出选项,并使用正确的选项重新安装。更多细节请参考GRUB2手册中的"GRUB only offers a rescue shell"部分。
GRUB2对设备与分区的命名规则举例如下,看看就能明白。需要说明的是磁盘从"0"开始计数,分区从"1"开始计数。
(fd0) 第一软盘 (hd0) 第一硬盘[大多数U盘与USB接口的移动硬盘以及SD卡也都被当作硬盘看待] (hd1,1) 第二硬盘的第一分区(通用于MBR与GPT分区) (hd0,msdos2) 第一硬盘的第二MBR分区,也就是传统的DOS分区表 (hd1,msdos5) 第二硬盘的第五MBR分区,也就是第一个逻辑分区 (hd0,gpt1) 第一硬盘的第一GPT分区 (cd) 启动光盘[仅在从光盘启动GRUB时可用] (cd0) 第一光盘
上面所举的例子仅是最常用的情形,更多高级的设备命名规则请参考GRUB2手册中的"Naming convention"与"How to specify devices"部分。此外,如果你想看看当前系统上有哪些设备可用,可以在GRUB SHELL中使用"ls"命令(可能需要先加载必要的驱动模块)。
文件的命名方法有两种:(1)绝对路径表示法,(2)相对路径表示法。举例如下:
(fd0)/grldr 第一软盘根目录下的"grldr"文件[绝对路径] (hd0,gpt1)/boot/vmlinuz 第一硬盘的第一GPT分区"boot"目录下的"vmlinuz"文件[绝对路径] /boot/vmlinuz 根设备"boot"目录下的"vmlinuz"文件[相对路径], 当"root"环境变量等于"(hd0,gpt1)"时,等价于"(hd0,gpt1)/boot/vmlinuz"
上面所举的例子仅是最常用的情形,更多高级的文件命名规则请参考GRUB2手册中的"How to specify files"部分。
磁盘块的命名方法同样也有两种:(1)绝对路径表示法,(2)相对路径表示法。举例如下:
(hd1,1)0+1 在第二硬盘的第一分区上,从第"0"个磁盘块(首扇区)起,长度为"1"的连续块。[绝对路径] (hd1,1)+1 含义与上一个相同,因为当从第"0"个磁盘块(首扇区)起时,"0"可以省略不写。[绝对路径] +1 在根设备上,从第"0"个磁盘块(首扇区)起,长度为"1"的连续块。[相对路径] 当"root"环境变量等于"(hd1,1)"时,等价于"(hd1,1)0+1"
磁盘块几乎只用于链式引导(chainloader)的场合。更多高级的磁盘块命名规则请参考GRUB2手册中的"How to specify block lists"部分。
GRUB2的环境变量大致可以分为两类,第一类是自动设置的变量,也就是这些变量的初始值由GRUB2自动设置,其值必定存在且不为空。第二类是手动设置的变量,它们没有初始值(或者初始值为空),需要经过手动明确设置之后才能使用。
大多数有特定含义的环境变量都是附属于特定附加模块的,只有加载了这些模块之后,这些环境变量才变得有意义。所以从模块的角度看,GRUB2的环境变量又可以分为三类:(1)核心变量,GRUB2核心提供的变量,不依赖于任何可加载模块,这样的变量只有"cmdpath prefix root"三个,而且它们的初始值都由GRUB2自动设置。(2)模块变量,绝大多数有特定含义的环境变量都属此类。(3)脚本变量,这是为了方便编写grub.cfg脚本而设置的变量,没有特殊含义,也不依赖于特定模块,与一般的bash脚本中的变量类似。有关GRUB2脚本的完整说明可以参考GRUB2手册中的"Writing full configuration files directly"部分,基本上其语法与bash脚本完全一致,上手非常容易。
下面列出的变量都是有特定含义的变量,这里只列出常用的一些变量,完整的列表可以参考GRUB2手册中的"Special environment variables"部分。
对于GRUB-2.0.2版本来说,所有可用的命令有大约200个之多,他们中的绝大多数由各种各样的模块提供。我们没有必要去了解所有这些200个命令,只需要了解一些常用的命令即可(实际上就连官方文档也没有给出全部的命令说明)。更多的命令说明可以参考GRUB2手册中的"The list of available commands"页面中列出的几个二级页面。
定义一个名为"title"的菜单项。当此菜单项被选中时,GRUB将会把环境变量"chosen"的值设为"id"(使用了[--id=id]选项)或"title"(未使用[--id=id]选项),然后执行花括号中的命令列表,如果列表中最后一个命令执行成功,并且已经载入了一个内核,那么将执行"boot"命令。
可以使用 --class 选项指定菜单项所属的"样式类"。从而可以使用指定的主题样式显示菜单项。
可以使用 --users 选项指定只允许特定的用户访问此菜单项。如果没有使用此选项,则表示允许所有用户访问。
可以使用 --unrestricted 选项指明允许所有用户访问此菜单项。
可以使用 --hotkey 选项设置访问此菜单项的热键(快捷键)。"key"可以是一个单独的字母,或者'backspace','tab','delete'之一。
可以使用 --id 选项为此菜单项设置一个全局唯一的标识符。"id"必须由ASCII字母/数字/下划线组成,且不得以数字开头。
[arg …]是可选的参数列表。你可以把它们理解为命令行参数。实际上"title"也是命令行参数,只不过这个参数是个必须参数而已。这些参数都可以在花括号内的命令列表中使用,"title"对应着"$1",其余的以此类推。
如果不带任何选项与参数,则表示列出当前激活的输入终端,以及所有其他可用的输入终端。
可以使用 --append 选项将指定的终端加入到激活的输入终端列表中,所有列表中的终端都可以用于向GRUB提供输入。
可以使用 --remove 选项将指定的终端从激活的输入终端列表中删除。
如果不使用任何选项,但是指定了一个或多个终端参数,则表示将当前激活的输入终端设置为参数指定的终端。
如果不带任何选项与参数,则表示列出当前激活的输出终端,以及所有其他可用的输出终端。
可以使用 --append 选项将指定的终端加入到激活的输出终端列表中,所有列表中的终端都将接受到GRUB的输出。
可以使用 --remove 选项将指定的终端从激活的输出终端列表中删除。
如果不使用任何选项,但是指定了一个或多个终端参数,则表示将当前激活的输出终端设置为参数指定的终端。
检查当前用户是否位于"userlist"或环境变量"superusers"中。[注意]如果环境变量"superusers"的值为空,此命令将返回'真'。
设置当前激活的输出终端的背景颜色。"color"可以使用HTML风格的颜色表示法("#RRGGBB"或"#RGB")。
[注意]仅在使用'gfxterm'作为输出终端的时候,才能改变背景色。
将当前激活的输出终端的背景图片设置为"file"文件。除非使用了"--mode 'normal'"选项,否则图片将被自动缩放以填满整个屏幕。
如果不带任何选项与参数,则表示删除背景图片。
[注意]仅在使用'gfxterm'作为输出终端的时候,才能改变背景图片。
启动已经被载入的OS或链式加载器。仅在运行于交互式命令行的时候才是需要的。在一个菜单项结束时是隐含的。
显示文件"file"的内容。如果使用了"--dos"选项,那么"回车/换行符"将被显示为一个简单的换行符。否则,回车符将被显示为一个控制符(<d>)。
链式加载"file"文件。通常使用磁盘块表示法,例如用'+1'表示当前根分区的第一个扇区。
可以使用 --force 选项强制载入文件,而不管它是否有正确的签名。通常用于加载有缺陷的启动载入器(例如 SCO UnixWare 7.1)。
将"file"作为配置文件加载。如果"file"中定义了菜单项,那么立即显示一个包含它们的菜单。
[注意]"file"文件对环境变量所做的任何变更都将在从此文件返回后失效。
检查CPU特性。仅在x86系统上可用。
如果使用了 -l 选项,那么如果CPU是64位则返回真,否则返回假。
如果不使用任何选项,表示将"from_drive"映射到"to_drive"。这主要用于链式加载Windows之类的操作系统,因为它们只能从第一个硬盘启动。出于方便的原因,分区后缀将被忽略,因此你可用安全地将"${root}"作为磁盘使用。
可以使用 -s 选项,执行反向映射,也就是交换这两个磁盘。例如: drivemap -s (hd0) (hd1)
可以使用 -l 选项,列出当前已有的映射。
可以使用 -r 选项,把映射重置为默认值,也就是撤销所有当前已有的映射。
显示所要求的文本并换行(除非使用了 -n 选项)。如果有多个字符串,依次输出它们,并用空格分隔每一个。
和bash的习惯一样,可以在双引号内使用"${var}"来引用变量的值,也可以使用 -e 选项激活对反斜杠转义符的解释( \\ \a \r \n \t ...)。
导出环境变量"envvar",以使其对于使用"configfile"命令载入的配置文件可见。
不做任何事,只返回一个失败的结果。主要用在if/while之类的控制构造中。
把"string"翻译为环境变量"lang"指定的语言。MO格式的翻译文件从环境变量"locale_dir"指定的目录加载。
关闭计算机。如果指定了 --no-apm 选项,表示不执行APM BIOS调用。否则,计算机使用APM关闭。
显示内建命令的帮助信息。如果没有指定"pattern",那么将显示所有可用命令的简短描述。
如果指定了"pattern",那么将只显示名字以这些"pattern"开头的命令的详细帮助信息。
为以32位协议启动的Linux内核载入一个"initial ramdisk",并在内存里的Linux设置区域设置合适的参数。
[注意]这个命令必须放在"linux"命令之后使用。
为以16位协议启动的Linux内核载入一个"initial ramdisk",并在内存里的Linux设置区域设置合适的参数。
[注意]这个命令必须放在"linux16"命令之后使用。
载入名为"module"的GRUB2模块。
使用32位启动协议从"file"载入一个Linux内核映像,并将其余的字符作为内核的命令行参数逐字传入。
[注意]使用32位启动协议意味着'vga='启动选项将会失效。如果你希望明确设置一个特定的视频模式,那么应该使用"gfxpayload"环境变量。虽然GRUB可以自动地检测某些'vga='参数,并把它们翻译为合适的"gfxpayload"设置,但是并不建议这样做。
以传统的16位启动协议从"file"载入一个Linux内核映像,并将其余的字符作为内核的命令行参数逐字传入。这通常用于启动一些遵守Linux启动协议的特殊工具(例如MEMDISK)。
[注意]使用传统的16位启动协议意味着:(1)'vga='启动选项依然有效,(2)不能启动纯64位内核(也就是内核必须要'CONFIG_IA32_EMULATION=y'才行)。
从指定的"file"加载字体,除非使用了绝对路径,否则"file"将被视为"$prefix/fonts/file.pf2"文件。
将"file"文件映射为"device"回环设备。例如:
loopback loop0 /path/to/image
ls (loop0)/
可以使用 -d 选项,删除先前使用这个命令创建的设备。
如果不使用参数,那么列出所有对GRUB已知的设备。
如果参数是包含在括号内的一个设备名,那么列出该设备根目录下的所有文件。
如果参数是以绝对路径给出的目录,那么列出这个目录的内容。
列出已经加载的所有字体
列出已经加载的所有模块
进入普通模式,并显示GRUB菜单。[说明]只要当前没有处于救援模式,其实就已经是在普通模式中了,所以通常并不需要明确使用此命令。
在普通模式中,命令模块[command.lst]与加密模块[crypto.lst]会被自动按需载入(无需使用"insmod"命令),并且可使用完整的GRUB脚本功能。但是其他模块则可能需要明确使用"insmod"命令来载入。
如果给出了"file"参数,那么将从这个文件中读入命令(也就是作为"grub.cfg"的替代),否则将从"$prefix/grub.cfg"中读入命令(如果存在的话)。你也可以理解为"file"的默认值是'$prefix/grub.cfg'。
可以在普通模式中嵌套调用此命令,以构建一个嵌套的环境。不过一般不这么做,而是使用"configfile"命令来达到这目的。
退出当前的普通模式。如果这个普通模式实例不是嵌套在另一个普通模式里的话,就会返回到救援模式。
对分区表进行各种修改。目前只能作用于MBR分区表(DOS分区表),而不能用于GPT分区表。目前仅支持以下三种用法:
(1)设置或去掉分区的激活标记(仅对Windows系统有意义)。
例如:"parttool (hd0,msdos2) +boot"表示为(hd0,msdos2)分区加上激活标记,而"parttool (hd0,msdos2) -boot"则表示去掉(hd0,msdos2)分区的激活标记。
(2)设置或去掉分区的隐藏标记(仅对Windows系统有意义)。
例如:"parttool (hd0,msdos2) +hidden"表示为(hd0,msdos2)分区加上隐藏标记,而"parttool (hd0,msdos2) -hidden"则表示去掉(hd0,msdos2)分区的隐藏标记。
(3)更改分区的类型。其值必须是0x00-0xFF范围内的值。且应该使用'0xNN'格式的十六进制数。
例如:"parttool (hd0,msdos2) type=0x83"表示将(hd0,msdos2)分区类型修改为'0x83'(Linux分区)。
定义一个名为user的用户,并使用明文口令'clear-password'。不建议使用此命令。
定义一个名为user的用户,并使用哈希口令'hashed-password'(通过"grub-mkpasswd-pbkdf2"工具生成)。这是建议使用的命令,因为它安全性更高。
提取"device"设备的特定信息。如果使用了 --set 选项,则表示将提取的结果保存在"var"变量中,否则将提取的结果直接显示出来。
从用户读取一行输入。如果给定环境变量"var",则把它设为所读取的行(不包括结尾的换行符)。
重新启动
卸载"module"模块
通过文件[--file]、卷标[--label]、文件系统UUID[--fs-uuid]来查找设备。
如果使用了 --set 选项,那么会将第一个找到的设备设置为环境变量"var"的值。默认的"var"是'root'。
可以使用 --no-floppy 选项来禁止查找软盘设备,因为这些设备非常慢。
将环境变量"envvar"的值设为'value'。如果没有使用参数,则打印出所有环境变量及其值。
直接将"file"文件的内容插入到当前位置。与"configfile"不同,此命令既不切换执行环境,也不会显示一个新的菜单。
计算"expression"的值,并在结果为真时返回零值,或者在结果为假时返回非零值,主要用在if/while之类的控制构造中。
可用的"expression"模式如下(与bash类似):
string1 == string2 [string1与string2完全相同] string1 != string2 [string1与string2不完全相同] string1 < string2 [string1在字母顺序上小于string2] string1 <= string2 [string1在字母顺序上小于string2或与string2完全相同] string1 > string2 [string1在字母顺序上大于string2] string1 >= string2 [string1在字母顺序上大于string2或与string2完全相同] integer1 -eq integer2 [integer1等于integer2] integer1 -ge integer2 [integer1大于或等于integer2] integer1 -gt integer2 [integer1大于integer2] integer1 -le integer2 [integer1小于或等于integer2] integer1 -lt integer2 [integer1小于integer2] integer1 -ne integer2 [integer1不等于integer2] prefixinteger1 -pgt prefixinteger2 [剔除非数字字符首部之后,integer1大于integer2] prefixinteger1 -plt prefixinteger2 [剔除非数字字符首部之后,integer1小于integer2] file1 -nt file2 [file1的修改时间比file2新] file1 -ot file2 [file1的修改时间比file2旧] -d file [file存在并且是一个目录] -e file [file存在] -f file [file存在并且不是一个目录] -s file [file存在并且文件尺寸大于零] -n string [string的长度大于零] string [string的长度大于零] -z string [string的长度等于零] ( expression ) 将expression视为一个整体(分组) ! expression 非(NOT) expression1 -a expression2 与(AND) expression1 -o expression2 或(OR)
不做任何事,只返回一个成功的结果。主要用在if/while之类的控制构造中。
撤销环境变量"envvar"
列出所有当前可用的视频模式。如果指定了分辨率(或者还附加了色深),那么仅显示与其匹配的模式。
在默认情况下,GRUB对于所有可以在物理上进入控制台的人都是可访问的。任何人都可以选择并编辑任意菜单项,并且可以直接访问GRUB SHELL。要启用认证支持,必须将环境变量"superusers"设置为一组用户名(可用空格/逗号/分号作为分隔符),这样,将仅允许"superusers"中的用户使用GRUB命令行、编辑菜单项、以及执行任意菜单项。而其他非"superusers"中的用户,只能执行那些没有设置 --users 选项的菜单,以及那些在 --users 选项中包含了该用户的菜单,但不能使用GRUB命令行、编辑菜单项。下面使用一个配置片段举例说明:
set superusers="root" password_pbkdf2 root grub.pbkdf2.sha512.10000.biglongstring password user1 insecure menuentry "所有人都可以执行此菜单" --unrestricted { ... } menuentry "仅允许超级用户执行此菜单" --users "" { ... } menuentry "允许 user1 和超级用户执行此菜单" --users user1 { ... }
有关GRUB2安全的更多详情,请参考GRUB2手册中的"Security"部分。
首先制作一张PNG格式的图片,分辨率最好是"1024x768"以保证较好的兼容性。然后将这张图片放到"$prefix/themes/1024x768.png"("$prefix"是GRUB2的安装目录)。然后在'grub.cfg'中加入如下内容:
set gfxmode=1024x768,auto insmod gfxterm insmod png terminal_output gfxterm background_image $prefix/themes/1024x768.png
由于GRUB2在内部使用UTF-8编码,并且所有文本文件(包括'grub.cfg')也都被假定为使用UTF-8编码,为了避免乱码,请务必以UTF-8编码保存'grub.cfg'文件。
set gfxterm_font=unicode set lang=zh_CN set locale_dir=$prefix/locale insmod gfxterm terminal_output gfxterm loadfont unicode
如果你认为默认的unicode字体在1024x768或更高分辨率的屏幕上显得太小,或者你认为默认的字体不好看,想换换口味,那么如何自己动手制作一个pf2字体呢?那就要用到"grub-mkfont"工具。下面的示例展示了如何从一个ttc字体(文泉驿等宽微米黑)制作一个24px大小的pf2字体:
grub-mkfont -i1 -n WenQuanYiMicroHeiMono24px -o WenQuanYiMicroHeiMono24px.pf2 -s24 -v wqy-microhei.ttc
将制作好的字体文件(WenQuanYiMicroHeiMono24px.pf2)放到"$prefix/fonts"目录中,修改'grub.cfg'文件中的两行:
set gfxterm_font=WenQuanYiMicroHeiMono24px loadfont WenQuanYiMicroHeiMono24px
[注意]你最好使用等宽中文字体(推荐使用文泉驿等宽正黑或者等宽微米黑),否则可能会让GRUB2的字体间距过大,十分难看。
GRUB4DOS有一个非常酷的'磁盘映射'功能,能够用于启动WinPE的ISO文件。其实,将GRUB2配合MEMDISK工具使用,同样可以引导各种映像文件,包括ISO文件与软/硬盘映像。
首先,你必须安装或者下载"syslinux"软件包,从中提取出"memdisk"文件(可能位于'/usr/share/syslinux/memdisk'或'bios/memdisk/memdisk'),然后将它复制到GRUB2的安装目录中(本文假定其位于"winpe"子目录中,也就是"$prefix/winpe/memdisk")。
然后,将你想要引导的WindowsPE的ISO文件放到某个地方(本文假定其位于"$prefix/winpe/WinPE.ISO")。当然,为了节约磁盘空间,你也可以用gzip对ISO文件进行压缩,不过这个示例中没有这么做。
最后,在'grub.cfg'中加入如下菜单项(如果你对ISO进行了gzip压缩,那么还需要额外再加上"insmod gzio"命令):
menuentry "[BIOS] Windows PE" --unrestricted { linux16 $prefix/winpe/memdisk iso raw initrd16 $prefix/winpe/WinPE.ISO }
[注意]与GRUB4DOS一样,由于MEMDISK对各种映像文件的模拟是通过在实模式下拦截BIOS的 INT 13h 与 INT 15h 调用来实现的,所以有很大的局限性:
[释疑]不要将MEMDISK与GRUB2的"memdisk.mod"模块混淆,他们是两个毫不相干的东西。"memdisk.mod"模块的作用是为'core.img'提供内存盘支持,其目的是为了让GRUB2能够正确识别启动设备。如果把'core.img'比作Linux内核,那么"memdisk.mod"模块的作用就相当于为'core.img'这个"内核"提供了'initramdisk',用以加载磁盘与文件系统驱动,从而让GRUB2可以访问磁盘,进而加载其他的模块。但是实际上,在绝大多数情况下,更本不需要使用"memdisk.mod",因为在'grub-install'的时候,这些驱动已经被嵌入到'core.img'中了。如果你还是不明白"memdisk.mod"模块的作用的话,那就忘记它的存在吧,因为你更本不需要知道有这个东西,就像你不需要了解全部GRUB2模块一样。仅仅是因为这两个东西的名字相同,我才在这里多罗嗦了几句而已。
由于GRUB2不能在UEFI模式下对ISO文件进行仿真,那么我们应该如何在UEFI模式下引导WindowsPE呢?答案是必须使用WIM格式的WindowsPE。
具体说来就是首先用GRUB2链式加载微软的"bootmgfw.efi"引导管理器,然后再由"bootmgfw.efi"根据BCD文件的指引去启动WindowsPE。
第一步,从例如"微PE"这样的作品中提取"WEPE64.WIM"与"WEPE.SDI"(BOOT.SDI)文件。
第二步,从Win10原版安装光盘中提取"bootmgfw.efi"文件([sources/install.wim]/Windows/Boot/EFI/bootmgfw.efi)。
第三步,将提取的三个文件放置到一个FAT32或NTFS磁盘分区上,这里假定放到'(hd0,gpt3)/winpe/'目录中。
第四步,仿照下面的命令序列编写一个BCD文件:
bcdedit /createstore BCD bcdedit /store BCD /create {dddddddd-dddd-dddd-dddd-dddddddddddd} /device bcdedit /store BCD /set {dddddddd-dddd-dddd-dddd-dddddddddddd} RAMDISKSDIDEVICE BOOT bcdedit /store BCD /set {dddddddd-dddd-dddd-dddd-dddddddddddd} RAMDISKSDIPATH "\winpe\WEPE.SDI" bcdedit /store BCD /create {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} /d "UEFI Windows PE x64" /application OSLOADER bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} HIGHESTMODE YES bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} DEVICE RAMDISK=[BOOT]\winpe\WEPE64.WIM,{dddddddd-dddd-dddd-dddd-dddddddddddd} bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} OSDEVICE RAMDISK=[BOOT]\winpe\WEPE64.WIM,{dddddddd-dddd-dddd-dddd-dddddddddddd} bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} SYSTEMROOT "\Windows" bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} WINPE YES bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} NX OptIn bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} PAE ForceEnable bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} detecthal Yes bcdedit /store BCD /set {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} LOCALE "zh-CN" bcdedit /store BCD /create {bootmgr} /d "Windows PE Boot Manager" bcdedit /store BCD /set {bootmgr} nointegritychecks Yes bcdedit /store BCD /set {bootmgr} DEVICE BOOT bcdedit /store BCD /set {bootmgr} DISPLAYBOOTMENU YES bcdedit /store BCD /set {bootmgr} LOCALE "zh-CN" bcdedit /store BCD /displayorder {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} bcdedit /store BCD /default {aaaaaaaa-aaaa-aaaa-aaaa-aaaaaaaaaaaa} bcdedit /store BCD /timeout 10第五步,将生成的BCD文件同样放置到'(hd0,gpt3)/winpe/'目录中。
menuentry 'UEFI Windows PE Boot Manager' --unrestricted { chainloader (hd0,gpt3)/winpe/bootmgfw.efi }
以"install-amd64-minimal-*.iso"为例。
第一步,将ISO中的'/boot/{gentoo,gentoo.igz}'、'/image.squashfs'三个文件放到'(hd0,gpt3)/os/gentoo/'目录中;
第二步,将ISO中的'/livecd'放到相同分区(hd0,gpt3)的根目录下;
最后,在'grub.cfg'中加入如下菜单项:
menuentry "Gentoo Minimal Install LiveCD" --unrestricted { linux (hd0,gpt3)/os/gentoo/gentoo cdroot looptype=squashfs loop=/os/gentoo/image.squashfs initrd (hd0,gpt3)/os/gentoo/gentoo.igz }
[提示]'livecd'是寻找'image.squashfs'所在磁盘分区的关键。
[提示] Gentoo LiveCD 亦可使用与其他Linux发行版的LiveCD类似的方法启动(也就是使用"isoboot="参数)。
首先需要说明的是,这里给出的方法,只适用于提供了"img_loop="或"iso-scan/filename="或"fromiso="或"isoboot="或"isoloop="之类参数的LiveCD。
下面以 Ubuntu 的 LiveCD 为例说明。首先,假定你将ISO文件放在'(hd0,gpt3)/ISO/Ubuntu.iso';然后,在'grub.cfg'中加入如下菜单项:
menuentry "Ubuntu LiveCD" --unrestricted { loopback loop0 (hd0,gpt3)/ISO/Ubuntu.iso linux (loop0)/casper/vmlinuz boot=casper iso-scan/filename=/ISO/Ubuntu.iso initrd (loop0)/casper/initrd.lz }
[说明]这里给出的方法,其实就是各种"硬盘安装 XX Linux"的翻版,只不过不再需要将"vmlinuz"与"initrd"从ISO中解压出来而已。
更多其它发行版的实例,请继续阅读下面的内容。
下面是本文作者实际使用的一个"grub.cfg"文件,通用于BIOS与UEFI模式,放在这里当作一个实例,供读者参考:
############################################################################### #因为"$prefix"与"$root"的初始值由"grub-install"在安装时确定,并且嵌入"core.img"中的模块也会随硬件状况而变化, #所以不要简单的直接复制"grub"目录到处使用,而是应该在每一个介质上都使用"grub-install"进行安装。 ############################################################################### #[1]因为传统BIOS只能识别磁盘前137GB范围,所以GRUB2的安装分区最好完全位于前137GB范围内。 #[2]建议将VFAT文件系统的卷标设为纯大写字母,以确保最大程度的兼容性。 #[bug]在GRUB2-SHELL中,会以纯小写字母形式显示VFAT文件系统上的纯大写字母文件名(导致视觉上的混淆)。 ############################################################################### #可以在 Linux 中使用如下命令格式化VFAT文件系统{卷标="ESP",单元(族)大小=64K[128扇区(最大值)]} # mkfs.vfat -v --codepage=437 -nESP -s128 /dev/sdx1 ############################################################################### #可以在windows上使用如下命令安装GRUB2 {安装目标来自于以管理员身份运行的"wmic diskdrive list brief"命令(DeviceID)} # grub-install.exe --disable-shim-lock --locales=zh_CN --themes= --boot-directory=k: --recheck --target=x86_64-efi --efi-directory=k: --removable --no-nvram # grub-install.exe --disable-shim-lock --locales=zh_CN --themes= --boot-directory=k: --recheck --target=i386-pc --disk-module=biosdisk \\?\PhysicalDrive5 ############################################################################### #https://github.com/thias/glim/tree/master/grub2 #https://bitbucket.org/gsls200808/chinese-opensource-mirror-site/src/master/ ############################################################################### #(1)GRUB2安装分区的卷标标记为"$esp"变量 #(2)在GRUB2安装分区的根目录下,存在与卷标("$esp")同名的目录(存放WinPE与Linux镜像) set esp=ESP set check_signatures=no set default=0 set fallback=1 set gfxmode=1024x768,auto set gfxterm_font=WenQuanYiMicroHeiMono set lang=zh_CN set locale_dir=$prefix/locale set pager=1 set timeout=5 insmod part_gpt insmod part_msdos insmod fat insmod exfat insmod ntfs insmod iso9660 insmod btrfs insmod ext2 insmod xfs insmod all_video insmod gfxterm insmod png terminal_output gfxterm background_image $prefix/themes/wmoon.png loadfont 24px #密码保护 set superusers=root password_pbkdf2 root grub.pbkdf2.sha512.69.3AE287EEFF7205CC.C0C0B1517369963D14FADE30BE588E56D88E86451C9D3FB9B51917DBBF2F8E8A #http://www.mistyprojects.co.uk/documents/BCDEdit/index.html menuentry '正常启动(Windows)' --unrestricted { set grbroot=$root if [ 'efi' == $grub_platform ] ; then if search --file --set /EFI/Microsoft/Boot/bootmgfw.efi ; then chainloader /EFI/Microsoft/Boot/bootmgfw.efi else echo '[UEFI]没有找到 Windows 启动文件' fi elif [ 'pc' == $grub_platform ] ; then if search --file --set /bootmgr ; then ntldr /bootmgr elif search --file --set /ntldr ; then ntldr /ntldr else echo '[BIOS]没有找到 Windows 启动文件' fi fi set root=$grbroot unset grbroot } #http://www.wepe.com.cn/download.html #[注意]务必确保在BCD中设置了正确的wim与sdi文件路径 if [ 'efi' == $grub_platform -a -f /$esp/WinPE/BCD -a -f /$esp/WinPE/bootx64.efi -a -f /$esp/WinPE/win10pe64.sdi -a -f /$esp/WinPE/win10pe64.wim ] ; then menuentry '[UEFI]启动64位 Win 10 PE →' --unrestricted { chainloader /$esp/WinPE/bootx64.efi } elif [ 'pc' == $grub_platform -a -f /$esp/WinPE/memdisk -a -f /$esp/WinPE/win10pe64.iso ] ; then menuentry '[BIOS]启动64位 Win 10 PE' --unrestricted { linux16 /$esp/WinPE/memdisk iso raw initrd16 /$esp/WinPE/win10pe64.iso } fi #https://www.system-rescue.org/Download/ #https://www.system-rescue.org/manual/Booting_SystemRescue/ #基于ArchLinux制作的功能丰富维护工具(安全/文件/磁盘/网络/硬件/引导),可选使用Xfce图形界面 if [ -f /$esp/systemrescue.iso ] ; then menuentry 'SystemRescue LiveCD [root/123]' --unrestricted { loopback loop0 /$esp/systemrescue.iso linux (loop0)/sysresccd/boot/x86_64/vmlinuz archisobasedir=sysresccd img_label=$esp img_loop=/$esp/systemrescue.iso vga=792 copytoram systemd.wants=gpm.service systemd.wants=sshd.service nofirewall setkmap=us rootpass=123 initrd (loop0)/sysresccd/boot/intel_ucode.img (loop0)/sysresccd/boot/amd_ucode.img (loop0)/sysresccd/boot/x86_64/sysresccd.img } fi #https://mirrors.kernel.org/gentoo/releases/amd64/autobuilds/current-admincd-amd64/ #https://mirrors.neusoft.edu.cn/gentoo/releases/amd64/autobuilds/current-admincd-amd64/ #https://wiki.gentoo.org/wiki/Handbook:AMD64/Installation/Media/zh-cn #[注意] Gentoo LiveCD 仅接受足够复杂的密码 if [ -f /$esp/admincd.iso ] ; then menuentry 'Gentoo AdminCD [root/abc#XYZ#345]' --unrestricted { loopback loop0 /$esp/admincd.iso linux (loop0)/boot/gentoo cdroot isoboot=/$esp/admincd.iso vga=792 docache dogpm dosshd keymap=us scandelay passwd=abc#XYZ#345 initrd (loop0)/boot/gentoo.igz } fi #https://mirror.rackspace.com/archlinux/ #https://opentuna.cn/archlinux/iso/latest/ #[airootfs.fs]/usr/share/doc/mkinitcpio-archiso/README.bootparams if [ -f /$esp/archlinux.iso ] ; then menuentry 'Arch Linux LiveCD [root/空]' --unrestricted { loopback loop0 /$esp/archlinux.iso linux (loop0)/arch/boot/x86_64/vmlinuz-linux archisobasedir=arch img_label=$esp img_loop=/$esp/archlinux.iso vga=792 copytoram systemd.wants=gpm.service systemd.wants=sshd.service initrd (loop0)/arch/boot/intel-ucode.img (loop0)/arch/boot/amd-ucode.img (loop0)/arch/boot/x86_64/initramfs-linux.img } fi #https://www.calculate-linux.org/main/en/download #https://mirrors.aliyun.com/calculate-linux/release/ #https://mirror.iscas.ac.cn/calculate/release/ #https://mirror-hk.koddos.net/calculate-linux/release/ #https://man7.org/linux/man-pages/man7/dracut.cmdline.7.html #基于Gentoo制作的LiveCD(包含编译工具),可用作LFS(Linux From Scratch)的宿主系统 #[注意]此LiveCD存在键盘按键失灵的BUG,仅适合SSH远程使用 if [ -f /$esp/cls.iso ] ; then menuentry 'Live GCC x64 [root/root](仅适合SSH远程使用)' --unrestricted { loopback loop0 /$esp/cls.iso linux (loop0)/boot/vmlinuz root=live iso-scan/filename=/$esp/cls.iso vga=792 rd.live.ram=1 calculate=lang:en_US,keymap:en_US initrd (loop0)/boot/initrd } fi #https://download.fedoraproject.org/pub/fedora/linux/releases/35/Spins/x86_64/iso/ #https://mirrors.163.com/fedora/releases/35/Spins/x86_64/iso/ #https://docs.fedoraproject.org/en-US/fedora/rawhide/install-guide/advanced/Boot_Options/ #只有 Xfce 可以设置中文界面 [Fedora35 LiveCD BUG] rd.live.ram=1 会导致启动失败 if [ -f /$esp/Fedora-Xfce-Live.iso ] ; then menuentry 'Fedora 35 Xfce LiveCD [root/空](自动liveuser登录)' --unrestricted { loopback loop0 /$esp/Fedora-Xfce-Live.iso linux (loop0)/isolinux/vmlinuz rd.live.image root=live:CDLABEL=Fedora-Xfce-Live-35-1-2 iso-scan/filename=/$esp/Fedora-Xfce-Live.iso vga=792 rd.live.ram=0 systemd.wants=sshd.service inst.lang=zh_CN locale.LANG=zh_CN.utf8 initrd (loop0)/isolinux/initrd.img } fi #https://download.rockylinux.org/pub/rocky/8/Live/x86_64/ #https://mirrors.sjtug.sjtu.edu.cn/rocky/8/Live/x86_64/ #https://mirrors.edge.kernel.org/pub/linux/utils/boot/dracut/dracut.html if [ -f /$esp/Rocky-Workstation.iso ] ; then menuentry 'Rocky 8.5 GNOME LiveCD [root/空](自动liveuser登录)' --unrestricted { loopback loop0 /$esp/Rocky-Workstation.iso linux (loop0)/isolinux/vmlinuz0 rd.live.image root=live:CDLABEL=Rocky-8.5-Workstation-20211114 iso-scan/filename=/$esp/Rocky-Workstation.iso vga=792 rd.live.ram=1 systemd.wants=sshd.service inst.lang=zh_CN systemd.setenv=LC_ALL=zh_CN.utf8 initrd (loop0)/isolinux/initrd0.img } fi #http://mirror.stream.centos.org/9-stream/BaseOS/x86_64/iso/ #https://mirrors.nju.edu.cn/centos-stream/9-stream/BaseOS/x86_64/iso/ #https://mirrors.tuna.tsinghua.edu.cn/centos-stream/9-stream/BaseOS/x86_64/iso/ #https://anaconda-installer.readthedocs.io/en/latest/boot-options.html #此安装方式通用于所有不含"LiveOS"目录的 Fedora/Rocky/CentOS 映像 #用于硬盘安装的额外ISO文件(包含"Packages"或"BaseOS"目录)必须存放于 xfs/ext234/exFAT 文件系统(为了方便查找,建议放在根目录) if [ -f /$esp/CentOS-Stream-boot.iso ] ; then menuentry '安装 CentOS Stream {网络|硬盘[/CentOS-Stream-dvd1.iso]}' --unrestricted { loopback loop0 /$esp/CentOS-Stream-boot.iso linux (loop0)/images/pxeboot/vmlinuz inst.stage2=hd:LABEL=$esp:/$esp/CentOS-Stream-boot.iso vga=792 rd.live.ram=1 inst.repo=https://mirrors.aliyun.com/centos-stream/9-stream/BaseOS/x86_64/os/ inst.lang=zh_CN inst.keymap=us nameserver=223.6.6.6 initrd (loop0)/images/pxeboot/initrd.img } fi #https://cdimage.debian.org/cdimage/unofficial/non-free/cd-including-firmware/current-live/amd64/iso-hybrid/ #https://mirrors.ustc.edu.cn/debian-cdimage/unofficial/non-free/cd-including-firmware/current-live/amd64/iso-hybrid/ #https://manpages.debian.org/testing/live-boot-doc/live-boot.7.en.html #也适用于'Kali LiveCD [root/toor]'( hostname=kali ) https://cdimage.kali.org/current/ #[BUG] verify-checksums 会导致启动失败 [提示]内存小于6G时不要使用"toram" if [ -f /$esp/debian-live.iso ] ; then menuentry 'Debian 11.2 LiveCD (NO SSH)' --unrestricted { loopback loop0 /$esp/debian-live.iso linux (loop0)/live/vmlinuz-5.10.0-10-amd64 boot=live findiso=/$esp/debian-live.iso components vga=792 toram noeject username=root locales=zh_CN.UTF-8 initrd (loop0)/live/initrd.img-5.10.0-10-amd64 } fi #https://www.debian.org/releases/stable/amd64/ch05s03.zh-cn.html #https://www.debian.org/releases/stable/amd64/ch06s03.zh-cn.html #https://deb.debian.org/debian/dists/stable/main/installer-amd64/current/images/hd-media/ #https://mirrors.bfsu.edu.cn/debian-cd/current/amd64/iso-dvd/ #最好将DVD映像存放在 ext234/VFAT 文件系统的根目录(方便自动搜索到DVD映像) #[注意]此方法不适用于 Ubuntu if [ -f /$esp/debian/vmlinuz -a -f /$esp/debian/initrd.gz ] ; then menuentry '硬盘安装 Debian [/debian-amd64-DVD-1.iso]' --unrestricted { linux /$esp/debian/vmlinuz vga=792 priority=low locale=zh_CN initrd /$esp/debian/initrd.gz } fi #https://deb.debian.org/debian/dists/stable/main/installer-amd64/current/images/netboot/ #https://mirrors.pku.edu.cn/debian/dists/stable/main/installer-amd64/current/images/netboot/ #[注意]此方法不适用于 Ubuntu if [ -f /$esp/debian/mini.iso ] ; then menuentry '网络安装 Debian [不支持WiFi]' --unrestricted { loopback loop0 /$esp/debian/mini.iso linux (loop0)/linux vga=792 priority=low locale=zh_CN initrd (loop0)/initrd.gz } fi #https://cdimage.ubuntu.com/lubuntu/releases/ #https://mirrors.cloud.tencent.com/ubuntu-cdimage/lubuntu/releases/ #也适用于"硬盘安装 Ubuntu Server" 与 'KDE neon LiveCD'(initrd.lz; apparmor=0) https://files.kde.org/neon/images/user/current/neon-user-current.iso if [ -f /$esp/lubuntu.iso ] ; then menuentry 'Ubuntu LXQt LiveCD (NO SSH)' --unrestricted { loopback loop0 /$esp/lubuntu.iso linux (loop0)/casper/vmlinuz boot=casper iso-scan/filename=/$esp/lubuntu.iso vga=792 toram noprompt username=root locale=zh_CN keyboard-configuration/layoutcode=us initrd (loop0)/casper/initrd } fi #https://download.opensuse.org/distribution/openSUSE-current/live/ #https://repo.huaweicloud.com/opensuse/distribution/openSUSE-current/live/ if [ -f /$esp/openSUSE-Leap-KDE-Live.iso ] ; then menuentry 'openSUSE Leap 15.3 KDE LiveCD [root/空]' --unrestricted { loopback loop0 /$esp/openSUSE-Leap-KDE-Live.iso linux (loop0)/boot/x86_64/loader/linux root=live:CDLABEL=openSUSE_Leap_15.3_KDE_Live iso-scan/filename=/$esp/openSUSE-Leap-KDE-Live.iso vga=792 systemd.wants=gpm.service systemd.wants=sshd.service initrd (loop0)/boot/x86_64/loader/initrd } fi #https://download.opensuse.org/distribution/openSUSE-current/iso/ #https://mirror.bjtu.edu.cn/opensuse/distribution/openSUSE-current/iso/ #https://doc.opensuse.org/documentation/leap/startup/html/book-startup/cha-boot-parameters.html #https://en.opensuse.org/SDB:Linuxrc #用于硬盘安装的DVD映像必须存放于 xfs/ext234/NTFS 文件系统的根目录 if [ -f /$esp/openSUSE-Leap-NET.iso ] ; then menuentry '硬盘安装 openSUSE Leap [/openSUSE-Leap-DVD.iso]' --unrestricted { loopback loop0 /$esp/openSUSE-Leap-NET.iso linux (loop0)/boot/x86_64/loader/linux install=hd:/openSUSE-Leap-DVD.iso vga=792 language=zh_CN initrd (loop0)/boot/x86_64/loader/initrd } menuentry '网络安装 openSUSE Leap' --unrestricted { loopback loop0 /$esp/openSUSE-Leap-NET.iso linux (loop0)/boot/x86_64/loader/linux install=https://mirrors.yun-idc.com/opensuse/distribution/openSUSE-current/repo/oss/ vga=792 language=zh_CN initrd (loop0)/boot/x86_64/loader/initrd } fi menuentry '关机' --unrestricted { halt ; } menuentry '重新启动' --unrestricted { reboot ; } if [ 'efi' == $grub_platform ] ; then menuentry '重启至 UEFI 设置' --unrestricted { fwsetup ; } fi