使用ramdisk初始化initrd

本文详细介绍了initrd的工作原理及其在系统启动过程中的作用。initrd允许系统加载ramdisk作为临时根文件系统,从而使得系统能够在完全加载之前执行必要的初始化任务。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


  
  initrd提供了在boot loader下加载ram disk的方法。该ram disk可以被作为根文件系统挂载进来,里面的程序也可以运行。然后,新的根文件系统可以从其他设备挂载。之前的根(来自initrd)可以被转移到一个目录然后被卸载。
  
  initrd主要设计用来使系统启动于两个条件,一个是内核来自于非常小的驱动器,一个是额外的模块需要从initrd中加载。
  
  本文给出initrd的概要描述,更具体的可以参考[1]。
  
   操作
  -------
  当使用initrd,典型的系统启动顺序如下:
  1. boot loader加载内核并初始化ram disk
  2. 内核把initrd转化成正常的ram disk并释放initrd使用的内存
  3. initrd作为root被挂载,赋予读写权限。
  4. /linuxrc被执行(这可以是任何可执行文件,如脚本,运行在uid 0,可以做任何初始化)。
  5. linuxrc挂载真正的根文件系统
  6. linuxrc使用pivot_root系统调用把根文件系统放在根目录。
  7. 正常的启动序列(/sbin/init)在根文件系统上执行。
  8. initrd文件系统被移去。
  
  注意,改变根目录不牵扯卸载他。
  挂载在initrd的文件系统仍然可以被访问。
  
   启动命令选项
  ------------------------
  initrd添加了如下新的选项:
   initrd= (e.g. LOADLIN)
  加载特定的文件作为初始的ram disk。当使用lilo,你应该在/etc/lilo.conf中指定
  ram disk镜像文件位置,使用INITRD变量。
  
  noinitrd
  initrd数据被保留却不转化成ram disk,正常的根文件系统被加载。initrd的数据可以 从/dev/initrd中读取。注意,在initrd中的数据可以是任意结构的,不一定要是文件系统镜像。该选项多用于调试。
  
  注意:/dev/initrd是只读的,而且只能被使用一次。只要最后一个进程关闭它,所有的数据将会释放掉,而/dev/initrd将不再被打开。
  
   root=/dev/ram0 (without devfs)
   root=/dev/rd/0 (with devfs)
  
  initrd作为根文件系统被挂载,随后正常的启动顺序中,ram disk仍然作为root。
  
   安装
  ------------
  首先,用于initrd文件系统的目录在根文件系统上被创建:
   mkdir /initrd
  
  名字不是很重要。更多的可以参考pivot_root(2)的man。
  
  在启动过程中,如果根文件系统被创建(如果你有软盘启动),那么根文件系统应该创建/initrd目录。
  
  如果initrd没有被挂载,他里面的内容仍然可以被访问,通过以下的方法(注意,这在使用devfs时不可用):
   # mknod /dev/initrd b 1 250
   # chmod 400 /dev/initrd
  
  第二步,内核编译的时候必须支持ram disk并启动ram disk使能。所有从initrd中执行的程序必须被编译到内核中。
  
  第三步,你必须创建ram disk镜像。这可以通过在块设备上创建文件系统实现,把所需的文件拷贝,然后把文件拷贝到initrd文件。根据最新的内核,至少三种设备可以用:
  l 软盘(慢,却随处可用)
  l ram disk(快,需要物理内存)
  l loopback device(最高档)
  
   我将介绍 loopback device方法:
  保证loopback device被内核支持
  创建一个空的文件系统,拥有合适的大小:
  # dd if=/dev/zero of=initrd bs=300k count=1
  # mke2fs -F -m0 initrd
  如果空间不多,你可以用minix fs而不是ext2
  挂载文件系统
  # mount -t ext2 -o loop initrd /mnt
  创建控制台设备(如果用devfs,就没必要)
  # mkdir /mnt/dev
  # mknod /mnt/dev/console c 5 1
  拷贝initrd环境需要的所有文件。不要忘掉最重要的文件/linuxrc.
  注意,/linuxrc必须有执行权限。
  正确的initrd环境可以不用重启就可以通过命令行测试:
  # chroot /mnt /linuxrc
  卸载文件系统
  # umount /mnt
  现在initrd被保存在initrd文件中。还可以压缩他。
  # gzip -9 initrd
  为了试验initrd,你可以创建一个急救盘,只需创建/linuxrc到/bin/sh的符号链接。
  你还可以试着把initrd更小。
  
  最后,你要启动内核并加载initrd.几乎所有的boot loader都支持initrd.
  虽然启动进程兼容老的机制,下面的启动命令行还是得给出:
  
   root=/dev/ram0 init=/linuxrc rw如果不使用devfs.或者
   root=/dev/rd/0 init=/linuxrc rw
  如果使用devfs.
  
  使用 LOADLIN,只要执行
  LOADLIN initrd=
  e.g. LOADLIN C:/LINUX/BZIMAGE initrd=C:/LINUX/INITRD.GZ root=/dev/ram0
  init=/linuxrc rw
  
  使用LILO,你添加选项INITRD= 到/etc/lilo.conf中,使用APPEDN命令即可。
  image = /bzImage
  initrd = /boot/initrd.gz
  append = "root=/dev/ram0 init=/linuxrc rw"
  然后运行 /sbin/lilo
  
  对于其他的启动程序,请参考相应的文档。
  
  现在你可以启动并享受使用initrd的乐趣了
  
   如何改变根设备
  ------------------
  当完成他的任务,linuxrc改变根设备并启动真正的根设备。
  
  整个过程包括如下几步:
  1. 挂载新的根文件系统
  2. 把他变成根文件系统
  3. 删除所有对旧(Initrd)文件系统的访问
  4. 卸载initrd文件系统,重定位ram disk
  
  挂载新的根文件系统很容易:只需把他挂载到当前根下。例如:
  # mkdir /new-root
  # mount -o ro /dev/hda1 /new-root
  
  根的转变伴随着pivot_root系统调用。pivot_root把当前的根转移到新根下的一个目录,并把新的根放到原来的地方。老根的目录必须在调用pivot_root前退出。例如:
  # cd /new-root
  # mkdir initrd
  # pivot_root . initrd
  
  现在,linuxrc进程可以仍然访问老的根。所有的这些引用可以通过如下命令释放:
  # exec chroot . what-follows dev/console 2>&1
  
  然后执行的就是新根下的,例如/sbin/init.
  如果新的根文件系统使用devfs,而/dev目录还不可用,devfs必须被挂载进来,
  在使用chroot之前,只有这样才有/dev/console设备。
  
  注意:privot_root执行的情况可能有所不同。为了保证兼容性,注意以下几点:调用pivot_root之前,调用进程的当前目录应指向新的根文件目录使用.作为第一个参数,老根的相对路径作为第二个参数。chroot程序必须在就老的和新的根下都可以使用。最后改变根到新的根下。在执行命令中使用dev/console的相对路径。
  
  
  注意,initrd可以被卸载,而ram disk使用的内存可以被释放:
  # umount /initrd
  # blockdev --flushbufs /dev/ram0 # /dev/rd/0 if using devfs
  
  initrd还可以使用NFS-mounted根,参考man pivot_root(8)。
  
  注意:如果linuxrc或者其他程序从终端执行,老的改变根机制必须调用。
  
   使用限制
  -----------
  引入initrd的主要动机是为了在系统安装中可以模块化配置内核。过程运行如下:
  1. 一个小内核的系统从软盘或其他介质上启动(支持ram disk,initrd,a.out, ext2fs),并加载initrd。
  2. /linuxrc决定如何挂载真正的根文件系统(设备类型,设备驱动,文件系统),并分布媒体(CDROM,network,tape)。这些可以询问用户或者自动探测,或是混合使用。
  3. /linuxrc加载必须的内核模块
  4. /linuxrc创建并组装根文件系统(还不一定能用)。
  5. /linuxrc调用pivot_root改变根文件系统,执行(chroot)安装程序.
  6. boot loader 安装完毕
  boot loader配置用于加载initrd,initrd中包含系统启动的必须模块。
  现在系统可以启动的,额外的安装任务被执行。
  
  initrd的主要任务是在正常的系统操作中复用配置数据,而不需重新编译链接内核。
  
  第二个限制是在linux运行在不同的硬件配置上的安装问题。这种情况下,只需产生一个小内核,分别配置。initrd中必须含有所有需要的模块。然后,/linuxrc或其他一个文件可能有所不同。
  
  第三个限制更方便恢复磁盘,因为象根文件系统分区的信息在系统启动的时候要用,而在启动时刻是得不到的。
  
  最后,CDROM发布使他更方便安装。
  
   荒废的根改变机制
  ----------------
  下面的机制在pivot_root之前被使用。当前的内核仍然支持他,但是你不要依赖他。
  
  他在linuxrc存在的情况下,把真正的根设备作为根文件系统。initrd文件系统然后被
  卸载,或者把他移动到/initrd目录,如果这个目录存在于新的根文件系统。
  
  为了使用这种机制,你不必设定启动参数root,init,rw(如果设定了,这将影响到真正的
  根文件系统)。

本文转自
http://blog.21ic.com/user1/1066/archives/2005/2331.html
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值