#include <windows.h>
#include <stdlib.h>

#include "tekkenhook/config.h"
#include "tekkenhook/tekken-dll.h"
#include "tekkenhook/jvs.h"

#include "amcus/amcus.h"

#include "hook/process.h"

#include "hooklib/serial.h"
#include "hooklib/debug.h"

#include "platform/platform.h"
#include "gfxhook/gfx.h"
#include "gfxhook/dxgi.h"
#include "gfxhook/d3d11.h"
#include "board/bpreader.h"

#include "util/dprintf.h"

static HMODULE tekken_hook_mod;
static process_entry_t tekken_startup;
static struct tekken_hook_config tekken_hook_cfg;

static DWORD CALLBACK tekken_pre_startup(void)
{
    HRESULT hr;

    dprintf("--- Begin tekken_pre_startup ---\n");

    tekken_hook_config_load(&tekken_hook_cfg, L".\\bananatools.ini");

    serial_hook_init();

    struct dongle_info dinfo;
    dinfo.vid = 0x0B9A;
    dinfo.pid = 0x0C10;
    wcscpy_s(dinfo.manufacturer, _countof(dinfo.manufacturer), L"BM");
    wcscpy_s(dinfo.product, _countof(dinfo.product), L"RUDI04GBN-274713");

    hr = platform_hook_init(&tekken_hook_cfg.platform, PLATFORM_ES3, tekken_jvs_init, tekken_hook_mod, dinfo);

    if (FAILED(hr)) {
        ExitProcess(EXIT_FAILURE);
    }

    hr = tekken_dll_init(&tekken_hook_cfg.dll, tekken_hook_mod);

    if (FAILED(hr)) {
        ExitProcess(EXIT_FAILURE);
    }

    hr = amcus_hook_init(&tekken_hook_cfg.amcus);

    if (FAILED(hr)) {
        ExitProcess(EXIT_FAILURE);
    }

    hr = bpreader_init(&tekken_hook_cfg.reader, 4);

    if (FAILED(hr)) {
        ExitProcess(EXIT_FAILURE);
    }

    debug_hook_init();

    gfx_hook_init(&tekken_hook_cfg.gfx);
    gfx_d3d11_hook_init(&tekken_hook_cfg.gfx, tekken_hook_mod);
    gfx_dxgi_hook_init(&tekken_hook_cfg.gfx, tekken_hook_mod);

    dprintf("---  End  tekken_pre_startup ---\n");

    return tekken_startup();

}

BOOL WINAPI DllMain(HMODULE mod, DWORD cause, void *ctx)
{
    HRESULT hr;

    if (cause != DLL_PROCESS_ATTACH) {
        return TRUE;
    }

    tekken_hook_mod = mod;

    hr = process_hijack_startup(tekken_pre_startup, &tekken_startup);

    if (!SUCCEEDED(hr)) {
        dprintf("Failed to hijack process startup: %x\n", (int) hr);
    }

    return SUCCEEDED(hr);
}