QUOTE: |
Int $0x80 |
QUOTE: |
_syscallN(parameters) |
QUOTE: |
_syscall1( int, setuid, uid_t, uid ) |
syscallN( )宏指令的第1个参数int说明产生的函数的返回值的类型是整型,第2个参数setuid说明产生的函数的名称。后面是系统调用所需要的每个参数。这一宏指令后面还有两个参数uid_t和uid分别用来指定参数的类型和名称。
另外,用作系统调用的参数的数据类型有一个限制,它们的容量不能超过四个字节。这是因为执行int $0x80指令进行系统调用时,所有的参数值都存在32位的CPU寄存器中。使用CPU寄存器传递参数带来的另一个限制是可以传送给系统调用的参数的数目。这个限制是最多可以传递5个参数。所以Linux一共定义了6个不同的_syscallN()宏指令,从_syscall0()、_syscall1()直到_syscall5()。
一旦_syscallN()宏指令用特定系统调用的相应参数进行了扩展,得到的结果是一个与系统调用同名的函数,它可以在用户程序中执行这一系统调用。
2 添加新的系统调用
如果用户在Linux中添加新的系统调用,应该遵循几个步骤才能添加成功,下面几个步骤详细说明了添加系统调用的相关内容。
(1) 添加源代码
第一个任务是编写加到内核中的源程序,即将要加到一个内核文件中去的一个函数,该函数的名称应该是新的系统调用名称前面加上sys_标志。假设新加的系统调用为mycall(int number),在/usr/src/linux/kernel/sys.c文件中添加源代码,如下所示:
QUOTE: |
asmlinkage int sys_mycall(int number) { return number; } |
作为一个最简单的例子,我们新加的系统调用仅仅返回一个整型值。
(2) 连接新的系统调用
添加新的系统调用后,下一个任务是使Linux内核的其余部分知道该程序的存在。为了从已有的内核程序中增加到新的函数的连接,需要编辑两个文件。
在我们所用的Linux内核版本(RedHat 6.0,内核为2.2.5-15)中,第一个要修改的文件是:
QUOTE: |
/usr/src/linux/include/asm-i386/unistd.h |
该文件中包含了系统调用清单,用来给每个系统调用分配一个唯一的号码。文件中每一行的格式如下:
QUOTE: |
#define __NR_name NNN |
其中,name用系统调用名称代替,而NNN则是该系统调用对应的号码。应该将新的系统调用名称加到清单的最后,并给它分配号码序列中下一个可用的系统调用号。我们的系统调用如下:
QUOTE: |
#define __NR_mycall 191 |
系统调用号为191,之所以系统调用号是191,是因为Linux-2.2内核自身的系统调用号码已经用到190。
第二个要修改的文件是:
QUOTE: |
/usr/src/linux/arch/i386/kernel/entry.S |
该文件中有类似如下的清单:
QUOTE: |
.long SYMBOL_NAME() |
该清单用来对sys_call_table[]数组进行初始化。该数组包含指向内核中每个系统调用的指针。这样就在数组中增加了新的内核函数的指针。我们在清单最后添加一行:
QUOTE: |
.long SYMBOL_NAME(sys_mycall) |
(3) 重建新的Linux内核
为使新的系统调用生效,需要重建Linux的内核。这需要以超级用户身份登录。
QUOTE: |
#pwd /usr/src/linux # |
超级用户在当前工作目录(/usr/src/linux)下,才可以重建内核。
QUOTE: |
#make config #make dep #make clearn #make bzImage |
编译完毕后,系统生成一可用于安装的、压缩的内核映象文件:
QUOTE: |
/usr/src/linux/arch/i386/boot/bzImage |
(4) 用新的内核启动系统
要使用新的系统调用,需要用重建的新内核重新引导系统。为此,需要修改/etc/lilo.conf文件,在我们的系统中,该文件内容如下:
QUOTE: |
boot=/dev/hda map=/boot/map install=/boot/boot.b prompt timeout=50 image=/boot/vmlinuz-2.2.5-15 label=linux root=/dev/hdb1 read-only other=/dev/hda1 label=dos table=/dev/had |
首先编辑该文件,添加新的引导内核:
QUOTE: |
image=/boot/bzImage-new label=linux-new root=/dev/hdb1 read-only |
添加完毕,该文件内容如下所示:
QUOTE: |
boot=/dev/hda map=/boot/map install=/boot/boot.b prompt timeout=50 image=/boot/bzImage-new label=linux-new root=/dev/hdb1 read-only image=/boot/vmlinuz-2.2.5-15 label=linux root=/dev/hdb1 read-only other=/dev/hda1 label=dos table=/dev/hda |
这样,新的内核映象bzImage-new成为缺省的引导内核。为了使用新的lilo.conf配置文件,还应执行下面的命令:
QUOTE: |
#cp /usr/src/linux/arch/i386/boot/zImage /boot/bzImage-new |
其次配置lilo:
QUOTE: |
# /sbin/lilo |
QUOTE: |
/* xtdy.c */ #include _syscall1(int,mycall,int,ret) main() { printf("%d \n",mycall(100)); } |
QUOTE: |
# cc -o xtdy xtdy.c |
QUOTE: |
# xtdy |
QUOTE: |
# 100 |
上一页 [1] [2] [3]