Linux 内核编译与移植

目录

1. 树莓派等芯片带操作系统的启动过程

1.1 各种操作系统启动流程

1.2 BootLoader 简介

2. Linux 源码目录树分析

2.1 Linux是一个开源的,支持多架构多平台代码

2.2 Linux内核源代码目录树结构

3. 树莓派 Linux 源码配置

3.1 第一种配置方式:使用源码里自带的config

3.2 第二种配置方式:使用make menuconfig  一项项配置

4. Linux内核编译

4.1 编译指令

4.2 编译成功显示

5. Linux内核镜像拷贝至树莓派并启动新内核

5.1 打包 zImage 成树莓派可用的 xxx.img

5.2 把树莓派的SD读卡器插到电脑,映射到ubuntu上:

 5.3 进行数据拷贝

 5.4 安装 modules, 设备驱动文件: 驱动功能比如:hdmi usb wifi io ...

5.5 更新 kernel.img 文件,注意镜像名字是kernel7.img


1. 树莓派等芯片带操作系统的启动过程

1.1 各种操作系统启动流程

(1)C51,STM32(裸机)》》》》》》C直接操控底层寄存器实现相关业务。 

业务流程型的裸机代码:遥控灯: while(1)、垃圾桶:WemosD1 LOOP、恩智浦智能车: stm32

(2)X86,Intel   windows

启动过程:  电源 -》 BIOS -》windows内核-》C,D盘-》 程序启动(QQ)

(3)嵌入式产品: 树莓派,mini2440, mini6410,nanopi,海思,RK(瑞芯微)------人脸识别打卡器,智能家居主控。。。
        启动过程:  电源 --> BootLoader(引导操作系统启动) --> Linux内核 --> 文件系统(根据功能性来组织文件夹,带访问权限) --> KTV点歌机(应用程序)

(4)文件系统

  •  / 根目录底下的文件系统;
  • lib:放库(动静态库)、
  • dev:放驱动设备、
  • proc:内核的数据信息、
  • home:工作目录、
  • sbin:通常是一些可执行的文件或者指令、
  • bin:通常是一些可执行的文件或者指令

(5)安卓

启动过程:  电源-》 fastBoot/Bootloader/-》Linux内核-》文件系统-》虚拟机-》HOME应用程序-》点某图标打开某APP

1.2 BootLoader 简介

  • 一阶段:初始化时钟,关闭看门狗,关中断,启动 ICACHE,关闭 DCACHE 和 TLB,关闭MMU,初始化 SDRAM,初始化 FLASH、串口,IIC、驱动等,重定位。
  • 二阶段:初始化一个串口,检测系统内存映射,将内核映象和根文件系统映象从 Flash上读到SDRAM空间中,为内核设置启动参数,调用内核引导Linux内核启动

2. Linux 源码目录树分析

2.1 Linux是一个开源的,支持多架构多平台代码

Linux可移植性非常高,但Linux内核编译出来一般就几M

因为Linux支持多平台,多架构,所以编译之前要配置,配置成适合的目标平台来用

那么Linux内核支持哪些架构呢?

ARM架构:(目前比较主流的架构)
海思,友善之臂,RK,树莓派,nanopi

X86架构

PowerPc架构(较早期)
MIPS架构     (较早期)

2.2 Linux内核源代码目录树结构

arch:包含和硬件体系结构相关的代码,每种平台占一个相应的目录。和32位PC相关的代码存放在i386目录下,其中比较重要的包括kernel(内核核心部分)、mm(内存管理)、math-emu(浮点单元仿真)、lib(硬件相关工具函数)、boot(引导程序)、pci(PCI总线)和 power(CPU相关状态)。

block:部分块设备驱动程序。

crypto:常用加密和散列算法(如AES、SHA等),还有一些压缩和CRC校验算法。

Documentation:关于内核各部分的通用解释和注释。

drivers:设备驱动程序,每个不同的驱动占用一个子目录。

fs:各种支持的文件系统,如ext、fat、ntfs等。

include:头文件。其中,和系统相关的头文件被放置在linux子目录下。

init:内核初始化代码(注意不是系统引导代码)。

ipc:进程间通信的代码。

kernel:内核的最核心部分,包括进程调度、定时器等,和平台相关的一部分代码放在arch/*/kernel目录下。

lib:库文件代码。

mm:内存管理代码,和平台相关的一部分代码放在arch/*/mm目录下。

net:网络相关代码,实现了各种常见的网络协议。

scripts:用于配置内核文件的脚本文件。

security:主要是一个SELinux的模块。

sound:常用音频设备的驱动程序等。

usr:实现了一个cpio。

3. 树莓派 Linux 源码配置

驱动代码编写的前提:驱动代码的编译需要一个提前编译好的内核(编译内核就必须配置)

配置的最终目标会生成 .config 文件,该文件指导Makefile去把有用的东西组织成内核

好比买了一个树莓派,那么厂家就会给配这个树莓派相关的 linux 内核源码

参考树莓派之Linux内核源码(基础)_树莓派源码_Love小羽的博客-CSDN博客

树莓派Linux源码配置有三种方式:

  1. cp   厂家.config    xxx.config
  2. make menuconfig  一项项配置,通常是基于厂家的 config 来配置
  3. 完全自己来

如何配置树莓派Linux内核:

(1)查看 Linux 内核 .config 版本,拿到树莓派芯片后厂家一般会告诉你用哪个.config:

find -name *_defconfig

(2)配置config
        linux 源码中有很多工程:
        树莓派1的工程是 bcmrpi_defconfig;
        树莓派2、3的工程是 bcm2709_defconfig。

3.1 第一种配置方式:使用源码里自带的config

ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make bcm2709_defconfig

arm:指定ARM架构

 COMPILE=arm-linux-gnueabihf-:指定编译器

KERNEL=kernel7:树莓派里面特别指定的kernel

make bcm2709_defconfig:核心指令(把厂家的config变成了.config)

如果在上述的命令中报错这个,是因为缺少关键库,下载相关的库

sudo apt-get install flex
sudo apt-get install bison

再次执行次命令:

ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make bcm2709_defconfig

如果执行上述的命令,依然报错:

scripts/Kconfig.include:44: Sorry, this compiler is not supported.
scripts/kconfig/Makefile:94: recipe for target 'bcm2709_defconfig' failed
make[1]: *** [bcm2709_defconfig] Error 1
Makefile:619: recipe for target 'bcm2709_defconfig' failed
make: *** [bcm2709_defconfig] Error 2

则升级gcc编译器的命令,就好了

sudo apt-get install gcc-arm-linux-gnueabihf

3.2 第二种配置方式:使用make menuconfig  一项项配置

(1)配置前先安装需要用到的库

sudo apt-get install bc
sudo apt-get install libncurses5-dev libncursesw5-dev
sudo apt-get install zlib1g:i386
sudo apt-get install libc6-i386 lib32stdc++6 lib32gcc1 lib32ncurses5

(2)输入内核配置指令

ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make menuconfig

执行结果显示

Enter:进入下一个子菜单

前面带Y的:把驱动编译进内核

M:驱动模块的方式加载

exclude:代表略过

module:也是让内核变小的一种方式

驱动两种加载方法:

        * 编译进内核,zImage 包含了驱动

        M 模块的方式,生成驱动文件,在系统启动后,通过命令inmosd  xxx.ko 加载

4. Linux内核编译

4.1 编译指令

ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make -j4 zImage modules dtbs

arm:代表指定arm架构

CROSS_COMPILE=arm-linux-gnueabihf-:编译工具

KERNEL=kernel7:内核

-j4:代表指定用多少电脑资源进行编译,j4指四核

zImage:生成内核镜像

modules:要生成驱动模块

dtbs:生成配置文件

4.2 编译成功显示

提示:编译过程需要一些时间,请耐心等待。。。。

编译成功显示:

(1)看到源码树目录多了一个 vmlinux文件 ,失败则无此文件

(2)生成的 zImage内核镜像 在  arch/arm/root目录底下

 

5. Linux内核镜像拷贝至树莓派并启动新内核

5.1 打包 zImage 成树莓派可用的 xxx.img

./scripts/mkknlimg arch/arm/boot/zImage ./kernel_new.img

5.2 把树莓派的SD读卡器插到电脑,映射到ubuntu上:

用 dmesg 命令查看SD卡是否成功映射到ubuntu上,如下图代表映射成功

 5.3 进行数据拷贝

树莓派的SD卡有两个分区:
一个 fat 分区,是 boot 相关的内容,kernel 的 img 文件就放在这个分区里;
一个是 ext4 分区,也就是系统的根目录分区。

(1)在 ubuntu 的根目录下创建两个新目录 mkdir data1 和mkdir data2

(2)挂载U盘,执行命令如下:

sudo mount /dev/sdb1 data1   //一个fat分区,是boot相关的内容,kernel的img文件
sudo mount /dev/sdb2 data2   //一个是ext4分区,也就是系统的根目录分区

挂载U盘后查看data1和data2目录的结果: 

 5.4 安装 modules, 设备驱动文件: 驱动功能比如:hdmi usb wifi io ...

sudo ARCH=arm CROSS_COMPILE=arm-linux-gnueabihf- KERNEL=kernel7 make INSTALL_MOD_PATH=/home/outside/data2 modules_install

注意:

(1)上述命令是执行在linux-rpi-5.15.y的目录下

(2)把该驱动安装在 SD 卡的 ext4 分区,也就是系统的根目录data2分区,即make INSTALL_MOD_PATH=/home/outside/data2(/home/outside是本人的跟目录,请写自己的根目录路径)

5.5 更新 kernel.img 文件,注意镜像名字是kernel7.img

(1)先备份原本就刷好的SD卡里的kernel7.img,以免刷机失败,那样就不能启动树莓派了:

在SD卡的fat分区中备份,也就是在data1目录下进行备份

cd /home/outside/data1
cp kernel7.img kernel7OLD.img

(2)再把编译新生成的kernel_new.img 拷贝到data1,起名kernel7.img

cp kernel_new.img /home/outside/data1/kernel7.img

(3)检查拷贝后的文件是否有损坏:md5sum

du kernel_new.img  
du kernel_new.img -h  //计算文件的大小
 
md5sum kernel_new.img //每个文件都有一个唯一编码号,以下为了检测是否拷贝有错
md5sum /home/lzy/data1/kernel7.img   //这是检查拷贝后的文件

(4)拷贝其他相关配置文件,不能漏,要拷贝到SD卡的fat分区中,也就拷贝到data1目录下

cp arch/arm/boot/dts/.*dtb* /home/outside/data1
cp arch/arm/boot/dts/overlays/.*dtb* /home/outside/data1/overlays/
cp arch/arm/boot/dts/overlays/README /home/outside/data1/overlays/

.*dtb*:指dtb前后都有文字的,意思是以"."开头的,"."和d中间有什么字符不管,然后以dtb在中间,并且b后面有什么字符也不管,这样的所有文件,全都拷贝下来

".":是指通配符

(5)更新完成后插回树莓派即可开机,开机后可以用 uname -a 命令查看kernel信息已经改变

 

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux内核源码编译是将Linux内核源代码编译成可执行内核的过程。编译过程需要先安装必要的编译工具和依赖库,然后使用make命令进行编译编译完成后,可以将生成的内核镜像文件安装到系统中,以更新现有的内核版本或者创建新的内核版本。编译过程需要一定的技术和经验,建议在进行编译前先了解相关知识和文档。 ### 回答2: Linux内核是一个开放源代码的操作系统,它的内核源码可以通过网络从github等网站下载,因此也允许用户对源码进行修改。当用户对内核源代码进行修改之后,需要编译新的内核并将其安装到系统中,以应用修改所带来的变化。 首先,下载内核源码,可以通过以下指令在Linux终端中下载 ``` $ git clone https://github.com/torvalds/linux.git ``` 下载完成之后,进入源码根目录 ``` $ cd linux ``` 然后,在终端中运行make menuconfig命令,进入配置界面。在这个界面,可以选择需要编译内核中的功能、驱动和选项,也可以取消需要的功能等选项。完成之后,保存并退出配置界面。 接下来就是正式编译的过程。 先清除上次编译的结果 ``` $ make clean ``` 编译时可以指定对应的平台,可以通过以下命令指定: ``` $ ARCH=arm make xyz ``` 其中xyz指的是编译的目标,比如all表示编译全部内容,zImage表示编译内核文件,modules表示编译驱动模块, bzImage表示编译完整的内核文件等等。 编译完成之后,安装内核和驱动模块等文件: ``` $ sudo make modules_install && sudo make install ``` 这个命令会将编译好的内核镜像文件bzImage复制到/boot目录下,并为其设置一个名为config-<版本号>的配置文件。 最后,重新启动计算机,新的内核就会生效了。 编译内核源码需要一定的编程知识和经验,一些依赖包等也需要提前安装。此外,除非你确实需要修改内核,否则,在编译之前最好想清楚是否真的需要对内核源代码进行多大的修改。 ### 回答3: Linux内核是开源软件,它的源代码是公开且可自由使用的。编译Linux内核源代码就是将源代码翻译成计算机可以执行的机器语言的过程。 Linux内核源代码编译主要包括以下几个步骤: 1. 下载内核源码:从官方网站或其他可靠资源中获取内核源码,根据需要选择合适的版本。 2. 配置内核:使用make menuconfig工具打开Kernel Configuration Utility,根据系统硬件和需要进行内核的配置。配置完成后,保存并退出。 3. 编译内核:使用make命令编译内核,这个过程可能需要一些时间,过程中会生成一些文件(包括压缩镜像文件和应用程序的固件二进制文件等)。 4. 安装内核:使用make install命令将编译后的内核安装到系统中,建议在安装前进行备份。 5. 更新启动引导程序:重新载入启动引导程序(如Grub),以确保新内核可以正常启动。 在编译内核时,需要注意以下问题: 1. 建议使用最新版的编译器。 2. 系统要求:要求系统已经安装了必要的开发工具和库文件,如GCC编译器、make、ncurses和zlib等库文件。 3. 如果编译失败,请查看编译日志,根据错误提示解决问题。 总之,Linux内核源码编译需要一定的技术水平和耐心,但是掌握这个技能对于研究和开发Linux操作系统具有重要的价值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值