博主去年双11购入AMD Vega显卡(为了黑苹果免驱以及硬解),发现Linux下无法驱动Vega显卡,4K显示器分辨率被锁定为1920x2160的奇葩分辨率,无法接受。于是查阅了一下资料发现当时的内核(Linux 4.12)并不支持AMD Vega显卡,于是打算编译内核升级到最新版本,毕竟Linux的舒适使用对博主十分重要。现在分享一下Linux Kernel的编译教程。
注意:由于时间久远,博主的Linux已经升级Ubuntu 18.04,自带内核版本为Linux 4.15,所以演示平台为Ubuntu 18.04,编译的内核版本为Linux 4.18.5。另外,本文中部分图片来源网络。因为经典体系架构图现画比较麻烦。
Linux内核简介
Linux Kernel Map
Linux 系统体系结构
如下图所示,Linux体系结构,从大的方面可以分为用户空间(User Space)和内核空间(Kernel Space)两部分。
- 用户空间中包含了C库,用户的应用程序。在某些体系结构图中还包含了shell,当然shell脚本也是Linux体系中不可缺少的一部分。
- 内核空间包括硬件平台、平台依赖代码、内核、系统调用接口。
- 在任何一个现代操作系统中,都是分层的。为什么需要分层呢?
从程序员的角度分析,将linux底层和应用分开,做应用的做应用,做底层的做底层,各干各的。经济学的基本原理是,分工产生效率。
从安全性的角度分析,是为了保护内核。现代CPU通常都实现了不同的工作模式。
以ARM为例:ARM实现了7种工作模式,不同模式下CPU可以执行的指令或者访问的寄存器不同:(1)用户模式 usr (2)系统模式 sys(3)管理模式 svc(4)快速中断 fiq(5)外部中断 irq(6)数据访问终止 abt(7)未定义指令异常。如果任何一个上层应用都可以调用寄存器,那样肯定是无法稳定执行的。而且因为出现了这个问题,出现了一个新的学科“现代操作系统”,如果大家感兴趣可以看一下“现代操作系统”相关文章或者书籍。
以X86为例:X86实现了4个不同级别的权限,Ring0—Ring3 ;Ring0下可以执行特权指令,可以访问IO设备;Ring3则有很多的限制。如果分析一下Android的,这方面做的更加“丧心病狂”,Android所有的APK应用程序,都是在Java虚拟机上面运行,应用程序更加远离底层。
另外,用户空间和内核空间是程序执行的两种不同状态,我们可以通过“系统调用”和“硬件中断”来完成用户空间到内核空间的转移。
Linux Kernel体系结构
如下图所示,是Linux内核结构图。
SCI层(System Call Interface)
这一层是给应用用户空间提供一套标准的系统调用函数来访问Linux。前面分析Linux体系结构的时候,介绍过任何一类现代操作系统都不会允许上层应用直接访问底层,在Linux中,内核提供了一套标准接口,上层应用就可以通过这一套标准接口来访问底层。
PM(Procees Management)
这一部分包括具体创建进程(fork、exec),停止进程(kill、exit),并控制他们之间的通信(signal等)。还包括进程调度,控制活动进程如何共享CPU。这一部分是Linux已经做好的,在写驱动的时候,只需要调用对应的函数即可实现这些功能,例如创建进程、进程通信等等。
MM(Memory Management)
内存管理的主要作用是控制多个进程安全的共享内存区域。
VFS(Virtual File Systems)
虚拟文件系统,隐藏各种文件系统的具体细节,为文件操作提供统一的接口。在Linux中“一切皆文件”,这些文件就是通过VFS来实现的。Linux提供了一个大的通用模型,使这个模型包含了所有文件系统功能的集合。如下图所示,是一个虚拟文件系统的结构图。
DD(Device Drivers)
设备驱动,Linux驱动一般分为网络设备、块设备、字符设备、杂项设备,需要我们编写的只有字符设备,杂项设备是不容易归类的一种驱动,杂项设备和字符设备有很多重合的地方。
PD(Physical Devices)
这一部分提供丰富的网络协议支持。
Linux Kernel源码
官网
目录结构
- arch:根据cpu体系结构不同而分的代码
- block:部分块设备驱动程序
- crypto:加密,压缩,CRC校验算法
- documentation:内核文档
- drivers:设备驱动程序
- fs(虚拟文件系统vfs):文件系统
- include:内核所需的头文件,(与平台无关的头文件在include/linux中)
- lib:库文件代码(与平台相关的)
- mm:实现内存管理,与硬件体系结构无关的(与硬件体系结构相关的在arch中)
- net:网络协议的代码
- samples:一些内核编程的范例
- scripts:配置内核的脚本
- security:SElinux的模块
- sound:音频设备的驱动程序
- usr:cpio命令实现,用于制作根文件系统的命令(文件系统与内核放到一块的命令)
- virt:内核虚拟机
Linux Kernel编译
下载Linux Kernel源码并拷贝到/usr/src目录
输入如下命令即可
1 | wget https://cdn.kernel.org/pub/linux/kernel/v4.x/linux-4.18.5.tar.xz && cp -f linux-4.18.5.tar.xz /usr/src |
解压Linux Kernel
输入如下命令即可
1 | cd /usr/src && tar -xvf linux-4.18.5.tar.xz && cd linux-4.18.5 |
Ubuntu 18.04下,要执行以下命令安装软件包
1 | sudo apt install fortune |
配置编译参数
执行以下命令,会弹出一个带有GUI的配置界面,一般情况下默认参数就好,光标移动到Save保存即可,如图:
1 | make menuconfig |
编译内核
1 | make -j12 |
注意:-j参数后面加本机CPU线程数可以加快编译速度,博主的8700K是6核心12线程,所以-j12。
安装内核
1 | make modules_install |
更新grub引导
1 | update-grub2 |
至此升级内核成功,重启电脑。
确认升级是否成功
重启后,执行如下命令查看内核版本,和编译版本相同即为成功
1 | uname -r |