Linux Boot Sequence
Linux Boot Sequence
Linux Boot Sequence
Introduction
This article describes how a Linux system boots up, what are the steps involved, which scripts run, what configuration files are read and their order, from turning on the system till getting the login prompt. Although this article projects a general view of booting a Linux system, but some configuration files and commands can be Red Hat specific.
BIOS Initialization
BIOS, the Basic Input Output System is a firmware program that performs a very basic level of interaction with hardware. This is the first program that takes control when the computer is powered on. The BIOS performs a test on all the hardware component and peripherals called POST, the Power On Self Test. It initializes required hardware for booting. After POST executes successfully, BIOS looks for a boot device from a list of devices. Modern BIOSs allow you to configure this order of devices (sometimes called boot preference) that BIOS checks for booting. These boot devices can be any one of floppy drive, CDROM, hard drive, a network interface or other removable media (such as USB flash drive). The BIOS checks for the boot sector on the bootable device. Boot sector is the first physical sector on the storage device, and contains the code required for booting the machine.
The first 446 bytes of MBR contain the code that locates the partition to boot from. The rest of booting process takes place from that partition. This partition contains a software program for booting the system called the 'bootloader'.
GRUB
According to gnu.org, "A bootloader is the first software program that runs when a computer starts". GRUB or GRand Unified Bootloader is the bootloader program for Linux like operating systems. This is a multi-stage bootloader. Generally, the first stage of GRUB resides in first 446 bytes of MBR, whose only purpose is to locate the second stage. So, the code (in first 446 bytes) explained in the previous section is in fact the first stage if GRUB is the bootloader. The second stage is loaded into the memory from the boot partition. There is an alternative way of invoking GRUB, as a secondary bootloader. The first stage of GRUB in this case does not reside in the MBR, but in the boot sector of some partition (remember that MBR is not located in any of the partitions). Now, MBR must contain some other bootloader code, which is configured to pass the control to Linux bootloader (i.e. GRUB). But even in this case, the first stage lies within the first 446 bytes of boot sector of that partition. The second stage of GRUB does most of the work for booting the Linux system. Once the second stage is loaded into the memory, GRUB configuration file is read. The GRUB configuration file for Red Hat Enterprise Linux is /boot/grub/grub.conf. Here is a sample grub.conf: # grub.conf generated by anaconda # # Note that you do not have to rerun grub after making changes to this file # NOTICE: You have a /boot partition. This means that # all kernel and initrd paths are relative to /boot/, eg. # root (hd0,0) # kernel /vmlinuz-version ro root=/dev/VolGroup00/LogVol00 # initrd /initrd-version.img #boot=/dev/sda default=0 timeout=5 splashimage=(hd0,0)/grub/splash.xpm.gz hiddenmenu title Red Hat Enterprise Linux Server (2.6.18-238.el5) root (hd0,0) kernel /vmlinuz-2.6.18-238.el5 ro root=/dev/VolGroup00/LogVol00
initrd /initrd-2.6.18-238.el5.img title Windows XP Pro rootnoverify (hd0,1) chainload +1 GRUB allows you to choose from a list of available operating systems. The different OSs have different title in grub.conf. Corresponding to this configuration file, the user at GRUB menu will be asked to choose from two options: Red Hat Enterprise Linux Server (2.6.18238.el5), and Windows XP Pro. The commands following the title are the commands that run in the background when an entry is selected from the GRUB menu. For example, when the user selects first option of Red Hat Enterprise Linux, following three commands execute: root (hd0,0) kernel /vmlinuz-2.6.18-238.el5 ro root=/dev/VolGroup00/LogVol00 initrd /initrd-2.6.18-238.el5.img Corresponding to this menu entry, the first command, i.e. "root (hd0,0)" specifies the partition on which a compressed kernel image and initrd file are located. The second command i.e. "kernel /vmlinuz-2.6.18-238.el5 ro root=/dev/VolGroup00/LogVol00" tells which kernel image to use (in this case vmlinuz-2.6.18-238.el5). The arguments to this command are 'ro' and 'root'. 'root' specifies the device on which root directory of the filesystem (i.e. / directory) is located; 'ro' means that this partition is to be mounted in read only mode (i.e. the kernel mounts the root partition in read only mode). Note that the partition for root filesystem and the partition on which this kernel image resides (i.e. boot partition) are different. The third command is the location of initrd. Before going into the details of what initrd is, let's look at a problem caused at the boot time. The Chicken/Egg Module Problem and initrd The kernel needs to mount the root filesystem (as specified in the second command above). This filesystem may be on some partition with one of the following capabilities: Logical Volume Management Software RAID NFS (Network File System) Some other encrypted partition
The Linux kernel does not have these features compiled into it. But they are present as modules. So the kernel needs to load these modules in order to mount the root filesystem. These modules are present under /lib/modules/ directory, which is present on the root filesystem itself. But this root filesystem is not mounted yet (that is what we have been trying to do so far). So how can kernel access the modules for mounting a filesystem
which are present on the filesystem itself (without mounting it)? The kernel and GRUB provide a solution through initrd, the initial RAM disk. It contains the modules needed to mount the filesystem, and only modules needed for that filesystem are included. GRUB also supports chainloading, the method used to pass control to other bootoader. Chainloading is used by GRUB to boot operating systems like windows. Thi s can be checked in the above configuration file, under the title Windows XP Pro.
Kernel Initialization:
After GRUB stage 2, the location of kernel and necessary modules through initrd are known. Now the kernel is loaded into memory and initialized. The initrd image is compiled and mounted into the memory. It serves as a temporary root file system and helps kernel to boot properly without mounting any root file system. Now that all the drivers are loaded into memory, and kernel has booted, kernel mounts the root filesystem in read only mode, and starts the first process.
Alternate single user mode Bypass rc.sysinit (discussed later in this article)
Summarizing runlevels: Runlevel 1 is used for maintenance purpose, because this is very limited runlevels. Only the minimum scripts run in this runlevel. Only root user can log in. No other user can log into this runlevel. Runlevel 2 is more relaxed than runlevel 1. Here all the users can login, but networking service is not running. Runlevel 3 provides a full working environment. All users can log in, networking is enabled. Runlevel 4 is for experiment purpose only. In runlevel 5, graphical console is available. Runlevel 0 is the halt state of system, and switching to runlevel 6 will reboot the system. Now that we are clear with runlevels, let's continue our discussion about init process and its configuration file, inittab. 'inittab' file for RHEL is: # # inittab This file describes how the INIT process should set up # the system in a certain run-level. # # Author: Miquel van Smoorenburg, <miquels@drinkel.nl.mugnet.org> # Modified for RHS Linux by Marc Ewing and Donnie Barnes # # Default runlevel. The runlevels used by RHS are: # 0 - halt (Do NOT set initdefault to this) # 1 - Single user mode # 2 - Multiuser, without NFS (The same as 3, if you do not have networking) # 3 - Full multiuser mode # 4 - unused # 5 - X11 # 6 - reboot (Do NOT set initdefault to this) # id:3:initdefault: # System initialization. si::sysinit:/etc/rc.d/rc.sysinit l0:0:wait:/etc/rc.d/rc 0 l1:1:wait:/etc/rc.d/rc 1 l2:2:wait:/etc/rc.d/rc 2
3 4 5 6
# Trap CTRL-ALT-DELETE ca::ctrlaltdel:/sbin/shutdown -t3 -r now # When our UPS tells us power has failed, assume we have a few minutes # of power left. Schedule a shutdown for 2 minutes from now. # This does, of course, assume you have powerd installed and your # UPS connected and working correctly. pf::powerfail:/sbin/shutdown -f -h +2 "Power Failure; System Shutting Down" # If power was restored before the shutdown kicked in, cancel it. pr:12345:powerokwait:/sbin/shutdown -c "Power Restored; Shutdown Cancelled"
# Run gettys in standard runlevels 1:2345:respawn:/sbin/mingetty tty1 2:2345:respawn:/sbin/mingetty tty2 3:2345:respawn:/sbin/mingetty tty3 4:2345:respawn:/sbin/mingetty tty4 5:2345:respawn:/sbin/mingetty tty5 6:2345:respawn:/sbin/mingetty tty6 # Run xdm in runlevel 5 x:5:respawn:/etc/X11/prefdm -nodaemon The general format of entries in this file is: id:runlevel(s):action:process "id" is the unique sequence of characters for identifying an entry in this file. "runlevels" are the runlevels for which that entry is applicable. "action" is the action that is to be taken for the runlevel. "process" specifies the process that is to be executed. Now let's look at some entries in this file. The first uncommented line, i.e. "id:3:initdefault:"
specifies the default runlevel in which the system will boot. According to this file, the system will boot into runlevel 3 by default. One more important line is system initialization line (the one with id: si) si::sysinit:/etc/rc.d/rc.sysinit This line tells init process to execute "/etc/rc.d/rc.sysinit" script. This is the first script executed by init during booting process.
rc.sysinit
When this script executes, it asks the user for interactive setup (Press 'I' to enter interactive startup). This script performs a lot of functions for the system that include: Setting hostname It checks and mounts filesystems (/proc, /sys, and others in /etc/fstab). Enables SWAP Sets SELinux Configures kernel parameters from sysctl.conf Sets system clock Loads required modules Setting a minimal path Initializes required serial and other ports Enables RAID and LVM And the root filesystem, which was mounted read-only earlier (as specified in GRUB configuration file), is checked and remounted read-write. After this script has executed, the runlevel specific files are run, according to "/etc/inittab". In our file, the default runlevel was 3. So the line corresponding to this runlevel is: l3:3:wait:/etc/rc.d/rc 3 There are directories corresponding to each runlevel under "/etc/rc.d" directory. According to this line, the scripts in /etc/rc.d/rc3.d directory are run. Lets list the files in this directory. (Other directories like rc1.d, rc2.d ... have similar files in them). [root@redhat-server ~]# ls -l /etc/rc.d/rc3.d/ total 304 lrwxrwxrwx 1 root root 17 Jul 3 03:51 K01dnsmasq -> ../init.d/dnsmasq lrwxrwxrwx 1 root root 24 Jul 3 03:51 K02avahi-dnsconfd -> ../init.d/avahi-dnsconfd lrwxrwxrwx 1 root root 24 Jul 3 03:54 K02NetworkManager ->
../init.d/NetworkManager lrwxrwxrwx 1 root root 16 ../init.d/conman lrwxrwxrwx 1 root root 19 ../init.d/saslauthd lrwxrwxrwx 1 root root 17 ../init.d/wdaemon lrwxrwxrwx 1 root root 16 ../init.d/psacct lrwxrwxrwx 1 root root 13 lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 19 ../init.d/vncserver lrwxrwxrwx 1 root root 17 ../init.d/winbind lrwxrwxrwx 1 root root 20 ../init.d/netconsole lrwxrwxrwx 1 root root 16 ../init.d/vsftpd lrwxrwxrwx 1 root root 20 ../init.d/rpcsvcgssd lrwxrwxrwx 1 root root 16 ../init.d/ypbind lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 15 ../init.d/kdump lrwxrwxrwx 1 root root 15 ../init.d/mdmpd lrwxrwxrwx 1 root root 20 ../init.d/multipathd lrwxrwxrwx 1 root root 24 ../init.d/wpa_supplicant lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 18 ../init.d/netplugd lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 15 ../init.d/rdisc lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 25 ../init.d/readahead_later
Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul
3 03:50 K05conman -> 3 03:51 K05saslauthd -> 3 03:52 K05wdaemon -> 3 03:50 K10psacct -> 3 03:53 K20nfs -> ../init.d/nfs 3 03:53 K24irda -> ../init.d/irda 3 03:53 K35vncserver -> 3 03:56 K35winbind -> 3 03:51 K50netconsole -> 5 10:10 K50vsftpd -> 3 03:53 K69rpcsvcgssd -> 3 03:56 K73ypbind -> 3 3 3 3 03:52 03:49 04:00 03:49 K74ipmi -> ../init.d/ipmi K74nscd -> ../init.d/nscd K74ntpd -> ../init.d/ntpd K80kdump ->
3 03:52 K85mdmpd -> 3 03:49 K87multipathd -> 3 03:51 K88wpa_supplicant -> 3 03:52 K89dund -> ../init.d/dund 3 03:49 K89netplugd -> 3 03:52 K89pand -> ../init.d/pand 3 03:49 K89rdisc -> 3 03:53 K91capi -> ../init.d/capi 3 03:52 K99readahead_later ->
lrwxrwxrwx 1 root root 23 ../init.d/microcode_ctl lrwxrwxrwx 1 root root 25 ../init.d/readahead_early lrwxrwxrwx 1 root root 15 ../init.d/kudzu lrwxrwxrwx 1 root root 16 ../init.d/iscsid lrwxrwxrwx 1 root root 19 ../init.d/ip6tables lrwxrwxrwx 1 root root 18 ../init.d/iptables lrwxrwxrwx 1 root root 18 ../init.d/mcstrans lrwxrwxrwx 1 root root 14 lrwxrwxrwx 1 root root 17 ../init.d/network lrwxrwxrwx 1 root root 16 ../init.d/auditd lrwxrwxrwx 1 root root 21 ../init.d/restorecond lrwxrwxrwx 1 root root 16 ../init.d/syslog lrwxrwxrwx 1 root root 18 ../init.d/cpuspeed lrwxrwxrwx 1 root root 20 ../init.d/irqbalance lrwxrwxrwx 1 root root 15 ../init.d/iscsi lrwxrwxrwx 1 root root 17 ../init.d/portmap lrwxrwxrwx 1 root root 17 ../init.d/nfslock lrwxrwxrwx 1 root root 19 ../init.d/mdmonitor lrwxrwxrwx 1 root root 19 ../init.d/rpcidmapd lrwxrwxrwx 1 root root 17 ../init.d/rpcgssd lrwxrwxrwx 1 root root 20 ../init.d/messagebus lrwxrwxrwx 1 root root 24 ../init.d/setroubleshoot
Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul Jul
3 03:52 S00microcod e_ctl -> 3 03:52 S04readahead_early -> 3 03:54 S05kudzu -> 3 03:51 S07iscsid -> 3 03:49 S08ip6tables -> 3 03:49 S08iptables -> 3 03:51 S08mcstrans -> 3 03:53 S09isdn -> ../init.d/isdn 3 03:51 S10network -> 3 03:49 S11auditd -> 3 03:51 S12restorecond -> 3 03:51 S12syslog -> 3 03:49 S13cpuspeed -> 3 03:52 S13irqbalance -> 3 03:51 S13iscsi -> 3 03:51 S13portmap -> 3 03:53 S14nfslock -> 3 03:52 S15mdmonitor -> 3 03:53 S18rpcidmapd -> 3 03:53 S19rpcgssd -> 3 03:51 S22messagebus -> 3 03:53 S23setroubleshoot ->
lrwxrwxrwx 1 root root ../init.d/bluetooth lrwxrwxrwx 1 root root ../init.d/netfs lrwxrwxrwx 1 root root ../init.d/pcscd lrwxrwxrwx 1 root root ../init.d/acpid lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root ../init.d/haldaemon lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root ../init.d/lvm2-monitor lrwxrwxrwx 1 root root ../init.d/autofs lrwxrwxrwx 1 root root ../init.d/hplip lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root ../init.d/rawdevices lrwxrwxrwx 1 root root ../init.d/xinetd lrwxrwxrwx 1 root root ../init.d/sendmail lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root ../init.d/crond lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root ../init.d/anacron lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root ../init.d/rhnsd lrwxrwxrwx 1 root root ../init.d/yum-updatesd lrwxrwxrwx 1 root root ../init.d/avahi-daemon lrwxrwxrwx 1 root root ../init.d/firstboot lrwxrwxrwx 1 root root lrwxrwxrwx 1 root root
19 Jul 15 Jul 15 Jul 15 Jul 14 Jul 19 Jul 14 Jul 22 Jul 16 Jul 15 Jul 14 Jul 14 Jul 20 Jul 16 Jul 18 Jul 13 Jul 15 Jul 13 Jul 17 Jul 13 Jul 15 Jul 22 Jul 22 Jul 19 Jul 11 Jul 16 Jul
3 03:52 S25bluetooth -> 3 03:51 S25netfs -> 3 03:51 S25pcscd -> 3 03:52 S26acpid -> 3 03:54 S26apmd -> ../init.d/apmd 3 03:54 S26haldaemon -> 3 03:52 S26hidd -> ../init.d/hidd 3 03:52 S26lvm2-monitor -> 3 03:50 S28autofs -> 3 03:52 S50hplip -> 3 03:52 S55sshd -> ../init.d/sshd 3 03:51 S56cups -> ../init.d/cups 3 03:51 S56rawdevices -> 3 03:52 S56xinetd -> 3 03:51 S80sendmail -> 3 03:49 S85gpm -> ../init.d/gpm 3 03:51 S90crond -> 3 03:53 S90xfs -> ../init.d/xfs 3 03:49 S95anacron -> 3 03:52 S95atd -> ../init.d/atd 3 03:54 S97rhnsd -> 3 03:52 S97yum-updatesd -> 3 03:51 S98avahi-daemon -> 3 03:54 S99firstboot -> 3 03:51 S99local -> ../rc.local 3 03:52 S99smartd ->
../init.d/smartd Some files in this directory start with S and others with K. The files starting with S correspond to the scripts that are to be 'started' in that particular runlevel, and the ones with K correspond to the ones that are to be 'killed'. These files are just soft links to scripts under "/etc/rc.d/init/d" directory (One soft link points to "/etc/rc.local" which itself is a soft link to "/etc/rc.d/rc/local"). The scripts in "/etc/rc.d/init.d/" are daemons. Daemons are the processes that run in background and provide some kind of service, like http daemon (httpd) provides web service. After all these scripts have been executed, then finally "/etc/rc.local" script runs. If there is some command or script that you want to be executed at system startup, you can put it in this script. [root@redhat-server ~]# cat /etc/rc.local #!/bin/sh # # This script will be executed *after* all the other init scripts. # You can put your own initialization stuff in here if you d on't # want to do the full Sys V style init stuff. touch /var/lock/subsys/local Now that all these scripts have been executed successfully, login prompt appears. Red Hat Enterprise Linux Server release 5.6 (Tikanga) Kernel 2.6.18-238.el5 on an i686 redhat-server login: