|
陷阱技术探秘----动态汉化Windows技术的分析(8) 四、“陷阱”技术 讨论“陷阱”技术,还要回到前面的两个发现。发现之二,已能解释为修改的Windows函数,而发现之一,却仍是一个迷。 数据段存放的是变量及常量等内容,如果这里面包含有重定位信息,那么,必定要在变量说明中将函数指针赋给一个FARPROC类型的变量,于是,在变量说明中写下: FARPROCFarProcFunc=ExtTextOut; 果然,我自己程序的数据段中也有了重定位信息。这样,当程序调入内存中时,变量FarProcFunc已是函数ExtTextOut的地址了。
要直接修改代码段的内容,还遇到一个难题,就是代码段是不可改写的。这时,需要用到一个未公开的Windows函数AllocCStoDSAlias取得与代码段有相同基址的可写数据段别名,其函数声明为 WORDFARPASCALAllocCStoDSAlias(WORDcode_sel); 参数是代码段的句柄,返回值是可写数据段别名句柄。
Windows中函数地址是32位,高字是其模块的内存句柄,低字是函数在模块内的偏移。将得到的可写数据段别名句柄锁定,再将函数偏移处的5个字节保留下来,然后将其改为转向替代函数(用EAJmp)
*(lpStr+wOffset)=0xEA; *(lpStr+wOffset+1)=lpFarProcReplace;
反汇编即是JmplpFarProcReplace,最后,内存解锁。 这就是我们为Windows设的“陷阱”,当所有对此函数的调用都无条件地转到我们规定的替代函数处。当程序结束之前,将保留的5字节内容再置回来,否则,系统会崩溃。
下面给出作者编写的使Windows的ExtTextOut函数落入自己函数“陷阱”的源程序。
//源程序 relocate.c #include #include
BOOL WINAPI MyExtTextOut(HDC hDC, int x, int y, UINT nInt1, const RECT FAR* lpRect,LPCSTR lpStr, UINT nInt2, int FAR* lpInt); WORD FAR PASCAL AllocCStoDSAlias(WORD code_sel);
typedef struct tagFUNC { FARPROC lpFarProcReplace;//替代函数地址 FARPROC lpFarProcWindows;//Windows函数地址 BYTEbOld;//保存原函数第一字节 LONGlOld;//保存原函数接后的四字节长值 }FUNC;
FUNCFunc={MyExtTextOut,ExtTextOut};
//Windows主函数 int PASCAL WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance, LPSTR lpCmdLine,int nCmdShow) { HANDLE hMemCode;//代码段句柄 WORD hMemData;//相同基址的可写数据段别名 WORD wOffset; //函数偏移 LPSTRlpStr; LPLONG lpLong; char lpNotice[96];
hMemCode=HIWORD((LONG) Func.lpFarProcWindows ); wOffset=LOWORD((LONG) Func.lpFarProcWindows );
wsprintf(lpNotice,"函数所在模块句柄 0x%4xH,偏移 0x%4xH", hMemCode,wOffset); MessageBox(NULL,lpNotice,"提示",MB_OK);
//取与代码段有相同基址的可写数据段别名 hMemData=AllocCStoDSAlias(hMemCode);
lpStr=GlobalLock(hMemData);
lpLong=(lpStr+wOffset+1 ); //保存原函数要替换的头几个字节 Func.bOld=*(lpStr+wOffset); Func.lOld=*lpLong;
|