美文网首页
关于Ubuntu20.04更换内核/添加系统调用

关于Ubuntu20.04更换内核/添加系统调用

作者: Wannay | 来源:发表于2021-04-16 04:37 被阅读0次

0. 说明

使用VMware安装虚拟机,采用的是Ubuntu20.04LTS,默认的Kernel版本为Linux-5.8.0,是2021年04月官网最新20.04LTS镜像。

我想要替换的内核为:Linux-5.4.1,通过编译源码的方式添加系统调用(如果不需要添加系统调用,单纯编译源码则跳过对应的步骤即可)。

1. 准备工作

下载内核,内核源码下载地址如下,我使用的是5.4.1,要使用其他版本的话可以借鉴,但是很有可能发生改变,比如5.8的源码中就有很多不同,选用过高版本或者过低版本的需要注意~

http://ftp.sjtu.edu.cn/sites/ftp.kernel.org/pub/linux/kernel/v4.x/  # 4.x的版本
http://ftp.sjtu.edu.cn/sites/ftp.kernel.org/pub/linux/kernel/v5.x/  # 5.x的版本

将源码下载到自己的电脑上,并通过直接拖拽到VMware(需要安装VMTools)中虚拟机对应的位置上或者使用XShell、FinalShell、Putty、宝塔面板等工具上传到自己的虚拟机上。

wanna@ubuntu:~$ pwd
/home/wanna
wanna@ubuntu:~$ ls
Desktop    Downloads   linux-5.4.1.tar.gz  linux-5.8.1         Music     Public  Templates  test.c
Documents  install.sh  linux-5.4.1.zip     linux-5.8.1.tar.gz  Pictures  snap    test       Videos

可以看到我已经将源码上传到自己的虚拟机的Home目录下。

2. 添加系统调用,不需要的可以跳过此步骤

tar -xvzf  linux-5.4.1.tar.gz   # 解压压缩包,如果下载的是zip的源码的话使用unzip  linux-5.4.1.tar.gz 解压
cd linux-5.4.1   # 切换到linux源码的根目录下
sudo vim arch/x86/include/asm/syscalls.h  # 使用vim编辑器编辑syscalls.h文件,也可以使用其他编辑器

在第47行左右的空行(可以使用vim命令:47直接跳转到47行),添加syscall的函数声明asmlinkage long sys_helloworld(void);

image.png

使用如下命令编辑sys.c,添加sys_helloworld函数的具体实现(对应上面添加的声明的实现)

sudo vim kernel/sys.c  # 使用vim编辑器打开文件

在2650行左右的右花括号之后,#endif之前添加函数主体

asmlinkage long sys_helloworld(void)
{       
        printk("hello_world\n");
        return 0;
} 
image.png

使用如下命令编辑64位的系统调用表

sudo vim arch/x86/entry/syscalls/syscall_64.tbl  # 使用vim编辑系统调用表

在第346行添加一行如下内容,第一列是系统调用号(可以自定义,但是别和系统已经有的冲突,还有测试的时候调用号别忘了改),第4列是需要执行的函数的名称。

336     64      helloworld              sys_helloworld
image.png

3. 准备编译内核

3.1 执行过程如下

依次执行如下命令

sudo make clean

#依次执行下面的命令安装依赖
sudo apt-get install gcc make libncurses5-dev openssl libssl-dev 
sudo apt-get install build-essential 
sudo apt-get install pkg-config
sudo apt-get install libc6-dev
sudo apt-get install bison
sudo apt-get install flex
sudo apt-get install libelf-dev

sudo make oldconfig  #仍采用原内核配置文件

# 执行下面的命令时,可能会有证书等问题,但是一定要先执行!!!
# 不然貌似没有配置文件存在,具体解决方法见下述
# 建议将虚拟机的核心数修改为4-8核心,内存修改为2G+
sudo make 

sudo mv  ~/linux-5.4.1  /usr/src/   # 将内核源码等移动到/usr/src/目录下
cd /usr/src/linux-5.4.1/     # 切换到cd /usr/src/linux-5.4.1/目录下

sudo make modules_install  # 安装模块
sudo make install  # 安装内核

# 下面两句命令执行过程可能会有问题,就是仍旧没切换到我们想要的内核,具体看后续
# (有可能可以有可能不行,建议可以尝试一波,按照别人的博客可以,我自己不行)
sudo mkinitramfs -o /boot/initrd.img-5.4.1
sudo update-initramfs -c -k 5.4.1

sudo update-grub   # 将内核启动项改为我们自己编译的内核

uname -r   # 在重启前看看自己的内核版本
sudo reboot   # 重启
uname -r  # 在重启后看看自己的内核版本,如果内核版本发生变化了,则替换成功!

3.2 关于sudo make命令过程中可能出现的问题:

No rule to make target 'debian/canonical-certs.pem', needed by 'certs/x509_c......

证书问题,解决方法如下:

cd /usr/src/linux-5.4.1/     # 切换到cd /usr/src/linux-5.4.1/目录下,如果已经在了则不用切换
sudo vim .config   # 使用vim编辑.config文件

将配置文件中的如下配置全部注释掉

# 使用vim的类似/CONFIG_MODULE_SIG_ALL等命令去进行搜索
# 也可以使用类似Ctrl+F的方式去修改
CONFIG_MODULE_SIG_ALL 
CONFIG_MODULE_SIG_KEY
CONFIG_SYSTEM_TRUSTED_KEYS
CONFIG_DEBUG_INFO

3.3 关于执行如下这段代码之后仍旧没有按照我们的内核启动的解决

sudo mkinitramfs -o /boot/initrd.img-5.4.1
sudo update-initramfs -c -k 5.4.1
sudo update-grub   # 将内核启动项改为我们自己编译的内核
uname -r   # 在重启前看看自己的内核版本
sudo reboot   # 重启
uname -r  # 在重启后看看自己的内核版本,如果内核版本发生变化了,则替换成功!

执行如下操作查看当前有哪些内核可以启动

grep 'menuentry' /boot/grub/grub.cfg
image.png

使用如下命令编辑启动项的配置文件

sudo vim /etc/default/grub   # 编辑配置文件

将配置文件中的如下配置项改为如下(修改我们默认的启动项):

# Ubuntu, with Linux 5.4.1这部分可以根据grep 'menuentry' /boot/grub/grub.cfg的结果来进行选择
# 指定为自己的内核名字就行,记得一定要加上前缀Advanced options for Ubuntu>,还有记得加上引号表示字符串
# 一定要加前缀的!至于为什么要加上这个前缀,看本博客第5点就明白了!
GRUB_DEFAULT="Advanced options for Ubuntu>Ubuntu, with Linux 5.4.1"

继续执行

sudo update-grub   # 将内核启动项改为我们自己编译的内核
uname -r   # 在重启前看看自己的内核版本
sudo reboot   # 重启
uname -r  # 在重启后看看自己的内核版本,如果内核版本发生变化了,则替换成功!
image.png

4. 编写测试代码验证我们的系统调用已经插入成功!

cd ~/   # 切换到Home目录下
vim test.c   #  使用vim编辑test.c文件
#include <stdio.h>
#include <linux/kernel.h>
#include <sys/syscall.h>
#include <unistd.h>
int main(int argc,char **argv){
        //336就是我们刚才添加的系统调用的id
        long int a=syscall(336);
        printf("System call sys_helloworld return %ld\n",a);
        return 0;
}

使用如下的命令编译并执行

gcc test.c -o test  # 将test.c编译成test的二进制文件
./test    # 执行编译好的文件

执行结果如下

image.png
最终也就是得到了return 0,注意这里如果return -1则说明我们添加的系统调用并没有成功,由于我们的系统调用函数是打印了一行日志文件,hello_world,因此我们可以使用dmesg查看到
dmesg | tail -1   # 只展示最后一行日志
image.png

5. 事后补充

VMware可以在开机的时候选择启动项(老版本的在开机时长按Shift键,新版本的在开机时长按Esc键,应该没人用老版本了吧),这个界面叫做grub引导界面。


image.png

看到这个界面我们似乎就明白了,为什么上面配置启动项的时候要加上前缀,因为那相当于是二级目录啊!在下面这个界面中我们就能选择对应的内核进行启动了。


image.png

相关文章

  • 关于Ubuntu20.04更换内核/添加系统调用

    0. 说明 使用VMware安装虚拟机,采用的是Ubuntu20.04LTS,默认的Kernel版本为Linux-...

  • 更新/etc/resolv.conf,服务不生效问题

    背景 linux操作系统 cpp程序添加了dnscache 逻辑(调用getaddrinfo()),想更换loca...

  • CentOS7 开启BBR 加速

    先查看系统内核版本,低于4.10的需要先更换内核版本 uname -r 更换内核版本 1 导入公钥 rpm --i...

  • 第一节 linux是什么

    1、linux是什么 组成: 硬件、内核、系统调用、应用程序。 操作系统组成: 内核+系统调用。 2、li...

  • Linux内核编译及添加系统调用

    首先让我们脑补一下什么是系统调用,个人理解系统调用就是调用系统函数(内核状态下的函数)。在linux系统中,用户通...

  • 在linux内核中添加系统调用

    下载linux源码,我用的是4.6.3 解压源码 在arch/x86/entry/syscalls/syscall...

  • Linux内核编译及添加系统调用

    1 总体设计思路 系统调用的本质是调用内核函数,以内核态运行程序。为了在内核态下运行,本实验针对Linux的内核进...

  • Linux内核编译及添加系统调用

    1 总体设计思路 系统调用的本质是调用内核函数,以内核态运行程序。为了在内核态下运行,本实验针对Linux的内核进...

  • 模块添加syscall

    在《添加一个syscall》一文中需要修改内核源码和重新编译安装内核才能添加自己的系统调用,这太浪费时间了,调试也...

  • Linux基础篇

    Linux是什么[1] 硬件>>内核>>系统调用>>应用程序 内核+系统调用(+部分应用程序)可以称之为操作系统 ...

网友评论

      本文标题:关于Ubuntu20.04更换内核/添加系统调用

      本文链接:https://www.haomeiwen.com/subject/gtxulltx.html