20135327郭皓——Linux内核分析第二周 操作系统是如何工作的
操作系统是如何工作的
上章重点回顾:
计算机是如何工作的?(总结)——三个法宝
存储程序计算机工作模型,计算机系统最最基础性的逻辑结构;
函数调用堆栈,高级语言得以运行的基础,只有机器语言和汇编语言的时候堆栈机制对于计算机来说并不那么重要,但有了高级语言及函数,堆栈成为了计算机的基础功能;
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内核分析第二周 操作系统是如何工作的的更多相关文章
- Linux内核分析第二周--操作系统是如何工作的
Linux内核分析第二周--操作系统是如何工作的 李雪琦 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course ...
- 20135327郭皓--Linux内核分析第九周 期中总结
Linux内核分析第九周 期中总结 一.知识概要 1. 计算机是如何工作的 存储程序计算机工作模型:冯诺依曼体系结构 X86汇编基础 会变一个简单的C程序分析其汇编指令执行过程 2. 操作系统是如何工 ...
- linux内核分析 第二周 操作系统是如何工作的
银雪纯 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.计算机是如何工作的 ...
- 20135327郭皓--Linux内核分析第三周 构造一个简单的Linux系统MenuOS
Linux内核分析第三周 构造一个简单的Linux系统MenuOS 前提回顾 1.计算机是如何工作的三个法宝 1.存储程序计算机 2.函数调用堆栈 3.中断 2.操作系统的两把宝剑 中断上下文的切换 ...
- 20135327郭皓--Linux内核分析第七周 可执行程序的装载
第七周 可执行程序的装载 郭皓 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-1000029000 ...
- 20135327郭皓--Linux内核分析第五周 扒开系统调用的三层皮(下)
Linux内核分析第五周 扒开系统调用的三层皮(下) 郭皓 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/U ...
- 20135327郭皓--Linux内核分析第四周 扒开系统调用的三层皮(上)
Linux内核分析第四周 扒开系统调用的三层皮(上) 郭皓 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/U ...
- 20135327郭皓--Linux内核分析第八周 进程的切换和系统的一般执行过程
第八周 进程的切换和系统的一般执行过程 一.进程切换的关键代码switch_to分析 1.进程调度与进程调度的时机分析 不同类型的进程有不同的调度需求 第一种分类: I/O-bound:频繁进行I/O ...
- 20135327郭皓--Linux内核分析第六周 进程的描述和进程的创建
进程的描述和进程的创建 一.进程的描述 操作系统三大功能: 进程管理 内存管理 文件系统 进程描述符task_struct数据结构 task _ struct:为了管理进程,内核必须对每个进程进行清晰 ...
随机推荐
- 关于MVC开发时,无法找到area的问题记录
解决方法: 检查area=admin 的dll是否生成,一般都是admin域生成dll导致
- 关于java中的使用通配符错误,错误信息Diamond types are not supported at language level '5‘
当时,我问了下大神,他们问我是不是jdk问题.因为jdk8才支持这样的棱形写法.当时自己的jdk版本是jdk8,然后就奇怪了,最后我发现原来在Language level中调成了5.0 5.0不支持6 ...
- linux 的常用命令---------第九阶段
Centos 7 系统启动及相关配置文件(面试题) 1. BIOS 初始化,开始post开机自检(主要检查磁盘.cpu.内存) 2. 加载 MBR 到内存 3. GRUB 阶段(可不说) 4. 加载内 ...
- iframe-metamask
iframe--require('iframe') higher level api for creating and removing iframes in browsers 用于创建或移除浏览器中 ...
- MyBatis实战之映射器
映射器是MyBatis最强大的工具,也是我们使用MyBatis时用得最多的工具,因此熟练掌握它十分必要.MyBatis是针对映射器构造的SQL构建的轻量级框架,并且通过配置生成对应的JavaBean返 ...
- Python中__init__()方法注意点
此文转自https://www.cnblogs.com/zyxstar2003/archive/2011/03/21/1989954.html 1.__init__并不相当于C#中的构造函数,执行它的 ...
- Linux下端口被占用确认
有时候关闭软件后,后台进程死掉,导致端口被占用.下面以JBoss端口8083被占用为例,列出详细解决过程. 解决方法: 1.查找被占用的端口 netstat -tln netstat -tln | g ...
- VBA读取、增加自定义和修改文档属性
读取系统文档属性 Sub read()On Error Resume Nextrw = 1Worksheets(1).ActivateFor Each p In ActiveWorkbook.Buil ...
- Python开发简单爬虫
简单爬虫框架: 爬虫调度器 -> URL管理器 -> 网页下载器(urllib2) -> 网页解析器(BeautifulSoup) -> 价值数据 Demo1: # codin ...
- Transaction Check Error:file /usr/libexec/getconf/default conflicts between attempted installs of gcc-6.4.1-1.fc25.i686 and gcc-6.4.1-1.fc25.x86_64
今天在我的ubuntu系统上使用yum来安装软件时出错了错误:Transaction Check Error:file /usr/libexec/getconf/default conflicts b ...