马哥教育网络班21期-第十周课程练习​

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

1.png

打开一个ks文件

2.png

根据需要配置好相关的内容,将文件另存为myks.cfg

(2)创建引导镜像

在/tmp 目录下创建目录myiso ,在myiso 目录下创建isolinux 目录;

#mkdir -p /tmp/myiso/isolinux

将光盘上的isolinux 下的文件都拷贝到新建的目录下;

#cp /media/isolinux/* /tmp/myiso/isolinux/ 
将myks.cfg文件复制到/tmp/myiso 目录下;

3.png

创建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条)

  • 马哥教育
    马哥教育 2016-09-27 09:31

    写的很好,如果脚本在添加一些判断条件的会更好一些