mumble/plugins/dods/dods.cpp
2009-12-17 21:15:07 -06:00

257 lines
5.7 KiB
C++

#include <stdio.h>
#include <stdlib.h>
#include <windows.h>
#include <tlhelp32.h>
#include <string>
#include <sstream>
#define _USE_MATH_DEFINES
#include <math.h>
#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<float>(M_PI / 180.0f);
v *= static_cast<float>(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<float>(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 << "<context>"
<< "<game>dods</game>"
<< "<hostport>" << chHostStr << "</hostport>"
<< "<team>" << sTeam << "</team>"
<< "</context>";
context = new_context.str();
/* TODO
new_identity << "<identity>"
<< "<name>" << "SAS" << "</name>"
<< "</identity>";
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<address> (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;
}