1
0
mirror of https://github.com/FunkyFr3sh/cnc-ddraw.git synced 2025-03-22 00:32:10 +01:00
cnc-ddraw/src/utils.c
2024-06-10 00:05:04 +02:00

751 lines
20 KiB
C

#include <windows.h>
#include <intrin.h>
#include <math.h>
#include "ddraw.h"
#include "debug.h"
#include "dd.h"
#include "ddsurface.h"
#include "hook.h"
#include "mouse.h"
#include "render_d3d9.h"
#include "utils.h"
#include "config.h"
/*
The following code is licensed under the MIT license:
DetourEnumerateModules
Copyright (c) Microsoft Corporation
https://github.com/microsoft/Detours
*/
#define MM_ALLOCATION_GRANULARITY 0x10000
HMODULE WINAPI util_enumerate_modules(_In_opt_ HMODULE hModuleLast)
{
PBYTE pbLast = (PBYTE)hModuleLast + MM_ALLOCATION_GRANULARITY;
MEMORY_BASIC_INFORMATION mbi;
ZeroMemory(&mbi, sizeof(mbi));
// Find the next memory region that contains a mapped PE image.
//
for (;; pbLast = (PBYTE)mbi.BaseAddress + mbi.RegionSize) {
if (VirtualQuery(pbLast, &mbi, sizeof(mbi)) <= 0) {
break;
}
// Skip uncommitted regions and guard pages.
//
if ((mbi.State != MEM_COMMIT) ||
((mbi.Protect & 0xff) == PAGE_NOACCESS) ||
(mbi.Protect & PAGE_GUARD)) {
continue;
}
__try {
PIMAGE_DOS_HEADER pDosHeader = (PIMAGE_DOS_HEADER)pbLast;
if (pDosHeader->e_magic != IMAGE_DOS_SIGNATURE ||
(DWORD)pDosHeader->e_lfanew > mbi.RegionSize ||
(DWORD)pDosHeader->e_lfanew < sizeof(*pDosHeader)) {
continue;
}
PIMAGE_NT_HEADERS pNtHeader = (PIMAGE_NT_HEADERS)((PBYTE)pDosHeader +
pDosHeader->e_lfanew);
if (pNtHeader->Signature != IMAGE_NT_SIGNATURE) {
continue;
}
return (HMODULE)pDosHeader;
}
#if defined(_MSC_VER)
#pragma prefast(suppress:28940, "A bad pointer means this probably isn't a PE header.")
#endif
__except (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION ?
EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH) {
continue;
}
}
return NULL;
}
BOOL util_is_bad_read_ptr(void* p)
{
MEMORY_BASIC_INFORMATION mbi = { 0 };
if (VirtualQuery(p, &mbi, sizeof(mbi)))
{
DWORD mask = (
PAGE_READONLY |
PAGE_READWRITE |
PAGE_WRITECOPY |
PAGE_EXECUTE_READ |
PAGE_EXECUTE_READWRITE |
PAGE_EXECUTE_WRITECOPY);
BOOL b = !(mbi.Protect & mask);
if (mbi.Protect & (PAGE_GUARD | PAGE_NOACCESS))
b = TRUE;
return b;
}
return TRUE;
}
BOOL util_is_minimized(HWND hwnd)
{
RECT rc = { 0 };
return IsIconic(hwnd) || (real_GetClientRect(hwnd, &rc) && (rc.right - rc.left == 0 || rc.bottom - rc.top == 0));
}
BOOL util_in_foreground()
{
DWORD process_id = 0;
return GetWindowThreadProcessId(real_GetForegroundWindow(), &process_id) && process_id == GetCurrentProcessId();
}
BOOL util_is_avx_supported()
{
const DWORD XMM_STATE_BIT = 1 << 1;
const DWORD YMM_STATE_BIT = 1 << 2;
const DWORD OS_AVX_BITS = XMM_STATE_BIT | YMM_STATE_BIT;
const DWORD AVX_BIT = 1 << 28;
const DWORD OSXSAVE_BIT = 1 << 27;
const DWORD XSAVE_BIT = 1 << 26;
const DWORD CPU_AVX_BITS = AVX_BIT | OSXSAVE_BIT | XSAVE_BIT;
BOOL result = FALSE;
__try
{
int info[4] = { 0 };
__cpuid(info, 0);
if (info[0] >= 1)
{
__cpuid(info, 1);
if ((info[2] & CPU_AVX_BITS) == CPU_AVX_BITS)
{
unsigned int xcr0 = 0;
#ifdef _MSC_VER
xcr0 = (unsigned int)_xgetbv(_XCR_XFEATURE_ENABLED_MASK);
#elif __AVX__
__asm__("xgetbv" : "=a" (xcr0) : "c" (0) : "%edx");
#endif
result = (xcr0 & OS_AVX_BITS) == OS_AVX_BITS;
}
}
}
__except (EXCEPTION_EXECUTE_HANDLER)
{
}
return result;
}
void util_limit_game_ticks()
{
if (GetCurrentThreadId() != g_ddraw.gui_thread_id)
return;
if (g_ddraw.ticks_limiter.htimer)
{
FILETIME ft = { 0 };
GetSystemTimeAsFileTime(&ft);
if (CompareFileTime((FILETIME*)&g_ddraw.ticks_limiter.due_time, &ft) == -1)
{
memcpy(&g_ddraw.ticks_limiter.due_time, &ft, sizeof(LARGE_INTEGER));
}
else
{
WaitForSingleObject(g_ddraw.ticks_limiter.htimer, g_ddraw.ticks_limiter.tick_length * 2);
}
g_ddraw.ticks_limiter.due_time.QuadPart += g_ddraw.ticks_limiter.tick_length_ns;
SetWaitableTimer(g_ddraw.ticks_limiter.htimer, &g_ddraw.ticks_limiter.due_time, 0, NULL, NULL, FALSE);
}
else
{
static DWORD next_game_tick;
if (!next_game_tick)
{
next_game_tick = timeGetTime();
return;
}
next_game_tick += g_ddraw.ticks_limiter.tick_length;
DWORD tick_count = timeGetTime();
int sleep_time = next_game_tick - tick_count;
if (sleep_time <= 0 || sleep_time > g_ddraw.ticks_limiter.tick_length)
{
next_game_tick = tick_count;
}
else
{
Sleep(sleep_time);
}
}
}
void util_update_bnet_pos(int new_x, int new_y)
{
static int old_x = -32000;
static int old_y = -32000;
if (old_x == -32000 || old_y == -32000 || !g_ddraw.bnet_active)
{
old_x = new_x;
old_y = new_y;
return;
}
POINT pt = { 0, 0 };
real_ClientToScreen(g_ddraw.hwnd, &pt);
RECT mainrc;
SetRect(&mainrc, pt.x, pt.y, pt.x + g_ddraw.width, pt.y + g_ddraw.height);
int adj_y = 0;
int adj_x = 0;
HWND hwnd = FindWindowEx(HWND_DESKTOP, NULL, "SDlgDialog", NULL);
while (hwnd != NULL)
{
RECT rc;
real_GetWindowRect(hwnd, &rc);
OffsetRect(&rc, new_x - old_x, new_y - old_y);
real_SetWindowPos(
hwnd,
0,
rc.left,
rc.top,
0,
0,
SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE);
if (rc.bottom - rc.top <= g_ddraw.height)
{
if (rc.bottom > mainrc.bottom && abs(mainrc.bottom - rc.bottom) > abs(adj_y))
{
adj_y = mainrc.bottom - rc.bottom;
}
else if (rc.top < mainrc.top && abs(mainrc.top - rc.top) > abs(adj_y))
{
adj_y = mainrc.top - rc.top;
}
}
if (rc.right - rc.left <= g_ddraw.width)
{
if (rc.right > mainrc.right && abs(mainrc.right - rc.right) > abs(adj_x))
{
adj_x = mainrc.right - rc.right;
}
else if (rc.left < mainrc.left && abs(mainrc.left - rc.left) > abs(adj_x))
{
adj_x = mainrc.left - rc.left;
}
}
hwnd = FindWindowEx(HWND_DESKTOP, hwnd, "SDlgDialog", NULL);
}
if (adj_x || adj_y)
{
HWND hwnd = FindWindowEx(HWND_DESKTOP, NULL, "SDlgDialog", NULL);
while (hwnd != NULL)
{
RECT rc;
real_GetWindowRect(hwnd, &rc);
OffsetRect(&rc, adj_x, adj_y);
real_SetWindowPos(
hwnd,
0,
rc.left,
rc.top,
0,
0,
SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE);
hwnd = FindWindowEx(HWND_DESKTOP, hwnd, "SDlgDialog", NULL);
}
}
old_x = new_x;
old_y = new_y;
}
BOOL util_get_lowest_resolution(
float ratio,
SIZE* out_res,
DWORD min_width,
DWORD min_height,
DWORD max_width,
DWORD max_height)
{
BOOL result = FALSE;
int org_ratio = (int)((ratio + 0.005f) * 100);
SIZE lowest = { .cx = max_width + 1, .cy = max_height + 1 };
DWORD i = 0;
DEVMODE m;
memset(&m, 0, sizeof(DEVMODE));
m.dmSize = sizeof(DEVMODE);
while (real_EnumDisplaySettingsA(NULL, i, &m))
{
if (m.dmPelsWidth >= min_width &&
m.dmPelsHeight >= min_height &&
m.dmPelsWidth <= max_width &&
m.dmPelsHeight <= max_height &&
m.dmPelsWidth <= lowest.cx &&
m.dmPelsHeight <= lowest.cy)
{
int res_ratio = (int)((((float)m.dmPelsWidth / m.dmPelsHeight) + 0.005f) * 100);
if (abs(res_ratio - org_ratio) <= 5)
{
result = TRUE;
out_res->cx = lowest.cx = m.dmPelsWidth;
out_res->cy = lowest.cy = m.dmPelsHeight;
}
}
memset(&m, 0, sizeof(DEVMODE));
m.dmSize = sizeof(DEVMODE);
i++;
}
return result;
}
void util_toggle_maximize()
{
RECT client_rc;
RECT dst_rc;
LONG style = real_GetWindowLongA(g_ddraw.hwnd, GWL_STYLE);
LONG exstyle = real_GetWindowLongA(g_ddraw.hwnd, GWL_EXSTYLE);
BOOL got_menu = GetMenu(g_ddraw.hwnd) != NULL;
if (real_GetClientRect(g_ddraw.hwnd, &client_rc) && SystemParametersInfo(SPI_GETWORKAREA, 0, &dst_rc, 0))
{
int width = (dst_rc.right - dst_rc.left);
int height = (dst_rc.bottom - dst_rc.top);
int x = dst_rc.left;
int y = dst_rc.top;
if (client_rc.right != g_ddraw.width || client_rc.bottom != g_ddraw.height)
{
dst_rc.left = 0;
dst_rc.top = 0;
dst_rc.right = g_ddraw.width;
dst_rc.bottom = g_ddraw.height;
AdjustWindowRectEx(&dst_rc, style, got_menu, exstyle);
}
else if (g_config.boxing)
{
dst_rc.left = 0;
dst_rc.top = 0;
dst_rc.right = g_ddraw.width;
dst_rc.bottom = g_ddraw.height;
for (int i = 20; i-- > 1;)
{
if (width >= g_ddraw.width * i && height - 20 >= g_ddraw.height * i)
{
dst_rc.right = g_ddraw.width * i;
dst_rc.bottom = g_ddraw.height * i;
break;
}
}
AdjustWindowRectEx(&dst_rc, style, got_menu, exstyle);
}
else if (g_config.maintas)
{
util_unadjust_window_rect(&dst_rc, style, got_menu, exstyle);
int w = dst_rc.right - dst_rc.left;
int h = dst_rc.bottom - dst_rc.top;
double dst_ar = (double)g_ddraw.height / g_ddraw.width;
double src_ar = (double)h / w;
dst_rc.top = 0;
dst_rc.left = 0;
dst_rc.right = w;
dst_rc.bottom = (LONG)round(dst_ar * w);
if (src_ar < dst_ar)
{
dst_rc.right = (LONG)round(((double)dst_rc.right / dst_rc.bottom) * h);
dst_rc.bottom = h;
}
AdjustWindowRectEx(&dst_rc, style, got_menu, exstyle);
}
RECT pos_rc;
pos_rc.left = (width / 2) - ((dst_rc.right - dst_rc.left) / 2) + x;
pos_rc.top = (height / 2) - ((dst_rc.bottom - dst_rc.top) / 2) + y;
pos_rc.right = (dst_rc.right - dst_rc.left);
pos_rc.bottom = (dst_rc.bottom - dst_rc.top);
util_unadjust_window_rect(&pos_rc, style, got_menu, exstyle);
util_unadjust_window_rect(&dst_rc, style, got_menu, exstyle);
util_set_window_rect(
pos_rc.left,
pos_rc.top,
dst_rc.right - dst_rc.left,
dst_rc.bottom - dst_rc.top,
0);
}
}
void util_toggle_fullscreen()
{
/* Disable ALT+ENTER on battle.net and Infantry Online Zone List Window */
if (g_ddraw.bnet_active || (g_config.infantryhack && GetMenu(g_ddraw.hwnd)))
return;
if (g_config.toggle_borderless && g_config.windowed)
{
if (!g_config.fullscreen)
{
mouse_unlock();
g_config.upscaled_state = g_config.fullscreen = TRUE;
dd_SetDisplayMode(0, 0, 0, 0);
mouse_lock();
}
else
{
mouse_unlock();
g_config.upscaled_state = g_config.fullscreen = FALSE;
dd_SetDisplayMode(0, 0, 0, 0);
//mouse_lock();
}
}
else
{
if (g_config.windowed)
{
mouse_unlock();
if (g_config.toggle_upscaled)
{
g_config.upscaled_state = g_config.fullscreen = TRUE;
}
g_config.window_state = g_config.windowed = FALSE;
dd_SetDisplayMode(0, 0, 0, SDM_LEAVE_WINDOWED);
util_update_bnet_pos(0, 0);
mouse_lock();
}
else
{
mouse_unlock();
if (g_config.toggle_upscaled)
{
g_config.upscaled_state = g_config.fullscreen = FALSE;
}
g_config.window_state = g_config.windowed = TRUE;
if (g_ddraw.renderer == d3d9_render_main && !g_config.nonexclusive)
{
d3d9_reset(g_config.windowed);
}
else
{
if (g_ddraw.render.thread)
{
EnterCriticalSection(&g_ddraw.cs);
g_ddraw.render.run = FALSE;
ReleaseSemaphore(g_ddraw.render.sem, 1, NULL);
LeaveCriticalSection(&g_ddraw.cs);
WaitForSingleObject(g_ddraw.render.thread, INFINITE);
g_ddraw.render.thread = NULL;
}
ChangeDisplaySettings(NULL, g_ddraw.bnet_active ? CDS_FULLSCREEN : 0);
}
dd_SetDisplayMode(0, 0, 0, SDM_LEAVE_FULLSCREEN);
//mouse_lock();
}
}
}
BOOL util_unadjust_window_rect(LPRECT prc, DWORD dwStyle, BOOL fMenu, DWORD dwExStyle)
{
RECT rc;
SetRectEmpty(&rc);
BOOL fRc = AdjustWindowRectEx(&rc, dwStyle, fMenu, dwExStyle);
if (fRc)
{
prc->left -= rc.left;
prc->top -= rc.top;
prc->right -= rc.right;
prc->bottom -= rc.bottom;
}
return fRc;
}
void util_set_window_rect(int x, int y, int width, int height, UINT flags)
{
if (g_config.windowed)
{
if (g_ddraw.render.thread)
{
EnterCriticalSection(&g_ddraw.cs);
g_ddraw.render.run = FALSE;
ReleaseSemaphore(g_ddraw.render.sem, 1, NULL);
LeaveCriticalSection(&g_ddraw.cs);
WaitForSingleObject(g_ddraw.render.thread, INFINITE);
g_ddraw.render.thread = NULL;
}
if ((flags & SWP_NOMOVE) == 0)
{
g_config.window_rect.left = x;
g_config.window_rect.top = y;
}
if ((flags & SWP_NOSIZE) == 0)
{
g_config.window_rect.bottom = height;
g_config.window_rect.right = width;
}
dd_SetDisplayMode(0, 0, 0, 0);
}
}
BOOL CALLBACK util_enum_thread_wnd_proc(HWND hwnd, LPARAM lParam)
{
RECT size = { 0 };
real_GetClientRect(hwnd, &size);
LONG sytle = real_GetWindowLongA(hwnd, GWL_STYLE);
if (!g_ddraw.hwnd && !(sytle & WS_DISABLED) && size.right > 0 && size.bottom > 0)
g_ddraw.hwnd = hwnd;
#ifdef _DEBUG
char class[MAX_PATH] = { 0 };
GetClassNameA(hwnd, class, sizeof(class) - 1);
char title[MAX_PATH] = { 0 };
GetWindowTextA(hwnd, title, sizeof(title) - 1);
RECT pos = { 0 };
real_GetWindowRect(hwnd, &pos);
LONG exsytle = real_GetWindowLongA(hwnd, GWL_EXSTYLE);
TRACE(
"%s(class=%s, title=%s, X=%d, Y=%d, nWidth=%d, nHeight=%d)\n",
__FUNCTION__, class, title, pos.left, pos.top, size.right, size.bottom);
dbg_dump_wnd_styles(sytle, exsytle);
#endif
return TRUE;
}
BOOL CALLBACK util_enum_child_proc(HWND hwnd, LPARAM lparam)
{
IDirectDrawSurfaceImpl* this = (IDirectDrawSurfaceImpl*)lparam;
RECT size;
RECT pos;
if (real_GetClientRect(hwnd, &size) &&
real_GetWindowRect(hwnd, &pos) &&
size.right > 1 &&
size.bottom > 1)
{
char class_name[MAX_PATH] = { 0 };
GetClassNameA(hwnd, class_name, sizeof(class_name) - 1);
LONG style = real_GetWindowLongA(hwnd, GWL_STYLE);
LONG exstyle = real_GetWindowLongA(hwnd, GWL_EXSTYLE);
#ifdef _DEBUG_X
HWND parent = GetParent(hwnd);
TRACE("util_enum_child_proc class=%s, hwnd=%p, width=%u, height=%u, left=%d, top=%d, parent=%p\n",
class_name, hwnd, size.right, size.bottom, pos.left, pos.top, parent);
dbg_dump_wnd_styles(style, exstyle);
if (parent != g_ddraw.hwnd)
return TRUE;
#endif
/* Atrox */
if (style == 0x500100C4 && strcmp(class_name, "Edit") == 0)
return TRUE;
if (g_config.fixchilds == FIX_CHILDS_DETECT_HIDE ||
strcmp(class_name, "VideoRenderer") == 0 ||
strcmp(class_name, "MCIQTZ_Window") == 0 ||
strcmp(class_name, "MCIAVI") == 0 ||
strcmp(class_name, "AVIWnd32") == 0 ||
strcmp(class_name, "MCIWndClass") == 0 ||
strcmp(class_name, "AVI Window") == 0)
{
if (g_config.fixchilds != FIX_CHILDS_DETECT_HIDE)
{
InterlockedExchangePointer((void*)&g_ddraw.video_window_hwnd, hwnd);
}
if (!(exstyle & WS_EX_TRANSPARENT))
{
real_SetWindowLongA(hwnd, GWL_EXSTYLE, exstyle | WS_EX_TRANSPARENT);
real_SetWindowPos(
hwnd,
0,
0,
0,
0,
0,
SWP_ASYNCWINDOWPOS | SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOOWNERZORDER
);
}
}
else if (!(exstyle & WS_EX_TRANSPARENT))
{
g_ddraw.got_child_windows = g_ddraw.child_window_exists = TRUE;
if (g_config.fixchilds == FIX_CHILDS_DETECT_PAINT)
{
HDC dst_dc = GetDC(hwnd);
HDC src_dc;
dds_GetDC(this, &src_dc);
real_MapWindowPoints(HWND_DESKTOP, g_ddraw.hwnd, (LPPOINT)&pos, 2);
real_BitBlt(dst_dc, 0, 0, size.right, size.bottom, src_dc, pos.left, pos.top, SRCCOPY);
ReleaseDC(hwnd, dst_dc);
}
}
}
#ifdef _DEBUG_X
return TRUE;
#else
return FALSE;
#endif
}
static unsigned char util_get_pixel(int x, int y)
{
return ((unsigned char*)dds_GetBuffer(
g_ddraw.primary))[y * g_ddraw.primary->pitch + x * g_ddraw.primary->bytes_pp];
}
BOOL util_detect_low_res_screen()
{
/* struct Copied from wkReSolution */
typedef struct
{
PVOID UnkTable1;
DWORD Unk1, Unk2, Unk3, Unk4;
PVOID UnkDD, UnkTable2;
DWORD Unk5;
DWORD RenderWidth, RenderHeight;
DWORD Unk6, Unk7;
DWORD WidthRT, HeightRT;
DWORD HalfWidth, HalfHeight;
DWORD Unk8;
PCHAR UnkC;
LPDIRECTDRAW lpDD;
} * LPW2DDSTRUCT;
static int* in_movie = (int*)0x00665F58;
static int* is_vqa_640 = (int*)0x0065D7BC;
static BYTE* should_stretch = (BYTE*)0x00607D78;
static LPW2DDSTRUCT* pW2DS;
if (!pW2DS)
pW2DS = (LPW2DDSTRUCT*)((DWORD)GetModuleHandleA(NULL) + 0x799C4);
if (g_ddraw.width <= g_ddraw.upscale_hack_width || g_ddraw.height <= g_ddraw.upscale_hack_height)
{
return FALSE;
}
if (g_ddraw.isredalert)
{
if ((*in_movie && !*is_vqa_640) || *should_stretch)
{
return TRUE;
}
return FALSE;
}
else if (g_ddraw.iscnc1)
{
return
util_get_pixel(g_ddraw.upscale_hack_width + 1, 0) == 0 ||
util_get_pixel(g_ddraw.upscale_hack_width + 5, 1) == 0;
}
else if (g_ddraw.iskkndx)
{
return util_get_pixel(g_ddraw.width - 3, 3) == 0;
}
else if (g_ddraw.isworms2)
{
DWORD w2_width = *pW2DS ? (*pW2DS)->RenderWidth : 0;
DWORD w2_height = *pW2DS ? (*pW2DS)->RenderHeight : 0;
if (w2_width && w2_width < g_ddraw.width && w2_height && w2_height < g_ddraw.height)
{
if (g_ddraw.upscale_hack_width != w2_width || g_ddraw.upscale_hack_height != w2_height)
{
g_ddraw.upscale_hack_width = w2_width;
g_ddraw.upscale_hack_height = w2_height;
InterlockedExchange(&g_ddraw.upscale_hack_active, FALSE);
}
return TRUE;
}
}
return FALSE;
}