Chinaunix首页 | 论坛 | 博客
  • 博客访问: 191906
  • 博文数量: 219
  • 博客积分: 1435
  • 博客等级: 中尉
  • 技术积分: 1388
  • 用 户 组: 普通用户
  • 注册时间: 2011-03-17 09:27
文章分类

全部博文(219)

文章存档

2013年(2)

2012年(183)

2011年(34)

我的朋友

分类:

2011-12-12 09:36:44

最近做 linux 系统实验,需要增加系统调用。我采用的方法是通过修改内核源码来增加系统调用。
实验环境为 Linux 3.1.0-4-ARCH,使用的内核源码为 linux-2.6.39.4.tar.bz2.
需要增加的系统调用函数为:

  1. int mycall(int num)
  2. {
  3.     printk("This is my syscall from kernel.\n");
  4.     printk("current pid is: %d.\n", current->pid);
  5.     return num;
  6. }
解压内核源码压缩包,然后修改如下文件:
1) linux-2.6.39.4/arch/x86/kernel/syscall_table_32.S
在最后添加:
  1. .long sys_mycall
2) linux-2.6.39.4/arch/x86/include/asm/unistd_32.h
可以看到原有的最后一个系统调用为:
  1. #define __NR_syncfs 344
在后面添加自己的系统调用:
  1. #define __NR_mycall 345
同时,修改总的系统调用数:
  1. #define NR_syscalls 346
3) linux-2.6.39.4/include/linux/syscalls.h
添加增加的系统调用的声明:
  1. asmlinkage long sys_mycall(int num);
4) linux-2.6.39.4/kernel/sys.c
添加系统调用函数:
  1. SYSCALL_DEFINE1(mycall, int, num)
  2. {
  3.     printk("This is my syscall from kernel.\n");
  4.     printk("current pid is: %d.\n", current->pid);
  5.     return (long)num;
  6. }
配置内核:
  1. $ make localmodconfig
编译内核:
  1. $ make bzImage -j4
编译模块:
  1. $ make modules
安装模块:
  1. $ sudo make modules_install
安装内核:
  1. $ sudo make install
生成引导镜像,我使用的是 Archlinux,生成方法有所不同:
  1. $ sudo mkinitcpio -k /boot/vmlinuz -g /boot/initramfs-2.6.39.img
修改 grub 的 menu.lst,添加新的内核选项,重启后使用新内核进入系统。
测试增加的系统调用,其系统调用号为 345。测试函数如下:
  1. #include <stdio.h>
  2. int main()
  3. {
  4.      printf("The return value is: %d.\n", syscall(345, 5));
  5.      return 0;
  6. }
由于没有编译 nVidia 显卡驱动模块,系统无法进入图形界面。
程序在 tty1 下运行,结果如图:

可见,增加的系统调用已经可以正确调用。

后记:
关于增加系统调用,网上有很多资料,然而大多数都是针对 2.6.2* 的内核的,和此次实验所使用的 2.6.3* 的内核有很多文件都不一样。而且,在 linux-2.6.39.4/kernel/sys.c 文件中增加系统调用那块,许多资料还在使用以前的 _syscallN(...) 方法。
另外,增加系统调用也可以通过内核模块实现,不过需要将 sys_call_table 改为可读可写并将其导出,因为自 linux 2.6.26 之后,为了系统安全,系统符号表默认是只读且不可导出的。我也尝试了这个方法,不过却是失败了。
阅读(325) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~