目录

系统调度过程

用户空间角度:

内核角度

1、调用fork创建一个新进程

2、使用_fo_fork创建新进程

3、父进程调用wake_up_new_task尝试唤醒新进程

4、CPU选择一个合适的进程来运行;

5、运行新进程

6、实现负载均衡


系统调度过程

分析在命令行下使用./test程序。程序在用户态和内核台中运行的流程

用户空间角度:

shell 运行test程序,调用fork()系统调用函数来创建一个新进程;

调用exec系统调用函数来装载test程序。

ret = fork()
if(ret > 0)
{
    //父进程
}
else if(ret == 0)//子进程
{
    execve()
}
else
{
    //失败
}

内核角度

do_fork ->新进程

1、调用fork创建一个新进程

2、使用_fo_fork创建新进程

  • 创建新进程的task_struct数据结构
  • 复制父进程的task_struct数据结构到新进程
  • 复制父进程相关的页表项到新进程
  • 设置新进程的内核栈

3、父进程调用wake_up_new_task尝试唤醒新进程

  • 调用调度类select_task_rq(),为新进程寻找一个负载最轻的CPU(如果CPU0有四个线程在运行,CPU有一个线程在运行,那么新进程在寻址负载最轻的CPU为CPU1)
  • 调用调度类的enqueue_task(),把新进程添加到CPU1的就绪队列里

4、CPU选择一个合适的进程来运行;

  • 每次时钟节拍到来时,scheduler_tick()检查是否需要重新调度;check_preempt_tick会做检查,当需要重新调度时会设置当前进程的thread_info中的TIF_NEED_RESCHED标志位。
  • 在中断返回前会检查当前进程是否需要调度。如需要调度,调用preempt_schedule_irq来切换进程运行。
  • 调度器的schedule函数会调用调度类的pick_next_task来选择下一个最合适的进程
  • switch_mm切换父进程和新进程的页表
  • switch_to切换进程来运行 

5、运行新进程

  • 新进程第一次调用时会调用ret_from_fork函数
  • 返回用户空间运行shell程序
  • shell程序调用exec来运行test程序,最终新进程编程test进程

6、实现负载均衡

  • 在每个时钟节拍到来,检查是否需要触发软中断来实现SMP负载均衡,调用函数为scheduler_tick->trigger_load_balabce()。下一次实现负载均衡的时间点存放在就绪队列的next_balance成员里。
  • 触发 SCHED_SOFTIRQ软中断
  • 在软中断函数run_rebalance_domains里,从当前CPU开始遍历CPU拓扑关系,寻找负载不均匀的调度组;
  • 找到调度组最繁忙的,然后计算迁移多少到CPU1上才能保持两个调度组负载平衡;
  • 从CPU0迁移部分进程到CPU1