idle.rs 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. use core::{
  2. intrinsics::unlikely,
  3. sync::atomic::{AtomicBool, Ordering},
  4. };
  5. use alloc::{sync::Arc, vec::Vec};
  6. use crate::{
  7. mm::{percpu::PerCpu, VirtAddr, IDLE_PROCESS_ADDRESS_SPACE},
  8. process::KernelStack,
  9. smp::{core::smp_get_processor_id, cpu::ProcessorId},
  10. };
  11. use super::{ProcessControlBlock, ProcessManager};
  12. static mut __IDLE_PCB: Option<Vec<Arc<ProcessControlBlock>>> = None;
  13. impl ProcessManager {
  14. /// 初始化每个核的idle进程
  15. pub fn init_idle() {
  16. static INIT_IDLE: AtomicBool = AtomicBool::new(false);
  17. if INIT_IDLE
  18. .compare_exchange(false, true, Ordering::SeqCst, Ordering::SeqCst)
  19. .is_err()
  20. {
  21. panic!("Idle process already initialized");
  22. }
  23. assert!(
  24. smp_get_processor_id() == ProcessorId::new(0),
  25. "Idle process must be initialized on the first processor"
  26. );
  27. let mut v: Vec<Arc<ProcessControlBlock>> = Vec::with_capacity(PerCpu::MAX_CPU_NUM as usize);
  28. for i in 0..PerCpu::MAX_CPU_NUM {
  29. let kstack = if unlikely(i == 0) {
  30. let stack_ptr =
  31. VirtAddr::new(Self::stack_ptr().data() & (!(KernelStack::ALIGN - 1)));
  32. // 初始化bsp的idle进程
  33. let mut ks = unsafe { KernelStack::from_existed(stack_ptr) }
  34. .expect("Failed to create kernel stack struct for BSP.");
  35. unsafe { ks.clear_pcb(true) };
  36. ks
  37. } else {
  38. KernelStack::new().unwrap_or_else(|e| {
  39. panic!("Failed to create kernel stack struct for AP {}: {:?}", i, e)
  40. })
  41. };
  42. let idle_pcb = ProcessControlBlock::new_idle(i, kstack);
  43. assert!(idle_pcb.basic().user_vm().is_none());
  44. unsafe {
  45. idle_pcb
  46. .basic_mut()
  47. .set_user_vm(Some(IDLE_PROCESS_ADDRESS_SPACE()))
  48. };
  49. assert!(idle_pcb.sched_info().on_cpu().is_none());
  50. idle_pcb.sched_info().set_on_cpu(Some(ProcessorId::new(i)));
  51. v.push(idle_pcb);
  52. }
  53. unsafe {
  54. __IDLE_PCB = Some(v);
  55. }
  56. }
  57. /// 获取当前的栈指针
  58. ///
  59. /// 请注意,该函数只是于辅助bsp核心的idle进程初始化
  60. fn stack_ptr() -> VirtAddr {
  61. #[cfg(target_arch = "x86_64")]
  62. return VirtAddr::new(x86::current::registers::rsp() as usize);
  63. #[cfg(target_arch = "riscv64")]
  64. {
  65. let stack_ptr: usize;
  66. unsafe {
  67. core::arch::asm!("mv {}, sp", out(reg) stack_ptr);
  68. }
  69. return VirtAddr::new(stack_ptr);
  70. }
  71. }
  72. /// 获取idle进程数组的引用
  73. pub fn idle_pcb() -> &'static Vec<Arc<ProcessControlBlock>> {
  74. unsafe { __IDLE_PCB.as_ref().unwrap() }
  75. }
  76. }