如何提取 x64 程式那些易失的方法參數

来源:https://www.cnblogs.com/huangxincheng/archive/2023/03/24/17250240.html
-Advertisement-
Play Games

一:背景 1. 講故事 最近經常遇到有朋友反饋,在 x64 環境下如何提取線程棧中的方法參數,熟悉 x64 調用協定的朋友應該知道,這種協定範圍下,方法的前四個參數都是用寄存器傳遞的,比如rcx,rdx,r8d,r9d 四個寄存器,由於寄存器存值的臨時性,它的值容易被後面的邏輯給徵用了,那這種情況下 ...


一:背景

1. 講故事

最近經常遇到有朋友反饋,在 x64 環境下如何提取線程棧中的方法參數,熟悉 x64 調用協定的朋友應該知道,這種協定範圍下,方法的前四個參數都是用寄存器傳遞的,比如rcx,rdx,r8d,r9d 四個寄存器,由於寄存器存值的臨時性,它的值容易被後面的邏輯給徵用了,那這種情況下還有沒有辦法提取出來呢? 說實話,全靠運氣,為什麼這麼說呢? 如果這個在方法的棧初始化過程中有臨時的保存線上程棧中的話,那恭喜你,可以成功給撈出來。

接下來通過一個小案例來深入的聊一下。

二:案例分析

1. 一個案例演示

為了方便講述,這裡我用 Marshal 在 ntheap 上分配堆塊,然後提取 Marshal.FreeHGlobal 方法的用戶句柄,參考代碼如下:


        static void Main(string[] args)
        {
            //1. 分配 堆塊
            IntPtr ptr = Marshal.AllocHGlobal(sizeof(int));
            Console.WriteLine("ptr= 0x{0:X2}", ptr);

            //2. 寫入數據
            var num = int.MaxValue;
            Marshal.WriteInt32(ptr, num);
            Console.WriteLine("num 已寫入 ptr= 0x{0:X2} 堆塊", ptr);
            Debugger.Break();

            //3. 釋放 堆塊
            Marshal.FreeHGlobal(ptr);
            Console.WriteLine("ptr= 0x{0:X2} 堆塊成功釋放", ptr);
        }

熟悉 ntheap 的朋友都知道,如果在調試的環境下使用 FreeHGlobal 方法會命中底層的 ntdll!RtlpValidateHeap 方法,只要在這地方下個斷點即可,參考代碼如下:


0:000> bp ntdll!RtlpValidateHeap
0:000> g
Breakpoint 0 hit
ntdll!RtlpValidateHeap:
00007ffe`8e92a784 48895c2410      mov     qword ptr [rsp+10h],rbx ss:00000021`2037e078=00007ffd00000000
0:000> k 10
 # Child-SP          RetAddr               Call Site
00 00000021`2037e068 00007ffe`8e9295f5     ntdll!RtlpValidateHeap
01 00000021`2037e070 00007ffe`8e855cc1     ntdll!RtlDebugFreeHeap+0x99
02 00000021`2037e0d0 00007ffe`8e855b74     ntdll!RtlpFreeHeap+0xc1
03 00000021`2037e280 00007ffe`8e8547b1     ntdll!RtlpFreeHeapInternal+0x464
04 00000021`2037e340 00007ffe`8c33934f     ntdll!RtlFreeHeap+0x51
05 00000021`2037e380 00007ffd`d4af5c7c     KERNELBASE!LocalFree+0x2f
06 00000021`2037e3c0 00007ffd`7b132a10     System_Private_CoreLib!System.Runtime.InteropServices.Marshal.FreeHGlobal+0x4c [/_/src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/Marshal.Windows.cs @ 144] 
07 00000021`2037e490 00007ffd`dacaae93     Example_18_1_1!Example_18_1_1.Program.Main+0xd0 [D:\skyfly\18.20230322\src\Example\Example_18_1_1\Program.cs @ 21] 
...

從代碼中可以看到當釋放堆塊時果然調用了這個函數,接下來有一個需求,我想知道 KERNELBASE!LocalFree 第一個參數到底是什麼,有朋友肯定想說,你可以取 rcx 寄存器呀,但不要忘了,此時代碼都跑到 ntdll!RtlpValidateHeap 方法了, rcx 中的值早就被其他方法給覆蓋了,那怎麼辦呢?

2. 還有希望嗎

有沒有希望真的看運氣了,這時候要詳細觀察 KERNELBASE!LocalFree 方法入口處的彙編代碼,看下它有沒有將 rcx 保存在棧中,如果真的有保存到棧中,那就萬幸了,有了思路之後說乾就乾。


0:000> u KERNELBASE!LocalFree
KERNELBASE!LocalFree:
00007ffe`8c339320 48895c2410      mov     qword ptr [rsp+10h],rbx
00007ffe`8c339325 4889742418      mov     qword ptr [rsp+18h],rsi
00007ffe`8c33932a 48894c2408      mov     qword ptr [rsp+8],rcx
00007ffe`8c33932f 57              push    rdi
00007ffe`8c339330 4883ec30        sub     rsp,30h
00007ffe`8c339334 488bd9          mov     rbx,rcx
00007ffe`8c339337 f6c308          test    bl,8
00007ffe`8c33933a 753f            jne     KERNELBASE!LocalFree+0x5b (00007ffe`8c33937b)

從彙編代碼看真的很萬幸,代碼將 rcx 保存到了 rsp+8 的棧位置,接下來急需要知道這裡的 rsp+8 指的是哪一塊記憶體地址?

3. rsp+8 到底指向哪裡

在 x64 平臺下,為了最大化的利用寄存器,方法棧幀使用一個 rsp 來標記棧空間,而不像 32bit 平臺用 ebpesp 兩個寄存器來聯合承載,參考 k 命令輸出。


0:000> k 8
 # Child-SP          RetAddr               Call Site
00 00000021`2037e068 00007ffe`8e9295f5     ntdll!RtlpValidateHeap
01 00000021`2037e070 00007ffe`8e855cc1     ntdll!RtlDebugFreeHeap+0x99
02 00000021`2037e0d0 00007ffe`8e855b74     ntdll!RtlpFreeHeap+0xc1
03 00000021`2037e280 00007ffe`8e8547b1     ntdll!RtlpFreeHeapInternal+0x464
04 00000021`2037e340 00007ffe`8c33934f     ntdll!RtlFreeHeap+0x51
05 00000021`2037e380 00007ffd`d4af5c7c     KERNELBASE!LocalFree+0x2f
06 00000021`2037e3c0 00007ffd`7b132a10     System_Private_CoreLib!System.Runtime.InteropServices.Marshal.FreeHGlobal+0x4c [/_/src/libraries/System.Private.CoreLib/src/System/Runtime/InteropServices/Marshal.Windows.cs @ 144] 
07 00000021`2037e490 00007ffd`dacaae93     Example_18_1_1!Example_18_1_1.Program.Main+0xd0 [D:\skyfly\18.20230322\src\Example\Example_18_1_1\Program.cs @ 21] 

接下來的問題是: Child-SP 和 KERNELBASE!LocalFree 方法中的 rsp 到底是什麼關係? 要回答這個問題,需要非常清楚 Child-SP 是如何標記棧幀的,畫個圖如下:

從圖中可以清晰的看到:Child-SP 標記的是子方法中第一個參數的位置,而方法入口處的 RSP 指向的是該方法的返回地址 RIP 的位置,比 Child-SP 小一個指針單元。

有朋友可能要問為什麼是 RIP 的位置,這是因為彙編的 call 指令會隱式的如下執行。


PUSH RIP
SUB ESP,8

有了這些基礎之後,接下來就好辦了,計算公式為:

rsp = Child-SP - 0x8

那麼 rsp + 0x8 = Child-SP - 0x8 + 0x8 = Child-SP = 000000212037e3c0,接下來用 windbg 來驗證下。


0:000> dp 000000212037e3c0 L1
00000021`2037e3c0  00000141`a81f1f20
0:000> !heap -x 00000141`a81f1f20
Entry             User              Heap              Segment               Size  PrevSize  Unused    Flags
-------------------------------------------------------------------------------------------------------------
00000141a81f1f10  00000141a81f1f20  00000141a8100000  00000141a8100000        40        90        3c  busy extra fill 

大家再回頭看下 Console 界面的輸出,果然就是我苦苦尋求的 ptr= 0x141A81F1F20 地址。

三:總結

這是一篇非常有用的經驗分享帖,相信你在dump分析中肯定會用的上,總的來說,由於方法參數是通過寄存器傳遞的,能不能成功撈取需要你仔細觀察彙編代碼才能知道。

世間美好,相信的人都能得到。

圖片名稱
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • [NOIP2002 普及組] 選數 洛谷傳送門 點擊查看題目 題目描述 已知 n 個整數 x1,x2,.....,xn,以及 1 個整數 k(k<n)。從 n 個整數中任選 k 個整數相加,可分別得到一系列的和。例如當 n=4,k=3,4 個整數分別為 3,7,12,19 時,可得全部的組合與它們的 ...
  • @ 先看一下導出的整體效果(如下圖),其中標註的區域都是通過後臺動態生成的: 一、先在Word中建立好表格模板 1.1、參數創建方法(Word和WPS) 1.1.1、Office中Word域的創建 1.1.1.1、選中指定的單元格 -> 點擊頭部工具欄中的”插入“ -> 選擇 ”文檔部件“ -> 選 ...
  • 一、什麼是敏感詞過濾? 敏感詞過濾是一種處理網路內容的技術,可以檢測和過濾出網路中的敏感/違禁辭彙。它通過給定的關鍵字或字元串,判斷網路內容是否包含某些敏感信息,從而防止違反法律法規的信息流通。 通常,可以使用兩種方法來過濾敏感詞: 黑名單過濾:即定義一個黑名單,將所有敏感詞擇記錄在其中,然後對輸入 ...
  • WPF(Windows Presentation Foundation)是由微軟開發的桌面應用程式框架,用於創建現代化、高度交互和具有視覺吸引力的用戶界面。它是 .NET Framework 的一部分,提供了一種基於 XAML(Extensible Application Markup Langua... ...
  • 前言 在程式設計中,我們會遇到各種各樣的異常問題,一個好的異常處理解決方案能夠幫助開發者快速的定位問題,也能夠給用戶更好的用戶體驗。 異常處理的幾種方式 1、通過異常過濾器捕獲異常進行處理 2、自定義異常處理中間件 在這裡我選擇自定義異常處理中間件,中間件依托於請求管道運行,並且中間件的執行是有序的 ...
  • 1.可空類型修飾符(?) 眾所周知,在C#中引用類型可以使用一個null引用來表示一個不存在的值,比如 string str = null 是正確的; 但是值類型卻不能為空,比如 int k = null 那麼編譯器就會報錯; 為了讓值類型也可以為空,就需要使用可空類型,即用可空類型修飾符 "?" ...
  • 泛型的學習 一、泛型的引入 泛型 泛:寬泛的--不確定的; 型:類型 不確定的類型 無處不在的 調用普通方法的時候,參數類型在聲明的時候就確定了,調用按照類型傳遞參數即可 a. 如果有100個類型 100個方法?--很累 b. 有沒有能夠做一個方法可以能夠滿足不同類型的需求呢? 傳統方法 publi ...
  • WebSocket 1.基於Html5,IIS8.0版本以上,前端代碼和伺服器都必須支持WebSocket才能使用; 2.請求必須以WS:開頭 下麵是後臺接收前端websocket申請的方法: /// <summary> /// WebSocket建立鏈接的方法 /// </summary> /// ...
一周排行
    -Advertisement-
    Play Games
  • GoF之工廠模式 @目錄GoF之工廠模式每博一文案1. 簡單說明“23種設計模式”1.2 介紹工廠模式的三種形態1.3 簡單工廠模式(靜態工廠模式)1.3.1 簡單工廠模式的優缺點:1.4 工廠方法模式1.4.1 工廠方法模式的優缺點:1.5 抽象工廠模式1.6 抽象工廠模式的優缺點:2. 總結:3 ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 本章將和大家分享ES的數據同步方案和ES集群相關知識。廢話不多說,下麵我們直接進入主題。 一、ES數據同步 1、數據同步問題 Elasticsearch中的酒店數據來自於mysql資料庫,因此mysql數據發生改變時,Elasticsearch也必須跟著改變,這個就是Elasticsearch與my ...
  • 引言 在我們之前的文章中介紹過使用Bogus生成模擬測試數據,今天來講解一下功能更加強大自動生成測試數據的工具的庫"AutoFixture"。 什麼是AutoFixture? AutoFixture 是一個針對 .NET 的開源庫,旨在最大程度地減少單元測試中的“安排(Arrange)”階段,以提高 ...
  • 經過前面幾個部分學習,相信學過的同學已經能夠掌握 .NET Emit 這種中間語言,並能使得它來編寫一些應用,以提高程式的性能。隨著 IL 指令篇的結束,本系列也已經接近尾聲,在這接近結束的最後,會提供幾個可供直接使用的示例,以供大伙分析或使用在項目中。 ...
  • 當從不同來源導入Excel數據時,可能存在重覆的記錄。為了確保數據的準確性,通常需要刪除這些重覆的行。手動查找並刪除可能會非常耗費時間,而通過編程腳本則可以實現在短時間內處理大量數據。本文將提供一個使用C# 快速查找並刪除Excel重覆項的免費解決方案。 以下是實現步驟: 1. 首先安裝免費.NET ...
  • C++ 異常處理 C++ 異常處理機制允許程式在運行時處理錯誤或意外情況。它提供了捕獲和處理錯誤的一種結構化方式,使程式更加健壯和可靠。 異常處理的基本概念: 異常: 程式在運行時發生的錯誤或意外情況。 拋出異常: 使用 throw 關鍵字將異常傳遞給調用堆棧。 捕獲異常: 使用 try-catch ...
  • 優秀且經驗豐富的Java開發人員的特征之一是對API的廣泛瞭解,包括JDK和第三方庫。 我花了很多時間來學習API,尤其是在閱讀了Effective Java 3rd Edition之後 ,Joshua Bloch建議在Java 3rd Edition中使用現有的API進行開發,而不是為常見的東西編 ...
  • 框架 · 使用laravel框架,原因:tp的框架路由和orm沒有laravel好用 · 使用強制路由,方便介面多時,分多版本,分文件夾等操作 介面 · 介面開發註意欄位類型,欄位是int,查詢成功失敗都要返回int(對接java等強類型語言方便) · 查詢介面用GET、其他用POST 代碼 · 所 ...
  • 正文 下午找企業的人去鎮上做貸後。 車上聽同事跟那個司機對罵,火星子都快出來了。司機跟那同事更熟一些,連我在內一共就三個人,同事那一手指桑罵槐給我都聽愣了。司機也是老社會人了,馬上聽出來了,為那個無辜的企業經辦人辯護,實際上是為自己辯護。 “這個事情你不能怪企業。”“但他們總不能讓銀行的人全權負責, ...