1. idle是什么

简单的说idle是一个进程,其pid号为 0。其前身是系统创建的第一个进程。也是唯一一个没有通过fork()产生的进程。

在smp系统中,每一个处理器单元有独立的一个执行队列,而每一个执行队列上又有一个idle进程,即有多少处理器单元。就有多少idle进程。

系统的空暇时间,事实上就是指idle进程的"执行时间"。既然是idle是进程。那我们来看看idle是怎样被创建,又详细做了哪些事情?

2. idle的创建  

我们知道系统是从BIOS加电自检,载入MBR中的引导程序(LILO/GRUB),再载入linux内核開始执行的,一直到指定shell開始执行告一段落,这时用户開始操作Linux。

而大致是在vmlinux的入口

startup_32(head.S)中为pid号为0的原始进程设置了运行环境,然后原是进程開始运行start_kernel()完毕Linux内核的初始化工作。

包含初始化页表,初始化中断向量表,初始化系统时间等。继而调用 fork(),创建第一个用户进程:  

kernel_thread(kernel_init, NULL, CLONE_FS | CLONE_SIGHAND); 这个进程就是著名的pid为1的init进程,它会继续完毕剩下的初始化工作,然后execve(/sbin/init), 成为系统中的其它全部进程的祖先。关于init我们这次先不研究,回过头来看pid=0的进程,在创建了init进程后,pid=0的进程调用 cpu_idle()演变成了idle进程。

current_thread_info()->status |= TS_POLLING;  

在 smp系统中。除了上面刚才我们讲的主处理器(运行初始化工作的处理器)上idle进程的创建,还有从处理器(被主处理器activate的处理器)上的idle进程,他们又是怎么创建的呢?接着看init进程,init在演变成/sbin/init之前,会运行一部分初始化工作,当中一个就是 smp_prepare_cpus(),初始化SMP处理器,在这过程中会在处理每一个从处理器时调用  

task = copy_process(CLONE_VM, 0, idle_regs(&regs), 0, NULL, NULL, 0);  

init_idle(task, cpu);  

即从init中复制出一个进程,并把它初始化为idle进程(pid仍然为0)。从处理器上的idle进程会进行一些Activate工作,然后运行cpu_idle()。   

整个过程简单的说就是,原始进程(pid=0)创建init进程(pid=1),然后演化成idle进程(pid=0)。init进程为每一个从处理器(执行队列)创建出一个idle进程(pid=0)。然后演化成/sbin/init。

3. idle的执行时机  

idle 进程优先级为MAX_PRIO,即最低优先级。

早先版本号中,idle是參

与调度的。所以将其优先级设为最低。当没有其它进程能够执行时,才会调度执行idle。而眼下的版本号中idle并不在执行队列中參与调度,而是在执行队列结构中含idle指针,指向idle进程,在调度器发现执行队列为空的时候执行,调入执行。

4. idle的workload   从上面的分析我们能够看出,idle在系统没有其它就绪的进程可执行的时候才会被调度。无论是主处理器。还是从处理器,最后都是执行的cpu_idle()函数。所以我们来看看cpu_idle做了什么事情。  由于idle进程中并不执行什么有意义的任务,所以通常考虑的是两点:1.节能,2.低退出延迟。

其核心代码例如以下:  void cpu_idle(void) {   int cpu = smp_processor_id();   current_thread_info()->status |= TS_POLLING;  /* endless idle loop with no priority at all */  while (1) {    tick_nohz_stop_sched_tick(1);   while (!need_resched()) {

check_pgt_cache();    rmb(); 

   if (rcu_pending(cpu))     rcu_check_callbacks(cpu, 0);    if (cpu_is_offline(cpu))     play_dead(); 

   local_irq_disable(); 

   __get_cpu_var(irq_stat).idle_timestamp = jiffies;    /* Don't trace irqs off for idle */    stop_critical_timings();    pm_idle(); 

   start_critical_timings();   } 

  tick_nohz_restart_sched_tick();   preempt_enable_no_resched();   schedule();   preempt_disable();  } }

循环推断need_resched以减少退出延迟,用idle()来节能。  默认的idle实现是hlt指令。hlt指令使CPU处于暂停状态,等待硬件中断发生的时候恢复,从而达到节能的目的。

即从处理器C0态变到C1态(见 ACPI标准)。这也是早些年windows平台上各种"处理器降温"工具的主要手段。当然idle也能够是在别的ACPI或者APM模块中定义的,甚至是自己定义的一个idle(比方说nop)。

小结:  

1.idle是一个进程,其pid为0。  

2.主处理器上的idle由原始进程(pid=0)演变而来。从处理器上的idle由init进程fork得到,可是它们的pid都为0。  

3.Idle进程为最低优先级。且不參与调度。仅仅是在执行队列为空的时候才被调度。  

4.Idle循环等待need_resched置位。默认使用hlt节能。

Linux内核的idle进程分析的更多相关文章

  1. Linux内核--网络栈实现分析(十一)--驱动程序层(下)

    本文分析基于Linux Kernel 1.2.13 原创作品,转载请标明http://blog.csdn.net/yming0221/article/details/7555870 更多请查看专栏,地 ...

  2. Linux内核态抢占机制分析(转)

    Linux内核态抢占机制分析  http://blog.sina.com.cn/s/blog_502c8cc401012pxj.html 摘 要]本文首先介绍非抢占式内核(Non-Preemptive ...

  3. Linux内核抢占实现机制分析【转】

    Linux内核抢占实现机制分析 转自:http://blog.chinaunix.net/uid-24227137-id-3050754.html [摘要]本文详解了Linux内核抢占实现机制.首先介 ...

  4. (转)Linux内核基数树应用分析

    Linux内核基数树应用分析 ——lvyilong316 基数树(Radix tree)可看做是以二进制位串为关键字的trie树,是一种多叉树结构,同时又类似多层索引表,每个中间节点包含指向多个节点的 ...

  5. Linux内核--网络栈实现分析(七)--数据包的传递过程(下)

    本文分析基于Linux Kernel 1.2.13 原创作品,转载请标明http://blog.csdn.net/yming0221/article/details/7545855 更多请查看专栏,地 ...

  6. 【转载】linux内核笔记之进程地址空间

    原文:linux内核笔记之进程地址空间 进程的地址空间由允许进程使用的全部线性地址组成,在32位系统中为0~3GB,每个进程看到的线性地址集合是不同的. 内核通过线性区的资源(数据结构)来表示线性地址 ...

  7. linux内核SPI总线驱动分析(一)(转)

    linux内核SPI总线驱动分析(一)(转) 下面有两个大的模块: 一个是SPI总线驱动的分析            (研究了具体实现的过程) 另一个是SPI总线驱动的编写(不用研究具体的实现过程) ...

  8. Linux内核--网络栈实现分析(二)--数据包的传递过程--转

    转载地址http://blog.csdn.net/yming0221/article/details/7492423 作者:闫明 本文分析基于Linux Kernel 1.2.13 注:标题中的”(上 ...

  9. linux内核中链表代码分析---list.h头文件分析(一)【转】

    转自:http://blog.chinaunix.net/uid-30254565-id-5637596.html linux内核中链表代码分析---list.h头文件分析(一) 16年2月27日17 ...

随机推荐

  1. [你必须知道的.NET]第二十四回:认识元数据和IL(上)

    发布日期:2009.02.24 作者:Anytao © 2009 Anytao.com ,Anytao原创作品,转贴请注明作者和出处. 说在,开篇之前 很早就有说说Metadata(元数据)和IL(中 ...

  2. python开发学习-day06(模块拾忆、面向对象)

    s12-20160130-day06 *:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: ...

  3. 七 使用list和tuple

    list Python内置的一种数据类型是列表:list.list是一种有序的集合,可以随时添加和删除其中的元素. 比如,列出班里所有同学的名字,就可以用一个list表示: >>> ...

  4. 十五oracle 触发器

    一.触发器简介 触发器的定义就是说某个条件成立的时候,触发器里面所定义的语句就会被自动的执行.因此触发器不需要人为的去调用,也不能调用.然后,触发器的触发条件其实在你定义的时候就已经设定好了.这里面需 ...

  5. Xcode代码提示里的字母含义

    P -- 协议 M -- 成员方法 C -- 类 K -- 枚举 .常量 V -- 成员变量 T -- typedef类型 G -- 全局变量 f -- 函数 # -- #define指令

  6. FastReport.Net使用:[38]关系的使用

    打印所有成绩 1. 数据源准备 接下来我们需要打印学生成绩,而成绩表中无姓名,我们通过建立Realtion关系来打印数据. 2. 创建Relation关系 在数据视图上的动作下拉菜单中选择“新建关系” ...

  7. 【BZOJ 2007】 2007: [Noi2010]海拔 (平面图转对偶图+spfa)

    2007: [Noi2010]海拔 Time Limit: 20 Sec  Memory Limit: 552 MBSubmit: 2504  Solved: 1195 Description YT市 ...

  8. 新浪微博API的使用Python

    本文记录了用新浪微博官方Python SDK调用API进行开发的流程. 准备工作 申请成为开发者并创建一个应用: 首先要有一个新浪微博的账号,然后去新浪微博开放平台(http://open.weibo ...

  9. iOS开发系列——内购、GameCenter、iCloud、Passbook系统服务开发汇总

    社交 Social 现在很多应用都内置“社交分享”功能,可以将看到的新闻.博客.广告等内容分享到微博.微信.QQ.空间等,其实从iOS6.0开始苹果官方就内置了Social.framework专门来实 ...

  10. 微信小程序导航栏,下面内容滑动,上册导航栏跟着滑动,内容随着导航栏滑动

    16.类似微信导航栏滑动.png 今日头条导航栏,下面滑动上面跟着滑动 index.wxml <swiper class="content" style="heig ...