嵌入式Linux關(guān)機(jī)時如何通知主板BIOS斷電
Linux電腦輸入poweroff退出操作系統(tǒng)后電源會自動切斷,而嵌入式Linux如果沒做特殊處理 輸入poweroff關(guān)閉系統(tǒng)后電源依舊保持著。敲擊鍵盤也不會有響應(yīng)。原因是CPU和主板之間有著行業(yè)標(biāo)準(zhǔn),比如ACPI(Advanced Configuration and Power Interface)、 APM(Advanced Power Management),都有相應(yīng)的硬件IO狀態(tài)指示。
當(dāng)CPU退出操作系統(tǒng)時會告訴BIOS:“保持內(nèi)存的電源,其他電源斷掉,下次喚醒我時記得提醒從硬盤里恢復(fù)?!盋PU掛起到內(nèi)存,晃動鼠標(biāo)、按壓鍵盤能在0.5秒恢復(fù)系統(tǒng)。
CPU告訴BIOS:“內(nèi)存電源也關(guān)閉,系統(tǒng)狀態(tài)(現(xiàn)場)已經(jīng)保存在硬盤?!彼追Q休眠模式,按鍵盤不能喚醒系統(tǒng),得按壓機(jī)箱開機(jī)鍵。
當(dāng)需要最徹底斷電時,CPU說:“全部斷電吧,你也歇歇?!边@下連同BIOS也斷電了,啟動需要按壓主機(jī)開機(jī)鍵,或者用內(nèi)資短接機(jī)箱電源針腳,主板才背重新供電。
CPU怎么說?用GPIO說。
為了實(shí)現(xiàn)嵌入式Linux在關(guān)閉GPIO通知BIOS,需要說明的是,由于沒有專用的電源管理GPIO的CPU,沒法做到CPU完全釋放資源后,CPU內(nèi)部硬件可以自動通知BIOS,軟件方式控制GPIO告知BIOS狀態(tài),有可能硬盤回寫還沒完成BIOS就把電源切斷。
這里的BIOS可以8051單片機(jī)負(fù)責(zé)。
源碼分析
首先在busybox上查詢poweroff是哪個系統(tǒng)調(diào)用,它向內(nèi)核傳遞一個宏。LINUX_REBOOT_CMD_POWER_OFF=0x4321FEDC
再在Linux源碼搜索響應(yīng)改宏命令位于kernel/reboot.c
繼續(xù)走讀kernel_power_off發(fā)現(xiàn)與平臺相關(guān)的接口machine_power_off。
既然是和架構(gòu)相關(guān)的,那代碼顯然應(yīng)該放在arch目錄下咯
x86架構(gòu)下的內(nèi)容,構(gòu)造與架構(gòu)相關(guān)的結(jié)構(gòu)體struct machine_ocf打印“ha ha”方便待會運(yùn)行QEMU調(diào)試觀察。
根據(jù)實(shí)際需求,把打印語句換成GPIO操作去通知8051單片機(jī)。
整個系統(tǒng)poweroff調(diào)用堆如下:
poweroff reboot LINUX_REBOOT_CMD_POWER_OFF kernel_power_off -> machine_power_off -> struct machine_ops.power_off do_exit
測試
構(gòu)建測試腳本 b.c 和 a.sh文件。我已經(jīng)向do_exit添加調(diào)試信息,打印被結(jié)束進(jìn)程的pid。
int main(){ printf("master pid %d\r\n", getpid()); if (!fork()) { printf("child pid %d\r\n", getpid()); } sleep(10); return 1;}
./b.out &./b.out &./b.out &./b.out &
系統(tǒng)啟動3個init進(jìn)程,記住他們的PID編號,待會會被釋放掉。
執(zhí)行a.sh派生出PID從58到64的進(jìn)程。
執(zhí)行poweroff,PID=67;
首先釋放掛載的文件系統(tǒng)umount PID=68;
關(guān)閉交換空間swapoff,PID=69;
結(jié)束3個init進(jìn)程和4個b.out進(jìn)程;
最后打印調(diào)試語句“ha ha ha”
shell進(jìn)程是運(yùn)行在init之前的,busybox分別發(fā)送SIGTERM和SIGKILL信號關(guān)閉它們。
使得斷電更恰到好處
注意到了嗎,控制GPIO是放在do_exit之前的,do_exit是不會返回調(diào)用線程的,在machine_power_off函數(shù)里控制GPIO,8051單片機(jī)延時一段時間才能關(guān)閉CPU電源,避免CPU資源未釋放完畢前被掉電。
建議單片機(jī)端添加CPU電流采樣功能,當(dāng)CPU資源釋放完畢后功耗可能會有所降低,單片機(jī)根據(jù)電流+GPIO雙重保險來判斷或許根可靠。
之前我曾今想過重寫do_exit,僅在結(jié)束最后一個進(jìn)程時作GPIO,實(shí)踐起來也不可靠。原因有2:
1、poweroff有一個操作是強(qiáng)制關(guān)機(jī),它不會調(diào)用do_exit,文稿末尾我上貼圖;
2、如果某應(yīng)用程序在內(nèi)核空間死鎖,將永遠(yuǎn)不能控制GPIO,只能手動強(qiáng)制關(guān)機(jī);