制作根文件系统并使用NFS挂载运行。
上位机准备:
- 准备busybox,安装menuconfig所需依赖的库:
- sudo apt-get install build-essential
- sudo apt-get install libncurses5
- sudo apt-get install libncurses5-dev
- 在busybox中执行make menuconfig
Linux Module Utilities ---> //按N键去除选项(insmod/lsmod/rmmod精简版命令) [*] Simplified modutils (NEW) 去除以上选项,立马出现完整版的命令选项: [*] insmod (NEW) │ │ [*] rmmod (NEW) │ │ [*] lsmod (NEW) [*] Pretty output (NEW) [*] Blacklist support │ │ [*] modprobe (NEW) │ │ [*] depmod (NEW) 保存退出 注意:目前busybox提供的命令已经足够使用
- 修改Makefile进行交叉编译:
- vim Makefile +164 //修改为指定的交叉编译器。
- vim Makefile +190 //修改为ARCH=arm。(指定架构)
- 保存退出
- 正式进行编译:
- make
- make install
- ls _install/ //查看编译生成的内容
- cd _install; linuxrc bin sbin usr //验证了busybox仅仅提供各种命令。
- 创建根文件目录:
- cp _install /opt/rootfs -frd
- cd /opt/rootfs //进入制作好的根文件系统根目录下。
- mkdir dev lib etc proc sys //创建必要的系统目录。
- mkdir home tmp var mnt //创建可选的目录
- 添加必要的依赖动态库:
//获取到编译busybox的交叉编译器的路径 which is arm-cortex_a9-linux-gnueabi-gcc //进入交叉编译器的根目录 cd /opt/toolchains //添加libc.so.6动态库 find . -name libc.so.6 得到: ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libc.so.6 ls ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libc.so.6 -lh 得到 ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libc.so.6 -> libc-2.18-2013.10.so 说明libc.so.6也仅仅是一个软连接文件,所以拷贝时务必将实体文件一块拷走 cp ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libc.so.6 /opt/rootfs/lib/ -d cp ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libc-2.18-2013.10.so /opt/rootfs/lib/ -d //添加libm.so.6动态库 cd /opt/toolchains find . -name libm.so.6 得到: ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libm.so.6 ls ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libm.so.6 -lh 得到 ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libm.so.6 -> libm-2.18-2013.10.so 说明libc.so.6也仅仅是一个软连接文件,所以拷贝时务必将实体文件一块拷走 cp ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libm.so.6 /opt/rootfs/lib/ -d cp ./arm-cortex_a9-linux-gnueabi/sysroot/lib/libm-2.18-2013.10.so /opt/rootfs/lib/ -d //切记切记切记:最后还要添加动态库使用时所需的加载器 cd /opt/toolchains find . -name ld-* 得到加载器 ./arm-cortex_a9-linux-gnueabi/sysroot/lib/ld-2.18-2013.10.so ./arm-cortex_a9-linux-gnueabi/sysroot/lib/ld-linux.so.3 cp ./arm-cortex_a9-linux-gnueabi/sysroot/lib/ld-* /opt/rootfs/lib/ -d
添加系统启动的必要配置文件和脚本
- inittab配置文件
cd /opt/rootfs/ vim etc/inittab 添加如下内容 ::sysinit:/etc/init.d/rcS ::respawn:-/bin/sh 保存退出
说明:系统启动流程
上电CPU从EMMC的512字节运行uboot
->uboot运行首先进行硬件初始化
然后根据bootcmd从某个地方加载
内核到内存并且启动内核
在启动内核之前给内核传递参数通过bootargs
->内核uImage启动,首先做7大子系统的初始化
最后根据bootargs到某个地方找根文件系统rootfs
->内核找到rootfs以后,内核启动rootfs中的/sbin/init
第一号进程,第一号进程init首先打开rootfs目录下的
etc/inittab文件,第一号进程init会解析inittab文件
首先找到sysinit关键字,一旦找到此关键字
第一号进程会创建一个子进程执行sysinit关键字指定的
脚本程序etc/init.d/rcS,父进程第一号进程会等待子进程
执行完rcS脚本程序
->子进程执行完rcS脚本程序以后,父进程init继续执行
继续解析inittab文件,找到respawn关键字,一旦找到
这个关键字,父进程init继续创建一个子进程,子进程就会
执行respawn对应的程序/bin/sh,父进程继续等待子进程
至此启动了一个shell程序,用户可以输入各种命令
添加系统启动脚本文件rcS
存于根文件系统rootfs的etc/init.d/目录下 cd /opt/rootfs mkdir etc/init.d/ vim etc/init.d/rcS 添加如下内容 mount -a mkdir /dev/pts mount -t devpts devpts /dev/pts echo /sbin/mdev > /proc/sys/kernel/hotplug mdev -s 保存退出即可
mount -a:系统会自动解析fstab配置文件,系统根据
此配置文件进行一系列的挂接动作
mount -t devpts devpts /dev/pts:将虚拟文件系统devpts
挂接到/dev/pts目录下
将来/dev/pts目录就可以作为
devpts虚拟文件系统的入口
此命令用于将来可以远程登录开发板,
例如:telnet
echo /sbin/mdev > /proc/sys/kernel/hotplug:
向文件/proc/sys/kernel/hotplug写入字符串"/sbin/mdev"
起始就是告诉内核驱动将来创建设备文件的程序(人)是/sbin/mdev
mdev -s:系统启动,将内核驱动对应的设备文件进行自动创建
添加系统启动的配置文件fstab
存于根文件系统rootfs的etc目录下 cd /opt/rootfs vim etc/fstab 添加如下内容 proc /proc proc defaults 0 0 sysfs /sys sysfs defaults 0 0 tmpfs /dev tmpfs defaults 0 0 保存退出
说明:
第一列:表示要挂接的设备
第二列:表示挂接点,将来的入口
第三列:表示指定的文件系统格式
第四,五,六:分别指定访问权限
结论:
将来/proc目录,/sys目录,/dev/目录分别作为
procfs,sysfs,tmpfs三种虚拟文件系统的入口
并且以上三种虚拟文件系统将来创建的内容都是
内核来创建,并且分别创建到/proc,/sys,/dev/三个
目录中,关键的关键,这三个目录下将来内核创建的
目录也好,文件也好,都是存在于内存中,掉电就会丢失!
下位机执行
- 分区
fdisk 2 2 0x100000:0x4000000 0x4100000:0x2f200000 #然后重新烧写rootfs_ext4.img
fdisk:uboot的分区工具命令 第一2:表示对EMMC进行重新分区,如果是SD0写0,如果是SD1写1 第二2:表示要分2两个分区,注意uboot分区不算,这两个分区分别表示uImage和rootfs所在的分区 0x100000:uImage所在分区的起始地址,为1M开始 0x4000000:uImage所在分区的大小,为64M(65M-1M) 0x4100000:rootfs所在分区的起始地址,65M 0x2f200000:rootfs所在分区的大小,754M(819M-65M)
- 烧写内核到mmc
#擦除当前mmc中数据 mmc erase 0x800 0x3000 tftp 48000000 uImage #注意:下载完毕以后,最后会提示uImage下载的文件大小 #烧写Linux内核 mmc write 48000000 0x800 0x3000 #48000000:内存的起始地址,按字节为单位 #0x800:要写的emmc的起始地址,按sector为单位,0x800=0x100000/512 #0x3000:要向emmc写入的数据大小,按sector为单位,0x3000=uImage文件大小/512
- 也可以先将一个文件系统镜像rootfs_ext4.img烧写到mmc中
tftp 480000000 rootfs_ext4.img #注意观察rootfs_ext4.img的文件大小 mmc write 48000000 0x20800 0x32000 # 0x20800=65M/512 #0x32000=rootfs_ext4.img文件大小/512 #至此:uboot/uImage/rootfs_ext4.img烧写完毕 #使用该文件系统,修改参数 setenv bootargs root=/dev/mmcblk0p2 init=/linuxrc console=ttySAC0,115200 rootfstype=ext4 maxcpus=1 lcd=vs070cxn(或者lcd=wy070ml) tp=gslx680-linux saveenv
说明: bootargs同样属于uboot的非常非常重要的一个环境变量 此变量就是给linux内核传递的启动参数,将来内核根据 这个启动参数去到某个地方找根文件系统rootfs root=/dev/mmcblk0p2:告诉linux内核根文件系统在emmc的第三分区 init=/linuxrc:启动运行的第一个脚本linuxrc,linuxrc会启动/sbin/init第一号进程 console=ttySAC0,115200:告诉linux内核,调试串口用第一个串口,注意一定要符合硬件的调试串口编号 rootfstype=ext4:文件系统格式为ext4 maxcpus=1:只启动CPU0 lcd=vs070cxn:指定LCD显示屏的型号, lcd=wy070ml:指定新板子的LCD显示屏型号 tp=gslx680-linux:指定触摸屏的型号
- 添加nfs文件系统读取参数
重启下位机,进入uboot命令行执行: setenv bootargs root=/dev/nfs nfsroot=192.168.0.104:/opt/rootfs ip=192.168.0.118:192.168.0.104:192.168.0.1:255.255.255.0 init=/linuxrc console=ttySAC0,115200 maxcpus=1 lcd=vs070cxn tp=gslx680-linux saveenv tftp 48000000 uImage bootm 48000000 //查看是否能够挂接自己制作的根文件系统 # 注意:shell终端之前的打印信息仔细看,是否有错误!
在自己制作的根文件系统中添加一个应用程序
上位机执行:
cd /opt/rootfs vim helloworld.c //最好是线程程序 arm-cortex_a9-linux-gnueabi-gcc -o helloworld helloworld.c
下位机测试:
cd / ls helloworld ./helloworld //看是否能够正常运行
是否出现类似:libxxx.so…找不到
问:如何解决呢?
答:只需到交叉编译器中找到对应的动态库并且
拷贝到根文件系统rootfs的lib目录下
注意软连接问题噢!
问:cannot run /etc/init.d/rcS: Permission denied
答:rcS脚本文件没有可执行权限,解决办法:
cd /opt/rootfs chmod 777 etc/init.d/rcS
案例.向根文件系统rootfs添加自己移植或者自己制作的动态库
实施步骤:
1.明确:自己移植或者自己制作的动态库一律不允许放到根文件系统rootfs
的必要目录lib下,要单独存放,注意设置环境变量
2.上位机执行:
mkdir /opt/rootfs/home/applib
cd /opt/rootfs/home/applib
vim test.h //声明 添加如下内容
#ifndef __TEST_H
#define __TEST_H
extern void my_test(void);
#endif
保存退出
vim test.c //定义 添加如下内容
#include <stdio.h>
void my_test(void)
{
printf("%s\n", func);
}
保存退出
vim main.c //调用
#include <stdio.h>
#include “test.h”
int main(void)
{
my_test(); //调用
return 0;
}
编译:
arm-cortex_a9-linux-gnueabi-gcc -shared -fpic -o libtest.so test.c
arm-cortex_a9-linux-gnueabi-gcc -o main main.c -L. -ltest
注意:不要将libtest.so拷贝到/opt/rootfs/lib下,相当危险!
下位机测试:
进入下位机的linux系统,执行:
cd /home/applib
ls
libtest.so main
./main //势必提示libtest.so找不到
解决办法:
export LD_LIBRARY_PATH=/home/applib:$LD_LIBRARY_PATH
然后
./main
在根文件系统中添加应用程序自启动功能
上位机实施步骤:
cd /opt/rootfs
vim etc/init.d/rcS 在文件最后添加如下内容:
export LD_LIBRARY_PATH=/home/applib:$LD_LIBRARY_PATH
/home/applib/main &
保存退出
重启下位机,看main程序是否能够自己运行
案例:问:rootfs_ext4.img从何而来?
答:rootfs_ext4.img仅仅是根文件系统rootfs一个镜像文件而已
此镜像中同样包含了根文件系统rootfs的内容
问:如何将自己制作的rootfs制作成一个单个二进制镜像文件呢
也就是将/opt/rootfs(目录)->rootfs_ext4.img(单个文件)
上位机实施步骤:
cd /opt/
sudo dd if=/dev/zero of=rootfs_ext4.img bs=1k count=8196
命令说明:
dd:用于创建一个单个镜像文件
if=/dev/zero:将来创建的单个镜像文件里面的内容全部来自设备/dev/zero
/dev/zero设备能够源源不断产生0数据
of=rootfs_ext4.img:指定将来创建单个镜像文件名
并且此文件里面填充全0
bs=1k:生成的rootfs_ext4.img以块为单位,一块1024字节
count=8196:总共8196块
结论:生成的rootfs_ext4.img数据块为8MB
sudo mkfs.ext4 rootfs_ext4.img //格式化镜像文件rootfs_ext4.img
//把rootfs_ext4.img比作U盘
指定的文件系统格式为ext4
//类似windows下格式化U盘
sudo mkdir /mnt/initrd //创建一个目录
sudo mount -t ext4 -o loop rootfs_ext4.img /mnt/initrd
命令说明:
挂接rootfs_ext4.img到目录/mnt/initrd,并且指定的文件系统类型ext4
挂接命令的结果就是将来只需要访问/mnt/initrd,本质
就是在访问rootfs_ext4.img里面的内容
sudo cp /opt/rootfs/* /mnt/initrd -frd //向目录initrd
拷贝rootfs的内容,本质上就是向rootfs_ext4.img
拷贝rootfs的内容
结果是拷贝完毕,rootfs_ext4.img里面的内容
就是/opt/rootfs里面的内容
sudo umount /mnt/initrd //卸载/mnt/initrd,将来initrd
不再作为rootfs_ext4.img的入口
cp rootfs_ext4.img /tftpboot
至此第一天烧写系统使用的rootfs_ext4.img就是这么来的!
向EMMC烧写战果rootfs_ext4.img
下位机重启,进入uboot命令行模式执行:
- 1.向下位机部署系统软件之前,切记记得要进行分区规划
- EMMC存储空间的划分:
- 0–512----------1M--------7M--------17M---------剩余
- uboot uImage rootfs 大片
- mmcblkboot0 mmcblk0p1 mmcblk0p2
- uboot已经完成 自己分 自己分
- 2.烧写uImage
- tftp 48000000 uImage
- mmc write 48000000 0x800 0x3000 //emmc地址以块单位,一块0x200=512字节
计算流程:
0x800起始地址=0x100000/0x200
0x3000分区大小=7M-1M=0x600000/0x200
- 3.烧写rootfs_ext4.img
- tftp 480000000 rootfs_ext4.img //下载自己制作的rootfs
- mmc write 48000000 3800 5000
计算流程:
0x3800起始地址=0x700000/0x200
0x5000分区大小=10M=0xa00000/0x200
- 4.设置系统启动参数
- setenv bootcmd mmc read 48000000 0x800 0x3000 ; bootm 48000000
- setenv bootargs root=/dev/mmcblk0p2 console=ttySAC0,115200 rootfstype=ext4
- maxcpus=1
- saveenv
8.6.切记:如果对分区进行了修改,记得要对EMMC重新分区
利用uboot提供的fdisk命令 fdisk 2 2 0x100000:0x600000 0x700000:0xa00000 说明: 第一个‘2’:表示emmc 第二个‘2’:表示分两个分区(uImage+rootfs) uboot分区不用做 0x100000:0x600000:第二个分区的起始地址和大小 0x700000:0xa00000:第三个分区的起始地址和大小
8.7.重新启动系统,测试rootfs_ext4.img
系统启动完毕,观察main程序是否启动,看main的打印的信息