您的位置:首页 > 其它

ubuntu10.04编译添加系统调用的内核

2011-10-06 16:36 731 查看
在美国的同学让我帮他完成一到os作业:添加系统调用(具体内容见后文)。查了一些资料后基本了解了操作流程,在经历了一些失败后,总算成功。我是在VMware中ubuntu10.04下完成这个小作业的。
首先从网上下载内核源码包,我下的是和ubuntu10.04默认内核接近的2.6.32。copy到/usr/src目录后,执行tar -xvf(现在不需要区分xzvf和xjvf了)解压,我的目录名是linux-source-2.6.32,以下操作均在该目录下进行。
任务是添加一个系统调用,在用户态通过这个系统调用传进一个结构体指针,在内核态获取一些数据后,填充到该指针指向的结构体中,再返回用户态,并输出。

一、新建或修改文件:
1、在include/linux中新建get_info.h定义结构体,出于某些考虑,简化了一些内容
struct str_info{
long state;
/* current state of process */
pid_t pid;
/* process id (input) */
unsigned long start_time;
/* process start time */
long user_time;
/* CPU time spent in user mode */
long sys_time;
/* CPU time spent in system mode */
long cutime;
/* total user time of children */
long cstime;
/* total system time of children */

};

2、修改include/linux/syscall.h,添加“#include <linux/get_info.h>”,并在最后一行加入“asmlinkage long sys_get_info(struct str_info *info);”即系统调用为get_info。
3、修改arch/x86/inclue/asm/unistd_32.h,添加“#define __NR_get_info
337”,并将“#define NR_syscalls 337”改为“#define NR_syscalls 338”。
4、修改arch/x86/kernel/syscall_table_32.S,在最后一行添加“
.long sys_get_info”。
5、最后修改kernel/sys.c。添加“asmlinkage long sys_get_info(struct str_info *info){........}”,遍历进程的方法网上可以很方便找到,主要用的就是for_each_process()、list_for_each()、list_entry()等等。

二、编译内核
1、make,相当于make bzImage和make modules。
2、make modules_install。
3、make install。
4、在根目录下,mkinitramfs 2.6.xx.xx -o /boot/initrd.img-2.6.xx.xx-generic,生成xx文件

三、修改启动项
因为我用的虚拟机编译,而虚拟机不能嵌套,所以无法再ubuntu里再模拟运行改内核。直接修改/boot/grub/grub.cfg,用新内核启动。

四、测试系统调用
....
        struct get_info p;
p.pid = 1806;    //1806为系统调用号,是具体情况指定
syscall(337, &p);

        ......
虽然这个过程不是很复杂,但是对于没有编译过内核的人来说还是会经历不少曲折,但是无数次失败的汗水才能造就成功后的喜悦。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息