資料介紹
Linux 下的關(guān)機(jī)和重啟流程對于一般的桌面應(yīng)用和網(wǎng)絡(luò)服務(wù)器來說并不重要,但是在用戶自己定義的嵌入式系統(tǒng)內(nèi)核中就有一定的研究意義,通過了解 Linux 關(guān)機(jī)重啟的流程,我們對它可以修改和自定義,甚至以此為基礎(chǔ)開發(fā)出全新的功能來。
1. 概述
在 Linux 下的關(guān)機(jī)和重啟可能由兩種行為引發(fā),一是通過用戶編程,一是系統(tǒng)自己產(chǎn)生的消息。用戶和系統(tǒng)進(jìn)行交互的方式也有兩個(gè),一個(gè)是系統(tǒng)調(diào)用:sys_reboot,另一個(gè)就是 apm 或則 acpi 的設(shè)備文件,通過對其操作也可以使系統(tǒng)關(guān)機(jī)或者重啟。
2. 通過系統(tǒng)調(diào)用 sys_reboot 的重啟
這個(gè)系統(tǒng)調(diào)用定義了一系列的 MAGIC_NUMBER,在調(diào)用的開始部分首先檢查 MAGIC_NUMBER 是否正確,只有正確才繼續(xù)向下運(yùn)行。在重啟的時(shí)候轉(zhuǎn)向分支
case Linux _REBOOT_CMD_RESTART:
首先使用 notifier_call_chain 向其它部分發(fā)出重啟的消息,然后調(diào)用 machine_restart 函數(shù)完成重啟。
machine_restart 函數(shù)的開始部分有一段SMP相關(guān)的代碼,主要完成多 CPU 時(shí)由一個(gè) CPU 完成重啟操作,其它 CPU 處于等待狀態(tài)。之后系統(tǒng)根據(jù)一個(gè)變量 reboot_thru_bios 的內(nèi)容判斷重啟方式,通過閱讀 reboot_setup 我們可以得知,這個(gè)參數(shù)的內(nèi)容是在系統(tǒng)啟動時(shí)指定的,決定了是否利用 bios,事實(shí)上是系統(tǒng)復(fù)位后的入口 (FFFF:0000) 地址的程序進(jìn)行重啟。在不通過 bios 進(jìn)行重啟的情況下,系統(tǒng)首先設(shè)定了重啟標(biāo)志,然后向端口 0xfe 寫入數(shù)字 0x64,這種重啟的具體原理我還不大清楚,似乎是模擬了一次 reset 鍵的按下,希望大家和我討論。在通過 bios 重啟的情況下,系統(tǒng)同樣先設(shè)定了重啟模式,然后切換到了實(shí)模式,通過一條 ljmp $0xffff,$0x0 完成了重啟。
3. 通過系統(tǒng)調(diào)用 sys_reboot 進(jìn)行關(guān)機(jī)
在系統(tǒng)調(diào)用的處理分支上,我們可以看到,首先同樣是檢查 MAGIC_NUMBER,然后在
case Linux _REBOOT_CMD_POWER_OFF:
的執(zhí)行流程里面,又是使用 notifier_call_chain 發(fā)出了關(guān)閉計(jì)算機(jī)電源的消息,緊接著執(zhí)行了 machine_power_off 函數(shù)。我們在 machine_power_off 函數(shù)中可以看到,如果 pm_power_off 這個(gè)函數(shù)指針不為空,那么系統(tǒng)就會通過調(diào)用這個(gè)函數(shù)進(jìn)行關(guān)機(jī)。在 apm 已經(jīng)加載的情況下 (SMP 除外),實(shí)際上 pm_power_off 函數(shù)實(shí)際上指向了 apm.c 中的 apm_power_off,在這個(gè)函數(shù)里系統(tǒng)通過 apm_info 結(jié)構(gòu)里的值,使用切換到實(shí)模式關(guān)機(jī),或者使用 apm_bios_call_simple 函數(shù)調(diào)用保護(hù)模式下的 apm 接口關(guān)機(jī)兩種方法。
4. apm 驅(qū)動本身的關(guān)機(jī)過程
apm 使用其注冊的設(shè)備的 ioctl 接口完成 apm 的操作,在 apm.c的do_ioctl 函數(shù)中可以看見處理的分支。這里只有 suspend 和 standby 的代碼,所以我們不能通過 ioctl 這種方法使用 apm 關(guān)機(jī)。
當(dāng)用戶按下 POWER 開關(guān)的時(shí)候,如果有 apm 模塊,那么關(guān)機(jī)流程是由 apm 來處理的。apm 驅(qū)動在初始化的時(shí)候啟動了一個(gè) apm 內(nèi)核線程:apm_mainloop,系統(tǒng)會在這里檢測到 POWEROFF 按鍵消息并且將其命名為 APM_SYS_SUSPEND,以區(qū)別 apm -s 設(shè)置的 APM_USER_SUSPEND 模式。緊接著進(jìn)入了 apm_event_handler 函數(shù),又從 apm_event_handler 函數(shù)進(jìn)入了 check_events 函數(shù),處理函數(shù)對應(yīng)的 case 分支上。系統(tǒng)同樣使用了 suspend 函數(shù)進(jìn)行關(guān)機(jī),不過由于其它參數(shù)的原因,suspend 最后調(diào)用的是關(guān)機(jī)的流程。
5. 解決問題實(shí)例
1) 按 POWER 鍵時(shí)某些主板死機(jī)
經(jīng)查只有某些特定的驅(qū)動裝載之后才會出現(xiàn)這樣的情況,并且當(dāng)使用關(guān)機(jī)系統(tǒng)調(diào)用 sys_reboot 的時(shí)候沒有這樣的問題。分析 apm 的處理流程,懷疑是在關(guān)機(jī)前驅(qū)動程序沒有正確處理 apm 發(fā)出的詢問消息造成的。由于部分驅(qū)動程序沒有源代碼,決定 hack 掉 apm.c 的關(guān)機(jī)部分,讓兩種方式的關(guān)機(jī)走同樣的流程。于是把 apm.c 的 check_events 函數(shù)中對 APM_SYS_SUSPEND 部分改寫為如下代碼:
ret = exec_usermodehelper(poweroff_helper_path, argv, envp); if (ret) { printk(KERN_ERR “apm.c: failed to exec %s , errno = %dn”, poweroff_helper_path, errno); } break;
定義了一個(gè)用戶態(tài)應(yīng)用程序 poweroff_helper_path,當(dāng) POWEROFF 鍵按下的時(shí)候系統(tǒng)運(yùn)行這個(gè) kernel_helper 程序。我們再寫一個(gè)通過 sys_reboot 系統(tǒng)調(diào)用關(guān)機(jī)的程序,放在指定的位置下。死機(jī)的問題就解決了。
2) 快速返回實(shí)模式重啟
主要可以參考了 process.c 中的返回實(shí)模式的代碼,比如我把 real_mode_switch 換成如下代碼:
// For fast reboot support static unsigned char fast_reboot_switch [] = { 0x66, 0x0f, 0x20, 0xc0, /* movl %cr0,%eax */ 0x66, 0x25, 0x10, 0x11, 0x11, 0x11, /* andl $0x11111110,%eax */ 0x66, 0x0f, 0x22, 0xc0, /* movl %eax,%cr0 */ 0xea, 0x00, 0x00, 0x00, 0x70 /* ljmp $0x7000,$0x0000 */ };
系統(tǒng)就可以切換到實(shí)模式中,然后跳轉(zhuǎn)到 7000H:0 位置開始執(zhí)行。
6. ACPI 概述
在 2.4.20 內(nèi)核中 ACPI 模塊被注明為試驗(yàn)和未完成,里面有一部分功能也許沒有實(shí)現(xiàn)。如果 APM 和 APCI 兩個(gè)模塊同時(shí)編譯進(jìn)內(nèi)核,APM 在 ACPI 前被加載,APM 起作用使 ACPI 退出。對于系統(tǒng)電量、電源實(shí)踐一類的支持(主要是在筆記本上有用),靠的是 acpid 這個(gè) daemon 程序。
沒有一個(gè)功能類似 apm 的應(yīng)用程序切換狀態(tài),acpi 的程序僅僅完成了對 acpi 狀態(tài)的查詢。用戶實(shí)現(xiàn) S0-S4 的功能可以直接向 /proc/acpi/sleep 文件中寫入數(shù)字來實(shí)現(xiàn)。通過讀出 (cat) 其中的內(nèi)容可以知道系統(tǒng)到底支持那些模式。
acpi 模塊的源代碼主程序在 linux/drivers/acpi/driver.c 中,如果向 sleep 文件寫東西,就轉(zhuǎn)到了 linux/drivers/acpi/ospm/system/sm_osl.c 文件的 sm_osl_proc_write_sleep 函數(shù)中,這個(gè)函數(shù)后來調(diào)用了 sm_osl_suspend 函數(shù)。在這個(gè)函數(shù)里完成了各種功能,包括保護(hù)各種狀態(tài)。最后真正的 sleep 是通過對 acpi_enter_sleep_state 的調(diào)用完成的,這個(gè)函數(shù)在 Linux /drivers/acpi/hardware/hwsleep.c 文件中,這里寫了 acpi 的寄存器使系統(tǒng)進(jìn)入 sleep 狀態(tài)。寫寄存器的指令在這個(gè)目錄下面的 hwregs.c 中。
7. 總結(jié)
本文對 acpi 的介紹非常簡略,實(shí)際上 ACPI 必定會成為將來 Linux 內(nèi)核中首選的電源管理方式。由于目前官方代碼中 ACPI 版本較低,所以沒有太詳細(xì)的論述,希望將來的內(nèi)核能有所改進(jìn)。
?
1. 概述
在 Linux 下的關(guān)機(jī)和重啟可能由兩種行為引發(fā),一是通過用戶編程,一是系統(tǒng)自己產(chǎn)生的消息。用戶和系統(tǒng)進(jìn)行交互的方式也有兩個(gè),一個(gè)是系統(tǒng)調(diào)用:sys_reboot,另一個(gè)就是 apm 或則 acpi 的設(shè)備文件,通過對其操作也可以使系統(tǒng)關(guān)機(jī)或者重啟。
2. 通過系統(tǒng)調(diào)用 sys_reboot 的重啟
這個(gè)系統(tǒng)調(diào)用定義了一系列的 MAGIC_NUMBER,在調(diào)用的開始部分首先檢查 MAGIC_NUMBER 是否正確,只有正確才繼續(xù)向下運(yùn)行。在重啟的時(shí)候轉(zhuǎn)向分支
case Linux _REBOOT_CMD_RESTART:
首先使用 notifier_call_chain 向其它部分發(fā)出重啟的消息,然后調(diào)用 machine_restart 函數(shù)完成重啟。
machine_restart 函數(shù)的開始部分有一段SMP相關(guān)的代碼,主要完成多 CPU 時(shí)由一個(gè) CPU 完成重啟操作,其它 CPU 處于等待狀態(tài)。之后系統(tǒng)根據(jù)一個(gè)變量 reboot_thru_bios 的內(nèi)容判斷重啟方式,通過閱讀 reboot_setup 我們可以得知,這個(gè)參數(shù)的內(nèi)容是在系統(tǒng)啟動時(shí)指定的,決定了是否利用 bios,事實(shí)上是系統(tǒng)復(fù)位后的入口 (FFFF:0000) 地址的程序進(jìn)行重啟。在不通過 bios 進(jìn)行重啟的情況下,系統(tǒng)首先設(shè)定了重啟標(biāo)志,然后向端口 0xfe 寫入數(shù)字 0x64,這種重啟的具體原理我還不大清楚,似乎是模擬了一次 reset 鍵的按下,希望大家和我討論。在通過 bios 重啟的情況下,系統(tǒng)同樣先設(shè)定了重啟模式,然后切換到了實(shí)模式,通過一條 ljmp $0xffff,$0x0 完成了重啟。
3. 通過系統(tǒng)調(diào)用 sys_reboot 進(jìn)行關(guān)機(jī)
在系統(tǒng)調(diào)用的處理分支上,我們可以看到,首先同樣是檢查 MAGIC_NUMBER,然后在
case Linux _REBOOT_CMD_POWER_OFF:
的執(zhí)行流程里面,又是使用 notifier_call_chain 發(fā)出了關(guān)閉計(jì)算機(jī)電源的消息,緊接著執(zhí)行了 machine_power_off 函數(shù)。我們在 machine_power_off 函數(shù)中可以看到,如果 pm_power_off 這個(gè)函數(shù)指針不為空,那么系統(tǒng)就會通過調(diào)用這個(gè)函數(shù)進(jìn)行關(guān)機(jī)。在 apm 已經(jīng)加載的情況下 (SMP 除外),實(shí)際上 pm_power_off 函數(shù)實(shí)際上指向了 apm.c 中的 apm_power_off,在這個(gè)函數(shù)里系統(tǒng)通過 apm_info 結(jié)構(gòu)里的值,使用切換到實(shí)模式關(guān)機(jī),或者使用 apm_bios_call_simple 函數(shù)調(diào)用保護(hù)模式下的 apm 接口關(guān)機(jī)兩種方法。
4. apm 驅(qū)動本身的關(guān)機(jī)過程
apm 使用其注冊的設(shè)備的 ioctl 接口完成 apm 的操作,在 apm.c的do_ioctl 函數(shù)中可以看見處理的分支。這里只有 suspend 和 standby 的代碼,所以我們不能通過 ioctl 這種方法使用 apm 關(guān)機(jī)。
當(dāng)用戶按下 POWER 開關(guān)的時(shí)候,如果有 apm 模塊,那么關(guān)機(jī)流程是由 apm 來處理的。apm 驅(qū)動在初始化的時(shí)候啟動了一個(gè) apm 內(nèi)核線程:apm_mainloop,系統(tǒng)會在這里檢測到 POWEROFF 按鍵消息并且將其命名為 APM_SYS_SUSPEND,以區(qū)別 apm -s 設(shè)置的 APM_USER_SUSPEND 模式。緊接著進(jìn)入了 apm_event_handler 函數(shù),又從 apm_event_handler 函數(shù)進(jìn)入了 check_events 函數(shù),處理函數(shù)對應(yīng)的 case 分支上。系統(tǒng)同樣使用了 suspend 函數(shù)進(jìn)行關(guān)機(jī),不過由于其它參數(shù)的原因,suspend 最后調(diào)用的是關(guān)機(jī)的流程。
5. 解決問題實(shí)例
1) 按 POWER 鍵時(shí)某些主板死機(jī)
經(jīng)查只有某些特定的驅(qū)動裝載之后才會出現(xiàn)這樣的情況,并且當(dāng)使用關(guān)機(jī)系統(tǒng)調(diào)用 sys_reboot 的時(shí)候沒有這樣的問題。分析 apm 的處理流程,懷疑是在關(guān)機(jī)前驅(qū)動程序沒有正確處理 apm 發(fā)出的詢問消息造成的。由于部分驅(qū)動程序沒有源代碼,決定 hack 掉 apm.c 的關(guān)機(jī)部分,讓兩種方式的關(guān)機(jī)走同樣的流程。于是把 apm.c 的 check_events 函數(shù)中對 APM_SYS_SUSPEND 部分改寫為如下代碼:
ret = exec_usermodehelper(poweroff_helper_path, argv, envp); if (ret) { printk(KERN_ERR “apm.c: failed to exec %s , errno = %dn”, poweroff_helper_path, errno); } break;
定義了一個(gè)用戶態(tài)應(yīng)用程序 poweroff_helper_path,當(dāng) POWEROFF 鍵按下的時(shí)候系統(tǒng)運(yùn)行這個(gè) kernel_helper 程序。我們再寫一個(gè)通過 sys_reboot 系統(tǒng)調(diào)用關(guān)機(jī)的程序,放在指定的位置下。死機(jī)的問題就解決了。
2) 快速返回實(shí)模式重啟
主要可以參考了 process.c 中的返回實(shí)模式的代碼,比如我把 real_mode_switch 換成如下代碼:
// For fast reboot support static unsigned char fast_reboot_switch [] = { 0x66, 0x0f, 0x20, 0xc0, /* movl %cr0,%eax */ 0x66, 0x25, 0x10, 0x11, 0x11, 0x11, /* andl $0x11111110,%eax */ 0x66, 0x0f, 0x22, 0xc0, /* movl %eax,%cr0 */ 0xea, 0x00, 0x00, 0x00, 0x70 /* ljmp $0x7000,$0x0000 */ };
系統(tǒng)就可以切換到實(shí)模式中,然后跳轉(zhuǎn)到 7000H:0 位置開始執(zhí)行。
6. ACPI 概述
在 2.4.20 內(nèi)核中 ACPI 模塊被注明為試驗(yàn)和未完成,里面有一部分功能也許沒有實(shí)現(xiàn)。如果 APM 和 APCI 兩個(gè)模塊同時(shí)編譯進(jìn)內(nèi)核,APM 在 ACPI 前被加載,APM 起作用使 ACPI 退出。對于系統(tǒng)電量、電源實(shí)踐一類的支持(主要是在筆記本上有用),靠的是 acpid 這個(gè) daemon 程序。
沒有一個(gè)功能類似 apm 的應(yīng)用程序切換狀態(tài),acpi 的程序僅僅完成了對 acpi 狀態(tài)的查詢。用戶實(shí)現(xiàn) S0-S4 的功能可以直接向 /proc/acpi/sleep 文件中寫入數(shù)字來實(shí)現(xiàn)。通過讀出 (cat) 其中的內(nèi)容可以知道系統(tǒng)到底支持那些模式。
acpi 模塊的源代碼主程序在 linux/drivers/acpi/driver.c 中,如果向 sleep 文件寫東西,就轉(zhuǎn)到了 linux/drivers/acpi/ospm/system/sm_osl.c 文件的 sm_osl_proc_write_sleep 函數(shù)中,這個(gè)函數(shù)后來調(diào)用了 sm_osl_suspend 函數(shù)。在這個(gè)函數(shù)里完成了各種功能,包括保護(hù)各種狀態(tài)。最后真正的 sleep 是通過對 acpi_enter_sleep_state 的調(diào)用完成的,這個(gè)函數(shù)在 Linux /drivers/acpi/hardware/hwsleep.c 文件中,這里寫了 acpi 的寄存器使系統(tǒng)進(jìn)入 sleep 狀態(tài)。寫寄存器的指令在這個(gè)目錄下面的 hwregs.c 中。
7. 總結(jié)
本文對 acpi 的介紹非常簡略,實(shí)際上 ACPI 必定會成為將來 Linux 內(nèi)核中首選的電源管理方式。由于目前官方代碼中 ACPI 版本較低,所以沒有太詳細(xì)的論述,希望將來的內(nèi)核能有所改進(jìn)。
?
下載該資料的人也在下載
下載該資料的人還在閱讀
更多 >
- Android 源碼添加長按電源鍵重啟功能
- linux 從入門到跑路-電源管理
- ideapad720s在接通電源情況下,關(guān)機(jī)后自動重啟的解決辦法
- Mac電腦選擇系統(tǒng)菜單中的關(guān)機(jī)或重啟無法關(guān)機(jī)或重啟,只能通過按電源鍵關(guān)機(jī)以及打開應(yīng)用經(jīng)??ㄋ绬栴}及解決方案
- [Linux] Ubuntu電源鍵軟關(guān)機(jī)設(shè)置
- 嵌入式Linux系統(tǒng)重啟后如何使得redis中的數(shù)據(jù)不再有
- 【嵌入式Linux(基礎(chǔ)篇)】嵌入式Linux底層系統(tǒng)開發(fā)流程和應(yīng)用開發(fā)流程
- 嵌入式Linux專題(一)——嵌入式Linux系統(tǒng)構(gòu)成及啟動流程
- 嵌入式 Linux 啟動流程和 bootloader 介紹
- STM8S單片機(jī)入門:關(guān)機(jī)模式開關(guān)機(jī)按鈕資料下載
- 如何實(shí)現(xiàn)遠(yuǎn)程FPGA的更新和重啟 21次下載
- 嵌入式Linux與物聯(lián)網(wǎng)軟件開發(fā)C語言內(nèi)核深度解析書籍的介紹
- halt poweroff reboot問題 0次下載
- 精彩易懂的LINUX講座 0次下載
- 基于ARM9的嵌入式Linux開發(fā)流程的研究
- python解析netflow數(shù)據(jù)到csv的流程詳解 516次閱讀
- 定時(shí)開關(guān)機(jī)在哪里設(shè)置 定時(shí)開關(guān)機(jī)對手機(jī)有利還是有害 6415次閱讀
- centos關(guān)機(jī)和重啟命令 883次閱讀
- Linux系統(tǒng)文件讀寫流程 597次閱讀
- kernel panic流程分析 893次閱讀
- Linux發(fā)送HTTP網(wǎng)絡(luò)包圖像 sk_buff數(shù)據(jù)結(jié)構(gòu)解析 2105次閱讀
- 一文解析鴻蒙LiteOS和LINUX比較 2.1w次閱讀
- 蘋果手機(jī)經(jīng)常死機(jī)或自動重啟應(yīng)該如何解決 5.4w次閱讀
- 單片機(jī)自關(guān)機(jī)電路原理圖 5187次閱讀
- 手機(jī)重啟和關(guān)機(jī)再開機(jī)到底有什么區(qū)別? 1.1w次閱讀
- 嵌入式Linux OS啟動流程 3638次閱讀
- 一文解析鈷酸鋰生產(chǎn)工藝流程 1.8w次閱讀
- 詳解bootloader的執(zhí)行流程與ARM Linux啟動過程分析 1w次閱讀
- 自動關(guān)機(jī)電路,自動關(guān)機(jī)原理分析 1.7w次閱讀
- 超詳細(xì)電腦加速關(guān)機(jī)100%絕技!教你幾招電腦關(guān)機(jī)慢的解決方法 4.5w次閱讀
下載排行
本周
- 1TC358743XBG評估板參考手冊
- 1.36 MB | 330次下載 | 免費(fèi)
- 2開關(guān)電源基礎(chǔ)知識
- 5.73 MB | 6次下載 | 免費(fèi)
- 3100W短波放大電路圖
- 0.05 MB | 4次下載 | 3 積分
- 4嵌入式linux-聊天程序設(shè)計(jì)
- 0.60 MB | 3次下載 | 免費(fèi)
- 5基于FPGA的光纖通信系統(tǒng)的設(shè)計(jì)與實(shí)現(xiàn)
- 0.61 MB | 2次下載 | 免費(fèi)
- 6基于FPGA的C8051F單片機(jī)開發(fā)板設(shè)計(jì)
- 0.70 MB | 2次下載 | 免費(fèi)
- 751單片機(jī)窗簾控制器仿真程序
- 1.93 MB | 2次下載 | 免費(fèi)
- 8基于51單片機(jī)的RGB調(diào)色燈程序仿真
- 0.86 MB | 2次下載 | 免費(fèi)
本月
- 1OrCAD10.5下載OrCAD10.5中文版軟件
- 0.00 MB | 234315次下載 | 免費(fèi)
- 2555集成電路應(yīng)用800例(新編版)
- 0.00 MB | 33564次下載 | 免費(fèi)
- 3接口電路圖大全
- 未知 | 30323次下載 | 免費(fèi)
- 4開關(guān)電源設(shè)計(jì)實(shí)例指南
- 未知 | 21548次下載 | 免費(fèi)
- 5電氣工程師手冊免費(fèi)下載(新編第二版pdf電子書)
- 0.00 MB | 15349次下載 | 免費(fèi)
- 6數(shù)字電路基礎(chǔ)pdf(下載)
- 未知 | 13750次下載 | 免費(fèi)
- 7電子制作實(shí)例集錦 下載
- 未知 | 8113次下載 | 免費(fèi)
- 8《LED驅(qū)動電路設(shè)計(jì)》 溫德爾著
- 0.00 MB | 6653次下載 | 免費(fèi)
總榜
- 1matlab軟件下載入口
- 未知 | 935054次下載 | 免費(fèi)
- 2protel99se軟件下載(可英文版轉(zhuǎn)中文版)
- 78.1 MB | 537796次下載 | 免費(fèi)
- 3MATLAB 7.1 下載 (含軟件介紹)
- 未知 | 420026次下載 | 免費(fèi)
- 4OrCAD10.5下載OrCAD10.5中文版軟件
- 0.00 MB | 234315次下載 | 免費(fèi)
- 5Altium DXP2002下載入口
- 未知 | 233046次下載 | 免費(fèi)
- 6電路仿真軟件multisim 10.0免費(fèi)下載
- 340992 | 191185次下載 | 免費(fèi)
- 7十天學(xué)會AVR單片機(jī)與C語言視頻教程 下載
- 158M | 183278次下載 | 免費(fèi)
- 8proe5.0野火版下載(中文版免費(fèi)下載)
- 未知 | 138040次下載 | 免費(fèi)
評論
查看更多