1、请详细描述CentOS系统的启动流程(详细到每个过程系统做了哪些事情)
流程顺序:
POST(加电自检) –> BootSequence (BIOS) –> Bootloader(MBR) –> kernel(ramdisk) –> rootfs(只读方式) –> switchroot –> /sbin/init –> (/etc/inittab,/etc/init/*.conf)(PS:用户空间,根据配置文件进行加载)
–> 设定运行级别 –> 系统初始脚本 –> 关闭或启动对应级别下的服务 –> 启动终端
各阶段说明:
(1)POST:加点自检,检查硬件设备是否存在
(2)BootSequence
(3)kernel自身初始化,实现功能
a 探测可识别到的所有硬件设备;
b 加载硬件驱动程序;(有可能会借助于ramdisk加载驱动)
c 以只读方式挂载根文件系统;
d 运行用户空间的第一个应用程序:/sbin/init
(4)/sbin/init管理用户空间服务进程
(5)启动运行级别初始化控制:/etc/rc.d/rc#.d
(6)系统初始化脚本:/etc/rc.d/rc.sysinit
(7)启动终端
启动开机流程总结:
内核级别:
1. POST做开机启动时候的硬件检测功能
2. BootSequence(BIOS)启动加载主引导分区MBR中的引导加载器程序BootLoader
在Linux现行的BootLoader是三段划分(打破446字节限制)的GRUB程序,
stage1:MBR(Master Boot Record);
stage1.5:MBR之后的扇区,让stage1中的bootloader能识别stage2所在的分区上的文件系统;
stage2:磁盘分区(/boot/grub/)
3.Kernel识别硬件、加载驱动、只读挂载根文件系统、同时交付给用户空间第一个程序/sbin/init
(PS:此处特别要注意,系统发行商为了适应多种硬件接口驱动调用,会在第一次安装系统时候,自动识别硬件接口, 并调用唯一驱动程序来生成ramdisk文件,以内存当磁盘做虚根,驱动接口后会切换到真实的根文件系统上)
CentOS 5系列是initrd,当磁盘映像文件会造成二次缓存缓冲
CentOS 6/7系列改进为initramfs,以文件系统形式可以不二次占用缓存和缓冲
从内核空间切换到用户空间
用户空间级别
4./sbin/init接管后,根据其配置文件来初始化;
5.根据/sbin/init中的配置会设置默认运行级别,以及一些在/etc/init.d/设置的开机服务;
6./etc/rc.d/rc.sysinit运行系统初始化脚本,完成系统初始化;
7.关闭对应级别下需要停止的服务,启动对应级别下需要开启的服务;
8.设置登录终端
2、为运行于虚拟机上的CentOS 6添加一块新硬件,提供两个主分区;
(1) 为硬盘新建两个主分区;并为其安装grub;
(2) 为硬盘的第一个主分区提供内核和ramdisk文件; 为第二个分区提供rootfs;
(3) 为rootfs提供bash、ls、cat程序及所依赖的库文件;
(4) 为grub提供配置文件;
(5) 将新的硬盘设置为第一启动项并能够正常启动目标主机;
新增一块硬盘,识别为/dev/sdc,
#fdisk /dev/sdc
(n;p;+5G;w)
新建两个主分区/dev/sdb1 /dev/sdb2
(1)
在/mnt 下新建目录boot rootfs,执行如下命令
#mkdir /mnt/boot /mnt/rootfs #mount /dev/sdc1 /mnt/boot #/mount /dev/sdc2 /mnt/rootfs #grub-install --root-directory=/mnt /dev/sdc #cd /mnt/boot/grub #ls
(2)
#cp /boot/vmlinuz-2.6.32-431.el6.x86_64 /mnt/boot/vmlinuz-2.6.32-431.el6.x86_64 #cp /boot/initramfs-2.6.32-431.el6.x86_64.img /mnt/boot/vmlinuz-2.6.32-431.el6.x86_64 #cd /mnt/rootfs #mkdir bin boot dev etc home lib lib64 media misc mnt opt proc root sbin srv sys users usr var
(3)
#cp /bin/bash /mnt/roofs/bin #cp /bin/ls /mnt/roofs/bin #cp /bin/cat /mnt/rootfs/bin #for i in `ldd /bin/bash | grep -o "[^[:space:]]*/lib[^[:space:]]*"`; do cp $i /mnt/rootfs/lib64/ ; done #for i in `ldd /bin/ls| grep -o "[^[:space:]]*/lib[^[:space:]]*"`; do cp $i /mnt/rootfs/lib64/ ; done #for i in `ldd /bin/cat | grep -o "[^[:space:]]*/lib[^[:space:]]*"`; do cp $i /mnt/rootfs/lib64/ ; done
(4)
#vim /mnt/boot/grub/grub.conf
default=0
timeout=5
title CentOS-hyc
root (hd0,0)
kernel /vmlinuz-2.6.32-431.el6.x86_64 ro root=/dev/sdc1 selinux=0 init=/bin/bash
initrd /initramfs-2.6.32-431.el6.x86_64.img
3、制作一个kickstart文件以及一个引导镜像。描述其过程。
(1)制作一个kickstart文件
安装system-config-kickstart
#yum install -y system-config-kickstart
#system-config-kickstart
打开一个ks文件
根据需要配置好相关的内容,将文件另存为myks.cfg
(2)创建引导镜像
在/tmp 目录下创建目录myiso ,在myiso 目录下创建isolinux 目录;
#mkdir -p /tmp/myiso/isolinux
将光盘上的isolinux 下的文件都拷贝到新建的目录下;
#cp /media/isolinux/* /tmp/myiso/isolinux/
将myks.cfg文件复制到/tmp/myiso 目录下;
创建ISO文件系统
# mkisofs -R -J -T -v –no-emul-boot –boot-load-size 4 –boot-info-table -V "CentOS 6.8 x86_64 boot" -b /tmp/myiso/isolinux/isolinux.bin -c /tmp/myiso/isolinux/boot.cat -o /root/boot.iso myiso
4、写一个脚本
(1) 能接受四个参数:start, stop, restart, status
start: 输出“starting 脚本名 finished.”
…
(2) 其它任意参数,均报错退出;
#!/bin/bash # case "$1" in start) echo "starting $0 finished" ;; stop) echo "$0 is stopped" ;; restart) echo "$0 is restarted" ;; status) echo "$0 status" ;; *) echo "Usage: $0 {start|stop|restart|status}" || true exit 1 esac
5、写一个脚本,判断给定的用户是否登录了当前系统;
(1) 如果登录了,则显示用户登录,脚本终止;
(2) 每3秒钟,查看一次用户是否登录;
#!/bin/bash # read -p "Please enter your user name:" username while true ;do currentU=`who | cut -d" " -f1|sort |uniq | xargs` if echo $currentU | grep -w ${username};then echo "$username is longin" exit fi sleep 3 done
6、写一个脚本,显示用户选定要查看的信息;
cpu) display cpu info
mem) display memory info
disk) display disk info
quit) quit
非此四项选择,则提示错误,并要求用户重新选择,只到其给出正确的选择为止;
#!/bin/bash # cat << EOF ------------------------------- cpu ) show cpu information; mem ) show memory information; disk) show disk information; quit) quit ------------------------------ EOF read -p "Please slect one option above:" option case "$option" in cpu) lscpu ;; mem) cat /proc/meminfo ;; disk) fdisk -l ;; quit) echo "Quit..." exit 0 ;; *) echo "wrong args,select option 1) cpu 2) mem 3) disk 4) quit" /bin/bash $0 esac
7、写一个脚本
(1) 用函数实现返回一个用户的UID和SHELL;用户名通过参数传递而来;
(2) 提示用户输入一个用户名或输入“quit”退出;
当输入的是用户名,则调用函数显示用户信息;
当用户输入quit,则退出脚本;进一步地:显示键入的用户相关信息后,再次提醒输出用户名或quit:
#!/bin/bash # info() { id $username &> /dev/null if [ $? -ne 0 ];then echo "the user is not exist!" else grep "^$username" /etc/passwd |awk -F: '{printf "USER ID:%u\nUSER SHELL:%s\n" ,$3,$NF}' fi } while true;do read -p "please input an username or quit:" username if [ $username == "quit" ];then exit 0 else info fi done
8、写一个脚本,完成如下功能(使用函数)
(1) 提示用户输入一个可执行命令的名字;获取此命令依赖的所有库文件(ldd);
(2) 复制命令文件至/mnt/sysroot目录下的对应的rootfs的路径上,例如,如果复制的文件原路径是/usr/bin/useradd,则复制到/mnt/sysroot/usr/bin/目录中;
(3) 复制此命令依赖的各库文件至/mnt/sysroot目录下的对应的rootfs的路径上;规则同上面命令相关的要求
#!/bin/bash # DEST_PATH=/mnt/sysroot read -p "please input one executable command:" CMD path=`which ${CMD}` pri_lib() { echo "the shared library dependencies:" ldd ${path} } cmd_lib() { CPATH=`echo $1 | grep -v "^alias" | grep -o "/.*/"` [ ! -d $DEST_PATH$CPATH ] && mkdir -p $DEST_PATH$CPATH [ ! -e $DEST_PATH$1 ] && cp $1 $DEST_PATH$CPATH } lib_cp() { for i in `ldd $1 | grep -o "/[^[:space:]]*"`; do cmd_lib $i done echo "lib copy complete" } pri_lib cmd_lib ${path} lib_cp ${path}
原创文章,作者:π,如若转载,请注明出处:http://www.178linux.com/49175
评论列表(1条)
写的很好,如果脚本在添加一些判断条件的会更好一些