本文约2500字,最近在做精简rootfs分区的工作,我们拿到sdk的时候,rootfs和busybox已经是精简过了的,由于快启功能的需要又得把我们的app和相关库放到rootfs分区中,于是还得在精简的基础上再精简,能strip的都要strip,不需要的能删除就删除。那么下载的全功能busybox应该如何来制作这个精简版本的rootfs镜像呢?我也没做过,本文梳理了这个裁剪流程, 一起来学习吧。读技术类文章的好处是就算你看完没有去实操,但是你知道原理是这么回事,等到工作需要时,可以很快找到思路,再辅以AI,就会很容易上手了。
我建了一个BSP学习交流群,想学BSP或者已经是BSP开发者可私信我,加入群,一起交流学习,共同进步。
关注公众号, 即可获得与Linux相关的电子书籍以及常用开发工具,文末有文档清单。
本文基于消费类嵌入式产品如爱芯AX615/君正T41等系列 ARM/MIPS 平台,Rootfs基础镜像需≤4MB 场景(不含自研app程序))下来说明如何做文件系统的裁剪。
【前言】:
在16MB NOR Flash 的 ARM 嵌入式设备中,分区通常如下:
boot: 256KB~512KB
内核 (zImage):2MB~3MB
配置分区:512KB~1MB
再刨除APP、env、spl等分区后Rootfs大约只剩4MB左右(如果为了快启,需要将app部分程序移动到rootfs分区,那就减少app分区大小,增加rootfs,也就是不含app的rootfs最多只可占用4MB左右)
默认 Rootfs+BusyBox 动辄 6MB~10MB,直接烧写必然溢出。
本文从编译、配置、目录结构到文件系统选型的完整流程裁剪方案描述。
目标:可用 Rootfs 压缩至3MB以内,BusyBox 静态编译后≤800KB,适配消费类摄像机、网关等无 GUI、无多余服务的场景。
一 BusyBox:剔骨式裁剪(核心:静态编译 + 最小 Applet + 编译优化)
[1]. 基础配置(make menuconfig 必选项)
>>编译选项(Build Options)
Busybox Settings --->Build Options --->[*] Build BusyBox as a static binary (no shared libs) # 静态编译,不依赖外部库[ ] Force NOMMU build # ARM有MMU,关闭[ ] Build with large file support # 无大文件需求,关闭(arm-linux-gnueabihf-) Cross compiler prefix # 交叉工具链前缀[*] Additional CFLAGS: -Os -ffunction-sections -fdata-sections # 体积优化[*] Additional LDFLAGS: -Wl,--gc-sections # 剔除未使用代码段
>>安装选项(Installation Options)
Installation Options --->[*] Don't use /usr # 不生成/usr目录,减少体积(../rootfs) BusyBox installation prefix # 安装到rootfs目录
>>核心必选 Applet
Shell 必选(无 Shell 系统无法启动)
Shells --->[*] ash # 轻量Shell,替代bash(bash体积大)[ ] hush # 可选,ash足够[ ] bash # 禁用,体积大
>>基础命令必选(缺一个都可能启动失败)
Coreutils --->[*] ls [*] cp [*] rm [*] mv [*] mkdir [*] rmdir [*] cat [*] echo [*] sync[ ] vi [ ] vim [ ] find [ ] grep [ ] awk # 禁用,用替代方案或后期添加Init Utilities --->[*] init [*] reboot [*] halt # 系统启停必需Linux System Utilities --->[*] mdev # 自动创建设备节点(替代udev,体积极小)[*] mount [*] umount [*] df [*] free # 挂载与系统信息[ ] blkid [ ] losetup # 禁用Networking Utilities --->(工业网络刚需,按需保留)[*] ifconfig [*] route [*] ping [*] dhcpcd # 网络配置+DHCP[ ] wget [ ] tftp [ ] ftpd [ ] httpd # 禁用,用轻量客户端替代
禁用所有冗余功能
关闭所有Editors(vi/vim)、Debug Utilities、Print Utilities
关闭IPv6(无需求时)、SSL/TLS、加密工具
关闭国际化支持:CONFIG_LOCALE_SUPPORT=n、CONFIG_WIDE_CHAR=n
[2]. 编译与瘦身(终极压缩)
# 1. 配置后编译make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- -j4# 2. 安装到rootfsmake install# 3. 终极瘦身:strip所有符号(关键!减少30%~50%体积)arm-linux-gnueabihf-strip --strip-all rootfs/bin/busybox# 4. 验证体积:静态编译后应≤800KBls -lh rootfs/bin/busybox # 理想值:600KB~750KB
[3]. 避坑要点
不要动态编译:动态编译需携带libc.so、libm.so等库,反而增加体积(约 1MB)
不要保留/usr:开启后会生成/usr/bin、/usr/share等目录,浪费空间
必选mdev:无mdev需手动创建设备节点,易遗漏且占用空间
说明:sdk提供的都是编译的动态的busybox,也许我们实操时可以尝试下静态编译。
二 Rootfs 目录结构:极简骨架(只留启动必需目录,删除所有冗余)
[1]. 最小目录清单(仅 10 个目录,总空间≤200KB)
rootfs/├── bin/ # BusyBox二进制文件(必需)├── sbin/ # 系统命令(必需)├── dev/ # 设备节点(必需,mdev自动生成)├── etc/ # 配置文件(必需,inittab、fstab等)├── proc/ # 进程虚拟文件系统(必需,内核挂载)├── sys/ # 系统虚拟文件系统(必需,内核挂载)├── tmp/ # 临时文件(必需,内存挂载)├── var/ # 运行时数据(必需,内存挂载)├── root/ # root用户目录(必需,空)└── linuxrc # 启动脚本(必需,指向/bin/ash)
[2]. 目录裁剪必杀技
>>删除所有非必需目录
# 安装后直接删除冗余目录rm -rf rootfs/usr rootfs/lib rootfs/mnt rootfs/media rootfs/opt rootfs/home
>>/etc目录极简配置(仅保留 4 个文件)
inittab(启动流程,必须)::sysinit:/etc/init.d/rcS::respawn:/bin/ash::shutdown:/bin/umount -a
rcS(初始化脚本,必须)
#!/bin/shmount -amdev -secho "Rootfs mounted successfully!"fstab(挂载虚拟文件系统,必须)proc /proc proc defaults 0 0sysfs /sys sysfs defaults 0 0tmpfs /tmp tmpfs defaults,size=1M 0 0tmpfs /var varfs defaults,size=512K 0 0passwd(极简用户配置,必须)root:x:0:0:root:/root:/bin/ash
>>/dev目录:空目录即可(mdev 自动生成)
mkdir -p rootfs/dev# 无需手动创建null、console等节点,mdev -s启动时自动生成
[3]. 体积控制结果
目录骨架 + 配置文件:≤150KB
BusyBox 二进制:≤800KB
合计:≤950KB(未压缩前)
三 文件系统选型:SquashFS(压缩率 50%~70%)
[1]. 为什么选 SquashFS?
高压缩率:zstd/gzip 压缩,1.5MB 未压缩→600KB~800KB 压缩后
只读:防止误写损坏,掉电安全(配合 WAL 配置区,参数独立)
支持 XIP:NOR Flash 直接执行,无需拷贝到 RAM,节省内存
启动快:挂载速度极快,无文件系统检查
[2]. 制作 SquashFS 镜像
# 安装工具sudo apt install squashfs-tools# 制作镜像(zstd压缩,最高压缩率)mksquashfs rootfs rootfs.squashfs -comp zstd -b 64K -no-xattrs# 验证体积:最终镜像≤3MB(含所有内容)ls -lh rootfs.squashfs
[3]. 内核配置支持 SquashFS(必须开启)
File systems --->Miscellaneous file systems --->[*] SquashFS 4.0+ support[*] SquashFS zstd compression support # 开启zstd压缩[ ] SquashFS XZ compression support # 禁用,体积大
四 进阶优化:极限压缩(再省 300KB~500KB)
[1]. 内核与 Rootfs 合并(initramfs,省去独立 Rootfs 分区)
若 Flash 极度紧张(Rootfs 分区≤2MB),可将 Rootfs 打包为initramfs,内嵌到内核 zImage 中:
# 1. 制作initramfs镜像cd rootfsfind . -print0 | cpio --null -ov --format=newc | gzip -9 > ../initramfs.cpio.gz# 2. 内核配置内嵌initramfsGeneral setup --->[*] Initial RAM filesystem and RAM disk (initramfs/initrd) support(initramfs.cpio.gz) Initramfs source file(s)# 3. 编译内核:zImage+initramfs合并,总大小≤3.5MB优势:省去独立 Rootfs 分区,直接用内核空间,16MB Flash 分区更灵活。
[2]. 工具替代:用二进制文件替代脚本
用C 语言编译的轻量工具替代 Shell 脚本(如自定义网络配置工具,≤50KB)
禁用syslogd:工业设备无需日志,直接关闭,节省 50KB
[3]. 编译器优化:-Os+LTO(再省 10% 体积)
# BusyBox编译时添加LTOCFLAGS+="-flto"LDFLAGS+="-flto"
五 避坑总结
[1].不要用 ext4/ubi:ext4 日志区 + 元数据占用大,掉电需 fsck,不适合小分区; ubi有自己的一套修复机制也比较浪费空间。
[2].不要动态编译 BusyBox:依赖库多,体积反而更大
[3].必选 mdev:无 mdev 无法自动创建设备节点,系统启动失败
[4].只读 SquashFS + 双备份或者独立 WAL 配置区:系统只读防损坏,参数可动态修改,兼顾稳定与灵活
结语
在 16MB NOR Flash 的 ARM Linux 设备中,裁剪不是 “可选优化”,而是"生存必需"。通过"BusyBox 静态最小编译、Rootfs 极简骨架和squashFS 高压缩"的组合,可将 Rootfs 控制在3MB以内,适配消费类摄像机、网关等嵌入式场景。
以上为全文内容。

这里是女程序员的笔记本
15年+嵌入式软件工程师兼二胎宝妈
分享读书心得、工作经验,自我成长和生活方式。
希望我的文字能对你有所帮助