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

當前位置:首頁 > 嵌入式 > 嵌入式客棧
[導讀]看了很多關于linux內存管理的文章還是云里霧里,聽了很多關于linux內存管理的課程還是一頭霧水。其實很多時候造成不懂的原因不是資料太少,恰恰是資料太多,而且各個內核版本的差異,32位64位的不同,文章的胡編亂造等都給讀者帶來疑惑。本著對內存深度剖析的

看了很多關于linux內存管理的文章還是云里霧里,聽了很多關于linux內存管理的課程還是一頭霧水。其實很多時候造成不懂的原因不是資料太少,恰恰是資料太多,而且各個內核版本的差異,32位64位的不同,文章的胡編亂造等都給讀者帶來疑惑。本著對內存深度剖析的態(tài)度,希望以版本kernel-4.14,架構AARCH64為專題做個內存管理的架構性整理。

這篇文章我們先來看下linux在啟動過程中的初始化。


創(chuàng)建啟動頁表:

在匯編代碼階段的head.S文件中,負責創(chuàng)建映射關系的函數是create_page_tablescreate_page_tables函數負責identity mapping和kernel image mapping。

  • identity map:是指把idmap_text區(qū)域的物理地址映射到相等的虛擬地址上,這種映射完成后,其虛擬地址等于物理地址。idmap_text區(qū)域都是一些打開MMU相關的代碼。

  • kernel image map:將kernel運行需要的地址(kernel txt、rodata、data、bss等等)進行映射。

arch/arm64/kernel/head.S:ENTRY(stext) bl preserve_boot_args bl el2_setup // Drop to EL1, w0=cpu_boot_mode adrp x23, __PHYS_OFFSET and x23, x23, MIN_KIMG_ALIGN - 1 // KASLR offset, defaults to 0 bl set_cpu_boot_mode_flag bl __create_page_tables /* * The following calls CPU setup code, see arch/arm64/mm/proc.S for * details. * On return, the CPU will be ready for the MMU to be turned on and * the TCR will have been set. */ bl __cpu_setup // initialise processor b __primary_switchENDPROC(stext)

__create_page_tables主要執(zhí)行的就是identity map和kernel image map:

 __create_page_tables:...... create_pgd_entry x0, x3, x5, x6 mov x5, x3 // __pa(__idmap_text_start) adr_l x6, __idmap_text_end // __pa(__idmap_text_end) create_block_map x0, x7, x3, x5, x6
/* * Map the kernel image (starting with PHYS_OFFSET). */ adrp x0, swapper_pg_dir mov_q x5, KIMAGE_VADDR + TEXT_OFFSET // compile time __va(_text) add x5, x5, x23 // add KASLR displacement create_pgd_entry x0, x5, x3, x6 adrp x6, _end // runtime __pa(_end) adrp x3, _text // runtime __pa(_text) sub x6, x6, x3 // _end - _text add x6, x6, x5 // runtime __va(_end) create_block_map x0, x7, x3, x5, x6 ......

其中調用create_pgd_entry進行PGD及所有中間level(PUD, PMD)頁表的創(chuàng)建,調用create_block_map進行PTE頁表的映射。關于四級頁表的關系如下圖所示,這里就不進一步解釋了。


匯編結束后的內存映射關系如下圖所示:


當執(zhí)行完上面的map之后,MMU就已經打開了并且開始進入C代碼運行階段,那么下一步就要對dtb進行映射了。


fixmap區(qū)之dtb map:

在執(zhí)行setup_arch中,會最先進行early_fixmap_init(),這個函數就是用來map dtb的,但是它只會建立dtb對應的這段物理地址中間level的頁表entry,而最后一個level的頁表映射則通過setup_machine_fdt函數里的fixmap_remap_fdt來創(chuàng)建。

void *__init fixmap_remap_fdt(phys_addr_t dt_phys){ void *dt_virt; int size;
dt_virt = __fixmap_remap_fdt(dt_phys, &size, PAGE_KERNEL_RO); if (!dt_virt) return NULL;
memblock_reserve(dt_phys, size); return dt_virt; }

fixmap_remap_fdt主要是為fdt建立地址映射,在該函數的最后,順便就調用memblock_reserve保留了該段內存。

可以看出dtb的映射采用的是fixmap,所謂fixmap就是固定映射,它需要我們明確的知道想要映射的物理地址,并把這段地址映射到想要映射的虛擬地址上。當然這里固定映射還有些片面,因為在fixmap機制實現上,也有支持動態(tài)分配虛擬地址的功能,這個功能主要用于臨時fixmap映射(這個臨時映射就是用來執(zhí)行early ioremap使用的。),而dtb的映射屬于永久映射。


fixmap區(qū)之early ioremap:

對于一些硬件需要在內存管理系統(tǒng)起來之前就要工作的,我們就可以使用這種機制來映射內存給這些硬件driver使用。各個模塊在使用完early ioremap的地址后,需要盡快把這段映射的虛擬地址釋放掉,這樣才能反復被其他模塊繼續(xù)申請使用。

early_ioremap_init會調用early_ioremap_setup:

可見它的實現是依賴fixmap的,所以它必須要在early_fixmap_init之后才能運行。

注意:如果想要在伙伴系統(tǒng)初始化之前進行設備寄存器的訪問,那么可以考慮early IO remap機制。

至此我們已經知道dtb和early ioremap都是在fixmap區(qū)的,如下圖:


系統(tǒng)內存的布局:

完成dtb的map之后,內核可以訪問這一段的內存了,通過解析dtb中的內容,內核可以勾勒出整個內存布局的情況,為后續(xù)內存管理初始化奠定基礎。這一步主要在setup_machine_fdt中完成。這里就不看代碼了,其調用流程是:setup_machine_fdt->early_init_dt_scan->early_init_dt_scan_nodes

就像注釋中所示內核根據dtb的不同node勾勒出choosen node,root node,memory node相應內存區(qū)域。

除了這3個node,還有一個reserved-memory node,它是在上面講到dtb map的時候fixmap_remap_fdt函數做的。下面我們看下這4個node的具體實現。

  • choosen node

該節(jié)點有一個bootargs屬性,該屬性定義了內核的啟動參數,比如mem= xx,此外,還處理initrd相關的property,并保存在initrd_start和initrd_end這兩個全局變量中。

  • root node

與內存無關,暫時不詳述,以后有機會講到device tree系列再詳述。

  • memory node

通過memblock_add加入到memblock.memory對應的memblock_type鏈表中進行管理。


接下來到arm64_memblock_init函數:

void __init arm64_memblock_init(void){...... memblock_reserve(__pa_symbol(_text), _end - _text); 1.kernel image保留區(qū) #ifdef CONFIG_BLK_DEV_INITRD if (initrd_start) { memblock_reserve(initrd_start, initrd_end - initrd_start); 2.initrd保留區(qū) /* the generic initrd code expects virtual addresses */ initrd_start = __phys_to_virt(initrd_start); initrd_end = __phys_to_virt(initrd_end); } #endif early_init_fdt_scan_reserved_mem(); 3.dts中配置為保留的區(qū)域......}
  1. reserve內核代碼、數據區(qū)等(_text到_end那一段,具體的內容可以參考內核鏈接腳本)

  2. 保留initital ramdisk image區(qū)域(從initrd_start到initrd_end區(qū)域)

  3. reserved-memory node 如下所示:


完成:

通過上面的一系列操作,需要動態(tài)管理的內存已經被放到了memory type和reserved type這兩個region中了,現在內存已經被memblock模塊所管理了,這只是啟動后的第一步,后續(xù)內存才會加入到伙伴系統(tǒng)去管理。

END

往期精彩推薦,點擊即可閱讀




▲Linux內核中I2C總線及設備長啥樣?  [墻裂推薦]
學習AI之機器學習概念篇
手把手教系列之IIR數字濾波器設計實現

麻煩給個好看!

免責聲明:本文內容由21ic獲得授權后發(fā)布,版權歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯系我們,謝謝!

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

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

關鍵字: 驅動電源

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

關鍵字: 工業(yè)電機 驅動電源

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

關鍵字: 驅動電源 照明系統(tǒng) 散熱

根據LED驅動電源的公式,電感內電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關鍵字: LED 設計 驅動電源

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

關鍵字: 電動汽車 新能源 驅動電源

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

關鍵字: 發(fā)光二極管 驅動電源 LED

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

關鍵字: LED 驅動電源 功率因數校正

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

關鍵字: LED照明技術 電磁干擾 驅動電源

開關電源具有效率高的特性,而且開關電源的變壓器體積比串聯穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現在的LED驅動電源

關鍵字: LED 驅動電源 開關電源

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

關鍵字: LED 隧道燈 驅動電源
關閉