从内核态切换到用户态

1.用户态切换到内核态有三种方式

(1)系统调用
这是用户态进程主动要求切换到内核态的一种方式,用户态进程通过系统调用申请使用操作系统提供的服务程序完成工作,比如fork()实际上就是执行了一个创建新进程的系统调用。而系统调用的机制其核心还是使用了操作系统为用户特别开放的一个中断来实现,例如0x80中断。
(2) 异常
当CPU在执行运行在用户态下的程序时,发生了某些事先不可知的异常,这时会触发由当前运行进程切换到处理此异常的内核相关程序中,也就转到了内核态,比如缺页异常。
(3)外围设备的中断
当外围设备完成用户请求的操作后,会向CPU发出相应的中断信号,这时CPU会暂停执行下一条即将要执行的指令转而去执行与中断信号对应的处理程序,如果先前执行的指令是用户态下的程序,那么这个转换的过程自然也就发生了由用户态到内核态的切换。比如硬盘读写操作完成,系统会切换到硬盘读写的中断处理程序中执行后续操作等。

这3种方式是系统在运行时由用户态转到内核态的最主要方式,其中系统调用可以认为是用户进程主动发起的,异常和外围设备中断则是被动的。

2.以系统调用函数open为例:简单图示

3.系统调用如何从用户空间切换到内存空间

(1)用户态: ① 触发0x80中断

            ②保存当前的运行位置,状态,数据

              ③将系统调用号保存到eax寄存器中

             触发0x80中断 由系统调用函数触发的中断 。实际上在内核中,  触发0x80中断之后才会有上面的两步 。

(2)每一个中断号在内核中都有一个中断处理程序,中断处理程序是被内核调用来响应中断的

          通过该中断处理程序陷入内核

(3)在内核中:  ①有系统调用表,用寄存器中的系统调用号对应一个系统调用函数

                          ②通过函数得到执行该系统调用函数会得到一个返回值fd

                          ③将fd的值用eax寄存器带出  

 

在函数调用时int fd=open();open调用完之后才会将eax中的得到的返回值赋给fd,相当于是两个步骤

只有0x80中断才是系统调用中断,类似的还有缺页中断它们所对应的中断号是不同的

4.返回值:在c语言中如何返回返回值:如果返回值小于四个字节通过eax寄存器带出  若在4-8之间用eax和ecx带出  大于8通过生成临时对象或者变量带出

 

你可能感兴趣的:(Linux)