1 引入buildroot#
Buildroot是Linux平台上一个开源的嵌入式Linux系统自动构建框架。用来制作根文件系统,我们还要自己去移植一些第三方软件和库,比如 alsa
、iperf
、mplayer
等等。
那么有没有一种傻瓜式的方法或软件,它不仅包含了 busybox
的功能,而且里面还集成了各种软件,需要什么软件就选择什么软件,不需要我们去移植。答案肯定是有的,buildroot
就是这样一种工具。
1.1 下载buildroot#
Buildroot版本每2个月,2月,5月,8月和11月发布一次。版本号的格式为YYYY.MM,例如2013.02、2014.08。
可以从http://buildroot.org/downloads/获得发行包。
也可通过github
仓库获取最新版本:
1 | git clone git://git.busybox.net/buildroot |
buildroot
和 uboot
、Linux kernel
一样也支持图形化配置:make menuconfig
1.2 buildroot目录结构#
1.2.0 buildroot源目录#
arch
: CPU架构相关的配置脚本board
: 在构建系统时,board默认的boot和Linux kernel配置文件,以及一些板级相关脚本boot
: uboot配置脚本目录configs
: 板级配置文件,该目录下的配置文件记录着该机器平台或者方案使用的工具链,boot, kernel,各种应用软件包的配置dl
: download的简写,下载一些开源包。第一次下载后,下次就不会再去从官网下载了,而是从dl/目录下拿开源包,以节约时间docs
:fs
: 各种文件系统的自动构建脚本linux
: 存放Linux kernel的自动构建脚本package
: 第三方开源包的自动编译构建脚本,用来配置编译dl目录下载的开源包support
:system
: 存放文件系统目录的和设备节点的模板,这些模板会被拷贝到output/
目录下,用于制作根文件系统rootfs
toolchain/
目录中存放着各种制作工具链的脚本
1.2.1 编译出的output输出目录介绍#
images/
存储所有映像(内核映像,引导加载程序和根文件系统映像)的位置。这些是您需要放在目标系统上的文件。build/
构建所有组件的位置(包括主机上Buildroot所需的工具和针对目标编译的软件包)。该目录为每个组件包含一个子目录。host/
包含为主机构建的工具和目标工具链。staging/
是到内部目标工具链host/的符号链接target/
它几乎包含了目标的完整根文件系统。除了设备文件/dev/
(Buildroot无法创建它们,因为Buildroot不能以root身份运行并且不想以root身份运行)之外,所需的一切都存在。
1.3 配置 Target options#
1 | Target options |
配置输出目标选项,架构,格式,浮点策略,指令集啊。配置好后如下:
1.4 配置工具链#
Buildroot为交叉编译工具链提供了两种解决方案:
1.4.1 内部工具链#
- 内部工具链,称为
Buildroot toolchain
。buildroot
其实是可以自动下载交叉编译器的,但是都是从国外服务器下载的, 鉴于国内的网络环境,推荐大家设置成自己所使用的交叉编译器(也就是外部工具链)。
1.4.2 外部工具链#
- 外部工具链
External toolchain
。
1 | Toolchain |
Toolchain
:设置为 Custom toolchain,表示使用用户自己的交叉编译器。Toolchain origin
:设置为 Pre-installed toolchain,表示使用预装的交叉编译器。Toolchain path
:设置自己安装的交叉编译器绝对路径!buildroot 要用到。Toolchain prefix
:设置交叉编译器前缀,要根据自己实际所使用的交叉编译器来设置,比如我们使用的是 arm-linux-gnueabihf-gcc
,因此前缀就是$(ARCH)-linux-gnueabihf
,其中 ARCH 我们前面已经设置为了 arm。
1.5 配置 build options#
编译选项,编译第三方插件使用静态还是动态链接等。
1.6 配置 System configuration#
系统配置,比如开发板名字、欢迎语、用户名、密码等。
1 | System configuration |
1.7 配置 Filesystem images#
根文件系统格式。
1 | -> Filesystem images |
1.8 禁止编译 Linux 内核和 uboot#
一版不建议uboot和kernel也用buildroot。buildroot 不仅仅能构建根文件系统,也可以编译 linux 内核和 uboot。
buildroot如果开启了uboot和Linux内核的编译,会自动下载最新的 linux 内核和 uboot,那么最新的内核和uboot会对编译器版本号有要求,可能导致编译失败。
1 | -> Kernel |
1.9 配置 Target packages#
配置要选择的第三方库或软件、比如 alsa-utils、ffmpeg、iperf
等工具。
2 编译buildroot#
2.1 make help#
可以看到buildroot
下make的使用细节,包括对package、uclibc、busybox、linux
以及文档生成等配置。
2.2 make print-version#
打印buildroot
版本号
#
或者(make linux-menuconfig
…):进行图形化配置
2.4 make xxxx_defconfig#
1 | Buildroot_2020.02.x/configs$ ls |
make 100ask_imx6ull_pro_ddr512m_systemV_core_defconfig
即可产生.config
和output目录
:
2.5 make#
sudo make //注意不能-jxxx,来指定多核编译
make命令通常将执行以下步骤:
- 下载源文件(根据需要);
- 配置、构建和安装交叉编译工具链,或仅导入外部工具链;
- 配置、构建和安装选定的目标软件包;
- 构建内核映像(如果选择);
- 构建引导加载程序映像(如果选择);
- 以选定的格式创建一个根文件系统
make clean
:delete all build products (including build directories, host, staging and target trees, the images and the toolchain)make distclean
: 等于make clean+删除配置make show-targets
:显示出本次配置所要编译所有的目标make pkg-target
:单独编译某个pkg模块make pkg-rebuild
:重新编译pkgmake pkg-extrac
t:只下载解压pkg,不编译,pkg解压后放在output/build/
对应的pkg-dir
目录下make pkg-source
:只下载某pkg,然后不做任何事情make list-defconfigs
:例举所有可用的defconfigs。make xxx_menuconfig
:比如make linux-menuconfig
rootfs.tar
就是编译出的根文件系统,解压缩后就能使用。
2.5.1 nfs 挂载根文件系统#
1 | setenv bootargs 'console=tty1 console=ttymxc0,115200 root=/dev/nfs nfsroot=192.168.1.253: |
可以看到能进入rootfs,但是驱动ko和第三方软件和库没有。
2.6 make show-targets#
make show-targets
显示出本次配置所要编译所有的目标。
3 buildroot框架原理#
Buildroot
提供了函数框架和变量命令框架,采用它的框架编写的app_pkg.mk
这种Makefile
格式的自动构建脚本
,将被package/pkg-generic.mk
这个核心脚本展开填充到buildroot
主目录下的Makefile
中去。
最后make all
执行Buildroot
主目录下的Makefile
,生成你想要的image。 package/pkg-generic.mk
中通过调用同目录下的pkg-download.mk
、pkg-utils.mk
文件,已经帮你自动实现了下载、解压、依赖包下载编译等一系列机械化的流程。
你只要需要按照格式写app_pkg.mk
,填充下载地址,链接依赖库的名字等一些特有的构建细节即可。 总而言之,Buildroot
本身提供构建流程的框架,开发者按照格式写脚本,提供必要的构建细节,配置整个系统,最后自动构建出你的系统。
3.1 添加自己的软件包#
3.1.1 package/Config.in总入口添加菜单#
添加如下语句:
1 | menu "myown(fuzidage) package" |
为自己的软件包添加入口,这样在make menuconfig
的时候就可以找到自己的软件包的Config.in
,如果在make menuconfig
的时候选中helloworld
,那么"BR2_PACKAGE_HELLOWORLD=y"
也会同步到.config
中去。
3.1.2 配置APP对应的Config.in和mk文件#
在package
中新增目录helloworld
,并在里面添加Config.in
和helloworld.mk
3.1.2.1 Config.in#
1 | config BR2_PACKAGE_HELLOWORLD |
helloworld/Config.in
文件,可以通过make menuconfig
可以对helloworld
进行选择。只有在BR2_PACKAGE_HELLOWORLD=y
条件下,才会调用helloworld.mk
进行编译
3.1.2.2 helloworld.mk#
1 | ################################################################################ |
helloworld.mk
包括源码位置、安装目录、权限设置等。
3.1.3 编写APP源码和Makefile#
创建一个work/helloworld
目录,建立hello_world.c
和makefile
。
1 | #include <stdio.h> |
#
通过上面对package/Config.in
入口的配置, 我们可以通过make menuconfig
,进入Target packages
可以看见多了一个"myown(fuzidage) package"
入口,选中,保存配置到.config
。
然后make savedefconfig
,对helloworld的配置就会保存到对应的xxx_defconfig
中。
3.1.5 编译使用APP#
可以和整个平台一起编译APP;或者make helloworld
单独编译。
编译过程中,会被拷贝到output/build/helloworld-1.0.0
文件夹中。然后生成的bin文件拷贝到output/target/bin/helloworld
,这个文件会打包到文件系统中。
如果需要清空相应的源文件,通过make helloworld-dirclean
。
3.2 如何重新编译软件包#
经过第一次完整编译后,如果我们需要对源码包重新配置,我们不能直接在buildroot
上的根目录下直接make,buildroot
是不知道你已经对源码进行重新配置,它只会将第一次编译出来的文件,再次打包成根文件系统镜像文件。
那么可以通过以下2种方式重新编译:
直接删除源码包,然后
make all
1
2例如我们要重新编译helloworld,那么可以直接删除output/build/helloworld目录,
那么当你make的时候,就会自动从dl文件夹下,解压缩源码包,并重新安装。这种效率偏低进行
xxx-rebuild
,然后make all
1
2也是以helloworld为例子,我们直接输入make helloworld-rebuild,
即可对build/helloworld/目录进行重新编译,然后还要进行make all(或者make world 或者 make target-post-image)如果要重新配置编译安装:
1
make <package>-reconfigure; make all
3.3 使能第三方软件和库#
前面 1.9配置Targetpackages 有引入介绍。
3.3.1 使能音频的ALSA库套件#
3.3.2 使能busybox套件#
使能后,buildroot
会自动下载 busybox
压缩包,buildroot
下载的源码压缩包都存 放在/dl
目录下,在 dl
目录下就有一个叫做“busybox”
的文件夹,此目录下保存着 busybox 压 缩包:
make all
编译完后, buildroot
将所有解压缩后的软件保存在/output/build
软件中,我们可以找到/output/build/busybox-1.29.3
这个文件夹,此文件夹就是解压后的 busybox 源码:
3.3.2.1 修改配置busybox套件#
修改busybox源码就直接在/output/build/busybox-1.29.3
修改。
make busybox-menuconfig
可以配置busybox套件选择哪些功能:
3.3.2.2 rebuild busybox套件#
make busybox
或者make busybox-rebuild
即可重新编译。
编译完后还要make
或者make target-post-image
对其进行打包进根文件系统。
3.3.3 PS1环境变量#
我们构建的根文件系统启动以后会发现, 输入命令的时候命令行前面一直都是“#”
,如果我们进入到某个目录的话前面并不会显示当前目录路径:
PS1 用于设置命令提示符格式,格式如下:
1 | PS1 = ‘命令列表’ |
我们打开/etc/profie,修改成如下:
1 | PS1='[\u@\h]:\w$:' |
3.4 单独生成目标(build out of tree)#
make O=/home/XXX/output
4 buildroot官方教程链接#
可以下载正点原子翻译的中文版buildroot手册。
5 附录#
5.1 buildroot编译log#
1 | make ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- imx_v7_defconfig V=1 > log.1 2>&1 |
见附件:https://files.cnblogs.com/files/fuzidage/buildroot.build.rar?t=1724326727&download=true