我是Linux Kernel新手,但是现在涉及到了修改调用的活,
赶鸭子上架了。通过看书和实践,到也做出了简单的东西,感谢
james_zhang和young_yang,写出我的经历,也算是一个入门过
程了。
首先我看了一本书《Linux 操作系统内核分析》(9711课本)。
没有太详细的阅读,直接找到了系统调用的一章,和自己的代码
一对比,发现不一样!看来2.4.x和2.2.x还是有变化的。
然后请教了james_zhang和young_yang,坚定了信心,大体的
原理不会变,结构不会变。
连上了google。搜索“编写系统调用”,够裸吧!
找到了几个网页,打开一看,大体也差不多:
// 以下引用:
需要以下几个基本步骤:
1、编写系统调用函数
编辑sys.c文件:
# cd /usr/src/linux/kernel
# vi sys.c
在文件的最后增加一个系统调用函数:
asmlinkage int sys_print_info(int testflag)
{
printk(" Its my syscall function!n");
return 0;
}
该函数有一个int型入口参数testflag,并返回整数0。
2、修改与系统调用号相关的文件
编辑入口表文件:
# cd /usr/src/linux/arch/i386/kernel
# vi entry.S
把函数的入口地址加到sys_call_table表中:
arch/i386/kernel/entry.S中的最后几行源代码修改前为:
......
.long SYMBOL_NAME(sys_sendfile)
.long SYMBOL_NAME(sys_ni_syscall) /* streams1 */
.long SYMBOL_NAME(sys_ni_syscall) /* streams2 */
.long SYMBOL_NAME(sys_vfork) /* 190 */
rept NR_syscalls-190
.long SYMBOL_NAME(sys_ni_syscall)
.endr
修改后为:
......
.long SYMBOL_NAME(sys_sendfile)
.long SYMBOL_NAME(sys_ni_syscall) /* streams1 */
.long SYMBOL_NAME(sys_ni_syscall) /* streams2 */
.long SYMBOL_NAME(sys_vfork) /* 190 */
.long SYMBOL_NAME(sys_print_info) /* added by I */
.rept NR_syscalls-191
.endr
修改相应的头文件:
# cd /usr/src/linux/include/asm
# vi unistd.h
把增加的sys_call_table表项所对应的向量,在include/asm/unistd.h
中进行必要申明,以供用户进程和其他系统进程查询或调用。
#define __NR_putpmsg 189
#define __NR_vfork 190
#define __NR_print_info 191 /* added by I */
// 引用结束
这一段有点差别,经过分析发现只要在arch/i386/kernel/entry.S单独加上
.long SYMBOL_NAME(sys_print_info) /* added by I */
就行了。
然后就是做过无数次的make了,重起,编个程序一试,不行
# vi test.c
#include
#include
extern int errno;
_syscall1(int,print_info,int,testflag)
main()
{
int i;
i= print_info(0);
if(i==0)
printf("i=%d , syscall success!n",i);
}
分析一下,发现还需要改/usr/include/linux/unistd.h,加上同样的东西
就行了。
然后试验,一切OK,我的Hello World系统调用就这样写出来了!
相信我能做很多人就能像我一样的。其实一些东西只要肯动手都不想想像的
那么难。