Linux系统调用概述与性能分析
1.什么是系统调用?
系统调用是操作系统提供给用户程序的一组特殊接口,用户程序可以利用这组接口来调用系统服务,例如磁盘I/O会用到的open,write,read等服务。
系统调用可以看作内核与用户空间程序交互的接口,把用户进程的请求传递给内核,待内核把请求处理完毕后再将处理结果送回给用户空间。
2.为什么需要系统调用?
1)将用户与底层硬件隔离,用户不需要了解底层硬件的具体控制方式,不需要面向具体的硬件编码,降低了开发的复杂度和难度。
2)极大地提高了系统的安全性,使用户程序和内核之间实现隔离,用户程序无法访问内核数据以及使用内核函数,访问内核的路径事先确定,只能从规定的位置进入内核,可以提高系统安全性。
3)提高系统的可移植性。
3.系统调用是如何实现的
系统调用在linux中是通过软件中断来实现的,通过软件指令来触发中断,linux内核响应中断后,进程从用户态切换到内核态,执行系统调用。
步骤如下:
1)在进程的内核态堆栈中保存大多数寄存器的内容(即保存恢复进程到用户态执行所需要的上下文)。
2)根据用户态传递的系统调用号,确定系统调用的执行程序。
3)调用相应的执行程序来处理系统调用
4)从系统调用返回
4.Linux中执行系统调用的具体方法
4.1使用glibc库函数实现系统调用
glibc库是linux下标准的c库,glibc实现操作系统提供的系统服务,即系统调用的封装。特点如下:
1)每个特定的系统调用对应了至少一个g’libc封装的库函数
2)多个api也可能只对应同一个系统调用
3)返回值-1在多数情况下表示内核不能满足进程的需求
4)glibc中定义的errno变量包含特定的出错码
4.2 libc库系统调用实例
1)示例代码chm.c
2)利用glibc更改权限前的test.txt
3)更改权限后
3)更改权限过程中发生的部分系统调用
4.3 Syscall实现系统调用
函数原型:
long int syscall(long int sysno,...)
参数列表说明:
sysno是系统调用号,唯一标识系统调用,详见sys/syscall.h。…代表系统调用所需要的参数,根据系统调用的不用,可带0~5个不等的参数。
返回值:
返回值为特定系统调用的返回值,调用成功后可将该返回值转化为特定的类型,如果系统调用失败则返回-1,错误代码存放在errno中。
4.4 Syscall系统调用实例
1)示例代码schm.c
2)更改前后test.txt权限对比
4.5 两种方法的对比
glibc函数库只需要知道api可以,无需知道其他的细节,具有较好的移植性。但是缺点就是部分系统调用没有被封装时无法使用;
syscall可以定制自己的系统调用,但是使用较为麻烦。
5.系统调用的开销
频繁使用系统调用会影响程序的执行效率。
主要原因在于用户态和内核态的切换需要消耗资源和时间,下面是切换的主要步骤:
1.保存用户进程的现场
2.合法性检查
3.参数传递
4.恢复现场
6.附录
权限说明:
S_IRWXU00700 权限,代表该文件所有者具有可读、可写及可执行的权限。
S_IRUSR 或S_IREAD,00400权限,代表该文件所有者具有可读取的权限。
S_IWUSR 或S_IWRITE,00200 权限,代表该文件所有者具有可写入的权限。
S_IXUSR 或S_IEXEC,00100 权限,代表该文件所有者具有可执行的权限。
S_IRWXG 00070权限,代表该文件用户组具有可读、可写及可执行的权限。
S_IRGRP 00040 权限,代表该文件用户组具有可读的权限。
S_IWGRP 00020权限,代表该文件用户组具有可写入的权限。
S_IXGRP 00010 权限,代表该文件用户组具有可执行的权限。
S_IRWXO 00007权限,代表其他用户具有可读、可写及可执行的权限。
S_IROTH 00004 权限,代表其他用户具有可读的权限
S_IWOTH 00002权限,代表其他用户具有可写入的权限。
S_IXOTH 00001 权限,代表其他用户具有可执行的权限。