Chinaunix首页 | 论坛 | 博客
  • 博客访问: 3115205
  • 博文数量: 685
  • 博客积分: 0
  • 博客等级: 民兵
  • 技术积分: 5303
  • 用 户 组: 普通用户
  • 注册时间: 2014-04-19 14:17
个人简介

文章分类

全部博文(685)

文章存档

2015年(116)

2014年(569)

分类: LINUX

2015-02-05 17:21:41

原文地址:http://blog.chinaunix.net/uid-24227137-id-3620045.html

  1. 调度器概述

由 于现在的计算机系统运行的任务的个数远远超过处理器核心的个数,因此导致了各任务在共享处理器、寄存器资源,为了实现处理器时间在各个任务之间公平的分 配,实现程序并行运行的假象,操作系统内核需要进程调度器来尽量公平的在各个进程之间分配运行时间。受到现实问题的影响,调度器实现变得很复杂:
  1. 需要在各进程间尽量公平的分配处理器时间
  2. 由于更重要的进程要比次重要的进程分配更多的处理器时间,因此需要时间优先级调度,差异化进程
  3. 进程的切换次数不能太频繁,否则导致处理器的效率降低,将时间消耗在进程切换上
  4. 两次进程切换的时间又不能太长,否则导致某些进程相应缓慢
上面的条件已经有各种矛盾了,是个难搞的活,下面看一下linux的内核调度器的框架:

linux通用的的调度器框架包括主调度器和周期性调度器,调度器类是实现了不同调度策略的实例,such as CFS、实时调度器,调度器类判断接下来要执行哪个进程。主调度器使用特定调度器类的选择进程,然后负责同底层CPU交互。下面来看看调度器相关的数据结构。

 2.  数据结构

先来看看熟悉的struct task_struct结构和调度器相关的成员:
  1. int on_rq;  
  2.   
  3. int prio, static_prio, normal_prio;  
  4. unsigned int rt_priority;  
  5. const struct sched_class *sched_class;  
  6. struct sched_entity se;  
  7. struct sched_rt_entity rt;  
  1. on_rq:表示进程是否在就绪队列上面
  2. prio,static_prio, normal_prio ,rt_priority和进程的优先级相关,prio,normal_prio是进程的动态优先级,由于内核有时候可能需要临时提高进程的优先级,因此增加了prio变量,such as 为了防止优先级反转rt_mutex提高当前持有锁进程的优先级就是设置prio变量然后引起重调度,normal_prio是用于调度器计算进程的 weight用的,在CFS中可以看到,normal_prio越高,weight越大,表示当前进程所占的权重较大,就可以获得跟多的处理器时间。 rt_priority是进程的实时优先级,在普通进程中没有用到。
  3. se 就是调度实体,是调度器作用的对象,因此task_struct中嵌入这个对象就可以被调度器调度啦
  4. sched_class 这个是调度器具体实现的接口,主要包含就绪队列的入队和出队操作(对于CFS来将就绪队列是红黑树),还有就是从就绪队列中选择下一个要执行的进程,周期性调度的底层操作,以及修改进程优先级的和内核抢占等操作。
具体看看struct sched_entity结构,这个包含了和调度器相关的重要成员
  1. struct sched_entity {  
  2.     struct load_weight  load;       /* for load-balancing */  
  3.     struct rb_node      run_node;  
  4.     struct list_head    group_node;  
  5.     unsigned int        on_rq;  
  6.   
  7.     u64         exec_start;  
  8.     u64         sum_exec_runtime;  
  9.     u64         vruntime;  
  10.     u64         prev_sum_exec_runtime;  
  11.   
  12.     u64         nr_migrations;  
  13.   
  14. #ifdef CONFIG_SCHEDSTATS  
  15.     struct sched_statistics statistics;  
  16. #endif  
  17.   
  18. #ifdef CONFIG_FAIR_GROUP_SCHED  
  19.     struct sched_entity *parent;  
  20.     /* rq on which this entity is (to be) queued: */  
  21.     struct cfs_rq       *cfs_rq;  
  22.     /* rq "owned" by this entity/group: */  
  23.     struct cfs_rq       *my_q;  
  24. #endif  
  25. };  
  1. load就是由进程优先级计算而来的表示进程权重的值
  2. run_node使得进程可以串在就绪队列上,CFS的就绪队列为一颗红黑树
  3. on_rq指示进程是否在就绪队列上,就绪队列上的进程表示进程可以运行,等待获得处理器时间,当进程被调度执行时进程从就绪队列上删除并且将on_rq设为0
  4. exec_start这是个动态更新的值,在进程被调度执行时更新为当前时间,表示此次调度开始执行的时间
  5. sum_exec_runtime表示总的在处理器上执行的时间,由于进程调度不能太频繁,内核保证每个进程都会执行一段时间才允许被抢占,sum_exec_runtime-prev_exec_runtime就表示此次调度执行的时间。
  6. vruntime是进程的在完全公平的优先级调度的情况下进程运行的时间,CFS调度中最重要的一个值了,CFS的就绪队列的黑红树的键值就是它了,每次调度就找vruntime最小的进程执行,应该是位于黑红树的最左边的进程。
  7. cfs_rq这个就是CFS的就绪队列,就绪队列是每一个处理器都有一个

3. 调度框架

先看看周期性调度,想想也可以差不多想出来周期性调度要干什么的,每一个 cpu的时钟周期都触发一次进程调度,由上面可以看到进程需要维护调度的当前时间,因此这个函数需要更新进程的当前调度时间,然后就是调用特定调度器类的 周期调度函数就ok啦,大体的框框应该是这样子,具体细节还有不少,看看代码:
  1. /*  
  2.  * This function gets called by the timer code, with HZ frequency.  
  3.  * We call it with interrupts disabled.  
  4.  */  
  5. void scheduler_tick(void)  
  6. {  
  7.     int cpu = smp_processor_id();  
  8.     struct rq *rq = cpu_rq(cpu);  
  9.     struct task_struct *curr = rq->curr;  
  10.   
  11.     sched_clock_tick();  /*处理硬件时钟的一些地方,和我们不相关*/  
  12.   
  13.     raw_spin_lock(&rq->lock);  
  14.     update_rq_clock(rq); /*更新就绪队列的时间*/  
  15.     update_cpu_load_active(rq);  
  16.     curr->sched_class->task_tick(rq, curr, 0); /* 调用调度器类的底层函数,这个函数会设置进程重调度标志TIF_NEED_RESCHED表示需要重新调度,然后内核会在适当的时机(比如系统调用结束重 返用户空间之前)调度,因此周期性调度并不执行真正的调度任务,只是设置一个重调度请求的标志而已*/  
  17.     raw_spin_unlock(&rq->lock);  
  18.   
  19.     perf_event_task_tick();  
  20.   
  21. #ifdef CONFIG_SMP  
  22.     rq->idle_balance = idle_cpu(cpu);  
  23.     trigger_load_balance(rq, cpu);  
  24. #endif  
  25. }  
周期性调度还是比较简单,没有涉及到处理真正的调度任务,下面看看主调器,也就是它来响应周期性调度器的TIF_NEED_RESCHED请求执行调度任务的。

在看主调度器之前先看看内核抢占,2.5版本的内核之前,在内核太运行的程序 是不能被抢占,只能等内核运行完成调度器才能调度其他的程序运行,这会造成很大的系统延时,在2.5中加入了内核抢占,在非重要的区域内核是可以被抢占, 每个进程都维护了一个抢占计数器,preempt_count,preempt_count为0时表示可以抢占,大于0是表示不能抢占,当需要禁止抢占的时候就调用inc_preempt_count将preempt_count加1。当前可以被抢占的时候且已经被抢占的时候将preempt_count加PREEMPT_ACTIVE表示这个进程是被内核抢占的,为了避免其他的inc_preempt_count调用影响此标志位,PREEMPT_ACTIVE = 0x1<<30
主调度器的任务就比较复杂了:
  1. /*  
  2.  * __schedule() is the main scheduler function.  
  3.  */  
  4. static void __sched __schedule(void)  
  5. {  
  6.     struct task_struct *prev, *next;  
  7.     unsigned long *switch_count;  
  8.     struct rq *rq;  
  9.     int cpu;  
  10.   
  11. need_resched:  
  12.     preempt_disable();  /*停止内核抢占,关键区域*/  
  13.     cpu = smp_processor_id();  
  14.     rq = cpu_rq(cpu);  /*获得当前cpu的就绪队列*/  
  15.     rcu_note_context_switch(cpu);  
  16.     prev = rq->curr;   /*在cpu上运行的当前进程,也就是准备被调度离开cpu的进程*/  
  17.   
  18.     schedule_debug(prev);  
  19.   
  20.     if (sched_feat(HRTICK))  
  21.         hrtick_clear(rq);  
  22.   
  23.     raw_spin_lock_irq(&rq->lock);  
  24.   
  25.     switch_count = &prev->nivcsw;  
  26.     if (prev->state && ! (preempt_count() & PREEMPT_ACTIVE)) {/*获取当前的进程内核抢占计数,如果设置 PREEMPT_ACTIVE表示被抢占,为了使得被抢占的进程可以快速恢复执行,不会执行下面的使进程停止活动的操作*/  
  27.         if (unlikely(signal_pending_state(prev->state, prev))) {  
  28.             prev->state = TASK_RUNNING;  
  29.         } else {  
  30.             deactivate_task(rq, prev, DEQUEUE_SLEEP); /*是进程停止活动*/  
  31.             prev->on_rq = 0;  
  32.   
  33.             /*  
  34.              * If a worker went to sleep, notify and ask workqueue  
  35.              * whether it wants to wake up a task to maintain  
  36.              * concurrency.  
  37.              *//*这个跟内核线程相关*/  
  38.             if (prev->flags & PF_WQ_WORKER) {  
  39.                 struct task_struct *to_wakeup;  
  40.   
  41.                 to_wakeup = wq_worker_sleeping(prev, cpu);  
  42.                 if (to_wakeup)  
  43.                     try_to_wake_up_local(to_wakeup);  
  44.             }  
  45.         }  
  46.         switch_count = &prev->nvcsw;  
  47.     }  
  48.   
  49.     pre_schedule(rq, prev);/*在CFS中无操作*/  
  50.   
  51.     if (unlikely(!rq->nr_running))  
  52.         idle_balance(cpu, rq);  
  53.   
  54.     put_prev_task(rq, prev); /*将让出处理器的进程加入到就绪队列中,并且统计就绪队列相关数据*/  
  55.     next = pick_next_task(rq); /*选择下一个要执行的进程,这两个操作的主体都是在具体的调度器类中实现,而不是在linux调度器框架中实现,因此实际实现将在CFS和实时调度中说明*/  
  56.     clear_tsk_need_resched(prev); /*由于调度已经完成,需要清除<span style="font-size: 18px;">TIF_NEED_RESCHED标志位*/span>  
  57.     rq->skip_clock_update = 0;  
  58.   
  59.     if (likely(prev != next)) {/*在调度器找不到需要运行的进程时才会相等*/  
  60.         rq->nr_switches++;  
  61.         rq->curr = next;  
  62.         ++*switch_count;  
  63.   
  64.         context_switch(rq, prev, next);/* unlocks the rq *//*处理底层的上下文切换的操作*/  
  65.         /*  
  66.          * The context switch have flipped the stack from under us  
  67.          * and restored the local variables which were saved when  
  68.          * this task called schedule() in the past. prev == current  
  69.          * is still correct, but it can be moved to another cpu/rq.  
  70.          */  
  71.         cpu = smp_processor_id(); /*由于切换了新的进程运行,新进程可能运行在不同的cpu上面*/  
  72.         rq = cpu_rq(cpu); /*同样的理由需要更新就绪队列*/  
  73.     } else  
  74.         raw_spin_unlock_irq(&rq->lock);  
  75.   
  76.     post_schedule(rq);  
  77.   
  78.     preempt_enable_no_resched();  
  79.     if (need_resched()) /*如果新的被切换上来的进程设置了<span style="font-size: 18px;">TIF_NEED_RESCHED标志,则又重新调度,这个是可能的,当这个进程设置了<span style="font-size: 18px;">TIF_NEED_RESCHED标志位之后被高优先级的进程抢占了就会发生这种情况*/span>span>  
  80.         goto need_resched;  
  81. }  
调度器的整体框架基本上就是这样了,这个调度器框架调度具体的调度类CFS和实时调度的相关操作调度进程。好晚了,睡觉。。。。

阅读(1108) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~