壳编写

来源:互联网 发布:淘宝详情页生成器 编辑:程序博客网 时间:2024/04/30 02:55
作为一种主要的软件保护手段大概可以分为压缩壳和加密壳两类。而现在
流行的加壳程序无论是压缩的还是加密的几乎都是针对应用层程序的,对于驱动
程序的保护壳则几乎是空白。笔者只在一些国外加密狗的驱动中见过类似应用层
的保护壳。本篇文章主要介绍驱动加壳程序与应用层加壳程序在编写上的区别以
及一些注意事项。

1.校验和的计算

    驱动程序被加壳后必须重新进行校验和的计算,否则加壳后的驱动加载会
失败

;*****************计算pe文件校验和*********************

CalcPECheckSum PROC lpBaseAddr:DWORD,dwFileSize:DWORD
LOCAL  CheckSum:DWORD
pushad
mov    ecx,dwFileSize
        inc    ecx
        shr    ecx,1
        xor    eax,eax
        clc
        mov    esi,lpBaseAddr
    cal_checksum:
        adc    ax,word ptr [esi]
inc    esi
inc    esi
        loop   cal_checksum
        mov    ebx,dwFileSize
        add    eax,ebx
        mov    CheckSum,eax
        popad
        mov    eax,CheckSum 
ret
CalcPECheckSum endp

;*******************************************************
  
2.原始IAT的处理

    由于原驱动程序被加上了我们的外壳,所以原驱动程序的IAT表的填写工作
要由我们的外壳程序来完成。应用层壳一般通过GetModuleHandle和GetProcAdd
ress两个API来完成这个工作,或者自己实现这两个API的功能。而驱动壳是要
随驱动程序一起被加载到内核当中去的,但内核里没有这两个函数,需要我们
自己对这两个函数做内核的实现。当然也可以用MmGetSystemRoutineAddress函
数,不过它只能得到ntoskrnl.exe和hal.dll两个模块的函数,对于其它模块则
无能为力了,影响壳的通用性。

    壳的GetModuleHandle函数可以通过遍历PsLoadedModuleList链表来实现,
关于遍历这个链表的方法可以参照Futo的代码,通过DRIVEROBJECT的DriverSec
tion成员来完成,而驱动对象可以从堆栈当中找到。

    壳的GetProcAddress函数的实现就很简单了,内核模块本身也是PE文件,
直接遍历一下PE的导出表就ok了。

    还有一点需要注意的就是UNICODE的转换,PE文件里面的字符串是以ASCII
方式存储的,而内核里的字符串多半是用UNICODE方式存放的,这点需要注意。

3.节表的处理

    在给程序加壳的时候一般都要添加新节,用于存放壳的代码,应用层程序
的节表的最后一项和第一个节之间一般是有一个很大的空间可以用来添加新的
节表项的,但一般情况下驱动程序节表的最后一项后面紧接着就是第一个节,
根本没有足够的0x28大小的空间存放新的节表项。解决的方法有两种,第一种
将所有的节向后移动,而第二种方法则是将PE头整体向前移动覆盖掉部分无用
的dos头,留出足够的空间存放新的节表项。

    另外一点需要注意的是,加壳后的驱动程序的每一个节表项必须满足如下
两个公式,才能被系统正常加载

    1) VirtualAddress == PointerToRawData
    2) SizeOfRawData >= VirtualSize

    至于为什么,笔者也没搞清楚,这只是笔者通过分析驱动加载代码及实验
的结果,哪位仁兄知道还请赐教:)

4.重定位表的处理

    由于驱动程序是要被加载到内核空间中,所以外壳必须实现原来由系统完
成的原驱动程序的重定位工作。原驱动程序的重定位表的处理方法跟应用层 D
LL 文件的处理方法完全一样,代码如下:

    mov  eax,dword ptr [ebp+OriginalRelocateAddr]
add  eax,dword ptr [ebp+ModuleHandle]
mov  ecx,dword ptr [ebp+OriginalRelocateSize]
mov  ebx,eax
mov  esi,dword ptr [ebp+ModuleHandle]
sub  esi,dword ptr [ebp+OriginalBaseAddr] ;esi=diff

NextRelocateBlock:
.if ecx == 0
jmp FixAllRelocate
.endif
assume ebx : ptr IMAGE_BASE_RELOCATION
push ecx
mov  ecx,dword ptr [ebx].SizeOfBlock
sub  ecx,sizeof IMAGE_BASE_RELOCATION
shr  ecx,1
mov  eax,ebx
add  eax,sizeof IMAGE_BASE_RELOCATION
NextRelocateEntry:
xor edi,edi
mov di,word ptr [eax]
shr edi,12
.if edi == IMAGE_REL_BASED_HIGHLOW
movzx edi,word ptr [eax]
and edi,0fffh
add edi,dword ptr [ebx].VirtualAddress
add edi,dword ptr [ebp+ModuleHandle]
add dword ptr [edi],esi
.endif
add eax,2
loop NextRelocateEntry
pop  ecx
sub  ecx,dword ptr [ebx].SizeOfBlock
add  ebx,dword ptr [ebx].SizeOfBlock
jmp NextRelocateBlock
FixAllRelocate:

    需要注意的一点:驱动程序被加壳后必须要有重定位表,否则驱动加载会失
败,解决的方法需要自己构造一个假的重定位表来替换原始的重定位表。
    
    另外,由于驱动壳的特殊性,shell的编写和驱动程序的编写没什么区别,

稍有错误就会蓝屏。

转载自:http://lwglucky.blog.51cto.com/1228348/291914

0 0
原创粉丝点击