linux reboot流程


在kernel根文件系統下做了實驗,在命令行輸入reboot,加log,跟蹤代碼。主要用於自己的學習總結。

SYSCALL_DEFINE4(reboot, int, magic1, int, magic2, unsigned int, cmd, void __user *, arg)
struct pid_namespace *pid_ns = task_active_pid_ns(current);
ret = reboot_pid_ns(pid_ns, cmd);
/*pid部分,先留個坑,在後面的文章和學習中再填上*/
kernel_restart(NULL); kernel_halt(); kernel_power_off(); ret = hibernate();
/*這裏有好幾個分支,我們主要看kernel_restart分支*/
kernel_restart(NULL)
kernel_restart_prepare(cmd);
blocking_notifier_call_chain(&reboot_notifier_list, SYS_RESTART, cmd);
/*調用之前註冊在notifier上的各個回調函數?
比如事先註冊的register_reboot_notifier(&xxx_wdt_reboot_notifier);
static struct notifier_block xxx_wdt_reboot_notifier = {
.notifier_call = xxx_wtd_reboot_notify,
};然後會調用到 xxx_wtd_reboot_notify這個函數*/
usermodehelper_disable(); /*不懂,又是一個坑*/
device_shutdown(); /*調用各個driver註冊的shutdown回調函數,比如會調用 到  .shutdown = stmmac_pltfr_shutdown,這個以太網驅動的函數,(如果系統使用的是這個以太網驅動)*/
migrate_to_reboot_cpu();
/*把當前運行的程序切換到boot cpu?*/
int cpu = reboot_cpu;
set_cpus_allowed_ptr(current, cpumask_of(cpu));
/*至於切換的細節,又是一個坑*/
syscore_shutdown();
/*調用通過register_syscore_ops(struct syscore_ops *ops)註冊的回調函數*/ 
/*關於syscore, 在系統suspend時,等到其他的的suspend回調函數調用完畢後,纔會調用syscore註冊的suspend回調函數,resume時,會最開始調用syscore註冊的resume回調函數,再調用其他的resume回調函數*/
kmsg_dump(KMSG_DUMP_RESTART);/*應該是把所有未打印緩存着的log全部打印出來*/
machine_restart(cmd);
local_irq_disable(); /*應該是關閉本cpu的irq*/
smp_send_stop(); /*關閉除了本cpu之外的其他cpu,細節又是一個坑*/

arm_pm_restart(reboot_mode, cmd);/*調用此函數重啓*/


arm_pm_restart函數在我們的項目中,是一個restart_driver_probe中賦值的函數,最終會調用到ASM指令,linux中嵌入了彙編,又一個坑。
 


















發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章