说明
GRUB2的官网是GNU GRUB - GNU Project - Free Software Foundation (FSF)。
GRUB2的源代码可以在这里下载到。
本文使用的编译环境是
Linux home 4.4.0-83-generic #106~14.04.1-Ubuntu SMP Mon Jun 26 18:10:19 UTC 2017 x86_64 x86_64 x86_64 GNU/Linux
gcc version 4.8.4 (Ubuntu 4.8.4-2ubuntu1~14.04.3)
补充20181002:
之后在Ubuntu18.04上也进行了测试:
Linux home 4.15.0-23-generic #25-Ubuntu SMP Wed May 23 18:02:16 UTC 2018 x86_64 x86_64 x86_64 GNU/Linux
(说明:正文中某些图片有不一致的情况,是因为在不同平台编译的结果。)
正文
首先下载到源代码,大致就是一个压缩文件,名字是grub-2.02.tar.gz。(版本号可能会有所不同)
使用tar命令解压文件,得到grub-2.02目录。
进入该目录,运行./configure命令,如下图所示:
可以看到有报错,原因是少装了某些依赖,通过apt-get命令安装对应的依赖即可。
这里需要安装的依赖有bison、flex,根据不同的平台和配置,需要安装的命令可能不同,这里不具体说明。
./configure成功后,显示如下的界面:
之后就可以通过make命令来编译。
编译时发生了如下的错误:
在网上找了一下,在c++ - How to overcome "'aclocal-1.15' is missing on your system" warning? - Stack Overflow上找到了一些说明。
之后又安装了automake,然后在根目录运行了autoreconf -f -i命令。
再之后使用make可以正常执行下去,但是在编译的时候还是报错了:
这个应该是一个代码问题,也可能是版本不兼容的问题,是出现在Ubuntu18.04上的,目前的修改方法如下:
// #define fprintf(...) 0
#define fprintf(...)
修改的文件是grub-2.02/grub-core/script/yylex.l。修改之后可以正常编译。
编译结束后如下所示:
可以看到生成了大量的新的文件:
这里包括了很多的工具,比如grub-mkconfig就可用来生成配置文件。关于这些工具的作用,这里不一一介绍。
GRUB for UEFI
下面简单说明下如何生成EFI格式的GRUB,它可以在UEFI She'll下直接使用,基本上支持UEFI的机器都会有一个单独的FAT格式的硬盘分区,里面放的就包括这么一个GURB。
这部分内容参考自GRUB - OSDev Wiki。
生成EFI格式的GRUB的方式如下:
1. configure命令如下:
./configure --target=x86_64 --with-platform=efi
这里就多了几个参数,其中--target参数用来指定对应的平台,这里指定了x86的64位平台。
2. make还是不变;
3. 使用gurb-mkstandalone工具:
grub-mkstandalone -O x86_64-efi -o bootx64.efi
完成上述操作之后就会生成一个UEFI可用的GURB,名称是bootx64.efi。
需要注意的是,上述命令其实是存在问题的,虽然命令是通过编译GRUB得到的grub-mkstandalone,但是这里使用了默认的库,所以实际而默认的库如下:
所以上面的命令实际上根本没有用到我们自己的GRUB源代码编译出来的东西。
4. 要使用真正的通过源代码编译出来的GRUB,可以使用grub-mkimage:
下面是生成GRUB二进制使用的命令:
./grub-mkimage -p . -d ./grub-core/ -O x86_64-efi -o bootx64.efi hello
这里比较重要的参数是-d,它指定了我们刚才编译出来的image和module,因为它们才是我们需要的,而不是系统中已经存在的。
另一个重要的参数是hello,它其实是一个模块,GRUB编译会在grub-core目录下生成非常多的可插拔的模块:
hello就是其中的一个模块。
GRUB二进制的这种方式,使得我们可以自由的合入我们需要的模块。
最后补上bootx64.efi在UEFI Shell下的运行结果: