openwrt在mips架构上添加自定义系统调用
操作步骤
1)添加系统调用号
2)在系统调用表中添加该调用的服务例程
3)实现系统调用服务例程
4)重新编译内核
5)编写用户态程序测试
下面详细说明
一、添加系统调用号
在源码目录arch/mips/include/uapi/asm/unistd.h中修改。
修改之前:
修改之后:
__NR_mycall就是我们添加的自定义系统调用号,添加之后还要修改两处,下面的两个红色框已标处,__NR_Linux_syscalls是最后一个系统调用号在系统调用表中的偏移,__NR_O32_Linux_syscalls是系统调用的个数。
二、在内核系统调用表中添加自定义系统调用的服务例程(即:系统调用入口)
根据你的内核位数(32/64bit)选择需要修改的文件:
arch/mips/kernel/scall32-o32.S ---->32bit kernel
arch/mips/kernel/scall64-64.S ---->64bit kernel
64bit内核,如果支持兼容32bit ABI(o32,n32)则还需要修改scall64-o32.S或scall64-n32.S
我这里用的是32bit内核,在arch/mips/kernel/scall32-o32.S中找到如下所示的地方:
添加自定义的系统调用入口,如下图所示:
注意此函数的系统调用号是4356,测试时要注意。
三、实现系统调用
在arch/mips/kernel/syscall.c中给出一个实现:
asmlinkage int sys_mycall(void)
{
printk(KERN_EMERG "kernel,wqj debug: into[%s],call[%s()]\n",__FILE__,__FUNCTION__);
return 100;
}
四、重新编译内核
五、编写应用程序测试
c语言测试:
#include <stdio.h>
#include <linux/unistd.h>
#include <sys/syscall.h>
#include <errno.h>
int main()
{
int n;
n = 0;
n = syscall(4356);//注意这里的调用号是4356,不是356
printf("Wqj: syscall return value is [%d]\n", n);
return 0;
}
汇编文件测试:
.text
.global main
.ent main
main:
li $2,4356
syscall
.end main
编译测试程序运行后,会在屏幕上输出 Wqj: syscall return value is[100],用dmesg查看会看到printk函数打印的信息。亲测可用,这里不粘贴结果了,因为结果忘记截图了。