forked from Dniel97/segatools
Merge branch 'gfx' of https://dev.s-ul.net/Felix/segatools
This commit is contained in:
commit
194ee1a1ce
@ -12,8 +12,9 @@
|
|||||||
|
|
||||||
#include "chunihook/config.h"
|
#include "chunihook/config.h"
|
||||||
|
|
||||||
|
#include "gfxhook/config.h"
|
||||||
|
|
||||||
#include "hooklib/config.h"
|
#include "hooklib/config.h"
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
|
|
||||||
#include "platform/config.h"
|
#include "platform/config.h"
|
||||||
#include "platform/platform.h"
|
#include "platform/platform.h"
|
||||||
|
@ -10,7 +10,7 @@
|
|||||||
#include "chunihook/chuni-dll.h"
|
#include "chunihook/chuni-dll.h"
|
||||||
#include "chunihook/slider.h"
|
#include "chunihook/slider.h"
|
||||||
|
|
||||||
#include "hooklib/gfx.h"
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
#include "platform/platform.h"
|
#include "platform/platform.h"
|
||||||
|
|
||||||
|
@ -12,9 +12,11 @@
|
|||||||
|
|
||||||
#include "chuniio/chuniio.h"
|
#include "chuniio/chuniio.h"
|
||||||
|
|
||||||
|
#include "gfxhook/d3d9.h"
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
#include "hook/process.h"
|
#include "hook/process.h"
|
||||||
|
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
#include "hooklib/serial.h"
|
#include "hooklib/serial.h"
|
||||||
#include "hooklib/spike.h"
|
#include "hooklib/spike.h"
|
||||||
|
|
||||||
@ -60,7 +62,8 @@ static DWORD CALLBACK chuni_pre_startup(void)
|
|||||||
|
|
||||||
/* Hook Win32 APIs */
|
/* Hook Win32 APIs */
|
||||||
|
|
||||||
gfx_hook_init(&chuni_hook_cfg.gfx, chuni_hook_mod);
|
gfx_hook_init(&chuni_hook_cfg.gfx);
|
||||||
|
gfx_d3d9_hook_init(&chuni_hook_cfg.gfx, chuni_hook_mod);
|
||||||
serial_hook_init();
|
serial_hook_init();
|
||||||
|
|
||||||
/* Initialize emulation hooks */
|
/* Initialize emulation hooks */
|
||||||
|
@ -14,6 +14,7 @@ shared_library(
|
|||||||
amex_lib,
|
amex_lib,
|
||||||
board_lib,
|
board_lib,
|
||||||
chuniio_lib,
|
chuniio_lib,
|
||||||
|
gfxhook_lib,
|
||||||
hooklib_lib,
|
hooklib_lib,
|
||||||
jvs_lib,
|
jvs_lib,
|
||||||
platform_lib,
|
platform_lib,
|
||||||
|
@ -13,7 +13,6 @@
|
|||||||
|
|
||||||
#include "hook/process.h"
|
#include "hook/process.h"
|
||||||
|
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
#include "hooklib/serial.h"
|
#include "hooklib/serial.h"
|
||||||
#include "hooklib/spike.h"
|
#include "hooklib/spike.h"
|
||||||
|
|
||||||
|
17
gfxhook/config.c
Normal file
17
gfxhook/config.c
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
|
#include "gfxhook/config.h"
|
||||||
|
|
||||||
|
void gfx_config_load(struct gfx_config *cfg, const wchar_t *filename)
|
||||||
|
{
|
||||||
|
assert(cfg != NULL);
|
||||||
|
assert(filename != NULL);
|
||||||
|
|
||||||
|
cfg->enable = GetPrivateProfileIntW(L"gfx", L"enable", 1, filename);
|
||||||
|
cfg->windowed = GetPrivateProfileIntW(L"gfx", L"windowed", 0, filename);
|
||||||
|
cfg->framed = GetPrivateProfileIntW(L"gfx", L"framed", 1, filename);
|
||||||
|
cfg->monitor = GetPrivateProfileIntW(L"gfx", L"monitor", 0, filename);
|
||||||
|
}
|
7
gfxhook/config.h
Normal file
7
gfxhook/config.h
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
|
void gfx_config_load(struct gfx_config *cfg, const wchar_t *filename);
|
195
gfxhook/d3d11.c
Normal file
195
gfxhook/d3d11.c
Normal file
@ -0,0 +1,195 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
#include <dxgi.h>
|
||||||
|
#include <d3d11.h>
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "gfxhook/d3d11.h"
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
#include "gfxhook/util.h"
|
||||||
|
|
||||||
|
#include "hook/table.h"
|
||||||
|
|
||||||
|
#include "hooklib/dll.h"
|
||||||
|
|
||||||
|
#include "util/dprintf.h"
|
||||||
|
|
||||||
|
typedef HRESULT (WINAPI *D3D11CreateDevice_t)(
|
||||||
|
IDXGIAdapter *pAdapter,
|
||||||
|
D3D_DRIVER_TYPE DriverType,
|
||||||
|
HMODULE Software,
|
||||||
|
UINT Flags,
|
||||||
|
const D3D_FEATURE_LEVEL *ppFeatureLevels,
|
||||||
|
UINT FeatureLevels,
|
||||||
|
UINT SDKVersion,
|
||||||
|
ID3D11Device **ppDevice,
|
||||||
|
D3D_FEATURE_LEVEL *pFeatureLevel,
|
||||||
|
ID3D11DeviceContext **ppImmediateContext);
|
||||||
|
typedef HRESULT (WINAPI *D3D11CreateDeviceAndSwapChain_t)(
|
||||||
|
IDXGIAdapter *pAdapter,
|
||||||
|
D3D_DRIVER_TYPE DriverType,
|
||||||
|
HMODULE Software,
|
||||||
|
UINT Flags,
|
||||||
|
const D3D_FEATURE_LEVEL *ppFeatureLevels,
|
||||||
|
UINT FeatureLevels,
|
||||||
|
UINT SDKVersion,
|
||||||
|
const DXGI_SWAP_CHAIN_DESC *pSwapChainDesc,
|
||||||
|
IDXGISwapChain **ppSwapChain,
|
||||||
|
ID3D11Device **ppDevice,
|
||||||
|
D3D_FEATURE_LEVEL *pFeatureLevel,
|
||||||
|
ID3D11DeviceContext **ppImmediateContext);
|
||||||
|
|
||||||
|
static struct gfx_config gfx_config;
|
||||||
|
static D3D11CreateDevice_t next_D3D11CreateDevice;
|
||||||
|
static D3D11CreateDeviceAndSwapChain_t next_D3D11CreateDeviceAndSwapChain;
|
||||||
|
|
||||||
|
static const struct hook_symbol d3d11_hooks[] = {
|
||||||
|
{
|
||||||
|
.name = "D3D11CreateDevice",
|
||||||
|
.patch = D3D11CreateDevice,
|
||||||
|
.link = (void **) &next_D3D11CreateDevice,
|
||||||
|
}, {
|
||||||
|
.name = "D3D11CreateDeviceAndSwapChain",
|
||||||
|
.patch = D3D11CreateDeviceAndSwapChain,
|
||||||
|
.link = (void **) &next_D3D11CreateDeviceAndSwapChain,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
void gfx_d3d11_hook_init(const struct gfx_config *cfg, HINSTANCE self)
|
||||||
|
{
|
||||||
|
HMODULE d3d11;
|
||||||
|
|
||||||
|
assert(cfg != NULL);
|
||||||
|
|
||||||
|
if (!cfg->enable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(&gfx_config, cfg, sizeof(*cfg));
|
||||||
|
hook_table_apply(NULL, "d3d11.dll", d3d11_hooks, _countof(d3d11_hooks));
|
||||||
|
|
||||||
|
if (next_D3D11CreateDevice == NULL || next_D3D11CreateDeviceAndSwapChain == NULL) {
|
||||||
|
d3d11 = LoadLibraryW(L"d3d11.dll");
|
||||||
|
|
||||||
|
if (d3d11 == NULL) {
|
||||||
|
dprintf("D3D11: d3d11.dll not found or failed initialization\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next_D3D11CreateDevice == NULL) {
|
||||||
|
next_D3D11CreateDevice = (D3D11CreateDevice_t) GetProcAddress(
|
||||||
|
d3d11,
|
||||||
|
"D3D11CreateDevice");
|
||||||
|
}
|
||||||
|
if (next_D3D11CreateDeviceAndSwapChain == NULL) {
|
||||||
|
next_D3D11CreateDeviceAndSwapChain = (D3D11CreateDeviceAndSwapChain_t) GetProcAddress(
|
||||||
|
d3d11,
|
||||||
|
"D3D11CreateDeviceAndSwapChain");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next_D3D11CreateDevice == NULL) {
|
||||||
|
dprintf("D3D11: D3D11CreateDevice not found in loaded d3d11.dll\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
if (next_D3D11CreateDeviceAndSwapChain == NULL) {
|
||||||
|
dprintf("D3D11: D3D11CreateDeviceAndSwapChain not found in loaded d3d11.dll\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self != NULL) {
|
||||||
|
dll_hook_push(self, L"d3d11.dll");
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (d3d11 != NULL) {
|
||||||
|
FreeLibrary(d3d11);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WINAPI D3D11CreateDevice(
|
||||||
|
IDXGIAdapter *pAdapter,
|
||||||
|
D3D_DRIVER_TYPE DriverType,
|
||||||
|
HMODULE Software,
|
||||||
|
UINT Flags,
|
||||||
|
const D3D_FEATURE_LEVEL *ppFeatureLevels,
|
||||||
|
UINT FeatureLevels,
|
||||||
|
UINT SDKVersion,
|
||||||
|
ID3D11Device **ppDevice,
|
||||||
|
D3D_FEATURE_LEVEL *pFeatureLevel,
|
||||||
|
ID3D11DeviceContext **ppImmediateContext)
|
||||||
|
{
|
||||||
|
dprintf("D3D11: D3D11CreateDevice hook hit\n");
|
||||||
|
|
||||||
|
return next_D3D11CreateDevice(
|
||||||
|
pAdapter,
|
||||||
|
DriverType,
|
||||||
|
Software,
|
||||||
|
Flags,
|
||||||
|
ppFeatureLevels,
|
||||||
|
FeatureLevels,
|
||||||
|
SDKVersion,
|
||||||
|
ppDevice,
|
||||||
|
pFeatureLevel,
|
||||||
|
ppImmediateContext);
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WINAPI D3D11CreateDeviceAndSwapChain(
|
||||||
|
IDXGIAdapter *pAdapter,
|
||||||
|
D3D_DRIVER_TYPE DriverType,
|
||||||
|
HMODULE Software,
|
||||||
|
UINT Flags,
|
||||||
|
const D3D_FEATURE_LEVEL *ppFeatureLevels,
|
||||||
|
UINT FeatureLevels,
|
||||||
|
UINT SDKVersion,
|
||||||
|
const DXGI_SWAP_CHAIN_DESC *pSwapChainDesc,
|
||||||
|
IDXGISwapChain **ppSwapChain,
|
||||||
|
ID3D11Device **ppDevice,
|
||||||
|
D3D_FEATURE_LEVEL *pFeatureLevel,
|
||||||
|
ID3D11DeviceContext **ppImmediateContext)
|
||||||
|
{
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc;
|
||||||
|
HWND hwnd;
|
||||||
|
UINT width;
|
||||||
|
UINT height;
|
||||||
|
|
||||||
|
dprintf("D3D11: D3D11CreateDeviceAndSwapChain hook hit\n");
|
||||||
|
|
||||||
|
desc = (DXGI_SWAP_CHAIN_DESC *) pSwapChainDesc;
|
||||||
|
|
||||||
|
if (desc != NULL) {
|
||||||
|
desc->Windowed = gfx_config.windowed;
|
||||||
|
|
||||||
|
hwnd = desc->OutputWindow;
|
||||||
|
width = desc->BufferDesc.Width;
|
||||||
|
height = desc->BufferDesc.Height;
|
||||||
|
} else {
|
||||||
|
hwnd = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hwnd != NULL) {
|
||||||
|
gfx_util_ensure_win_visible(hwnd);
|
||||||
|
gfx_util_borderless_fullscreen_windowed(hwnd, width, height);
|
||||||
|
}
|
||||||
|
|
||||||
|
return next_D3D11CreateDeviceAndSwapChain(
|
||||||
|
pAdapter,
|
||||||
|
DriverType,
|
||||||
|
Software,
|
||||||
|
Flags,
|
||||||
|
ppFeatureLevels,
|
||||||
|
FeatureLevels,
|
||||||
|
SDKVersion,
|
||||||
|
pSwapChainDesc,
|
||||||
|
ppSwapChain,
|
||||||
|
ppDevice,
|
||||||
|
pFeatureLevel,
|
||||||
|
ppImmediateContext);
|
||||||
|
}
|
||||||
|
|
7
gfxhook/d3d11.h
Normal file
7
gfxhook/d3d11.h
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
|
void gfx_d3d11_hook_init(const struct gfx_config *cfg, HINSTANCE self);
|
251
gfxhook/d3d9.c
Normal file
251
gfxhook/d3d9.c
Normal file
@ -0,0 +1,251 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
#include <d3d9.h>
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "hook/com-proxy.h"
|
||||||
|
#include "hook/table.h"
|
||||||
|
|
||||||
|
#include "hooklib/dll.h"
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
#include "gfxhook/util.h"
|
||||||
|
|
||||||
|
#include "util/dprintf.h"
|
||||||
|
|
||||||
|
typedef IDirect3D9 * (WINAPI *Direct3DCreate9_t)(UINT sdk_ver);
|
||||||
|
typedef HRESULT (WINAPI *Direct3DCreate9Ex_t)(UINT sdk_ver, IDirect3D9Ex **d3d9ex);
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDirect3D9_CreateDevice(
|
||||||
|
IDirect3D9 *self,
|
||||||
|
UINT adapter,
|
||||||
|
D3DDEVTYPE type,
|
||||||
|
HWND hwnd,
|
||||||
|
DWORD flags,
|
||||||
|
D3DPRESENT_PARAMETERS *pp,
|
||||||
|
IDirect3DDevice9 **pdev);
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDirect3D9Ex_CreateDevice(
|
||||||
|
IDirect3D9Ex *self,
|
||||||
|
UINT adapter,
|
||||||
|
D3DDEVTYPE type,
|
||||||
|
HWND hwnd,
|
||||||
|
DWORD flags,
|
||||||
|
D3DPRESENT_PARAMETERS *pp,
|
||||||
|
IDirect3DDevice9 **pdev);
|
||||||
|
|
||||||
|
static struct gfx_config gfx_config;
|
||||||
|
static Direct3DCreate9_t next_Direct3DCreate9;
|
||||||
|
static Direct3DCreate9Ex_t next_Direct3DCreate9Ex;
|
||||||
|
|
||||||
|
static const struct hook_symbol gfx_hooks[] = {
|
||||||
|
{
|
||||||
|
.name = "Direct3DCreate9",
|
||||||
|
.patch = Direct3DCreate9,
|
||||||
|
.link = (void **) &next_Direct3DCreate9,
|
||||||
|
}, {
|
||||||
|
.name = "Direct3DCreate9Ex",
|
||||||
|
.patch = Direct3DCreate9Ex,
|
||||||
|
.link = (void **) &next_Direct3DCreate9Ex,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
void gfx_d3d9_hook_init(const struct gfx_config *cfg, HINSTANCE self)
|
||||||
|
{
|
||||||
|
HMODULE d3d9;
|
||||||
|
|
||||||
|
assert(cfg != NULL);
|
||||||
|
|
||||||
|
if (!cfg->enable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(&gfx_config, cfg, sizeof(*cfg));
|
||||||
|
hook_table_apply(NULL, "d3d9.dll", gfx_hooks, _countof(gfx_hooks));
|
||||||
|
|
||||||
|
if (next_Direct3DCreate9 == NULL || next_Direct3DCreate9Ex == NULL) {
|
||||||
|
d3d9 = LoadLibraryW(L"d3d9.dll");
|
||||||
|
|
||||||
|
if (d3d9 == NULL) {
|
||||||
|
dprintf("Gfx: d3d9.dll not found or failed initialization\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next_Direct3DCreate9 == NULL) {
|
||||||
|
next_Direct3DCreate9 = (Direct3DCreate9_t) GetProcAddress(d3d9, "Direct3DCreate9");
|
||||||
|
}
|
||||||
|
if (next_Direct3DCreate9Ex == NULL) {
|
||||||
|
next_Direct3DCreate9Ex = (Direct3DCreate9Ex_t) GetProcAddress(d3d9, "Direct3DCreate9Ex");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next_Direct3DCreate9 == NULL) {
|
||||||
|
dprintf("Gfx: Direct3DCreate9 not found in loaded d3d9.dll\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
if (next_Direct3DCreate9Ex == NULL) {
|
||||||
|
dprintf("Gfx: Direct3DCreate9Ex not found in loaded d3d9.dll\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self != NULL) {
|
||||||
|
dll_hook_push(self, L"d3d9.dll");
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (d3d9 != NULL) {
|
||||||
|
FreeLibrary(d3d9);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
IDirect3D9 * WINAPI Direct3DCreate9(UINT sdk_ver)
|
||||||
|
{
|
||||||
|
struct com_proxy *proxy;
|
||||||
|
IDirect3D9Vtbl *vtbl;
|
||||||
|
IDirect3D9 *api;
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
dprintf("Gfx: Direct3DCreate9 hook hit\n");
|
||||||
|
|
||||||
|
api = NULL;
|
||||||
|
|
||||||
|
if (next_Direct3DCreate9 == NULL) {
|
||||||
|
dprintf("Gfx: next_Direct3DCreate9 == NULL\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
api = next_Direct3DCreate9(sdk_ver);
|
||||||
|
|
||||||
|
if (api == NULL) {
|
||||||
|
dprintf("Gfx: next_Direct3DCreate9 returned NULL\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr = com_proxy_wrap(&proxy, api, sizeof(*api->lpVtbl));
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("Gfx: com_proxy_wrap returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
vtbl = proxy->vptr;
|
||||||
|
vtbl->CreateDevice = my_IDirect3D9_CreateDevice;
|
||||||
|
|
||||||
|
return (IDirect3D9 *) proxy;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (api != NULL) {
|
||||||
|
IDirect3D9_Release(api);
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WINAPI Direct3DCreate9Ex(UINT sdk_ver, IDirect3D9Ex **d3d9ex)
|
||||||
|
{
|
||||||
|
struct com_proxy *proxy;
|
||||||
|
IDirect3D9ExVtbl *vtbl;
|
||||||
|
IDirect3D9Ex *api;
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
dprintf("Gfx: Direct3DCreate9Ex hook hit\n");
|
||||||
|
|
||||||
|
api = NULL;
|
||||||
|
|
||||||
|
if (next_Direct3DCreate9Ex == NULL) {
|
||||||
|
dprintf("Gfx: next_Direct3DCreate9Ex == NULL\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr = next_Direct3DCreate9Ex(sdk_ver, d3d9ex);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("Gfx: next_Direct3DCreate9Ex returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
api = *d3d9ex;
|
||||||
|
hr = com_proxy_wrap(&proxy, api, sizeof(*api->lpVtbl));
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("Gfx: com_proxy_wrap returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
vtbl = proxy->vptr;
|
||||||
|
vtbl->CreateDevice = my_IDirect3D9Ex_CreateDevice;
|
||||||
|
|
||||||
|
*d3d9ex = (IDirect3D9Ex *) proxy;
|
||||||
|
|
||||||
|
return S_OK;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (api != NULL) {
|
||||||
|
IDirect3D9Ex_Release(api);
|
||||||
|
}
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDirect3D9_CreateDevice(
|
||||||
|
IDirect3D9 *self,
|
||||||
|
UINT adapter,
|
||||||
|
D3DDEVTYPE type,
|
||||||
|
HWND hwnd,
|
||||||
|
DWORD flags,
|
||||||
|
D3DPRESENT_PARAMETERS *pp,
|
||||||
|
IDirect3DDevice9 **pdev)
|
||||||
|
{
|
||||||
|
struct com_proxy *proxy;
|
||||||
|
IDirect3D9 *real;
|
||||||
|
|
||||||
|
dprintf("Gfx: IDirect3D9::CreateDevice hook hit\n");
|
||||||
|
|
||||||
|
proxy = com_proxy_downcast(self);
|
||||||
|
real = proxy->real;
|
||||||
|
|
||||||
|
if (gfx_config.windowed) {
|
||||||
|
pp->Windowed = TRUE;
|
||||||
|
pp->FullScreen_RefreshRateInHz = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (gfx_config.framed) {
|
||||||
|
gfx_util_frame_window(hwnd);
|
||||||
|
}
|
||||||
|
|
||||||
|
dprintf("Gfx: Using adapter %d\n", gfx_config.monitor);
|
||||||
|
|
||||||
|
return IDirect3D9_CreateDevice(real, gfx_config.monitor, type, hwnd, flags, pp, pdev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDirect3D9Ex_CreateDevice(
|
||||||
|
IDirect3D9Ex *self,
|
||||||
|
UINT adapter,
|
||||||
|
D3DDEVTYPE type,
|
||||||
|
HWND hwnd,
|
||||||
|
DWORD flags,
|
||||||
|
D3DPRESENT_PARAMETERS *pp,
|
||||||
|
IDirect3DDevice9 **pdev)
|
||||||
|
{
|
||||||
|
dprintf("Gfx: IDirect3D9Ex::CreateDevice hook forwarding to my_IDirect3D9_CreateDevice\n");
|
||||||
|
|
||||||
|
return my_IDirect3D9_CreateDevice(
|
||||||
|
(IDirect3D9 *) self,
|
||||||
|
adapter,
|
||||||
|
type,
|
||||||
|
hwnd,
|
||||||
|
flags,
|
||||||
|
pp,
|
||||||
|
pdev);
|
||||||
|
}
|
7
gfxhook/d3d9.h
Normal file
7
gfxhook/d3d9.h
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
|
void gfx_d3d9_hook_init(const struct gfx_config *cfg, HINSTANCE self);
|
364
gfxhook/dxgi.c
Normal file
364
gfxhook/dxgi.c
Normal file
@ -0,0 +1,364 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
#include <dxgi.h>
|
||||||
|
#include <dxgi1_3.h>
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "gfxhook/dxgi.h"
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
|
#include "hook/com-proxy.h"
|
||||||
|
#include "hook/table.h"
|
||||||
|
|
||||||
|
#include "hooklib/dll.h"
|
||||||
|
|
||||||
|
#include "util/dprintf.h"
|
||||||
|
|
||||||
|
typedef HRESULT (WINAPI *CreateDXGIFactory_t)(REFIID riid, void **factory);
|
||||||
|
typedef HRESULT (WINAPI *CreateDXGIFactory1_t)(REFIID riid, void **factory);
|
||||||
|
typedef HRESULT (WINAPI *CreateDXGIFactory2_t)(
|
||||||
|
UINT flags,
|
||||||
|
REFIID riid,
|
||||||
|
void **factory);
|
||||||
|
|
||||||
|
static HRESULT hook_factory(REFIID riid, void **factory);
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDXGIFactory_CreateSwapChain(
|
||||||
|
IDXGIFactory *self,
|
||||||
|
IUnknown *device,
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc,
|
||||||
|
IDXGISwapChain **swapchain);
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDXGIFactory1_CreateSwapChain(
|
||||||
|
IDXGIFactory1 *self,
|
||||||
|
IUnknown *device,
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc,
|
||||||
|
IDXGISwapChain **swapchain);
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDXGIFactory2_CreateSwapChain(
|
||||||
|
IDXGIFactory2 *self,
|
||||||
|
IUnknown *device,
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc,
|
||||||
|
IDXGISwapChain **swapchain);
|
||||||
|
|
||||||
|
static struct gfx_config gfx_config;
|
||||||
|
static CreateDXGIFactory_t next_CreateDXGIFactory;
|
||||||
|
static CreateDXGIFactory1_t next_CreateDXGIFactory1;
|
||||||
|
static CreateDXGIFactory2_t next_CreateDXGIFactory2;
|
||||||
|
|
||||||
|
static const struct hook_symbol dxgi_hooks[] = {
|
||||||
|
{
|
||||||
|
.name = "CreateDXGIFactory",
|
||||||
|
.patch = CreateDXGIFactory,
|
||||||
|
.link = (void **) &next_CreateDXGIFactory,
|
||||||
|
}, {
|
||||||
|
.name = "CreateDXGIFactory1",
|
||||||
|
.patch = CreateDXGIFactory1,
|
||||||
|
.link = (void **) &next_CreateDXGIFactory1,
|
||||||
|
}, {
|
||||||
|
.name = "CreateDXGIFactory2",
|
||||||
|
.patch = CreateDXGIFactory2,
|
||||||
|
.link = (void **) &next_CreateDXGIFactory2,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
void gfx_dxgi_hook_init(const struct gfx_config *cfg, HINSTANCE self)
|
||||||
|
{
|
||||||
|
HMODULE dxgi;
|
||||||
|
|
||||||
|
assert(cfg != NULL);
|
||||||
|
|
||||||
|
if (!cfg->enable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(&gfx_config, cfg, sizeof(*cfg));
|
||||||
|
hook_table_apply(NULL, "dxgi.dll", dxgi_hooks, _countof(dxgi_hooks));
|
||||||
|
|
||||||
|
if (next_CreateDXGIFactory == NULL || next_CreateDXGIFactory1 == NULL) {
|
||||||
|
dxgi = LoadLibraryW(L"dxgi.dll");
|
||||||
|
|
||||||
|
if (dxgi == NULL) {
|
||||||
|
dprintf("DXGI: dxgi.dll not found or failed initialization\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next_CreateDXGIFactory == NULL) {
|
||||||
|
next_CreateDXGIFactory = (CreateDXGIFactory_t) GetProcAddress(
|
||||||
|
dxgi,
|
||||||
|
"CreateDXGIFactory");
|
||||||
|
}
|
||||||
|
if (next_CreateDXGIFactory1 == NULL) {
|
||||||
|
next_CreateDXGIFactory1 = (CreateDXGIFactory1_t) GetProcAddress(
|
||||||
|
dxgi,
|
||||||
|
"CreateDXGIFactory1");
|
||||||
|
}
|
||||||
|
if (next_CreateDXGIFactory2 == NULL) {
|
||||||
|
next_CreateDXGIFactory2 = (CreateDXGIFactory2_t) GetProcAddress(
|
||||||
|
dxgi,
|
||||||
|
"CreateDXGIFactory2");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next_CreateDXGIFactory == NULL) {
|
||||||
|
dprintf("DXGI: CreateDXGIFactory not found in loaded dxgi.dll\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
if (next_CreateDXGIFactory1 == NULL) {
|
||||||
|
dprintf("DXGI: CreateDXGIFactory1 not found in loaded dxgi.dll\n");
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* `CreateDXGIFactory2` was introduced in Windows 8.1 and the original
|
||||||
|
* Nu runs Windows 8, so do not require it to exist */
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self != NULL) {
|
||||||
|
dll_hook_push(self, L"dxgi.dll");
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (dxgi != NULL) {
|
||||||
|
FreeLibrary(dxgi);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WINAPI CreateDXGIFactory(REFIID riid, void **factory)
|
||||||
|
{
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
dprintf("DXGI: CreateDXGIFactory hook hit\n");
|
||||||
|
|
||||||
|
hr = next_CreateDXGIFactory(riid, factory);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("DXGI: CreateDXGIFactory returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr = hook_factory(riid, factory);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WINAPI CreateDXGIFactory1(REFIID riid, void **factory)
|
||||||
|
{
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
dprintf("DXGI: CreateDXGIFactory1 hook hit\n");
|
||||||
|
|
||||||
|
hr = next_CreateDXGIFactory1(riid, factory);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("DXGI: CreateDXGIFactory1 returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr = hook_factory(riid, factory);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT WINAPI CreateDXGIFactory2(UINT flags, REFIID riid, void **factory)
|
||||||
|
{
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
dprintf("DXGI: CreateDXGIFactory2 hook hit\n");
|
||||||
|
|
||||||
|
if (next_CreateDXGIFactory2 == NULL) {
|
||||||
|
dprintf("DXGI: CreateDXGIFactory2 not available, forwarding to CreateDXGIFactory1\n");
|
||||||
|
|
||||||
|
return CreateDXGIFactory1(riid, factory);
|
||||||
|
}
|
||||||
|
|
||||||
|
hr = next_CreateDXGIFactory2(flags, riid, factory);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("DXGI: CreateDXGIFactory2 returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr = hook_factory(riid, factory);
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HRESULT hook_factory(REFIID riid, void **factory)
|
||||||
|
{
|
||||||
|
struct com_proxy *proxy;
|
||||||
|
IDXGIFactory *api_0;
|
||||||
|
IDXGIFactory1 *api_1;
|
||||||
|
IDXGIFactory2 *api_2;
|
||||||
|
IDXGIFactoryVtbl *vtbl_0;
|
||||||
|
IDXGIFactory1Vtbl *vtbl_1;
|
||||||
|
IDXGIFactory2Vtbl *vtbl_2;
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
api_0 = NULL;
|
||||||
|
api_1 = NULL;
|
||||||
|
api_2 = NULL;
|
||||||
|
|
||||||
|
if (memcmp(riid, &IID_IDXGIFactory, sizeof(*riid)) == 0) {
|
||||||
|
api_0 = *factory;
|
||||||
|
hr = com_proxy_wrap(&proxy, api_0, sizeof(*api_0->lpVtbl));
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("DXGI: com_proxy_wrap returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
vtbl_0 = proxy->vptr;
|
||||||
|
vtbl_0->CreateSwapChain = my_IDXGIFactory_CreateSwapChain;
|
||||||
|
|
||||||
|
*factory = proxy;
|
||||||
|
} else if (memcmp(riid, &IID_IDXGIFactory1, sizeof(*riid)) == 0) {
|
||||||
|
api_1 = *factory;
|
||||||
|
hr = com_proxy_wrap(&proxy, api_1, sizeof(*api_1->lpVtbl));
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("DXGI: com_proxy_wrap returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
vtbl_1 = proxy->vptr;
|
||||||
|
vtbl_1->CreateSwapChain = my_IDXGIFactory1_CreateSwapChain;
|
||||||
|
|
||||||
|
*factory = proxy;
|
||||||
|
} else if (memcmp(riid, &IID_IDXGIFactory2, sizeof(*riid)) == 0) {
|
||||||
|
api_2 = *factory;
|
||||||
|
hr = com_proxy_wrap(&proxy, api_2, sizeof(*api_2->lpVtbl));
|
||||||
|
|
||||||
|
if (FAILED(hr)) {
|
||||||
|
dprintf("DXGI: com_proxy_wrap returned %x\n", (int) hr);
|
||||||
|
|
||||||
|
goto fail;
|
||||||
|
}
|
||||||
|
|
||||||
|
vtbl_2 = proxy->vptr;
|
||||||
|
vtbl_2->CreateSwapChain = my_IDXGIFactory2_CreateSwapChain;
|
||||||
|
|
||||||
|
*factory = proxy;
|
||||||
|
}
|
||||||
|
|
||||||
|
return S_OK;
|
||||||
|
|
||||||
|
fail:
|
||||||
|
if (api_0 != NULL) {
|
||||||
|
IDXGIFactory_Release(api_0);
|
||||||
|
}
|
||||||
|
if (api_1 != NULL) {
|
||||||
|
IDXGIFactory1_Release(api_1);
|
||||||
|
}
|
||||||
|
if (api_2 != NULL) {
|
||||||
|
IDXGIFactory2_Release(api_2);
|
||||||
|
}
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDXGIFactory_CreateSwapChain(
|
||||||
|
IDXGIFactory *self,
|
||||||
|
IUnknown *device,
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc,
|
||||||
|
IDXGISwapChain **swapchain)
|
||||||
|
{
|
||||||
|
struct com_proxy *proxy;
|
||||||
|
IDXGIFactory *real;
|
||||||
|
HWND hwnd;
|
||||||
|
UINT width;
|
||||||
|
UINT height;
|
||||||
|
|
||||||
|
dprintf("DXGI: IDXGIFactory::CreateSwapChain hook hit\n");
|
||||||
|
|
||||||
|
proxy = com_proxy_downcast(self);
|
||||||
|
real = proxy->real;
|
||||||
|
|
||||||
|
if (desc != NULL) {
|
||||||
|
desc->Windowed = gfx_config.windowed;
|
||||||
|
|
||||||
|
hwnd = desc->OutputWindow;
|
||||||
|
width = desc->BufferDesc.Width;
|
||||||
|
height = desc->BufferDesc.Height;
|
||||||
|
} else {
|
||||||
|
hwnd = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hwnd != NULL) {
|
||||||
|
/*
|
||||||
|
* Ensure window is maximized to avoid a Windows 10 issue where a
|
||||||
|
* fullscreen swap chain is not created because the window is minimized
|
||||||
|
* at the time of creation.
|
||||||
|
*/
|
||||||
|
ShowWindow(hwnd, SW_RESTORE);
|
||||||
|
|
||||||
|
if (!gfx_config.framed && width > 0 && height > 0) {
|
||||||
|
dprintf("DXGI: Resizing window to %ldx%ld\n", width, height);
|
||||||
|
|
||||||
|
SetWindowLongPtrW(hwnd, GWL_STYLE, WS_POPUP);
|
||||||
|
SetWindowLongPtrW(hwnd, GWL_EXSTYLE, WS_EX_TOPMOST);
|
||||||
|
|
||||||
|
SetWindowPos(
|
||||||
|
hwnd,
|
||||||
|
HWND_TOP,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
(int) width,
|
||||||
|
(int) height,
|
||||||
|
SWP_FRAMECHANGED | SWP_NOSENDCHANGING);
|
||||||
|
|
||||||
|
ShowWindow(hwnd, SW_SHOWMAXIMIZED);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return IDXGIFactory_CreateSwapChain(real, device, desc, swapchain);
|
||||||
|
}
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDXGIFactory1_CreateSwapChain(
|
||||||
|
IDXGIFactory1 *self,
|
||||||
|
IUnknown *device,
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc,
|
||||||
|
IDXGISwapChain **swapchain)
|
||||||
|
{
|
||||||
|
dprintf("DXGI: IDXGIFactory1::CreateSwapChain hook forwarding to my_IDXGIFactory_CreateSwapChain\n");
|
||||||
|
|
||||||
|
return my_IDXGIFactory_CreateSwapChain(
|
||||||
|
(IDXGIFactory *) self,
|
||||||
|
device,
|
||||||
|
desc,
|
||||||
|
swapchain);
|
||||||
|
}
|
||||||
|
|
||||||
|
static HRESULT STDMETHODCALLTYPE my_IDXGIFactory2_CreateSwapChain(
|
||||||
|
IDXGIFactory2 *self,
|
||||||
|
IUnknown *device,
|
||||||
|
DXGI_SWAP_CHAIN_DESC *desc,
|
||||||
|
IDXGISwapChain **swapchain)
|
||||||
|
{
|
||||||
|
dprintf("DXGI: IDXGIFactory2::CreateSwapChain hook forwarding to my_IDXGIFactory_CreateSwapChain\n");
|
||||||
|
|
||||||
|
return my_IDXGIFactory_CreateSwapChain(
|
||||||
|
(IDXGIFactory *) self,
|
||||||
|
device,
|
||||||
|
desc,
|
||||||
|
swapchain);
|
||||||
|
}
|
7
gfxhook/dxgi.h
Normal file
7
gfxhook/dxgi.h
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
|
void gfx_dxgi_hook_init(const struct gfx_config *cfg, HINSTANCE self);
|
48
gfxhook/gfx.c
Normal file
48
gfxhook/gfx.c
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include <assert.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
|
#include "hook/table.h"
|
||||||
|
|
||||||
|
#include "util/dprintf.h"
|
||||||
|
|
||||||
|
typedef BOOL (WINAPI *ShowWindow_t)(HWND hWnd, int nCmdShow);
|
||||||
|
|
||||||
|
static BOOL WINAPI hook_ShowWindow(HWND hWnd, int nCmdShow);
|
||||||
|
|
||||||
|
static struct gfx_config gfx_config;
|
||||||
|
static ShowWindow_t next_ShowWindow;
|
||||||
|
|
||||||
|
static const struct hook_symbol gfx_hooks[] = {
|
||||||
|
{
|
||||||
|
.name = "ShowWindow",
|
||||||
|
.patch = hook_ShowWindow,
|
||||||
|
.link = (void **) &next_ShowWindow,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
void gfx_hook_init(const struct gfx_config *cfg)
|
||||||
|
{
|
||||||
|
assert(cfg != NULL);
|
||||||
|
|
||||||
|
if (!cfg->enable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(&gfx_config, cfg, sizeof(*cfg));
|
||||||
|
hook_table_apply(NULL, "user32.dll", gfx_hooks, _countof(gfx_hooks));
|
||||||
|
}
|
||||||
|
|
||||||
|
static BOOL WINAPI hook_ShowWindow(HWND hWnd, int nCmdShow)
|
||||||
|
{
|
||||||
|
dprintf("Gfx: ShowWindow hook hit\n");
|
||||||
|
|
||||||
|
if (!gfx_config.framed && nCmdShow == SW_RESTORE) {
|
||||||
|
nCmdShow = SW_SHOW;
|
||||||
|
}
|
||||||
|
|
||||||
|
return next_ShowWindow(hWnd, nCmdShow);
|
||||||
|
}
|
@ -1,7 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <windows.h>
|
|
||||||
|
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
struct gfx_config {
|
struct gfx_config {
|
||||||
@ -11,4 +9,4 @@ struct gfx_config {
|
|||||||
int monitor;
|
int monitor;
|
||||||
};
|
};
|
||||||
|
|
||||||
void gfx_hook_init(const struct gfx_config *cfg, HINSTANCE self);
|
void gfx_hook_init(const struct gfx_config *cfg);
|
28
gfxhook/meson.build
Normal file
28
gfxhook/meson.build
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
gfxhook_lib = static_library(
|
||||||
|
'gfxhook',
|
||||||
|
include_directories : inc,
|
||||||
|
implicit_include_directories : false,
|
||||||
|
c_pch : '../precompiled.h',
|
||||||
|
dependencies : [
|
||||||
|
capnhook.get_variable('hook_dep'),
|
||||||
|
dxguid_lib,
|
||||||
|
],
|
||||||
|
link_with : [
|
||||||
|
hooklib_lib,
|
||||||
|
util_lib,
|
||||||
|
],
|
||||||
|
sources : [
|
||||||
|
'config.c',
|
||||||
|
'config.h',
|
||||||
|
'd3d9.c',
|
||||||
|
'd3d9.h',
|
||||||
|
'd3d11.c',
|
||||||
|
'd3d11.h',
|
||||||
|
'dxgi.c',
|
||||||
|
'dxgi.h',
|
||||||
|
'gfx.c',
|
||||||
|
'gfx.h',
|
||||||
|
'util.c',
|
||||||
|
'util.h',
|
||||||
|
],
|
||||||
|
)
|
116
gfxhook/util.c
Normal file
116
gfxhook/util.c
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include "gfxhook/util.h"
|
||||||
|
|
||||||
|
#include "util/dprintf.h"
|
||||||
|
|
||||||
|
void gfx_util_ensure_win_visible(HWND hwnd)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Ensure window is maximized to avoid a Windows 10 issue where a
|
||||||
|
* fullscreen swap chain is not created because the window is minimized
|
||||||
|
* at the time of creation.
|
||||||
|
*/
|
||||||
|
ShowWindow(hwnd, SW_RESTORE);
|
||||||
|
}
|
||||||
|
|
||||||
|
void gfx_util_borderless_fullscreen_windowed(HWND hwnd, UINT width, UINT height)
|
||||||
|
{
|
||||||
|
BOOL ok;
|
||||||
|
HRESULT hr;
|
||||||
|
|
||||||
|
dprintf("Gfx: Resizing window to %ldx%ld\n", width, height);
|
||||||
|
|
||||||
|
SetWindowLongPtrW(hwnd, GWL_STYLE, WS_POPUP);
|
||||||
|
SetWindowLongPtrW(hwnd, GWL_EXSTYLE, WS_EX_TOPMOST);
|
||||||
|
|
||||||
|
ok = SetWindowPos(
|
||||||
|
hwnd,
|
||||||
|
HWND_TOP,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
(int) width,
|
||||||
|
(int) height,
|
||||||
|
SWP_FRAMECHANGED | SWP_NOSENDCHANGING);
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
/* come on... */
|
||||||
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
||||||
|
dprintf("Gfx: SetWindowPos failed: %x\n", (int) hr);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ok = ShowWindow(hwnd, SW_SHOWMAXIMIZED);
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
/* come on... */
|
||||||
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
||||||
|
dprintf("Gfx: ShowWindow failed: %x\n", (int) hr);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
HRESULT gfx_util_frame_window(HWND hwnd)
|
||||||
|
{
|
||||||
|
HRESULT hr;
|
||||||
|
DWORD error;
|
||||||
|
LONG style;
|
||||||
|
RECT rect;
|
||||||
|
BOOL ok;
|
||||||
|
|
||||||
|
SetLastError(ERROR_SUCCESS);
|
||||||
|
style = GetWindowLongW(hwnd, GWL_STYLE);
|
||||||
|
error = GetLastError();
|
||||||
|
|
||||||
|
if (error != ERROR_SUCCESS) {
|
||||||
|
hr = HRESULT_FROM_WIN32(error);
|
||||||
|
dprintf("Gfx: GetWindowLongPtrW(%p, GWL_STYLE) failed: %x\n",
|
||||||
|
hwnd,
|
||||||
|
(int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
ok = GetClientRect(hwnd, &rect);
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
||||||
|
dprintf("Gfx: GetClientRect(%p) failed: %x\n", hwnd, (int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
style |= WS_BORDER | WS_CAPTION | WS_MINIMIZEBOX | WS_SYSMENU;
|
||||||
|
ok = AdjustWindowRect(&rect, style, FALSE);
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
/* come on... */
|
||||||
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
||||||
|
dprintf("Gfx: AdjustWindowRect failed: %x\n", (int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This... always seems to set an error, even though it works? idk */
|
||||||
|
SetWindowLongW(hwnd, GWL_STYLE, style);
|
||||||
|
|
||||||
|
ok = SetWindowPos(
|
||||||
|
hwnd,
|
||||||
|
HWND_TOP,
|
||||||
|
rect.left,
|
||||||
|
rect.top,
|
||||||
|
rect.right - rect.left,
|
||||||
|
rect.bottom - rect.top,
|
||||||
|
SWP_FRAMECHANGED | SWP_NOMOVE);
|
||||||
|
|
||||||
|
if (!ok) {
|
||||||
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
||||||
|
dprintf("Gfx: SetWindowPos(%p) failed: %x\n", hwnd, (int) hr);
|
||||||
|
|
||||||
|
return hr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return S_OK;
|
||||||
|
}
|
7
gfxhook/util.h
Normal file
7
gfxhook/util.h
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
void gfx_util_ensure_win_visible(HWND hwnd);
|
||||||
|
void gfx_util_borderless_fullscreen_windowed(HWND hwnd, UINT width, UINT height);
|
||||||
|
HRESULT gfx_util_frame_window(HWND hwnd);
|
@ -5,20 +5,8 @@
|
|||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
|
|
||||||
#include "hooklib/config.h"
|
#include "hooklib/config.h"
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
|
|
||||||
void gfx_config_load(struct gfx_config *cfg, const wchar_t *filename)
|
|
||||||
{
|
|
||||||
assert(cfg != NULL);
|
|
||||||
assert(filename != NULL);
|
|
||||||
|
|
||||||
cfg->enable = GetPrivateProfileIntW(L"gfx", L"enable", 1, filename);
|
|
||||||
cfg->windowed = GetPrivateProfileIntW(L"gfx", L"windowed", 0, filename);
|
|
||||||
cfg->framed = GetPrivateProfileIntW(L"gfx", L"framed", 1, filename);
|
|
||||||
cfg->monitor = GetPrivateProfileIntW(L"gfx", L"monitor", 0, filename);
|
|
||||||
}
|
|
||||||
|
|
||||||
void dvd_config_load(struct dvd_config *cfg, const wchar_t *filename)
|
void dvd_config_load(struct dvd_config *cfg, const wchar_t *filename)
|
||||||
{
|
{
|
||||||
assert(cfg != NULL);
|
assert(cfg != NULL);
|
||||||
|
@ -1,10 +1,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
|
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
|
|
||||||
void gfx_config_load(struct gfx_config *cfg, const wchar_t *filename);
|
|
||||||
void dvd_config_load(struct dvd_config *cfg, const wchar_t *filename);
|
void dvd_config_load(struct dvd_config *cfg, const wchar_t *filename);
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
|
217
hooklib/gfx.c
217
hooklib/gfx.c
@ -1,217 +0,0 @@
|
|||||||
#include <windows.h>
|
|
||||||
#include <d3d9.h>
|
|
||||||
|
|
||||||
#include <assert.h>
|
|
||||||
#include <stdbool.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
#include "hook/com-proxy.h"
|
|
||||||
#include "hook/table.h"
|
|
||||||
|
|
||||||
#include "hooklib/config.h"
|
|
||||||
#include "hooklib/dll.h"
|
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
|
|
||||||
#include "util/dprintf.h"
|
|
||||||
|
|
||||||
typedef IDirect3D9 * (WINAPI *Direct3DCreate9_t)(UINT sdk_ver);
|
|
||||||
|
|
||||||
static HRESULT STDMETHODCALLTYPE my_CreateDevice(
|
|
||||||
IDirect3D9 *self,
|
|
||||||
UINT adapter,
|
|
||||||
D3DDEVTYPE type,
|
|
||||||
HWND hwnd,
|
|
||||||
DWORD flags,
|
|
||||||
D3DPRESENT_PARAMETERS *pp,
|
|
||||||
IDirect3DDevice9 **pdev);
|
|
||||||
static HRESULT gfx_frame_window(HWND hwnd);
|
|
||||||
|
|
||||||
static struct gfx_config gfx_config;
|
|
||||||
static Direct3DCreate9_t next_Direct3DCreate9;
|
|
||||||
|
|
||||||
static const struct hook_symbol gfx_hooks[] = {
|
|
||||||
{
|
|
||||||
.name = "Direct3DCreate9",
|
|
||||||
.patch = Direct3DCreate9,
|
|
||||||
.link = (void **) &next_Direct3DCreate9
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
void gfx_hook_init(const struct gfx_config *cfg, HINSTANCE self)
|
|
||||||
{
|
|
||||||
HMODULE d3d9;
|
|
||||||
|
|
||||||
assert(cfg != NULL);
|
|
||||||
|
|
||||||
if (!cfg->enable) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
memcpy(&gfx_config, cfg, sizeof(*cfg));
|
|
||||||
hook_table_apply(NULL, "d3d9.dll", gfx_hooks, _countof(gfx_hooks));
|
|
||||||
|
|
||||||
if (next_Direct3DCreate9 == NULL) {
|
|
||||||
d3d9 = LoadLibraryW(L"d3d9.dll");
|
|
||||||
|
|
||||||
if (d3d9 == NULL) {
|
|
||||||
dprintf("Gfx: d3d9.dll not found or failed initialization\n");
|
|
||||||
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
next_Direct3DCreate9 = (Direct3DCreate9_t) GetProcAddress(d3d9, "Direct3DCreate9");
|
|
||||||
|
|
||||||
if (next_Direct3DCreate9 == NULL) {
|
|
||||||
dprintf("Gfx: Direct3DCreate9 not found in loaded d3d9.dll\n");
|
|
||||||
|
|
||||||
FreeLibrary(d3d9);
|
|
||||||
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (self != NULL) {
|
|
||||||
dll_hook_push(self, L"d3d9.dll");
|
|
||||||
}
|
|
||||||
|
|
||||||
fail:
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
IDirect3D9 * WINAPI Direct3DCreate9(UINT sdk_ver)
|
|
||||||
{
|
|
||||||
struct com_proxy *proxy;
|
|
||||||
IDirect3D9Vtbl *vtbl;
|
|
||||||
IDirect3D9 *api;
|
|
||||||
HRESULT hr;
|
|
||||||
|
|
||||||
dprintf("Gfx: Direct3DCreate9 hook hit\n");
|
|
||||||
|
|
||||||
if (next_Direct3DCreate9 == NULL) {
|
|
||||||
dprintf("Gfx: next_Direct3DCreate9 == NULL\n");
|
|
||||||
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
api = next_Direct3DCreate9(sdk_ver);
|
|
||||||
|
|
||||||
if (api == NULL) {
|
|
||||||
dprintf("Gfx: next_Direct3DCreate9 returned NULL\n");
|
|
||||||
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
hr = com_proxy_wrap(&proxy, api, sizeof(*api->lpVtbl));
|
|
||||||
|
|
||||||
if (FAILED(hr)) {
|
|
||||||
dprintf("Gfx: com_proxy_wrap returned %x\n", (int) hr);
|
|
||||||
|
|
||||||
goto fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
vtbl = proxy->vptr;
|
|
||||||
vtbl->CreateDevice = my_CreateDevice;
|
|
||||||
|
|
||||||
return (IDirect3D9 *) proxy;
|
|
||||||
|
|
||||||
fail:
|
|
||||||
if (api != NULL) {
|
|
||||||
IDirect3D9_Release(api);
|
|
||||||
}
|
|
||||||
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static HRESULT STDMETHODCALLTYPE my_CreateDevice(
|
|
||||||
IDirect3D9 *self,
|
|
||||||
UINT adapter,
|
|
||||||
D3DDEVTYPE type,
|
|
||||||
HWND hwnd,
|
|
||||||
DWORD flags,
|
|
||||||
D3DPRESENT_PARAMETERS *pp,
|
|
||||||
IDirect3DDevice9 **pdev)
|
|
||||||
{
|
|
||||||
struct com_proxy *proxy;
|
|
||||||
IDirect3D9 *real;
|
|
||||||
|
|
||||||
dprintf("Gfx: IDirect3D9::CreateDevice hook hit\n");
|
|
||||||
|
|
||||||
proxy = com_proxy_downcast(self);
|
|
||||||
real = proxy->real;
|
|
||||||
|
|
||||||
if (gfx_config.windowed) {
|
|
||||||
pp->Windowed = TRUE;
|
|
||||||
pp->FullScreen_RefreshRateInHz = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (gfx_config.framed) {
|
|
||||||
gfx_frame_window(hwnd);
|
|
||||||
}
|
|
||||||
|
|
||||||
dprintf("Gfx: IDirect3D9:: Using Display No %x\n", gfx_config.monitor);
|
|
||||||
|
|
||||||
return IDirect3D9_CreateDevice(real, gfx_config.monitor, type, hwnd, flags, pp, pdev);
|
|
||||||
}
|
|
||||||
|
|
||||||
static HRESULT gfx_frame_window(HWND hwnd)
|
|
||||||
{
|
|
||||||
HRESULT hr;
|
|
||||||
DWORD error;
|
|
||||||
LONG style;
|
|
||||||
RECT rect;
|
|
||||||
BOOL ok;
|
|
||||||
|
|
||||||
SetLastError(ERROR_SUCCESS);
|
|
||||||
style = GetWindowLongW(hwnd, GWL_STYLE);
|
|
||||||
error = GetLastError();
|
|
||||||
|
|
||||||
if (error != ERROR_SUCCESS) {
|
|
||||||
hr = HRESULT_FROM_WIN32(error);
|
|
||||||
dprintf("Gfx: GetWindowLongPtrW(%p, GWL_STYLE) failed: %x\n",
|
|
||||||
hwnd,
|
|
||||||
(int) hr);
|
|
||||||
|
|
||||||
return hr;
|
|
||||||
}
|
|
||||||
|
|
||||||
ok = GetClientRect(hwnd, &rect);
|
|
||||||
|
|
||||||
if (!ok) {
|
|
||||||
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
||||||
dprintf("Gfx: GetClientRect(%p) failed: %x\n", hwnd, (int) hr);
|
|
||||||
|
|
||||||
return hr;
|
|
||||||
}
|
|
||||||
|
|
||||||
style |= WS_BORDER | WS_CAPTION | WS_MINIMIZEBOX | WS_SYSMENU;
|
|
||||||
ok = AdjustWindowRect(&rect, style, FALSE);
|
|
||||||
|
|
||||||
if (!ok) {
|
|
||||||
/* come on... */
|
|
||||||
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
||||||
dprintf("Gfx: AdjustWindowRect failed: %x\n", (int) hr);
|
|
||||||
|
|
||||||
return hr;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* This... always seems to set an error, even though it works? idk */
|
|
||||||
SetWindowLongW(hwnd, GWL_STYLE, style);
|
|
||||||
|
|
||||||
ok = SetWindowPos(
|
|
||||||
hwnd,
|
|
||||||
HWND_TOP,
|
|
||||||
rect.left,
|
|
||||||
rect.top,
|
|
||||||
rect.right - rect.left,
|
|
||||||
rect.bottom - rect.top,
|
|
||||||
SWP_FRAMECHANGED | SWP_NOMOVE);
|
|
||||||
|
|
||||||
if (!ok) {
|
|
||||||
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
||||||
dprintf("Gfx: SetWindowPos(%p) failed: %x\n", hwnd, (int) hr);
|
|
||||||
|
|
||||||
return hr;
|
|
||||||
}
|
|
||||||
|
|
||||||
return S_OK;
|
|
||||||
}
|
|
@ -17,8 +17,6 @@ hooklib_lib = static_library(
|
|||||||
'dvd.h',
|
'dvd.h',
|
||||||
'fdshark.c',
|
'fdshark.c',
|
||||||
'fdshark.h',
|
'fdshark.h',
|
||||||
'gfx.c',
|
|
||||||
'gfx.h',
|
|
||||||
'path.c',
|
'path.c',
|
||||||
'path.h',
|
'path.h',
|
||||||
'reg.c',
|
'reg.c',
|
||||||
|
@ -7,6 +7,8 @@
|
|||||||
#include "board/config.h"
|
#include "board/config.h"
|
||||||
#include "board/sg-reader.h"
|
#include "board/sg-reader.h"
|
||||||
|
|
||||||
|
#include "gfxhook/config.h"
|
||||||
|
|
||||||
#include "hooklib/config.h"
|
#include "hooklib/config.h"
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
|
|
||||||
@ -42,9 +44,10 @@ void idz_hook_config_load(
|
|||||||
platform_config_load(&cfg->platform, filename);
|
platform_config_load(&cfg->platform, filename);
|
||||||
amex_config_load(&cfg->amex, filename);
|
amex_config_load(&cfg->amex, filename);
|
||||||
aime_config_load(&cfg->aime, filename);
|
aime_config_load(&cfg->aime, filename);
|
||||||
|
dvd_config_load(&cfg->dvd, filename);
|
||||||
|
gfx_config_load(&cfg->gfx, filename);
|
||||||
idz_dll_config_load(&cfg->dll, filename);
|
idz_dll_config_load(&cfg->dll, filename);
|
||||||
zinput_config_load(&cfg->zinput, filename);
|
zinput_config_load(&cfg->zinput, filename);
|
||||||
dvd_config_load(&cfg->dvd, filename);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void zinput_config_load(struct zinput_config *cfg, const wchar_t *filename)
|
void zinput_config_load(struct zinput_config *cfg, const wchar_t *filename)
|
||||||
|
@ -7,6 +7,8 @@
|
|||||||
|
|
||||||
#include "board/sg-reader.h"
|
#include "board/sg-reader.h"
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
|
|
||||||
#include "idzhook/idz-dll.h"
|
#include "idzhook/idz-dll.h"
|
||||||
@ -19,6 +21,7 @@ struct idz_hook_config {
|
|||||||
struct amex_config amex;
|
struct amex_config amex;
|
||||||
struct aime_config aime;
|
struct aime_config aime;
|
||||||
struct dvd_config dvd;
|
struct dvd_config dvd;
|
||||||
|
struct gfx_config gfx;
|
||||||
struct idz_dll_config dll;
|
struct idz_dll_config dll;
|
||||||
struct zinput_config zinput;
|
struct zinput_config zinput;
|
||||||
};
|
};
|
||||||
|
@ -1,11 +1,18 @@
|
|||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
#include <shlwapi.h>
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <wchar.h>
|
||||||
|
|
||||||
#include "amex/amex.h"
|
#include "amex/amex.h"
|
||||||
|
|
||||||
#include "board/sg-reader.h"
|
#include "board/sg-reader.h"
|
||||||
|
|
||||||
|
#include "gfxhook/d3d11.h"
|
||||||
|
#include "gfxhook/dxgi.h"
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
#include "hook/process.h"
|
#include "hook/process.h"
|
||||||
|
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
@ -20,6 +27,7 @@
|
|||||||
#include "platform/platform.h"
|
#include "platform/platform.h"
|
||||||
|
|
||||||
#include "util/dprintf.h"
|
#include "util/dprintf.h"
|
||||||
|
#include "util/lib.h"
|
||||||
|
|
||||||
static HMODULE idz_hook_mod;
|
static HMODULE idz_hook_mod;
|
||||||
static process_entry_t idz_startup;
|
static process_entry_t idz_startup;
|
||||||
@ -27,6 +35,8 @@ static struct idz_hook_config idz_hook_cfg;
|
|||||||
|
|
||||||
static DWORD CALLBACK idz_pre_startup(void)
|
static DWORD CALLBACK idz_pre_startup(void)
|
||||||
{
|
{
|
||||||
|
wchar_t *module_path;
|
||||||
|
wchar_t *file_name;
|
||||||
HRESULT hr;
|
HRESULT hr;
|
||||||
|
|
||||||
dprintf("--- Begin idz_pre_startup ---\n");
|
dprintf("--- Begin idz_pre_startup ---\n");
|
||||||
@ -35,9 +45,31 @@ static DWORD CALLBACK idz_pre_startup(void)
|
|||||||
|
|
||||||
idz_hook_config_load(&idz_hook_cfg, L".\\segatools.ini");
|
idz_hook_config_load(&idz_hook_cfg, L".\\segatools.ini");
|
||||||
|
|
||||||
|
module_path = module_file_name(NULL);
|
||||||
|
|
||||||
|
if (module_path != NULL) {
|
||||||
|
file_name = PathFindFileNameW(module_path);
|
||||||
|
|
||||||
|
_wcslwr(file_name);
|
||||||
|
|
||||||
|
if (wcsstr(file_name, L"serverbox") != NULL) {
|
||||||
|
dprintf("Executable filename contains 'ServerBox', disabling full-screen mode\n");
|
||||||
|
|
||||||
|
idz_hook_cfg.gfx.windowed = true;
|
||||||
|
idz_hook_cfg.gfx.framed = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(module_path);
|
||||||
|
|
||||||
|
module_path = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/* Hook Win32 APIs */
|
/* Hook Win32 APIs */
|
||||||
|
|
||||||
serial_hook_init();
|
serial_hook_init();
|
||||||
|
gfx_hook_init(&idz_hook_cfg.gfx);
|
||||||
|
gfx_d3d11_hook_init(&idz_hook_cfg.gfx, idz_hook_mod);
|
||||||
|
gfx_dxgi_hook_init(&idz_hook_cfg.gfx, idz_hook_mod);
|
||||||
zinput_hook_init(&idz_hook_cfg.zinput);
|
zinput_hook_init(&idz_hook_cfg.zinput);
|
||||||
dvd_hook_init(&idz_hook_cfg.dvd, idz_hook_mod);
|
dvd_hook_init(&idz_hook_cfg.dvd, idz_hook_mod);
|
||||||
|
|
||||||
|
@ -1,6 +1,11 @@
|
|||||||
LIBRARY idzhook
|
LIBRARY idzhook
|
||||||
|
|
||||||
EXPORTS
|
EXPORTS
|
||||||
|
CreateDXGIFactory
|
||||||
|
CreateDXGIFactory1
|
||||||
|
CreateDXGIFactory2
|
||||||
|
D3D11CreateDevice
|
||||||
|
D3D11CreateDeviceAndSwapChain
|
||||||
aime_io_get_api_version
|
aime_io_get_api_version
|
||||||
aime_io_init
|
aime_io_init
|
||||||
aime_io_led_set_color
|
aime_io_led_set_color
|
||||||
|
@ -8,12 +8,14 @@ shared_library(
|
|||||||
dependencies : [
|
dependencies : [
|
||||||
capnhook.get_variable('hook_dep'),
|
capnhook.get_variable('hook_dep'),
|
||||||
capnhook.get_variable('hooklib_dep'),
|
capnhook.get_variable('hooklib_dep'),
|
||||||
|
shlwapi_lib,
|
||||||
xinput_lib,
|
xinput_lib,
|
||||||
],
|
],
|
||||||
link_with : [
|
link_with : [
|
||||||
aimeio_lib,
|
aimeio_lib,
|
||||||
amex_lib,
|
amex_lib,
|
||||||
board_lib,
|
board_lib,
|
||||||
|
gfxhook_lib,
|
||||||
hooklib_lib,
|
hooklib_lib,
|
||||||
idzio_lib,
|
idzio_lib,
|
||||||
jvs_lib,
|
jvs_lib,
|
||||||
|
@ -50,6 +50,8 @@ subdir('jvs')
|
|||||||
subdir('platform')
|
subdir('platform')
|
||||||
subdir('util')
|
subdir('util')
|
||||||
|
|
||||||
|
subdir('gfxhook')
|
||||||
|
|
||||||
subdir('aimeio')
|
subdir('aimeio')
|
||||||
subdir('chuniio')
|
subdir('chuniio')
|
||||||
subdir('divaio')
|
subdir('divaio')
|
||||||
|
@ -3,9 +3,10 @@
|
|||||||
|
|
||||||
#include "board/config.h"
|
#include "board/config.h"
|
||||||
|
|
||||||
|
#include "gfxhook/config.h"
|
||||||
|
|
||||||
#include "hooklib/config.h"
|
#include "hooklib/config.h"
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
|
|
||||||
#include "mu3hook/config.h"
|
#include "mu3hook/config.h"
|
||||||
|
|
||||||
|
@ -4,8 +4,9 @@
|
|||||||
|
|
||||||
#include "board/config.h"
|
#include "board/config.h"
|
||||||
|
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
#include "hooklib/gfx.h"
|
|
||||||
|
|
||||||
#include "mu3hook/mu3-dll.h"
|
#include "mu3hook/mu3-dll.h"
|
||||||
|
|
||||||
|
@ -6,6 +6,11 @@
|
|||||||
#include "board/sg-reader.h"
|
#include "board/sg-reader.h"
|
||||||
#include "board/vfd.h"
|
#include "board/vfd.h"
|
||||||
|
|
||||||
|
#include "gfxhook/d3d9.h"
|
||||||
|
#include "gfxhook/d3d11.h"
|
||||||
|
#include "gfxhook/dxgi.h"
|
||||||
|
#include "gfxhook/gfx.h"
|
||||||
|
|
||||||
#include "hook/process.h"
|
#include "hook/process.h"
|
||||||
|
|
||||||
#include "hooklib/dvd.h"
|
#include "hooklib/dvd.h"
|
||||||
@ -38,7 +43,10 @@ static DWORD CALLBACK mu3_pre_startup(void)
|
|||||||
/* Hook Win32 APIs */
|
/* Hook Win32 APIs */
|
||||||
|
|
||||||
dvd_hook_init(&mu3_hook_cfg.dvd, mu3_hook_mod);
|
dvd_hook_init(&mu3_hook_cfg.dvd, mu3_hook_mod);
|
||||||
gfx_hook_init(&mu3_hook_cfg.gfx, mu3_hook_mod);
|
gfx_hook_init(&mu3_hook_cfg.gfx);
|
||||||
|
gfx_d3d9_hook_init(&mu3_hook_cfg.gfx, mu3_hook_mod);
|
||||||
|
gfx_d3d11_hook_init(&mu3_hook_cfg.gfx, mu3_hook_mod);
|
||||||
|
gfx_dxgi_hook_init(&mu3_hook_cfg.gfx, mu3_hook_mod);
|
||||||
serial_hook_init();
|
serial_hook_init();
|
||||||
|
|
||||||
/* Initialize emulation hooks */
|
/* Initialize emulation hooks */
|
||||||
|
@ -13,6 +13,7 @@ shared_library(
|
|||||||
link_with : [
|
link_with : [
|
||||||
aimeio_lib,
|
aimeio_lib,
|
||||||
board_lib,
|
board_lib,
|
||||||
|
gfxhook_lib,
|
||||||
hooklib_lib,
|
hooklib_lib,
|
||||||
mu3io_lib,
|
mu3io_lib,
|
||||||
platform_lib,
|
platform_lib,
|
||||||
|
@ -1,6 +1,11 @@
|
|||||||
LIBRARY mu3hook
|
LIBRARY mu3hook
|
||||||
|
|
||||||
EXPORTS
|
EXPORTS
|
||||||
|
CreateDXGIFactory
|
||||||
|
CreateDXGIFactory1
|
||||||
|
CreateDXGIFactory2
|
||||||
|
D3D11CreateDevice
|
||||||
|
D3D11CreateDeviceAndSwapChain
|
||||||
Direct3DCreate9
|
Direct3DCreate9
|
||||||
aime_io_get_api_version
|
aime_io_get_api_version
|
||||||
aime_io_init
|
aime_io_init
|
||||||
|
29
util/lib.c
Normal file
29
util/lib.c
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
wchar_t *module_file_name(HMODULE module)
|
||||||
|
{
|
||||||
|
size_t buf_len;
|
||||||
|
DWORD len;
|
||||||
|
wchar_t *buf;
|
||||||
|
|
||||||
|
buf_len = MAX_PATH;
|
||||||
|
buf = malloc(buf_len * sizeof(*buf));
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
len = GetModuleFileNameW(module, buf, buf_len);
|
||||||
|
|
||||||
|
if (GetLastError() != ERROR_INSUFFICIENT_BUFFER) {
|
||||||
|
buf_len = len;
|
||||||
|
buf = realloc(buf, buf_len * sizeof(*buf));
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
buf_len *= 2;
|
||||||
|
buf = realloc(buf, buf_len * sizeof(*buf));
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf;
|
||||||
|
}
|
5
util/lib.h
Normal file
5
util/lib.h
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
|
||||||
|
wchar_t *module_file_name(HMODULE module);
|
@ -17,6 +17,8 @@ util_lib = static_library(
|
|||||||
'dprintf.h',
|
'dprintf.h',
|
||||||
'dump.c',
|
'dump.c',
|
||||||
'dump.h',
|
'dump.h',
|
||||||
|
'lib.c',
|
||||||
|
'lib.h',
|
||||||
'str.c',
|
'str.c',
|
||||||
'str.h',
|
'str.h',
|
||||||
],
|
],
|
||||||
|
Loading…
Reference in New Issue
Block a user