一. 综述: 系统调用为用户态的进程提供了一组与硬件设备进行交互的接口。
POSIX API不同于系统调用,其间不存在一一对应的关系,后者属于内核而前者属于库函数的范畴。
二. 系统调用:
1.系统调用的一般过程:
为了把系统调用号(位于寄存器eax中)与相应的服务例程联系起来,内核提供了一个系统调用派分表存储其对应关系。
2. 系统调用进入如退出的两种方式:
int $oX80----iret 汇编语言指令:最基本的,老版本Linux内核系统调用的唯一方式。
sysenter----sysexit 汇编语言指令: 从奔2开始新添加的,增加了一致性及安全性检查部分。
3. 系统调用的参数传递
eax寄存器进入时保存系统调用号,返回时保存返回码(大于等于0表示成功,小于0时为相应的出错码)。
其余参数是通过有用户态堆栈拷贝到CPU“寄存器(当参数个数小于等于6时),然后再经CPU寄存器拷贝到内核态堆栈来实现的。不直接从用户态向内核态进行拷贝是因为同时操作两个栈比较复杂且通过使用寄存器有能够使系统调用处理程序的结构与其他异常处理程序的结构类似。
若参数个数大于6个,则用一个单独的寄存器指向进程地址空间中参数所在的内存区,然由有封装例程处理(将其保存到栈中)。
参数的有效性验证:类型,权限的相关因素。
三. 内核封装例程:
内核线程可以通过内核封装例程来调用系统调用。系统中定义了7个内核封装例程,实际参数个数为:2+2*n。