说明

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命令,如下图所示:

编译grafana面板_#define

可以看到有报错,原因是少装了某些依赖,通过apt-get命令安装对应的依赖即可。

这里需要安装的依赖有bison、flex,根据不同的平台和配置,需要安装的命令可能不同,这里不具体说明。

./configure成功后,显示如下的界面:

编译grafana面板_GNU_02

之后就可以通过make命令来编译。

编译时发生了如下的错误:

编译grafana面板_Ubuntu_03

在网上找了一下,在c++ - How to overcome "'aclocal-1.15' is missing on your system" warning? - Stack Overflow上找到了一些说明。

之后又安装了automake,然后在根目录运行了autoreconf -f -i命令。

再之后使用make可以正常执行下去,但是在编译的时候还是报错了:

编译grafana面板_GNU_04

这个应该是一个代码问题,也可能是版本不兼容的问题,是出现在Ubuntu18.04上的,目前的修改方法如下:

// #define fprintf(...) 0
#define fprintf(...)

修改的文件是grub-2.02/grub-core/script/yylex.l。修改之后可以正常编译。

编译结束后如下所示:

编译grafana面板_#define_05

可以看到生成了大量的新的文件:

编译grafana面板_编译grafana面板_06

这里包括了很多的工具,比如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,但是这里使用了默认的库,所以实际而默认的库如下:

编译grafana面板_GNU_07

所以上面的命令实际上根本没有用到我们自己的GRUB源代码编译出来的东西。

4. 要使用真正的通过源代码编译出来的GRUB,可以使用grub-mkimage:

编译grafana面板_GNU_08

下面是生成GRUB二进制使用的命令:

./grub-mkimage -p . -d ./grub-core/ -O x86_64-efi -o bootx64.efi hello

这里比较重要的参数是-d,它指定了我们刚才编译出来的image和module,因为它们才是我们需要的,而不是系统中已经存在的。

另一个重要的参数是hello,它其实是一个模块,GRUB编译会在grub-core目录下生成非常多的可插拔的模块:

编译grafana面板_Ubuntu_09

hello就是其中的一个模块。

GRUB二进制的这种方式,使得我们可以自由的合入我们需要的模块。

最后补上bootx64.efi在UEFI Shell下的运行结果:

编译grafana面板_编译grafana面板_10

编译grafana面板_#define_11