C# 通過 inline-asm 解決嵌入x86匯編
"嵌入"是指什么?資源?注入進(jìn)程?如果是嵌入資源,那跟嵌入任何其他內(nèi)容是一樣的,vs中只要拖拽就能完成嵌入資源。如果是注入進(jìn)程,則必須得先將匯編碼轉(zhuǎn)為機(jī)器碼。雖然托管的C#也是能辦到,但這似乎是所有人都不推薦的方式。
C#可不可以嵌入?yún)R編 可以 在我眼中C#作為一個(gè)介于中上層語言是不可能不可以置入?yún)R編代碼的 為什么會(huì)被我認(rèn)為中上層語言呢 從C#保留指針就可以看出 我知道有很多人一定不會(huì)相信C#可以使用匯編代碼
不過C#會(huì)比較麻煩C#不可以直接內(nèi)聯(lián)匯編(inline-asm)準(zhǔn)確的說C#只可以使用(auto-asm)動(dòng)態(tài)匯編 這種技術(shù)不是
C#獨(dú)有的 易語言、VB、C++ 三種語言都可以 不過動(dòng)態(tài)匯編我見過最多的是被應(yīng)用在外掛方面 及遠(yuǎn)程匯編注入 實(shí)際上是屬于動(dòng)態(tài)匯編技術(shù)的一種擴(kuò)展 不過很難說JIT在編譯代碼后是通過在遠(yuǎn)程把匯編代碼寫入托管進(jìn)程執(zhí)行的 又或者說是一種寄生在外殼程序中運(yùn)行的技術(shù)及“內(nèi)存運(yùn)行” 懶得討論這些一想到就頭大。
從上圖中你可以看見一份簡(jiǎn)單的x86 / call匯編在C#中內(nèi)嵌并被調(diào)用執(zhí)行一看你會(huì)發(fā)現(xiàn)并不是太難 我的一篇博文 寫了一大堆廢話就是說這個(gè)東西不過是易語言的
我們知道軟件運(yùn)行時(shí)所有代碼會(huì)放在虛擬內(nèi)存中 而可執(zhí)行的代碼在內(nèi)存中
內(nèi)存保護(hù)一般是PAGE_EXECUTE_READ及32不過經(jīng)過我研究.NET上的可執(zhí)行代碼應(yīng)該是PAGE_EXECUTE_READWIRTE及64 如果是P/invoke上執(zhí)行DLL中的保護(hù)是32 就可以我們?cè)趦?nèi)嵌匯編時(shí)不可以使用只讀保護(hù)
如果我們需要使用由.NET去委托去Call那么必須是可讀可寫 如果通過Win32API去Call那么使用32就可以 有些區(qū)別 、我曾研究過易語言上字節(jié)集在內(nèi)存中的內(nèi)存保護(hù)到底是多少結(jié)果與C#是一致 4 / PAGE_READWRITE 不過為什么易語言可以CALL而C#不可以CALL一直是讓我感到較為迷惑的事情 可能是托管堆與非托管堆之間不同造成的 不過我更希望有大神出來幫忙指點(diǎn)一下下。
由于是X86匯編 首先需要把目標(biāo)平臺(tái)切換為x86 這樣才不會(huì)造成C#調(diào)用匯編代碼時(shí)出錯(cuò) 一定不要省略這個(gè)步驟
首先你需要定義一個(gè)有參數(shù)的委托 重點(diǎn)在于在匯編中有這樣一句話
call dword ptr[ebp+8] // call 參數(shù)一 [UnmanagedFunctionPointer(CallingConvention.Cdecl)] public delegate IntPtr CallMethod(IntPtr ptr);
由于是在VC下內(nèi)聯(lián)的匯編 最后移植到C# 一般在VC下函數(shù)的調(diào)用方式是cdcel
何況下面的是按照cdcel導(dǎo)出函數(shù)格式進(jìn)行的 所以不可以使用__stdcall的方式
[STAThread] static void Main(string[] args) { byte[] buf_asm = { // push ebp // mov ebp,esp // sub esp,0C0h // push ebx // push esi // push edi // lea edi,[ebp-0C0h] // mov ecx,30h // mov eax,0CCCCCCCCh // rep stos dword ptr es:[edi] 85, 139, 236, 129, 236, 192, 0, 0, 0, 83, 86, 87, 141, 189, 64, 255, 255, 255, 185, 48, 0, 0, 0, 184, 204, 204, 204, 204, 243, 171, // call dword ptr[ebp+8] 255, 85, 8, // pop edi // pop esi // pop ebx // mov esp,ebp // pop ebp // ret 95, 94, 91, 139, 229, 93, 195 }; IntPtr ptr_asm = SetHandleCount(buf_asm); VirtualProtect(ptr_asm, buf_asm.Length); CallMethod call_method = Marshal.GetDelegateForFunctionPointer(ptr_asm, typeof(CallMethod)) as CallMethod; call_method(Marshal.GetFunctionPointerForDelegate(new Action(Hello_x86))); }
首先把你需要嵌入的匯編以字節(jié)數(shù)組的格式寫出來 然后通過
SetHandleCount函數(shù)是用于取地址指針的
static void VirtualProtect(IntPtr ptr, int size) { int outMemProtect; if (!VirtualProtect(ptr, size, 64, out outMemProtect)) throw new Exception("Unable to modify memory protection."); }
上面的函數(shù)用于修改內(nèi)存保護(hù) 不過是為了讓委托可以進(jìn)行交互 包括匯編代碼可以被互調(diào)用
static void Hello_x86() { Console.Title = ((new StackFrame()).GetMethod()).Name; Console.WriteLine("I was x86 assembly call a test function."); Console.ReadKey(false); }
上面的函數(shù)是一個(gè)測(cè)試函數(shù) 這個(gè)函數(shù)沒有太大意義 只是表現(xiàn)利用了匯編調(diào)用
本函數(shù) 然后本函數(shù)輸出一個(gè)回應(yīng)的信息 用于提示該函數(shù)被寫入內(nèi)存匯編調(diào)用
依賴的外部函數(shù)
[DllImport("kernel32.dll", CharSet = CharSet.Auto)] public static extern IntPtr SetHandleCount(byte[] value); [DllImport("kernel32.dll", SetLastError = true)] public static extern bool VirtualProtect(IntPtr lpAddress, int dwSize, int flNewProtect, out int lpflOldProtect);
依賴的命名空間
using System; using System.Runtime.InteropServices; using System.Diagnostics;
通過inline-asm技術(shù)解決C#語言解決嵌入x86匯編,希望大家能夠喜歡。
相關(guān)文章
c#異步讀取數(shù)據(jù)庫與異步更新ui的代碼實(shí)現(xiàn)
這篇文章主要介紹了c#從數(shù)據(jù)庫里取得數(shù)據(jù)并異步更新ui的方法,大家參考使用吧2013-12-12Quartz.Net實(shí)現(xiàn)原理及使用方法詳解
這篇文章主要介紹了Quartz.Net實(shí)現(xiàn)原理及使用方法詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-12-12C#使用semaphore來管理異步下載請(qǐng)求的方法
這篇文章主要介紹了C#使用semaphore來管理異步下載請(qǐng)求的方法,涉及C#使用semaphore實(shí)現(xiàn)多線程管理的技巧,需要的朋友可以參考下2015-06-06C#調(diào)用FFplay實(shí)現(xiàn)播放視頻功能
這篇文章主要為大家詳細(xì)介紹了C#如何調(diào)用FFplay實(shí)現(xiàn)播放視頻功能,文中的示例代碼講解詳細(xì),具有一定的參考價(jià)值,有需要的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-10-10Unity實(shí)現(xiàn)人物旋轉(zhuǎn)和移動(dòng)效果
這篇文章主要為大家詳細(xì)介紹了Unity實(shí)現(xiàn)人物旋轉(zhuǎn)和移動(dòng)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-01-01