扫一扫
分享文章到微信
扫一扫
关注官方公众号
至顶头条
在讨论引导新的驱动器之前,需要了解一些引导加载程序理论。
引导加载程序通常安装在计算机第一个硬盘的 MBR 中。调用引导加载程序时(BIOS 自动执行 MBR 中的代码),它通常显示可以引导的 OS 的菜单。选择一个给定 OS 引导。
关于此场景应该注意两点:
- OS 选择菜单(通常)从磁盘加载。
- 要引导相关 OS,引导加载程序需要从磁盘读取相关内核。
由于以上操作在加载 OS 之前发生,它意味着所有磁盘读取都必须通过 BIOS 调用的方式发生。这会涉及严重的问题:即为了直接引导磁盘,您的 BIOS 必须支持通过 FireWire 或 USB 连接的磁盘。这通常可以看作从这些类型的磁盘引导的一个 BIOS 选项。实际上 FireWire BIOS 支持当前很少见,但 USB 支持正在变得相当普遍。因此,如果您在相对较新的计算机中使用 USB,应该可以直接在 Linux 中引导驱动器。
在外置驱动器的 MBR 中安装了 GRUB 之后,当通过 USB 连接时,我可以直接引导该驱动器。当引导连接的磁盘时很简单地进入了 BIOS 设置程序。外置磁盘将显示为普通的硬盘驱动器:移动该磁盘使它在引导顺序中位于内置驱动器之前。
我也可以在内置驱动器的 MBR 中安装引导加载程序,并使用它引导 USB 驱动器(这时它在 GRUB 中显示为 hd1 in GRUB)。如果您使用 FireWire,有可能 BIOS 不能直接引导驱动器,将需要一些其他操作。
幸运地是,因为 Linux 的灵活性,如果您不能直接引导(使用 PCMCIA FireWire 卡,我的情况肯定是这样!),会有相当简单的解决方案。可以从支持的设备(如软盘驱动器、CD、USB key 或主驱动器上的微小分区)执行初始引导步骤,然后使用外置驱动器进行其他操作。
构建引导映像
可以使用两种方法引导:
一阶段引导
内核引导、安装根文件系统,并通过调用初始化脚本(通常是 /sbin/init)继续进行初始化。
两阶段(initrd)引导
内核引导、安装初始 ram 磁盘(initrd),执行进一步的可定制初始化,然后安装根文件系统并继续进行初始化(通常也是通过调用 /sbin/init)
这两种方法都有自己的优点和缺点。
一阶段引导
为了使用一阶段引导,我们需要构建内核,其具有安装内置根文件系统所需的所有驱动器(其他任何驱动器都可以在正常初始化过程中,在能够从根分区加载的模块中构建)。
如果我们要从非常小的设备引导(如软盘),最好的方法是构建的内核仅具有足够使我们可以安装根外置文件系统的内置驱动器 —— 然后将其他所有项构建为模块。例如,我内置了 SCSI 支持、PCMCIA 支持、IEE1394、SBP 和类似支持,但是其他所有项(包括显卡支持、网络设备支持等等)都作为模块构建,这些模块存储在根分区(在外置驱动器上)中,而不是软盘上。
使用简单(一阶段)引导过程,我们应该能够构建具有所需支持的内核,将其放在软盘驱动器中,在软盘中安装引导加载程序(我使用 GRUB,但还有其他选择,如 LILO),然后使用与此内核(对于 GRUB)相似的内核引导:
root (fd0)
kernel (fd0)/boot/bzImage root=/dev/sda1
这种方法基本上可以工作,但有两个问题:
1. 因为 SBP 支持使用 SCSI 仿真,为了检测磁盘和允许安装 /dev/sda1,需要“重新扫描”仿真的 SCSI 总线。这种扫描使用一组简单的命令执行。不过,遗憾的是,使用一阶段引导,我们不能运行任何命令,直到内核已经完成引导,而内核直到安装了根文件系统才能完成引导 —— 典型的自相矛盾困境。令人感到高兴的是,对于导致 SCSI 总线在启动时被扫描的 2.4 内核有可用的修补程序(有关更多详细信息,请参阅参考资料)。通过应用此修补程序,我可以使外置驱动器在引导过程中由内核自动检测,而不需要任何重新扫描命令。这使我们进入了下一个问题。
2. 内核中有定时窗口,这意味着内核经常在其能够被正确的监测和初始化之前尝试安装根设备。对于此问题,也有可用的修补程序(请参阅参考资料获得相关链接),它只是使内核在启动时等待很短的时间,并使其在安装根文件系统失败时重试(为外置驱动器提供时间识别)。
通过应用这两个修补程序,我可以成功地在可引导软盘上构建内核,其将引导,然后使用外置 FireWire 驱动器作为根。
这种方法的主要问题是需要我们给内核源码打补丁 —— 这最多是一件痛苦之事(当发行新的内核版本时),严重时会是个大问题(如果没有维护补丁程序与内核发生的其他更改保持一致的话)
您可能已经想到如果我们的 BIOS 支持 USB 或 FireWire 且我们直接引导,我们就可以避免这两个问题。不幸的是,情况并不是这样:虽然此方法在引导过程中使用 BIOS 调用来访问磁盘,一旦内核开始初始化,将不再使用 BIOS,而是使用内核驱动器访问磁盘 —— 这样就会遇到相同的问题。
两阶段引导
到了内核版本 2.0.X,向 Linux 内核添加了一项引人注意的能力 —— 使用“initial RAM disk”(或 initrd)提供两阶段引导过程。
简而言之,内核像平常一样引导;但不安装“真实的”根文件系统,而是在 RAM 中创建微型根文件系统并安装该系统。在安装真实的根、切换为使用真实的根并销毁 initial RAM disk 之前,任何步骤都可以在此初始环境中执行。
这在各种环境中都有用,但是为了便于说明,我们将仅使用我们的迷你环境重新扫描 SCSI 总线,等待外置磁盘被识别,然后切换为使用该磁盘作为真实的根继续引导。
为了使用这种方法,我们需要创建两项,内核和 initrd 映像。
内核就是具有内置 initrd 支持的普通内核。initrd 映像是包含我们的迷你根文件系统的回送文件系统映像(此映像可以使用 gzip 进行压缩以减少其大小)。
有关创建或定制自己的 initrd 映像的详细信息,可以查看参考资料部分。
在 initrd 映像中,有一个名为 linuxrc 的文件。当加载 initrd 时会执行此文件,所以确保其具有执行权限!我们为了进行说明,所以 linuxrc 非常简单:
清单1. initrd linuxrc
#!/bin/sh REAL_ROOT=/dev/sda1 # mount the /proc filesystem
mount -t proc none /proc #for scsi-emulation # modprobe sd_mod #for pcmcia
# modprobe pcmcia_core #for FireWire # modprobe ieee1394 # modprobe ohci1394
# modprobe raw1394 # modprobe sbp2 #for USB # modprobe usbcore # modprobe
ohci-hcd # modprobe uhci-hcd# modprobe usb-storage # loop rescanning the
scsi bus + rerunning devfsd retries=5 i=1 until [ -e $REAL_ROOT ] do if
[ $i -gt $retries ] then echo "Unable to mount real root ($REAL_ROOT)
- Giving up!" /bin/ash exit fi echo "Real root ($REAL_ROOT) not
found, retrying ($i)" sleep 1 echo "scsi addsingle-device 0 0 0" > /proc/scsi
/scsi echo "scsi add-single-device 1 0 0"> /proc/scsi/scsi echo "scsi add-
single-device 2 0"> /proc/scsi/scsi /bin/devfsd /dev -np i=$((i+1))
done #umount /proc asit will be remounted by the normal init process umount
/proc #now we simply exit, and the normal boot process should continue
exit 0
我们做的所有操作都是加载适当的模块来支持外置驱动器:它们应该根据需要被解注。(我在内核中构建了所有必需的支持,因此不需要任何模块。)然后我们进行循环,重新扫描 SCSI 总线(通过将命令回送到 /proc pseudo-filesystem 中的特殊文件,并调用 devfsd),直到出现根设备(我的例子中为 /dev/sda1)。在我的例子中,讨论的仿真 FireWire SCSI 总线是 1 0 0,不过也可以尝试其他的,而不会有任何负面影响 —— 如果您知道要使用的总线,可以裁剪脚本。同样,如果您有其他 SCSI 设备(或仿真 SCSI 设备),驱动器可能会有不同的字母(例如,/dev/sdb1)。如果不使用外置驱动器的第一个分区,则需要使用不同的编号(例如,/dev/sda2)。
现在所需要做的就是将相关文件复制到 initrd 映像中(可以使用 mount -o loop 命令安装未压缩的映像)。特别地,需要确保具有 linuxrc 文件、在其中使用的所有命令和那些命令依靠的所有库。然后,(未装载的)映像可以进行压缩。
接着把内核(bzImage)和 initrd 映像(initrd.gz)复制到(bootable, ext3)软盘中。
最后一步是在软盘中安装引导加载程序,并使用下列选项引导内核:kernel bzImage root=/dev/sda1 initrd=initrd.gz。
现在应该可以使用软盘进行引导:它将从软盘加载内核,将 initrd 映像加载到 RAM 中,等待识别根设备,然后像平常一样从那里继续引导。从此以后,可以移除软盘。
如果软盘不适合(例如,如果计算机没有软盘驱动器),则可以使用能够通过 BIOS 引导的任何设备。就个人而言,为了写作本文,我使用小的 32Mb USB 盘。或者,如果您不介意改变内置硬盘驱动器的话,为了更便于引导,可以在其中创建小的分区。
如果您非常迫切的想了解IT领域最新产品与技术信息,那么订阅至顶网技术邮件将是您的最佳途径之一。
现场直击|2021世界人工智能大会
直击5G创新地带,就在2021MWC上海
5G已至 转型当时——服务提供商如何把握转型的绝佳时机
寻找自己的Flag
华为开发者大会2020(Cloud)- 科技行者