windowsHOOK

// HOOK_IAT.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <windows.h>

bool    hookIat( const char* pszDllName , const char* pszFunction , LPVOID pNewFunction );


// 盗版的MessageBox
DWORD WINAPI MyMessageBox( HWND hWnd , TCHAR* pText , TCHAR* pTitle , DWORD type ) {

    // 还原IAT
    hookIat( "User32.dll" , 
             "MessageBoxW" , 
             GetProcAddress(GetModuleHandleA("User32.dll"),"MessageBoxW" )
             );

    // 调用原版函数
    MessageBox( 0 , L"在盗版的MessageBox中弹出此框" , L"提示" , 0 );


    // HOOK IAT
    hookIat( "User32.dll" , "MessageBoxW" , &MyMessageBox );

    return 0;
}



int _tmain(int argc, _TCHAR* argv[])
{
    MessageBox( 0 , L"正版API" , L"提示" , 0 );
    // HOOK IAT
    hookIat( "User32.dll" , "MessageBoxW" , &MyMessageBox );

    MessageBox( 0 , L"正版API" , L"提示" , 0 );
    MessageBox( 0 , L"正版API" , L"提示" , 0 );
    return 0;
}


bool hookIat( const char* pszDllName ,
              const char* pszFunction ,
              LPVOID pNewFunction ) {

    // PE文件中,所有的API的地址都保存到了导入表中.

    // 程序调用一个API时, 先会从导入表中得到API
    // 的地址, 再调用这个地址.
    // 如果将导入表中的API地址替换掉, 那么调用
    // API时, 就会调用被替换的地址.

    // HOOK IAT的步骤:
    // 1. 解析PE文件,找到导入表
    // 2. 找到导入表中对应的模块
    // 3. 找到对应模块的对应函数.
    // 4. 修改函数地址.

    // 导入表中有两张表, 一张是导入名称表, 一张是导入
    // 地址表, 这两张表的元素一一对应的.
    // 导入名称表中存放的是函数名
    // 导入地址表中存放的是函数地址.

    HANDLE hProc = GetCurrentProcess( );
    
    PIMAGE_DOS_HEADER           pDosHeader; // Dos头
    PIMAGE_NT_HEADERS           pNtHeader;  // Nt头
    PIMAGE_IMPORT_DESCRIPTOR    pImpTable;  // 导入表
    PIMAGE_THUNK_DATA           pInt;       // 导入表中的导入名称表
    PIMAGE_THUNK_DATA           pIat;       // 导入表中的导入地址表
    DWORD                       dwSize;
    DWORD                       hModule;
    char*                       pFunctionName;
    DWORD                       dwOldProtect;

    hModule = ( DWORD)GetModuleHandle( NULL );

    // 读取dos头
    pDosHeader = (PIMAGE_DOS_HEADER)hModule;

    // 读取Nt头
    pNtHeader = (PIMAGE_NT_HEADERS)( hModule + pDosHeader->e_lfanew );


    // 获取导入表
    pImpTable = ( PIMAGE_IMPORT_DESCRIPTOR )
        (hModule + pNtHeader->OptionalHeader.DataDirectory[1].VirtualAddress);

    // 遍历导入表
    while( pImpTable->FirstThunk != 0 && pImpTable->OriginalFirstThunk != 0 ) {


        // 判断是否找到了对应的模块名
        if( _stricmp( (char*)(pImpTable->Name+hModule) , pszDllName ) != 0 ) {
            ++pImpTable;
            continue;
        }
        
        // 遍历名称表,找到函数名
        pInt = (PIMAGE_THUNK_DATA)( pImpTable->OriginalFirstThunk + hModule );
        pIat = (PIMAGE_THUNK_DATA)( pImpTable->FirstThunk + hModule );

        while( pInt->u1.AddressOfData != 0 ) {

            // 判断是以名称导入还是以需要导入
            if( pInt->u1.Ordinal & 0x80000000 == 1 ) {
                // 以序号导入

                // 判断是否找到了对应的函数序号
                if( pInt->u1.Ordinal == ( (DWORD)pszFunction ) & 0xFFFF ) {
                    // 找到之后,将钩子函数的地址写入到iat
                    VirtualProtect( &pIat->u1.Function ,
                                    4 ,
                                    PAGE_READWRITE ,
                                    &dwOldProtect
                                    );

                    pIat->u1.Function = (DWORD)pNewFunction;

                    VirtualProtect( &pIat->u1.Function ,
                                    4 ,
                                    dwOldProtect ,
                                    &dwOldProtect
                                    );
                    return true;
                }
            }
            else {
                // 以名称导入
                pFunctionName = (char*)( pInt->u1.Function + hModule + 2);

                // 判断是否找到了对应的函数名
                if( strcmp( pszFunction , pFunctionName ) == 0 ) {

                    VirtualProtect( &pIat->u1.Function ,
                                    4 ,
                                    PAGE_READWRITE ,
                                    &dwOldProtect
                                    );

                    // 找到之后,将钩子函数的地址写入到iat
                    pIat->u1.Function = (DWORD)pNewFunction;

                    VirtualProtect( &pIat->u1.Function ,
                                    4 ,
                                    dwOldProtect ,
                                    &dwOldProtect
                                    );

                    return true;
                }
            }
            
            ++pIat;
            ++pInt;
        }


        ++pImpTable;
    }

    return false;
    
}

========================

// HOOK_内联HOOK.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <windows.h>

BYTE    g_jmpShellCode[5] = { "\xe9" };
BYTE    g_oldOpcode[ 5 ] = { 0 };

#include "hookFunction.h"


// 盗版的MessageBox
DWORD WINAPI MyMessageBox( HWND hWnd , TCHAR* pText , TCHAR* pTitle , DWORD type ) 
{

    DWORD dwOldProtect = 0;
    DWORD dwWrite = 0;
    VirtualProtectEx( GetCurrentProcess( ) ,
                      &MessageBox ,
                      4 ,
                      PAGE_EXECUTE_READWRITE ,
                      &dwOldProtect
                      );


    // 恢复函数原来的内容
    WriteProcessMemory( GetCurrentProcess( ) ,
                        &MessageBox ,
                        g_oldOpcode ,
                        sizeof( g_jmpShellCode ) ,
                        &dwWrite
                        );

    // 调用原版函数
    MessageBox( 0 , L"在盗版的MessageBox中弹出此框" , L"提示" , 0 );



    // 调用完原版函数之后,再次HOOK这个函数
    WriteProcessMemory( GetCurrentProcess( ) ,
                        &MessageBox ,
                        g_jmpShellCode ,
                        sizeof( g_jmpShellCode ) ,
                        &dwWrite
                        );

    //  6. 恢复内存分页的属性
    VirtualProtectEx( GetCurrentProcess( ) ,
                      &MessageBox ,
                      4 ,
                      dwOldProtect ,
                      &dwOldProtect
                      );

    return 0;
}

// 盗版的MessageBox
DWORD WINAPI MyMessageBox2( HWND hWnd , TCHAR* pText , TCHAR* pTitle , DWORD type ) {
    //MessageBox( hWnd , pText , pTitle , type );
    MessageBox( hWnd , L"--------------", pTitle , type );
    return 0;
}

int _tmain(int argc, _TCHAR* argv[])
{

    // 内联HOOK的示意:
    // 未被拦截的函数
    // MessageBox:
    //      +--------------+
    // ==>  | mov edi,edi  |
    //      | push ebp     |
    //      | mov ebp,esp  |
    //      |              |
    //      |  XXXXXXXXX   |
    //      +--------------+
    // 
    // 被拦截的函数   
    //                                      function
    //  MessageBox:               +--->>-------------------+            
    //      +--------------+      |    |    push ebp       |
    // ==>  | jmp function>>------+    |   mov ebp,esp     |
    //      | push ebp     |           |                   |
    //      | mov ebp,esp  |           |     XXXXXX        |
    //      |              |           |     XXXXXX        |
    //      |  XXXXXXXXX   |           |     XXXXXX        |
    //      +--------------+           |     XXXXXX        |
    //                                 | jmp  MessageBox+5 |
    //                                 +-------------------+
    //
    // 所谓HOOK, 就是拦截, HOOK API就是在API的真正代码被执行前拦截它
    // 跳转到另一个地方执行代码, 执行后再跳转回去,或不跳转.
    // HOOK的概念比简单, 但一些细节需要注意
    // 例如:
    // 1. 原函数是没有跳转到XXX地址的代码的,jmp XXX是我们通过修改内存
    //    加入进去的, 在修改内存时,需要注意的是,一般能够被执行的内存分页
    //    都没有可写的属性,因此,在修改内存前,需要修改内存分页属性.
    // 2. jmp 指令需要用到一个跳转偏移, 而非一个绝对的地址. 将jmp XXX的
    //    opcode写入到内存时, 必须知道怎么计算出这个跳转偏移. 一般跳转
    //    偏移需要用到以下公式来计算:
    //      跳转偏移 = 目标地址 - 当前跳转指令所在地址 - 跳转指令的总体长度
    //    如:  
    //      指令要跳转到 011AB61F
    //      jmp 011AB61F 这条跳转指令所在地址为 011AB618
    //      jmp 011AB61F 这条指令的总体长度为2, 因为它的opcode是EB 05,只有2个字节
    //      所以它的偏移为: 011AB61F - 011AB61F - 2  => 5
    //         
    //  +-<011AB618 | EB 05 | JMP 011AB61F 
    //  |  011AB61A | 31 C0 | XOR EAX , EAX 
    //  |  011AB61C | 31 DB | XOR EBX , EBX 
    //  |  011AB61E | 40    | INC EAX 
    //  +> 011AB61F | 43    | INC EBX 


    // 内联HOOK步骤:
    //  1. 设置要修改的内存的分页属性为可写
    //  2. 准备跳转的shellcode
    //  3. 计算跳转偏移,并将计算好的跳转偏移写入到shellcode中
    //  4. 将函数开始地址处的opcode备份,字节数和shellcode等长.
    //  5. 将opcode写入到要HOOK的函数
    //  6. 恢复内存分页的属性

    // 1. 设置要修改的内存的分页属性为可写
    DWORD dwOldProtect = 0;
    VirtualProtectEx( GetCurrentProcess( ) ,
                      &MessageBox ,
                      4 ,
                      PAGE_EXECUTE_READWRITE ,
                      &dwOldProtect
                      );

    // 2. 准备跳转的shellcode
    //  BYTE    g_jmpShellCode[5] = { "\xe9" };


    // 3. 计算跳转偏移,并将计算好的跳转偏移写入到shellcode中
    DWORD dwJmpOffset = (DWORD)&MyMessageBox - (DWORD)&MessageBoxW - 5;
    *(DWORD*)( g_jmpShellCode + 1 ) = dwJmpOffset;



    // 4. 将函数开始地址处的opcode备份,字节数和shellcode等长.
    //    BYTE  g_oldOpcode[ 5 ] = { 0 };
    DWORD   dwRead = 0;
    ReadProcessMemory( GetCurrentProcess( ) ,
                       &MessageBox ,
                       g_oldOpcode ,
                       sizeof( g_jmpShellCode ) ,
                       &dwRead
                       );


    //  5. 将opcode写入到要HOOK的函数
    WriteProcessMemory( GetCurrentProcess( ) ,
                        &MessageBox ,
                        g_jmpShellCode ,
                        sizeof( g_jmpShellCode ) ,
                        &dwRead
                        );

    //  6. 恢复内存分页的属性
    VirtualProtectEx( GetCurrentProcess( ) ,
                      &MessageBox ,
                      4 ,
                      dwOldProtect ,
                      &dwOldProtect
                      );


    // 函数被调用后,将跑到钩子函数中执行代码
    MessageBox( NULL , L"我是正版MessageBox" , L"提示" , 0 );
    return 0;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容