initial_kthread.rs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. //! 这个文件内放置初始内核线程的代码。
  2. use core::sync::atomic::{compiler_fence, Ordering};
  3. use alloc::{ffi::CString, string::ToString};
  4. use log::{debug, error};
  5. use system_error::SystemError;
  6. use crate::{
  7. arch::{interrupt::TrapFrame, process::arch_switch_to_user},
  8. driver::{net::e1000e::e1000e::e1000e_init, virtio::virtio::virtio_probe},
  9. filesystem::vfs::core::mount_root_fs,
  10. net::net_core::net_init,
  11. process::{
  12. exec::ProcInitInfo, kthread::KernelThreadMechanism, stdio::stdio_init, ProcessFlags,
  13. ProcessManager,
  14. },
  15. smp::smp_init,
  16. syscall::Syscall,
  17. };
  18. use super::{cmdline::kenrel_cmdline_param_manager, initcall::do_initcalls};
  19. const INIT_PROC_TRYLIST: [&str; 3] = ["/bin/dragonreach", "/bin/init", "/bin/sh"];
  20. pub fn initial_kernel_thread() -> i32 {
  21. kernel_init().unwrap_or_else(|err| {
  22. log::error!("Failed to initialize kernel: {:?}", err);
  23. panic!()
  24. });
  25. switch_to_user();
  26. }
  27. fn kernel_init() -> Result<(), SystemError> {
  28. KernelThreadMechanism::init_stage2();
  29. kenrel_init_freeable()?;
  30. #[cfg(target_arch = "x86_64")]
  31. crate::driver::disk::ahci::ahci_init()
  32. .inspect_err(|e| log::error!("ahci_init failed: {:?}", e))
  33. .ok();
  34. virtio_probe();
  35. mount_root_fs().expect("Failed to mount root fs");
  36. e1000e_init();
  37. net_init().unwrap_or_else(|err| {
  38. error!("Failed to initialize network: {:?}", err);
  39. });
  40. debug!("initial kernel thread done.");
  41. return Ok(());
  42. }
  43. #[inline(never)]
  44. fn kenrel_init_freeable() -> Result<(), SystemError> {
  45. do_initcalls().unwrap_or_else(|err| {
  46. panic!("Failed to initialize subsystems: {:?}", err);
  47. });
  48. stdio_init().expect("Failed to initialize stdio");
  49. smp_init();
  50. return Ok(());
  51. }
  52. /// 切换到用户态
  53. #[inline(never)]
  54. fn switch_to_user() -> ! {
  55. let current_pcb = ProcessManager::current_pcb();
  56. // 删除kthread的标志
  57. current_pcb.flags().remove(ProcessFlags::KTHREAD);
  58. current_pcb.worker_private().take();
  59. *current_pcb.sched_info().sched_policy.write_irqsave() = crate::sched::SchedPolicy::CFS;
  60. drop(current_pcb);
  61. let mut proc_init_info = ProcInitInfo::new("");
  62. proc_init_info.envs.push(CString::new("PATH=/").unwrap());
  63. proc_init_info.args = kenrel_cmdline_param_manager().init_proc_args();
  64. proc_init_info.envs = kenrel_cmdline_param_manager().init_proc_envs();
  65. let mut trap_frame = TrapFrame::new();
  66. if let Some(path) = kenrel_cmdline_param_manager().init_proc_path() {
  67. log::info!("Boot with specified init process: {:?}", path);
  68. try_to_run_init_process(
  69. path.as_c_str().to_str().unwrap(),
  70. &mut proc_init_info,
  71. &mut trap_frame,
  72. )
  73. .unwrap_or_else(|e| {
  74. panic!(
  75. "Failed to run specified init process: {:?}, err: {:?}",
  76. path, e
  77. )
  78. });
  79. } else {
  80. let mut ok = false;
  81. for path in INIT_PROC_TRYLIST.iter() {
  82. if try_to_run_init_process(path, &mut proc_init_info, &mut trap_frame).is_ok() {
  83. ok = true;
  84. break;
  85. }
  86. }
  87. if !ok {
  88. panic!("Failed to run init process: No working init found.");
  89. }
  90. }
  91. drop(proc_init_info);
  92. // 需要确保执行到这里之后,上面所有的资源都已经释放(比如arc之类的)
  93. compiler_fence(Ordering::SeqCst);
  94. unsafe { arch_switch_to_user(trap_frame) };
  95. }
  96. fn try_to_run_init_process(
  97. path: &str,
  98. proc_init_info: &mut ProcInitInfo,
  99. trap_frame: &mut TrapFrame,
  100. ) -> Result<(), SystemError> {
  101. proc_init_info.proc_name = CString::new(path).unwrap();
  102. proc_init_info.args.insert(0, CString::new(path).unwrap());
  103. if let Err(e) = run_init_process(proc_init_info, trap_frame) {
  104. if e != SystemError::ENOENT {
  105. error!(
  106. "Failed to run init process: {path} exists but couldn't execute it (error {:?})",
  107. e
  108. );
  109. }
  110. proc_init_info.args.remove(0);
  111. return Err(e);
  112. }
  113. Ok(())
  114. }
  115. fn run_init_process(
  116. proc_init_info: &ProcInitInfo,
  117. trap_frame: &mut TrapFrame,
  118. ) -> Result<(), SystemError> {
  119. compiler_fence(Ordering::SeqCst);
  120. let path = proc_init_info.proc_name.to_str().unwrap();
  121. Syscall::do_execve(
  122. path.to_string(),
  123. proc_init_info.args.clone(),
  124. proc_init_info.envs.clone(),
  125. trap_frame,
  126. )?;
  127. Ok(())
  128. }