#include "host.h" #define C_LUNA_API extern "C" __declspec(dllexport) BOOL APIENTRY DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved) { switch (ul_reason_for_call) { case DLL_PROCESS_ATTACH: case DLL_THREAD_ATTACH: case DLL_THREAD_DETACH: case DLL_PROCESS_DETACH: break; } return TRUE; } typedef void (*ProcessEvent)(DWORD pid); typedef void (*ThreadEvent_maybe_embed)(const wchar_t *hookcode, const char *hookname, ThreadParam, bool isembedable); typedef void (*ThreadEvent)(const wchar_t *hookcode, const char *hookname, ThreadParam); typedef bool (*OutputCallback)(const wchar_t *hookcode, const char *hookname, ThreadParam, const wchar_t *text); typedef void (*HostInfoHandler)(HOSTINFO type, const wchar_t *log); typedef void (*HookInsertHandler)(DWORD pid, uint64_t address, const wchar_t *hookcode); typedef void (*EmbedCallback)(const wchar_t *text, ThreadParam); typedef void (*findhookcallback_t)(wchar_t *hookcode, const wchar_t *text); template std::optional checkoption(bool check, T &&t) { if (check) return std::move(t); return {}; } C_LUNA_API void Luna_Start(ProcessEvent Connect, ProcessEvent Disconnect, ThreadEvent_maybe_embed Create, ThreadEvent Destroy, OutputCallback Output, HostInfoHandler hostinfo, HookInsertHandler hookinsert, EmbedCallback embed) { Host::StartEx( checkoption(Connect, std::function(Connect)), checkoption(Disconnect, std::function(Disconnect)), checkoption(Create, [=](const TextThread &thread) { Create(thread.hp.hookcode, thread.hp.name, thread.tp, thread.hp.type & EMBED_ABLE); }), checkoption(Destroy, [=](const TextThread &thread) { Destroy(thread.hp.hookcode, thread.hp.name, thread.tp); }), checkoption(Output, [=](const TextThread &thread, std::wstring &output) { return Output(thread.hp.hookcode, thread.hp.name, thread.tp, output.c_str()); }), false, checkoption(hostinfo, [=](HOSTINFO type, const std::wstring &output) { hostinfo(type, output.c_str()); }), checkoption(hookinsert, [=](DWORD pid, uint64_t addr, const std::wstring &hookcode) { hookinsert(pid, addr, hookcode.c_str()); }), checkoption(embed, [=](const std::wstring &output, const ThreadParam &tp) { embed(output.c_str(), tp); })); } #if 0 C_LUNA_API void Luna_Inject(DWORD pid, LPCWSTR basepath) { Host::InjectProcess(pid, basepath); } #endif C_LUNA_API bool Luna_CreatePipeAndCheck(DWORD pid) { return Host::CreatePipeAndCheck(pid); } C_LUNA_API void Luna_Detach(DWORD pid) { Host::DetachProcess(pid); } C_LUNA_API void Luna_Settings(int flushDelay, bool filterRepetition, int defaultCodepage, int maxBufferSize, int maxHistorySize) { TextThread::flushDelay = flushDelay; TextThread::filterRepetition = filterRepetition; Host::defaultCodepage = defaultCodepage; TextThread::maxBufferSize = maxBufferSize; TextThread::maxHistorySize = maxHistorySize; } C_LUNA_API void Luna_SetLanguage(const char* lang) { Host::SetLanguage(lang); } C_LUNA_API void Luna_InsertPCHooks(DWORD pid, int which) { Host::InsertPCHooks(pid, which); } C_LUNA_API bool Luna_InsertHookCode(DWORD pid, LPCWSTR hookcode) { auto hp = HookCode::Parse(hookcode); if (hp) Host::InsertHook(pid, hp.value()); return hp.has_value(); } C_LUNA_API void Luna_QueryThreadHistory(ThreadParam tp, void (*callback)(const wchar_t *)) { auto s = Host::GetThread(tp).storage.Acquire(); callback(s->c_str()); } C_LUNA_API void Luna_RemoveHook(DWORD pid, uint64_t addr) { Host::RemoveHook(pid, addr); } C_LUNA_API void Luna_FindHooks(DWORD pid, SearchParam sp, findhookcallback_t findhookcallback) { Host::FindHooks(pid, sp, [=](HookParam hp, std::wstring text) { wchar_t hookcode[HOOKCODE_LEN]; wcscpy_s(hookcode,HOOKCODE_LEN, hp.hookcode); findhookcallback(hookcode,text.c_str()); }); } C_LUNA_API void Luna_EmbedSettings(DWORD pid, UINT32 waittime, UINT8 fontCharSet, bool fontCharSetEnabled, wchar_t *fontFamily, Displaymode displaymode, bool fastskipignore) { auto sm = Host::GetCommonSharedMem(pid); if (!sm) return; sm->waittime = waittime; sm->fontCharSet = fontCharSet; sm->fontCharSetEnabled = fontCharSetEnabled; wcscpy_s(sm->fontFamily, 100, fontFamily); sm->displaymode = displaymode; sm->fastskipignore = fastskipignore; } C_LUNA_API bool Luna_checkisusingembed(ThreadParam tp) { return Host::CheckIsUsingEmbed(tp); } C_LUNA_API void Luna_useembed(ThreadParam tp, bool use) { auto sm = Host::GetCommonSharedMem(tp.processId); if (!sm) return; sm->codepage = Host::defaultCodepage; for (int i = 0; i < ARRAYSIZE(sm->embedtps); i++) { if (sm->embedtps[i].use && (sm->embedtps[i].tp == tp)) if (!use) ZeroMemory(sm->embedtps + i, sizeof(sm->embedtps[i])); } if (use) for (int i = 0; i < ARRAYSIZE(sm->embedtps); i++) { if (!sm->embedtps[i].use) { sm->embedtps[i].use = true; sm->embedtps[i].tp = tp; break; } } } C_LUNA_API void Luna_embedcallback(ThreadParam tp, LPCWSTR text, LPCWSTR trans) { auto sm = Host::GetCommonSharedMem(tp.processId); if (!sm) return; wcsncpy(sm->text, trans, ARRAYSIZE(sm->text)); char eventname[1000]; sprintf(eventname, LUNA_EMBED_notify_event, tp.processId, simplehash::djb2_n2((const unsigned char *)(text), wcslen(text) * 2)); win_event event1(eventname); event1.signal(true); } C_LUNA_API void Luna_SyncThread(ThreadParam tp, bool sync) { // 必须放到线程里去异步做,不然GetThread死锁 std::thread([=]() { try { auto &&t=Host::GetThread(tp); if (sync) TextThread::syncThreads->insert(&t); else TextThread::syncThreads->erase(&t); } catch (...) { } }) .detach(); }