向linux内核增加新的系统调用,为Linux内核添加系统调用

目标:向内核添加系统调用long get_shed_times(unsigned long * num),程序调用此函数时,将此进程被调度的次数存入num指向的内存单元中,32位整数。

系统环境:CentOS 5.5 32bit + 2.6.18 source code + i386架构

首先在task_struct中添加调度计数变量unsigned long sched_times;

include/linux/sched.h

task_struct {

.........

unsigned long sched_times;

..........

};

在创建新进程时将sched_times初始化为0

kernel/fork.c   do_fork()函数

longdo_fork(unsignedlongclone_flags,

unsignedlongstack_start,

structpt_regs *regs,

unsignedlongstack_size,

int__user *parent_tidptr,

int__user *child_tidptr)

{

structtask_struct *p;

inttrace = 0;

structpid *pid = alloc_pid();

longnr;

............

p = copy_process(clone_flags, stack_start, regs, stack_size, parent_tidptr, child_tidptr, nr);

p->sched_times = 0;

................

}

进程调度时,sched_times ++

kernel/sched.c  scheduale()函数

asmlinkagevoid__sched schedule(void)

{

........

idx = sched_find_first_bit(array->bitmap);

queue = array->queue + idx;

next = list_entry(queue->next,structtask_struct, run_list);

next->sched_times ++;

if(!rt_task(next) && interactive_sleep(next->sleep_type)) {

unsignedlonglongdelta = now - next->timestamp;

if(unlikely((longlong)(now - next->timestamp) 

delta = 0;

.........

}

添加系统调用

kernel/sys.c 最后边

asmlinkagelongsys_get_sched_times(unsignedlong* addr)

{

printk(KERN_ALERT "get_sched_times called,sched_times=%d",current->sched_times);

returncopy_to_user(addr,&(current->sched_times),sizeof(long));

}

添加系统调用号 include/asm/unistd.h

3da8d7c63a8c860dca4dd1302da4aa1a.png

在最后,加入#define __NR_get_sched_times   318,并将NR_syscalls改为319

ed5f65b5042df101429178b9bc6a91be.png

在文件中arch/i386/kernel/syscall_table.S最后添加如下内容:

8f1a43d1ebf0f8d079a0d2e8e4bcb5d4.png

OK,重新编译内核,重启。

测试:

通常,系统调用需要靠C库支持。但是,也可以直接用syscall或者用Linux本身提供的宏

宏:  __syscalln()  n的范围从0到6,代表需要传给系统调用的参数个数,例如open

long open(const char *filename, int flags,int mode);

#define  __NR_open     5

__syscall3(long,open,const char*,filename,int,flags,int,mode)

上述方法在2.6.20以后就去掉了,因为有安全漏洞。

直接使用syscall(系统调用号,参数...)

#include 

intmain()

{

unsignedlongnum;

inti,x,sum=0;

scanf("%d",&x);

for(i=0;i

sum = sum +i;

printf("%d\n",sum);

if(syscall(318,&num))

printf("Failed\n");

else

printf("sched_times =%d\n",num);

return0;

}

运行结果:

用户层

531ab1cfe9503e3da572b89aa455f68b.png

内核层

aa993d95818f43e48fe59e5eb1aa4f68.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值