【Linux】编译用于Exynos4412(ARM)的Linux-3.14内核

【Linux】编译用于Exynos4412(ARM)的Linux-3.14内核

零、准备

在准备之前需要配置好交叉编译环境,本文不做介绍。

1、下载

Linux-3.14内核源代码

下载后得到以下文件:

yu@Yubuntu:~/kernel$ ls -l
总计 76568
-rw-rw-r-- 1 yu yu 78399152  4月  9 01:21 linux-3.14.tar.xz

alt text

2、解压

yu@Yubuntu:~/kernel$ tar -vxf linux-3.14.tar.xz

解压后得到如下文件:

yu@Yubuntu:~/kernel$ ls -l
总计 76572
drwxrwxr-x 23 yu yu     4096  3月 31  2014 linux-3.14
-rw-rw-r--  1 yu yu 78399152  4月  9 01:21 linux-3.14.tar.xz

壹、编译内核

1、设置CPU架构和交叉编译器

设置CPU架构和交叉编译器的方法有几种,临时设置、通过环境变量设置和修改Makefile设置。基于实际情况,在本次编译中,我希望我把我修改好的内核发给别人同样有效,而且我只针对一种CPU指令集来配置的,所以我选择修改Makefile

使用vi编辑Makefile

yu@Yubuntu:~/kernel$ cd linux-3.14/
  
yu@Yubuntu:~/kernel/linux-3.14$ vi Makefile

把198、199行的

ARCH        ?= $(SUBARCH)
CROSS_COMPILE   ?= $(CONFIG_CROSS_COMPILE:"%"=%)

改为

ARCH        ?= arm
CROSS_COMPILE   ?= $(CONFIG_CROSS_COMPILE:"%"=%)

alt text
保存并退出~

注:这里关于交叉编译器的安装和配置就不再介绍了。

2、设置处理器

使用如下命令设置处理器:

yu@Yubuntu:~/kernel/linux-3.14$ make exynos_defconfig

难搞,报警告了,具体情况如下:
alt text

In file included from scripts/kconfig/zconf.tab.c:2537:
scripts/kconfig/menu.c: In function ‘get_symbol_str’:
scripts/kconfig/menu.c:587:46: warning: ‘jump’ may be used uninitialized in this function [-Wmaybe-uninitialized]
  587 |                                 jump->offset = strlen(r->s);
      |                                 ~~~~~~~~~~~~~^~~~~~~~~~~~~~
scripts/kconfig/menu.c:548:26: note: ‘jump’ was declared here
  548 |         struct jump_key *jump;
      |                          ^~~~

没关系,Linux论坛上已有解决方案了:
https://patchwork.kernel.org/project/linux-kbuild/patch/1415098919-21836-1-git-send-email-syntheticpp@gmx.net/
alt text

我们按照这个页面上的来修改我们的代码:

yu@Yubuntu:~/kernel/linux-3.14$ vi scripts/kconfig/menu.c

把548行的struct jump_key *jump;改为struct jump_key *jump = NULL;
alt text
把586行的if (head && location && menu == location)改为if (jump && menu == location)(由于我上一个修改我是注释掉原有行,再在下面添加的新行,所以这边我的第二个位置显示的是587行了):
alt text
保存退出~

重新使用如下命令设置处理器:

yu@Yubuntu:~/kernel/linux-3.14$ make exynos_defconfig

alt text
搞定!

3、配置Linux系统内核

1. 配置工具
Linux系统内核的配置文件在上一个步骤中,被我们写到了.config文件中了,感兴趣的同学可以打开看看,很复杂,新手不建议修改。对于大多数情况,我们可以使用Linux内核源码这边给我们提供的配置工具来配置,配置工具在README中有介绍,大致有这么些:
alt text
其中,我们比较常用的是make menuconfig,对于远程使用SSH连接的同学友好些。
2. 前置准备
make menuconfig需要一些依赖,可以使用如下命令安装对应的依赖:

sudo apt-get install libncurses5-dev

libncurses5-dev是一个基于文本的GUI开发库,用于支持在C等编程语言中开发基于文本终端的交互式应用程序,提供了屏幕绘制、键盘输入处理等功能。

另外,make menuconfig对屏幕大小有一定要求,我们最好把当前命令窗口拉伸到最大
3. 配置命令
执行如下命令开始配置:

yu@Yubuntu:~/kernel/linux-3.14$ make menuconfig

alt text
界面上方的文本即是使用帮助,同学们可以自己阅读一下,本文不介绍如何配置Linux内核,同学们根据自己的需要在此界面修改好Linux内核配置后再往下。
4. 配置参考
这里给出我的参考配置:
1). 网卡驱动
alt text

    Device Drivers  --->             
[*]   Network device support  --->   
[*]     Ethernet driver support  --->
<*>     DM9000 support               

2). 网络支持
alt text

[*] Networking support  --->                
      Networking options  --->              
<*>     Packet socket                       
<*>     Unix domain sockets                 
[*]     TCP/IP networking                   
[*]       IP: kernel level autoconfiguration

3). 文件系统
alt text

    File systems  --->                                             
[*]   Network File Systems  --->                                   
<*>     NFS client support                                         
<*>       NFS client support for NFS version 3                     
[*]         NFS client support for the NFSv3 ACL protocol extension
[*]     Root file system on NFS                                    

4). 串口配置
alt text

    System Type  --->                       
(2)   S3C UART to use for low-level messages

5). 交叉编译
alt text

    General setup  --->                             
(arm-none-linux-gnueabi-)   Cross-compiler tool prefix

配置完成后,使用方向键选择“Save”以保存配置。

alt text
配置并保存成功~

4、编译Linux系统内核

因为是针对的exynos4412编译的Linux内核,而exynos4412使用的引导程序是U-Boot,所以把编译好的Linux内核文件封装成uImage格式是比较好的选择。
uImage是一种经过封装的内核镜像格式,它在嵌入式系统中较为常用,特别是使用U-Boot作为引导加载器的系统。这种格式在普通的内核镜像基础上添加了一个头部信息,该头部信息包含了镜像的加载地址、入口地址、镜像大小等内容,便于U-Boot识别和加载内核。
命令make uImage的意思是让make工具依据Makefile里的规则来构建uImage格式的内核镜像。

那么,我们执行如下代码开始编译Linux系统内核:

yu@Yubuntu:~/kernel/linux-3.14$ make uImage

难搞,刚开始编译就报错了:
alt text

/usr/bin/ld: scripts/dtc/dtc-parser.tab.o:(.bss+0x50): multiple definition of `yylloc'; scripts/dtc/dtc-lexer.lex.o:(.bss+0x0): first defined here
collect2: error: ld returned 1 exit status
make[2]: *** [scripts/Makefile.host:127:scripts/dtc/dtc] 错误 1
make[1]: *** [scripts/Makefile.build:455:scripts/dtc] 错误 2
make: *** [Makefile:527:scripts] 错误 2

经查询,是因为GCC版本太高了导致的,我们可以稍作修改,使用如下命令编辑dtc-lexer.lex.c_shipped文件:

yu@Yubuntu:~/kernel/linux-3.14$ vi scripts/dtc/dtc-lexer.lex.c_shipped

在640行,在YYLTYPE yylloc;前面加上extern,即把YYLTYPE yylloc;改为extern YYLTYPE yylloc;
alt text
保存并退出。

重新编译:

yu@Yubuntu:~/kernel/linux-3.14$ make uImage

好,这次编译了大概3分钟,现在又又出错了:
alt text

"mkimage" command not found - U-Boot images will not be built
make[1]: *** [/home/yu/kernel/linux-3.14/arch/arm/boot/Makefile:80:arch/arm/boot/uImage] 错误 1
make: *** [/home/yu/kernel/linux-3.14/arch/arm/Makefile:305:uImage] 错误 2

本次出错的原因在于找不到mkimage命令,我们需要安装U-Boot工具(u-boot-tools),这个是用于U-Boot引导加载程序的辅助工具集,可帮助我们进行U-Boot的配置、编译、调试以及映像文件处理等工作。

我们使用如下命令安装U-Boot工具:

sudo apt-get install u-boot-tools

alt text

好,继续编译:

yu@Yubuntu:~/kernel/linux-3.14$ make uImage

经过大约4分钟,编译完成~
alt text

编译好的内核二进制文件在arch/arm/boot目录下:
alt text

我们可以使用如下命令复制到TFTP服务器上去,方便等下开发板下载运行:

yu@Yubuntu:~/kernel/linux-3.14/arch/arm/boot$ cp uImage ~/share/tftp/linux

下面的步骤是可选的,本文仅做介绍。

5、编译内核模块

在使用make menuconfig配置Linux内核时,有的功能被设置为模块(M),模块不会在上一个步骤被编译进内核,我们需要单独编译,使用如下命令开始编译内核模块:

yu@Yubuntu:~/kernel/linux-3.14$ make modules

编译内核模块成功:
alt text
其中,以.ko结尾的即为内核模块(Kernel Object)文件。
我们将来在Linux中可以使用insmodmodprobe命令来加载.ko内核模块,使用rmmod 命令卸载内核模块。

6、查看编译耗时

使用如下命令可以查看编译耗时:

yu@Yubuntu:~/kernel/linux-3.14$ $(which time) -v make uImage

alt text
我这边耗时约3分钟。

贰、编译设备树

ARM芯片中很多功能,每个芯片的功能都不一样,为了解决硬件多样性与内核可移植性之间的矛盾,我们引入了设备树。设备树相当于一份硬件说明书,告诉内核什么设备在哪里。

1、创建设备树文件

我们使用的芯片是Exynos4412,在Linux中已存在相关的设备树文件了,但是已存在的不一定能直接使用,需要结合实际对其进行修改。
我们使用如下命令复制一份设备树文件:

yu@Yubuntu:~/kernel/linux-3.14$ cp arch/arm/boot/dts/exynos4412-origen.dts arch/arm/boot/dts/exynos4412-ex4412.dts

设备树文件与C语言文件一样,需要编译,故我们编辑对应的Makefile文件:

yu@Yubuntu:~/kernel/linux-3.14$ vi arch/arm/boot/dts/Makefile

alt text
我们同样的把“exynos4412-origen”的复制一份修改一下即可。

2、修改设备树文件

我们需要针对我们自己的硬件对设备树进行修改:

yu@Yubuntu:~/kernel/linux-3.14$ vi arch/arm/boot/dts/exynos4412-ex4412.dts

alt text
在根节点内添加网卡的硬件信息。

3、编译设备树文件

使用如下命令开始编译设备树文件:

yu@Yubuntu:~/kernel/linux-3.14$ make dtbs

alt text
编译好后的文件为arch/arm/boot/dts/exynos4412-ex4412.dtb,这个文件我们后面启动Linux内核时需要用到的,我们可以复制到TFTP服务器上去。

使用如下命令复制设备树文件到TFTP服务器上去:

yu@Yubuntu:~/kernel/linux-3.14$ cp arch/arm/boot/dts/exynos4412-ex4412.dtb ~/share/tftp/dt.dtb

叁、优化和运行

1、优化

我们需要对内核代码进行一下小小的修改,以便更适配我们的硬件:

1). 忽略无用的时钟,使用如下命令编辑相关代码:

yu@Yubuntu:~/kernel/linux-3.14$ vi drivers/clk/clk.c

alt text
修改完成后保存退出~

2). 优化eMMC,使用如下命令编辑相关代码:

yu@Yubuntu:~/kernel/linux-3.14$ vi drivers/mmc/core/mmc.c

alt text
修改完成后保存退出~

2、重新编译

1). 编译Linux内核:

yu@Yubuntu:~/kernel/linux-3.14$ make uImage

alt text
编译完成后使用如下命令把uImage复制到TFTP服务器上,方便开发板使用:

yu@Yubuntu:~/kernel/linux-3.14$ cp arch/arm/boot/uImage ~/share/tftp/linux

2). 编译设备树:

yu@Yubuntu:~/kernel/linux-3.14$ make dtbs

alt text
编译完成后使用如下命令把设备树文件复制到TFTP服务器上,方便开发板使用:

yu@Yubuntu:~/kernel/linux-3.14$ cp arch/arm/boot/dts/exynos4412-ex4412.dtb ~/share/tftp/dt.dtb

3、运行

配置好网络和U-Boot相关设置,启动开发板:
alt text
成功在开发板上运行我们自己编译的Linux内核~

肆、参考资料

  1. https://patchwork.kernel.org/project/linux-kbuild/patch/1415098919-21836-1-git-send-email-syntheticpp@gmx.net/
  2. https://www.kernel.org/pub/linux/kernel/v3.x/
  3. https://www.kernel.org/
  4. https://blog.csdn.net/zhoukaiqili/article/details/126191871
  5. https://blog.csdn.net/eibo51/article/details/51901480
posted @ 2025-04-09 14:29  清风来叙  阅读(206)  评论(2)    收藏  举报