聊一聊對一個 C# 商業程式的反反調試

来源:https://www.cnblogs.com/huangxincheng/archive/2022/11/02/16849808.html
-Advertisement-
Play Games

一:背景 1.講故事 前段時間有位朋友在微信上找到我,說他對一個商業的 C# 程式用 WinDbg 附加不上去,每次附加之後那個 C# 程式就自動退出了,問一下到底是怎麼回事?是不是哪裡搞錯了,有經驗的朋友應該知道,其實這是 商業程式 的反調試機制搗鬼的,為了保護程式隱私,一般都不希望他人對自己做逆 ...


一:背景

1.講故事

前段時間有位朋友在微信上找到我,說他對一個商業的 C# 程式用 WinDbg 附加不上去,每次附加之後那個 C# 程式就自動退出了,問一下到底是怎麼回事?是不是哪裡搞錯了,有經驗的朋友應該知道,其實這是 商業程式 的反調試機制搗鬼的,為了保護程式隱私,一般都不希望他人對自己做逆向分析,那能不能破解它的反調試呢?當然是可以的,難易程度就看對方的誠意了。

經過和朋友的技術搗鼓之後,發現還好,對方只是用了 KERNELBASE!IsDebuggerPresent 做的反調試判斷,難度不大,這裡就不細聊那個程式,我們做一個簡單的案例來說下如何反反調試,老規矩,上 WinDbg 說話。

二:WinDbg 分析

1. 案例演示

為了方便講述,先上一個例子。


    internal class Program
    {
        [DllImport("kernelbase.dll", SetLastError = true)]
        static extern bool IsDebuggerPresent();

        static void Main(string[] args)
        {
            Console.ReadLine();

            var isAttached = IsDebuggerPresent();

            if (isAttached)
            {
                Console.WriteLine("/(ㄒoㄒ)/~~ 小心,我被附加了 調試器!");
            }
            else
            {
                Console.WriteLine("O(∩_∩)O 程式很安全!");
            }

            Console.ReadLine();
        }
    }

在沒有 WinDbg 的情況下是這樣輸出的。

有 WinDbg 的情況下是這樣輸出的。

有朋友肯定要懟了,C# 中有一個 Debugger.IsAttached 屬性為什麼不用,我試了下,這玩意很差勁,檢測不到 WinDbg 這種非托管調試器的附加。

2. 簡述 IsDebuggerPresent 方法

其實 IsDebuggerPresent 方法提取的是 PEB 中的 BeingDebugged 欄位,這個欄位定義在 KernelBase.dll 中,那怎麼驗證呢? 可以用 !peb 查看進程環境塊的地址,然後用 dt 觀察即可。


0:001> !peb
PEB at 000000000035b000
    InheritedAddressSpace:    No
    ReadImageFileExecOptions: No
    BeingDebugged:            Yes
    ImageBaseAddress:         00007ff719030000
    NtGlobalFlag:             4000
    NtGlobalFlag2:            0
    Ldr                       00007ffb1259b4c0
    ...

0:001> dt ntdll!_PEB 000000000035b000
   +0x000 InheritedAddressSpace : 0 ''
   +0x001 ReadImageFileExecOptions : 0 ''
   +0x002 BeingDebugged    : 0x1 ''
   +0x003 BitField         : 0x4 ''
   +0x003 ImageUsesLargePages : 0y0
   +0x003 IsProtectedProcess : 0y0
   +0x003 IsImageDynamicallyRelocated : 0y1
   +0x003 SkipPatchingUser32Forwarders : 0y0
   ...

從上面的 BeingDebugged : 0x1 可以看到,當前程式被附加了調試器。

3. 反反調試思路

找到 IsDebuggerPresent() 方法的讀取來源,這問題就好辦了,通常有兩種做法。

  1. 修改 IsDebuggerPresent() 方法的反彙編代碼

只要讓 IsDebuggerPresent() 方法一直返回 false,那我們就可以成功破解反調試,首先用 x 命令找到 IsDebuggerPresent() 的彙編代碼,輸出如下:


0:007> x KernelBase!IsDebuggerPresent
00007ffb`0fe468a0 KERNELBASE!IsDebuggerPresent (IsDebuggerPresent)
0:007> u 00007ffb`0fe468a0
KERNELBASE!IsDebuggerPresent:
00007ffb`0fe468a0 65488b042560000000 mov   rax,qword ptr gs:[60h]
00007ffb`0fe468a9 0fb64002        movzx   eax,byte ptr [rax+2]
00007ffb`0fe468ad c3              ret
00007ffb`0fe468ae cc              int     3
00007ffb`0fe468af cc              int     3
00007ffb`0fe468b0 cc              int     3
00007ffb`0fe468b1 cc              int     3
00007ffb`0fe468b2 cc              int     3

按照 stdcall 協定, eax 會作為方法的返回值,接下來使用 WinDbg 的 a 命令修改 00007ffb0fe468a0 處的彙編代碼,鍵入完彙編代碼之後,按 Enter 即可,輸出如下:


0:007> a 00007ffb`0fe468a0
00007ffb`0fe468a0 mov eax , 0
00007ffb`0fe468a5 ret 
00007ffb`0fe468a6 

0:007> u 00007ffb`0fe468a0
KERNELBASE!IsDebuggerPresent:
00007ffb`0fe468a0 b800000000      mov     eax,0
00007ffb`0fe468a5 c3              ret
00007ffb`0fe468a6 0000            add     byte ptr [rax],al
00007ffb`0fe468a8 000f            add     byte ptr [rdi],cl
00007ffb`0fe468aa b640            mov     dh,40h
00007ffb`0fe468ac 02c3            add     al,bl
00007ffb`0fe468ae cc              int     3
00007ffb`0fe468af cc              int     3

可以看到 WinDbg 已成功修改了 KERNELBASE!IsDebuggerPresent 方法的代碼,哈哈,接下來繼續 go,截圖如下:

可以看到已成功的反反調試,看到程式很開心,我也挺開心的。

  1. 使用bp斷點攔截

這種做法就是使用 bp + script 攔截,大概就是在 KERNELBASE!IsDebuggerPresent的ret 處用腳本自動修改 eax 值,這也是可以的,當然也是最安全的。

首先觀察一下 uf KERNELBASE!IsDebuggerPresent 函數的彙編代碼。


0:004> uf KERNELBASE!IsDebuggerPresent
KERNELBASE!IsDebuggerPresent:
00007ffb`0fe468a0 65488b042560000000 mov   rax,qword ptr gs:[60h]
00007ffb`0fe468a9 0fb64002        movzx   eax,byte ptr [rax+2]
00007ffb`0fe468ad c3              ret

接下來在 00007ffb0fe468ad 處下一個斷點,即位置 KERNELBASE!IsDebuggerPresent + 0xd ,然後使用寄存器修改命令 r 修改 eax 的值,再讓程式 gc 即可,腳本代碼如下:


0:004> bp KERNELBASE!IsDebuggerPresent+0xd "r eax =0; gc"
0:004> g

可以看到,此時的程式又是笑哈哈的。

三: 總結

這篇文章無意對抗,只是對一個疑難問題尋求解決方案的探索,大家合理使用。


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

-Advertisement-
Play Games
更多相關文章
  • 文章目錄 🦠一、前言 🦠二、軟體開發架構 🍀2.1、C/S架構 🍀2.2、B/S架構 🍀2.3、服務端與客戶端 🦠三、ip與埠號 🍀3.1、IP地址與埠號常識 🍀3.2、MAC和IP的概念與不同 🦠四、tcp協議和udp協議 🦠五、乙太網:區域網與交換機 🍀5.1、什麼是局 ...
  • 前言 環境使用 Python 3.8 Pycharm 模塊使用 requests jieba 結巴分詞 wordcloud 詞雲 數據來源分析 明確需求 <數據來源分析> 採集數據是什麼東西? 通過那個url地址得到想要數據的內容 抓包分析: 瀏覽器自帶工具 --> 開發者工具I. F12 或者 鼠 ...
  • 邏輯導航 1.當一訪問127.0.0.1:8000時,就會向某一地址發送請求 2.請求介面需要返迴首頁所需要的輪播圖片 3.前端vue輪播圖組件迴圈一下後端發送的圖片連接列表,依次展示輪播圖 輪播圖表設計 輪播圖中一些共有的欄位,我們可以創建一個公共的基礎表,以後需要該欄位直接基礎該表就行 基表,可 ...
  • 疫情尚未結束,我們需要做好自己,時刻防範,不給別人添麻煩。 今天我們來嘗試用Python抓取世界疫情,實現可視化地圖展示。 話不多說直接開搞! 採集數據 1、數據來源 數據來源於TX新聞,鏈接展示不了,就只貼圖了。 2、模塊 import requests import csv # Python學習 ...
  • 1. Seata Server 部署 Seata分TC、TM和RM三個角色,TC(Server端)為單獨服務端部署,TM和RM(Client端)由業務系統集成。 首先,下載最新的安裝包 也可以下載源碼,然後本地編譯。最新的版本是1.5.2 下載後的啟動包(或者源碼)中有個scripts目錄,裡面有各 ...
  • 前言 在 Java 和 Kotlin 中, 除了使用Spring Boot創建微服務外,還有很多其他的替代方案。 | 名稱 | 開發商 | | : : | : : | | Helidon SE | 甲骨文 | | Ktor | JetBrains | | Micronaut | Object Com ...
  • namespace Qingyun.QingBaiFang.V2.APITest.Web { using Microsoft.VisualStudio.TestTools.UnitTesting; using System.Collections.Generic; using System.Text ...
  • 運行一個.NET Core 項目 報錯:引發的異常:“sqlsugar.sqlsugarexception” 位於 system.private.corelib.dll 中 。 我運行的項目是 : 核心商城系統(CoreShop),本地運行起來項目後,輸入用戶名和密碼進行登錄,登錄後也沒有報錯,但是 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...