280 lines
9.9 KiB
C++
Raw Normal View History

2024-04-02 15:36:52 +08:00
static std::wstring StringToWideString(const std::string &text, UINT encoding = CP_UTF8)
2024-03-23 18:33:58 +08:00
{
std::vector<wchar_t> 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);
}
2024-04-02 15:36:52 +08:00
std::string WideStringToString(const std::wstring &text, UINT cp = CP_UTF8)
2024-03-23 18:33:58 +08:00
{
2024-04-02 15:36:52 +08:00
std::vector<char> buffer((text.size() + 1) * 4);
2024-03-23 18:33:58 +08:00
WideCharToMultiByte(cp, 0, text.c_str(), -1, buffer.data(), buffer.size(), nullptr, nullptr);
return buffer.data();
}
2024-04-02 15:36:52 +08:00
HANDLE runexe(const std::wstring &exe, const std::optional<std::wstring> &startup_argument)
2024-03-23 18:33:58 +08:00
{
STARTUPINFOW si;
PROCESS_INFORMATION pi;
2024-04-02 15:36:52 +08:00
ZeroMemory(&si, sizeof(si));
2024-03-23 18:33:58 +08:00
si.cb = sizeof(si);
2024-04-02 15:36:52 +08:00
ZeroMemory(&pi, sizeof(pi));
std::vector<wchar_t> argu;
if (startup_argument.has_value())
{
argu.resize(startup_argument.value().size() + 1);
wcscpy(argu.data(), startup_argument.value().c_str());
2024-03-23 18:33:58 +08:00
}
2024-04-02 15:36:52 +08:00
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
2024-03-23 18:33:58 +08:00
return pi.hProcess;
}
2024-04-02 15:36:52 +08:00
std::wstring stolower(const std::wstring &s1)
{
auto s = s1;
std::transform(s.begin(), s.end(), s.begin(), tolower);
return s;
2024-03-23 18:33:58 +08:00
}
2024-04-02 15:36:52 +08:00
std::vector<DWORD> EnumerateProcesses(const std::wstring &exe)
2024-03-23 18:33:58 +08:00
{
2024-04-02 15:36:52 +08:00
2024-03-23 18:33:58 +08:00
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
if (hSnapshot == INVALID_HANDLE_VALUE)
{
return {};
}
2024-04-02 15:36:52 +08:00
2024-03-23 18:33:58 +08:00
PROCESSENTRY32 pe32;
pe32.dwSize = sizeof(PROCESSENTRY32);
if (!Process32First(hSnapshot, &pe32))
{
CloseHandle(hSnapshot);
return {};
}
std::vector<DWORD> pids;
do
{
2024-04-02 15:36:52 +08:00
if (stolower(exe) == stolower(pe32.szExeFile))
2024-03-23 18:33:58 +08:00
pids.push_back(pe32.th32ProcessID);
} while (Process32Next(hSnapshot, &pe32));
2024-04-02 15:36:52 +08:00
2024-03-23 18:33:58 +08:00
CloseHandle(hSnapshot);
return pids;
}
2024-04-02 15:36:52 +08:00
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
};
2024-03-23 18:33:58 +08:00
struct ThreadParam
{
2024-04-02 15:36:52 +08:00
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
2024-03-23 18:33:58 +08:00
};
typedef void (*ProcessEvent)(DWORD);
typedef void (*ThreadEvent)(wchar_t *, char *, ThreadParam);
typedef void (*OutputCallback)(wchar_t *, char *, ThreadParam, const wchar_t *);
typedef void (*ConsoleHandler)(const wchar_t *);
typedef void (*HookInsertHandler)(uint64_t, const wchar_t *);
typedef void (*EmbedCallback)(const wchar_t *, ThreadParam);
nlohmann::json config;
std::map<std::string, std::string> translation;
std::unordered_set<DWORD> connectedpids;
void (*Luna_Start)(ProcessEvent Connect, ProcessEvent Disconnect, ThreadEvent Create, ThreadEvent Destroy, OutputCallback Output, ConsoleHandler console, HookInsertHandler hookinsert, EmbedCallback embed);
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::set<std::string> notranslation;
HANDLE hsema;
2024-04-02 15:36:52 +08:00
class lunapatch
{
2024-03-23 18:33:58 +08:00
public:
HANDLE hMessage;
HANDLE hwait;
lunapatch(std::wstring dll, nlohmann::json &&_translation, nlohmann::json &&_config)
2024-04-02 15:36:52 +08:00
{
translation = _translation;
config = _config;
2024-04-02 15:36:52 +08:00
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(
[](DWORD pid)
{
auto font = StringToWideString(config["embedsettings"]["font"]);
auto insertspace_policy = config["embedsettings"]["insertspace_policy"];
auto keeprawtext = config["embedsettings"]["keeprawtext"];
Luna_EmbedSettings(pid, 1000, 2, false, font.data(), insertspace_policy, keeprawtext, false);
connectedpids.insert(pid);
},
[](DWORD pid)
{
connectedpids.erase(pid);
ReleaseSemaphore(hsema, 1, NULL);
},
[](auto, auto, auto) {},
[](auto, auto, auto) {},
[](auto, auto, auto, auto) {},
[](auto) {},
[](uint64_t addr, const wchar_t *output)
{
std::wstring newhookcode = output;
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, addr, _ctx1, _ctx2, true);
}
}
}
},
[](const wchar_t *output, ThreadParam tp)
{
std::wstring text = output;
for (auto pid : connectedpids)
{
if ((Luna_checkisusingembed(pid, tp.addr, tp.ctx, tp.ctx2)))
{
auto trans = findtranslation(text);
Luna_embedcallback(pid, output, trans.c_str());
}
}
});
2024-03-23 18:33:58 +08:00
}
2024-04-02 15:36:52 +08:00
void run()
{
auto target_exe = StringToWideString(config["target_exe"]);
auto _startup_argument = config["startup_argument"];
2024-03-23 18:33:58 +08:00
std::optional<std::wstring> startup_argument;
2024-04-02 15:36:52 +08:00
if (_startup_argument.is_null())
startup_argument = {};
2024-03-23 18:33:58 +08:00
else
2024-04-02 15:36:52 +08:00
startup_argument = StringToWideString(config["startup_argument"]);
hwait = runexe(target_exe, startup_argument);
2024-03-23 18:33:58 +08:00
}
2024-04-02 15:36:52 +08:00
~lunapatch()
{
if (notranslation.size())
{
for (auto &text : notranslation)
{
translation[text] = "";
2024-03-23 18:33:58 +08:00
}
2024-04-02 15:36:52 +08:00
auto notrs = nlohmann::json(translation).dump(4);
2024-03-23 18:33:58 +08:00
std::ofstream of;
2024-03-23 20:24:03 +08:00
of.open(std::string(config["translation_file"]));
2024-04-02 15:36:52 +08:00
of << notrs;
2024-03-23 18:33:58 +08:00
of.close();
}
}
2024-04-02 15:36:52 +08:00
void wait()
{
WaitForSingleObject(hwait, INFINITE);
while (connectedpids.size())
WaitForSingleObject(hsema, INFINITE);
2024-03-23 18:33:58 +08:00
}
2024-04-02 15:36:52 +08:00
void inject()
{
// chrome has multi process
2024-03-23 18:33:58 +08:00
Sleep(config["inject_timeout"]);
2024-04-02 15:36:52 +08:00
for (auto exe : std::set<std::string>{config["target_exe"], config["target_exe2"]})
2024-03-23 20:24:03 +08:00
{
2024-04-02 15:36:52 +08:00
auto pids = EnumerateProcesses(StringToWideString(exe));
for (auto pid : pids)
{
wprintf(L"%d\n", pid);
Luna_Inject(pid, L"");
2024-03-23 20:24:03 +08:00
}
2024-03-23 18:33:58 +08:00
}
}
static std::wstring findtranslation(const std::wstring &text)
2024-04-02 15:36:52 +08:00
{
auto utf8text = WideStringToString(text);
if (translation.find(utf8text) == translation.end())
{
// wprintf(L"%s\n",text.c_str());
2024-03-23 18:33:58 +08:00
notranslation.insert(utf8text);
return {};
}
return StringToWideString(translation.at(utf8text));
}
};
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"";
}
2024-04-02 15:36:52 +08:00
bool checkisapatch()
{
auto curr = std::filesystem::path(GetExecutablePath());
auto config = curr / "LunaPatch.json";
if (std::filesystem::exists(config) == false)
2024-03-23 18:33:58 +08:00
{
return false;
}
std::ifstream jsonfile;
jsonfile.open(config);
2024-04-02 15:36:52 +08:00
auto configjson = nlohmann::json::parse(jsonfile);
2024-03-23 18:33:58 +08:00
jsonfile.close();
2024-04-02 15:36:52 +08:00
std::string translation_file = configjson["translation_file"];
2024-03-23 18:33:58 +08:00
jsonfile.open(translation_file);
2024-04-02 15:36:52 +08:00
std::map<std::string, std::string> translation = nlohmann::json::parse(jsonfile);
2024-03-23 18:33:58 +08:00
jsonfile.close();
2024-04-02 15:36:52 +08:00
auto LunaHost = (curr / (std::wstring(L"LunaHost") + std::to_wstring(8 * sizeof(void *)))).wstring();
2024-03-23 18:33:58 +08:00
2024-04-02 15:36:52 +08:00
lunapatch _lunapatch(LunaHost, std::move(translation), std::move(configjson));
2024-03-23 18:33:58 +08:00
_lunapatch.run();
_lunapatch.inject();
_lunapatch.wait();
return true;
}