一、用open()举例说明
在main函数中,使用open()函数,我们知道这个函数是系统调用,所以在main函数调用该函数的时候,会产生一个从用户空间到内核空间的转换,而此时的系统调用就是用户到内核的一个桥梁
内核向用户提供系统调用,会给每个系统调用一个编号,这个编号就是系统调用号(在用户空间),以后都不会改变。
系统调用表(在内核空间):内核中会把系统调用的入口地址存在里面。
现在说说从用户空间到内核空间是如何转换的
- main中使用open(),所以先去查找对应的系统调用号,查到以后存在exa寄存器中此时产生中断。
- 中断以后陷入内核,此时的main程序不能再继续执行,CPU从main程序中换下去(该过程中还存在一个现场保护,即把即存在中的信息记录下来,因为寄存器是共享的,所以要记录此时它里面的值),换到内核
- 这时内核上来工作(产生中断的时候从用户态到内核态,产生中断的时候也会产生一个中断号(0X80号),内核会根据这个中断号,系统调用处理中断),到内核工作的时候,它首先会从exa寄存器中读取系统调用号,然后在系统调用表中查找对应的系统调用,然后调用它,并将调用该系统调用后返回的文件描述符再写入exa寄存器中带回给main函数中的fd
- 得到返回的文件描述符之后,在刚刚产生中断的地方继续执行
二、讨论printf缓冲区
1、为什么有缓冲区的?
答:因为如果每次只写一个a,然后再写入b,再写入c,它的每一个写入都会经过下面这些步骤:
系统调用——>中断——>陷入内核——>去系统调用表中查找——>返回 每次都这样执行,消耗太大,效率也低,所以我们就把它们都先存在缓冲区中,一次性的只走一遍上述的步骤即可