日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁(yè) > > IOT物聯(lián)網(wǎng)小鎮(zhèn)
[導(dǎo)讀]作?者:道哥,10年的嵌入式開(kāi)發(fā)老兵。公眾號(hào):【IOT物聯(lián)網(wǎng)小鎮(zhèn)】,專注于:C/C、Linux操作系統(tǒng)、應(yīng)用程序設(shè)計(jì)、物聯(lián)網(wǎng)、單片機(jī)和嵌入式開(kāi)發(fā)等領(lǐng)域。?公眾號(hào)回復(fù)【書(shū)籍】,獲取Linux、嵌入式領(lǐng)域經(jīng)典書(shū)籍。轉(zhuǎn)?載:歡迎轉(zhuǎn)載文章,轉(zhuǎn)載需注明出處。目錄bootloader跳轉(zhuǎn)到操...



目錄


  • bootloader 跳轉(zhuǎn)到操作系統(tǒng)


  • 操作系統(tǒng)跳轉(zhuǎn)到應(yīng)用程序


  • 應(yīng)用程序調(diào)用操作系統(tǒng)中的函數(shù)


不論是在x86平臺(tái)上,還是在嵌入式平臺(tái)上,系統(tǒng)的啟動(dòng)一般都經(jīng)歷了 bootloader操作系統(tǒng),再到應(yīng)用程序,這樣的三級(jí)跳過(guò)程。


每一個(gè)相互交接的過(guò)程,都是我們學(xué)習(xí)的重點(diǎn)。


這篇文章,我們?nèi)匀灰詘86平臺(tái)為例,一起來(lái)看一下:從上電之后,系統(tǒng)是如何一步一步的進(jìn)入應(yīng)用程序的入口地址


bootloader 跳轉(zhuǎn)到操作系統(tǒng)

在上一篇文章中,討論了bootloader在進(jìn)入保護(hù)模式之后,在地址0x0001_0000處創(chuàng)建了全局描述符表(GDT),表中創(chuàng)建了3個(gè)段描述符:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序只要在GDT中創(chuàng)建了這3個(gè)描述符,然后把GDT的地址(eg: 0x0001_0000)設(shè)置到GDTR寄存器中,此時(shí)就可以進(jìn)入保護(hù)模式工作了(設(shè)置CR0寄存器的bit0為1)。


之前的第6篇文章中Linux從頭學(xué)06:16張結(jié)構(gòu)圖,徹底理解【代碼重定位】的底層原理,我們是假設(shè)bootloader把操作系統(tǒng)程序讀取到內(nèi)存0x0002_0000的位置,這里繼續(xù)使用這個(gè)示例:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序關(guān)于文件頭header的內(nèi)容,與實(shí)模式下是不同的。


在實(shí)模式下,header的布局如下圖:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序bootloader在把操作系統(tǒng),從硬盤(pán)加載到內(nèi)存中之后,從header中取得3個(gè)段的匯編地址(即:段的開(kāi)始地址相對(duì)于文件開(kāi)始的偏移量),然后計(jì)算得到段的基地址,最后把段基地址寫(xiě)回到header的這3個(gè)段地址空間中。


這樣的話,操作系統(tǒng)開(kāi)始執(zhí)行時(shí),就可以從header中準(zhǔn)確的獲取到每一個(gè)段的基地址了,然后就可以設(shè)置相應(yīng)的段寄存器,進(jìn)入正確的執(zhí)行上下文了。


那么在保護(hù)模式下呢,操作系統(tǒng)需要的就不是段的基地址了,而是要獲取到每一個(gè)段的描述符才行。


很顯然,需要借助bootloader才可以完成這個(gè)目標(biāo),也就是:


  1. 在 GDT 中為操作系統(tǒng)程序中的三個(gè)段,建立相應(yīng)的描述符;


  2. 把每一個(gè)段的描述符索引號(hào),寫(xiě)回到操作系統(tǒng)程序的 header 中;


注意:


這里描述的僅僅是一個(gè)可能的過(guò)程,主要用來(lái)理解原理。


有些系統(tǒng)可以用不同的實(shí)現(xiàn)方式,例如:在進(jìn)入操作系統(tǒng)之后,在另外一個(gè)位置存放GDT,并重新創(chuàng)建其中的段描述符。


操作系統(tǒng)的 header 布局

既然header需要作為媒介,來(lái)接收bootloader往其中寫(xiě)入段索引號(hào),所以bootloader與OS就要協(xié)商好,寫(xiě)在什么位置?


可以按照之前的方式,直接覆寫(xiě)在每個(gè)段的匯編地址位置,也可以寫(xiě)在其他的位置,例如:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序其中,最后的3個(gè)位置可以用來(lái)接收操作系統(tǒng)的三個(gè)段索引號(hào)。


建立操作系統(tǒng)的三個(gè)段描述符

bootloader把OS加載到內(nèi)存中之后,會(huì)解析OS的header中數(shù)據(jù),得到每個(gè)段的基地址以及界限。


雖然header中沒(méi)有明確的記錄每個(gè)段的界限,可以根據(jù)下一個(gè)段的開(kāi)始地址,來(lái)計(jì)算得到上一個(gè)段的長(zhǎng)度。


我們可以聯(lián)想一下:


現(xiàn)代Linux系統(tǒng)中ELF文件的格式,在文件頭部中記錄了每一個(gè)段的長(zhǎng)度,具體解析請(qǐng)參考這篇文章:Linux系統(tǒng)中編譯、鏈接的基石-ELF文件:扒開(kāi)它的層層外衣,從字節(jié)碼的粒度來(lái)探索。


此時(shí),bootloader就可以利用這幾個(gè)信息:段基地址、界限、類型以及其他屬性,來(lái)構(gòu)造出相應(yīng)的段描述符了(下圖橙色部分):


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序
PS:這里的示例只為操作系統(tǒng)創(chuàng)建了 3 個(gè)段描述符,實(shí)際情況也許有更多的段。


OS段描述符建立之后,bootloader再把這3個(gè)段描述符在GDT中的索引號(hào),填寫(xiě)到OS的header中相應(yīng)的位置:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序上圖中,“入口地址”下面的那個(gè)4,本質(zhì)上是不需要的,加上更有好處,好處如下:


當(dāng)從bootloader跳入到操作系統(tǒng)的入口地址時(shí),需要告訴處理器兩件事情:


  1. 代碼段的索引號(hào);


  2. 代碼的入口地址;


因此,把入口地址和索引號(hào)放在一起,有助于bootloader直接使用跳轉(zhuǎn)語(yǔ)句,進(jìn)入到OS的start標(biāo)記處開(kāi)始執(zhí)行。


操作系統(tǒng)跳轉(zhuǎn)到應(yīng)用程序

從現(xiàn)代操作系統(tǒng)來(lái)看,這個(gè)標(biāo)題是有錯(cuò)誤的:


操作系統(tǒng)是應(yīng)用程序的下層支撐,相當(dāng)于是應(yīng)用程序的runtime,怎么能叫做跳轉(zhuǎn)到應(yīng)用程序呢?


其實(shí)我想表達(dá)的意思是:操作系統(tǒng)是如何加載、執(zhí)行一個(gè)應(yīng)用程序的。


既然是保護(hù)模式,那么操作系統(tǒng)就承擔(dān)起重要的職責(zé):保護(hù)系統(tǒng)不會(huì)受到每一個(gè)應(yīng)用程序的惡意破壞!


因此,操作系統(tǒng):把應(yīng)用程序從硬盤(pán)上復(fù)制到內(nèi)存中之后,跳入應(yīng)用程序的第一條指令之前,需要為應(yīng)用程序分配好內(nèi)存資源:


  1. 代碼段的基地址、界限、類型和權(quán)限等信息;


  2. 數(shù)據(jù)段的基地址、界限、類型和權(quán)限等信息;


  3. 棧段的基地址、界限、類型和權(quán)限等信息;


以上這些信息,都以段描述符的形式,創(chuàng)建在GDT中。


PS: 在現(xiàn)代操作系統(tǒng)中,應(yīng)用程序都會(huì)有一個(gè)自己私有的局部描述符表 LDT,專門(mén)存儲(chǔ)應(yīng)用程序自己的段描述符。


還記得之前討論過(guò)的下面這張圖嗎?


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序段寄存器的bit2位TI標(biāo)志,就說(shuō)明了需要到GDT中查找段描述符?還是到LDT中去查找?


為了方便起見(jiàn),我們就把所有的段描述符都放在GDT中。


就猶如bootloader為OS創(chuàng)建段描述符一樣,OS也以同樣的步驟為應(yīng)用程序來(lái)創(chuàng)建每一個(gè)段描述符。


此時(shí)的GDT就是下面這樣:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序從這張圖中已經(jīng)可以看出一個(gè)問(wèn)題了:


如果所有應(yīng)用程序的段描述符都放在全局的GDT中,當(dāng)應(yīng)用程序結(jié)束之后,還得去更新GDT,勢(shì)必給操作系統(tǒng)的代碼帶來(lái)很多麻煩。


因此,更合理的方式應(yīng)該是放在應(yīng)用程序私有的LDT中,這個(gè)問(wèn)題,以后還會(huì)進(jìn)一步討論到。


不管怎樣,OS 啟動(dòng)應(yīng)用程序的整體流程如下:


  1. 操作系統(tǒng)把應(yīng)用程序讀取到內(nèi)存中的某個(gè)空閑位置;


  2. 操作系統(tǒng)分析應(yīng)用程序 header 部分的信息;


  3. 操作系統(tǒng)為應(yīng)用程序創(chuàng)建每一個(gè)段描述符,并且把索引號(hào)寫(xiě)回到 header 中;


  4. 跳轉(zhuǎn)到應(yīng)用程序的入口地址,應(yīng)用程序從 header 中獲取到每個(gè)段索引號(hào),設(shè)置好自己的執(zhí)行上下文(即:設(shè)置好各種寄存器);


應(yīng)用程序調(diào)用操作系統(tǒng)中的函數(shù)

這里的函數(shù)可以理解成系統(tǒng)調(diào)用,也就是操作系統(tǒng)為所有的應(yīng)用程序提供的公共函數(shù)。


在Linux系統(tǒng)中,系統(tǒng)調(diào)用是通過(guò)中斷來(lái)實(shí)現(xiàn)的,在中斷處理器程序中,再通過(guò)一個(gè)寄存器來(lái)標(biāo)識(shí):當(dāng)前應(yīng)用程序想調(diào)用哪一個(gè)系統(tǒng)函數(shù),也就是說(shuō):每一個(gè)系統(tǒng)函數(shù)都有一個(gè)固定的數(shù)字編號(hào)


再回到我們當(dāng)前討論的x86處理器中,操作系統(tǒng)提供系統(tǒng)函數(shù)的最簡(jiǎn)單的方法就是:


把所有的系統(tǒng)函數(shù)都放在一個(gè)單獨(dú)的代碼段中,把這個(gè)段的索引號(hào)以及每一個(gè)系統(tǒng)函數(shù)的偏移地址告訴應(yīng)用程序。


這樣的話,應(yīng)用程序就可以通過(guò)這2個(gè)信息調(diào)用到系統(tǒng)函數(shù)了。


假如:有2個(gè)系統(tǒng)函數(shù)os_func1和os_func2,放在一個(gè)獨(dú)立的段中:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序既然OS中多了一個(gè)代碼段,那么bootloader就需要幫助它在GDT中多創(chuàng)建一個(gè)段描述符:


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序在應(yīng)用程序的header中,預(yù)留一個(gè)足夠大的空間來(lái)存放每一個(gè)系統(tǒng)函數(shù)的跳轉(zhuǎn)信息(系統(tǒng)函數(shù)的段索引號(hào)和函數(shù)的偏移地址):


Linux從頭學(xué)10:三級(jí)跳過(guò)程詳解-從 bootloader 到 操作系統(tǒng),再到應(yīng)用程序應(yīng)用程序有了這個(gè)信息之后,當(dāng)需要調(diào)用os_func1時(shí),就直接跳轉(zhuǎn)到相應(yīng)的 段索引號(hào):函數(shù)偏移地址,就可以調(diào)用到這個(gè)系統(tǒng)函數(shù)了。


這里同樣的會(huì)引出2個(gè)問(wèn)題:


  1. 如果操作系統(tǒng)提供的系統(tǒng)函數(shù)很多,應(yīng)用程序也很多,那么操作系統(tǒng)在加載每一個(gè)應(yīng)用程序時(shí),豈不是要忙死了?而且應(yīng)用程序也不知道應(yīng)該保留多大的空間來(lái)存放這些系統(tǒng)函數(shù)的跳轉(zhuǎn)信息;


  2. 在執(zhí)行系統(tǒng)函數(shù)時(shí),此時(shí)代碼段、數(shù)據(jù)段都是屬于操作系統(tǒng)的勢(shì)力范圍,但是?;泛蜅m斨羔樖褂玫娜匀皇菓?yīng)用程序擁有的棧,這樣合理嗎?


對(duì)于第一個(gè)問(wèn)題,所以Linux中通過(guò)中斷,提供一個(gè)統(tǒng)一的調(diào)用入口地址,然后通過(guò)一個(gè)寄存器來(lái)區(qū)分是哪一個(gè)函數(shù)。


對(duì)于第二個(gè)問(wèn)題,Linux在加載每一個(gè)應(yīng)用程序時(shí),會(huì)在內(nèi)核中建立與該應(yīng)用程序相關(guān)的數(shù)據(jù)結(jié)構(gòu),并且在內(nèi)核中創(chuàng)建一塊內(nèi)存空間,專門(mén)用作:從這個(gè)應(yīng)用程序跳轉(zhuǎn)到內(nèi)核中執(zhí)行代碼時(shí),所使用的棧空間。


但是,還有一些問(wèn)題依然存在,例如:


  1. 應(yīng)用程序雖然可以調(diào)用操作系統(tǒng)提供的函數(shù)了,但是操作系統(tǒng)如何對(duì)內(nèi)核代碼進(jìn)行保護(hù)?;


  2. Linux 為應(yīng)用程序建立內(nèi)部棧的底層支撐是什么


這就涉及到 x86 中復(fù)雜的特權(quán)級(jí)相關(guān)內(nèi)容了,下一篇文章,我們就向這些細(xì)節(jié)問(wèn)題繼續(xù)探索。


------ End ------


本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢(shì)抑制與過(guò)流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問(wèn)題卻十分常見(jiàn),不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問(wèn)題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車場(chǎng)照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢(shì)逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問(wèn)題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對(duì)周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來(lái)解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開(kāi)關(guān)電源具有效率高的特性,而且開(kāi)關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開(kāi)關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉