操作系统是如何工作的

上章重点回顾:

计算机是如何工作的?(总结)——三个法宝

  • 存储程序计算机工作模型,计算机系统最最基础性的逻辑结构;

  • 函数调用堆栈,高级语言得以运行的基础,只有机器语言和汇编语言的时候堆栈机制对于计算机来说并不那么重要,但有了高级语言及函数,堆栈成为了计算机的基础功能;

    • enter

      • pushl %ebp

      • movl %esp,%ebp

    • leave

      • movl %ebp,%esp

      • popl %ebp

    • 函数参数传递机制和局部变量存储

  • 中断,多道程序操作系统的基点,没有中断机制程序只能从头一直运行结束才有可能开始运行其他程序。

本章操作系统是如何工作的主要针对的是函数调用堆栈。

一、函数调用堆栈

1.堆栈

2.堆栈寄存器和堆栈操作

3.函数的堆栈框架

4.函数堆栈框架的形成

5.关于一个小例子

p2函数的调用详解:

main中的局部变量:

6.C代码中嵌入汇编代码 内嵌式语法

二、一个简单的时间轮转多道程序实验

1.进入实验楼,进入指定文件夹,运行程序

2.打开mymain.c myinterrupt.c

3.对核心代码分析

 mymain.c

 void __init my_start_kernel(void) //操作系统的入口 启动操作系统
{
int i = ;
while()
{
i++;
if(i% == ) //每循环10万次打印一次
printk(KERN_NOTICE "my_start_kernel here %d \n",i); }
}
 myinterrupt.c

 #include <trace/events/timer.h>

 /*
* Called by timer interrupt.
*/
void my_timer_handler(void)//每一次中断都执行一次
{
printk(KERN_NOTICE "\n>>>>>>>>>>>>>>>>>my_timer_handler here<<<<<<<<<<<<<<<<<<\n\n");
}

时间轮转多道程序就是在这个函数的基础上,加入自己的要求,下面是多进程时间片轮转代码

mymain.c: 内核初始化和进程的启动

#include <linux/types.h>
#include <linux/string.h>
#include <linux/ctype.h>
#include <linux/tty.h>
#include <linux/vmalloc.h> #include "mypcb.h" tPCB task[MAX_TASK_NUM]; //声明一个task数组,tPCB结构体类型在mypcb.h中有定义
tPCB * my_current_task = NULL; //表示当前task指针
volatile int my_need_sched = ; //表示是否需要调度 void my_process(void); void __init my_start_kernel(void)
{
int pid = ; //初始进程号,修改方便
int i;
/* Initialize process 0*/ //0号进程的初始化过程
task[pid].pid = pid;
task[pid].state = ; //0表示可运行
task[pid].task_entry = task[pid].thread.ip = (unsigned long)my_process; //入口是my_process
task[pid].thread.sp = (unsigned long)&task[pid].stack[KERNEL_STACK_SIZE-]; //堆栈的栈顶位置
task[pid].next = &task[pid]; //下一个进程next还是指向自己(刚启动时没有其他进程) /*fork more process */ //创建更多进程
for(i=;i<MAX_TASK_NUM;i++)
{
memcpy(&task[i],&task[],sizeof(tPCB)); //复制0号进程的状态
task[i].pid = i; //创建的新进程号
task[i].state = -; //-1等待
task[i].thread.sp = (unsigned long)&task[i].stack[KERNEL_STACK_SIZE-]; //新进程的堆栈
task[i].next = task[i-].next; //指向下一个进程
task[i-].next = &task[i]; //新进程放在链表的尾部
} //进程的启动
/* start process 0 by task[0] */ //执行0号进程
pid = ; //初始进程号,修改方便
my_current_task = &task[pid];
asm volatile( //嵌入式汇编代码
"movl %1,%%esp\n\t" /* set task[pid].thread.sp to esp */ //将进程的sp赋给esp寄存器
"pushl %1\n\t" /* push ebp */ //当前的栈是空栈,ebp等于esp,所以push的%1是esp也就是ebp
"pushl %0\n\t" /* push task[pid].thread.ip */ //ip入栈
"ret\n\t" /* pop task[pid].thread.ip to eip */ //ip弹栈赋给eip,0号进程正式启动
"popl %%ebp\n\t"
:
: "c" (task[pid].thread.ip),"d" (task[pid].thread.sp) /* input c or d mean %ecx/%edx*/
);
} //内核初始化完成,0号进程开始执行 void my_process(void) //进程的起点
{
int i = ;
while()
{
i++;
if(i% == ) //执行10 000 000次才判断一次是否需要调度
{
printk(KERN_NOTICE "this is process %d -\n",my_current_task->pid);
if(my_need_sched == ) //1为需要调度,这是一个主动调度机制
{
my_need_sched = ;
my_schedule();
}
printk(KERN_NOTICE "this is process %d +\n",my_current_task->pid);
}
}
}
  • 首先是先创建第一个进程,定义其进程编号为0,状态为可执行,设置进程入口的地址为该进程的eip,设置进程的堆栈为该进程的堆栈,task[0].next=&task[0];因为此时系统之中仅仅只要一个进程,该进程的下一个进程就只能暂且指向自己了。
  • 接着是创建更多的进程,步骤是通过一个for循环进程依次创建进程的编号、状态和堆栈。将新创建的进程加入上一个进程的尾部。
  • 接下来就是要启动0号进程也就是第一个进程,将该进程的堆栈标记so存入寄存器esp中,紧接着将ebp压入栈中,再将该进程的eip压入栈中,接着将myprocess的eip弹出给eip,也就是启动0号进程。
  • 当运行完后则需要运行popl %ebp退回上一状态。

*1f指的是标号1所在的位置。

myinterrupt.c:进程的切换

 #include <linux/types.h>
#include <linux/string.h>
#include <linux/ctype.h>
#include <linux/tty.h>
#include <linux/vmalloc.h> #include "mypcb.h" tPCB task[MAX_TASK_NUM]; //声明一个task数组
tPCB * my_current_task = NULL; //表示当前task指针
volatile int my_need_sched = ; //表示是否需要调度 void my_process(void); void __init my_start_kernel(void)
{
int pid = ; //初始进程号,修改方便
int i;
/* Initialize process 0*/ //0号进程的初始化过程
task[pid].pid = pid;
task[pid].state = ; //0表示可运行
task[pid].task_entry = task[pid].thread.ip = (unsigned long)my_process; //入口是my_process
task[pid].thread.sp = (unsigned long)&task[pid].stack[KERNEL_STACK_SIZE-]; //堆栈的栈顶位置
task[pid].next = &task[pid]; //下一个进程next还是指向自己(刚启动时没有其他进程) /*fork more process */ //创建更多进程
for(i=;i<MAX_TASK_NUM;i++)
{
memcpy(&task[i],&task[],sizeof(tPCB)); //复制0号进程的状态
task[i].pid = i; //创建的新进程号
task[i].state = -; //-1等待
task[i].thread.sp = (unsigned long)&task[i].stack[KERNEL_STACK_SIZE-]; //新进程的堆栈
task[i].next = task[i-].next; //指向下一个进程
task[i-].next = &task[i]; //新进程放在链表的尾部
} //进程的启动
/* start process 0 by task[0] */ //执行0号进程
pid = ; //初始进程号,修改方便
my_current_task = &task[pid];
asm volatile( //嵌入式汇编代码
"movl %1,%%esp\n\t" /* set task[pid].thread.sp to esp */ //将进程的sp赋给esp寄存器
"pushl %1\n\t" /* push ebp */ //当前的栈是空栈,ebp等于esp,所以push的%1是esp也就是ebp
"pushl %0\n\t" /* push task[pid].thread.ip */ //ip入栈
"ret\n\t" /* pop task[pid].thread.ip to eip */ //ip弹栈赋给eip,0号进程正式启动
"popl %%ebp\n\t"
:
: "c" (task[pid].thread.ip),"d" (task[pid].thread.sp) /* input c or d mean %ecx/%edx*/
);
} //内核初始化完成,0号进程开始执行 void my_process(void) //进程的起点
{
int i = ;
while()
{
i++;
if(i% == ) //执行10 000 000次才判断一次是否需要调度
{
printk(KERN_NOTICE "this is process %d -\n",my_current_task->pid);
if(my_need_sched == ) //1为需要调度,这是一个主动调度机制
{
my_need_sched = ;
my_schedule();
}
printk(KERN_NOTICE "this is process %d +\n",my_current_task->pid);
}
}
}
  • 当前进程的下一个进程赋给next,当前进程赋给prev。如果下一个进程状态为0的话,即正在执行,则需要在两个正在运行的进程之间做进程上下文切换:

    • 保存当前进程的ebp
    • 将当前进程的esp赋给prev->thread.sp,从而将其保存起来
    • 将下一个进程的sp放置esp中
    • 保存当前进程的eip至当前进程所属的pcb中
    • 将下一个进程的eip保存至当前栈中
    • 通过ret,即可调用下一个进程
  • 若下一个进程的状态还未执行过,则进行以下步骤:
    • 将下一个进程设置为运行时状态,并将其设置为当前进程
    • 保存当前进程的ebp,将其压入栈中
    • 保存当前进程的esp至当前进程的pcb中
    • 由于该程序并未执行过,所以它的堆栈为空栈,空栈的创建则是通过将ebp和esp都赋予同一个值。
    • 保存当前进程的eip
    • 将当前进程的eip压入栈中
    • 通过ret,即可调用下一个进程

总结:

  本节讲的是操作系统是如何工作的,首先总结了一下计算机是如何工作的,计算机有三大法宝:存储程序计算机工作模型、函数调用堆栈和中断机制,而操作系统也是有两把利剑:中断上下文和进程上下文的切换。

  

郭皓原创作品转载请注明出处 《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000

20135327郭皓——Linux内核分析第二周 操作系统是如何工作的的更多相关文章

  1. Linux内核分析第二周--操作系统是如何工作的

    Linux内核分析第二周--操作系统是如何工作的 李雪琦 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course ...

  2. 20135327郭皓--Linux内核分析第九周 期中总结

    Linux内核分析第九周 期中总结 一.知识概要 1. 计算机是如何工作的 存储程序计算机工作模型:冯诺依曼体系结构 X86汇编基础 会变一个简单的C程序分析其汇编指令执行过程 2. 操作系统是如何工 ...

  3. linux内核分析 第二周 操作系统是如何工作的

    银雪纯 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.计算机是如何工作的 ...

  4. 20135327郭皓--Linux内核分析第三周 构造一个简单的Linux系统MenuOS

    Linux内核分析第三周  构造一个简单的Linux系统MenuOS 前提回顾 1.计算机是如何工作的三个法宝 1.存储程序计算机 2.函数调用堆栈 3.中断 2.操作系统的两把宝剑 中断上下文的切换 ...

  5. 20135327郭皓--Linux内核分析第七周 可执行程序的装载

    第七周 可执行程序的装载 郭皓 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-1000029000 ...

  6. 20135327郭皓--Linux内核分析第五周 扒开系统调用的三层皮(下)

    Linux内核分析第五周 扒开系统调用的三层皮(下) 郭皓 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/U ...

  7. 20135327郭皓--Linux内核分析第四周 扒开系统调用的三层皮(上)

    Linux内核分析第四周 扒开系统调用的三层皮(上) 郭皓 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/U ...

  8. 20135327郭皓--Linux内核分析第八周 进程的切换和系统的一般执行过程

    第八周 进程的切换和系统的一般执行过程 一.进程切换的关键代码switch_to分析 1.进程调度与进程调度的时机分析 不同类型的进程有不同的调度需求 第一种分类: I/O-bound:频繁进行I/O ...

  9. 20135327郭皓--Linux内核分析第六周 进程的描述和进程的创建

    进程的描述和进程的创建 一.进程的描述 操作系统三大功能: 进程管理 内存管理 文件系统 进程描述符task_struct数据结构 task _ struct:为了管理进程,内核必须对每个进程进行清晰 ...

随机推荐

  1. Test传送门(更新中)

    一.Codeforces传送门: Avito Code Challenge 2018 题解传送门 Codeforces Round #485 (Div. 2)     题解传送门 二.hihocode ...

  2. 关于$namespace$和重载运算符

    $namespace$ 还记得列队和天天爱跑步吗?记得当时写部分分写的非常艰难,一大原因就是部分分之间有很多重名的数组,而且大小还不一样大,经常写着写着就串了,而且$maxn$有一次提交时用错了直接全 ...

  3. Android与js交互拍照上传资料

    应用场景:h5通知android端拍照,选相册,然后将图片路径上传成功之后,获取到网络路径,将此路径返还给h5界面,并展示出来. android与js快速交互 效果图如下:   1.在Activity ...

  4. php可逆加密解密

    函数: function encrypt($data, $key) { $prep_code = serialize($data); $block = mcrypt_get_block_size('d ...

  5. linux问题解答

    1.Linux如何查询进程?杀死一个进程? ps命令提供进程的一次性查看(瞬时信息),结果不是动态的:top对命令实时监控 ps只是查看进程,而top还可以监视系统性能,如平均负载,cpu和内存的消耗 ...

  6. (转载)c++引用

    引用,顾名思义是某一个变量或对象的别名,对引用的操作与对其所绑定的变量或对象的操作完全等价 语法:类型 &引用名=目标变量名: 特别注意: 1.&不是求地址运算符,而是起标志作用 2. ...

  7. Java http协议概述

    一.http协议用于定义客户端与web服务端通讯的格式 二.HTTP1.0与HTTP1.1的区别 1.在HTTP1.0协议中,客户端与web服务器建立链接后只能获取一个web资源 2.HTTP1.1协 ...

  8. Android源代码下载以及异常查找网站推荐

    源代码下载:https://github.com/ 异常查找:http://stackoverflow.com/

  9. SystemView SEGGER FreeRTOS 移植和使用

    /* 官方帮助英文翻译文档参考:https://blog.csdn.net/bjr2016/article/category/7275877. */ /* 移植文档参考:https://blog.cs ...

  10. Dozer 使用小结

    这篇文章是本人在阅读Dozer官方文档(5.5.1版本,官网已经一年多没更新了)的过程中,整理下来我认为比较基础的应用场景. 本文中提到的例子应该能覆盖JavaBean映射的大部分场景,希望对你有所帮 ...