实验环境:服务器版Ubuntu,内核版本
●添加系统调用的入口参数
进入解压得到的内核文件夹linux-2.6.30.6中,在linux-2.6.30.6/arch/x86/include/asm/unistd_32.h文件中增加:
#define __NR_zzr_calculator 335
添加系统调用的入口参数(注意:其中会顺序定义入口参数的序号,添加的序号是在原有最大值的基础上+1)
●在linux-2.6.30.6/arch/x86/kernel/syscall_table_32.S 中添加:.long sys_zzr_calculator/* 335 */
●添加自定义系统响应函数
修改linux-2.6.30.6/kernel/sys.c文件,在文件末尾添加自定义的系统响应函数。函数实现如下:
/* The system call function Added in by ZZR. */ asmlinkage int sys_zzr_calculator(int *result, int first, int second, char op) { switch(op){ case '+': *result = first + second; break; case '-': *result = first - second; break; case '*': *result = first * second; break; case '/': if(second == 0){ printk("divisor can't be 0.\n"); return -1; } *result = first / second; break; default: printk("operator is illegal.\n"); return -1; } return 0; }●编译内核
●编写测试代码
#include <stdio.h> #include <sys/syscall.h> #include <unistd.h> #include <time.h> #define __NR_zzr_calculator 335 int main(){ clock_t start,finish; double duration1,duration2; start = clock(); int result; syscall(__NR_zzr_calculator,&result,3,4,'+'); printf("+:\t%d\n",result); syscall(__NR_zzr_calculator,&result,3,4,'-'); printf("-:\t%d\n",result); syscall(__NR_zzr_calculator,&result,3,4,'*'); printf("*:\t%d\n",result); syscall(__NR_zzr_calculator,&result,3,4,'/'); printf("/:\t%d\n",result); finish = clock(); duration1 = (double)(finish-start)/CLOCKS_PER_SEC; printf("Kernel:\t%f\n",duration1); return 0; }
注意:代码是通过系统调用syscall()函数来调用我们自己新增的系统调用,该函数的参数与新增的系统调用一致。
●查看结果
结果正确,实验成功。●dmesg –c
查看是否有添加系统调用成功信息。dmesg 是一个显示内核缓冲区系统控制信息的工具;比如系统在启动时的信息会写到/var/log/
注:dmesg 工具并不是专门用来查看硬件芯片组标识的工具,但通过这个工具能让我们知道机器中的硬件的一些参数;因为系统在启动的时候,会写一些硬件相关的日志到 /var/log/message* 或/var/log/boot* 文件中;
如果我们用这个工具来查看一些硬件的信息;这个工具信息量太大,的确需要耐心。
$ dmesg
$ dmesg -c 注:清理掉缓冲区,下次开机的时候还会自动生成。