在Linux内核中添加系统调用,并编译内核
1 环境准备
运行系统:vmware下安装的ubuntu10.10 32bit桌面版。
编译内核版本: linux-2.6.32.63
内核目录: /home/wanchouchou/linuxKernel/linux-2.6.32.63
为了方便,使用chmod 777 -R linux-2.6.32.63/ 将所有的内核文件都改为全权限,这样就可以在非root状态修改文件进行编译了。
2 添加系统调用
首先进入linux-2.6.32.63目录下。以后的文件路径都是以此目录为“根目录”的。
2.1 在系统函数表中添加表项
进入arch/x86/kernel目录下,然后vim syscall_table_32.S,在此文件的最后一行添加自己的系统调用表项:
.long sys_rt_tgsigqueueinfo /* 335 */
.long sys_perf_event_open
.long sys_mycall //这是我们自己添加的表项
好了,下面开始添加系统调用号。
2.2 添加自己的系统调用号
现在进入目录 arch/x86/include/asm,该目录下有三个文件unistd_32.h, unistd_64.h, unistd.h。由于我们编译的是32位内核,所以需要在unistd_32.h中添加系统调用号。
vim unistd_32.h,在最后添加代码:
#define __NR_perf_event_open 336
#define __NR_mycall 337 //添加的
#ifdef __KERNEL__ /* 原本为337,但是由于我们添加了一行,所以改338*/
#define NR_syscalls 338
2.3 编写自己系统调用的实现函数
由于系统调用必须编译到核心的内核镜像中去。所以我们将此函数的实现写到kernel/sys_i386_32.c中:
//引入头文件
#include <asm/page.h>
........
//添加函数代码
asmlinkage long sys_mycall(void){
return THREAD_SIZE;
}
3 编译内核
现在就可以回到linux-2.6.32.63目录进行编译了。如果以前编译过内核,最好使用make mrproper和make clean命令来清除之前的编译残留文件。如果是第一次编译内核,那么就需要对内核的编译选项进行配置,这里推荐使用make menuconfig。如果键入该命令后提示缺少ncurses库文件的话,就使用sudo apt-get install libncurses5-dev命令安装该库。之后就可以进行menuconfig配置了,建议保持默认值即可。
一切准备就绪,键入make命令开始编译内核!现在可以泡一桶coffee来慢慢喝了,为什么是一桶?因为时间真的很长~~~
4 编译并安装内核模块
make modules ,这个相对来说要快一点。10分钟左右。然后安装内核模块: make modules_install 以及安装内核: make install
5 让新旧内核均可以加载
sudo mkinitramfs -o /book/initrd.img-2.6.32.63 //此命令会在/boot目录下生成initrd.img-2.6.32.63等文件
sudo update-initramfs -c -k 2.6.32.63 //根据/lib/modules/2.6.32.63文件进行更新。如果此命令出现FATAL: could not load /lib/modules/2.6.32.63的话,就将命令后面的2.6.32.63改为此目录下的新内核文件名即可(注意:旧内核文件名为2.6.35-22-generic)。
sudo update-grub2 //自动修改系统的引导配置文件,主要是更新/book/grup/grup.cfg启动文件。此命令执行完后会在grup.cfg中添加新内核的启动项
6 进入grup选择需要启动的内核
在ubuntu10.10默认grup菜单是不会显示的,也就是说无法选择启动的内核。要想显示的话,需要按照如下步骤进行操作:
1、把/etc/default/grub文件中的GRUB_HIDDEN_TIMEOUT=0改为大于0的数字,比如4;
2、把/etc/grub.d/30_os-prober文件中所有的set timeout=0中的0改为10.仔细找,不要漏过了;
3、上述修改完成后,执行命令更新grup2: sudo update-grub //更新/boot/grub/grub.cfg文件
7 重启
reboot。这时候在启动的时候就会停留在grup界面了,里面有4个选项,每个内核对应2个,选择进入2.6.32.63内核。启动完成后在终端输入: uname -a 显示的就是新内核版本了。
8 验证自定义的系统调用
现在我们可以编写代码来验证自定义的系统调用是否被加入到了内核中:
#include<sdtio.h>
#include<unistd.h>
#include<sys/syscall.h> #define SYS_mycall 337 //同我们前面定义的系统调用号相同 int main(){
long ret = ;
ret = syscall(SYS_mycall); //根据系统调用号调用对应的系统调用
printf("ThreadSize is: %ld\n", ret);
return ;
}
需要说明的是,在2.6.20以后的版本是没有以前的那些系统调用宏定义的(如_syscall0, _syscall1等),必须使用syscall来进行系统调用(可以通过man syscall获取参考信息)。同时由于是系统调用,所以需要root模式下编译、运行程序。程序运行结果: 8192。
注意,按理来说,我们是不需要自己定义SYS_mycall = 337的,但是,如果不这样的话编译的时候会提示找不到SYS_mycall,也就是说,它并没有被显示地加入到syscall.h文件中~解决办法如下:
1、修改/usr/include/asm/unistd_32.h文件,将最后的#define __NR_recvmmsg 337 改为 #define __NR_mycall 337 //貌似系统中根本就没有recvmmsg只有recvmsg~
2、修改/usr/include/bits/syscall.h文件,在文件开头位置加入代码: #define SYS_mycall __NR_mycall ;
这样,我们就可以直接在程序中使用SYS_mycall了。
在Linux内核中添加系统调用,并编译内核的更多相关文章
- 向linux内核中添加外部中断驱动模块
本文主要介绍外部中断驱动模块的编写,包括:1.linux模块的框架及混杂设备的注册.卸载.操作函数集.2.中断的申请及释放.3.等待队列的使用.4.工作队列的使用.5.定时器的使用.6.向linux内 ...
- linux内核学习之二:编译内核
在linux内核学习系列的第一课中讲述了搭建学习环境的过程(http://www.cnblogs.com/xiongyuanxiong/p/3523306.html),环境搭好后,马上就进入到下一环节 ...
- 42.Linux应用调试-初步制作系统调用(用户态->内核态)
1首先来讲讲应用程序如何实现系统调用(用户态->内核态)? 我们以应用程序的write()函数为例: 1)首先用户态的write()函数会进入glibc库,里面会将write()转换为swi(S ...
- Linux VFS中write系统调用实现原理【转】
转自:http://blog.chinaunix.net/uid-28362602-id-3425881.html 目录 用户空间的write函数在内核里面的服务例程为sys_write Vfs_wr ...
- 如何利用webmin在Linux主机中添加网站
Linux系统因其高效稳定而受到广大用户的推崇与青睐,然后其管理的复杂性也使很多用户望而却步,动弹不得.为了降低 Linux系统的管理难度,更有效方便的使用该系统,我司所有Linux主机或VPS系统均 ...
- 编译linux内核以及添加系统调用的全过程
参考链接: https://www.zybuluo.com/hakureisino/note/514321# 北京邮电大学操作系统实验: https://wenku.baidu.com/view/6d ...
- linux centos中添加删除修改环境变量,设置java环境变量
前言 安装完软件必要添加环境变量.指令很少,然而长时间不写就会不自信:我写的对吗?于是百度开始,于是发现又是各有千秋.好吧,好记星不如烂笔头.当然,最重要的是,百度出来的都他妈的是如何添加环境变量,只 ...
- 操作系统开发系列—12.f.在内核中添加中断处理 ●
因为CPU只有一个,同一时刻要么是客户进程在运行,要么是操作系统在运行,如果实现进程,需要一种控制权转换机制,这种机制便是中断. 要做的工作有两项:设置8259A和建立IDT. /*========= ...
- Linux程序设计中的curses.h编译报错,无法找到curses.h和ncurses.h
源程序screen.c如下: #include <stdio.h> #include <term.h> #include <curses.h> #include & ...
随机推荐
- (转)为什么国外 MMORPG 中不采用自动寻路等功能?
不只是自动寻路,现在网游中的教学引导系统,辅助系统的功能强大程度,友好程度都可以说到了变态的程度,开发这些功能投入的资源甚至要超过游戏内容本身.究其原因,还是竞争越来越激烈,人心越来越浮躁,游戏商家为 ...
- TA-LIB】之MACD
移动平滑异同平均线(Moving Average Convergence Divergence,简称MACD指标)策略.MACD是查拉尔·阿佩尔(Geral Appel)于1979年提出的,由一快及一 ...
- PHP continue和break的用法(深入理解)
对于刚入门的PHP童鞋们来说,在循环体中的continue和break的作用总是分不清到底是什么意思, 怎么用, 两者到底有什么区别? 接下来说几个例子,其实它们是很好区分的. <?php $t ...
- TCP/IP协议之http和https协议
一.TCP/IP协议 TCP/IP 是不同的通信协议的大集合. 1.TCP - 传输控制协议 TCP 用于从应用程序到网络的数据传输控制. TCP 负责在数据传送之前将它们分割为 IP 包,然后在它们 ...
- scrapy笔记2
cookies的使用: 使用 scrapy.http.cookie.CookieJar 类的extract_cookies方法,CookieJar._cookies就是我们需要的cookies,是一个 ...
- perl语言入门总结-第3章-列表与数组
1-列表list指的是标题的有序集合, 而数组(array)则是存储列表的变量. 更精确地说,列表指的是数据,而数组指的是变量. 访问数组中的元素 ] = "yabba"; ] = ...
- Android面试收集录14 Android进程间通信方式
一.使用 Intent Activity,Service,Receiver 都支持在 Intent 中传递 Bundle 数据,而 Bundle 实现了 Parcelable 接口,可以在不同的进程间 ...
- 第四模块:网络编程进阶&数据库开发 口述
进程即正在执行的一个过程.进程是对正在运行程序的一个抽象. 子进程死了之后 ,父进程关闭的时候要清理掉子进程的僵尸进程(收尸),孤儿进程是指父进程先死掉了的,交给init管理. join() 等待子进 ...
- 5. css定位 居中
1.准备工作 (1)添加背景图片 background: url('images/grass.png') (2)背景图片格式 background-size:contain; #完全限制在方框 #co ...
- 《Cracking the Coding Interview》——第17章:普通题——题目2
2014-04-28 22:05 题目:写个程序判断三连棋哪一方赢了. 解法:三个相同的棋子连成一条横线,竖线或者对角线就判断为赢了. 代码: // 17.2 Write an algorithm t ...