詳解驅動開發中內核PE結構VA與FOA轉換 環球微動態
摘要:本文將探索內核中解析PE文件的相關內容。
本文分享自華為云社區《驅動開發:內核PE結構VA與FOA轉換》,作者: LyShark 。
本章將探索內核中解析PE文件的相關內容,PE文件中FOA與VA、RVA之間的轉換也是很重要的,所謂的FOA是文件中的地址,VA則是內存裝入后的虛擬地址,RVA是內存基址與當前地址的相對偏移,本章還是需要用到封裝的KernelMapFile()映射函數,在映射后對其PE格式進行相應的解析,并實現轉換函數。
(資料圖)
首先先來演示一下內存VA地址與FOA地址互相轉換的方式,通過使用WinHEX打開一個二進制文件,打開后我們只需要關注如下藍色注釋為映像建議裝入基址,黃色注釋為映像裝入后的RVA偏移。
通過上方的截圖結合PE文件結構圖我們可得知0000158B為映像裝入內存后的RVA偏移,緊隨其后的00400000則是映像的建議裝入基址,為什么是建議而不是絕對?別急后面慢來來解釋。
通過上方的已知條件我們就可以計算出程序實際裝入內存后的入口地址了,公式如下:
VA(實際裝入地址) = ImageBase(基址) + RVA(偏移) => 00400000 + 0000158B = 0040158B
找到了程序的OEP以后,接著我們來判斷一下這個0040158B屬于那個節區,以.text節區為例,下圖我們通過觀察區段可知,第一處橙色位置00000B44 (節區尺寸),第二處紫色位置00001000 (節區RVA),第三處00000C00 (文件對齊尺寸),第四處00000400 (文件中的偏移),第五處60000020 (節區屬性)。
得到了上方text節的相關數據,我們就可以判斷程序的OEP到底落在了那個節區中,這里以.text節為例子,計算公式如下:
虛擬地址開始位置:節區基地址 + 節區RVA => 00400000 + 00001000 = 00401000虛擬地址結束位置:text節地址 + 節區尺寸 => 00401000 + 00000B44 = 00401B44
經過計算得知 .text 節所在區間(401000 - 401B44) 你的裝入VA地址0040158B只要在區間里面就證明在本節區中,此處的VA地址是在401000 - 401B44區間內的,則說明它屬于.text節。
經過上面的公式計算我們知道了程序的OEP位置是落在了.text節,此時你興致勃勃的打開x64DBG想去驗證一下公式是否計算正確不料,這地址根本不是400000開頭啊,這是什么鬼?
上圖中出現的這種情況就是關于隨機基址的問題,在新版的VS編譯器上存在一個選項是否要啟用隨機基址(默認啟用),至于這個隨機基址的作用,猜測可能是為了防止緩沖區溢出之類的爛七八糟的東西。
為了方便我們調試,我們需要手動干掉它,其對應到PE文件中的結構為 IMAGE_NT_HEADERS -> IMAGE_OPTIONAL_HEADER -> DllCharacteristics 相對于PE頭的偏移為90字節,只需要修改這個標志即可,修改方式 x64:6081 改 2081 相對于 x86:4081 改 0081 以X86程序為例,修改后如下圖所示。
經過上面對標志位的修改,程序再次載入就能夠停在0040158B的位置,也就是程序的OEP,接下來我們將通過公式計算出該OEP對應到文件中的位置。
.text(節首地址) = ImageBase + 節區RVA => 00400000 + 00001000 = 00401000VA(虛擬地址) = ImageBase + RVA(偏移) => 00400000 + 0000158B = 0040158BRVA(相對偏移) = VA - (.text節首地址) => 0040158B - 00401000 = 58BFOA(文件偏移) = RVA + .text節對應到文件中的偏移 => 58B + 400 = 98B
經過公式的計算,我們找到了虛擬地址0040158B對應到文件中的位置是98B,通過WinHEX定位過去,即可看到OEP處的機器碼指令了。
接著我們來計算一下.text節區的結束地址,通過文件的偏移加上文件對齊尺寸即可得到.text節的結束地址400+C00= 1000,那么我們主要就在文件偏移為(98B - 1000)在該區間中找空白的地方,此處我找到了在文件偏移為1000之前的位置有一段空白區域,如下圖:
接著我么通過公式計算一下文件偏移為0xF43的位置,其對應到VA虛擬地址是多少,公式如下:
.text(節首地址) = ImageBase + 節區RVA => 00400000 + 00001000 = 00401000VPK(實際大小) = (text節首地址 - ImageBase) - 實際偏移 => 401000-400000-400 = C00VA(虛擬地址) = FOA(.text節) + ImageBase + VPK => F43+400000+C00 = 401B43
計算后直接X64DBG跳轉過去,我們從00401B44的位置向下全部填充為90(nop),然后直接保存文件。
再次使用WinHEX查看文件偏移為0xF43的位置,會發現已經全部替換成了90指令,說明計算正確。
到此文件偏移與虛擬偏移的轉換就結束了,那么這些功能該如何實現呢,接下來將以此實現這些轉換細節。
FOA轉換為VA:首先來實現將FOA地址轉換為VA地址,這段代碼實現起來很簡單,如下所示,此處將dwFOA地址0x84EC00轉換為對應內存的虛擬地址。
// 署名權// right to sign one"s name on a piece of work// PowerBy: LyShark// Email: me@lyshark.comNTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath){DbgPrint("hello lyshark.com \n");NTSTATUS status = STATUS_SUCCESS;HANDLE hFile = NULL;HANDLE hSection = NULL;PVOID pBaseAddress = NULL;UNICODE_STRING FileName = { 0 };// 初始化字符串RtlInitUnicodeString(&FileName, L"\\??\\C:\\Windows\\System32\\ntoskrnl.exe");// 內存映射文件status = KernelMapFile(FileName, &hFile, &hSection, &pBaseAddress);if (!NT_SUCCESS(status)){return 0;}// 獲取PE頭數據集PIMAGE_DOS_HEADER pDosHeader = (PIMAGE_DOS_HEADER)pBaseAddress;PIMAGE_NT_HEADERS pNtHeaders = (PIMAGE_NT_HEADERS)((PUCHAR)pDosHeader + pDosHeader->e_lfanew);PIMAGE_SECTION_HEADER pSection = IMAGE_FIRST_SECTION(pNtHeaders);PIMAGE_FILE_HEADER pFileHeader = &pNtHeaders->FileHeader;DWORD64 dwFOA = 0x84EC00;DWORD64 ImageBase = pNtHeaders->OptionalHeader.ImageBase;DWORD NumberOfSectinsCount = pNtHeaders->FileHeader.NumberOfSections;DbgPrint("鏡像基址 = %p | 節表數量 = %d \n", ImageBase, NumberOfSectinsCount);for (int each = 0; each < NumberOfSectinsCount; each++){DWORD64 PointerRawStart = pSection[each].PointerToRawData; // 文件偏移開始位置DWORD64 PointerRawEnds = pSection[each].PointerToRawData + pSection[each].SizeOfRawData; // 文件偏移結束位置// DbgPrint("文件開始偏移 = %p | 文件結束偏移 = %p \n", PointerRawStart, PointerRawEnds);if (dwFOA >= PointerRawStart && dwFOA <= PointerRawEnds){DWORD64 RVA = pSection[each].VirtualAddress + (dwFOA - pSection[each].PointerToRawData); // 計算出RVADWORD64 VA = RVA + pNtHeaders->OptionalHeader.ImageBase; // 計算出VADbgPrint("FOA偏移 [ %p ] --> 對應VA地址 [ %p ] \n", dwFOA, VA);}}ZwUnmapViewOfSection(NtCurrentProcess(), pBaseAddress);ZwClose(hSection);ZwClose(hFile);Driver->DriverUnload = UnDriver;return STATUS_SUCCESS;}
運行效果如下所示,此處之所以出現兩個結果是因為沒有及時返回,一般我們取第一個結果就是最準確的;
VA轉換為FOA:將VA內存地址轉換為FOA文件偏移,代碼與如上基本保持一致。
// 署名權// right to sign one"s name on a piece of work// PowerBy: LyShark// Email: me@lyshark.comNTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath){DbgPrint("hello lyshark.com \n");NTSTATUS status = STATUS_SUCCESS;HANDLE hFile = NULL;HANDLE hSection = NULL;PVOID pBaseAddress = NULL;UNICODE_STRING FileName = { 0 };// 初始化字符串RtlInitUnicodeString(&FileName, L"\\??\\C:\\Windows\\System32\\ntoskrnl.exe");// 內存映射文件status = KernelMapFile(FileName, &hFile, &hSection, &pBaseAddress);if (!NT_SUCCESS(status)){return 0;}// 獲取PE頭數據集PIMAGE_DOS_HEADER pDosHeader = (PIMAGE_DOS_HEADER)pBaseAddress;PIMAGE_NT_HEADERS pNtHeaders = (PIMAGE_NT_HEADERS)((PUCHAR)pDosHeader + pDosHeader->e_lfanew);PIMAGE_SECTION_HEADER pSection = IMAGE_FIRST_SECTION(pNtHeaders);PIMAGE_FILE_HEADER pFileHeader = &pNtHeaders->FileHeader;DWORD64 dwVA = 0x00007FF6D3389200;DWORD64 ImageBase = pNtHeaders->OptionalHeader.ImageBase;DWORD NumberOfSectinsCount = pNtHeaders->FileHeader.NumberOfSections;DbgPrint("鏡像基址 = %p | 節表數量 = %d \n", ImageBase, NumberOfSectinsCount);for (DWORD each = 0; each < NumberOfSectinsCount; each++){DWORD Section_Start = ImageBase + pSection[each].VirtualAddress; // 獲取節的開始地址DWORD Section_Ends = ImageBase + pSection[each].VirtualAddress + pSection[each].Misc.VirtualSize; // 獲取節的結束地址DbgPrint("Section開始地址 = %p | Section結束地址 = %p \n", Section_Start, Section_Ends);if (dwVA >= Section_Start && dwVA <= Section_Ends){DWORD RVA = dwVA - pNtHeaders->OptionalHeader.ImageBase; // 計算RVADWORD FOA = pSection[each].PointerToRawData + (RVA - pSection[each].VirtualAddress); // 計算FOADbgPrint("VA偏移 [ %p ] --> 對應FOA地址 [ %p ] \n", dwVA, FOA);}}ZwUnmapViewOfSection(NtCurrentProcess(), pBaseAddress);ZwClose(hSection);ZwClose(hFile);Driver->DriverUnload = UnDriver;return STATUS_SUCCESS;}
運行效果如下所示,此處沒有出現想要的結果是因為我們當前的VA內存地址并非實際裝載地址,僅僅是PE磁盤中的地址,此處如果換成內存中的PE則可以提取出正確的結果;
RVA轉換為FOA:將相對偏移地址轉換為FOA文件偏移地址,此處僅僅只是多了一步pNtHeaders->OptionalHeader.ImageBase + dwRVARVA轉換為VA的過程其轉換結果與VA轉FOA一致。
// 署名權// right to sign one"s name on a piece of work// PowerBy: LyShark// Email: me@lyshark.comNTSTATUS DriverEntry(IN PDRIVER_OBJECT Driver, PUNICODE_STRING RegistryPath){DbgPrint("hello lyshark.com \n");NTSTATUS status = STATUS_SUCCESS;HANDLE hFile = NULL;HANDLE hSection = NULL;PVOID pBaseAddress = NULL;UNICODE_STRING FileName = { 0 };// 初始化字符串RtlInitUnicodeString(&FileName, L"\\??\\C:\\Windows\\System32\\ntoskrnl.exe");// 內存映射文件status = KernelMapFile(FileName, &hFile, &hSection, &pBaseAddress);if (!NT_SUCCESS(status)){return 0;}// 獲取PE頭數據集PIMAGE_DOS_HEADER pDosHeader = (PIMAGE_DOS_HEADER)pBaseAddress;PIMAGE_NT_HEADERS pNtHeaders = (PIMAGE_NT_HEADERS)((PUCHAR)pDosHeader + pDosHeader->e_lfanew);PIMAGE_SECTION_HEADER pSection = IMAGE_FIRST_SECTION(pNtHeaders);PIMAGE_FILE_HEADER pFileHeader = &pNtHeaders->FileHeader;DWORD64 dwRVA = 0x89200;DWORD64 ImageBase = pNtHeaders->OptionalHeader.ImageBase;DWORD NumberOfSectinsCount = pNtHeaders->FileHeader.NumberOfSections;DbgPrint("鏡像基址 = %p | 節表數量 = %d \n", ImageBase, NumberOfSectinsCount);for (DWORD each = 0; each < NumberOfSectinsCount; each++){DWORD Section_Start = pSection[each].VirtualAddress; // 計算RVA開始位置DWORD Section_Ends = pSection[each].VirtualAddress + pSection[each].Misc.VirtualSize; // 計算RVA結束位置if (dwRVA >= Section_Start && dwRVA <= Section_Ends){DWORD VA = pNtHeaders->OptionalHeader.ImageBase + dwRVA; // 得到VA地址DWORD FOA = pSection[each].PointerToRawData + (dwRVA - pSection[each].VirtualAddress); // 得到FOADbgPrint("RVA偏移 [ %p ] --> 對應FOA地址 [ %p ] \n", dwRVA, FOA);}}ZwUnmapViewOfSection(NtCurrentProcess(), pBaseAddress);ZwClose(hSection);ZwClose(hFile);Driver->DriverUnload = UnDriver;return STATUS_SUCCESS;}
運行效果如下所示;
點擊關注,第一時間了解華為云新鮮技術~
圖片
-
詳解驅動開發中內核PE結構VA
屬羊女的命運不好是真的嗎
流通協會:4月二手車交易146
-
全球觀點:【物流618】申通6
重慶油菜喜獲豐收 總產量實
焦點熱門:ads是什么意思怎么
中方要美拿出誠意,減少周邊
交銀施羅德基金管理有限公司
世界動態:電動焦慮下的日系
-
熱點評!智能表面產品和新型
安溪:讓農家書屋成為鄉村的
每日動態!2023高考數學開考
環球關注:世茂能源06月07日
天天新資訊:加拿大野火煙霧
每日觀點:蘋果手機通話記錄
-
漣源市市場監督管理局:247
每日訊息!廣聚能源:公司主
【新要聞】豐原藥業:公司沒
2個8相乘的積是多少_2個7的
23安徽債61今日發布發行公告
深圳多部門開通高考服務熱線
精彩推送
- BIAME汽車制博會上驚現“吸睛王”,跨越為汽車產業高質量發展“劃重點”
- 詳解驅動開發中內核PE結構VA與FOA轉換 環球微動態
- 天天看熱訊:白酒概念板塊漲0.19% 順鑫農業漲3.34%居首
- 全球熱頭條丨本月底前,北京疾控中心新址完成基坑墊層施工
- 當前關注:收評:滬指午后拉升收漲0.49% 建筑裝飾板塊領漲
- 易實精密北交所上市首日漲39% 募1.08億金元證券保薦|環球快看點
- 鋰電池板塊跌0.57% 日播時尚漲10%居首
- 生物醫藥板塊跌1.35% 退市輔仁漲7.04%居首
- 廣東自貿區固定資產投資累計超1萬億元|全球動態
- 貴州發揮金融“活水”優勢支持消費恢復擴大 全球今亮點
- 全國首條長距離新能源運輸試點項目落地山西臨汾
- TechInsights:全球5G智能手機出貨量同比增長3.9%|快訊
- 全球速讀:5月份國內汽車投訴量逾1.3萬宗創新高 自主品牌插電混動車型投訴量環比漲超3倍
- 智能報警記錄儀亮相中國國際社會公共安全產品博覽會
- 當前視點!一線城市房地產限購 應適時優化調整
- 環球熱消息:藝術特色學校創建自查報告(藝術特色)
- 弘視科技黑科技亮相2023PT|全息互動 4K顯示 毫秒延時
- 廣發基金王瑞冬:醫療領域關注“三低長尾行業龍頭”
- 李云澤:抓緊恢復和擴大有效需求關鍵手 持續優化和改善金融服務|每日焦點
- 全球快消息!廊坊大力發展現代商貿物流產業
- 多地下調首套房貸利率至4%以下
- 李海濤離任蜂巢添盈純債
- 世界實時:八仙過海是神話故事嗎
- 屬羊女的命運不好是真的嗎 1967年屬羊女命運婚姻|世界關注
- 全國“三夏”麥收進度過半 已收冬小麥達1.64億畝_環球播報
- vivo手機電池不耐用怎么辦教你一招(vivo手機電池不耐用怎么辦)
- 微讀圣經官方網站_圣經電腦版官方|天天快看
- 美麗的神話歌詞完整版_美麗的神話歌詞_世界熱頭條
- 流通協會:4月二手車交易146萬輛,市場呈下沉趨勢 每日視訊
- 食道癌擴散快嗎能治好嗎_食道癌擴散快嗎
- 短訊!護航逐夢,海底撈開展愛心助考行動
- 天天動態:江鈴E400電池包屢修屢壞無法解決 廠家稱未達到換新條件
- 國旗旗桿標準尺寸是多少_關于國旗旗桿標準尺寸介紹
- 全球觀點:【物流618】申通618第一波單日攬收6000萬件
- 焦點短訊!錫林郭勒盟發放內蒙古自治區首筆“帶押過戶”住房公積金貸款
- 每日快看:廣東新豐:厚植生態底色 提升綠色顏值
- 每日速看!創新經營方式 促進農民增收——山西省開展農業生產托管服務紀略
- 【熱聞】四川:今年前5月完成投資1447.5億元
- 洲際、萬達布局環球影城,強IP主題樂園如何做好酒店產品?_世界快看點
- 推動文旅賦能鄉村振興,湖南發布最新措施
- 浙江蒼南構建自然保護地新體系
- 午評:三大指數早間低位震蕩 豬肉板塊漲幅居前
- 廣汽本田最新公益短片《美在流溪——為生態保護注入源頭活水》上線 天天熱頭條
- 環球今日訊!國家林草局發布黃河上中游天然林保護修復效益監測國家報告
- 榆林治沙帶來生態空間顏色之變
- 空調使用旺季來臨,教你找到正規的大金空調官方售后維修中心
- 底部放量下跌意味著什么?底部放量下跌是好還是壞?
- 人民幣兌美元中間價報7.1280 調貶84個基點
- 底部放量滯漲說明什么?底部放量注意建倉什么意思?
- 環球新動態:福建華通銀行被罰97萬 未按規定報送可疑交易報告等
- 標準普爾500指數實時行情介紹?標準普爾500指數和道瓊斯指數區別在哪?
- 標準普爾500指數是什么意思?標準普爾500指數計算方法為多少?
- 轉增股本是什么意思?已發行股本是什么意思非上市公司?
- 重慶油菜喜獲豐收 總產量實現連續16年增長|天天最資訊
- 股本是什么意思?股本和實收資本的區別是什么?
- 萬達集團回應“19億股權被凍結”:正通過法律途徑申訴
- 天天看點:桂陽:“六月楊梅滿枝頭 “采梅止渴”正當時
- 交通銀行:林驊劉建軍王文進高管任職資格獲核準
- 我愛我家:預計今年二手房市場交易量將迎來穩健復蘇
- 環球觀速訊丨請注意!多家銀行今起下調存款利率,5年期存款利率2.5%
- 觀天下!武漢7宗地塊集中拍賣時間推遲至6月26日 起拍總價42.26億元
- 未來鋰供給結構的三大發展方向
- 5月地產銷售繼續分化 優質房企增長好于平均值
- 環球簡訊:新增化學測試指標 根據嬰童頭圍設計尺寸、重量 嬰童運動頭盔將有標可依
- 拍領導牽手的攝影師立功還是擔責?公司領導拍員工照片可以嗎?
- 除息日買入股票劃算嗎?除權除息日應該買入還是賣出?
- 石油與化工指數漲跌互現
- 環球關注:江蘇出臺14條措施推動外貿穩規模優結構
- 除息日股票為什么會跌?除息日是什么意思?除息日特點介紹
- 納指納斯達克指數開盤時間介紹?納指基金賣出凈值怎么算?
- 納指100實時行情一覽 納指和etf哪個好一點
- 納指期貨和納斯達克有關系嗎?納指是什么意思?
- 納指ETF是指什么?納指etf交易規則及費用是什么?
- 焦點熱門:ads是什么意思怎么理解ads是什么意思
- 股票交易系統投票怎么操作?股票交易系統投票有什么用?
- 佳能60d配什么鏡頭最實用(佳能60d能配70-200嗎?)-世界速看料
- 首超日本,我國成世界第一大汽車出口國!_聚看點
- 集資詐騙的定罪標準是什么?集資詐騙案退賠程序怎么走?
- etf期權和股指期權區別是什么?上證etf期權開戶條件有哪些?
- 每日熱訊!今日上市:雙元科技、易實精密
- 50etf期權持倉量多少說明什么?etf期權交易規則及費用詳解
- 中方要美拿出誠意,減少周邊駐軍,你們的艦機,離我們領土太近了|全球報資訊
- 他們用“超級天平”給原子核稱“體重”
- 軋空行情是上漲還是下跌?軋空的反義詞是什么?
- 軋空什么意思?軋空怎么造句?游戲驛站軋空事件是咋回事?
- 全國“三夏”麥收進度過半 當前頭條
- 中國人壽財產保險是國企嗎?中國人壽財產保險和中國人保區別在哪?
- 環球視點!別總想著“培育”消費者
- 全國首個百度文心千帆大模型賦能中心基地落地無錫梁溪
- 為什么說帶血的流量該好好管管?奪命直播亂象是什么意思?
- 金帝股份過會:今年IPO過關第136家 國信證券過2單
- 存款利率向下 大額存單熱銷 環球報道
- 電視劇坐莊講了什么?坐莊賭博構成什么犯罪?
- 李顯揚現在哪里任職?李顯揚學的什么專業?新加坡李顯揚個人簡介
- 光伏逆變器發展面臨三大挑戰 即時看
- 2023年即將倒閉的銀行有哪些?國家不允許倒閉的銀行有哪幾個?
- 天天觀速訊丨做好“加減法” 增糧有實招——部分糧食主產區一線掃描
- 工信部:前瞻布局下一代互聯網 全面推進6G技術研發 世界熱門
- 環球熱點評!林業上“云” 生態保護有“數”——數字技術守護綠水青山一線見聞
- 萬億美元發債潮將至 美國金融體系或醞釀流動性風險