【操作系统】实验四 增加Linux系统调用

简介: 【操作系统】实验四 增加Linux系统调用

😘欢迎关注:👍点赞🙌收藏✍️留言

🏇码字不易,你的👍点赞🙌收藏❤️关注对我真的很重要,有问题可在评论区提出,感谢支持!!!

实验四

实验内容

  1. 实验名称:增加Linux系统调用
  2. 实验任务:

学习如何产生一个系统调用以及怎样同过往内核中增加一个新函数从而在内核空间中实现对用户空间的读/写。
部分A
添加一个新的内核系统调用,具体完成某个你希望实现的功能。
部分B
重新编译内核,使你的系统调用可用。
部分C
编写一个用户态的程序,验证你增加的系统调用。

实验过程

部分A: 添加新的内核系统调用

在最新版本的 Linux 内核源代码中,系统调用入口表通常不再在 entry_64.S文件中。系统调用入口表的位置在不同的内核版本中可能会有所变化。通常,它会在 arch/x86/entry/syscalls/syscall_64.tbl 文件中定义。

新下载一个内核源代码,然后解压再进行下面步骤:

  1. 打开终端并进入内核源代码目录。根据内核版本,可能会有所不同:
    cd /usr/src/linux-6.0.1 # 根据你的内核版本进入相应的目录
  2. 查找系统调用入口表的位置。可以使用以下命令找到它:

find . -name syscalls.h

这将在整个内核源代码目录中搜索 syscalls.h文件,如图所示。

vim ./arch/x86/include/asm/syscalls.h

图1 找到的syscalls.h 文件

打开找到的 syscalls.h 文件,在里面声明自己的系统调用函数,如图2所示。

图2打开syscalls.h文件

  1. 添加asmlinkage long sys_helloworld(void);声明,如图3所示。

    图3 添加声明
  2. 添加系统调用id:文件为 syscall_64.tbl它包含系统调用号和系统调用函数之间的映射,如图4所示。
vim ./arch/x86/entry/syscalls/syscall_64.tbl

图4 添加系统调用

在文件中找到一个空的系统调用入口,通常为sys_ni_syscall

使用命令 /sys_ni_syscall 搜索,找到以后回车,然后 点击 i 即可修改

修改内容,如图5所示。:

图5 添加系统调用入口

这表示将系统调用号 156映射到名为 sys_my_call 的系统调用函数。

  1. 保存文件并退出编辑器。
:wq

编写新的系统调用函数,先打开sys.c文件,如图6所示。

图6 打开sys.c文件

6. 在内核源代码目录中,打开文件 kernel/sys.c 以编辑,如图7所示。

图7 执行 vim kernel/sys.c命令

7. 在文件末尾添加你的新系统调用函数 sys_my_call的定义。例如,你可以添加一个简单的示例系统调用,如图8所示。

asmlinkage long sys_helloworld(void) {
printk(“hello world”);
return 1; 
}

图8 添加系统调用

注意:这里函数定义部分有些教程中使用的是 asmlinkage helloworld(void)在高版本的内核中很可能会出现以下报错:

arch/x86/entry/syscall_64.o:(.rodata+0xa78):undefined reference to '__x64_sys_helloworld'

在我的版本下使用如下函数形式,如图9所示。

SYSCALL_DEFINE0(helloworld) {
Printk(“hello world!\n”);
return 0; 
}

图9 添加系统调用

8. 保存文件并退出编辑器。

:wq

部分B: 重新编译和安装内核

  1. 在终端中编译内核:
  2. 在此之前需要 make clean 清理一下旧文件
  3. 重新配置.config文件 make menuconfig

make menuconfig 时将general setup -> localversion 修改成新的名称,如 “wikiKernel”

  1. 然后make -j8 编译内核 (时间较长)
    编译成功,没有报错信息,如图10所示。

图10 编译内核完毕

  1. 安装内核
    编译成功,没有报错信息,如图11所示。

图11 执行sudo make modules_install命令安装内核成功

图12 执行sudo make install命令

图13 reboot重启机器

部分C编写一个用户态的程序

验证系统调用,如图14、15所示。

新建hello.c文件,编写程序如下:

#include <stdio.h>
#include<linux/kernel.h>
#include<sys/syscall.h>
#include<unistd.h>
 
int main()
{
       long int a = syscall(156);
       printf("System call sys_helloworld return %ld\n", a);
       return 0;
}

图14 验证系统调用的代码

编译:

Gcc -o a hello.c

运行:

./a.out

图15 执行命令,验证系统调用添加成功

实验小结

在本次实验中,我成功地增加了Linux系统调用,并通过重新编译内核和编写用户态程序的方式实现了对用户空间的读/写功能。在实验过程中,我遇到了以下一些问题和挑战:

存在问题:

  1. 在添加新的系统调用时,需要深入理解Linux内核的机制和源代码,对于初学者来说有一定的难度。
  2. 在重新编译内核时,需要安装并配置相应的编译工具和环境,对于操作系统的了解和熟练掌握程度有一定要求。
  3. 在编写用户态程序时,需要熟悉Linux的系统调用接口,以及如何与内核交互,需要进行比较深入的学习和理解。

注意事项:

  1. 在整个实验过程中,要特别注意安全性原则,避免操作系统内核被破坏或者数据被篡改。
  2. 在修改和编译内核时,务必备份原有的内核文件,以防止操作失误导致系统无法启动。
  3. 在进行用户态程序开发时,需要考虑异常情况的处理和错误信息的输出,确保程序的健壮性和可靠性。

有待提高的能力:

  1. 理解并掌握操作系统的内核机制和体系结构,提高对内核源代码的理解和阅读能力。
  2. 熟悉Linux系统调用的接口和使用方法,能够灵活地进行系统调用操作。
  3. 学会使用编译工具和环境,能够独立完成内核的编译和安装过程。
  4. 提高代码的编写能力和调试技巧,加强对程序异常情况的处理能力。
  5. 培养独立思考和解决实际问题的能力,寻找并探索更多的实践机会,不断提高自己的技术水平和能力。

相关文章
|
2天前
|
存储 缓存 安全
Linux基础——冯诺依曼体系结构与操作系统
Linux基础——冯诺依曼体系结构与操作系统
Linux基础——冯诺依曼体系结构与操作系统
|
3天前
|
网络协议 Linux 数据安全/隐私保护
【Linux操作系统】权限管理和粘滞位
【Linux操作系统】权限管理和粘滞位
【Linux操作系统】权限管理和粘滞位
|
3天前
|
Linux Shell 程序员
【Linux操作系统】命令的运行原理
【Linux操作系统】命令的运行原理
|
3天前
|
Linux
【Linux操作系统】基本指令-2
【Linux操作系统】基本指令
【Linux操作系统】基本指令-2
|
3天前
|
安全 Linux
【Linux操作系统】基本指令-1
【Linux操作系统】基本指令
|
1天前
|
算法 网络协议 Linux
探索Linux命令idn:处理国际化域名
`idn`命令在Linux中用于处理国际化域名,转换成ASCII兼容的ACE格式或反之。它支持Punycode算法,提供命令行接口及多种参数,如`-a`转ASCII,`-d`转回国际化域名。示例包括将`xn--zhonggu-wu9d.com`转换。使用时注意有效输入,考虑版本兼容性,并可与其他工具结合使用。
|
1天前
|
Linux 数据处理 数据库
深入解析Linux命令id:理解用户身份与权限
`id`命令在Linux中用于显示用户身份(UID, GID和附加组)。它查看系统用户数据库获取信息。参数如`-u`显示UID,`-g`显示GID,`-G`显示附加组,结合`-n`显示名称而非ID。用于确认命令执行者身份,确保权限正确。在脚本中使用时注意权限管理,遵循最小权限原则。
|
1天前
|
Linux 数据处理
Linux命令iconv:字符编码转换的利器
`iconv`是Linux下的字符编码转换工具,支持多种编码如UTF-8、ISO-8859-1等。它允许用户指定源(-f)和目标(-t)编码,转换文件或输出到指定文件(-o)。使用`-l`可列出所有支持的编码。示例:将UTF-8文件转为ISO-8859-1编码:`iconv -f UTF-8 -t ISO-8859-1 input.txt -o output.txt`。在转换前确认源编码,测试小样本,备份数据,并注意特殊字符处理。
|
1天前
|
网络协议 Linux 网络安全
Linux命令hostnamectl:掌握系统主机信息的利器
`hostnamectl`是Linux系统管理的关键工具,用于查看和设置主机名、内核信息等。它集成在`systemd`中,通过修改配置文件交互。命令特点包括综合显示多种信息、简单语法和设置功能。例如,`hostnamectl status`显示系统详情,`sudo hostnamectl set-hostname NEWHOSTNAME`用于更改主机名。使用时注意权限、备份配置、更新网络和重启相关服务,避免频繁更改。

热门文章

最新文章