摘要:本期重点和大家讨论系统调用机制。其中涉及到了一些及系统调用的性能、上下文深层问题,同时也穿插着讲述了一些内核调试方法。并且最后试验部分我们利用系统调用与相关内核服务完成了一个搜集系统调用序列的特定任务,该试验具有较强的实用和教学价值。
顾名思意,系统调用说的是操作系统提供给用户程序调用的一组“特殊”接口。用户程序可以通过这组“特殊”接口来获得操作系统内核提供的服务,比如用户可以通过文件系统相关的调用请求系统打开文件、关闭文件或读写文件,可以通过时钟相关的系统调用获得系统时间或设置系统时间等。
从逻辑上来说,系统调用可被看成是一个内核与用户空间程序交互的接口——它好比一个中间人,把用户进程的请求传达给内核,待内核把请求处理完毕后再将处理结果送回给用户空间。
系统服务之所以需要通过系统调用提供给用户空间的根本原因是为了对系统“保护”,因为我们知道Linux的运行空间分为内核空间与用户空间,它们各自运行在不同的级别中,逻辑上相互隔离。所以用户进程在通常情况下不允许访问内核数据,也无法使用内核函数,它们只能在用户空间操作用户数据,调用户用空间函数。比如我们熟悉的“hello world”程序(执行时)就是标准的户空间进程,它使用的打印函数printf就属于用户空间函数,打印的字符“hello word”字符串也属于用户空间数据。
但是很多情况下,用户进程需要获得系统服务(调用系统程序),这时就必须利用系统提供给用户的“特殊”接口——系统调用了,它的特殊性主要在于规定了用户进程进入内核的具体位置;换句话说用户访问内核的路径是事先规定好的,只能从规定位置进入内核,而不准许肆意跳入内核。有了这样的陷入内核的统一访问路径限制才能保证内核安全无虞。我们可以形象地描述这种机制:作为一个游客,你可以买票要求进入野生动物园,但你必须老老实实的坐在观光车上,按照规定的路线观光游览。当然,不准下车,因为那样太危险,不是让你丢掉小命,就是让你吓坏了野生动物。
对于现代操作系统,系统调用是一种内核与用户空间通讯的普遍手段,Linux系统也不例外。但是Linux系统的系统调用相比很多Unix和windows等系统具有一些独特之处,无处不体现出Linux的设计精髓——简洁和高效。
Linux系统调用很多地方继承了Unix的系统调用(但不是全部),但Linux相比传统Unix的系统调用做了很多扬弃,它省去了许多Unix系统冗余的系统调用,仅仅保留了最基本和最有用的系统调用,所以Linux全部系统调用只有250个左右(而有些操作系统系统调用多达1000个以上)。
这些系统调用按照功能逻辑大致可分为“进程控制”、“文件系统控制”、“系统控制”、“存管管理”、“网络管理”、“socket控制”、“用户管理”、“进程间通信”几类,详细情况可参阅文章系统调用列表
如果你想详细看看系统调用的说明,可以使用man 2 syscalls 命令查看,或干脆到 <内核源码目录>/include/asm-i386/unistd.h源文件种找到它们的原本。
熟练了解和掌握上面这些系统调用是对系统程序员的必备要求,但对于一个开发内核者或内核开发者来[1]说死记硬背下这些调用还远远不够。如果你仅仅知道存在的调用而不知道为什么它们会存在,或只知道如何使用调用而不知道这些调用在系统中的主要用途,那么你离驾驭系统还有不小距离。
要弥补这个鸿沟,第一,你必须明白系统调用在内核里的主要用途。虽然上面给出了数种分类,不过总的概括来讲系统调用主要在系统中的用途无非以下几类:
l 控制硬件——系统调用往往作为硬件资源和用户空间的抽象接口,比如读写文件时用到的write/read调用。
l 设置系统状态或读取内核数据——因为系统调用是用户空间和内核的唯一通讯手段[2],所以用户设置系统状态,比如开/关某项内核服务(设置某个内核变量),或读取内核数据都必须通过系统调用。比如getpgid、getpriority、setpriority、sethostname
l 进程管理——一系列调用接口是用来保证系统中进程能以多任务,在虚拟内存环境下得以运行。比如 fork、clone、execve、exit等
第二,什么服务应该存在于内核;或者说什么功能应该实现在内核而不是在用户空间。这个问题并不没有明确的答案,有些服务你可以选择在内核完成,也可以在用户空间完成。选择在内核完成通常基于以下考虑:
l 服务必须获得内核数据,比如一些服务必须获得中断或系统时间等内核数据。
l 从安全角度考虑,在内核中提供的服务相比用户空间提供的毫无疑问更安全,很难被非法访问到。
l 从效率考虑,在内核实现服务避免了和用户空间来回传递数据以及保护现场等步骤,因此效率往往要比实现在用户空间高许多。比如,httpd等服务。
l 如果内核和用户空间都需要使用该服务,那么最好实现在内核空间,比如随机数产生。
理解上述道理对掌握系统调用本质意义很大,希望网友们能从使用中多总结,多思考。
系统调用并非直接和程序员或系统管理员打交道,它仅仅是一个通过软中断机制(我们后面讲述)向内核提交请求,获取内核服务的接口。而在实际使用中程序员调用的多是用户编程接口——API,而管理员使用的则多是系统命令。
用户编程接口其实是一个函数定义,说明了如何获得一个给定的服务,比如read()、malloc()、free()、abs()等。它有可能和系统调用形式上一致,比如read()接口就和read系统调用对应,但这种对应并非一一对应,往往会出现几种不同的API内部用到统一个系统调用,比如malloc()、free()内部利用brk( )系统调用来扩大或缩小进程的堆;或一个API利用了好几个系统调用组合完成服务。更有些API甚至不需要任何系统调用——因为它不必需要内核服务,如计算整数绝对值的abs()接口。
另外要补充的是Linux的用户编程接口遵循了在Unix世界中最流行的应用编程界面标准——POSIX标准,这套标准定义了一系列API。在Linux中(Unix也如此)这些API主要是通过C库(libc)实现的,它除了定义的一些标准的C函数外,一个很重要的任务就是提供了一套封装例程(wrapper routine)将系统调用在用户空间包装后供用户编程使用。
不过封装并非必须的,如果你愿意直接调用,Linux内核也提供了一个syscall()函数来实现调用,我们看个例子来对比一下通过C库调用和直接调用的区别。
#include <syscall.h>
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int main(void) {
long ID1, ID2;
/*-----------------------------*/
/* 直接系统调用*/
/* SYS_getpid (func no. is 20) */
/*-----------------------------*/
ID1 = syscall(SYS_getpid);
printf ("syscall(SYS_getpid)=%ld/n", ID1);
/*-----------------------------*/
/* 使用"libc"封装的系统调用 */
/* SYS_getpid (Func No. is 20) */
/*-----------------------------*/
ID2 = getpid();
printf ("getpid()=%ld/n", ID2);
return(0);
}
系统命令相对编程接口更高了一层,它是内部引用API的可执行程序,比如我们常用的系统命令ls、hostname等。Linux的系统命令格式遵循系统V的传统,多数放在/bin和/sbin下(相关内容可看看shell等章节)。
有兴趣的话可以通过strace ls或strace hostname 命令查看一下它们用到的系统调用,你会发现诸如open、brk、fstat、ioctl 等系统调用被用在系统命令中。
下一个需要解释一下的问题是内核函数和系统调用的关系,内核函数大家不要想像的过于复杂,其实它们和普通函数很像,只不过在内核实现,因此要满足一些内核编程的要求[3]。系统调用是一层用户进入内核的接口,它本身并非内核函数,进入内核后,不同的系统调用会找到对应到各自的内核函数——换个专业说法就叫:系统调用服务服务例程。实际对请求服务的是内核函数而非调用接口。
比如系统调用 getpid实际就是调用内核函数sys_getpid。
asmlinkage long sys_getpid(void)
{
return current->tpid;
}
Linux系统种存在许多的内核函数,有些是内核文件种自己使用的,有些则是可以export出来供内核其他部分共同使用的,具体情况自己决定。
内核公开的内核函数——export出来的——可以使用命令ksyms 或 cat /proc/ksyms来查看。另外网上还有一本归纳分类内核函数的书叫作《The Linux Kernel API Book》,有兴趣的读者可以去看看。
总而言之,从用户角度向内核看,依次是系统命令、编程接口、系统调用和内核函数。再讲述了系统调用实现后,我们会回过头来看看整个执行路径。
Linux中实现系统调用利用了0x86体系结构中的软件中断[4]。软件中断和我们常说的中断(硬件中断)不同之处在于——它是通过软件指令触发而并非外设,也就是说又编程人员出发的一种异常,具体的讲就是调用int $0x80汇编指令,这条汇编指令将产生向量为128的编程异常。
之所以系统调用需要借助异常实现,是因为当用户态的进程调用一个系统调用时,CPU便被切换到内核态执行内核函数[5],而我们在i386体系结构部分已经讲述过了进入内核——进入高特权级别——必须经过系统的门机制,这里异常实际上就是通过系统门陷入内核(除了int 0x80外用户空间还可以通过int3——向量3、into——向量4 、bound——向量5等异常指令进入内核,而其他异常用户空间程序无法利用,都是由系统使用的)。
我们更详细的解释一下这个过程。int $0x80指令目的是产生一个编号为128的编程异常,这个编程异常对应的中断描述符表IDT中的第128项——也就是对应的系统门描述符。门描述符中含有一个预设的内核空间地址,它指向了系统调用处理程序:system_call()(别和系统调用服务程序混淆,这个程序在entry.S文件中用汇编语言编写)。
很显然所有的系统调用都会统一的转到这个地址,但Linux一共有2、3百个系统调用都从这里进入内核后又该如何派发它们到各自的服务程序去呢?别发昏,解决这个问题的方法非常简单:首先Linux为每个系统调用都进行了编号(0—NR_syscall),同时在内核中保存了一张系统调用表,该表中保存了系统调用编号和其对应的服务例程,因此在系统调入通过系统门陷入内核前,需要把系统调用号一并传入内核,在x86上,这个传递动作是通过在执行int0x80前把调用号装入eax寄存器实现的。这样系统调用处理程序一旦运行,就可以从eax中得到数据,然后再去系统调用表中寻找相应服务例程了。
除了需要传递系统调用号以外,许多系统调用还需要传递一些参数到内核,比如sys_write(unsigned int fd, const char * buf, size_t count)调用就需要传递文件描述符号fd和要写入的内容buf和写入字节数count等几个内容到内核。碰到这种情况,Linux会有6个寄存器使用来传递这些参数:eax (存放系统调用号)、 ebx、ecx、edx、esi及edi来存放这些额外的参数(以字母递增的顺序)。具体做法是在system_call( )中使用SAVE_ALL宏把这些寄存器的值保存在内核态堆栈中。
有始便有终,当服务例程结束时,system_call( ) 从eax获得系统调用的返回值,并把这个返回值存放在曾保存用户态 eax寄存器栈单元的那个位置上。然后跳转到ret_from_sys_call( ),终止系统调用处理程序的执行。
当进程恢复它在用户态的执行前,RESTORE_ALL宏会恢复用户进入内核前被保留到堆栈中的寄存器值。其中eax返回时会带回系统调用的返回码。(负数说明调用错误,0或正数说明正常完成)
我们可以通过分析一下getpid系统调用的真是过程来将上述概念具体化,分析getpid系统调用一个办法是查看entry.s中的代码细节,逐步跟踪源码来分析运行过程,另外就是可借助一些内核调试工具,动态跟踪运行路径。
假设我们的程序源文件名为getpid.c,内容是:
#include <syscall.h>
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int main(void) {
long ID;
ID = getpid();
printf ("getpid()=%ld/n", ID);
return(0);
}
将其编译成名为getpid的执行文件”gcc –o getpid <路径>/getpid.c”, 我们使用KDB来产看它进入内核后的执行路径。
l 激活KDB (按下pause键,当然你必须已经给内核打了KDB补丁);设置内核断点 “bp sys_getpid” ;退出kdb “go”;然后执行./getpid 。瞬间,进入内核调试状态,执行路径停止在断点sys_getpid处。
l 在KDB>提示符下,执行bt命令观察堆栈,发现调用的嵌套路径,可以看到在sys_getpid是在内核函数system_call中被嵌套调用的。
l 在KDB>提示符下,执行rd命令查看寄存器中的数值,可以看到eax中存放的getpid调用号——0x00000014(=20).
l 在KDB>提示符下,执行ssb(或ss)命令跟踪内核代码执行路径,可以发现sys_getpid执行后,会返回system_call函数,然后接者转入ret_from_sys_call例程。(再往后还有些和调度有关其他例程,我们这里不说了它们了。)
结合用户空间的执行路径,大致该程序可归结为一下几个步骤:
1 该程序调用libc库的封装函数getpid。该封装函数中将系统调用号_NR_getpid(第20个)压入EAX寄存器,
2 调用软中断 int 0x80 进入内核。
(以下进入内核态)
3 在内核中首先执行system_call,接着执行根据系统调用号在调用表中查找到对应的系统调用服务例程sys_getpid。
4.执行sys_getpid服务例程。
5.执行完毕后,转入ret_from_sys_call例程,系统调用中返回。
内核调试是一个很有趣的话题,方法多种多样,我个人认为比较好用的是UML(user mode linux+gdb)和 KDB 这两个工具。尤其KDB对于调试小规模内核模块或查看内核运行路径很有效,对于它的使用方法可以看看Linux 内核调试器内幕这片文章。
系统调用的内在过程并不复杂,我们不再多说了,下面这节我们主要就系统调用所涉及的一些重要问题作一些讨论和分析,希望这样能更有助了解系统调用的精髓。
系统调用虽说是要进入内核执行,但它并非一个纯粹意义上的内核例程。首先它是代表用户进程的,这点决定了虽然它会陷入内核执行,但是上下文仍然是处于进程上下文中,因此可以访问进程的许多信息(比如current结构——当前进程的控制结构),而且可以被其他进程抢占(在从系统调用返回时,由system_call函数判断是否该再调度),可以休眠,还可接收信号[6]等等。
所有这些特点都涉及到了进程调度的问题,我们这里不做深究,只要大家明白系统调用完成后,再回到或者说把控制权交回到发起调用的用户进程前,内核会有一次调度。如果发现有优先级别更高的进程或当前进程的时间片用完,那么就会选择高优先级的进程或重新选择进程运行。除了再调度需要考虑外,再就是内核需要检查是否有挂起的信号,如果发现当前进程有挂起的信号,那么还需要先返回用户空间处理信号处理例程(处于用户空间),然后再回到内核,重新返回用户空间,有些麻烦但这个反复过程是必须的。
系统调用需要从用户空间陷入内核空间,处理完后,又需要返回用户空间。其中除了系统调用服务例程的实际耗时外,陷入/返回过程和系统调用处理程序(查系统调用表、存储/恢复用户现场)也需要花销一些时间,这些时间加起来就是一个系统调用的响应速度。系统调用不比别的用户程序,它对性能要求很苛刻,因为它需要陷入内核执行,所以和其他内核程序一样要求代码简洁、执行迅速。幸好Linux具有令人难以置信的上下文切换速度,使得其进出内核都被优化得简洁高效;同时所有Linux系统调用处理程序和每个系统调用本身也都非常简洁。
绝大多数情况下,Linux系统调用性能是可以接受的,但是对于一些对性能要求非常高的应用来说,它们虽然希望利用系统调用的服务,但却希望加快相应速度,避免陷入/返回和系统调用处理程序带来的花销,因此采用由内核直接调用系统调用服务例程,最好的例子就HTTPD——它为了避免上述开销,从内核调用socket等系统调用服务例程。
系统调用是用户空间和内核空间交互的唯一手段,但是这并非时说要完成交互功能非要添加新系统调用不可。添加系统调用需要修改内核源代码、重新编译内核,因此如果想灵活的和内核交互信息,最好使用一下几种方法。
l 编写字符驱动程序
利用字符驱动程序可以完成和内核交互数据的功能。它最大的好处在于可以模块式加载,这样以来就避免了编译内核等手续,而且调用接口固定,容易操作。
l 使用proc 文件系统
利用proc文件系统修订系统状态是一种很常见的手段,比如通过修改proc文件系统下的系统参数配置文件(/proc/sys),我们可以直接在运行时动态更改内核参数;再如,通过下面这条指令:echo 1 > /proc/sys/net/ip_v4/ip_forward开启内核中控制IP转发的开关。类似的,还有许多内核选项可以直接通过proc文件系统进行查询和调整。
l 使用虚拟文件系统
有些内核开发者认为利用ioctl()系统调用(字符设备驱动接口)往往会似的系统调用意义不明确,而且难控制。而将信息放入到proc文件系统中会使信息组织混乱,因此也不赞成过多使用。他们建议实现一种孤立的虚拟文件系统来代替ioctl()和/proc,因为文件系统接口清楚,而且便于用户空间访问,同时利用虚拟文件系统使得利用脚本执行系统管理任务更家方便、有效。
我们希望收集Linux系统运行时系统调用被执行的信息,既实时获取系统调用日志。这些日志信息将能以可读形式实时的返回给用户空间,以便用户观察或做近一步的日志分析(如入侵检测等)。
所以简单的讲实验代码集需要完成以下几个基本功能:
第一:记录系统调用日志,将其写入缓冲区(内核中),以便用户读取;
第二:建立新的系统调用,以便将内核缓冲中的系统调用日志返回到用户空间。
第三:循环利用系统调用,以便能动态实时返回系统调用日志。
代码功能一节介绍中的基本功能对应程序代码集中的三个子程序。它们分别是syscall_auydit、Sys_audit和auditd。接下来我们介绍代码具体结构。
syscall_audit该程序是一个内核态的服务例程,该例程负责记录系统调用的运行日志。
记录系统调用日志的具体做法是在内核中修改系统调用处理程序system_call[7],在其中需要监控的每个调用(在我们例子钟222个系统调用都监控了,当然你也可以根据自己需求有选择的监控)执行完毕后都插入一个日志记录指令,该指令会转去调用内核服务函数syscall_audit来记录该次调用的信息[8]。
Syscall_audit内核服务例程会建立了一个内核缓冲区来存放被记录的函数。当搜集的数据量到达一定阀值时(比如设定为到达缓冲区总大小的%80,这样作可避免在丢失新调用),唤醒系统调用进程取回数据。否则继续搜集,这时系统调用程序会堵塞在一个等待队列上,直到被唤醒,也就是说如果缓冲区还没接近满时,系统调用会等待(被挂起)它被填充。
由于系统调用是在内核中被执行,因此记录其执行日志也应该在内核态收集,所以我们需要利用一个新的系统调用来完成将内核信息带回到用户空间——sys_audit就是我们新填加的系统调用,它功能非常简单,就是从缓冲区中取数据返回用户空间。
为了保证数据连续性,防止丢失。我们会建立一个内核缓冲区存放每刻搜集到的日志数据,并且当搜集的数据量到达一定阀值时(比如设定为到达缓冲区总大小的%80),系统调用进程就会被唤醒[9],以取回数据。否则在日志搜集时,系统调用程序会堵塞在等待队列上,直到被唤醒,也就是说如果缓冲区还没接近满时,系统调用会等待它被填充。
不用多说,我们需要一个用户空间服务进程来不断的调用audit系统调用,取回系统中搜集到的的调用日志信息。要知道,长时间的调用日志序列对于分析入侵或系统行为等才有价值。
除了上面介绍的内容外,我们还需要一些辅助性,但却很必要的工作,这些工作将帮助我们将上述代码灵活地机结成一体,完成需要的功能。
n 其一是修改entry.S汇编代码,该代码中含有系统调用表和系统调用入口代码system_call。我们首先需要在系统调用表中加入新的系统调用(名为sys_audit,223号。.long SYMBOL_NAME(sys_audit));下来在系统调用入口中加入跳转到日志记录服务例程中(跳转 “je auditsys”, 而auditsys代码段会真正调用系统调用记录例程syscall_audit);
n 其二是填加代码文件audit.c,该文件中包含syscall_audit与系统调用sys_audit两个函数体,我们这里只说包含函数体,而并非函数,是因为这里我们并不想把函数的实现在内核中写死,而是希望利用了函数指针,即做了两个钩子函数,来完成把具体函数实现放在模块中完成,以便能动态加载,方便调试(请见下一节介绍)。
u 其三是修改i386_ksyms.c文件,再最后加入
extern void (*my_audit)(int,int);
EXPORT_SYMBOL(my_audit);
extern int(*my_sysaudit)(unsigned char,unsigned char*,unsigned short,unsigned char);
EXPORT_SYMBOL(my_sysaudit);
,这样做是为了导出内核符号表,以便能模块代码中能挂接上以上函数指针。
n 其四是修改内核原代码目录下/kernel自目录下的Makefile文件,很简单,只需要在obj-y := 。。。。。最后加上audit.o,告诉编译内核是把audit.o编进去。
我们的日志收集例程与取日志系统调用这两个关键函数的实现是放在内核模块中实现。其中有些需要解释的地方:
1. 模块编程的必要原则,如初始化、注销等都应该实现,所不同的是我们在初始化与注销时会分别挂上或卸下[10]了两个钩子函数的实现。
2. 我们系统调用日志记录采用了一个结构体:syscall_buf,它含有诸如系统调用号——syscall、进程ID——pid、调用程序名——comm[COMM_SIZE]等字段,共52字节;我们的内核缓冲区为audit_buf,它是一个可容纳100个syscall_buf的数组。
3. 系统调用实现极简单,要做的仅仅是利用__copy_to_user[11]将内核缓冲中的日志数据取到用户空间。为了提高效率,在缓冲区未满时(未到%80的阀值时),系统调用会挂起等待wait_event_interruptible(buffer_wait, current_pos >= AUDIT_BUF_SIZE*8/10);相应地当缓冲区收集快满时,则唤醒系统调用继续收集日志wake_up_interruptible(&buffer_wait)。
4. 最后要补充说明一下,在auditd用户服务程序中调用我们新加的系统调用前必须利用宏_syscall4(int, audit, u8, type, u8 *, buf, u16, len, u8, reset)来“声明”该调用——展开成audit函数原形,以便进行格式转换和参数传递,否则系统不能识别。
取回数据 |
Audit系统调用 |
用户程序auditd |
系统调用服务例程sys_audit |
系统调用日志缓冲 |
Audit.o模块 |
my_sysaudit钩子函数 |
my_audit钩子函数 |
用户空间 |
内核空间 |
日志记录例程syscall_audit |
描述日志数据流向
|
描述系统调用关系
|
程序体系图 |
下面具体讲述一下如何添加这个调用。
1 修改entry.S ——在其中的添加audit调用,并且在system_call中加入搜集例程。(该函数位于<内核源代码>/arch/i386/kernel/下)
2 添加audit.c文件到<内核源代码>/arch/i386/kernel/下——该文件中定义了
sys_audit和syscall_audit 两个函数需要的钩子函数(my_audit和my_sysaudit),它们会在entry.S中被使用。
3 修改<内核源代码>/arch/i386/kernel/i386-kysms.c文件,在其中导出my_audit与my_sysaudit两个钩子函数。因为只有在内核符号表里导出,才可被其他内核函数使用,也就是说才能在模块中被挂上。
4 修改<内核源代码>/arch/i386/kernel/Makefile文件,将audit.c编译入内核。
到这可以重新编译内核了,新内核已经加入了检测点了。下一步是编写模块来实现系统调用与内核搜集服务例程的功能了。
1 编写名为audit的模块,其中除了加载、卸载模块函数以外主要实现了mod_sys_audit与mod_syscall_audit两个函数。它们会分别挂载到my_sysaudit和my_audit两个钩子上。
2 编译后将模块加载 insmod audit.o。(你可通过dmesg查看是加载信息)
3 修改/usr/include/asm/unistd.h ——在其中加入audit的系统调用号。这样用户空间才可找到audit系统调用了。
4 最后,我们写一个用户deamon程序,来循环调用audit系统调用,并把搜集到的信息打印到屏幕上。
完了。系统调用还有许多细节,请大家查看有关书记吧。不罗索了。再见。
相关代码请下载 auditexample.tar (实现于 2.4.18 内核)。
感谢SAL的开发者,例子程序基本框架来自于它们的灵感。
[1]我们说的开发内核者指开发系统内核,比如开发驱动模块机制、开发系统调用机制;而内核开发者则是指在内核基础之上进行的开发,比如驱动开发、系统调用开发、文件系统开发、网络通讯协议开发等。我们杂志所关注的问题主要在内核开发层次,即利用内核提供的机制进行开发。
[2]对Linux而言,系统调用是用户程序访问内核的唯一手段,无论是/proc方式或设备文件方式归根到底都是利用系统调用完成的。
[3]内核编程相比用户程序编程有一些特点,简单的讲内核程序一般不能引用C库函数(除非你自己实现了,比如内核实现了不少C库种的String操作函数);缺少内存保护措施;堆栈有限(因此调用嵌套不能过多);而且由于调度关系,必须考虑内核执行路径的连续性,不能有长睡眠等行为。
[4]软件中断虽然叫中断,但实际上属于异常(更准确说是陷阱)——CPU发出的中断——而且是由编程者触发的一种特殊异常。
[5]系统调用过程可被理解成——由内核在核心态代表应用程序执行任务。
[6]除了进程上下文外,Linux系统中还有另一种上下文——它被成为中断上下文。中断上下文不同于进程上下文,它代表中断执行,所以和进程是异步进行而且可以说毫不相干的。这种上下文中的程序,要避免睡眠因为无法被抢占。
[7]System_call是个通用的系统调用服务程序,或说系统调用入口程序,因为任何一个系统调用都要经过system_call统一处理(查找系统调用表,跳转到相应调用的服务例程),所以任何一次系统调用的信息都可被syscall_audit记录下来。
[8] 这里我们主要记录诸如调用时刻、调用者PID、程序名等信息,这些信息可从xtime或current这些全局变量处取得。
[9] 这里需要利用等待队列,具体声明见DECLARE_WAIT_QUEUE_HEAD(buffer_wait)。
[10] 所谓挂上或卸下其实就是将函数指针指向模块中实现的函数或指向空函数,但要知道这些函数指针一定是要导出到内核符号表中的,否则找不到。
[11] 这是一个系统提供的内核函数,目的就是从内核向用户空间传递数据。