实验环境:服务器版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 注:清理掉缓冲区,下次开机的时候还会自动生成。