在美国的同学让我帮他完成一到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);
......
虽然这个过程不是很复杂,但是对于没有编译过内核的人来说还是会经历不少曲折,但是无数次失败的汗水才能造就成功后的喜悦。