boywhp
驱动中牛
驱动中牛
  • 注册日期2007-08-09
  • 最后登录2015-04-24
  • 粉丝2
  • 关注0
  • 积分1105分
  • 威望515点
  • 贡献值0点
  • 好评度254点
  • 原创分1分
  • 专家分0分
阅读:3466回复:8

花了一天时间搞定通用Inline Hook代码,共享与大家

楼主#
更多 发布于:2008-04-22 23:04
#include "inlinehook.h"

typedef struct _INLINE_HOOK_ITEM
{
    PBYTE    HookAddress;        //inlinehook 的位置
    DWORD    OrgBytesSize;        //
    //PBYTE    OrgBytes;            //原始函数字节
    PBYTE    HookBytes;            //HOOK代码 = OrgBytesSize + 14字节
}INLINE_HOOK_ITEM, *PINLINE_HOOK_ITEM;

////////////////////////////////
// DisassembleMem32
////////////////////////////////
DWORD DisassembleMem32(PBYTE pbCode)
{
 BYTE      bmodrm;

 //----------------
 bmodrm = *pbCode;

 if (bmodrm >= 0xC0)
   return 1;

 if (bmodrm >= 0x80)
   return ((bmodrm & 0x07) == 0x04 ? 6 : 5);

 if (bmodrm >= 0x40)
   return ((bmodrm & 0x07) == 0x04 ? 3 : 2);

 if ((bmodrm & 0x07) == 0x05)
   return 5;

 if ((bmodrm & 0x07) == 0x04)
   return ((pbCode[1] & 0x07) == 0x05 ? 6 : 2);

 return 1;
} //DisassembleMem32()


////////////////////////////////
// DisassembleProlog
////////////////////////////////
DWORD DisassembleProlog(
 PBYTE      pbCode,
 DWORD      cbMinimumRequired)
{
 PBYTE      pb;
 DWORD      cboperand;

 //----------------
 cboperand = 4;

 for (pb = pbCode; (DWORD)(pb - pbCode) < cbMinimumRequired; )
 {
   // Potemkin's Hackers Group rocks heavy metal-style (OPCODE.LST)

   switch (*pb++)
   {
   case 0x00:      // 00h: ADD mem8, reg8
   case 0x01:      // 01h: ADD mem, reg
   case 0x02:      // 02h: ADD reg8, mem8
   case 0x03:      // 03h: ADD reg, mem
   case 0x08:      // 08h: OR mem8, reg8
   case 0x09:      // 09h: OR mem, reg
   case 0x0A:      // 0Ah: OR reg8, mem8
   case 0x0B:      // 0Bh: OR reg, mem
   case 0x10:      // 10h: ADC mem8, reg8
   case 0x11:      // 11h: ADC mem, reg
   case 0x12:      // 12h: ADC reg8, mem8
   case 0x13:      // 13h: ADC reg, mem
   case 0x18:      // 18h: SBB mem8, reg8
   case 0x19:      // 19h: SBB mem, reg
   case 0x1A:      // 1Ah: SBB reg8, mem8
   case 0x1B:      // 1Bh: SBB reg, mem
   case 0x20:      // 20h: AND mem8, reg8
   case 0x21:      // 21h: AND mem, reg
   case 0x22:      // 22h: AND reg8, mem8
   case 0x23:      // 23h: AND reg, mem
   case 0x28:      // 28h: SUB mem8, reg8
   case 0x29:      // 29h: SUB mem, reg
   case 0x2A:      // 2Ah: SUB reg8, mem8
   case 0x2B:      // 2Bh: SUB reg, mem
   case 0x30:      // 30h: XOR mem8, reg8
   case 0x31:      // 31h: XOR mem, reg
   case 0x32:      // 32h: XOR reg8, mem8
   case 0x33:      // 33h: XOR reg, mem
   case 0x38:      // 38h: CMP mem8, reg8
   case 0x39:      // 39h: CMP mem, reg
   case 0x3A:      // 3Ah: CMP reg8, mem8
   case 0x3B:      // 3Bh: CMP reg, mem
   case 0x84:      // 84h: TEST mem8, reg8
   case 0x85:      // 85h: TEST mem, reg
   case 0x86:      // 86h: XCHG mem8, reg8
   case 0x87:      // 87h: XCHG mem, reg
   case 0x88:      // 88h: MOV mem8, reg8
   case 0x89:      // 89h: MOV mem, reg
   case 0x8A:      // 8Ah: MOV reg8, mem8
   case 0x8B:      // 8Bh: MOV reg, mem
   case 0x8C:      // 8Ch: mem, sreg
   case 0x8D:      // 8Dh: LEA reg, mem
   case 0x8E:      // 8Eh: sreg, mem
   case 0x8F:      // 8Fh: POP mem
   case 0xC4:      // C4h: LES reg, mem
   case 0xC5:      // C5h: LDS reg, mem
   case 0xD0:      // D0h: mem8, 1
   case 0xD1:      // D1h: mem, 1
   case 0xD2:      // D2h: mem8, CL
   case 0xD3:      // D3h: mem, CL
   case 0xFE:      // FEh: mem8
   case 0xFF:      // FFh: mem
     pb += DisassembleMem32(pb);
     break;

   case 0x04:      // 04h: ADD AL, imm8
   case 0x0C:      // 0Ch: OR AL, imm8
   case 0x14:      // 14h: ADC AL, imm8
   case 0x1C:      // 1Ch: SBB AL, imm8
   case 0x24:      // 24h: AND AL, imm8
   case 0x2C:      // 2Ch: SUB AL, imm8
   case 0x34:      // 34h: XOR AL, imm8
   case 0x3C:      // 3Ch: CMP AL, imm8
   case 0x6A:      // 6Ah: PUSH simm8
   case 0xA8:      // A8h: TEST AL, imm8
   case 0xB0:      // B0h: MOV AL, imm8
   case 0xB1:      // B1h: MOV CL, imm8
   case 0xB2:      // B2h: MOV DL, imm8
   case 0xB3:      // B3h: MOV BL, imm8
   case 0xB4:      // B4h: MOV AH, imm8
   case 0xB5:      // B5h: MOV CH, imm8
   case 0xB6:      // B6h: MOV DH, imm8
   case 0xB7:      // B7h: MOV BH, imm8
   case 0xD4:      // D4h: AAM imm8
   case 0xD5:      // D5h: AAD imm8
     pb++;
     break;

   case 0x05:      // 05h: ADD EAX, imm
   case 0x0D:      // 0Dh: OR EAX, imm
   case 0x15:      // 15h: ADC EAX, imm
   case 0x1D:      // 1Dh: SBB EAX, imm
   case 0x25:      // 25h: AND EAX, imm
   case 0x2D:      // 2Dh: SUB EAX, imm
   case 0x35:      // 35h: XOR EAX, imm
   case 0x3D:      // 3Dh: CMP EAX, imm
   case 0x68:      // 68h: PUSH imm
   case 0xA9:      // A9h: TEST EAX, imm
   case 0xB8:      // B8h: MOV EAX, imm
   case 0xB9:      // B9h: MOV ECX, imm
   case 0xBA:      // BAh: MOV EDX, imm
   case 0xBB:      // BBh: MOV EBX, imm
   case 0xBC:      // BCh: MOV ESP, imm
   case 0xBD:      // BDh: MOV EBP, imm
   case 0xBE:      // BEh: MOV ESI, imm
   case 0xBF:      // BFh: MOV EDI, imm
     pb += cboperand;
     break;

   case 0x06:      // 06h: PUSH ES
   case 0x07:      // 07h: POP ES
   case 0x0E:      // 0Eh: PUSH CS
   case 0x16:      // 16h: PUSH SS
   case 0x17:      // 17h: POP SS
   case 0x1E:      // 1Eh: PUSH DS
   case 0x1F:      // 1Fh: POP DS
   case 0x26:      // 26h: ES:
   case 0x27:      // 27h: DAA
   case 0x2E:      // 2Eh: CS:
   case 0x2F:      // 2Fh: DAS
   case 0x36:      // 36h: SS:
   case 0x37:      // 37h: AAA
   case 0x3E:      // 3Eh: DS:
   case 0x3F:      // 3Fh: AAS
   case 0x40:      // 40h: INC EAX
   case 0x41:      // 41h: INC ECX
   case 0x42:      // 42h: INC EDX
   case 0x43:      // 43h: INC EBX
   case 0x44:      // 44h: INC ESP
   case 0x45:      // 45h: INC EBP
   case 0x46:      // 46h: INC ESI
   case 0x47:      // 47h: INC EDI
   case 0x48:      // 48h: DEC EAX
   case 0x49:      // 49h: DEC ECX
   case 0x4A:      // 4Ah: DEC EDX
   case 0x4B:      // 4Bh: DEC EBX
   case 0x4C:      // 4Ch: DEC ESP
   case 0x4D:      // 4Dh: DEC EBP
   case 0x4E:      // 4Eh: DEC ESI
   case 0x4F:      // 4Fh: DEC EDI
   case 0x50:      // 50h: PUSH EAX
   case 0x51:      // 51h: PUSH ECX
   case 0x52:      // 52h: PUSH EDX
   case 0x53:      // 53h: PUSH EBX
   case 0x54:      // 54h: PUSH ESP
   case 0x55:      // 55h: PUSH EBP
   case 0x56:      // 56h: PUSH ESI
   case 0x57:      // 57h: PUSH EDI
   case 0x58:      // 58h: POP EAX
   case 0x59:      // 59h: POP ECX
   case 0x5A:      // 5Ah: POP EDX
   case 0x5B:      // 5Bh: POP EBX
   case 0x5C:      // 5Ch: POP ESP
   case 0x5D:      // 5Dh: POP EBP
   case 0x5E:      // 5Eh: POP ESI
   case 0x5F:      // 5Fh: POP EDI
   case 0x60:      // 60h: PUSHAD
   case 0x61:      // 61h: POPAD
   case 0x64:      // 64h: FS:
   case 0x90:      // 90h: NOP
   case 0x91:      // 91h: XCHG EAX, ECX
   case 0x92:      // 92h: XCHG EAX, EDX
   case 0x93:      // 93h: XCHG EAX, EBX
   case 0x94:      // 94h: XCHG EAX, ESP
   case 0x95:      // 95h: XCHG EAX, EBP
   case 0x96:      // 96h: XCHG EAX, ESI
   case 0x97:      // 97h: XCHG EAX, EDI
   case 0x98:      // 98h: CWDE
   case 0x99:      // 99h: CDQ
   case 0x9C:      // 9Ch: PUSHFD
   case 0x9D:      // 9Dh: POPFD
   case 0x9E:      // 9Eh: SAHF
   case 0x9F:      // 9Fh: LAHF
   case 0xA4:      // A4h: MOVSB
   case 0xA5:      // A5h: MOVSD
   case 0xA6:      // A6h: CMPSB
   case 0xA7:      // A7h: CMPSD
   case 0xAA:      // AAh: STOSB
   case 0xAB:      // ABh: STOSD
   case 0xAC:      // ACh: LODSB
   case 0xAD:      // ADh: LODSD
   case 0xAE:      // AEh: SCASB
   case 0xAF:      // AFh: SCASD
   case 0xC9:      // C9h: LEAVE
   case 0xD6:      // D6h: SETALC
   case 0xD7:      // D7h: XLAT
   case 0xF0:      // F0h: LOCK
   case 0xF2:      // F2h: REPNZ
   case 0xF3:      // F3h: REP
   case 0xF5:      // F5h: CMC
   case 0xF8:      // F8h: CLC
   case 0xF9:      // F9h: STC
   case 0xFC:      // FCh: CLD
   case 0xFD:      // FDh: STD
     break;

   case 0x66:      // 66h: memory access size prefix
     cboperand = 2;
     continue;

   case 0x69:      // 69h: IMUL reg, imm, mem
   case 0x81:      // 81h: mem, imm
   case 0xC7:      // C7h: MOV mem, imm
     pb += DisassembleMem32(pb) + cboperand;
     break;

   case 0x6B:      // 6Bh: IMUL reg8, imm8, mem8
   case 0x80:      // 80h: mem8, imm8
   case 0x82:      // 82h: mem8, simm8
   case 0x83:      // 83h: mem, simm8
   case 0xC0:      // C0h: mem8, imm8
   case 0xC1:      // C1h: mem, imm8
   case 0xC6:      // C6h: MOV mem8, imm8
     pb += DisassembleMem32(pb) + 1;
     break;

   case 0xA0:      // A0h: MOV AL, [ofs]
   case 0xA1:      // A1h: MOV EAX, [ofs]
   case 0xA2:      // A2h: MOV [ofs], AL
   case 0xA3:      // A3h: MOV [ofs], EAX
     pb += 4;
     break;

   case 0xC8:      // C8h: ENTER imm16, imm8
     pb += 3;
     break;

   case 0xF6:      // F6h/0: TEST mem8, imm8; F6h/{1..7}: mem8
     pb += DisassembleMem32(pb) + ((*pb & 0x38) == 0x00 ? 1 : 0);
     break;

   case 0xF7:      // F7h/0: TEST mem, imm; F7h/{1..7}: mem
     pb += DisassembleMem32(pb) + ((*pb & 0x38) == 0x00 ? cboperand : 0);
     break;

   case 0x0F:
     switch (*pb++)
     {
     case 0x0D:    // 0Fh/0Dh: mem
     case 0x18:    // 0Fh/18h: mem
     case 0x90:    // 0Fh/90h: SETO mem8
     case 0x91:    // 0Fh/91h: SETNO mem8
     case 0x92:    // 0Fh/92h: SETC mem8
     case 0x93:    // 0Fh/93h: SETNC mem8
     case 0x94:    // 0Fh/94h: SETZ mem8
     case 0x95:    // 0Fh/95h: SETNZ mem8
     case 0x96:    // 0Fh/96h: SETNA mem8
     case 0x97:    // 0Fh/97h: SETA mem8
     case 0x98:    // 0Fh/98h: SETS mem8
     case 0x99:    // 0Fh/99h: SETNS mem8
     case 0x9A:    // 0Fh/9Ah: SETP mem8
     case 0x9B:    // 0Fh/9Bh: SETNP mem8
     case 0x9C:    // 0Fh/9Ch: SETL mem8
     case 0x9D:    // 0Fh/9Dh: SETNL mem8
     case 0x9E:    // 0Fh/9Eh: SETNG mem8
     case 0x9F:    // 0Fh/9Fh: SETG mem8
     case 0xA3:    // 0Fh/A3h: BT mem, reg
     case 0xA5:    // 0Fh/A5h: SHLD mem, reg, CL
     case 0xAB:    // 0Fh/ABh: BTS mem, reg
     case 0xAD:    // 0Fh/ADh: SHRD mem, reg, CL
     case 0xAF:    // 0Fh/AFh: IMUL reg, mem
     case 0xB3:    // 0Fh/B3h: BTR mem, reg
     case 0xB4:    // 0Fh/B4h: LFS reg, mem
     case 0xB5:    // 0Fh/B5h: LGS reg, mem
     case 0xB6:    // 0Fh/B6h: MOVZX reg, mem8
     case 0xB7:    // 0Fh/B7h: MOVZX reg, mem16
     case 0xBB:    // 0Fh/BBh: BTC mem, reg
     case 0xBC:    // 0Fh/BCh: BSF mem, reg
     case 0xBD:    // 0Fh/BDh: BSR mem, reg
     case 0xBE:    // 0Fh/BEh: MOVSX reg, mem8
     case 0xBF:    // 0Fh/BFh: MOVSX reg, mem16
     case 0xC0:    // 0Fh/C0h: XADD mem8, reg8
     case 0xC1:    // 0Fh/C1h: XADD mem, reg
     case 0xC7:    // 0Fh/C7h/0: CMPXCHG8B mem
       pb += DisassembleMem32(pb);
       break;

     case 0xA0:    // 0Fh/A0h: PUSH FS
     case 0xA1:    // 0Fh/A1h: POP FS
     case 0xA8:    // 0Fh/A8h: PUSH GS
     case 0xA9:    // 0Fh/A9h: POP GS
     case 0xC8:    // 0Fh/C8h: BSWAP EAX
     case 0xC9:    // 0Fh/C9h: BSWAP ECX
     case 0xCA:    // 0Fh/CAh: BSWAP EDX
     case 0xCB:    // 0Fh/CBh: BSWAP EBX
     case 0xCC:    // 0Fh/CCh: BSWAP ESP
     case 0xCD:    // 0Fh/CDh: BSWAP EBP
     case 0xCE:    // 0Fh/CEh: BSWAP ESI
     case 0xCF:    // 0Fh/CFh: BSWAP EDI
       break;

     case 0xA4:    // 0Fh/A4h: SHLD mem, reg, imm8
     case 0xAC:    // 0Fh/ACh: SHRD mem, reg, imm8
     case 0xBA:    // 0Fh/BAh: mem, imm8
       pb += DisassembleMem32(pb) + 1;
       break;

     default:
       return 0;
     }
     break; //case 0x0F

   default:
     return 0;
   } //switch(*pb)

   cboperand = 4;
 } //for(pb)

 return (DWORD)(pb - pbCode);
} //DisassembleProlog()

__declspec(naked) void InLineHookHead()
{
    __asm
    {
        POP EAX;
        PUSH 0xAAAAAAAA;            //OrgFunction
        PUSH EAX;
        _emit 0xea;
        _emit 0xbb;
        _emit 0xbb;
        _emit 0xbb;
        _emit 0xbb;
        _emit 0x08;
        _emit 0x00;
        //POP EAX;                    
        //MOV EAX, 0xBBBBBBBB;        //MyHookFunction - 注意调用方式
        //CALL EAX;
        //INT 3;
        //ADD ESP, 8;
        //PUSH dword ptr [ESP - 8];
    }
}
void InLineHookHeadEnd(){}

DWORD
SetupInlineHook(
                IN OUT PBYTE FunctionAddress,
                IN PBYTE NewAddress
    )
{
    PINLINE_HOOK_ITEM hook_item = NULL;
    PBYTE jmp_back = NULL;

    UINT i, call_head_len;
    
    if (!FunctionAddress || !NewAddress)
        return 0;

    hook_item = ExAllocatePool(NonPagedPool, sizeof(INLINE_HOOK_ITEM));
    hook_item->HookAddress = FunctionAddress;
    hook_item->OrgBytesSize = DisassembleProlog(FunctionAddress, 7);    //JMP FAR 0008:0xffffffff
    call_head_len = (ULONG)InLineHookHeadEnd - (ULONG)InLineHookHead;//头字节数

    hook_item->HookBytes = ExAllocatePool(NonPagedPool, hook_item->OrgBytesSize + call_head_len + 7);
    //------------------------------------------------------------
    //填写HookHeader Bytes
    memcpy(hook_item->HookBytes, (PBYTE)InLineHookHead, call_head_len);
    for (i=0; i<call_head_len; i++)
    {
        if (*(PULONG)&hook_item->HookBytes == 0xAAAAAAAA)
        {
            *(PULONG)&hook_item->HookBytes = (ULONG)hook_item->HookBytes + call_head_len;
            i += 4;
        }
        if (*(PULONG)&hook_item->HookBytes == 0xBBBBBBBB)
        {
            *(PULONG)&hook_item->HookBytes = (ULONG)NewAddress;
            i += 4;
        }
    }
    //here is old bytes
    memcpy(hook_item->HookBytes + call_head_len, FunctionAddress, hook_item->OrgBytesSize);
    //7 bytes jmp back
    jmp_back = hook_item->HookBytes + call_head_len + hook_item->OrgBytesSize;
    jmp_back[0] = 0xEA;
    *(PULONG)&jmp_back[1] = (ULONG)(FunctionAddress + hook_item->OrgBytesSize);
    jmp_back[5] = 0x08;
    jmp_back[6] = 0x00;
    //------------------------------------------------------------
    //setup inline hook
    //------------------------------------------------------------
    _asm
    {
        MOV EAX, CR0;            //move CR0 register into EAX
        AND EAX, NOT 10000H;    //disable WP bit
        MOV CR0, EAX;            //write register back
    }
    hook_item->HookAddress[0] = 0xEA;//JMP FAR 0008:0xffffffff
    *(PULONG)&hook_item->HookAddress[1] = (ULONG)hook_item->HookBytes;
    hook_item->HookAddress[5] = 0x08;
    hook_item->HookAddress[6] = 0x00;
    for (i=7; i<hook_item->OrgBytesSize; i++)//fill nop
        hook_item->HookAddress = 0x90;
    _asm
    {
        MOV EAX, CR0;
        OR  EAX, 10000H;
        MOV CR0, EAX;
    }
    //------------------------------------------------------------
    KdPrint(("Hook At %08x\n", FunctionAddress));
    
    return (DWORD)hook_item;
}

//下面是简单的测试--win2000以及win2003SP1测试通过

SetupInlineHook((PBYTE)PacketSend, (PBYTE)InLineHookPacketSend);

NDIS_STATUS
InLineHookPacketSend(
                     SEND_HANDLER                OrgPacketSend,
                     IN    NDIS_HANDLE                MacBindingHandle,
                     IN    PNDIS_PACKET            Packet
                     )
{

    KdPrint(("InLineHookPacketSend %08x, %08x, %08x!!!\n", OrgPacketSend, MacBindingHandle, Packet));
    return OrgPacketSend(MacBindingHandle, Packet);    
}

最新喜欢:

SamSunSamSun
boywhp
驱动中牛
驱动中牛
  • 注册日期2007-08-09
  • 最后登录2015-04-24
  • 粉丝2
  • 关注0
  • 积分1105分
  • 威望515点
  • 贡献值0点
  • 好评度254点
  • 原创分1分
  • 专家分0分
沙发#
发布于:2008-04-22 23:07
我写的时候走了好多弯路,在堆栈一块出了不少的问题,另外我没有写Unhook的函数,会用这个代码的人自然会完善它
eleqi
驱动小牛
驱动小牛
  • 注册日期2005-12-20
  • 最后登录2014-01-03
  • 粉丝4
  • 关注2
  • 积分172分
  • 威望1475点
  • 贡献值0点
  • 好评度115点
  • 原创分0分
  • 专家分0分
板凳#
发布于:2008-04-22 23:14
学习,谢谢!
GoodOnline
驱动小牛
驱动小牛
  • 注册日期2007-04-11
  • 最后登录2009-02-28
  • 粉丝0
  • 关注0
  • 积分4分
  • 威望204点
  • 贡献值0点
  • 好评度191点
  • 原创分0分
  • 专家分0分
地板#
发布于:2008-04-23 10:11
赞扬一下,加油啊. 要是能发个可编译,可测试的就更好了。

有一个问题就是 DEP(执行数据保护),估计会击败你的code。
所以你可以使用一些不再被系统使用的地址例如:Vidin**来替换你的ExAllocatePool函数。不过万一被别人再次利用就。。。。
Hook 看来确实值得鄙视。哈哈
boywhp
驱动中牛
驱动中牛
  • 注册日期2007-08-09
  • 最后登录2015-04-24
  • 粉丝2
  • 关注0
  • 积分1105分
  • 威望515点
  • 贡献值0点
  • 好评度254点
  • 原创分1分
  • 专家分0分
地下室#
发布于:2008-04-23 19:33
BS 卡巴的HOOK!!!居然不断检查自己的HOOK,造成死循环
 
classfree
驱动小牛
驱动小牛
  • 注册日期2004-05-23
  • 最后登录2019-01-06
  • 粉丝1
  • 关注1
  • 积分873分
  • 威望276点
  • 贡献值0点
  • 好评度85点
  • 原创分0分
  • 专家分0分
  • 社区居民
5楼#
发布于:2008-05-25 09:33
老大, 差个头文件哦
低调点!
WQXNETQIQI
驱动大牛
驱动大牛
  • 注册日期2006-06-12
  • 最后登录2010-10-26
  • 粉丝0
  • 关注0
  • 积分13分
  • 威望1076点
  • 贡献值0点
  • 好评度895点
  • 原创分1分
  • 专家分0分
6楼#
发布于:2008-05-25 18:08
inline drop是史上最垃圾最挫的HOOK方式了
驱动开发者 呵呵
boywhp
驱动中牛
驱动中牛
  • 注册日期2007-08-09
  • 最后登录2015-04-24
  • 粉丝2
  • 关注0
  • 积分1105分
  • 威望515点
  • 贡献值0点
  • 好评度254点
  • 原创分1分
  • 专家分0分
7楼#
发布于:2008-05-26 10:58
QIQI一般用啥好办法?能不跟卡巴冲突也行
safejmp
驱动牛犊
驱动牛犊
  • 注册日期2008-07-27
  • 最后登录2010-09-21
  • 粉丝0
  • 关注0
  • 积分6分
  • 威望25点
  • 贡献值0点
  • 好评度0点
  • 原创分0分
  • 专家分0分
8楼#
发布于:2008-07-28 10:37
什么东西?
还会冲突
游客

返回顶部