mirror of
https://github.com/HIllya51/LunaHook.git
synced 2025-01-01 07:54:11 +08:00
82 lines
3.0 KiB
C++
82 lines
3.0 KiB
C++
|
#include"AB2Try.h"
|
|||
|
|
|||
|
/********************************************************************************************
|
|||
|
AkabeiSoft2Try hook:
|
|||
|
Game folder contains YaneSDK.dll. Maybe we should call the engine Yane(屋<EFBFBD> = roof)?
|
|||
|
This engine is based on .NET framework. This really makes it troublesome to locate a
|
|||
|
valid hook address. The problem is that the engine file merely contains bytecode for
|
|||
|
the CLR. Real meaningful object code is generated dynamically and the address is randomized.
|
|||
|
Therefore the easiest method is to brute force search whole address space. While it's not necessary
|
|||
|
to completely search the whole address space, since non-executable pages can be excluded first.
|
|||
|
The generated code sections do not belong to any module(exe/dll), hence they do not have
|
|||
|
a section name. So we can also exclude executable pages from all modules. At last, the code
|
|||
|
section should be long(>0x2000). The remain address space should be several MBs in size and
|
|||
|
can be examined in reasonable time(less than 0.1s for P8400 Win7x64).
|
|||
|
Characteristic sequence is 0F B7 44 50 0C, stands for movzx eax, word ptr [edx*2 + eax + C].
|
|||
|
Obviously this instruction extracts one unicode character from a string.
|
|||
|
A main shortcoming is that the code is not generated if it hasn't been used yet.
|
|||
|
So if you are in title screen this approach will fail.
|
|||
|
|
|||
|
********************************************************************************************/
|
|||
|
namespace { // unnamed
|
|||
|
|
|||
|
typedef struct _NSTRING
|
|||
|
{
|
|||
|
PVOID vfTable;
|
|||
|
DWORD lenWithNull;
|
|||
|
DWORD lenWithoutNull;
|
|||
|
WCHAR str[1];
|
|||
|
} NSTRING;
|
|||
|
|
|||
|
// qsort correctly identifies overflow.
|
|||
|
int cmp(const void * a, const void * b)
|
|||
|
{ return *(int*)a - *(int*)b; }
|
|||
|
|
|||
|
void SpecialHookAB2Try(hook_stack* stack, HookParam *, uintptr_t *data, uintptr_t *split, size_t *len)
|
|||
|
{
|
|||
|
//DWORD test = *(DWORD*)(esp_base - 0x10);
|
|||
|
DWORD edx = stack->edx;
|
|||
|
if (edx != 0)
|
|||
|
return;
|
|||
|
|
|||
|
//NSTRING *s = *(NSTRING **)(esp_base - 8);
|
|||
|
if (const NSTRING *s = (NSTRING *)stack->eax) {
|
|||
|
*len = s->lenWithoutNull << 1;
|
|||
|
*data = (DWORD)s->str;
|
|||
|
//*split = 0;
|
|||
|
*split = FIXED_SPLIT_VALUE; // 8/3/2014 jichi: change to single threads
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
bool FindCharacteristInstruction()
|
|||
|
{
|
|||
|
const BYTE bytes[] = { 0x0F, 0xB7, 0x44, 0x50, 0x0C, 0x89 };
|
|||
|
for (auto addr : Util::SearchMemory(bytes, sizeof(bytes), PAGE_EXECUTE_READWRITE))
|
|||
|
{
|
|||
|
//GROWL_DWORD(addr);
|
|||
|
HookParam hp;
|
|||
|
hp.address = addr;
|
|||
|
hp.text_fun = SpecialHookAB2Try;
|
|||
|
hp.type = USING_STRING | NO_CONTEXT | CODEC_UTF16;
|
|||
|
ConsoleOutput("INSERT AB2Try");
|
|||
|
//ConsoleOutput("Please adjust text speed to fastest/immediate.");
|
|||
|
//RegisterEngineType(ENGINE_AB2T);
|
|||
|
return NewHook(hp, "AB2Try");
|
|||
|
}
|
|||
|
return false;
|
|||
|
}
|
|||
|
} // unnamed namespace
|
|||
|
bool InsertAB2TryHook()
|
|||
|
{
|
|||
|
bool ret = FindCharacteristInstruction();
|
|||
|
if (ret)
|
|||
|
ConsoleOutput("AB2Try: found characteristic sequence");
|
|||
|
else
|
|||
|
ConsoleOutput("AB2Try: cannot find characteristic sequence. Make sure you have start the game and have seen some text on the screen.");
|
|||
|
return ret;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
bool AB2Try::attach_function() {
|
|||
|
return InsertAB2TryHook();
|
|||
|
}
|