第四章 进程调度
一、抢占与非抢占
1.非抢占式进程调度
进程会一直执行直到自己主动停止运行
2.抢占式进程调度
Linux/Unix使用的是抢占式的方式,强制的挂起进程的动作就叫做抢占。
二、进程优先级
1.进程的消耗类型
I/O消耗型进程
处理器耗费型
2.进程优先级
基于优先级的调度:优先极高的进程先运行;相同优先级的进程按照轮转方式进行调度
优先级分为两类:
nice值(从-20——+19):默认值为0;数值越大意味着优先级越低;可以通过 ps-el查看系统进程列表并找到NI标记列对应的优先级
实时优先级(从0——99):越高的实时优先级级数意味着进程优先级越高
三、Linux调度算法
1.调度器类
Linux调度器是以模块方式提供的(也就是调度器类),目的是允许不同类型的进程可以有针对性地选择调度算法
调度器类允许多种不同的可动态添加的调度算法并存,调度属于自己范畴的进程
调度器代码会按照优先级顺序遍历调度类,拥有一个可执行进程的最高优先级的调度器类胜出,去选择下面要执行的那一个程序。
2.Unix中的系统调度
将nice值映射到时间片的话,就必须将nice值对应到处理器的绝对时间;这样会导致进程切换无法最优进行。如果使用相对nice值,所带来的效果将会极大取决于其nice的初始值;如果执行nice值到时间片的映射,时间片极大受制于定时器。
3.公平调度
CFS基于一个简单的理念:进程调度的效果应当如同系统具备一个理想中的完美任务处理器。
CFS的做法如下:
允许每个进程运行一段时间、循环轮转、选择运行最少的进程作为下一个运行进程
nice值作为进程获得的处理器运行比的权重
每个进程都按照其权重在全部的可运行进程中所占的比例对应的“时间片”来运行
四、Linux调度的实现
1.时间记账
所有的调度器都必须对进程的运行时间做记账;
CFS使用调度器实体结构来追踪运行记账
2.虚拟实时
update_curr()函数实现了记账功能;计算了当前进程的执行时间并将其存放在data_exec中;然后将运行时间传递给了_update_curr(),由后者再根据当前可运行进程总数对运行时间进行计算,最终确定上述的权重值与当前运行进程的vrntime。
3.进程选择
CFS算法核心:选择具有最小vrntime的任务。
4.进程调度入口
进程调度的主要入口点是函数schedule(),定义在kernel/sched.c中;这正是内和其他部分用于调度进程调度器的入口。
五、抢占和上下文切换
1.上下文切换由定义在kernel/sched.c中的context_switch()函数负责,每当一个新的进程被选出来准备运行的时候,schedule()就会调用该函数:
调用switch_mm(),负责把虚拟内存从上一个进程映射切换到新的进程中
调用switch_to(),负责从上一个进程的处理器状态切换到新进程的处理器状态
2.Linux系统支持内核抢占
内核抢占发生在:
中断处理程序正在执行且返回内核空间之前
内核代码再一次具有可抢占性的时候
内核中的任务显式地调用schedule函数