- 外壳函数执行一条中断机器指令(int 0x80),引发处理器从用户态切换到核心态,并执行系统中断0x80的中断矢量所指向的代码。(在2.6内核及glib 2.3.2之后的版本都支持sysenter指令,进入内核的速度更快)
- 为响应中断0x80,内核会调用system_call()例程(对于x86-32硬件平台,位于arch/i386/entry.S)来处理这次中断,以系统调用编号对存放所有调用服务例程的列表(内核变量sys_call_table)进行索引,发现并调用相应的系统调用服务例程。(在linux中,系统调用服务例程的命名通常会采取sys_xyz()的形式,其中,xyz()正是所论及的系统调用)
- 嵌入式系统C函数库:uClibc(www.uclibc.org)和dietlibc(www.fefe.de/dietlibc/)
- 确定系统glibc版本:/lib/libc.so.6
- 少数几个系统函数在调用时从不失败。例如,getpid()总能成功返回进程的ID,而_exit()总能终止进程。无需对此类系统调用的返回值进行检测。
- 系统调用返回值为-1表示出错。
- 系统调用失败时,将全局整形变量errno设置为一个正值,以标示具体的错误。程序应包含<errno.h>,该文件提供了对errno的声明,以及一组对各种错误编号而定义的常量。
- 如果调用系统调用和库函数成功,errno绝不会被重置为0,故此,该变量值不为0,可能是之前调用失败造成的。
- 少数系统调用(比如,getpriority())在调用成功后,也会返回-1。要判断此类系统调用是否发生错误,应在调用前将errno设置为0,并在调用之后对其值进行检测。
- #include<stdio.h>
void perror(const char *msg);//打印其msg参数所指向的字符串,紧跟一条与当前errno值对应的消息
- #include<string.h>
char * strerror(int errnum);//针对errnum参数中所给定的错号,返回相应的错误字符串。对于无法识别的错误编号,返回“Unknown errno nnn”
- 从错误处理的角度来说,可将库函数划分为几类:
(1)某些库函数返回错误信息的方式与系统调用完全相同,返回值-1,拌之以errno号来表示具体的错误。
(2)某些库函数在出错时返回-1之外的其他值,但仍会设置errno来表明具体的出错情况。例如,fopen()在出错时会返回一个NULL指针
(3)还有些函数根本不使用errno