#include #include #include #include #include #include #define _USE_MATH_DEFINES #include #include "../mumble_plugin.h" using namespace std; HANDLE h; BYTE *posptr; BYTE *rotptr; BYTE *stateptr; BYTE *hostptr; BYTE *teamptr; static DWORD getProcess(const wchar_t *exename) { PROCESSENTRY32 pe; DWORD pid = 0; pe.dwSize = sizeof(pe); HANDLE hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0); if (hSnap != INVALID_HANDLE_VALUE) { BOOL ok = Process32First(hSnap, &pe); while (ok) { if (wcscmp(pe.szExeFile, exename)==0) { pid = pe.th32ProcessID; break; } ok = Process32Next(hSnap, &pe); } CloseHandle(hSnap); } return pid; } static BYTE *getModuleAddr(DWORD pid, const wchar_t *modname) { MODULEENTRY32 me; BYTE *addr = NULL; me.dwSize = sizeof(me); HANDLE hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPMODULE, pid); if (hSnap != INVALID_HANDLE_VALUE) { BOOL ok = Module32First(hSnap, &me); while (ok) { if (wcscmp(me.szModule, modname)==0) { addr = me.modBaseAddr; break; } ok = Module32Next(hSnap, &me); } CloseHandle(hSnap); } return addr; } static bool peekProc(VOID *base, VOID *dest, SIZE_T len) { SIZE_T r; BOOL ok=ReadProcessMemory(h, base, dest, len, &r); return (ok && (r == len)); } static void about(HWND h) { ::MessageBox(h, L"Reads audio position information from Day of Defeat: Source (Build 4057). IP:Port context with team discriminator.", L"Mumble DODS Plugin", MB_OK); } static bool calcout(float *pos, float *rot, float *opos, float *front, float *top) { float h = rot[0]; float v = rot[1]; if ((v < -360.0f) || (v > 360.0f) || (h < -360.0f) || (h > 360.0f)) return false; h *= static_cast(M_PI / 180.0f); v *= static_cast(M_PI / 180.0f); // Seems DODS is in inches. INCHES?!? opos[0] = pos[0] / 39.37f; opos[1] = pos[2] / 39.37f; opos[2] = pos[1] / 39.37f; front[0] = cos(v) * cos(h); front[1] = -sin(h); front[2] = sin(v) * cos(h); h -= static_cast(M_PI / 2.0f); top[0] = cos(v) * cos(h); top[1] = -sin(h); top[2] = sin(v) * cos(h); return true; } static int fetch(float *avatar_pos, float *avatar_front, float *avatar_top, float *camera_pos, float *camera_front, float *camera_top, string &context, wstring &identity) { for (int i=0;i<3;i++) avatar_pos[i] = avatar_front[i] = avatar_top[i] = 0; float ipos[3], rot[3]; bool ok; char state; char chHostStr[40]; BYTE bTeam; string sTeam; wostringstream new_identity; ostringstream new_context; ok = peekProc(posptr, ipos, 12) && peekProc(rotptr, rot, 12) && peekProc(stateptr, &state, 1) && peekProc(hostptr, chHostStr, 40) && peekProc(teamptr, &bTeam, 1); if (!ok) return false; chHostStr[39] = 0; switch (bTeam) { case 114: sTeam = "U.S. Army"; break; case 100: sTeam = "Wermacht"; break; default: sTeam = "Unknown"; break; } new_context << "" << "dods" << "" << chHostStr << "" << "" << sTeam << "" << ""; context = new_context.str(); /* TODO new_identity << "" << "" << "SAS" << "" << ""; identity = new_identity.str(); */ // Check to see if you are in a server if (state == 0 || state == 2) return true; // Deactivate plugin if (ok) { int res = calcout(ipos, rot, avatar_pos, avatar_front, avatar_top); if (res) { for (int i=0;i<3;++i) { camera_pos[i] = avatar_pos[i]; camera_front[i] = avatar_front[i]; camera_top[i] = avatar_top[i]; } return res; } } return false; } static int trylock() { h = NULL; posptr = rotptr = NULL; DWORD pid=getProcess(L"hl2.exe"); if (!pid) return false; BYTE *mod=getModuleAddr(pid, L"client.dll"); if (!mod) return false; BYTE *mod_engine=getModuleAddr(pid, L"engine.dll"); if (!mod_engine) return false; BYTE *mod_vphysics=getModuleAddr(pid, L"vphysics.dll"); if (!mod_vphysics) return false; h=OpenProcess(PROCESS_VM_READ, false, pid); if (!h) return false; // Check if we really have DODS running /* position tuple: client.dll+0x3f62a0 (x,y,z, float) orientation tuple: client.dll+0x3f6220 (v,h float) ID string: client.dll+0x3f6d91 = "DODSpectatorGUI@@" (17 characters, text) spawn state: client.dll+0x3e2b94 (0 when at main menu, 2 when at team selection menu or when not spawned, and between 5 and 6 when spawned, byte) host string: client.dll+0x
(ip:port zero-terminated string) */ // Remember addresses for later posptr = mod + 0x3FB358; rotptr = mod + 0x3FB2CC; stateptr = mod + 0x3E7BDC; hostptr = mod_engine + 0x3C91A4; teamptr = mod_vphysics + 0x489769; // Gamecheck char sMagic[17]; if (!peekProc(mod + 0x3FBE49, sMagic, 17) || strncmp("DODSpectatorGUI@@", sMagic, 17)!=0) return false; // Check if we can get meaningful data from it float apos[3], afront[3], atop[3]; float cpos[3], cfront[3], ctop[3]; wstring sidentity; string scontext; if (fetch(apos, afront, atop, cpos, cfront, ctop, scontext, sidentity)) return true; // If it failed clean up CloseHandle(h); h = NULL; return false; } static void unlock() { if (h) { CloseHandle(h); h = NULL; } return; } static const std::wstring longdesc() { return std::wstring(L"Supports DODS build 4057. No identity support yet."); } static std::wstring description(L"Day of Defeat: Source (Build 4057)"); static std::wstring shortname(L"Day of Defeat: Source"); static MumblePlugin dodsplug = { MUMBLE_PLUGIN_MAGIC, description, shortname, about, NULL, trylock, unlock, longdesc, fetch }; extern "C" __declspec(dllexport) MumblePlugin *getMumblePlugin() { return &dodsplug; }