#include #include #include #include #include #include #include #include #include static std::wstring StringToWideString(const std::string& text, UINT encoding=CP_UTF8) { std::vector buffer(text.size() + 1); int length = MultiByteToWideChar(encoding, 0, text.c_str(), text.size() + 1, buffer.data(), buffer.size()); return std::wstring(buffer.data(), length - 1); } std::string WideStringToString(const std::wstring& text,UINT cp=CP_UTF8) { std::vector buffer((text.size() + 1) * 4); WideCharToMultiByte(cp, 0, text.c_str(), -1, buffer.data(), buffer.size(), nullptr, nullptr); return buffer.data(); } HANDLE runexe(const std::wstring &exe,const std::optional &startup_argument) { STARTUPINFOW si; PROCESS_INFORMATION pi; ZeroMemory( &si, sizeof(si) ); si.cb = sizeof(si); ZeroMemory( &pi, sizeof(pi) ); std::vectorargu; if(startup_argument.has_value()){ argu.resize(startup_argument.value().size()+1); wcscpy(argu.data(),startup_argument.value().c_str()); } CreateProcessW( exe.c_str(), // No module name (use command line) startup_argument.has_value()?argu.data(): NULL, NULL, // Process handle not inheritable NULL, // Thread handle not inheritable FALSE, // Set handle inheritance to FALSE 0, // No creation flags NULL, // Use parent's environment block NULL, // Use parent's starting directory &si, // Pointer to STARTUPINFO structure &pi ); // Pointer to PROCESS_INFORMATION structure return pi.hProcess; } std::wstring stolower(const std::wstring& s1){ auto s=s1; std::transform(s.begin(), s.end(), s.begin(), tolower); return s; } std::vector EnumerateProcesses(const std::wstring& exe) { HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0); if (hSnapshot == INVALID_HANDLE_VALUE) { return {}; } PROCESSENTRY32 pe32; pe32.dwSize = sizeof(PROCESSENTRY32); if (!Process32First(hSnapshot, &pe32)) { CloseHandle(hSnapshot); return {}; } std::vector pids; do { if(stolower(exe)==stolower(pe32.szExeFile)) pids.push_back(pe32.th32ProcessID); } while (Process32Next(hSnapshot, &pe32)); CloseHandle(hSnapshot); return pids; } enum { STRING = 12, MESSAGE_SIZE = 500, PIPE_BUFFER_SIZE = 50000, SHIFT_JIS = 932, MAX_MODULE_SIZE = 120, PATTERN_SIZE = 30, HOOK_NAME_SIZE = 60, FIXED_SPLIT_VALUE = 0x10001 , HOOKCODE_LEN=500}; struct ThreadParam { bool operator==(ThreadParam other) const { return processId == other.processId && addr == other.addr && ctx == other.ctx && ctx2 == other.ctx2; } DWORD processId; uint64_t addr; uint64_t ctx; // The context of the hook: by default the first value on stack, usually the return address uint64_t ctx2; // The subcontext of the hook: 0 by default, generated in a method specific to the hook }; struct messagelist{ bool read; int type; DWORD pid; char name[HOOK_NAME_SIZE]; wchar_t hookcode[HOOKCODE_LEN]; ThreadParam tp; wchar_t* stringptr; uint64_t addr; }; class lunapatch{ public: HANDLE hMessage; HANDLE hwait; nlohmann::json config; std::maptranslation; std::unordered_setconnectedpids; void (*Luna_Start)( HANDLE* hRead ); void (*Luna_Inject)(DWORD pid,LPCWSTR basepath); void (*Luna_EmbedSettings)(DWORD pid,UINT32 waittime,UINT8 fontCharSet,bool fontCharSetEnabled,wchar_t *fontFamily,UINT32 spaceadjustpolicy,UINT32 keeprawtext,bool fastskipignore); void (*Luna_useembed)(DWORD pid,uint64_t address,uint64_t ctx1,uint64_t ctx2,bool use); bool (*Luna_checkisusingembed)(DWORD pid,uint64_t address,uint64_t ctx1,uint64_t ctx2); void (*Luna_embedcallback)(DWORD pid,LPCWSTR text,LPCWSTR trans); std::setnotranslation; HANDLE hsema; lunapatch(std::wstring dll,nlohmann::json&&_translation,nlohmann::json&&_config):translation(_translation),config(_config){ auto LunaHost=LoadLibraryW(dll.c_str()); Luna_Start=(decltype(Luna_Start))GetProcAddress(LunaHost,"Luna_Start"); Luna_EmbedSettings=(decltype(Luna_EmbedSettings))GetProcAddress(LunaHost,"Luna_EmbedSettings"); Luna_Inject=(decltype(Luna_Inject))GetProcAddress(LunaHost,"Luna_Inject"); Luna_useembed=(decltype(Luna_useembed))GetProcAddress(LunaHost,"Luna_useembed"); Luna_checkisusingembed=(decltype(Luna_checkisusingembed))GetProcAddress(LunaHost,"Luna_checkisusingembed"); Luna_embedcallback=(decltype(Luna_embedcallback))GetProcAddress(LunaHost,"Luna_embedcallback"); hsema=CreateSemaphore(NULL,0,100,NULL); Luna_Start(&hMessage); std::thread([&](){Parsehostmessage();}).detach(); } void run(){ auto target_exe=StringToWideString(config["target_exe"]); auto _startup_argument=config["startup_argument"]; std::optional startup_argument; if(_startup_argument.is_null()) startup_argument={}; else startup_argument=StringToWideString(config["startup_argument"]); hwait=runexe(target_exe,startup_argument); } ~lunapatch(){ if(notranslation.size()){ for(auto &text:notranslation){ translation[text]=""; } auto notrs=nlohmann::json(translation).dump(4); std::ofstream of; of.open(std::string(config["translation_file"])); of<{config["target_exe"],config["target_exe2"]}) { auto pids=EnumerateProcesses(StringToWideString(exe)); for(auto pid:pids){ wprintf(L"%d\n",pid); Luna_Inject(pid,L""); } } } std::wstring findtranslation(const std::wstring& text){ auto utf8text=WideStringToString(text); if(translation.find(utf8text)==translation.end()){ //wprintf(L"%s\n",text.c_str()); notranslation.insert(utf8text); return {}; } return StringToWideString(translation.at(utf8text)); } void Parsehostmessage(){ while (true) { messagelist message; DWORD _; ReadFile(hMessage,&message,sizeof(message),&_,NULL); switch (message.type) { case 0: { auto font =StringToWideString(config["embedsettings"]["font"]); auto insertspace_policy=config["embedsettings"]["insertspace_policy"]; auto keeprawtext=config["embedsettings"]["keeprawtext"]; Luna_EmbedSettings(message.pid,1000,2,false,font.data(),insertspace_policy,keeprawtext,false); connectedpids.insert(message.pid); } break; case 1: { connectedpids.erase(message.pid); ReleaseSemaphore(hsema,1,NULL); } break; case 7: { std::wstring text=message.stringptr; auto tp=message.tp; for(auto pid:connectedpids) { if((Luna_checkisusingembed(pid,tp.addr,tp.ctx,tp.ctx2))) { auto trans=findtranslation(text); Luna_embedcallback(pid,text.c_str(),trans.c_str()); } } } break; case 6: { std::wstring newhookcode=message.stringptr; for(auto hook:config["embedhook"]){ auto hookcode= StringToWideString(hook[0]); uint64_t _addr=hook[1]; uint64_t _ctx1=hook[2]; uint64_t _ctx2=hook[3]; if(hookcode==newhookcode){ for(auto pid:connectedpids){ Luna_useembed(pid,message.addr,_ctx1,_ctx2,true); } } } } break; default: break; } if(message.stringptr) free(message.stringptr); } } }; std::wstring GetExecutablePath() { WCHAR buffer[MAX_PATH]; GetModuleFileNameW(NULL, buffer, MAX_PATH); std::wstring fullPath(buffer); size_t pos = fullPath.find_last_of(L"\\/"); if (pos != std::wstring::npos) { return fullPath.substr(0, pos); } return L""; } bool checkisapatch(){ auto curr=std::filesystem::path(GetExecutablePath()); auto config=curr/"LunaPatch.json"; if(std::filesystem::exists(config)==false) { return false; } std::ifstream jsonfile; jsonfile.open(config); auto configjson=nlohmann::json::parse(jsonfile); jsonfile.close(); std::string translation_file=configjson["translation_file"]; jsonfile.open(translation_file); std::map translation=nlohmann::json::parse(jsonfile); jsonfile.close(); auto LunaHost=(curr/(std::wstring(L"LunaHost")+std::to_wstring(8*sizeof(void*)))).wstring(); lunapatch _lunapatch(LunaHost,std::move(translation),std::move(configjson)); _lunapatch.run(); _lunapatch.inject(); _lunapatch.wait(); return true; }