转自:http://bjshywang1.blog.163.com/blog/static/127915267200910246464068/
系统调用是应用程序和操作系统内核之间的功能接口。其主要目的是使得用户可以使用操作系统提供的有关设备管理、输入/输入系统、文件系统和进程控制、通 信以及存储管理等方面的功能,而不必了解系统程序的内部结构和有关硬件细节,从而起到减轻用户负担和保护系统以及提高资源利用率的作用。
Linux操作系统作为自由软件的代表,它优良的性能使得它的应用日益广泛,不仅得到专业人士的肯定,而且商业化的应用也是如火如荼。在Linux 中,大部分的系统调用包含在Linux的libc库中,通过标准的C函数调用方法可以调用这些系统调用。那么,对Linux的发烧友来说,如何在 Linux中增加新的系统调用呢?
我们通过添加一个虚构的系统调用foo()来说明。
(1)把sys-foo加入到系统调用表里去(系统调用的命名规则要求加sys前缀)。对于大多数体系结构来说,该表位于entry.s(在新的linux版本中可能为syscall_table.s)中。
入口位置:/usr/src/linux/arch/i386/kernel/syscall_table.S,形式如下:
.long sys_exit
.long sys_fork
...
.long sys_foo(我们在文件末尾所添加的行,虽然没有明确指定编号,但我们加入的这个系统调用会被按照次序分配给一个系统调用号。对于每种需要支持的体系结构,都必须添加进去,但每个体系结构中的系统调用号不需要相同。)
(2)定义系统调用号,/usr/src/linux/include/asm-i386/unistd.h
#define _NR_foo ***
//不能与前面已有的重复,***为我们在syscall_table.s中添加的位置(行数),注意行数从0开始
#define _NR_syscalls 319//修改系统中所用系统调用数目
(3)添加系统调用定义在include/linux/syscalls.h
asmlinkage long sys_foo(int a);
特别注意如果使用用户态的指针,需要添加"__use”r标志
(4)在合适的地方添加系统调用实现函数,位置的选择:
(1)无论何种配置,该系统调用都必须编译到核心的内核映像中去,所以可以把它放到kernel/sys.c文件中
(2)看它的功能与linux何种功能相关,就添加到相应的目录中
asmlinkage int sys_foo()
{
return THREAD_SIZE;
}
最后我们可以写一个测试程序来使用这个系统调用:
#define _NR_foo ***
_syscall0(long,foo)
int main()
{
long stack_size=foo();
printf("The kernel stack size is %ld ",stack_size);
return 0;
}