0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

C++寫殼詳細(xì)教程(下)

jf_78858299 ? 來(lái)源:看雪論壇九陽(yáng)道人 ? 作者:看雪論壇九陽(yáng)道人 ? 2023-03-17 14:49 ? 次閱讀

2. 處理stub.dll

配置stub工程

將工程設(shè)置release版本,如果不想代碼被優(yōu)化,可以禁止優(yōu)化。

大概流程如下:

① 將數(shù)據(jù)段,只讀數(shù)據(jù)段和代碼段進(jìn)行合并

② 編寫代碼獲取API的地址

③ 加入混淆指令,反調(diào)試

④ 解密/解壓縮

⑤ 加密IAT等等

之后會(huì)把存根文件stub.dll的.data,.rdata這2個(gè)區(qū)段合并到.text段并設(shè)置為可讀可寫可執(zhí)行屬性,需要前置代碼

//把數(shù)據(jù)段融入代碼段
#pragma comment(linker,"/merge:.data=.text")
//把只讀數(shù)據(jù)段融入代碼段
#pragma comment(linker,"/merge:.rdata=.text")
//設(shè)置代碼段為可讀可寫可執(zhí)行
#pragma comment(linker,"/section:.text,RWE")

根據(jù)之前說(shuō)的已經(jīng)知道殼區(qū)段就是新添加的區(qū)段了,里面將保存移植過(guò)來(lái)的stub的.text段里的所有內(nèi)容,稱之為殼代碼。

而使用殼代碼的時(shí)候要注意,因?yàn)榧油隁ず?,在殼代碼中無(wú)法使用導(dǎo)入表,因此,需要自己動(dòng)態(tài)獲取需要使用的API函數(shù)的地址。

只要獲取到LoadLibraryExA和GetProcAddress兩個(gè)函數(shù)的地址,我們就可以根據(jù)LoadLibraryExA來(lái)獲取任意模塊dll的基地址,再使用GetProcAddress函數(shù)獲取到任意API函數(shù)的地址了。

根據(jù)kernel32基址可獲取到GetProcAddress地址。

下面是我獲取kernel32基址的內(nèi)聯(lián)匯編代碼。

__asm
{
push esi;
mov esi, fs:[0x30]; //得到PEB地址
mov esi, [esi + 0xc]; //指向PEB_LDR_DATA結(jié)構(gòu)的首地址
mov esi, [esi + 0x1c];//一個(gè)雙向鏈表的地址
mov esi, [esi]; //得到第2個(gè)條目kernelBase的鏈表
mov esi, [esi]; //得到第3個(gè)條目kernel32的鏈表(win10系統(tǒng))
mov esi, [esi + 0x8]; //kernel32.dll地址
mov g_hKernel32, esi;
pop esi;
}

然后是獲取GetProcAddress函數(shù)的匯編代碼,可以使用C語(yǔ)言方式獲取,但我覺得用匯編寫,它就這樣赤裸裸呈現(xiàn),能更加清晰的了解找到一個(gè)函數(shù)地址的過(guò)程。

//獲取GetProcAddress函數(shù)地址
void MyGetFunAddress()
{
__asm
{
pushad;
mov ebp, esp;
sub esp, 0xc;
mov edx, g_hKernel32;
mov esi, [edx + 0x3c]; //NT頭的RVA
lea esi, [esi + edx]; //NT頭的VA
mov esi, [esi + 0x78]; //Export的Rva
lea edi, [esi + edx]; //Export的Va

mov esi, [edi + 0x1c]; //Eat的Rva
lea esi, [esi + edx]; //Eat的Va
mov[ebp - 0x4], esi; //保存Eat

mov esi, [edi + 0x20]; //Ent的Rva
lea esi, [esi + edx]; //Ent的Va
mov[ebp - 0x8], esi; //保存Ent

mov esi, [edi + 0x24]; //Eot的Rva
lea esi, [esi + edx]; //Eot的Va
mov[ebp - 0xc], esi; //保存Eot

xor ecx, ecx;
jmp _First;
_Zero:
inc ecx;
_First:
mov esi, [ebp - 0x8]; //Ent的Va
mov esi, [esi + ecx * 4]; //FunName的Rva

lea esi, [esi + edx]; //FunName的Va
cmp dword ptr[esi], 050746547h;// 47657450 726F6341 64647265 7373;
jne _Zero; // 上面的16進(jìn)制是GetProcAddress的ASCII
cmp dword ptr[esi + 4], 041636f72h;
jne _Zero;
cmp dword ptr[esi + 8], 065726464h;
jne _Zero;
cmp word ptr[esi + 0ch], 07373h;
jne _Zero;

xor ebx,ebx
mov esi, [ebp - 0xc]; //Eot的Va
mov bx, [esi + ecx * 2]; //得到序號(hào)

mov esi, [ebp - 0x4]; //Eat的Va
mov esi, [esi + ebx * 4]; //FunAddr的Rva
lea eax, [esi + edx]; //FunAddr
mov MyGetProcAddress, eax;
add esp, 0xc;
popad;
}
}

然后再獲取下MessageBoxW函數(shù),彈出一個(gè)對(duì)話框,測(cè)試是否成功。

//運(yùn)行函數(shù)
void RunFun()
{
MyLoadLibraryExA = (FuLoadLibraryExA)MyGetProcAddress(g_hKernel32, "LoadLibraryExA");
g_hUser32 = MyLoadLibraryExA("user32.dll", 0, 0);
MyMessageBoxW = (FuMessageBoxW)MyGetProcAddress(g_hUser32, "MessageBoxW");
MyMessageBoxW(0, L"大家好我是一個(gè)殼", L"提示", 0);
}

它在運(yùn)行原代碼之前先運(yùn)行了殼代碼,測(cè)試成功。

圖片

四、代碼段加密

我們?cè)谀嫦蚱平獾臅r(shí)候通常第一方法是找到關(guān)鍵字符串,關(guān)鍵代碼等,他們都是存在于代碼段的,那么只要把代碼段進(jìn)行加密,這種方式就不可行了。

先在加殼器中加密,這使用簡(jiǎn)單的亦或加密。

//加密代碼段
//1.獲取代碼段首地址
char* pTarText = GetSecHeader(pTarBuff, ".text")->PointerToRawData + pTarBuff;
//2.獲取代碼段實(shí)際大小
int nSize = GetSecHeader(pTarBuff, ".text")->Misc.VirtualSize;
for (int i = 0; i < nSize; ++i)
{
pTarText[i] ^= 0x15;
}

再到殼代碼里解密,自己寫了一個(gè)對(duì)比字符串的函數(shù)。

//自寫strcmp
int StrCmpText(const char* pStr, char* pBuff)
{
int nFlag = 1;
__asm
{
mov esi, pStr;
mov edi, pBuff;
mov ecx, 0x6;
cld;
repe cmpsb;
je _end;
mov nFlag, 0;
_end:
}
return nFlag;
}

//解密
void Decryption()
{
//獲取.text的區(qū)段頭
auto pNt = GetNtHeader((char*)g_hModule);
DWORD dwSecNum = pNt->FileHeader.NumberOfSections;
auto pSec = IMAGE_FIRST_SECTION(pNt);

//找到代碼區(qū)段
for (size_t i = 0; i < dwSecNum; i++)
{
if (StrCmpText(".text", (char*)pSec[i].Name))
{
pSec += i;
break;
}
}

//獲取代碼段首地址
char* pTarText = pSec->VirtualAddress + (char*)g_hModule;
int nSize = pSec->Misc.VirtualSize;
DWORD old = 0;
//解密代碼段
MyVirtualProtect(pTarText, nSize, PAGE_READWRITE, &old);
for (int i = 0; i < nSize; ++i) {
pTarText[i] ^= 0x15;
}
MyVirtualProtect(pTarText, nSize, old, &old);
}

五、壓縮

壓縮是一個(gè)比較復(fù)雜的過(guò)程,對(duì)于一個(gè)主要功能的加密的殼來(lái)說(shuō),壓縮也有一定的加密效果,如果使用了一些加密庫(kù)加密,即使你壓縮了,會(huì)發(fā)現(xiàn)加殼后的文件比沒加殼之前還要大!

圖片

這說(shuō)一下壓縮大概思路,首先不能壓縮頭部,考慮到后面要處理TLS,還有一個(gè)程序的圖標(biāo)在資源段,所以不壓縮這兩個(gè)段。

在加殼器中把原文件的中除了.tls和.rsrc段的其他段的數(shù)據(jù)一個(gè)一個(gè)的按順序取出來(lái),然后拼接在一起,然后對(duì)這份拼接后數(shù)據(jù)進(jìn)行一個(gè)整體的壓縮,之后需要再添加一個(gè)區(qū)段專門用于存放壓縮后的數(shù)據(jù),這個(gè)過(guò)程中,需要把壓縮后的區(qū)段的文件偏移和文件大小都清零,如下圖所示,把.tsl段和.rsrc段移動(dòng)到頭部的后面。

值得注意的是沒有處理TLS時(shí)要把TLS表的RVA和大小清零,TLS在數(shù)據(jù)目錄表的第九項(xiàng)。

auto pData = GetOptHeader(pTarBuff)->DataDirectory;
pData[9].Size = 0;
pData[9].VirtualAddress = 0;

運(yùn)行時(shí),先在殼代碼中進(jìn)行解壓縮,再解密,然后程序就能正常運(yùn)行了。

到此一個(gè)簡(jiǎn)單的加密壓縮殼就完成了,在這個(gè)過(guò)程中實(shí)際出現(xiàn)了很多bug,因?yàn)樯婕暗紻LL文件無(wú)法用VS調(diào)試, 所以使用OD或者x64dbg進(jìn)行調(diào)試,推薦使用x64dbg(x32dbg),這個(gè)軟件一直在更新,而且字符串提示更友好,更方便快捷。OD主要用于脫殼破解,逆向還是x64dbg更方便。

最后再說(shuō)一下VS2017使用配置:

有2個(gè)工程文件 一個(gè)是加殼器,一個(gè)是sutb。

加殼器使用x32debug編譯

sutb使用x32Release編譯

找到工程所在文件夾,新建一個(gè)bin目錄,把這兩個(gè)工程屬性中的輸出目錄改為bin,這樣操作起來(lái)方便一些,不改也行,但是加載stub時(shí)路徑就要填寫正確才行。

圖片

一個(gè)殼的基本框架就搭建完成了,而加殼主要是為了防止被別人破解,所以接下來(lái)就可以執(zhí)行加密操作了,下一次再說(shuō)說(shuō)IAT加密,Hash加密,動(dòng)態(tài)解密,反調(diào)試等技術(shù)吧。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • WINDOWS
    +關(guān)注

    關(guān)注

    4

    文章

    3564

    瀏覽量

    89104
  • C++
    C++
    +關(guān)注

    關(guān)注

    22

    文章

    2114

    瀏覽量

    73784
  • PE文件
    +關(guān)注

    關(guān)注

    0

    文章

    4

    瀏覽量

    5458
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    C++一個(gè)http服務(wù)器

    本篇文章不會(huì)涉及到很多復(fù)雜的概念,也沒有很難讀懂的模板函數(shù),代碼簡(jiǎn)單可讀,本篇文章送給每一個(gè)想自己用C++一個(gè)http服務(wù)器的小伙伴!高手們、大佬們當(dāng)然可以不用看的啦!
    發(fā)表于 09-30 10:47 ?2261次閱讀

    詳細(xì)講解C++串口的相關(guān)知識(shí)

    大家可以先參考一這篇blog,C++串口通信里面詳細(xì)講解了C++串口的相關(guān)知識(shí),以及一些函數(shù)的講解。下面我也會(huì)根據(jù)他的blog再講解。二、實(shí)現(xiàn)過(guò)程1、打開串口:使用函數(shù):HANDLE
    發(fā)表于 08-24 06:56

    詳述不用c++網(wǎng)頁(yè)的理由

    C++在web開發(fā)中地位如何,進(jìn)來(lái)一探究竟吧。
    的頭像 發(fā)表于 12-22 10:07 ?5978次閱讀

    C++課程資料詳細(xì)資料合集包括了:面向?qū)ο蟪绦蛟O(shè)計(jì)與C++,算法,函數(shù)等

    本文檔的主要內(nèi)容詳細(xì)介紹的是C++課程資料資料合集包括了:面向?qū)ο蟪绦蛟O(shè)計(jì)與C++,算法,函數(shù),概述, C++語(yǔ)言基礎(chǔ),構(gòu)造數(shù)據(jù)類型,數(shù)據(jù)類型,C+
    發(fā)表于 07-09 08:00 ?18次下載
    <b class='flag-5'>C++</b>課程資料<b class='flag-5'>詳細(xì)</b>資料合集包括了:面向?qū)ο蟪绦蛟O(shè)計(jì)與<b class='flag-5'>C++</b>,算法,函數(shù)等

    基于fpgrowth的c++實(shí)現(xiàn)詳細(xì)資料免費(fèi)下載

    本文檔的主要內(nèi)容詳細(xì)介紹的是基于fpgrowth的c++實(shí)現(xiàn)詳細(xì)資料免費(fèi)下載
    發(fā)表于 08-02 08:00 ?1次下載

    如何提高cc++的安全編程能力?《CC++安全編碼》帶你詳細(xì)學(xué)習(xí)

    ,既詳細(xì)闡述了C/C++語(yǔ)言及其相關(guān)庫(kù)固有的安全問(wèn)題和陷阱,系統(tǒng)總結(jié)了導(dǎo)致軟件漏洞的各種常見編碼錯(cuò)誤,并給出了應(yīng)對(duì)錯(cuò)誤的解決方案;又對(duì)C/C++
    發(fā)表于 08-28 08:00 ?0次下載

    如何進(jìn)行高質(zhì)量的C、C++編程?高質(zhì)量C++C編程指南詳細(xì)資料免費(fèi)下載

    本文檔的作用內(nèi)容詳細(xì)介紹的是如何進(jìn)行高質(zhì)量的C、C++編程?高質(zhì)量C++C編程指南詳細(xì)資料免費(fèi)
    發(fā)表于 09-10 08:00 ?30次下載

    VISUAL C++教程之VISUAL C++的安裝和使用方法

    本文檔的主要內(nèi)容詳細(xì)介紹的是VISUAL C++教程之VISUAL C++的安裝和使用方法資料免費(fèi)下載。
    發(fā)表于 12-27 16:32 ?19次下載
    VISUAL <b class='flag-5'>C++</b>教程之VISUAL <b class='flag-5'>C++</b>的安裝和使用方法

    C++的cast最完整最詳細(xì)的解釋資料說(shuō)明

    本文檔的主要內(nèi)容詳細(xì)介紹的是C++的cast最完整最詳細(xì)的解釋資料說(shuō)明。
    發(fā)表于 01-29 15:26 ?0次下載
    <b class='flag-5'>C++</b>的cast最完整最<b class='flag-5'>詳細(xì)</b>的解釋資料說(shuō)明

    C++程序設(shè)計(jì)教程之C++的初步知識(shí)的詳細(xì)資料說(shuō)明

    C++程序設(shè)計(jì)教程之C++的初步知識(shí)的詳細(xì)資料說(shuō)明包括了:1. 從CC++,2 . 最簡(jiǎn)單的C++
    發(fā)表于 03-14 14:48 ?31次下載
    <b class='flag-5'>C++</b>程序設(shè)計(jì)教程之<b class='flag-5'>C++</b>的初步知識(shí)的<b class='flag-5'>詳細(xì)</b>資料說(shuō)明

    C++程序設(shè)計(jì)教程之C++工具的詳細(xì)資料說(shuō)明

    本文檔的詳細(xì)介紹的是C++程序設(shè)計(jì)教程之C++工具的詳細(xì)資料說(shuō)明主要內(nèi)容包括了:1. 異常處理,2. 命名空間,3. 使用早期的函數(shù)庫(kù)
    發(fā)表于 03-14 16:39 ?4次下載
    <b class='flag-5'>C++</b>程序設(shè)計(jì)教程之<b class='flag-5'>C++</b>工具的<b class='flag-5'>詳細(xì)</b>資料說(shuō)明

    C++語(yǔ)言編碼規(guī)范詳細(xì)說(shuō)明

    本文檔的主要內(nèi)容詳細(xì)介紹的是C++語(yǔ)言編碼規(guī)范詳細(xì)說(shuō)明。
    發(fā)表于 01-07 16:19 ?14次下載
    <b class='flag-5'>C++</b>語(yǔ)言編碼規(guī)范<b class='flag-5'>詳細(xì)</b>說(shuō)明

    C語(yǔ)言和C++的特點(diǎn)與用法詳細(xì)說(shuō)明

    本文檔的主要內(nèi)容詳細(xì)介紹的是C語(yǔ)言和C++的特點(diǎn)與用法詳細(xì)說(shuō)明。
    的頭像 發(fā)表于 12-26 10:58 ?4456次閱讀

    qt用C++的2048小游戲源代碼

    qt用C++的2048小游戲源代碼
    發(fā)表于 09-27 11:48 ?1次下載

    C++詳細(xì)教程(上)

    本文基于Windows平臺(tái)對(duì)PE文件加的項(xiàng)目,經(jīng)過(guò)一個(gè)月的緩沖,決定復(fù)習(xí)總結(jié)及分享的我的心得。
    的頭像 發(fā)表于 03-17 14:49 ?853次閱讀
    <b class='flag-5'>C++</b><b class='flag-5'>寫</b><b class='flag-5'>殼</b><b class='flag-5'>詳細(xì)</b>教程(上)