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. Download failed : Oracle JDK 7 is NOT installed,解决oracle jdk7的问题

    先了解下概念: jdk(java development kit),就是java的开发工具集,顾名思义就是做开发用的,其中包括javac,也就是java compiler等.jre(java runt ...

  2. es6的Map()构造函数

    普通的object对象是键值对的集合,但对于它的键却有着严苛的要求,必须是字符串,这给我们平时带来很多的不方便 Map函数类似于对象,但它是一个更加完美的简直对集合,键可以是任意类型 set()方法可 ...

  3. 湖南大学ACM程序设计新生杯大赛(同步赛)C - Do you like Banana ?

    题目描述 Two endpoints of two line segments on a plane are given to determine whether the two segments a ...

  4. python笔记六:进程与线程

    1.进程 1)调用unix/linux系统中的进程函数fork(),用法和linux相同,调用成功返回0,失败返回-1: import os print 'Process (%s) start...' ...

  5. Maven的安装及修改为阿里云下载依赖

    使用JAVA工程管理越来越多的jar包,担心导错了,多导了,漏导了怎么办? 换一个IDE项目后项目会不会出一堆BUG,看的头皮发麻? 自己写的代码放在别人的机器上运行会不会出问题? Maven的强大毋 ...

  6. 【SQL】175. Combine Two Tables

    Table: Person +-------------+---------+ | Column Name | Type | +-------------+---------+ | PersonId ...

  7. FastReport.Net使用:[33]高亮显示

    1.首先来看下初始报表,很简单很普通. 2.下面对报表改进,90分以上的成绩以绿色显示,60~70分的以橙色斜体显示. 报表设计中选择数据成绩文本框,然后点击工具栏上的“ab突出显示”按钮打开“高亮显 ...

  8. 【BZOJ 2665】 2665: [cqoi2012]编号(暴力?)

    2665: [cqoi2012]编号 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 347  Solved: 217 Description 你需要给一 ...

  9. wpf企业级开发中的几种常见业务场景

    前阵子在公司弄个内部的进销存管理系统,从了解需求.系统设计到编码,大约耗费了两个月时间,后来公司有了其他的安排,这东西就算黄了.顺便吐槽一下,厂里的一些人说话真心不顾别人感受,邮件啥的没一句舒服的.不 ...

  10. 升级到php7和安装拓展(mac centos)

    Mac升级到php7 使用homebrew安装php7 brew update #更新源 brew search php #查找源中的php,发现有php7.1版本,安装最新的php7.1 brew ...