色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

gcc編譯時,鏈接器安排的【虛擬地址】是如何計算出來的?

微云疏影 ? 來源:道哥分享 ? 作者:道哥分享 ? 2023-01-24 16:46 ? 次閱讀

問題描述

昨天下午,旁邊的同事在學習Linux系統(tǒng)中的虛擬地址映射(經典書籍《程序員的自我修養(yǎng)-鏈接、裝載與庫》),在看到6.4章節(jié)的時候,對于一個可執(zhí)行的ELF文件中,虛擬地址的值百思不得其解!

例如下面這段C代碼:

首先編譯出32位的可執(zhí)行程序(為了避開一些與主題無關的干擾因素,采用了靜態(tài)鏈接):

gcc -m32 -static test.c -o test

編譯得到ELF格式的可執(zhí)行文件:test。

這個時候,使用readelf工具來查看這個可執(zhí)行文件中的段信息(segment):

上圖中的紅色矩形框中,第二個段的地址為什么是 0x080e_9f5c?

這篇文章主要根據書中的解釋,來具體的分析這個值的來龍去脈。

ELF 文件格式

在Linux系統(tǒng)中,有4種類型的文件都是ELF格式,包括:目標文件,可執(zhí)行文件,動態(tài)鏈接庫文件、核心轉儲文件。

如果想系統(tǒng)掌握Linux系統(tǒng)中的底層知識,研究ELF的格式是避免不了的事情。

很久之前總結過這篇文章:《Linux系統(tǒng)中編譯、鏈接的基石-ELF文件:扒開它的層層外衣,從字節(jié)碼的粒度來探索》,里面詳細總結了ELF文件的內部結構。

這里就不再贅述了,只要記住2點:

1.從編譯器的角度看,ELF 文件是由很多的節(jié)(Section)組成的;

2.從程序加載器的角度看,ELF 文件是又很多的段(Segment)組成的;

其實它倆沒有本質區(qū)別,只不過是鏈接器在鏈接階段,把不同目標文件中相同的section組織在一起,形成一個 segment。

對于剛才編譯出的test可執(zhí)行文件,其加載視圖如下:

可以看到該文件一共有5個段(segment),前2個需要LOAD到內存的段,它們屬性分別是:讀、執(zhí)行(R E) 和 讀、寫(RW),它們分別是代碼段和數據段。

綠色的箭頭反映出:代碼段中包含了很多的 section;黃色的箭頭反映出數據段也包含了很多的 section。

地址轉換和內存映射

從地址轉換的角度來看:

Linux 系統(tǒng)中CPU中使用的都是虛擬地址,該虛擬地址在尋址的時候,需要經過MMU地址轉換,得到實際的物理地址,然后才能在物理內存中讀取指令,或者讀取、寫入數據。

在現代操作系統(tǒng)中,MMU地址轉換單元基本上都是通過頁表來進行地址轉換的:

當然了,有些系統(tǒng)是兩級轉換(頁目錄、頁表),有些系統(tǒng)是三級或者四級頁表。

從內存映射的角度來看:

操作系統(tǒng)在把一個可執(zhí)行程序加載到系統(tǒng)中時,把ELF文件中每個段的內容讀取到物理內存中,然后把這個物理內存映射到該段對應的虛擬地址上(VirtAddr)。

假設一個可執(zhí)行程序中的代碼段長度是1.2K字節(jié), 數據段長度是1.3K字節(jié)。

操作系統(tǒng)在把它倆讀取到內存中時,需要 2 個物理內存頁來分別存儲它們(每 1 個物理頁的長度是4K):

雖然每一個物理內存頁的大小是 4K,但是代碼段和數據段實際上只使用了每個頁面剛開始的一段空間。

當CPU中需要讀取物理內存上代碼段中的指令時,使用的虛擬地址是 0x0000_1000 ~ 0x0000_1000 + 1.2K這個區(qū)間的地址,MMU單元經過頁表轉換之后,就會得到這個存放著代碼段的物理頁的物理地址。

數據段的尋址方式也是如此:當CPU中需要讀寫物理內存上數據段中的數據時,使用的虛擬地址是 0x0000_2000 ~ 0x0000_2000 + 1.3K這個區(qū)間的地址。

MMU單元經過頁表轉換之后,就會得到存放著數據段的物理頁的物理地址。

可以看出在這樣的安排下,每一個段的虛擬地址,都是按照4K(0x1000)對齊的。

如果操作系統(tǒng)都是這樣簡單映射的話,那么事情就簡單多了。

如果按照這樣的安排,來分析一下文章開頭的 test 可執(zhí)行程序中的虛擬地址安排:

1.代碼段安排的開始虛擬地址是 0x0804_8000,這是 4K 對齊的;

2.代碼段的結束虛擬地址就應該是 0x0804_8000 + 0xa0725 = 0x080e_8725;

3.那么數據段的開始地址就可以安排在 0x080e_8725 之后的下一個 4K 對齊的邊界地址,即:0x080e_9000。

但是這樣的地址安排,嚴重浪費了物理內存空間!

1.2K 字節(jié)的代碼段加上1.3K字節(jié)的數據段,本來只需要1個物理頁就夠了(4KB),但是這里卻消耗掉2個物理頁(8KB)。

為了減少物理內存的浪費,Linux操作系統(tǒng)就采用了一些巧妙的辦法來減少物理內存的浪費,那就是: 把文件中接壤部分的代碼段和數據段,讀取到同一個物理內存頁中,然后在虛擬地址空間中映射兩次,詳述如下。

Linux 中的內存重復映射

先來看一下test文件的結構:

代碼段在文件中的開始位置是:0x00000,長度是 0xa0725。

數據段的開始位置是:0xa0f5c,長度是0x1024。

可以看到它倆之間有一個空白區(qū)間,長度是: 0xa0f5c - 0xa0725 = 0x837(十進制:2103字節(jié))。

由于操作系統(tǒng)在把test文件讀取到物理內存的時候,從文件開始代碼段的0x00000地址開始讀取,按照4KB為一個單位存放到一個物理頁中。

1.文件中代碼段的 0x00000 ~ 0x00FFF 讀取到一個物理頁中;

2.文件中代碼段的 0x01000 ~ 0x01FFF 讀取到物理頁中;

3.下面的內容都是如此分割、復制;

也就是說:相當于把test文件從開始位置,按照4KB為一個單位進行"切割",然后復制到不同的物理內存頁中,如下所示:

注意:這些物理頁的地址很可能是不連續(xù)的。

這里有意思的是:代碼段與數據段接壤的這個4KB的空間,它的開始地址是0xA0000,結束地址是0xA0FFF,被復制到物理內存中最上面的橙色物理頁中。

再來看一下代碼段的虛擬地址:在執(zhí)行gcc指令的的時候,鏈接器把代碼段的虛擬地址安排在0x0804_8000處:

也就是說:當CPU中(或者說程序代碼中),使用0x0804_8000 ~ 0x0804_7FFF 這個區(qū)間的地址時,經過地址映射,就會找到物理內存中淺綠色的物理頁,而這個物理頁也對應著test可執(zhí)行文件開始的第一個4KB的空間。

而且,從虛擬地址的角度看,它的地址都是連續(xù)的,對應著test文件中連續(xù)的內容,這也是虛擬地址映射的本質。

把代碼段的開始位置安排在 0x0804_8000 地址,這是 Linux 操作系統(tǒng)確定的。

那么考慮一下:代碼段的最后一部分指令相應的4K頁面,其對應的開始虛擬地址是多少呢?

上圖中已經標記出來了,就是虛擬地址中橙色部分:0x080e_8000,計算如下:

通過代碼段的開始地址0x0804_8000,再加上代碼段在內存中的長度0xa0725,結果就是 0x080e_8725。

按照4K (0x1000)對齊之后,最后一個虛擬頁就應該是0x080e_8000。

也就是說:虛擬地址中0x080e_8000 ~ 0x080e_8724 這個區(qū)間就對應著test文件中代碼段的最后一部分指令(0x725個字節(jié))。

此外,上圖中最右側:test文件結構中的2個紅色地址:0xA0000, 0xA1000,是如何計算得到的?

代碼段的長度是 0xA0725,按照4K為一個單位來進行分割,也就是把0xA0725對0x1000進行整除,就得到這個4KB的開始地址0xA0000。

同理,下一個4KB的開始地址就是0xA1000。

把文件中這部分4K的數據(包括:一部分代碼段內容 + 0x837 字節(jié)空洞 + 一部分數據段內容),復制到上圖中物理內存中最上面的橙色物理頁中。

又因為虛擬地址空間中,0x080E_8000開始的這個4KB空間映射到這個物理頁中,所以:在這個虛擬地址空間中,也有一個0x837字節(jié)的空洞,如下所示:

空洞的下方,是代碼段的指令;空洞的上方,是數據段的數據。

現在,這個物理頁中即存放了代碼,又存放了數據。

那么CPU中在查找部分的代碼和數據的時候,必須都能夠找得到才行!

對于代碼段比較好理解:從這個物理頁開始的前0x725個字節(jié)是有效的,從虛擬地址的角度看,就是從0x080e_8000開始的前0x725個字節(jié)是有效的。

因此,對于這部分代碼的尋址,使用的虛擬地址處于0x080e_8000 ~ 0x080e_8724這個區(qū)間中。

那么數據段呢?

重點來了:Linux系統(tǒng)把虛擬地址空間 0x080e_9000 ~ 0x080e_9FFF 也映射到圖中物理內存中最上面的橙色物理頁上!

如下所示:

因為物理頁中,是從0x837個字節(jié)空洞的上面開始,才是真正的數據段內容,那么相應的: 虛擬地址0x080e_9000 ~ 0x080e_9FFF空間中,0x837字節(jié)上面的內容才是數據段內容。

那么在虛擬地址空間中,這個數據段的開始地址應該是多少呢?

只要計算出0x837字節(jié)空洞的上方,距離這個4K頁面開始地址的偏移量就可以了,然后再加上這個4K頁面的起始地址 0x080E_9000,就得到了數據段的開始地址(虛擬地址)。

因為虛擬地址、物理地址、test文件中,都是按照4K的單位進行劃分的,因此這個偏移量就等于:test文件中數據段的開始地址(0xA0F5C) 距離 這個頁面的開始地址(0xA0000) 的偏移量。

0xA0F5C - 0xA0000 = 0xF5C 。

即:從這個4K頁面的開始地址,偏移量為0xF5C的地方,才是數據段內容的開始。

因此對于虛擬地址來說,從0x080e_9000地址開始,偏移量為0xF5C之后的內容才是數據段的內容,這個地址值就是:0x080e_9000 + 0xF5C = 0x080e_9F5C,如下所示:

這個地址正是readelf工具讀所顯示的:數據段加載到虛擬地址空間中的開始地址,如下所示:

至此,就解釋了文章開頭提出的問題!

再來看一下整個數據段的內容:在內存中數據段占據的空間是 0x01e48(readelf 工具讀取到的 MemSiz),那么數據段的結束地址就是(虛擬地址):

0x080e_9F5C + 0x01e48 = 0x080e_bda4

如下所示:

小結

Linux系統(tǒng)中的這個操作:對屬于不同段的內容進行重復映射,有點類似于共享內存的味道了。

只不過這里重復映射之后,每個段的虛擬地址還是需要修正為該段的合法地址。

經過這樣的操作之后,在虛擬地址中每一個段的界限是涇渭分明的,但是映射到的物理內存頁,則有可能是同一個。

審核編輯 :李倩

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • Linux
    +關注

    關注

    87

    文章

    11342

    瀏覽量

    210164
  • 操作系統(tǒng)

    關注

    37

    文章

    6889

    瀏覽量

    123610
收藏 人收藏

    評論

    相關推薦

    LMP90100放大倍數大于等于16后計算的電壓不對,是哪里出了問題?

    *4089.0/8.0/8388608.0 計算出來的電壓與實際電壓一致但是有0.1mv的跳動????。如下圖 問題2 數據輸出速度214.65SPS,放大倍數8,使用緩沖(CH0_CONFIG
    發(fā)表于 01-10 08:24

    ADS1146芯片采集電壓,用SPI通信讀到是數據計算出來和用電壓表測得的值總有誤差,怎么解決?

    我用到ADS1146芯片采集電壓,我的芯片REF電壓是0.6163V,芯片工作電壓是3.3V。用SPI通信讀到是數據計算出來和用電壓表測得的值總有誤差,最大誤差是3mv,誤差是線性誤差。請問是什么原因導致的呢?如果是需要校準 那么怎么校準呢?
    發(fā)表于 01-07 06:34

    使用430驅動TMP006紅外傳感計算出來的Tobj一直是358~360之間波動,為什么?

    使用430驅動TMP006紅外傳感,各寄存讀取數據正常,經過方程式計算后Tdie正常在26.5度左右,Vobj數值8.18312447E-3 計算出來的Tobj一直是358~36
    發(fā)表于 01-02 07:13

    ads1115在采集距離傳感輸入的0.9v時誤差較大,怎么解決?

    ads1115在采集距離傳感輸入的0.9v時誤差較大,為0.03v,在0.6和1.2左右都是0.01v的誤差,請問這是為什么;在誤差為0.03v時候,計算出來的電流和距離傳感的標準電流差了0.04ma,對應實際距離差了十幾
    發(fā)表于 11-29 14:51

    TPA2013D1功率是如何計算出來的,和哪些因素有關?

    有關,如何計算? 和輸入有關么,比如IN+ 接輸入P,IN-接地 和IN+接輸入P, IN-接輸入N輸出功率應該相差4倍吧?? 請答疑,謝謝! 主要想知道功率是如何計算出來的。和哪些因素有關
    發(fā)表于 11-07 07:52

    使用TAS5756做一款低音炮,需要隨時調節(jié)低通濾波寬帶,請問這些數據是怎么算出來的?

    你好,使用TAS5756做一款低音炮。需要隨時調節(jié)低通濾波寬帶,現在用purepath Console可以調節(jié)(low pass,butterworth模式),但是不知道BO,B1,B2,A1,A2的值怎么計算出來的,可否提供計算公式。謝謝
    發(fā)表于 10-28 06:09

    PCM1808從模擬信號輸入到I2S信號輸出的延遲時間是多少?怎樣計算出來

    PCM1808從模擬信號輸入到I2S信號輸出的延遲時間是多少,或者怎樣計算出來?我想選一個低延遲的音頻ADC是否還有別的更好推薦,謝謝。
    發(fā)表于 10-14 06:25

    TAS5805內部的硬件CRC8對應標準的哪種? CRC校驗的數據是offset和value值嗎?

    TAS5805 內部的硬件 CRC8 對應標準的哪種? CRC 校驗的數據是 offset 和 value 值嗎?用 PPC 加載文件計算出來的值和在線CRC計算工具計算的值不一樣。 測試校驗的數據: 58 18 00 80
    發(fā)表于 10-12 06:19

    LPV521 Vout是怎么計算得出的呢?

    麻煩分析一下如下電路圖的工作原理:Vout是怎么計算得出的呢? 圖中測量電流的范圍能達多少呢?怎么計算出來的呢? 圖中三極管工作在什么狀態(tài)?怎么進行選型呢? 謝謝!
    發(fā)表于 09-05 07:37

    LTSR 25-NP計算出來的電流誤差特別大,為什么?

    大家好,我現在使用的電流傳感是lem的LTSR 25-NP,其中電流和輸出電壓關系是:v=2.5+0.025*I,現在的AD采樣輸入范圍是0-3V,中間有一個放大倍數,但是即使這樣,電壓稍微波動一些,計算出來的電流誤差就特別大,請大家支招,謝謝!
    發(fā)表于 09-03 08:18

    st2.0的電機庫,hall電角度是怎么計算出來

    小弟最近在看電機庫,是利用定時3捕獲hall信號,但是看代碼不是很明白,為什么計算電角度跟hall值1,2,4無關呢,只是判斷3,5,6這幾個hall值 bPrevHallState
    發(fā)表于 07-23 08:30

    ESP8266_RTOS在線升級FOTA CRC遇到的疑問求解

    背景:用ESP8266模組,云端自己搭建(參考Upgrade.c文件)。 問題: 1. 做FOTA時,從Server端下載文件成功,但調用upgrade_crc_check函數時,計算出來
    發(fā)表于 07-15 06:25

    dd馬達偏差量怎么計算出來

    反映了電機的實際位置與期望位置之間的差異。 計算DD馬達偏差量的基本步驟如下: 確定期望位置:期望位置是系統(tǒng)希望電機達到的目標位置。這可以是用戶輸入的值,也可以是系統(tǒng)根據某種算法計算出來的值。 獲取實際位置:實際位置是電機當前的實際位置。這可
    的頭像 發(fā)表于 07-11 15:05 ?734次閱讀

    MCSDK5.4.8中STEVAL-ESC001V1過流保護里邊的current threshold是怎么計算出來的?

    MCSDK5.4.8中,STEVAL-ESC001V1過流保護里邊的 current threshold是怎么計算出來的? 在voltage threshold 0.6V時33.636Apk 0.3V時0.707Apk 這是什么公式算出來的? STEVAL-ESC001V
    發(fā)表于 05-21 07:26

    隔離開關的額定電流怎么計算出來

    隔離開關的額定電流是指設備或元件所能承受的最大連續(xù)工作電流。它是由許多因素決定的,包括電流容量、電壓容量、環(huán)境溫度和使用條件等。 隔離開關是一種常見的電氣設備,用于控制和切斷電路中的電流。它主要由可分離的觸點組成,可將電路從電源分離以實現安全控制。在選擇隔離開關時,需要考慮電流容量以確保設備的安全運行。 首先,我們需要了解什么是電流容量。電流容量是指設備或元件所能連續(xù)承受的電流大小。它是設備設計和制造
    的頭像 發(fā)表于 02-05 15:29 ?2235次閱讀
    主站蜘蛛池模板: 国产久久re6免费热在线 | 在线看片福利无码网址 | 国产又爽又黄又不遮挡视频 | 亚洲国产成人在线 | 久久囯产精品777蜜桃传媒 | china年轻小帅脸直播飞机 | 蜜桃成人在线 | 国产亚洲精品 在线视频 香蕉 | 无修肉动漫在线观看影片 | 欧美亚洲精品真实在线 | 日韩 亚洲 欧美 中文 高清 | 欧美xxxxx18| 亚洲视频精品在线观看 | 在线 | 果冻国产传媒61国产免费 | 午夜伦伦电影理论片费看 | 国产ts调教 | 99久久免费国产精精品 | 蜜桃99影院 | AAA级精品无码久久久国片 | 折磨比基尼美女挠肚子 | 男人和女人全黄一级毛片 | 午夜一个人在线观看完整版 | 妹妹的第一次有点紧 | 好男人视频免费高清在线观看www | 亚洲欧美韩国综合色 | 91精品国产色综合久久不 | 亚洲无线码一区在线观看 | 99久久国产露脸精品竹菊传煤 | 麻豆一区二区免费播放网站 | 亚洲九九视频 | 亚洲中文无码AV在线观看 | 色情内射少妇兽交 | 中文字幕一区在线观看视频 | 亚洲电影成人 成人影院 | 97精品少妇偷拍AV | 亚洲国产精品日本无码网站 | 欧美激情一区二区三区视频 | 小柔的性放荡羞辱日记 | 国语自产精品一区在线视频观看 | 高清国产mv视频在线观看 | 穿着丝袜被男生强行啪啪 |