Linux内核分析之操作系统是如何工作的

在本周的课程中,孟老师主要讲解了操作系统是如何工作的,我根据自己的理解写了这篇博客,请各位小伙伴多多指正。

一、知识点总结

1. 三个法宝

存储程序计算机:所有计算机基础性的逻辑框架。

堆栈:高级语言的起点,函数调用需要堆栈机制。

中断机制:多道系统的基础,是计算机效率提升的关键。



2. 函数调用堆栈

堆栈是C语言程序运行时必须的一个记录调用路径和参数的空间,即CPU内已经集成好了很多功能。

堆栈含以下元素:

函数调用框架

传递参数

保存返回地址(%eax)

提供局部变量空间 等等

C语言编译器对堆栈的使用有一套的规则。



堆栈相关的寄存器:

esp,堆栈指针

ebp,基址指针



堆栈操作:

push,栈顶地址减少4个字节(32位)

pop,栈顶地址增加4个字节 注:ebp在C语言中用做记录当前函数调用基址。



其他关键寄存器:

cs:eip:指向地址连续的下一条指令

顺序执行:总是指向地址连续的下一条指令

跳转/分支:执行这样的指令的时候,cs:eip的值会根据程序需要被修改

call:将当前cs:eip的值压入栈顶,cs:eip指向被调用函数的入地址

ret:从栈顶弹出原来保存在这里的cs:eip的值,放入cs:eip中 发生中断时



二、Mykernel精简内核程序实验

1.实验过程

首先用下面两条命令

1 cd LinuxKernel/linux-3.9.4
2 qemu -kernel arch/x86/boot/bzImage

由图可知:每执行my_ start_ kernel函数一次或两次,my_ time_ hander函数就执行一次。

mymain.c ——系统中唯一的进程。mystartkernel之前的都是硬件初始化的工作,之后是整个操作系统的入口,开始执行操作系统。其中代码完成的工作是每循环10000次,打印一句话。

myinterrupt.c ——时间中断处理程序

每执行一次,都会执行一次时钟中断,每次时钟中断都调用printk并输出。



以上的实验环境仅仅是模拟了时钟中断,一直都是一个进程的run。需要加入下面的代码,才能开始进程切换调度。

需要在mykernel下重新更新文件:myinterrupt.c,主要负责中断以及进程切换;mymain.c,初始化系统环境,mypcb.h描述了进程控制块的定义。

再次到Linux3.9.4目录下重新编译,用下面命令:

1 make allnoconfig
2 make
3 qemu -kernel arch/x86/boot/bzImage

2.进程的启动和进程的切换机制

进程的启动:从void__init my_start_kernel(void)开始启动,函数前面是一些初始化工作,包括对于进程的fork。每个进程初始时,除了pid之外,对于所定义的pcb来说,其他都是相同的,以一个链表来组织各进程。嵌入式汇编代码的主要作用就是初始化第0号进程,将0号进程的esp写入esp寄存器,由于进程还未开始执行,因此进程的esp以及ebp都应该是指向栈底的(空栈),另外,0号进程的eip在这里已经指向了my_process代码段地址,ret之后,开始执行0号进程。

 1 void __init my_start_kernel(void)
 2 {
 3     int pid = 0;
 4     int i;
 5     /* Initialize process 0*/
 6     task[pid].pid = pid;
 7     task[pid].state = 0;/* -1 unrunnable, 0 runnable, >0 stopped */
 8     task[pid].task_entry = task[pid].thread.ip = (unsigned long)my_process;
 9     task[pid].thread.sp = (unsigned long)&task[pid].stack[KERNEL_STACK_SIZE-1];
10     task[pid].next = &task[pid];
11     /*fork more process */
12     for(i=1;i<MAX_TASK_NUM;i++)
13     {
14         memcpy(&task[i],&task[0],sizeof(tPCB));
15         task[i].pid = i;
16         task[i].state = -1;
17         task[i].thread.sp = (unsigned long)&task[i].stack[KERNEL_STACK_SIZE-1];
18         task[i].next = task[i-1].next;
19         task[i-1].next = &task[i];
20     }
21     /* start process 0 by task[0] */
22     pid = 0;
23     my_current_task = &task[pid];
24     asm volatile(
25         "movl %1,%%esp\n\t"     /* set task[pid].thread.sp to esp */
26         "pushl %1\n\t"          /* push ebp */
27         "pushl %0\n\t"          /* push task[pid].thread.ip */
28         "ret\n\t"               /* pop task[pid].thread.ip to eip */
29         "popl %%ebp\n\t"
30         :
31         : "c" (task[pid].thread.ip),"d" (task[pid].thread.sp)   /* input c or d mean %ecx/%edx*/
32     );
33 }     

进程的切换:通过schedule()函数进行进程切换,挡发生时钟中断的时候,中断处理程序会将需要调度的flag置为1,而与此同时,每一个当前执行的进程都在轮询,如果一旦flag被置为了1,则开始进行进程切换。case1: 进程开始切换的时候,是切换到一个进程状态为已经在运行的进程。在切换进程之前,将1f存到prev的eip中,1f指的是被标记为1的地址,ret之后,弹栈next的eip,开始执行next进程。

 1 if(next->state == 0)/* -1 unrunnable, 0 runnable, >0 stopped */
 2    {
 3     /* switch to next process */
 4     asm volatile(
 5         "pushl %%ebp\n\t"       /* save ebp */
 6         "movl %%esp,%0\n\t"     /* save esp */
 7         "movl %2,%%esp\n\t"     /* restore  esp */
 8         "movl $1f,%1\n\t"       /* save eip */
 9         "pushl %3\n\t"
10         "ret\n\t"               /* restore  eip */
11         "1:\t"                  /* next process start here */
12         "popl %%ebp\n\t"
13         : "=m" (prev->thread.sp),"=m" (prev->thread.ip)
14         : "m" (next->thread.sp),"m" (next->thread.ip)
15     );
16     my_current_task = next;
17     printk(KERN_NOTICE ">>>switch %d to %d<<<\n",prev->pid,next->pid);
18    }  

case2:进程切换到一个新进程。由于这里所切换的进程是一个新进程,所以这里的state置为0(running状态),这里的movl $1f,%1\n\t中的1f在进程再次调度回来的时候,走的是case1的分支代码块,所以case2中的 中没有标记为1的代码块,也没有pop ebp的操作。

 1 else
 2     {
 3         next->state = 0;
 4         my_current_task = next;
 5         printk(KERN_NOTICE ">>>switch %d to %d<<<\n",prev->pid,next->pid);
 6         /* switch to new process */
 7         asm volatile(
 8             "pushl %%ebp\n\t"       /* save ebp */
 9             "movl %%esp,%0\n\t"     /* save esp */
10             "movl %2,%%esp\n\t"     /* restore  esp */
11             "movl %2,%%ebp\n\t"     /* restore  ebp */
12             "movl $1f,%1\n\t"       /* save eip */
13             "pushl %3\n\t"
14             "ret\n\t"               /* restore  eip */
15             : "=m" (prev->thread.sp),"=m" (prev->thread.ip)
16             : "m" (next->thread.sp),"m" (next->thread.ip)
17         );
18     }   

三、对操作系统如何工作的理解

操作系统是管理计算机系统的全部硬件资源包括软件资源及数据资源;控制程序运行;改善人机界面;为其它应用软件提供支持等,使计算机系统所有资源最大限度地发挥作用,为用户提供方便有效的服务界面。linux内核从一个初始化上下文环境的函数开始执行,即start_kernel函数,创建很多进程或者fork若干进程,当中断发生的时候,如mykernel中就是时钟中断发生之后,接下来OS就会为各进程进行调度,在调度队列中选取出一个适合的进程,由CPU和内核堆栈保存前一个进程的各寄存器信息(进程描述块中的thread结构体保存大部分CPU寄存器,但是诸如eax、ebx等等这些通用寄存器是保存在内和堆栈中的),将eip指向要调度的进程执行的地址,开始执行。在本次课程中,我对于嵌入汇编的内容仍不熟练,所以为了更深入理解Linux需加强对汇编和操作系统的理解。

刘帅

原创作品转载请注明出处

《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000

时间: 2024-08-01 22:45:33

Linux内核分析之操作系统是如何工作的的相关文章

《Linux内核分析》 操作系统是如何工作的

范闻泽   原创作品转载请注明出处   <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 根据实验要求,在实验楼中的运行. 时间片轮转法 (输入vi main.c查看mykernel的源代码) mymain.c中最核心的代码 (myinterrupt.c的代码,每次时钟中断调用一次printk) 2.操作系统内核源代码[分析见注释] 1.mypcb.h//头文件,在其他的.c文件中引用:主要是数据结构的定义 1

linux内核分析作业:操作系统是如何工作的进行:完成一个简单的时间片轮转多道程序内核代码

计算机如何工作 三个法宝:存储程序计算机.函数调用堆栈.中断机制. 堆栈 函数调用框架 传递参数 保存返回地址 提供局部变量空间 堆栈相关的寄存器 Esp 堆栈指针  (stack pointer) Ebp 基址指针 (base pointer) 堆栈操作 Push:pop Ebp用作记录当前函数调用基址- 其他关键寄存器   中断 Call指令:1.将eip中下一条指令的地址A保存在栈顶:2.设置eip指向被调用程序代码开始处 1.Call xxx 2.进入xxx pushl %ebp mov

20135327郭皓——Linux内核分析第二周 操作系统是如何工作的

操作系统是如何工作的 上章重点回顾: 计算机是如何工作的?(总结)——三个法宝 存储程序计算机工作模型,计算机系统最最基础性的逻辑结构: 函数调用堆栈,高级语言得以运行的基础,只有机器语言和汇编语言的时候堆栈机制对于计算机来说并不那么重要,但有了高级语言及函数,堆栈成为了计算机的基础功能: enter pushl %ebp movl %esp,%ebp leave movl %ebp,%esp popl %ebp 函数参数传递机制和局部变量存储 中断,多道程序操作系统的基点,没有中断机制程序只能

LINUX内核分析第二周学习总结:操作系统是如何工作的?

马启扬 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.函数调用堆栈 1. 小结:计算机是怎样工作的 三个法宝:存储程序计算机.函数调用堆栈.中断机制. 存储程序计算机工作模型,计算机系统最最基础性的逻辑结构. 函数调用堆栈,高级语言得以运行的基础,只有机器语言和汇编语言的时候堆栈机制对于计算机来说并不那么重要,但有了高级语言及函数,堆栈成为了计算机的基础功能.(函数参数传递

Linux内核分析——操作系统是如何工作的

姓名:王晨光 学号:20133232 王晨光+原创作品转载请注明出处+<Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.进程的启动和切换源代码及分析 typedef struct PCB用于表示定义了一个进程,定义了进程管理相关的数据结构.同时也设置了用于保存EIP和ESP的数据类型. 下列代码为mymain.c部分代码: void __init my_start_kernel(void) { int pid

魏昊卿——《Linux内核分析》第二周作业:了解操作系统是怎样工作的

魏昊卿——<Linux内核分析>第二周作业:了解操作系统是怎样工作的 一.实验部分 使用实验楼的虚拟机打开shell cd LinuxKernel/linux-3.9.4 qemu -kernel arch/x86/boot/bzImage 然后cd mykernel 您可以看到qemu窗口输出的内容的代码mymain.c和myinterrupt.c 使用自己的Linux系统环境搭建过程参见mykernel,其中也可以找到一个简单的时间片轮转多道程序内核代码 mymain.c myinterr

Linux内核分析--操作系统是如何工作的

“平安的祝福 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 ” 一.初始化进程 操作系统内核启动入口函数是void __init my_start_kernel(void): 在这里简单定义进程的的两个cpu状态: struct Thread {    unsigned long        ip; //表示eip指令    unsigned long        sp;/

《Linux内核分析》 第二节 操作系统是如何工作的

黄胤凯   原创作品转载请注明出处   <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 根据实验要求,在实验楼中的运行.(也不知道是我的问题还是实验楼的问题,运行代码时久久出不来结果,所以部分截图出自教学视频中) (时间片轮转,可以看出是my start kernel的代码) (输入相应指令查看mykernel相关的源代码) (mymain.c的代码,其中头文件已忽略,之前都是硬件的载入,从这里开始是操作系统的

linux内核分析--计算机是如何工作的

知识点: 冯诺依曼体系结构结构,它最核心的思想是存储程序计算机. cpu通过总线与内存连接,依靠ip指针依次从内存中取出一条指令执行. API:程序员与计算机的接口界面. ABI:程序与CPU的接口界面. 寄存器.寻址方式和各类指令在学习<深入理解计算机>和<汇编>时已做了详细了解,不再赘述. 实验: 截图入下: 原代码为: int g(int x) { return x+3; } int f(int x) { return g(x); } a int main(void) { r