Ubuntu增加一个系统调用

本文详细介绍如何在Ubuntu系统上通过下载内核源码并进行修改来实现自定义系统调用的方法,包括从源码下载、配置、编译到最终测试的全过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


1.首先,作为一个Ubuntu系统,需要下载一套内核源码,网址是www.kernel.org.  下载了一个版本后解压到自己想要的位置。

我是解压到的/usr/src这个文件夹下面。

2.进入 linux-版本号/arch/x86/syscalls这个文件夹,打开这个文件,在文件的最末尾写上一行.

格式大概是这样子

545(相当于标识符)     64(表示64位系统)      myservice(自己命名)           sys_myservice(函数名字)

3.

前面的编写相当于一个声明,但是我们还需要对头文件声明一下

进入linux-版本号/include/linux  打开,在#endif后面加上asmlinkage int sys_myservice(void);


4.

之前声明完了很多东西,现在开始写函数的内容

进入linux-版本号/kernel,建立一个myservice.c的文件



5.

还是这个文件夹

这里要创建一个Kconfig file(我也不知道是什么,但是应该是和这个c文件相关的)

命名为Kconfig.myservice

6.

继续在这个文件夹下(如果不是创建一个新的文件,就不需要这步骤了)

编辑一个叫做Makefile的文件,这里应该是包含了我们写的函数的描述之类的东西

7.

回到上级文件夹,编辑它里面的Makefile文件,需要改变的就一行,这个就是和你编译出来的内核名字相关了(随意写)

8.

所有的基础工作已经完成了,现在开始编译我们的内核

a).

首先需要建立一个.config文件,可以通过make menuconfig指令来执行,但是需要设置大量的参数,所以我们可以利用/boot 里面的文件

把它复制到linux-版本号这个文件夹下面,然后运行make menuconfig指令,

看到这样的界面,运行<Load>,输入的名字


,然后save,命名为.config,然后看看有没有出现这个文件(这个文件默认是隐藏的,所以需要Ctrl+H)

b).

既然已经配置好了,我们就可以开始编译安装了。

因为我是Ubuntu系统,所以很方便

配置完内核之后,接下来要执行真正的编译过程。通常我们可以这样下命令:

make-kpkg  --initrd --revision wwang.001 --append-to-version -20110107 kernel_image

1、--initrd选项会让make-kpkg自动帮我们生成initramfs;

2、--revision会给生成的deb文件加上一个版本信息。这个参数只是影响到文件名,如果不指定,默认会是“10.00.Custom”;

3、--append-to-version也是一种版本信息,它不仅出现在deb安装包的文件名里,也会影响到kernel的名称,比如本例中,内核更新完成之后,用“uname -r”察看会得到“2.6.36-20110107”;

4、kernel_image表示生成内核和默认模块的安装包,另外您也可以加上kernel_headers,这样make-kpkg会再生成一个内核头文件的安装包。

如果我们用普通用户来执行make-kpkg,需要加上fakeroot运行。

fakeroot make-kpkg  --initrd --revision wwang.001 --append-to-version -20110107 kernel_image

编译过程执行完毕之后,会在上层目录里生成一个deb安装包,本例中生成的安装包的文件名是“linux-image-2.6.36-20110107_wwang.001_i386.deb”。

上面这一段是百度到的,需要下载一个包,然后就可以用make-kpkg指令了。(编译时间很长)


End.

重启之后选择Ubuntu高级选项,可以看到我们的内核

进来之后grep myservice /proc/kallsyms  ,可以判断我们的系统调用增加成功了没有。


接着我们就可以来写个测试程序来调用我们的[系统调用]了。

  1. #include <linux/unistd.h>  
  2. #include <sys/syscall.h>
  1. int main()  
  2. {  
  3.     syscall(545);     //还记得一开始增加的这个数字吗?
  4.     printf("成功!");
  5.    

然后我们去看看/var/log 里面的kern.log文件,搜索 my service(也就是printk里面打印的东西),检查是不是成功的!


相关资料:http://www.franksthinktank.com/howto/addsyscall/

http://blog.youkuaiyun.com/rk2900/article/details/8281335

http://www.cnblogs.com/wwang/archive/2011/01/07/1929486.html


### 添加自定义系统调用 #### 准备工作环境 为了在Ubuntu虚拟机中添加新的系统调用,首先需要准备合适的工作环境。这包括获取合适的Linux内核源码以及安装必要的编译工具链。 对于获取特定版本的Linux内核源码,在命令行执行如下操作来下载并解压指定版本的内核文件[^4]: ```bash sudo mv linux-5.6.15.tar.xz /usr/src/ cd /usr/src/ sudo tar -xvf linux-5.6.15.tar.xz ``` 接着安装用于构建新内核所需的各种开发库和支持程序: ```bash sudo apt-get update && sudo apt-get install build-essential libncurses-dev bison flex libssl-dev libelf-dev ``` #### 修改内核配置 进入已解压好的内核源代码目录 `/usr/src/linux-5.6.15` 后,可以利用 `make menuconfig` 或者其他方式调整内核配置选项以适应个人需求。如果只是简单测试,则可以直接沿用默认设置继续下一步骤。 #### 实现新的系统调用函数 创建一个新的C语言源文件放置于适当位置(比如放在arch/x86/kernel/syscall_example.c),编写想要实现的新功能逻辑。这里假设要加入的是一个简单的返回字符串信息给用户空间的应用程序作为例子: ```c // syscall_example.c #include <linux/linkage.h> #include <linux/kernel.h> asmlinkage long sys_myexample(void){ printk(KERN_INFO "System call myexample was called.\n"); return 0; } ``` 此部分实现了名为`sys_myexample()` 的系统调用接口,当被调用时会在日志里记录一条消息,并向应用程序返回成功状态码0[^2]。 #### 更新系统调用表 为了让操作系统识别这个新增加系统调用,还需要编辑体系结构相关的头文件中的系统调用表格。针对X86架构而言通常是修改位于 arch/x86/entry/syscalls/syscall_64.tbl 文件的内容,在最后一行之后追加上述所写入系统的编号及其对应的处理函数名: | 地址 | 系统调用号 | ABI | 名字 | |------|------------|-----|--------------| | ... | | | | | 新增 | 379 | 64 | sys_myexample| 注意这里的“379”只是一个示例数值,实际应用当中应当查询最新的官方文档确认具体的可用ID范围[^1]。 #### 编译与替换现有内核 完成上述更改后就可以开始重新编译整个内核镜像了。考虑到时间成本较高建议先运行 make olddefconfig 来快速同步最新改动前后的差异再正式build: ```bash make olddefconfig time make -j$(nproc) sudo make modules_install sudo cp arch/x86/boot/bzImage /boot/vmlinuz-linux-custom sudo mkinitcpio -p linux-custom sudo grub-mkconfig -o /boot/grub/grub.cfg ``` 最后重启计算机选择刚制作出来的定制化内核启动即可生效新添加进去的那个系统调用了[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值