Linux內核系列—12.c.操作系統開發之從Loader進入內核

来源:http://www.cnblogs.com/joey-hua/archive/2016/04/17/5401760.html
-Advertisement-
Play Games

實際上,我們要做的工作是根據內核的Program header table的信息進行類似下麵這個C語言語句的記憶體複製: memcpy(p_vaddr, BaseOfLoaderPhyAddr+p_offset, p_filesz); 複製可能不止一次,如果Program header有n個,複製就進 ...


實際上,我們要做的工作是根據內核的Program header table的信息進行類似下麵這個C語言語句的記憶體複製:

memcpy(p_vaddr, BaseOfLoaderPhyAddr+p_offset, p_filesz);

複製可能不止一次,如果Program header有n個,複製就進行n次。

每一個Program header都描述一個段,語句中的P_offset為段在文件中的偏移,p_filesz為段在文件中的長度,p_vaddr為段在記憶體中的虛擬地址。

由ld生成的可執行文件中p_vaddr的值總是一個類似於0x8048XXX的值,至少我們的例子中是一個這樣的值。可是我們啟動分頁機制時地址都是對等映射的,記憶體地址0x8048XXX已經處在128MB記憶體以外(128MB的十六進位表示是0x8000000),如果電腦的記憶體小於128MB的話,這個地址顯然已經超出了記憶體大小。

即便電腦有足夠大的記憶體,顯然,我們也不能讓編譯器來決定內核載入到什麼地方。解決它有兩個辦法,一是通過修改頁表讓0x8048XXX映射到較低的地址,另一種方法就是通過修改ld的選項讓它生成的可執行代碼中p_vaddr的值變小。

nasm -f elf -o kernel.o kernel.asm

ld -m elf_i386 -s -Ttext 0x30400 -o kernel.bin kernel.o

程式的入口地址就變成0x30400了,ELF header等信息會位於0x30400之前。此時的ELF header和Program header table的情況如下表所示:

根據上表,我們應該這樣放置內核:

memcpy(30000h, 90000h+0, 40Dh);

也就是說,我們應該把文件從開頭開始40Dh位元組的內容放到記憶體30000h處。由於程式的入口在30400h處,所以從這裡就可以看出,實際上代碼只有0Dh+1個位元組。下麵是Kernel.bin的內容:

上面被星號省去的部分都是0.從中可以看出,從400h到40Dh是僅有的代碼,0xEBFE正是代碼最後的“jmp $”。

下麵的代碼實現了將Kernel.bin根據ELF文件信息轉移到正確的位置。它很簡單,找出每個Program header,根據其信息進行記憶體複製:

; InitKernel ---------------------------------------------------------------------------------
; 將 KERNEL.BIN 的內容經過整理對齊後放到新的位置
; 遍歷每一個 Program Header,根據 Program Header 中的信息來確定把什麼放進記憶體,放到什麼位置,以及放多少。
; --------------------------------------------------------------------------------------------
InitKernel:
        xor   esi, esi
        mov   cx, word [BaseOfKernelFilePhyAddr+2Ch];`. ecx <- pELFHdr->e_phnum
        movzx ecx, cx                               ;/
        mov   esi, [BaseOfKernelFilePhyAddr + 1Ch]  ; esi <- pELFHdr->e_phoff
        add   esi, BaseOfKernelFilePhyAddr;esi<-OffsetOfKernel+pELFHdr->e_phoff
.Begin:
        mov   eax, [esi + 0]
        cmp   eax, 0                      ; PT_NULL
        jz    .NoAction
        push  dword [esi + 010h]    ;size ;`.
        mov   eax, [esi + 04h]            ; |
        add   eax, BaseOfKernelFilePhyAddr; | memcpy((void*)(pPHdr->p_vaddr),
        push  eax		    ;src  ; |      uchCode + pPHdr->p_offset,
        push  dword [esi + 08h]     ;dst  ; |      pPHdr->p_filesz;
        call  MemCpy                      ; |
        add   esp, 12                     ;/
.NoAction:
        add   esi, 020h                   ; esi += pELFHdr->e_phentsize
        dec   ecx
        jnz   .Begin

        ret
; InitKernel ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

接下來就是向內核跳轉

;***************************************************************
	jmp	SelectorFlatC:KernelEntryPointPhyAddr	; 正式進入內核 *
	;***************************************************************

KernelEntryPointPhyAddr定義在頭文件load.inc中,其值為0x30400.當然,它必須跟我們的ld的參數-Ttext指定的值是一致的。將來如果我們想將內核放在另外的位置,只需改動這兩個地方就可以了。

運行結果如下:

成功了,出現字元“K”,這表明我們的內核在執行了。Loader的使命圓滿結束。

 

一個碼農的日常 

源碼


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • AppCan MEAP企業移動應用平臺 AppCan MEAP企業移動應用平臺是為企業移動信息化戰略提供標準技術支撐的平臺級產品。AppCan MEAP提供整體的、開放標準的、具有前瞻性的移動應用技術方案,幫助企業高效低成本地完成移動應用的開發、測試、發佈、部署和管理工作,同時依然保持應用的高體驗性 ...
  • 1、volley 項目地址 https://github.com/smanikandan14/Volley-demo (1) JSON,圖像等的非同步下載;(2) 網路請求的排序(scheduling)(3) 網路請求的優先順序處理(4) 緩存(5) 多級別取消請求(6) 和Activity和生命周期的 ...
  • 字典是一種存儲相同類型多重數據的存儲器。每個值(value)都關聯獨特的鍵(key),鍵作為字典中的這個值數據的標識符。和數組中的數據項不同,字典中的數據項並沒有具體順序。 字典寫作Dictionary<Key, Value>。也可以寫作[Key: Value] 創建空字典 類型推斷寫作[:] 創建 ...
  • 預覽: 需要許可權: 1 <uses-permission android:name="com.android.launcher.permission.INSTALL_SHORTCUT" /> 配置文件:AndroidManifest.xml 在應用的閃屏頁面Activity的 oncreate方法調 ...
  • 一個Activity 對應 多個Fragment; 每一個類 extends Fragment , 一個Activity 可以同時顯示多個 Fragment; ...
  • 1、關於代理對象的設計小技巧 在設計一個類,需要通過代理和協議來從外部獲取需要的動態的數據。那麼在這裡設計使用代理會有兩種方法。 <第一種方法> 也是比較常見的: 在你設計的類中,聲明一個代理屬性 然後外部使用的時候 最後根據那個<...Protocol>協議,去遵循這個協議並實現協議的方法。 <第 ...
  • 本文轉載自345大神。。。。 "查看原文" 先上個圖形化界面GIT工具 Git 常用命令 git clone git remote git fetch git pull git push 1. git clone 遠程操作的第一步,通常是從遠程主機克隆一個版本庫,這時就要用到git clone命令。 ...
  • 每建一個Activity都要註冊許可權Manifest.xml但是有時候自動註冊好了,註意!不然的話是不能調用的!!!!!<activity android:name=".MainView"></activity>安卓Fragment的調用不用註冊! ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...