kernel笔记——进程调度

news/2024/7/4 1:17:57

调度器完成以下任务:

 

  • 时钟中断(或类似的定时器)时间内刷新进程的时间片,设置进程调度标志
  • 系统调用返回或中断完成时检查调度标志

 

 

schedule函数

内核代码中完成进程调度的函数为schedule(),该函数中包含以下调用:

 

  1. put_prev_task(rq, prev);
  2. next = pick_next_task(rq);
  3. context_switch(rq, prev, next);

 

schedule首先将当前执行函数放入运行队列,然后选择下一个要运行的进程(怎么选择下一个进程,这部分就是调度算法的工作),最后完成进程上下文切换。

 

 

由A进程进入schedule函数,从函数出来将变成B进程,A进程调用schedule相当于放弃cpu资源,进程从R状态变成S等其他状态都会经过schedule。在vmcore中,我们可以看到S状态进程进入S状态前最后一个调用的函数为schedule:

 

  1. crash> ps | grep sshd
  2. 1300 1 0 ffff81024fba1810 IN 0.0 26088 1380 sshd
  3. crash> bt 1300
  4. PID: 1300 TASK: ffff81024fba1810 CPU: 0 COMMAND: "sshd"
  5. #0 [ffff81026a2e9cd8] schedule at ffffffff802d9025
  6. #1 [ffff81026a2e9da0] schedule_timeout at ffffffff802d9a6b
  7. #2 [ffff81026a2e9df0] do_select at ffffffff801935f2
  8. #3 [ffff81026a2e9ed0] sys_select at ffffffff80193880
  9. #4 [ffff81026a2e9f80] system_call at ffffffff8010ad3e
  10. ……

进程可以是“被调度”,而有的进程或线程主动调用schedule,放弃cpu,例如ksoftirqd内核线程。

 

进程优先级

早期的调度算法根据进程优先级计算进程运行的时间片、选择被调度的进程。对于普通进程(normal process)而言,nice表示其优先级,nice的取值范围为-20~19,nice值越大优先级越低;对于实时进程(real time process)而言,优先级的取值范围为0~99,同样值越大优先级越低。

 

内核代码中nice取值范围对应100~139,MAX_RT_PRIO宏定义了实时进程优先级与nice取值的分界点,实时进程的优先级比普通进程优先级高。

 

通过top、ps等命令我们可以查到进程的优先级,以下为top命令输出的下半部分,进程信息的查询结果:

 

  1.    PID USER   PR  NI    VIRT RES  SHR   S  %CPU %MEM      TIME+  COMMAND
  2.      22  root   20    0        0     0      0   R      1.9      0.0  60:23.34  ksoftirqd/9
  3.   8689  root   20    0  273m 67m  11m  S       0.0      0.4   2:04.01  java
  4. 11058  root   39   19       0     0      0   S       0.0      0.0   1:45.68  kipmi0
  5. 11771  root  -98    0 20388 19m 7256   S       0.0      0.1   0:16.06  had
  6.       3  root   RT    0        0     0      0   S       0.0      0.0   0:00.00  migration/0

 

其中(PR列值+100)指示进程优先级,以上输出中,java进程优先级为120,为普通进程;had进程优先级为2,为实时进程。RT表示该进程为实时进程,并且优先级为0。

 

可以使用renice命令设置普通进程的优先级:

 

  1. linux # renice 10 -p 21691
  2. 21691: old priority -20, new priority 10

 

 

调度算法

从最初的遍历O(n)算法,到2.5版kernel的O(1)算法,再到2.6版kernel的CFS(completely fair scheduler),调度算法被不断地改进。

 

O(1)调度算法使得调度器选择下一个执行进程的时间变为常量,不随可运行进程数量变化,但其有以下缺点:

 

  1. 动态调整进程优先级的算法很复杂,不易于代码维护
  2. 不“偏袒”交互型程序,假如响应鼠标移动的进程与某后台进程有同一优先级,在鼠标被移动时,另一进程可能更先获得调度,这使得鼠标移动响应慢或有重影

 

 

因而应用于服务器,O(1)调度算法表现尚可,在交互性更强的桌面应用方面就强差人意。

 

CFS解决了以上O(1)算法问题,它的核心数据结构为红黑树,该红黑树的key为进程的虚拟运行时间(vruntime)。每次调度时选择具有最小vruntime的进程执行,即红黑树最左边的结点相应的进程。

 

CFS调度算法中,交互型的进程将获得更大的权重、更多被执行的机会。假如现在有一个vruntime为1ms的交互型进程,另有一个vruntime为10ms的cpu消耗型进程,调度器连续调度交互型进程10次,再调度vruntime为10ms的cpu消耗型进程。Android中使用的调度方法亦为CFS。

 

之前的调度算法中,nice优先级决定进程的时间片大小,在CFS中,nice与进程权重关联,权重大的进程获得的运行时间就长。prio_to_weight数组定义了nice与权重的对应关系,该数组在中定义。

 

调度策略

kernel提供了5种调度策略,针对普通进程,有以下3种:

 

  1. SCHED_OTHER
  2. SCHED_BATCH
  3. SCHED_IDLE

 

这3种调度策略对应的进程,按CFS方式调度,优先级依次降低。

 

针对实时进程,有以下2种:

 

  1. SCHED_FIFO
  2. SCHED_RR

 

FIFO即先到先服务,具备该调度策略的进程将一直运行,直至其被I/O请求挂起或被具有更高优先级的实时进程抢占。

RR即轮叫调度(Round Robin),除设定了进程运行的最大时间片外,其他调度方式与FIFO相同。

 

因实时进程的优先级比普通进程的优先级都要高,具备以上调度策略的进程变为可运行状态时,都将抢占以CFS方式调度的普通进程。

 

CFS调度算法相关的代码在sched_fair.c中,实时进程相关的调度算法在sched_rt.c中定义。

 

与进程调度策略相关的函数调用有以下两个,分别用于获取、设置进程调度策略:

 

  1. sched_getscheduler
  2. sched_setscheduler

以下程序示例使用sched_setscheduler调用,将程序自身的优先级设置为最高的实时优先级:

  1. // sched_test.c
  2. #include
  3. #include
  4. #include
  5. int main()
  6. {
  7.     struct sched_param param;
  8.     param.sched_priority = sched_get_priority_max(SCHED_FIFO);
  9.     sched_setscheduler(getpid(), SCHED_FIFO, ?m);
  10.     printf("running...\n");
  11.     while(1);
  12.     return 0;
  13. }

 

执行程序之后,我们在top中可以看到

 

  1.     PID USER  PR  NI   VIRT   RES  SHR  S  %CPU  %MEM    TIME+    COMMAND
  2. 24484   root  RT    0   3704   400   316  R     100      0.0   1:32.51    sched_test

 

 

设置实时进程的优先级只能通过函数调用,不像设置nice值,可使用renice命令。

 

负载均衡

对于多核cpu,kernel的调度器还需要解决另一个问题:如何在多核间做到负载平衡。若一个核忙于跑程序,而其他核是空闲的,那么就失去了多核存在的意义。

 

对于对称多处理(symmetric multiprocessing,SMP)架构的cpu,同一物理cpu中,多核间可共享cache,物理cpu之间则不能。因而在同一物理cpu多核间移动进程,相比物理cpu间移动进程,开销要小。

 

migration内核线程完成平衡多核工作负载的工作,migration以以下方式拉起:

 

  1. 时钟中断处理函数timer_interrupt调用scheduler_tick函数
  2. 在scheduler_tick中,trigger_load_balance函数被调用
  3. 若满足调整cpu负载条件,trigger_load_balance通过raise_softirq调用产生一个软中断
  4. 后续软中断得到处理, migration线程被拉起,完成负载平衡工作

另外我们可以通过taskset命令设置某进程在某个特定cpu上执行,并且不参与负载平衡调度:

 

  1. linux # taskset -cp 0 ./sched_test

 

 

以上命令设置sched_test在0号cpu上运行,使用top我们可以看到执行的效果:

 

  1. linux # ps -elf | grep test | grep -v grep
  2. 4 R root 24502 24053 99 -40 - - 633 - 12:49 pts/7 00:00:15 ./sched_test
  3. linux # top -p 24502
  4. top - 12:50:19 up 5 days, 3:04, 7 users, load average: 1.55, 1.86, 2.16
  5. Tasks: 1 total, 1 running, 0 sleeping, 0 stopped, 0 zombie
  6. Cpu0 :100.0%us, 0.0%sy, 0.0%ni, 0.0%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
  7. Cpu1 : 0.0%us, 0.0%sy, 0.0%ni,100.0%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
  8. Cpu2 : 0.0%us, 0.0%sy, 0.0%ni,100.0%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
  9. Cpu3 : 0.0%us, 0.0%sy, 0.0%ni,100.0%id, 0.0%wa, 0.0%hi, 0.0%si, 0.0%st
  10. Mem: 23980M total, 375M used, 23604M free, 37M buffers
  11. Swap: 2055M total, 0M used, 2055M free, 221M cached
  12.   PID USER PR NI VIRT RES SHR S %CPU %MEM   TIME+ COMMAND
  13. 24502 root RT  0 2532 364 292 R  100  0.0 0:40.73 sched_test

 

Reference: Chapter 4 - Process Scheduling,  Linux kernel development.3rd.Edition

转载于:https://www.cnblogs.com/felixzh/p/9047730.html


http://www.niftyadmin.cn/n/4008818.html

相关文章

活力 J2ME 一

几个基本概念的学习。应用程序管理器(Java Application Manager,JAM),在规范中也称做Application Management Software(AMS)。这是一个用来执行J2ME应用程序的原生程序(Native,代表通…

瑞星安全随身WiFi:为用户WiFi上网安全保驾护航

近年来我国互联网的发展真的可以用“飞快”来形容,各类网站、技术层出不穷,市场规模呈几何式增长,与此同时,使用无线WiFi的网民也在不断增多。现在,几乎每个家庭中都有无线路由器,很多公共场所,…

Common Lisp学习笔记(七)

7 Applicative Programming 7.2 funcall7.3 mapcar7.4 manipulating tables with mapcar7.5 lambda expressions7.6 find-if7.7 my-assoc7.8 remove-if, remove-if-not7.9 reduce7.10 everydebug tool: trace7.11 operating on multiple lists7.12 function函数7.13 kwargs for…

市值能否全面反映行业影响力?

跌宕起伏的发展过后,CDN行业终于迎来了井喷时代。在2014亚太全媒体CDN峰会上,国家广电总局科技委副主任杜百川表示,2017年超过一半的互联网流量将由CDN提供,全球CDN的收入将达到50亿美金。在这个传统企业电商化、视频抢占移动终端…

[转]Mac 科研常用软件

转自:http://bbs.feng.com/read-htm-tid-7698336.html 我的 Mac 是 2012 年的 Pro Retina,现在主要用的是 Mac 系统,Windows 已经不常上了经常用的软件有以下几个: 1. 科学绘图软件:SciDAVis 和 Plot 目前这两个软件更…

C# 插入、删除Excel分页符

引言 对Excel表格设置分页对我们预览、打印文档时是很方便的,特别是一些包含很多复杂数据的、不规则的表格,为保证打印时每一页的排版美观性或者数据的前后连接的完整性,此时的分页符就发挥了极大的作用。因此,本文将介绍C#设置Ex…

认识UniJa技术

2004年11月1日,中国联通在其高速的 CDMA 网络上正式推出了基于Java技术的新的下载类增值服务-UniJa,为中国的CDMA 用户提供新颖、独特的 Java 应用服务。 中国联通已经建成世界上最大的CDMA移动通信网络,拥有2000多万CDMA用户。CDMA 1x技术的…

Robotium 测试多个activity

2019独角兽企业重金招聘Python工程师标准>>> How to test two activities with Robotium:http://stackoverflow.com/questions/16019011/how-to-test-two-activities-with-robotium //Click on add ident buttonsolo.clickOnButton("Tap to get ano…