diff options
Diffstat (limited to 'aoe2hd/src')
-rwxr-xr-x | aoe2hd/src/CodeGenerator.cpp | 100 | ||||
-rwxr-xr-x | aoe2hd/src/CodeInjector.cpp | 165 | ||||
-rwxr-xr-x | aoe2hd/src/CodePatcher.cpp | 111 | ||||
-rwxr-xr-x | aoe2hd/src/ModuleMemory.cpp | 118 | ||||
-rwxr-xr-x | aoe2hd/src/main.cpp | 184 | ||||
-rwxr-xr-x | aoe2hd/src/native.c | 173 | ||||
-rwxr-xr-x | aoe2hd/src/utils.cpp | 19 |
7 files changed, 870 insertions, 0 deletions
diff --git a/aoe2hd/src/CodeGenerator.cpp b/aoe2hd/src/CodeGenerator.cpp new file mode 100755 index 0000000..74149dd --- /dev/null +++ b/aoe2hd/src/CodeGenerator.cpp @@ -0,0 +1,100 @@ +#include "CodeGenerator.h" + +#include <assert.h> +#include <sstream> +#include <iomanip> + +#include "native.h" +#include "utils.h" + + +std::vector<unsigned char> x86_relJump(unsigned long dst, + unsigned long src) +{ + std::vector<unsigned char> code(5); + code[0] = 0xE9; + unsigned long addr = dst - src; + code[1] = (*((unsigned char *)(&addr)+0)); + code[2] = (*((unsigned char *)(&addr)+1)); + code[3] = (*((unsigned char *)(&addr)+2)); + code[4] = (*((unsigned char *)(&addr)+3)); + return code; +} + +CodeGenerator::CodeGenerator(const native_data& nd) + : nd(nd), codes() +{ +} + +CodeGenerator::~CodeGenerator() +{ +} + +CodeGenerator& CodeGenerator::addCode(const std::vector<unsigned char>& code) +{ + codes.push_back(code); + return *this; +} + +CodeGenerator& CodeGenerator::setCode(int index, const std::vector<unsigned char>& code) +{ + codes.at(index) = code; + return *this; +} + +CodeGenerator& CodeGenerator::setCodeSized(int index, const std::vector<unsigned char>& code) +{ + assert(codes.at(index).size() == code.size()); + return setCode(index, code); +} + +CodeGenerator& CodeGenerator::setRel32JMP(int index, unsigned long dst, unsigned long src, bool reversed) +{ + if (!reversed) + { + dst += nd.proc.modbase - diffRel32JMP(reversed, index); + } + else + { + src += nd.proc.modbase + diffRel32JMP(reversed, index); + } + auto jmp = x86_relJump(dst, src); + setCodeSized(index, jmp); + return *this; +} + +std::vector<unsigned char>::size_type CodeGenerator::buildSize(int maxCodes) +{ + std::vector<unsigned char>::size_type total = 0; + for (auto& code : codes) + { + total += code.size(); + if (maxCodes-- == 0) + break; + } + return total; +} + +std::vector<unsigned char> CodeGenerator::build() +{ + std::vector<unsigned char> result; + for (auto& code : codes) + { + result.insert(result.end(), code.begin(), code.end()); + } + return result; +} + +std::vector<unsigned char> CodeGenerator::buildAndClear() +{ + auto result = build(); + clear(); + return result; +} + +std::string CodeGenerator::toString() +{ + std::stringstream out; + out << "CodeBin: " << utils::convertBinToHexstr(build()) << std::endl; + return out.str(); +} diff --git a/aoe2hd/src/CodeInjector.cpp b/aoe2hd/src/CodeInjector.cpp new file mode 100755 index 0000000..b62f969 --- /dev/null +++ b/aoe2hd/src/CodeInjector.cpp @@ -0,0 +1,165 @@ +#include <assert.h> + +#include <sstream> +#include <iomanip> +#include <algorithm> + +#include "CodeInjector.h" + +CodeInjector::CodeInjector(const native_data& nd) + : nd(nd) +{ + assert(nd.alloc_fn && nd.write_fn); +} + +CodeInjector::~CodeInjector() +{ +} + +bool CodeInjector::allocCodeSegment(const std::string& name, unsigned long siz) +{ + if (codeSegExists(name)) + return false; + code_seg seg = {0}; + seg.siz = siz; + seg.addr = nd.alloc_fn(&nd, siz); + if (!seg.addr) + return false; + code_map[name] = seg; + return true; +} + +bool CodeInjector::addCode(const std::string& name, const std::string& code_name, + const std::vector<unsigned char>& code) +{ + assert(code.size()); + if (!codeSegExists(name) + || codeBinExists(name, code_name)) + return false; + auto cave = findCodeCave(name, code.size()); + if (!cave) + return false; + code_bin bin = {0}; + bin.addr = cave; + bin.siz = code.size(); + if (!nd.write_fn(&nd, bin.addr, &code[0], bin.siz)) + return false; + code_map[name].children[code_name] = bin; + return true; +} + +bool CodeInjector::addCode(const std::string& name, const std::string& code_name, + unsigned long siz) +{ + assert(siz); + std::vector<unsigned char> code(siz, 0x90); + return addCode(name, code_name, code); +} + +bool CodeInjector::setCode(const std::string& name, const std::string& code_name, + const std::vector<unsigned char>& code, + unsigned long offset) +{ + assert(code.size()); + if (!codeSegExists(name) + || !codeBinExists(name, code_name)) + return false; + code_bin bin = {0}; + if (!getCodeBin(name, code_name, &bin)) + return false; + assert(bin.addr && bin.siz); + if (bin.addr + offset + code.size() > bin.addr + bin.siz) + return false; + return nd.write_fn(&nd, bin.addr + offset, &code[0], code.size()); +} + +bool CodeInjector::delCode(const std::string& name, const std::string& code_name) +{ + if (!codeBinExists(name, code_name)) + return false; + code_map[name].children[code_name]; + return code_map[name].children.erase(code_name) > 0; +} +unsigned long CodeInjector::getCodeAddr(const std::string& name, const std::string& code_name) +{ + assert(codeBinExists(name, code_name)); + assert(code_map[name].children[code_name].addr); + return code_map[name].children[code_name].addr; +} + +bool CodeInjector::getCodeSeg(const std::string& name, code_seg *seg) +{ + assert(seg); + if (!codeSegExists(name)) + return false; + *seg = code_map[name]; + return true; +} + +bool CodeInjector::getCodeBin(const std::string& name, const std::string& code_name, code_bin *bin) +{ + assert(bin); + if (!codeBinExists(name, code_name)) + return false; + *bin = code_map[name].children[code_name]; + return true; +} + +std::string CodeInjector::toString() +{ + std::stringstream out; + for (auto& code : code_map) + { + out << std::setw(16) << code.first << "[ " + << std::setw(8) << std::hex << code.second.addr << " , " + << std::setw(8) << std::hex << code.second.siz + << " ]" << std::endl; + for (auto& child : code.second.children) + { + out << std::setw(18) << child.first << "[ " + << std::setw(8) << std::hex << child.second.addr << " , " + << std::setw(6) << std::hex << child.second.siz + << " ]" << std::endl; + } + } + return out.str(); +} + +bool CodeInjector::codeBinExists(const std::string& name, const std::string& code_name) +{ + if (!codeSegExists(name)) + return false; + return code_map[name].children.find(code_name) + != code_map[name].children.end(); +} + +std::vector<code_bin> CodeInjector::convertCodeSegChildren(const std::string& name) +{ + auto cs = code_map[name].children; + std::vector<code_bin> ret; + ret.reserve(cs.size()); + std::for_each(cs.begin(), cs.end(), [&ret](std::pair<const std::string, code_bin> element) + { + ret.push_back(element.second); + }); + return ret; +} + +unsigned long CodeInjector::findCodeCave(const std::string& name, unsigned long siz) +{ + auto& cs = code_map[name]; + auto cl = convertCodeSegChildren(name); + std::sort(cl.begin(), cl.end()); + unsigned long end_addr = cs.addr; + for (auto& el : cl) + { + if (el.addr >= end_addr + siz) + { + return end_addr; + } + end_addr = el.addr + el.siz; + } + if (end_addr <= cs.addr + cs.siz) + return end_addr; + return 0; +} diff --git a/aoe2hd/src/CodePatcher.cpp b/aoe2hd/src/CodePatcher.cpp new file mode 100755 index 0000000..b96a2ab --- /dev/null +++ b/aoe2hd/src/CodePatcher.cpp @@ -0,0 +1,111 @@ +#include <assert.h> + +#include <sstream> +#include <iomanip> + +#include "CodePatcher.h" +#include "utils.h" +#include "native.h" + + +CodePatcher::CodePatcher(const native_data& nd) + : nd(nd) +{ + assert(nd.read_fn && nd.write_fn && nd.suspend_fn); +} + +CodePatcher::~CodePatcher() +{ +} + +bool CodePatcher::addPatch(const std::string& name, + unsigned long addr, + const std::vector<unsigned char>& old_code, + const std::vector<unsigned char>& new_code, + long new_offset) +{ + assert(addr); + assert(old_code.size() == new_code.size()); + if (codePatchExists(name)) + return false; + code_patch patch = {0}; + patch.addr = nd.proc.modbase + addr; + patch.old_code = old_code; + patch.new_code = new_code; + patch.new_offset = new_offset; + patch_map[name] = patch; + return true; +} + +void CodePatcher::setPatchSuspend(const std::string& name, long doSuspend) +{ + if (codePatchExists(name)) + patch_map[name].suspend = doSuspend; +} + +bool CodePatcher::doPatch(const std::string& name, int doUnPatch) +{ + if (!codePatchExists(name)) + return false; + auto& patch = patch_map[name]; + if (codeCmp(patch.addr + patch.new_offset, patch.new_code)) + return false; + if (patch.suspend) + nd.suspend_fn(&nd, 0); + bool ret = false; + if (doUnPatch) + { + ret = nd.write_fn(&nd, patch.addr, &patch.old_code[0], patch.old_code.size()); + } + else + { + ret = nd.write_fn(&nd, patch.addr, &patch.new_code[0], patch.new_code.size()); + } + if (patch.suspend) + nd.suspend_fn(&nd, 1); + return ret; +} + +bool CodePatcher::autoPatch(const std::string& name) +{ + if (!doPatch(name, 0)) + { + if (!doPatch(name, 1)) + return false; + return doPatch(name, 0); + } + return true; +} + +std::string CodePatcher::toString() +{ + std::stringstream out; + for (auto& patch : patch_map) + { + out << std::setw(16) << patch.first << "[ " + << std::setw(8) << std::hex << patch.second.addr << " , " + << std::setw(8) << std::hex << patch.second.new_offset + << " ]" << std::endl + << std::setw(23) << "Old: " + << utils::convertBinToHexstr(patch.second.old_code) << std::endl + << std::setw(23) << "New: " + << utils::convertBinToHexstr(patch.second.new_code) << std::endl; + } + return out.str(); +} + +bool CodePatcher::codeCmp(unsigned long addr, std::vector<unsigned char> code) +{ + if (code.size() == 0) + return true; + unsigned char buf[code.size()] = {0}; + if (!nd.read_fn(&nd, addr, &buf[0], code.size())) + return false; + /* TODO: replace with memcmp? */ + for (unsigned i = 0; i < code.size(); ++i) + { + if (buf[i] != code[i]) + return false; + } + return true; +} diff --git a/aoe2hd/src/ModuleMemory.cpp b/aoe2hd/src/ModuleMemory.cpp new file mode 100755 index 0000000..19e501b --- /dev/null +++ b/aoe2hd/src/ModuleMemory.cpp @@ -0,0 +1,118 @@ +#include <assert.h> + +#include <sstream> +#include <iomanip> + +#include "ModuleMemory.h" + +ModuleMemory::ModuleMemory(const native_data& nd) + : nd(nd), ptr_map(), ptr_read_count(0), ptr_invalid_count(0) +{ + assert(nd.read_fn); +} + +ModuleMemory::~ModuleMemory() +{ +} + +unsigned long ModuleMemory::getPtr(const std::string& name) +{ + bool valid = ptrExists(name) && ptrValid(name); + if (!valid) + return 0; + return ptr_map[name].ptr; +} + +unsigned long ModuleMemory::getPtr(const std::string& name, unsigned long *dest_ptr) +{ + assert(dest_ptr); + unsigned long ret = getPtr(name); + *dest_ptr = ret; + return ret; +} + +unsigned long ModuleMemory::getPtr(const std::string& name, unsigned long base, + unsigned long offset) +{ + target_ptr out = {0}; + out.base = base; + out.offset = offset; + out.valid = nd.read_fn(&nd, base + offset, &out.ptr, sizeof(out.ptr)); + ptr_map[name] = out; + if (out.valid) + { + ++ptr_read_count; + return out.ptr; + } + else + { + return 0; + } +} + +unsigned long ModuleMemory::recheckPtr(const std::string& name) +{ + if (!ptrExists(name)) + return 0; + target_ptr old = ptr_map[name]; + unsigned long new_ptr = getPtr(name, old.base, old.offset); + if (!new_ptr) + { + ptr_map[name].valid = false; + } + return new_ptr; +} + +void ModuleMemory::revalidateAllPtr() +{ +} + +bool ModuleMemory::getData(const std::string& name, void *buffer, unsigned long siz) +{ + assert(buffer); + if (!getPtr(name)) + return false; + if (!nd.read_fn(&nd, ptr_map[name].ptr, buffer, siz)) + { + ptr_map[name].valid = false; + } + return ptr_map[name].valid; +} + +bool ModuleMemory::ptrSetDependency(const std::string& name, const std::string& dependency) +{ + if (!getPtr(name) || !getPtr(dependency)) + return false; + ptr_map[name].dependency = dependency; + ptr_map[dependency].children.insert(name); + return true; +} + +std::string ModuleMemory::toString() +{ + std::stringstream out; + for (auto& ptr : ptr_map) + { + out << std::setw(16) << ptr.first << "[ " + << std::setw(8) << std::hex << ptr.second.base << " + " + << std::setw(8) << std::hex << ptr.second.offset << " = " + << std::setw(8) << std::hex << ptr.second.ptr << " , " + << std::setw(5) << (ptr.second.valid ? "TRUE" : "FALSE") << " , " + << std::setw(16) << (ptr.second.dependency.c_str() ? ptr.second.dependency.c_str() : "") + << " ]"; + for (auto& child : ptr.second.children) + { + out << ", " << child; + } + out << std::endl; + } + return out.str(); +} + +std::string ModuleMemory::toStringStats() +{ + std::stringstream out; + out << "PtrReadCount: " << ptr_read_count << " , " + << "PtrInvalidCount: " << ptr_invalid_count; + return out.str(); +} diff --git a/aoe2hd/src/main.cpp b/aoe2hd/src/main.cpp new file mode 100755 index 0000000..2e064f0 --- /dev/null +++ b/aoe2hd/src/main.cpp @@ -0,0 +1,184 @@ +/************************************ + * AoE2 HD Steam + * Minimap/NoFog-Hack (with a restriction) + * + * coded by lnslbrty/dev0, \x90 + * + * This hack may cause desyncs! + ************************************/ + +#include <windows.h> +#include <psapi.h> +#include <stdio.h> +#include <assert.h> + +#include <vector> +#include <iostream> +#include <iomanip> +#include <string> +#include <sstream> + +#include "CodeGenerator.h" +#include "CodePatcher.h" +#include "CodeInjector.h" +#include "ModuleMemory.h" +extern "C" { +#include "native.h" +#include "aoe2hd.h" +} + + +int main(int argc, char **argv) +{ + using namespace std; + + (void) argc; + (void) argv; + + native_data nd = {0}; + initNativeData(&nd); + + assert(get_module_proc(&nd, "Age of Empires II: HD Edition")); + assert(get_module_base(&nd, "AoK HD.exe")); + + ModuleMemory mm(nd); + mm.getPtr("MainClass", nd.proc.modbase, 0x009C7774); + mm.getPtr("GameClass", mm.getPtr("MainClass"), 0x4); + mm.getPtr("PlayerArray", mm.getPtr("GameClass"), 0x184); + mm.getPtr("PlayerNameBase", nd.proc.modbase, 0x006DB62C); + mm.getPtr("PlayerNamePtr0", mm.getPtr("PlayerNameBase"), 0x794); + mm.getPtr("PlayerNamePtr1", mm.getPtr("PlayerNamePtr0"), 0x5C); + + mm.ptrSetDependency("GameClass", "MainClass"); + mm.ptrSetDependency("PlayerArray", "GameClass"); + mm.ptrSetDependency("PlayerNamePtr0", "PlayerNameBase"); + mm.ptrSetDependency("PlayerNamePtr1", "PlayerNamePtr0"); + + for (unsigned long i = 1; i < 9; ++i) + { + stringstream player; + player << "Player" << i; + mm.getPtr(player.str(), (unsigned long)mm.getPtr("PlayerArray"), 0x8 * i); + mm.ptrSetDependency(player.str(), "PlayerArray"); + + stringstream player_res; + player_res << "Player" << i << "Resources"; + mm.getPtr(player_res.str(), (unsigned long)mm.getPtr(player.str()), 0x3C); + mm.ptrSetDependency(player_res.str(), player.str()); + + stringstream player_name; + player_name << "Player" << i << "Name"; + mm.getPtr(player_name.str(), (unsigned long)mm.getPtr("PlayerNamePtr1"), 0xBC + (0x60 * (i-1))); + mm.ptrSetDependency(player_name.str(), "PlayerNamePtr1"); + } + + CodeInjector ci(nd); + assert(ci.allocCodeSegment("MapCode")); + CodePatcher cp(nd); + CodeGenerator original(nd), injected(nd); + + { + original.addCode({DUMMY5}); + injected.addCode({MAP_NOFOG0}).addCode({MAP_NOFOGI}).addCode({MAP_NOFOG1}).addCode({DUMMY5}); + + assert(ci.addCode("MapCode", "NoFog", injected.buildSize())); + injected.setRel32JMP(3, MAP_NOFOG, ci.getCodeAddr("MapCode", "NoFog")); + ci.setCode("MapCode", "NoFog", injected.buildAndClear()); + + original.setRel32JMP(0, ci.getCodeAddr("MapCode", "NoFog"), MAP_NOFOG, true).addCode({0x90}); + assert(cp.addPatch("NoFog", MAP_NOFOG, {MAP_NOFOG0,MAP_NOFOG1}, original.buildAndClear())); + cp.setPatchSuspend("NoFog", 1); + assert(cp.autoPatch("NoFog")); + } + { + original.addCode({DUMMY5}); + injected.addCode({MAP_MINI0}).addCode({MAP_MINII}).addCode({MAP_MINI1}).addCode({DUMMY5}); + + assert(ci.addCode("MapCode", "MiniMap", injected.buildSize())); + injected.setRel32JMP(3, MAP_MINI, ci.getCodeAddr("MapCode", "MiniMap")); + ci.setCode("MapCode", "MiniMap", injected.buildAndClear()); + + original.setRel32JMP(0, ci.getCodeAddr("MapCode", "MiniMap"), MAP_MINI, true).addCode({0x90,0x90,0x90,0x90}); + assert(cp.addPatch("MiniMap", MAP_MINI, {MAP_MINI0,MAP_MINI1}, original.buildAndClear())); + cp.setPatchSuspend("MiniMap", 1); + assert(cp.autoPatch("MiniMap")); + } + { + original.addCode({DUMMY5}); + injected.addCode({MAP_SMTH0}).addCode({MAP_SMTHI}).addCode({MAP_SMTH1}).addCode({DUMMY5}) + .addCode({DUMMY5,DUMMY5,DUMMY5,DUMMY5,DUMMY5}); + + assert(ci.addCode("MapCode", "Smth", injected.buildSize())); + injected.setRel32JMP(3, MAP_SMTH, ci.getCodeAddr("MapCode", "Smth")); + ci.setCode("MapCode", "Smth", injected.buildAndClear()); + + original.setRel32JMP(0, ci.getCodeAddr("MapCode", "Smth"), MAP_SMTH, true).addCode({0x90,0x90,0x90,0x90}); + assert(cp.addPatch("Smth", MAP_SMTH, {MAP_SMTH0,MAP_SMTH1}, original.buildAndClear())); + cp.setPatchSuspend("Smth", 1); + assert(cp.autoPatch("Smth")); + } + { + original.addCode({DUMMY5}); + injected.addCode({MAP_UNIT0}).addCode({MAP_UNITI}).addCode({MAP_UNIT1}).addCode({DUMMY5}); + + assert(ci.addCode("MapCode", "Units", injected.buildSize())); + injected.setRel32JMP(3, MAP_UNIT, ci.getCodeAddr("MapCode", "Units")); + ci.setCode("MapCode", "Units", injected.buildAndClear()); + + original.setRel32JMP(0, ci.getCodeAddr("MapCode", "Units"), MAP_UNIT, true).addCode({0x90,0x90,0x90,0x90,0x90}); + assert(cp.addPatch("Units", MAP_UNIT, {MAP_UNIT0,MAP_UNIT1}, original.buildAndClear())); + cp.setPatchSuspend("Units", 1); + assert(cp.autoPatch("Units")); + } + + cout << ci.toString() << endl; + cout << cp.toString() << endl; + cout << "[PRESS A KEY TO CONTINUE]" << endl; + system("pause"); + + while (1) + { + cls( GetStdHandle( STD_OUTPUT_HANDLE )); + + while (!mm.recheckPtr("MainClass")) + { + Sleep(1000); + } + mm.revalidateAllPtr(); + + for (unsigned long i = 1; i < 9; ++i) + { + + stringstream player_res; + player_res << "Player" << i << "Resources"; + struct resources res = {0}; + if (!mm.getData(player_res.str(), &res, sizeof(res))) + { + continue; + } + + cout << "player[" << i << "]: " + << "wood.: " << setw(8) << dec << (unsigned long)res.wood << ", " + << "food.: " << setw(8) << dec << (unsigned long)res.food << ", " + << "gold.: " << setw(8) << dec << (unsigned long)res.gold << ", " + << "stone: " << setw(8) << dec << (unsigned long)res.stone << endl; + cout << " " << setw(8) + << "rpop.: " << setw(8) << dec << res.remainingPop << ", " + << "tpop.: " << setw(8) << dec << res.currentPop << endl; + + stringstream player_name; + player_name << "Player" << i << "Name"; + char name[32] = {0}; + mm.getData(player_name.str(), &name, 31); + cout << " " << name << endl; + } + + //cout << mm.toString() << endl; + //cout << mm.toStringStats() << endl; + //system("pause"); + Sleep(1000); + } + CloseHandle(nd.proc.hndl); + + return 0; +} diff --git a/aoe2hd/src/native.c b/aoe2hd/src/native.c new file mode 100755 index 0000000..d086555 --- /dev/null +++ b/aoe2hd/src/native.c @@ -0,0 +1,173 @@ +#include <windows.h> +#include <psapi.h> +#include <stdio.h> +#include <stdbool.h> +#include <assert.h> + +#include "native.h" + + +typedef LONG (NTAPI *NtSuspendProcess)(IN HANDLE ProcessHandle); +typedef LONG (NTAPI *NtResumeProcess)(IN HANDLE ProcessHandle); + +/* Standard error macro for reporting API errors */ +#define PERR(bSuccess, api){if(!(bSuccess)) printf("%s:Error %ld from %s \ + on line %ld\n", __FILE__, GetLastError(), api, (long)__LINE__);} + +void initNativeData(native_data *nd) +{ + assert(nd); + nd->alloc_fn = mem_alloc; + nd->read_fn = read_procmem; + nd->write_fn = write_procmem; + nd->suspend_fn = suspendProcess; +} + +/* see: https://support.microsoft.com/en-us/help/99261/how-to-performing-clear-screen-cls-in-a-console-application */ +void cls(HANDLE hConsole) +{ + COORD coordScreen = { 0, 0 }; /* here's where we'll home the + cursor */ + BOOL bSuccess; + DWORD cCharsWritten; + CONSOLE_SCREEN_BUFFER_INFO csbi; /* to get buffer info */ + DWORD dwConSize; /* number of character cells in + the current buffer */ + + /* get the number of character cells in the current buffer */ + + bSuccess = GetConsoleScreenBufferInfo( hConsole, &csbi ); + PERR( bSuccess, "GetConsoleScreenBufferInfo" ); + dwConSize = csbi.dwSize.X * csbi.dwSize.Y; + + /* fill the entire screen with blanks */ + + bSuccess = FillConsoleOutputCharacter( hConsole, (TCHAR) ' ', + dwConSize, coordScreen, &cCharsWritten ); + PERR( bSuccess, "FillConsoleOutputCharacter" ); + + /* get the current text attribute */ + + bSuccess = GetConsoleScreenBufferInfo( hConsole, &csbi ); + PERR( bSuccess, "ConsoleScreenBufferInfo" ); + + /* now set the buffer's attributes accordingly */ + + bSuccess = FillConsoleOutputAttribute( hConsole, csbi.wAttributes, + dwConSize, coordScreen, &cCharsWritten ); + PERR( bSuccess, "FillConsoleOutputAttribute" ); + + /* put the cursor at (0, 0) */ + + bSuccess = SetConsoleCursorPosition( hConsole, coordScreen ); + PERR( bSuccess, "SetConsoleCursorPosition" ); + return; +} + +bool get_module_proc(native_data *nd, LPCTSTR window_name) +{ + HWND hwnd; + + assert(window_name); + hwnd = FindWindow(NULL, window_name); + if (!hwnd) + goto error; + GetWindowThreadProcessId(hwnd, &nd->proc.pid); + if (!nd->proc.pid) + goto error; + nd->proc.hndl = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_OPERATION + | PROCESS_VM_READ | PROCESS_VM_WRITE, 0, nd->proc.pid); +error: + return nd->proc.hndl != NULL; +} + +bool get_module_base(native_data *nd, LPCTSTR module_name) +{ + HMODULE hMods[1024]; + DWORD cbNeeded; + unsigned int i; + + assert(module_name); + if (EnumProcessModules(nd->proc.hndl, hMods, sizeof(hMods), &cbNeeded)) + { + for (i = 0; i < (cbNeeded / sizeof(HMODULE)); i++) + { + TCHAR szModName[MAX_PATH]; + TCHAR szModPath[MAX_PATH]; + + if (GetModuleBaseName(nd->proc.hndl, hMods[i], szModName, sizeof(szModName) / sizeof(TCHAR)) + && GetModuleFileNameEx(nd->proc.hndl, hMods[i], szModPath, + sizeof(szModPath) / sizeof(TCHAR))) + { + if (strncmp(szModName, module_name, MAX_PATH) == 0) + { + nd->proc.modbase =(unsigned long)(hMods[i]); + return true; + } + } + } + } + return false; +} + +bool read_procmem(const native_data *nd, unsigned long addr, + void *buffer, unsigned long siz) +{ + SIZE_T bytes_read = 0; + unsigned long *vmptr = (unsigned long *)addr; + + assert(addr && buffer && siz); + if (!ReadProcessMemory(nd->proc.hndl, vmptr, buffer, siz, &bytes_read)) + return false; + if (bytes_read != siz) + return false; + + return true; +} + +bool write_procmem(const native_data *nd, unsigned long addr, const void *buffer, unsigned long siz) +{ + SIZE_T bytes_written = 0; + unsigned long *vmptr = (unsigned long *)addr; + + assert(addr && buffer && siz); + if (!WriteProcessMemory(nd->proc.hndl, vmptr, buffer, siz, &bytes_written)) + return false; + if (bytes_written != siz) + return false; + + return true; +} + +unsigned long mem_alloc(const native_data *nd, unsigned long siz) +{ + return (unsigned long)VirtualAllocEx(nd->proc.hndl, NULL, siz, + MEM_COMMIT | MEM_RESERVE, + PAGE_EXECUTE_READWRITE); +} + +/* see: https://github.com/mridgers/clink/issues/420 */ +bool suspendProcess(const native_data *nd, int doResume) +{ + bool ret = false; + NtSuspendProcess pfnNtSuspendProcess = + (NtSuspendProcess)GetProcAddress(GetModuleHandle("ntdll"), "NtSuspendProcess"); + NtResumeProcess pfnNtResumeProcess = + (NtResumeProcess)GetProcAddress(GetModuleHandle("ntdll"), "NtResumeProcess"); + + HANDLE processHandle = OpenProcess(PROCESS_ALL_ACCESS, FALSE, nd->proc.pid); + if (!processHandle) + return false; + if (doResume) + { + if (pfnNtResumeProcess(processHandle) == 0) + ret = true; + } + else + { + if (pfnNtSuspendProcess(processHandle) == 0) + ret = true; + } + CloseHandle(processHandle); + return ret; +} diff --git a/aoe2hd/src/utils.cpp b/aoe2hd/src/utils.cpp new file mode 100755 index 0000000..7639ab7 --- /dev/null +++ b/aoe2hd/src/utils.cpp @@ -0,0 +1,19 @@ +#include "utils.h" + +#include <sstream> +#include <iomanip> + + +namespace utils +{ +std::string convertBinToHexstr(const std::vector<unsigned char>& bin) +{ + std::stringstream buffer; + for (auto byte : bin) + { + buffer << std::hex << std::setfill('0'); + buffer << std::setw(2) << static_cast<unsigned>(byte); + } + return buffer.str(); +} +} |