首先参看下《TCP/IP 详解卷2》的“15.4 系统调用”章节。
下面直接开始实际的操作吧^_^:
1.首先在 /usr/src/sys/kern/init_sysent.c 中添加系统调用的入口:
先声明 hello 是一个函数,然后在 sysent 表中添加 hello 的项目:
3 表示 hello 系统调用有三个参数,系统调用号为 205.
2.然后是添加一个实现文件,当然,把系统调用的实现放在已有的文件里也是可以的。添加的实现文件为:/usr/src/sys/kern/kern_hello.c:
3.由于添加了一个文件,所以需要修改下对应的编译文件 /usr/src/sys/conf/files:
系统调用到这里就添加完成了,重新编译内核,替换旧的内核就可以了。
但是,要怎么调用呢?如果可以直接使用汇编语言调用的话,会相对简单点,不过,呃,NetBSD-1.0下怎么用汇编语言调用系统调用还没掌握(-_-),所以只能先用C语言的方式调用了。
在《TCP/IP 详解卷2》的“1.6 系统调用和库函数”章节有这样一段话:
“在各种Unix系统中,每个系统调用在标准C函数库中都有一个相同名字的函数。一个应用程序用标准C的调用序列来调用此函数。这个函数再调用相应的内核服务,所使用的技术依赖于所在系统。”
先验证下一个简单的系统调用:
在NetBSD-1.0上编译运行:
看看 write 依赖的库文件:
为什么 libc 库提供的链接符号是 _write 而不是 write 呢?看看 write.c 对应的汇编文件:
汇编语言中调用的就是 _write ,想必这是C函数库提供的机制吧。
知道了C语言系统调用的接口是放在 libc 库文件的,下面就直接描述怎么添加这个接口,至于相关的细节先跳过了:
1. 在 /usr/src/sys/sys/syscall.h 最后添加对应的系统调用号:
2.在 /usr/src/lib/libc/sys/Makefile.inc 中添加 hello 系统调用的编译:
(请原谅我目前使用的截图软件没有办法给 hello.o 添加红框......)
3.在 /usr/src/include/unistd.h 中添加系统调用的声明:
最后要重点提一下,syscall.h 和 unistd.h 文件除了要同步到对应的源码路径下,还需要同步到对应的头文件目录下:
/usr/src/sys/sys/syscall.h --------------- /usr/include/sys/syscall.h
/usr/src/include/unistd.h --------------- /usr/include/unistd.h
这样编译 libc 库文件的时候使用的头文件才是正确的。
libc 库文件编译出来之后,要直接覆盖掉 /usr/lib 目录下的原有文件: mv libc* /usr/lib/ 。
如果只是把 /usr/lib 目录下的原有文件重命名的话,可能会出现仍然加载旧文件的情况,可以使用 ldconfig 命令确认:
新的库文件覆盖好之后,重启NetBSD(不重启能否生效的情况我没测试过...)。然后验证下系统调用是否正常了:
编译运行: