mirror of
https://github.com/mumble-voip/mumble.git
synced 2025-10-26 11:19:16 +00:00
218 lines
4.9 KiB
C++
218 lines
4.9 KiB
C++
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <windows.h>
|
|
#include <tlhelp32.h>
|
|
|
|
#define _USE_MATH_DEFINES
|
|
#include <math.h>
|
|
|
|
#include "../mumble_plugin.h"
|
|
|
|
HANDLE h;
|
|
BYTE *posptr;
|
|
BYTE *rotptr;
|
|
BYTE *stateptr;
|
|
BYTE *ccontextptr;
|
|
|
|
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 Age of Chivalry (Build 3740)", L"Mumble AOC 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 AOC 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 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 *mod2=getModuleAddr(pid, L"server.dll");
|
|
if (!mod2)
|
|
return false;
|
|
BYTE *mod3=getModuleAddr(pid, L"engine.dll");
|
|
if (!mod3)
|
|
return false;
|
|
h=OpenProcess(PROCESS_VM_READ, false, pid);
|
|
if (!h)
|
|
return false;
|
|
|
|
// Check if we really have AOC running
|
|
/*
|
|
position tuple: client.dll+0xba54a8 (x,y,z, float)
|
|
orientation tuple: client.dll+0xb98e84 (v,h float)
|
|
ID string: client.dll+0xb60fc0 = "ageofchivalry" (13 characters, text)
|
|
spawn state: client.dll+0xb33ae8 (0 when at main menu, 1 when at team selection, 2 when not spawned,
|
|
6 when spawned on red team, 7 when spawned on blue team, byte)
|
|
context offsets: server.dll + 0x65e538 = team info
|
|
engine.dll + 0x5acd40 = ip:port
|
|
*/
|
|
char sMagic[13];
|
|
if (!peekProc(mod + 0xb60fc0, sMagic, 13) || strncmp("ageofchivalry", sMagic, 13)!=0)
|
|
return false;
|
|
|
|
// Remember addresses for later
|
|
posptr = mod + 0xba54a8;
|
|
rotptr = mod + 0xb98e84;
|
|
stateptr = mod + 0xb33ae8;
|
|
ccontextptr = mod2 + 0x65e538;
|
|
ccontextptr = mod3 + 0x5acd40;
|
|
|
|
float pos[3];
|
|
float rot[3];
|
|
float opos[3], top[3], front[3];
|
|
|
|
bool ok = peekProc(posptr, pos, 12) &&
|
|
peekProc(rotptr, rot, 12);
|
|
|
|
if (ok)
|
|
return calcout(pos, rot, opos, top, front);
|
|
// If it failed clean up
|
|
CloseHandle(h);
|
|
h = NULL;
|
|
return false;
|
|
}
|
|
|
|
static void unlock() {
|
|
if (h) {
|
|
CloseHandle(h);
|
|
h = NULL;
|
|
}
|
|
return;
|
|
}
|
|
|
|
static int fetch(float *avatar_pos, float *avatar_front, float *avatar_top, float *camera_pos, float *camera_front, float *camera_top, std::string &context, std::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 ccontext[256];
|
|
|
|
context.assign(ccontext);
|
|
|
|
ok = peekProc(posptr, ipos, 12) &&
|
|
peekProc(rotptr, rot, 12) &&
|
|
peekProc(stateptr, &state, 1) &&
|
|
peekProc(ccontextptr, ccontext, 256);
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
// Check to see if you are spawned
|
|
if (state == 0 || state == 1 || state == 2)
|
|
return true; // Deactivate plugin
|
|
|
|
return false;
|
|
}
|
|
|
|
static const std::wstring longdesc() {
|
|
return std::wstring(L"Supports AOC Build 3740. Context support only.");
|
|
}
|
|
|
|
static std::wstring description(L"Age of Chivalry (Build 3740)");
|
|
static std::wstring shortname(L"Age of Chivalry");
|
|
|
|
static MumblePlugin aocplug = {
|
|
MUMBLE_PLUGIN_MAGIC,
|
|
description,
|
|
shortname,
|
|
about,
|
|
NULL,
|
|
trylock,
|
|
unlock,
|
|
longdesc,
|
|
fetch
|
|
};
|
|
|
|
extern "C" __declspec(dllexport) MumblePlugin *getMumblePlugin() {
|
|
return &aocplug;
|
|
}
|