基于Linux Kernel 2.6 的源码, 分析其进程模型
文章目录
1. 简介
本文的内容是基于Linux Kernel 2.6 的源码, 深入分析其进程模型, 主要分析的内容如下:
- 什么是进程(进程的概念)
- 操作系统是如何组织进程的
- 进程状态之间是如何转换的
- 进程是如何进行调度的
- 对该操作系统进程模型的看法
2. 什么是进程
既然我们要分析操作系统的进程模型, 那么我们首先需要了解一下什么是进程。 首先, 我们先看下百度上对进程(Process)的定义:
进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,是操作系统结构的基础。在早期面向进程设计的计算机结构中,进程是程序的基本执行实体;在当代面向线程设计的计算机结构中,进程是线程的容器。程序是指令、数据及其组织形式的描述,进程是程序的实体。
接着我们看下维基百科上对进程(Process)的定义:
In computing, a process is an instance of a computer program that is being executed. It contains the program code and its current activity. Depending on the operating system (OS), a process may be made up of multiple threads of execution that execute instructions concurrently.
Google翻译一波, 内容如下:
在计算中,进程是正在执行的计算机程序的一个实例。它包含程序代码及其当前活动。根据操作系统(OS),一个进程可能由多个执行线程并发执行指令组成。
在对进程有了一些基础的了解之后, 我们便可以进行下一步的分析了。
3. 操作系统是如何组织进程的
在Linux系统中, 进程在/linux/include/linux/sched.h
头文件中被定义为task_struct
, 它是一个结构体, 一个它的实例化即为一个进程, task_struct
由许多元素构成, 下面列举一些重要的元素进行分析。
- 标识符:与进程相关的唯一标识符,用来区别正在执行的进程和其他进程。
- 状态:描述进程的状态,因为进程有挂起,阻塞,运行等好几个状态,所以都有个标识符来记录进程的执行状态。
- 优先级:如果有好几个进程正在执行,就涉及到进程被执行的先后顺序的问题,这和进程优先级这个标识符有关。
- 程序计数器:程序中即将被执行的下一条指令的地址。
- 内存指针:程序代码和进程相关数据的指针。
- 上下文数据:进程执行时处理器的寄存器中的数据。
- I/O状态信息:包括显示的I/O请求,分配给进程的I/O设备和被进程使用的文件列表等。
- 记账信息:包括处理器的时间总和,记账号等等。
3.1 进程状态(STATE)
在task_struct
结构体中, 定义进程的状态语句为
|
|
valatile
关键字的作用是确保本条指令不会因编译器的优化而省略, 且要求每次直接读值, 这样保证了对进程实时访问的稳定性。
进程在/linux/include/linux/sched.h
头文件中我们可以找到state
的可能取值如下
/*
* Task state bitmask. NOTE! These bits are also
* encoded in fs/proc/array.c: get_task_state().
* We have two separate sets of flags: task->state
* is about runnability, while task->exit_state are
* the task exiting. Confusing, but this way
* modifying one set can’t modify the other one by
* mistake. */ define TASK_RUNNING 0 define TASK_INTERRUPTIBLE 1 define TASK_UNINTERRUPTIBLE 2 define TASK_STOPPED 4 define TASK_TRACED 8
/* in tsk->exit_state */ define EXIT_ZOMBIE 16 define EXIT_DEAD 32
/* in tsk->state again */ define TASK_NONINTERACTIVE 64 define TASK_DEAD 128
根据state
后面的注释, 可以得到当state<0时,表示此进程是处于不可运行的状态, 当state=0时, 表示此进程正处于运行状态, 当state>0时, 表示此进程处于停止运行状态。
以下列举一些state的常用取值
状态 描述 0(TASK_RUNNING) 进程处于正在运行或者准备运行的状态中 1(TASK_INTERRUPTIBLE) 进程处于可中断睡眠状态, 可通过信号唤醒 2(TASK_UNINTERRUPTIBLE) 进程处于不可中断睡眠状态, 不可通过信号进行唤醒 4( TASK_STOPPED) 进程被停止执行 8( TASK_TRACED) 进程被监视 16( EXIT_ZOMBIE) 僵尸状态进程, 表示进程被终止, 但是其父程序还未获取其被终止的信息。 32(EXIT_DEAD) 进程死亡, 此状态为进程的最终状态
3.2 进程标识符(PID)
|
|
反应进程状态的信息,但不是运行状态,用于内核识别进程当前的状态,以备下一步操作
flags成员的可能取值如下,这些宏以PF(ProcessFlag)开头
|
|
3.4 进程之间的关系
|
|
在Linux系统中,所有进程之间都有着直接或间接地联系,每个进程都有其父进程,也可能有零个或多个子进程。拥有同一父进程的所有进程具有兄弟关系。
real_parent指向其父进程,如果创建它的父进程不再存在,则指向PID为1的init进程。 parent指向其父进程,当它终止时,必须向它的父进程发送信号。它的值通常与 real_parent相同。 children表示链表的头部,链表中的所有元素都是它的子进程(进程的子进程链表)。 sibling用于把当前进程插入到兄弟链表中(进程的兄弟链表)。 group_leader指向其所在进程组的领头进程。
3.5 进程调度
3.5.1 优先级
|
|
3.5.2 调度策略
|
|
policy
表示进程调度策略, 目前主要有以下五种策略
|
|
字段 描述 所在调度器类 SCHED_NORMAL (也叫SCHED_OTHER)用于普通进程,通过CFS调度器实现。SCHED_BATCH用于非交互的处理器消耗型进程。SCHED_IDLE是在系统负载很低时使用 CFS SCHED_FIFO 先入先出调度算法(实时调度策略),相同优先级的任务先到先服务,高优先级的任务可以抢占低优先级的任务 RT SCHED_RR 轮流调度算法(实时调度策略),后 者提供 Roound-Robin 语义,采用时间片,相同优先级的任务当用完时间片会被放到队列尾部,以保证公平性,同样,高优先级的任务可以抢占低优先级的任务。不同要求的实时任务可以根据需要用sched_setscheduler()API 设置策略 RT SCHED_BATCH SCHED_NORMAL普通进程策略的分化版本。采用分时策略,根据动态优先级(可用nice()API设置),分配 CPU 运算资源。注意:这类进程比上述两类实时进程优先级低,换言之,在有实时进程存在时,实时进程优先调度。但针对吞吐量优化 CFS SCHED_IDLE 优先级最低,在系统空闲时才跑这类进程(如利用闲散计算机资源跑地外文明搜索,蛋白质结构分析等任务,是此调度策略的适用者) CFS
3.6 进程的地址空间
进程都拥有自己的资源,这些资源指的就是进程的地址空间,每个进程都有着自己的地址空间,在task_struct中,有关进程地址空间的定义如下:
|
|
如果当前内核线程被调度之前运行的也是另外一个内核线程时候,那么其mm和avtive_mm都是NULL 以上即为操作系统是怎么组织进程的一些分析, 有了这些作为基础, 我们就可以进行下一步的分析了
4. 进程状态之间是如何转换的
关于linux进程状态(STATE)的定义, 取值以及描述都在进程状态中进行了详细的分析, 这里就不做过多的赘述。 下面给出进程的各种状态之间是如何进行互相转换的关系图:
(图片来源网上)
5.进程是如何进行调度的
5.1 与进程调度有关的数据结构
在了解进程是如何进行调度之前, 我们需要先了解一些与进程调度有关的数据结构。
###5.1.1 可运行队列(runqueue)
在/kernel/sched.c
文件下, 可运行队列被定义为struct rq
, 每一个CPU都会拥有一个struct rq
, 它主要被用来存储一些基本的用于调度的信息, 包括及时调度和CFS调度。在Linux kernel 2.6中, struct rq
是一个非常重要的数据结构, 接下来我们介绍一下它的部分重要字段:
|
|
###5.1.2 优先级数组(prio_array)
Linux kernel 2.6版本中, 在rq中多加了两个按优先级排序的数组active array
和expired array
。
这两个队列的结构是struct prio_array
, 它被定义在/kernel/sched.c
中, 其数据结构为:
|
|
Active array
表示的是CPU选择执行的运行进程队列, 在这个队列里的进程都有时间片剩余, *active
指针总是指向它。
Expired array
则是用来存放在Active array
中使用完时间片的进程, *expired指针总是指向它。
一旦在active array
里面的某一个普通进程的时间片使用完了, 调度器将重新计算该进程的时间片与优先级, 并将它从active array
中删除, 插入到expired array
中的相应的优先级队列中 。
当active array内的所有task都用完了时间片, 这时只需要将*active
与*expired
这两个指针交换下, 即可切换运行队列。
###5.1.3 调度器主函数(schedule())
schedule
函数存在/kernel/sched.c
中, 是Linux kernel很重要的一个函数, 它的作用是用来挑选出下一个应该执行的进程, 并且完成进程的切换工作, 是进程调度的主要执行者。
5.2 调度算法(O(1)算法)
5.2.1 介绍O(1)算法
何为O(1)算法: 该算法总能够在有限的时间内选出优先级最高的进程然后执行, 而不管系统中有多少个可运行的进程, 因此命名为O(1)算法。
5.2.2 O(1)算法的原理
在前面我们提到了两个按优先级排序的数组active array
和expired array
, 这两个数组是实现O(1)算法的关键所在。
O(1)调度算法每次都是选取在active array数组中且优先级最高的进程来运行。
那么该算法如何找到优先级最高的进程呢? 大家还记得前面prio_array
内的DECLARE_BITMAP(bitmap, MAX_PRIO+1);
字段吗?这里它就发挥出作用了(详情看代码注释), 这里只要找到bitmap
内哪一个位被设置为了1, 即可得到当前系统所运行的task的优先级(idx, 通过sehed_find_first_bit()方法实现), 接下来找到idx所对应的进程链表(queue), queue内的所有进程都是目前可运行的并且拥有最高优先级的进程, 接着依次执行这些进程,。
该过程定义在schedule
函数中, 主要代码如下:
|
|
6. 对该操作系统进程模型的看法
该算法模型解决了在Linux kernel 2.4下找寻优先级最高的进程时(时间复杂度为O(n)), 花费时间过多的问题, 并且能够在O(1)的时间复杂度内实现其功能, 可以说是已经非常的优秀了, 但是此算法仍然存在着一些不足之处, 比如当系统中交互式进程过多时( 如桌面系统),此时系统运行的就不太理想, 希望今后能够有更多的大佬们可以改善此算法, 精益求精。