首页 > 代码库 > Linux内核设计基础(七)之系统调用
Linux内核设计基础(七)之系统调用
我理解的系统调用就是内核提供的一组用户进程与内核进行交互的接口。除异常和陷入外,系统调用是内核唯一的合法入口。像/proc也是通过系统调用进行访问的。
系统调用的意义:
- 让用户进程受限地访问硬件设备
- 为用户空间提供一种硬件的抽象借口
- 提供了创建新进程并与已有进程进行通信的机制
- 提供了申请操作系统其他资源的能力
- 保证系统稳定可靠,避免应用程序恣意妄为
系统调用的基本原理:
系统调用通常的入口是C库中定义的函数,也可以是自定义的函数(通过syscall进行调用)。每个系统调用被赋予一个系统调用号,通过这个独一无二的号就可以关联系统调用。如果一个系统调用被删除,它所占用的系统调用号也不允许被回收利用,否则,以前编译过的代码会调用这个系统调用,但事实上却调用另一个系统调用。内核中用sys_call_table记录所有已注册过的系统调用。
既然系统调用要从用户空间切换到内核态,那应用程序是如何通知内核的?软中断。通过引发一个异常来促使系统切换到内核态去执行异常处理程序,不过这里异常处理程序就是系统调用的处理程序。在x86上用int 0x80进行软中断的触发,执行第128号异常处理程序system_call()。
系统调用添加过程:
- 实现自定义的系统调用并编译进内核映像,可以放在kernel/sys.c文件中,也可以依据具体功能放在相关的文件中。实现格式如下:
asmlinkage long sys_mysyscall(void) { ... }
- 在系统调用表(entry.S)的最后加入一个表项,本例中为.long sys_mysyscall,其对应的系统调用号(338)为其在文件中的次序。
- 对于所支持的各种体系结构,系统调用号都必须定义于asm/unistd.h中——#define __NR_mysyscall 338。
从用户空间访问系统调用
拿系统调用open()来说,
我们可以借助C库,以
long open(const char *filename, int flags, int mode);
的形式调用此系统调用。也可以不靠库支持:
#define NR_open 5 _syscall3(long, open, const char*, filename, int, flags, int, mode);
用宏的方式,这样在我们的程序中,不用引入C头文件,直接使用open()即可。
常用系统调用
- exec
- fork
- open
- reboot
- getpid
- read
- write
- ioctl
与一般函数的区别
- 系统调用由操作系统核心提供,运行于内核状态,而库函数或自定义函数由用户调用,运行于用户态。
- 部分libc库函数的实现借助系统调用(如printf调用了write这样的系统调用),而另一些则不会使用系统调用(如strlen, strcat, memcpy等)。
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。