initial
This commit is contained in:
		
							
								
								
									
										238
									
								
								inc/tsfuncs/BlFuncs.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										238
									
								
								inc/tsfuncs/BlFuncs.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,238 @@ | ||||
|  | ||||
| ////////////////////////////////////////////////// | ||||
| // BlFuncs Version 1.0 | ||||
|  | ||||
|  | ||||
| // Includes | ||||
|  | ||||
| #include "BlHooks.hpp" | ||||
| #include "BlFuncs.hpp" | ||||
|  | ||||
| #include <stdio.h> | ||||
|  | ||||
|  | ||||
| // Scanned structures | ||||
|  | ||||
| ADDR tsf_mCacheSequence; | ||||
| ADDR tsf_mCacheAllocator; | ||||
| ADDR tsf_gIdDictionary; | ||||
| ADDR tsf_gEvalState_globalVars; | ||||
|  | ||||
| BlFunctionDefIntern(tsf_BlStringTable__insert        ); | ||||
| BlFunctionDefIntern(tsf_BlNamespace__find            ); | ||||
| BlFunctionDefIntern(tsf_BlNamespace__createLocalEntry); | ||||
| BlFunctionDefIntern(tsf_BlDataChunker__freeBlocks    ); | ||||
| BlFunctionDefIntern(tsf_BlCon__evaluate              ); | ||||
| BlFunctionDefIntern(tsf_BlCon__executef              ); | ||||
| BlFunctionDefIntern(tsf_BlCon__executefSimObj        ); | ||||
| BlFunctionDefIntern(tsf_BlCon__getVariable           ); | ||||
| BlFunctionDefIntern(tsf_BlDictionary__addVariable    ); | ||||
| BlFunctionDefIntern(tsf_BlSim__findObject_name       ); | ||||
| BlFunctionDefIntern(tsf_BlStringStack__getArgBuffer  ); | ||||
| BlFunctionDefIntern(tsf_BlSimObject__getDataField    ); | ||||
| BlFunctionDefIntern(tsf_BlSimObject__setDataField    ); | ||||
| BlFunctionDefIntern(tsf_BlCon__getReturnBuffer       ); | ||||
|  | ||||
|  | ||||
| // C->TS Args | ||||
|  | ||||
| char* tsf_GetIntArg(signed int value) { | ||||
| 	char* ret = tsf_BlStringStack__getArgBuffer(16); | ||||
| 	snprintf(ret, 16, "%d", value); | ||||
| 	return ret; | ||||
| } | ||||
| char* tsf_GetFloatArg(float value) { | ||||
| 	char* ret = tsf_BlStringStack__getArgBuffer(32); | ||||
| 	snprintf(ret, 32, "%g", value); | ||||
| 	return ret; | ||||
| } | ||||
| char* tsf_GetStringArg(char* value) { | ||||
| 	int len = strlen(value)+1; | ||||
| 	char* ret = tsf_BlStringStack__getArgBuffer(len); | ||||
| 	memcpy(ret, value, len); | ||||
| 	return ret; | ||||
| } | ||||
| char* tsf_GetThisArg(ADDR obj) { | ||||
| 	return tsf_GetIntArg(*(signed int *)(obj + 32)); | ||||
| } | ||||
|  | ||||
|  | ||||
| // Eval | ||||
|  | ||||
| const char* tsf_Eval(const char *code) { | ||||
| 	const char *argv[] = {nullptr, code}; | ||||
| 	return tsf_BlCon__evaluate(0, 2, argv); | ||||
| } | ||||
|  | ||||
| const char* tsf_Evalf(const char *fmt, ...) { | ||||
| 	va_list args; | ||||
| 	char code[4096]; | ||||
| 	va_start(args, fmt); | ||||
| 	vsnprintf(code, 4096, fmt, args); | ||||
| 	va_end(args); | ||||
| 	 | ||||
| 	return tsf_Eval((const char*)code); | ||||
| } | ||||
|  | ||||
|  | ||||
| // Objects | ||||
|  | ||||
| ADDR tsf_FindObject(unsigned int id) { | ||||
| 	ADDR obj = *(ADDR*)(*(ADDR*)(tsf_gIdDictionary) + 4*(id & 0xFFF)); | ||||
| 	if(!obj) return 0; | ||||
| 	 | ||||
| 	while(obj && *(unsigned int *)(obj + 32) != id) { | ||||
| 		obj = *(ADDR*)(obj + 16); | ||||
| 		if(!obj) return 0; | ||||
| 	} | ||||
| 	 | ||||
| 	return obj; | ||||
| } | ||||
|  | ||||
| ADDR tsf_FindObject(const char* name) { | ||||
| 	return (ADDR)tsf_BlSim__findObject_name(name); | ||||
| } | ||||
|  | ||||
| ADDR tsf_LookupNamespace(const char* ns, const char* package) { | ||||
| 	const char* ste_package; | ||||
| 	if(package) { | ||||
| 		ste_package = tsf_BlStringTable__insert(package, 0); | ||||
| 	} else { | ||||
| 		ste_package = nullptr; | ||||
| 	} | ||||
| 	 | ||||
| 	if(ns) { | ||||
| 		const char* ste_namespace = tsf_BlStringTable__insert(ns, 0); | ||||
| 		return tsf_BlNamespace__find(ste_namespace, ste_package); | ||||
| 	} else { | ||||
| 		return tsf_BlNamespace__find(nullptr, ste_package); | ||||
| 	} | ||||
| } | ||||
| ADDR tsf_LookupNamespace(const char* ns) { | ||||
| 	return tsf_LookupNamespace(ns, nullptr); | ||||
| } | ||||
|  | ||||
|  | ||||
| // Object Fields | ||||
|  | ||||
| const char* tsf_GetDataField(ADDR simObject, const char* slotName, const char* array) { | ||||
| 	const char *ste_slotName; | ||||
| 	if(slotName) { | ||||
| 		ste_slotName = tsf_BlStringTable__insert(slotName, 0); | ||||
| 	} else { | ||||
| 		ste_slotName = nullptr; | ||||
| 	} | ||||
| 	 | ||||
| 	return tsf_BlSimObject__getDataField(simObject, ste_slotName, array); | ||||
| } | ||||
|  | ||||
| void tsf_SetDataField(ADDR simObject, const char* slotName, const char* array, const char* value) { | ||||
| 	const char* ste_slotName; | ||||
| 	if(slotName) { | ||||
| 		ste_slotName = tsf_BlStringTable__insert(slotName, 0); | ||||
| 	} else { | ||||
| 		ste_slotName = nullptr; | ||||
| 	} | ||||
| 	 | ||||
| 	tsf_BlSimObject__setDataField(simObject, ste_slotName, array, value); | ||||
| } | ||||
|  | ||||
|  | ||||
| // TS Global Variables | ||||
|  | ||||
| const char *tsf_GetVar(const char* name) { | ||||
| 	return tsf_BlCon__getVariable(name); | ||||
| } | ||||
|  | ||||
| void tsf_AddVarInternal(const char* name, signed int varType, void* data) { | ||||
| 	tsf_BlDictionary__addVariable((ADDR *)tsf_gEvalState_globalVars, name, varType, data); | ||||
| } | ||||
|  | ||||
| void tsf_AddVar(const char* name, const char** data) { | ||||
| 	tsf_AddVarInternal(name, 10, data); | ||||
| } | ||||
| void tsf_AddVar(const char* name, signed int* data) { | ||||
| 	tsf_AddVarInternal(name,  4, data); | ||||
| } | ||||
| void tsf_AddVar(const char* name, float* data) { | ||||
| 	tsf_AddVarInternal(name,  8, data); | ||||
| } | ||||
| void tsf_AddVar(const char* name, bool* data) { | ||||
| 	tsf_AddVarInternal(name,  6, data); | ||||
| } | ||||
|  | ||||
|  | ||||
| // TS->C Functions | ||||
|  | ||||
| ADDR tsf_AddConsoleFuncInternal(const char* pname, const char* cname, const char* fname, signed int cbtype, const char* usage, signed int mina, signed int maxa) { | ||||
| 	const char *ste_fname = tsf_BlStringTable__insert(fname, 0); | ||||
| 	ADDR ns = tsf_LookupNamespace(cname, pname); | ||||
| 	ADDR ent = tsf_BlNamespace__createLocalEntry(ns, ste_fname); | ||||
| 	 | ||||
| 	*(signed int *)tsf_mCacheSequence += 1; | ||||
| 	tsf_BlDataChunker__freeBlocks(*(ADDR *)tsf_mCacheAllocator); | ||||
| 	 | ||||
| 	*(const char**)(ent + 24) = usage ; | ||||
| 	*(signed int* )(ent + 16) = mina  ; | ||||
| 	*(signed int* )(ent + 20) = maxa  ; | ||||
| 	*(signed int* )(ent + 12) = cbtype; | ||||
| 	 | ||||
| 	return ent; | ||||
| } | ||||
|  | ||||
| void tsf_AddConsoleFunc(const char* pname, const char* cname, const char* fname, tsf_StringCallback sc, const char* usage, signed int mina, signed int maxa) { | ||||
| 	ADDR ent = tsf_AddConsoleFuncInternal(pname, cname, fname, 1, usage, mina, maxa); | ||||
| 	*(tsf_StringCallback *)(ent + 40) = sc; | ||||
| } | ||||
| void tsf_AddConsoleFunc(const char* pname, const char* cname, const char* fname, tsf_IntCallback    ic, const char*  usage, signed int mina, signed int maxa) { | ||||
| 	ADDR ent = tsf_AddConsoleFuncInternal(pname, cname, fname, 2, usage, mina, maxa); | ||||
| 	*(tsf_IntCallback    *)(ent + 40) = ic; | ||||
| } | ||||
| void tsf_AddConsoleFunc(const char* pname, const char* cname, const char* fname, tsf_FloatCallback  fc, const char*  usage, signed int mina, signed int maxa) { | ||||
| 	ADDR ent = tsf_AddConsoleFuncInternal(pname, cname, fname, 3, usage, mina, maxa); | ||||
| 	*(tsf_FloatCallback  *)(ent + 40) = fc; | ||||
| } | ||||
| void tsf_AddConsoleFunc(const char* pname, const char* cname, const char* fname, tsf_VoidCallback   vc, const char*  usage, signed int mina, signed int maxa) { | ||||
| 	ADDR ent = tsf_AddConsoleFuncInternal(pname, cname, fname, 4, usage, mina, maxa); | ||||
| 	*(tsf_VoidCallback   *)(ent + 40) = vc; | ||||
| } | ||||
| void tsf_AddConsoleFunc(const char* pname, const char* cname, const char* fname, tsf_BoolCallback   bc, const char*  usage, signed int mina, signed int maxa) { | ||||
| 	ADDR ent = tsf_AddConsoleFuncInternal(pname, cname, fname, 5, usage, mina, maxa); | ||||
| 	*(tsf_BoolCallback   *)(ent + 40) = bc; | ||||
| } | ||||
|  | ||||
|  | ||||
| // Initialization | ||||
|  | ||||
| bool tsf_InitInternal() { | ||||
| 	BlScanFunctionText(tsf_BlStringTable__insert        , "83 EC 0C 80 3D ? ? ? ? ?"                                                                                                         ); | ||||
| 	BlScanFunctionText(tsf_BlNamespace__find            , "55 8B EC 6A FF 68 ? ? ? ? 64 A1 ? ? ? ? 50 83 EC 0C 53 56 57 A1 ? ? ? ? 33 C5 50 8D 45 F4 64 A3 ? ? ? ? 8B DA 8B D1"              ); | ||||
| 	BlScanFunctionText(tsf_BlNamespace__createLocalEntry, "55 8B EC 6A FF 68 ? ? ? ? 64 A1 ? ? ? ? 50 83 EC 08 53 56 57 A1 ? ? ? ? 33 C5 50 8D 45 F4 64 A3 ? ? ? ? 89 4D F0"                 ); | ||||
| 	BlScanFunctionText(tsf_BlDataChunker__freeBlocks    , "55 8B EC 6A FF 68 ? ? ? ? 64 A1 ? ? ? ? 50 51 53 56 57 A1 ? ? ? ? 33 C5 50 8D 45 F4 64 A3 ? ? ? ? 8B D9 8B 33"                    ); | ||||
| 	BlScanFunctionText(tsf_BlCon__evaluate              , "55 8B EC 6A FF 68 ? ? ? ? 64 A1 ? ? ? ? 50 56 57 A1 ? ? ? ? 33 C5 50 8D 45 F4 64 A3 ? ? ? ? 8B 75 10"                             ); | ||||
| 	BlScanFunctionText(tsf_BlCon__executef              , "81 EC ? ? ? ? A1 ? ? ? ? 33 C4 89 84 24 ? ? ? ? 53 55 56 8B B4 24 ? ? ? ? 33 C9"                                                  ); | ||||
| 	BlScanFunctionText(tsf_BlCon__executefSimObj        , "81 EC ? ? ? ? A1 ? ? ? ? 33 C4 89 84 24 ? ? ? ? 53 56 8B B4 24 ? ? ? ? 33 C9"                                                     ); | ||||
| 	BlScanFunctionText(tsf_BlCon__getVariable           , "53 56 8B F1 57 85 F6 0F 84 ? ? ? ?"                                                                                               ); | ||||
| 	BlScanFunctionText(tsf_BlDictionary__addVariable    , "8B 44 24 04 56 57 8B F9"                                                                                                          ); | ||||
| 	BlScanFunctionText(tsf_BlSim__findObject_name       , "57 8B F9 8A 17"                                                                                                                   ); | ||||
| 	BlScanFunctionText(tsf_BlStringStack__getArgBuffer  , "55 8B EC 83 E4 F8 8B 0D ? ? ? ? A1 ? ? ? ? 56 57 8B 7D 08 8D 14 01 03 D7 3B 15 ? ? ? ? 72 2C 8B 0D"                               ); | ||||
| 	BlScanFunctionText(tsf_BlSimObject__getDataField    , "51 53 8B D9 55 56 8B 74 24 14"                                                                                                    ); | ||||
| 	BlScanFunctionText(tsf_BlSimObject__setDataField    , "81 EC ? ? ? ? A1 ? ? ? ? 33 C4 89 84 24 ? ? ? ? 8B 84 24 ? ? ? ? 53 8B D9 89 44 24 04"                                            ); | ||||
| 	BlScanFunctionText(tsf_BlCon__getReturnBuffer       , "81 F9 ? ? ? ? 76 2B"                                                                                                              ); | ||||
| 	 | ||||
| 	ADDR BlScanText   (tsf_mCacheSequenceLoc            , "FF 05 ? ? ? ? B9 ? ? ? ? 8B F8 E8 ? ? ? ? 8B 44 24 1C 89 47 18 8B 44 24 14"                                                       ); | ||||
| 	ADDR BlScanText   (tsf_mCacheAllocatorLoc           , "89 35 ? ? ? ? C7 06 ? ? ? ? A1 ? ? ? ? 68 ? ? ? ? C7 40 ? ? ? ? ? E8 ? ? ? ? 83 C4 04 8B 4D F4 64 89 0D ? ? ? ? 59 5E 8B E5 5D C3"); | ||||
| 	ADDR BlScanText   (tsf_gIdDictionaryLoc             , "89 15 ? ? ? ? E8 ? ? ? ? 8B F0 89 75 F0"                                                                                          ); | ||||
| 	ADDR BlScanText   (tsf_gEvalState_globalVarsLoc     , "B9 ? ? ? ? E8 ? ? ? ? 68 ? ? ? ? 6A 0A 68 ? ? ? ? B9 ? ? ? ? E8 ? ? ? ? E8 ? ? ? ?"                                               ); | ||||
| 	 | ||||
| 	tsf_mCacheSequence        = *(ADDR*)(tsf_mCacheSequenceLoc        + 2); | ||||
| 	tsf_mCacheAllocator       = *(ADDR*)(tsf_mCacheAllocatorLoc       + 2); | ||||
| 	tsf_gIdDictionary         = *(ADDR*)(tsf_gIdDictionaryLoc         + 2); | ||||
| 	tsf_gEvalState_globalVars = *(ADDR*)(tsf_gEvalState_globalVarsLoc + 1); | ||||
| 	 | ||||
| 	return true; | ||||
| } | ||||
|  | ||||
| bool tsf_DeinitInternal() { | ||||
| 	return true; | ||||
| } | ||||
							
								
								
									
										109
									
								
								inc/tsfuncs/BlFuncs.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										109
									
								
								inc/tsfuncs/BlFuncs.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,109 @@ | ||||
|  | ||||
| ////////////////////////////////////////////////// | ||||
| // BlFuncs Version 1.0 | ||||
|  | ||||
| #ifndef _H_BLFUNCS | ||||
| #define _H_BLFUNCS | ||||
|  | ||||
| // Require BlHooks to be included before this header | ||||
| #ifndef _H_BLHOOKS | ||||
| 	#error "BlFuncs.hpp: You must include BlHooks.hpp first" | ||||
| #else | ||||
|  | ||||
| typedef const char * (*tsf_StringCallback)(ADDR, signed int, const char *[]); | ||||
| typedef signed int   (*tsf_IntCallback   )(ADDR, signed int, const char *[]); | ||||
| typedef float        (*tsf_FloatCallback )(ADDR, signed int, const char *[]); | ||||
| typedef void         (*tsf_VoidCallback  )(ADDR, signed int, const char *[]); | ||||
| typedef bool         (*tsf_BoolCallback  )(ADDR, signed int, const char *[]); | ||||
|  | ||||
| /* These functions are used for tsf_BlCon__executefSimObj. | ||||
|    They refer to a special buffer for the argument stack. | ||||
|    For tsf_BlCon__executef, you need to use your own buffers. */ | ||||
| char *tsf_GetIntArg(int); | ||||
| char *tsf_GetFloatArg(float); | ||||
| char *tsf_ScriptThis(ADDR); | ||||
|  | ||||
| const char *tsf_Eval(const char *); | ||||
| const char *tsf_Evalf(const char *, ...); | ||||
|  | ||||
| ADDR tsf_FindObject(unsigned int); | ||||
| ADDR tsf_FindObject(const char *); | ||||
|  | ||||
| ADDR tsf_LookupNamespace(const char *, const char *); | ||||
|  | ||||
| const char *tsf_GetDataField(ADDR, const char *, const char *); | ||||
| void tsf_SetDataField(ADDR, const char *, const char *, const char *); | ||||
|  | ||||
| const char *tsf_GetVar(const char *); | ||||
|  | ||||
| void tsf_AddVarInternal(const char *, signed int, void *); | ||||
| void tsf_AddVar(const char *, const char **); | ||||
| void tsf_AddVar(const char *, signed int *); | ||||
| void tsf_AddVar(const char *, float *); | ||||
| void tsf_AddVar(const char *, bool *); | ||||
|  | ||||
| ADDR tsf_AddConsoleFuncInternal(const char *, const char *, const char *, signed int, const char *, signed int, signed int); | ||||
| void tsf_AddConsoleFunc(const char *, const char *, const char *, tsf_StringCallback, const char *, signed int, signed int); | ||||
| void tsf_AddConsoleFunc(const char *, const char *, const char *, tsf_IntCallback, const char *, signed int, signed int); | ||||
| void tsf_AddConsoleFunc(const char *, const char *, const char *, tsf_FloatCallback, const char *, signed int, signed int); | ||||
| void tsf_AddConsoleFunc(const char *, const char *, const char *, tsf_VoidCallback, const char *, signed int, signed int); | ||||
| void tsf_AddConsoleFunc(const char *, const char *, const char *, tsf_BoolCallback, const char *, signed int, signed int); | ||||
|  | ||||
| bool tsf_InitInternal(); | ||||
|  | ||||
| extern ADDR tsf_mCacheSequence; | ||||
| extern ADDR tsf_mCacheAllocator; | ||||
| extern ADDR tsf_gIdDictionary; | ||||
| extern ADDR tsf_gEvalState_globalVars; | ||||
|  | ||||
| BlFunctionDefExtern(const char *, __stdcall, tsf_BlStringTable__insert, const char *, bool); | ||||
| BlFunctionDefExtern(ADDR, __fastcall, tsf_BlNamespace__find, const char *, const char *); | ||||
| BlFunctionDefExtern(ADDR, __thiscall, tsf_BlNamespace__createLocalEntry, ADDR, const char *); | ||||
| BlFunctionDefExtern(void, __thiscall, tsf_BlDataChunker__freeBlocks, ADDR); | ||||
| BlFunctionDefExtern(const char *, , tsf_BlCon__evaluate, ADDR, signed int, const char **); | ||||
| BlFunctionDefExtern(const char *, , tsf_BlCon__executef, signed int, ...); | ||||
| BlFunctionDefExtern(const char *, , tsf_BlCon__executefSimObj, ADDR *, signed int, ...); | ||||
| BlFunctionDefExtern(const char *, __thiscall, tsf_BlCon__getVariable, const char *); | ||||
| BlFunctionDefExtern(void, __thiscall, tsf_BlDictionary__addVariable, ADDR *, const char *, signed int, void *); | ||||
| BlFunctionDefExtern(ADDR *, __thiscall, tsf_BlSim__findObject_name, const char *); | ||||
| BlFunctionDefExtern(char *, __stdcall, tsf_BlStringStack__getArgBuffer, unsigned int); | ||||
| BlFunctionDefExtern(const char *, __thiscall, tsf_BlSimObject__getDataField, ADDR, const char *, const char *); | ||||
| BlFunctionDefExtern(void, __thiscall, tsf_BlSimObject__setDataField, ADDR, const char *, const char *, const char *); | ||||
| BlFunctionDefExtern(char *, __fastcall, tsf_BlCon__getReturnBuffer, unsigned int); | ||||
|  | ||||
|  | ||||
| // Function short names | ||||
|  | ||||
| #define BlEval tsf_Eval | ||||
| #define BlEvalf tsf_Evalf | ||||
|  | ||||
| #define BlIntArg tsf_GetIntArg | ||||
| #define BlFloatArg tsf_GetFloatArg | ||||
| #define BlThisArg tsf_GetThisArg | ||||
| #define BlStringArg(x) tsf_GetStringArg((char*)x) | ||||
| #define BlReturnBuffer tsf_BlCon__getReturnBuffer | ||||
|  | ||||
| #define BlObject tsf_FindObject | ||||
| #define BlNamespace tsf_LookupNamespace | ||||
|  | ||||
| #define BlGetField tsf_GetDataField | ||||
| #define BlSetField tsf_SetDataField | ||||
|  | ||||
| #define BlGetVar tsf_GetVar | ||||
| #define BlAddVar tsf_AddVar | ||||
|  | ||||
| #define BlAddFunction tsf_AddConsoleFunc | ||||
|  | ||||
| #define __22ND_ARGUMENT(a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11, a12, a13, a14, a15, a16, a17, a18, a19, a20, a21, a22, ...) a22 | ||||
| #define __NUM_LIST(...) __22ND_ARGUMENT(dummy, ##__VA_ARGS__, 20, 19, 18, 17, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0) | ||||
| #define BlCall(...) \ | ||||
| 	tsf_BlCon__executef(__NUM_LIST(__VA_ARGS__), __VA_ARGS__) | ||||
| #define BlCallObj(obj, ...) \ | ||||
| 	tsf_BlCon__executefSimObj((ADDR*)obj, __NUM_LIST(__VA_ARGS__), __VA_ARGS__) | ||||
|  | ||||
| #define BlFuncsInit() if(!tsf_InitInternal()) { return false; } | ||||
| #define BlFuncsDeinit() if(!tsf_DeinitInternal()) { return false; } | ||||
|  | ||||
|  | ||||
| #endif | ||||
| #endif | ||||
							
								
								
									
										207
									
								
								inc/tsfuncs/BlHooks.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										207
									
								
								inc/tsfuncs/BlHooks.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,207 @@ | ||||
|  | ||||
| ////////////////////////////////////////////////// | ||||
| // RedoBlHooks Version 3.0 | ||||
|  | ||||
|  | ||||
| // Includes | ||||
|  | ||||
| #include <Windows.h> | ||||
| #include <Psapi.h> | ||||
| //#include <map> | ||||
|  | ||||
| #include "BlHooks.hpp" | ||||
|  | ||||
|  | ||||
| // Scanned structures | ||||
|  | ||||
| BlFunctionDefIntern(tsh_BlPrintf); | ||||
|  | ||||
|  | ||||
| // Sig Scanning | ||||
|  | ||||
| ADDR ImageBase; | ||||
| ADDR ImageSize; | ||||
|  | ||||
| void tsh_i_InitScanner(){ | ||||
| 	HMODULE module = GetModuleHandle(NULL); | ||||
| 	if(module) { | ||||
| 		MODULEINFO info; | ||||
| 		GetModuleInformation(GetCurrentProcess(), module, &info, sizeof(MODULEINFO)); | ||||
| 		ImageBase = (ADDR)info.lpBaseOfDll; | ||||
| 		ImageSize = info.SizeOfImage; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| bool tsh_i_CompareData(BYTE *data, BYTE *pattern, char* mask){ | ||||
| 	for (; *mask; ++data, ++pattern, ++mask){ | ||||
| 		if (*mask=='x' && *data!=*pattern) | ||||
| 			return false; | ||||
| 	} | ||||
| 	return (*mask)==0; | ||||
| } | ||||
|  | ||||
| ADDR tsh_i_FindPattern(ADDR imageBase, ADDR imageSize, BYTE *pattern, char *mask){ | ||||
| 	for (ADDR i=imageBase; i < imageBase+imageSize; i++){ | ||||
| 		if(tsh_i_CompareData((PBYTE)i, pattern, mask)){ | ||||
| 			return i; | ||||
| 		} | ||||
| 	} | ||||
| 	return 0; | ||||
| } | ||||
|  | ||||
| // Convert a text-style pattern into code-style | ||||
| void tsh_i_PatternTextToCode(char* text, char** opatt, char** omask) { | ||||
| 	unsigned int len = strlen(text); | ||||
| 	char* patt = (char*)malloc(len); | ||||
| 	char* mask = (char*)malloc(len); | ||||
| 	 | ||||
| 	int outidx = 0; | ||||
| 	int val = 0; | ||||
| 	bool uk = false; | ||||
| 	for(unsigned int i=0; i<len; i++){ | ||||
| 		char c = text[i]; | ||||
| 		if(c=='?'){ | ||||
| 			uk = true; | ||||
| 		}else if(c>='0' && c<='9'){ | ||||
| 			val = (val<<4) + (c-'0'); | ||||
| 		}else if(c>='A' && c<='F'){ | ||||
| 			val = (val<<4) + (c-'A'+10); | ||||
| 		}else if(c>='a' && c<='f'){ | ||||
| 			val = (val<<4) + (c-'a'+10); | ||||
| 		}else if(c==' '){ | ||||
| 			patt[outidx] = uk ? 0 : val; | ||||
| 			mask[outidx] = uk ? '?' : 'x'; | ||||
| 			val = 0; | ||||
| 			uk = false; | ||||
| 			outidx++; | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
| 	patt[outidx] = uk ? 0 : val; | ||||
| 	mask[outidx] = uk ? '?' : 'x'; | ||||
| 	outidx++; | ||||
| 	patt[outidx] = 0; | ||||
| 	mask[outidx] = 0; | ||||
| 	 | ||||
| 	*opatt = patt; | ||||
| 	*omask = mask; | ||||
| } | ||||
|  | ||||
|  | ||||
| // Public functions for sig scanning | ||||
|  | ||||
| // Scan using code-style pattern | ||||
| ADDR tsh_ScanCode(char* pattern, char* mask) { | ||||
| 	return tsh_i_FindPattern(ImageBase, ImageSize-strlen(mask), (BYTE*)pattern, mask); | ||||
| } | ||||
|  | ||||
| // Scan using a text-style pattern | ||||
| ADDR tsh_ScanText(char* text) { | ||||
| 	char* patt; | ||||
| 	char* mask; | ||||
| 	tsh_i_PatternTextToCode(text, &patt, &mask); | ||||
| 	 | ||||
| 	ADDR res = tsh_ScanCode(patt, mask); | ||||
| 	 | ||||
| 	free(patt); | ||||
| 	free(mask); | ||||
| 	 | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
|  | ||||
| // Call Patching and Hooking | ||||
|  | ||||
| // Remove protection from address | ||||
| //std::map<ADDR, std::pair<ADDR, ADDR>> tsh_DeprotectedAddresses; | ||||
|  | ||||
| void tsh_DeprotectAddress(ADDR length, ADDR location) { | ||||
| 	DWORD oldProtection; | ||||
| 	VirtualProtect((void*)location, length, PAGE_EXECUTE_READWRITE, &oldProtection); | ||||
| 	//tsh_DeprotectedAddresses[location] = {length, oldProtection}; | ||||
| } | ||||
|  | ||||
| // Patch a string of bytes by deprotecting and then overwriting | ||||
| void tsh_PatchBytes(ADDR length, ADDR location, BYTE* repl) { | ||||
| 	tsh_DeprotectAddress(length, location); | ||||
| 	memcpy((void*)location, (void*)repl, (size_t)length); | ||||
| } | ||||
|  | ||||
| void tsh_PatchByte(ADDR location, BYTE value) { | ||||
| 	tsh_PatchBytes(location, 1, &value); | ||||
| } | ||||
|  | ||||
| void tsh_ReplaceInt(ADDR addr, int rval) { | ||||
| 	tsh_PatchBytes(4, addr, (BYTE*)(&rval)); | ||||
| } | ||||
|  | ||||
| int tsh_i_CallOffset(ADDR instr, ADDR func) { | ||||
| 	return func - (instr+4); | ||||
| } | ||||
|  | ||||
| void tsh_i_ReplaceCall(ADDR instr, ADDR target) { | ||||
| 	tsh_ReplaceInt(instr, tsh_i_CallOffset(instr, target)); | ||||
| } | ||||
|  | ||||
| void tsh_i_PatchCopy(ADDR dest, ADDR src, unsigned int len) { | ||||
| 	for(unsigned int i=0; i<len; i++){ | ||||
| 		tsh_PatchByte(dest+i, *((BYTE*)(src+i))); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| void tsh_HookFunction(ADDR victim, ADDR detour, BYTE* origbytes) { | ||||
| 	memcpy(origbytes, (BYTE*)victim, 6); //save old data | ||||
| 	 | ||||
| 	*(BYTE*)victim = 0xE9; //jmp rel32 | ||||
| 	*(ADDR*)(victim+1) = (detour - (victim+5)); // jump offset | ||||
| 	*(BYTE*)(victim+5) = 0xC3; //retn | ||||
| } | ||||
|  | ||||
| void tsh_UnhookFunction(ADDR victim, BYTE* origbytes){ | ||||
| 	tsh_i_PatchCopy(victim, (ADDR)origbytes, 6); //restore old data | ||||
| } | ||||
|  | ||||
| int tsh_PatchAllMatchesCode(ADDR len, char* patt, char* mask, char* replace, bool debugprint){ | ||||
| 	int numpatched = 0; | ||||
| 	for(ADDR i=ImageBase; i<ImageBase+ImageSize-len; i++){ | ||||
| 		if(tsh_i_CompareData((BYTE*)i, (BYTE*)patt, mask)){ | ||||
| 			if(debugprint) BlPrintf("RedoBlHooks: Patching call at %08x", i); | ||||
| 			 | ||||
| 			numpatched++; | ||||
| 			tsh_DeprotectAddress(i, len); | ||||
| 			for(ADDR c=0; c<len; c++){ | ||||
| 				tsh_PatchByte(i+c, replace[c]); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	return numpatched; | ||||
| } | ||||
|  | ||||
| int tsh_PatchAllMatchesHex(ADDR len, char* text, char* replace, bool debugprint) { | ||||
| 	char* patt; | ||||
| 	char* mask; | ||||
| 	tsh_i_PatternTextToCode(text, &patt, &mask); | ||||
| 	 | ||||
| 	int res = tsh_PatchAllMatchesCode(len, patt, mask, replace, debugprint); | ||||
| 	 | ||||
| 	free(patt); | ||||
| 	free(mask); | ||||
| 	 | ||||
| 	return res; | ||||
| } | ||||
|  | ||||
|  | ||||
| // Initialization | ||||
|  | ||||
| bool tsh_InitInternal(){ | ||||
| 	tsh_i_InitScanner(); | ||||
| 	 | ||||
| 	BlScanFunctionText(tsh_BlPrintf, "8D 44 24 08 33 D2 50 FF 74 24 08 33 C9 E8 ? ? ? ? 83 C4 08 C3"); | ||||
| 	 | ||||
| 	return true; | ||||
| } | ||||
|  | ||||
| bool tsh_DeinitInternal() { | ||||
| 	return true; | ||||
| } | ||||
|  | ||||
							
								
								
									
										122
									
								
								inc/tsfuncs/BlHooks.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								inc/tsfuncs/BlHooks.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,122 @@ | ||||
|  | ||||
| ////////////////////////////////////////////////// | ||||
| // RedoBlHooks Version 3.0 | ||||
|  | ||||
| #ifndef _H_BLHOOKS | ||||
| #define _H_BLHOOKS | ||||
|  | ||||
|  | ||||
| // Typedefs | ||||
|  | ||||
| typedef unsigned char BYTE; | ||||
| typedef unsigned int  ADDR; | ||||
|  | ||||
|  | ||||
| // Prototypes | ||||
|  | ||||
| bool tsh_InitInternal(); | ||||
| bool tsh_DeinitInternal(); | ||||
| ADDR tsh_ScanCode(char*, char*); | ||||
| ADDR tsh_ScanText(char*); | ||||
| void tsh_HookFunction(ADDR, ADDR, BYTE*); | ||||
| void tsh_UnhookFunction(ADDR, BYTE*); | ||||
| int  tsh_PatchAllMatches(unsigned int, char*, char*, char*, bool); | ||||
| void tsh_PatchByte(ADDR, BYTE); | ||||
| void tsh_PatchBytes(unsigned int, ADDR, BYTE*); | ||||
| void tsh_PatchInt(ADDR, int); | ||||
|  | ||||
|  | ||||
| // Debug print settings | ||||
|  | ||||
| #ifndef TSH_NO_DEBUG_PRINT | ||||
| 	#define tsh_DEBUGPRINT false | ||||
| #else | ||||
| 	#define tsh_DEBUGPRINT true | ||||
| #endif | ||||
|  | ||||
|  | ||||
| // Function short names | ||||
|  | ||||
| // Use in code when the def is not shared in a header | ||||
| #define BlFunctionDef(returnType, convention, name, ...) \ | ||||
| 	typedef returnType (convention *tsh_##name##FnT)(__VA_ARGS__); \ | ||||
| 	tsh_##name##FnT name; | ||||
| // Use in header for shared function defs when a BlFunctionDefIntern exists in code | ||||
| #define BlFunctionDefExtern(returnType, convention, name, ...) \ | ||||
| 	typedef returnType (convention *tsh_##name##FnT)(__VA_ARGS__); \ | ||||
| 	extern tsh_##name##FnT name; | ||||
| // Use in code for shared function defs when a BlFunctionDefExtern exists in header | ||||
| #define BlFunctionDefIntern(name) \ | ||||
| 	tsh_##name##FnT name; | ||||
|  | ||||
| // Scan for and assign the pattern to the variable, or err and return if not found | ||||
| #define BlScanFunctionCode(target, patt, mask) \ | ||||
| 	target = (tsh_##target##FnT)tsh_ScanCode((char*)patt, (char*)mask); \ | ||||
| 	if(!target){ \ | ||||
| 		BlPrintf("RedoBlHooks | Cannot find function "#target"!"); \ | ||||
| 		return false; \ | ||||
| 	}else{ \ | ||||
| 		if(tsh_DEBUGPRINT) BlPrintf("RedoBlHooks | Found function "#target" at %08x", (int)target); \ | ||||
| 	} | ||||
| #define BlScanFunctionText(target, text) \ | ||||
| 	target = (tsh_##target##FnT)tsh_ScanText((char*)text); \ | ||||
| 	if(!target){ \ | ||||
| 		BlPrintf("RedoBlHooks | Cannot find function "#target"!"); \ | ||||
| 		return false; \ | ||||
| 	}else{ \ | ||||
| 		if(tsh_DEBUGPRINT) BlPrintf("RedoBlHooks | Found function "#target" at %08x", (int)target); \ | ||||
| 	} | ||||
| #define BlScanCode(target, patt, mask) \ | ||||
| 	target = tsh_ScanCode((char*)patt, (char*)mask); \ | ||||
| 	if(!target){ \ | ||||
| 		BlPrintf("RedoBlHooks | Cannot find pattern "#target"!"); \ | ||||
| 		return false; \ | ||||
| 	}else{ \ | ||||
| 		if(tsh_DEBUGPRINT) BlPrintf("RedoBlHooks | Found "#target" at %08x", (int)target); \ | ||||
| 	} | ||||
| #define BlScanText(target, text) \ | ||||
| 	target = tsh_ScanText((char*)text); \ | ||||
| 	if(!target){ \ | ||||
| 		BlPrintf("RedoBlHooks | Cannot find "#target"!"); \ | ||||
| 		return false; \ | ||||
| 	}else{ \ | ||||
| 		if(tsh_DEBUGPRINT) BlPrintf("RedoBlHooks | Found "#target" at %08x", (int)target); \ | ||||
| 	} | ||||
|  | ||||
| // Use in code to define the data and functions for hooking a function | ||||
| #define BlFunctionHookDef(func) \ | ||||
| 	BYTE tsh_BlFunctionHook##func##Data[6]; \ | ||||
| 	void func##HookOn(){ tsh_HookFunction((ADDR)func, (ADDR)func##Hook, tsh_BlFunctionHook##func##Data); } \ | ||||
| 	void func##HookOff(){ tsh_UnhookFunction((ADDR)func, tsh_BlFunctionHook##func##Data); } | ||||
| // Use in code to initialize the hook once | ||||
| #define BlFunctionHookInit(func) \ | ||||
| 	tsh_DeprotectAddress(6, (ADDR)func); \ | ||||
| 	func##HookOn(); | ||||
|  | ||||
| // Replace all matches of the pattern with the given byte string | ||||
| #define BlPatchAllMatchesCode(len, patt, mask, repl) \ | ||||
| 	tsh_PatchAllMatchesCode((ADDR)len, (char*)patt, (char*)mask, (char*)repl, tsh_DEBUGPRINT); | ||||
| #define BlPatchAllMatchesText(len, text, repl) \ | ||||
| 	tsh_PatchAllMatchesCode((ADDR)len, (char*)text, (char*)repl, tsh_DEBUGPRINT); | ||||
|  | ||||
| // Deprotect and replace one byte | ||||
| #define BlPatchByte(addr, byte) \ | ||||
| 	tsh_PatchByte((ADDR)addr, (BYTE)byte); | ||||
| // Deprotect and replace a byte string | ||||
| #define BlPatchBytes(len, addr, repl) \ | ||||
| 	tsh_PatchBytes((ADDR)len, (ADDR)addr, (BYTE*)repl); | ||||
|  | ||||
| // BlPrintf(char* format, ...) | ||||
| #define BlPrintf(...) if(tsh_BlPrintf) { tsh_BlPrintf(__VA_ARGS__); } | ||||
| // BlHooksInit() -> bool: success | ||||
| #define BlHooksInit() if(!tsh_InitInternal()) { BlPrintf("BlHooksInit failed"); return false; } | ||||
| // BlHooksDeinit() -> bool: success | ||||
| #define BlHooksDeinit() if(!tsh_DeinitInternal()) { BlPrintf("BlHooksDeinit failed"); return false; } | ||||
|  | ||||
|  | ||||
| // Scanned structures | ||||
|  | ||||
| BlFunctionDefExtern(void, , tsh_BlPrintf, const char*, ...); | ||||
|  | ||||
|  | ||||
| #endif | ||||
		Reference in New Issue
	
	Block a user
	 Redo
					Redo