Linux外围文件系统的定制方法

前言

一般来说,我们所说的Linux系统指的是各种基于Linux Kernel和GNU Project的操作系统发行版。为了掌握Linux操作系统的使用,了解 Linux操作系统的运作过程,理解内核与外围支撑系统的关系,加深对开源操作系统的认识,我决定造个轮子——自己定制一个Linux文件系统。

这里有两种实现方法:

直接自己实现init**\*(M1)***

加载bios 的硬件信息-> 读取MBR –>执行Grub ->加载kernel–> 加载驱动–> init –> 执行bash

利用系统/sbin/init**\*(M2)***

加载bios 的硬件信息-> 读取MBR –>执行Grub ->加载kernel–> 加载驱动–> init –> /sbin/init -> 取得run-level信息 -> /etc/rc.d/rc.sysinit -> services –> /etc/rc.d/rc.local –> mingetty –> login

我们先选择*M1*。

思路

  • 利用原有系统复制必备部件到新存储器
  • 利用initrd.img机制在RAM Disk中测试
  • 搭配原文件内核和模块启动

Step1:获得shell版本的initrd.img

首先,我们可以写一个脚本init,使得内核用该文件系统启动后能够直接获得一个Bash。

创建脚本 init

其中:/bin目录下是常用命令,init是自己写的脚本,/lib64目录下是应用程序所依赖的动态库。

init 内容

现在我们需要使用命令行,创建bin和sbin目录,向其中添加bash、ls、rm、cp、mv、echo、cat、less等基础命令。由于这些命令需要依赖/lib64等目录下的一些动态链接的共享库,所以需要将依赖的库拷贝到小系统对应的目录下,用ldd命令查询应用程序及其依赖的动态库。完成之后,执行:

find . | cpio -H newc -o | gzip > /boot/initrd.img

将根文件系统打包成initrd.img放到/boot目录下。启动时系统会自动执行initrd.img中的init。

费了这么大劲生成initrd.img,如何测试新建的initrd.img呢,需要在grub启动配置文件当中增加一个入口用于测试。

title CentOS 6 Mini
root (hd0,0)
kernel /vmlinuz-2.6.32-642.el6.x86_64
initrd /initrd.img

这样重启之后就会出现启动选项了。

Step2:完成挂载原系统能力

为了能挂载原系统,必须在initrd.img中加载原系统运行所必须的驱动模块,比如ext4文件系统的驱动、scsi设备的相关驱动等,/sbin/modinfo 配合/sbin/insmod,驱动放到/module

Step3:完成拥有管理设备能力(udev)

利用管理、监控主机设备的服务程序udevd来自动加载所需的驱动模块,比我们自己实现更加可靠。udevd的规则文件在/lib/udev/目录下,配置文件在/etc/udev/目录下,同时还需要/etc/nsswitch.conf配置的名称服务交换,其依赖的库为/lib目录下以libnss开头的文件,将上述文件拷贝到我们的目录下,然后使用/sbin/start\_udev命令可以启动udevd服务。(udevd需要调用一些其他的系统命令,如/sbin/modprobe,可用strace进行跟踪获取)。

小系统的目录文件

其中:/dev目录下是系统存放可用设备的目录,/log是使用strace命令生成的log记录文件。

Step4:完成拥有login登录能力

由于login的机制比较复杂,涉及进程管理机制和进程组、控制台等许多方面,因此我们采用*M2*,将/sbin/init命令拷到小系统目录下,init脚本改为

#!/bin/bash
exec /sbin/init

将控制权交给/sbin/init之后,系统启动时就必须等到它完成一系列调用之后,进入login界面,用户才能重新获得控制权。

/sbin/init的过程大致分为三块:第一块是udevd加载驱动模块、文件系统检查和根切换,相关配置在/etc/rc.sysinit中;第二块是启动各项服务,相关配置在/etc/rc.d/目录下;第三块是登录部分,需要调用/sbin/mingetty和/bin/login等命令。将上述所涉及的命令及文件拷贝到小系统对应的目录下,并对配置进行修改。

由于小系统启动之后initrd.img作为临时根文件系统直接在内存中运行,而我们小系统不需要进行根切换,故将/etc/rc.sysinit中remount\_needed()函数体注释掉,这样就不会根切换了。

由于系统采用了全新的Upstart启动方式(/sbin/init程序已经改由upstart软件包提供),将与Upstart启动相关的配置文件拷贝至小系统目录下:

/etc/inittab 配置默认运行级别
/etc/init/rcS.conf 加载rc.sysinit脚本,完成系统初始化任务
/etc/init/rc.conf 兼容脚本,负责各运行级别的调用处理
/etc/init/rcS-sulogin.conf 为单用户模式启动/sbin/sushell环境
/etc/init/control-alt-delete.conf 控制终端下的Ctrl+Alt+Del热键操作
/etc/init/start-ttys.conf 配置tty终端的开启数量、设备文件
/etc/sysconfig/init 控制tty终端的开启数量、终端颜色方案
/etc/init/tty.conf 控制tty终端的开启

将bootmini/etc/inittab的运行优先级改为2,那么系统启动时/sbin/init将执行bootmini/etc/rc.d/rc2.d/目录下以S开头的文件,将一些不需要开启的服务文件名改为K开头。

在bootmini/etc/rc.d/rc.local文件中可以加入用户需要系统开机启动后自动执行的操作。

login程序基于认证体系PAM, 配置文件在/etc/pam.d/目录下,相关库文件有/lib64/security/及其依赖的库文件;login还涉及用户组管理/bin/chgrp、/bin/chown、/bin/chmod等,保存用户名的文件/etc/passwd、/etc/group,用户密码文件为/etc/shadow。其他一些涉及的文件可通过strace来帮助分析。

可在真机上运行的完整版小系统

部分目录文件:

/etc

/bin

/sbin

/usr/bin

/usr/sbin

至此,文件系统算是可以跑了。下一篇我们再造个轮子——进行Linux内核的裁剪。真机效果也将在下篇看到。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Linux常见问题解决方案汇总

    1.VMware下的Centos7联网并设置固定ip 1).首先右击虚拟镜像名,点击"设置" 2).打开终端,查看网络 3).使用root用户编辑此文件 4).重启网卡,查看网络 5).再编辑一下之前编辑的文件 [root@nginx ~]# vi /etc/sysconfig/network-scripts/ifcfg-ens33 #IPADDRO和PREFIXO如上图,GATWAYO和DNS1只需要把IPADDRO最后改为1 IPADDRO=192.168.126.136 PREF

  • pyqt远程批量执行Linux命令程序的方法

    写了个小程序: 功能 1.测试远程ssh连接是否成功, 2.批量执行远程ssh命令 效果如下: 代码如下: #-*- coding:utf-8 -*- import sys from PyQt4 import QtCore, QtGui, uic import locale import re import os from PyQt4.QtCore import * from PyQt4.QtGui import * import paramiko qtCreatorFile = "test.u

  • 让DOSBox启动后自动执行命令的方法讲解

    使用DOSBox,可以在win下模拟DOS,自些好玩的工作.例如,学习8086汇编. 每次启动DOSBox后,都要挂载.转盘符.遇上调试的程序老死,就不好玩了. 可以想想办法,让这些固定"套路"自动化. 注意到DOSBox初启时,有一个窗口,如下显示: 就这个文件,掌管DOSBox启动后执行的命令. 找到这个文件. 用记事本就可以编辑. 拉到最下面,找到[autoexec]部分,补充命令如下: 然后重启DOSBox就行了. 截屏?不给. 自己做吧! 总结 以上就是这篇文章的全部内容了,

  • Linux shell环境下Zabbix Api的使用

    在linux shell环境下直接调用就可以,根据官网所述:在访问Zabbix中的任何数据之前,你需要登录并获取身份验证令牌.这可以使用该 user.login 方法完成. [root@localhost ~]# curl -i -X POST -H 'Content-Type: application/json' -d '{"jsonrpc": "2.0","method":"user.login","params

  • Linux shell数组与关联数组的用法实例

    1. 关联数组 使用 declare -A(declare 的用法请使用 help 进行查看,help declare) 进行声明关联数组变量: $ declare -A fruits_price $ fruits_price=([apple]='$100' [orange]='$150') 列出关联数组的索引(也就是 key): $ echo ${!fruits_price[*]} $ echo ${!fruits_price[@]} 2. 序列数组 seq 方法创建 基本用法: $ a_nu

  • Linux下误删messages文件的找回方法

    如果有进程正在使用的文件,如果被误删了,可以找回.如果没有进程在使用,就无法找回被误删的文件了. 假如/var/log/messages文件被误删了: 1.查询正在使用该文件的进程. [root@www]# lsof |grep message rsyslogd 1717 root 1w REG 8,2 243321 654968 /var/log/messages 2.根据查询结果,是PID为1717的进程正在使用该文件.进入该进程/proc下的目录/proc/1717/fd: [root@w

  • Linux加载vmlinux调试

    使用gdb加载内核符号表 arm-eabi-gdb out/target/product/msm8625/obj/KERNEL_OBJ/vmlinux 在内核的.config里面要打开 DEBUG_INFO和DEBUG_VM 定位故障代码 (gdb) l * qrd7627a_add_io_devices+0x100 0xc07cd05c is in qrd7627a_add_io_devices (/home/yejialong/GH700C/kernel/arch/arm/mach-msm/

  • Linux更改账户密码实例详解

    更改个人账户密码 普通用户想要更改自己的个人帐户密码,只需要运行passwd命令,不用带任何其他的命令: $ passwd 示例输出: Changing password for nick (current) UNIX password: Enter new UNIX password: Retype new UNIX password: passwd: password updated successfully 系统将提示我们需要先输入当前密码,如果密码正确,则会要求重新输入并确认新密码.在下

  • Linux内核宏container_of的深度剖析

    1.前面说的 我在好几年前读linux 驱动代码的时候看到这个宏,百度了好久,知道怎么用了,但是对实现过程和原理还是一知半解. container_of宏 在linux内核代码里面使用次数非常非常多,对于喜欢linux编程的同学来说,了解其实现方法,对以后看内核代码,写内核驱动的帮助都非常大,当然,我不是说了解这个就可以为所欲为了,内核博大精深,先宏观再微观去学习,不积跬步何以致千里,不要想着一口就能吃成一个胖子,我这篇文章主要剖析一下这个函数的实现原理,希望对大家学习过程中有所帮助. andr

  • Android之在linux终端执行shell脚本直接打印当前运行app的日志的实现方法

    1.问题 我们一般很多时候会需要在ubuntu终端上打印当前运行app的日志,我们一般常见的做法是 1).获取包名 打开当前运行的app,然后输入如下命令,然后在第一行TASK后面的就可以看到包名 adb shell dumpsys activity top 2).我们的终端安装了pidcat.py脚本,然后执行如下的命令就可以打印当前运行app的全日志, pidcat.py packageName 3).思考,为什么每次都需要这样重复的操作呢?一说到重复,我们应该立马想到是否可以用脚本解决重复

随机推荐