mirror of
https://github.com/libretro/RetroArch.git
synced 2025-01-23 09:55:48 +00:00
609 lines
16 KiB
C
609 lines
16 KiB
C
/* RetroArch - A frontend for libretro.
|
|
* Copyright (C) 2011-2017 - Daniel De Matteis
|
|
*
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
*
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <Windows.h>
|
|
|
|
#include "../input_driver.h"
|
|
#include "../input_keymaps.h"
|
|
#include "../video_driver.h"
|
|
#include "../../verbosity.h"
|
|
|
|
#define WINRAW_LOG(msg) RARCH_LOG("[WINRAW]: "msg"\n")
|
|
#define WINRAW_ERR(err) RARCH_ERR("[WINRAW]: "err"\n")
|
|
|
|
#define WINRAW_SYS_WRN(fun)\
|
|
RARCH_WARN("[WINRAW]: "fun" failed with error %lu.\n", GetLastError())
|
|
#define WINRAW_SYS_ERR(fun)\
|
|
RARCH_ERR("[WINRAW]: "fun" failed with error %lu.\n", GetLastError())
|
|
|
|
typedef struct
|
|
{
|
|
uint8_t keys[256];
|
|
} winraw_keyboard_t;
|
|
|
|
typedef struct
|
|
{
|
|
HANDLE hnd;
|
|
LONG x, y, dlt_x, dlt_y;
|
|
LONG whl_u, whl_d;
|
|
bool btn_l, btn_m, btn_r;
|
|
} winraw_mouse_t;
|
|
|
|
typedef struct
|
|
{
|
|
winraw_keyboard_t keyboard;
|
|
winraw_mouse_t *mice;
|
|
const input_device_driver_t *joypad;
|
|
HWND window;
|
|
bool kbd_mapp_block;
|
|
bool mouse_grab;
|
|
} winraw_input_t;
|
|
|
|
static winraw_keyboard_t *g_keyboard = NULL;
|
|
static winraw_mouse_t *g_mice = NULL;
|
|
static unsigned g_mouse_cnt = 0;
|
|
|
|
static HWND winraw_create_window(WNDPROC wnd_proc)
|
|
{
|
|
HWND wnd;
|
|
WNDCLASSA wc = {0};
|
|
|
|
wc.hInstance = GetModuleHandleA(NULL);
|
|
|
|
if (!wc.hInstance)
|
|
{
|
|
WINRAW_SYS_ERR("GetModuleHandleA");
|
|
return NULL;
|
|
}
|
|
|
|
wc.lpfnWndProc = wnd_proc;
|
|
wc.lpszClassName = "winraw-input";
|
|
if (!RegisterClassA(&wc) && GetLastError() != ERROR_CLASS_ALREADY_EXISTS)
|
|
{
|
|
WINRAW_SYS_ERR("RegisterClassA");
|
|
return NULL;
|
|
}
|
|
|
|
wnd = CreateWindowExA(0, wc.lpszClassName, NULL, 0, 0, 0, 0, 0,
|
|
HWND_MESSAGE, NULL, NULL, NULL);
|
|
if (!wnd)
|
|
{
|
|
WINRAW_SYS_ERR("CreateWindowExA");
|
|
goto error;
|
|
}
|
|
|
|
return wnd;
|
|
|
|
error:
|
|
UnregisterClassA(wc.lpszClassName, NULL);
|
|
return NULL;
|
|
}
|
|
|
|
static void winraw_destroy_window(HWND wnd)
|
|
{
|
|
BOOL r;
|
|
|
|
if (!wnd)
|
|
return;
|
|
|
|
r = DestroyWindow(wnd);
|
|
|
|
if (!r)
|
|
WINRAW_SYS_WRN("DestroyWindow");
|
|
|
|
r = UnregisterClassA("winraw-input", NULL);
|
|
|
|
if (!r)
|
|
WINRAW_SYS_WRN("UnregisterClassA");
|
|
}
|
|
|
|
static bool winraw_set_keyboard_input(HWND window)
|
|
{
|
|
RAWINPUTDEVICE rid;
|
|
BOOL r;
|
|
|
|
rid.dwFlags = window ? 0 : RIDEV_REMOVE;
|
|
rid.hwndTarget = window;
|
|
rid.usUsagePage = 0x01; /* generic desktop */
|
|
rid.usUsage = 0x06; /* keyboard */
|
|
|
|
r = RegisterRawInputDevices(&rid, 1, sizeof(RAWINPUTDEVICE));
|
|
|
|
if (!r)
|
|
{
|
|
WINRAW_SYS_ERR("RegisterRawInputDevices");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool winraw_init_devices(winraw_mouse_t **mice, unsigned *mouse_cnt)
|
|
{
|
|
UINT r, i;
|
|
POINT crs_pos;
|
|
winraw_mouse_t *mice_r = NULL;
|
|
unsigned mouse_cnt_r = 0;
|
|
RAWINPUTDEVICELIST *devs = NULL;
|
|
UINT dev_cnt = 0;
|
|
|
|
r = GetRawInputDeviceList(NULL, &dev_cnt, sizeof(RAWINPUTDEVICELIST));
|
|
if (r == (UINT)-1)
|
|
{
|
|
WINRAW_SYS_ERR("GetRawInputDeviceList");
|
|
goto error;
|
|
}
|
|
|
|
devs = (RAWINPUTDEVICELIST*)malloc(dev_cnt * sizeof(RAWINPUTDEVICELIST));
|
|
if (!devs)
|
|
{
|
|
WINRAW_ERR("malloc failed");
|
|
goto error;
|
|
}
|
|
|
|
dev_cnt = GetRawInputDeviceList(devs, &dev_cnt, sizeof(RAWINPUTDEVICELIST));
|
|
if (dev_cnt == (UINT)-1)
|
|
{
|
|
WINRAW_SYS_ERR("GetRawInputDeviceList");
|
|
goto error;
|
|
}
|
|
|
|
for (i = 0; i < dev_cnt; ++i)
|
|
mouse_cnt_r += devs[i].dwType == RIM_TYPEMOUSE ? 1 : 0;
|
|
|
|
if (mouse_cnt_r)
|
|
{
|
|
mice_r = (winraw_mouse_t*)calloc(1, mouse_cnt_r * sizeof(winraw_mouse_t));
|
|
if (!mice_r)
|
|
{
|
|
WINRAW_ERR("calloc failed");
|
|
goto error;
|
|
}
|
|
|
|
if (!GetCursorPos(&crs_pos))
|
|
{
|
|
WINRAW_SYS_WRN("GetCursorPos");
|
|
goto error;
|
|
}
|
|
|
|
for (i = 0; i < mouse_cnt_r; ++i)
|
|
{
|
|
mice_r[i].x = crs_pos.x;
|
|
mice_r[i].y = crs_pos.y;
|
|
}
|
|
}
|
|
|
|
/* count is already checked, so this is safe */
|
|
for (i = mouse_cnt_r = 0; i < dev_cnt; ++i)
|
|
{
|
|
if (devs[i].dwType == RIM_TYPEMOUSE)
|
|
mice_r[mouse_cnt_r++].hnd = devs[i].hDevice;
|
|
}
|
|
|
|
*mice = mice_r;
|
|
*mouse_cnt = mouse_cnt_r;
|
|
|
|
return true;
|
|
|
|
error:
|
|
free(devs);
|
|
free(mice_r);
|
|
*mice = NULL;
|
|
*mouse_cnt = 0;
|
|
return false;
|
|
}
|
|
|
|
static bool winraw_set_mouse_input(HWND window, bool grab)
|
|
{
|
|
RAWINPUTDEVICE rid;
|
|
BOOL r;
|
|
|
|
if (window)
|
|
rid.dwFlags = grab ? RIDEV_CAPTUREMOUSE : 0;
|
|
else
|
|
rid.dwFlags = RIDEV_REMOVE;
|
|
|
|
rid.hwndTarget = window;
|
|
rid.usUsagePage = 0x01; /* generic desktop */
|
|
rid.usUsage = 0x02; /* mouse */
|
|
|
|
r = RegisterRawInputDevices(&rid, 1, sizeof(RAWINPUTDEVICE));
|
|
|
|
if (!r)
|
|
{
|
|
WINRAW_SYS_ERR("RegisterRawInputDevices");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int16_t winraw_mouse_state(winraw_mouse_t *mouse, bool abs, unsigned id)
|
|
{
|
|
switch (id)
|
|
{
|
|
case RETRO_DEVICE_ID_MOUSE_X:
|
|
return abs ? mouse->x : mouse->dlt_x;
|
|
case RETRO_DEVICE_ID_MOUSE_Y:
|
|
return abs ? mouse->y : mouse->dlt_y;
|
|
case RETRO_DEVICE_ID_MOUSE_LEFT:
|
|
return mouse->btn_l ? 1 : 0;
|
|
case RETRO_DEVICE_ID_MOUSE_RIGHT:
|
|
return mouse->btn_r ? 1 : 0;
|
|
case RETRO_DEVICE_ID_MOUSE_WHEELUP:
|
|
return mouse->whl_u ? 1 : 0;
|
|
case RETRO_DEVICE_ID_MOUSE_WHEELDOWN:
|
|
return mouse->whl_d ? 1 : 0;
|
|
case RETRO_DEVICE_ID_MOUSE_MIDDLE:
|
|
return mouse->btn_m ? 1 : 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int16_t winraw_joypad_state(winraw_input_t *wr,
|
|
rarch_joypad_info_t joypad_info,
|
|
const struct retro_keybind *binds,
|
|
unsigned port, unsigned id)
|
|
{
|
|
const struct retro_keybind *bind = &binds[id];
|
|
unsigned key = rarch_keysym_lut[(enum retro_key)bind->key];
|
|
|
|
if (!wr->kbd_mapp_block && (bind->key < RETROK_LAST) && wr->keyboard.keys[key])
|
|
return 1;
|
|
|
|
return input_joypad_pressed(wr->joypad, joypad_info, port, binds, id);
|
|
}
|
|
|
|
static void winraw_update_mouse_state(winraw_mouse_t *mouse, RAWMOUSE *state)
|
|
{
|
|
POINT crs_pos;
|
|
|
|
if (state->usFlags & MOUSE_MOVE_ABSOLUTE)
|
|
{
|
|
mouse->x = state->lLastX;
|
|
mouse->y = state->lLastY;
|
|
}
|
|
else if (state->lLastX || state->lLastY)
|
|
{
|
|
InterlockedExchangeAdd(&mouse->dlt_x, state->lLastX);
|
|
InterlockedExchangeAdd(&mouse->dlt_y, state->lLastY);
|
|
|
|
if (!GetCursorPos(&crs_pos))
|
|
WINRAW_SYS_WRN("GetCursorPos");
|
|
else if (!ScreenToClient((HWND)video_driver_window_get(), &crs_pos))
|
|
WINRAW_SYS_WRN("ScreenToClient");
|
|
else
|
|
{
|
|
mouse->x = crs_pos.x;
|
|
mouse->y = crs_pos.y;
|
|
}
|
|
}
|
|
|
|
if (state->usButtonFlags & RI_MOUSE_LEFT_BUTTON_DOWN)
|
|
mouse->btn_l = true;
|
|
else if (state->usButtonFlags & RI_MOUSE_LEFT_BUTTON_UP)
|
|
mouse->btn_l = false;
|
|
|
|
if (state->usButtonFlags & RI_MOUSE_MIDDLE_BUTTON_DOWN)
|
|
mouse->btn_m = true;
|
|
else if (state->usButtonFlags & RI_MOUSE_MIDDLE_BUTTON_UP)
|
|
mouse->btn_m = false;
|
|
|
|
if (state->usButtonFlags & RI_MOUSE_RIGHT_BUTTON_DOWN)
|
|
mouse->btn_r = true;
|
|
else if (state->usButtonFlags & RI_MOUSE_RIGHT_BUTTON_UP)
|
|
mouse->btn_r = false;
|
|
|
|
if (state->usButtonFlags & RI_MOUSE_WHEEL)
|
|
{
|
|
if ((SHORT)state->usButtonData > 0)
|
|
InterlockedExchange(&mouse->whl_u, 1);
|
|
else if ((SHORT)state->usButtonData < 0)
|
|
InterlockedExchange(&mouse->whl_d, 1);
|
|
}
|
|
}
|
|
|
|
static LRESULT CALLBACK winraw_callback(HWND wnd, UINT msg, WPARAM wpar, LPARAM lpar)
|
|
{
|
|
static uint8_t data[1024];
|
|
UINT r;
|
|
unsigned i;
|
|
RAWINPUT *ri = (RAWINPUT*)data;
|
|
UINT size = sizeof(data);
|
|
|
|
if (msg != WM_INPUT)
|
|
return DefWindowProcA(wnd, msg, wpar, lpar);
|
|
|
|
if (GET_RAWINPUT_CODE_WPARAM(wpar) != RIM_INPUT) /* app is in the background */
|
|
goto end;
|
|
|
|
r = GetRawInputData((HRAWINPUT)lpar, RID_INPUT, data, &size, sizeof(RAWINPUTHEADER));
|
|
if (r == (UINT)-1)
|
|
{
|
|
WINRAW_SYS_WRN("GetRawInputData");
|
|
goto end;
|
|
}
|
|
|
|
if (ri->header.dwType == RIM_TYPEKEYBOARD)
|
|
{
|
|
if (ri->data.keyboard.Message == WM_KEYDOWN)
|
|
g_keyboard->keys[ri->data.keyboard.VKey] = 1;
|
|
else if (ri->data.keyboard.Message == WM_KEYUP)
|
|
g_keyboard->keys[ri->data.keyboard.VKey] = 0;
|
|
}
|
|
else if (ri->header.dwType == RIM_TYPEMOUSE)
|
|
{
|
|
for (i = 0; i < g_mouse_cnt; ++i)
|
|
{
|
|
if (g_mice[i].hnd == ri->header.hDevice)
|
|
{
|
|
winraw_update_mouse_state(&g_mice[i], &ri->data.mouse);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
end:
|
|
DefWindowProcA(wnd, msg, wpar, lpar);
|
|
return 0;
|
|
}
|
|
|
|
static void *winraw_init(const char *joypad_driver)
|
|
{
|
|
bool r;
|
|
video_viewport_t dst_vid_info;
|
|
struct retro_system_av_info *src_vid_info;
|
|
winraw_input_t *wr = (winraw_input_t *)calloc(1, sizeof(winraw_input_t));
|
|
g_keyboard = (winraw_keyboard_t*)calloc(1, sizeof(winraw_keyboard_t));
|
|
|
|
if (!wr || !g_keyboard)
|
|
{
|
|
WINRAW_ERR("calloc failed.");
|
|
goto error;
|
|
}
|
|
|
|
WINRAW_LOG("Initializing input driver ...");
|
|
|
|
input_keymaps_init_keyboard_lut(rarch_key_map_winraw);
|
|
|
|
wr->window = winraw_create_window(winraw_callback);
|
|
if (!wr->window)
|
|
{
|
|
WINRAW_ERR("winraw_create_window failed.");
|
|
goto error;
|
|
}
|
|
|
|
r = winraw_init_devices(&g_mice, &g_mouse_cnt);
|
|
if (!r)
|
|
{
|
|
WINRAW_ERR("winraw_init_devices failed.");
|
|
goto error;
|
|
}
|
|
|
|
if (!g_mouse_cnt)
|
|
WINRAW_LOG("Mouse unavailable.");
|
|
else
|
|
{
|
|
wr->mice = (winraw_mouse_t*)malloc(g_mouse_cnt * sizeof(winraw_mouse_t));
|
|
if (!wr->mice)
|
|
{
|
|
WINRAW_ERR("malloc failed.");
|
|
goto error;
|
|
}
|
|
|
|
memcpy(wr->mice, g_mice, g_mouse_cnt * sizeof(winraw_mouse_t));
|
|
}
|
|
|
|
r = winraw_set_keyboard_input(wr->window);
|
|
if (!r)
|
|
{
|
|
WINRAW_ERR("winraw_set_keyboard_input failed.");
|
|
goto error;
|
|
}
|
|
|
|
r = winraw_set_mouse_input(wr->window, false);
|
|
if (!r)
|
|
{
|
|
WINRAW_ERR("winraw_set_mouse_input failed.");
|
|
goto error;
|
|
}
|
|
|
|
wr->joypad = input_joypad_init_driver(joypad_driver, wr);
|
|
|
|
WINRAW_LOG("Input driver initialized.");
|
|
return wr;
|
|
|
|
error:
|
|
if (wr && wr->window)
|
|
{
|
|
winraw_set_mouse_input(NULL, false);
|
|
winraw_set_keyboard_input(NULL);
|
|
winraw_destroy_window(wr->window);
|
|
}
|
|
free(g_keyboard);
|
|
free(g_mice);
|
|
if (wr)
|
|
free(wr->mice);
|
|
free(wr);
|
|
WINRAW_ERR("Input driver initialization failed.");
|
|
return NULL;
|
|
}
|
|
|
|
static void winraw_poll(void *d)
|
|
{
|
|
unsigned i;
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
memcpy(&wr->keyboard, g_keyboard, sizeof(winraw_keyboard_t));
|
|
|
|
/* following keys are not handled by windows raw input api */
|
|
wr->keyboard.keys[VK_LCONTROL] = GetAsyncKeyState(VK_LCONTROL) >> 1 ? 1 : 0;
|
|
wr->keyboard.keys[VK_RCONTROL] = GetAsyncKeyState(VK_RCONTROL) >> 1 ? 1 : 0;
|
|
wr->keyboard.keys[VK_LMENU] = GetAsyncKeyState(VK_LMENU) >> 1 ? 1 : 0;
|
|
wr->keyboard.keys[VK_RMENU] = GetAsyncKeyState(VK_RMENU) >> 1 ? 1 : 0;
|
|
wr->keyboard.keys[VK_LSHIFT] = GetAsyncKeyState(VK_LSHIFT) >> 1 ? 1 : 0;
|
|
wr->keyboard.keys[VK_RSHIFT] = GetAsyncKeyState(VK_RSHIFT) >> 1 ? 1 : 0;
|
|
|
|
for (i = 0; i < g_mouse_cnt; ++i)
|
|
{
|
|
wr->mice[i].x = g_mice[i].x;
|
|
wr->mice[i].y = g_mice[i].y;
|
|
wr->mice[i].dlt_x = InterlockedExchange(&g_mice[i].dlt_x, 0);
|
|
wr->mice[i].dlt_y = InterlockedExchange(&g_mice[i].dlt_y, 0);
|
|
wr->mice[i].whl_u = InterlockedExchange(&g_mice[i].whl_u, 0);
|
|
wr->mice[i].whl_d = InterlockedExchange(&g_mice[i].whl_d, 0);
|
|
wr->mice[i].btn_l = g_mice[i].btn_l;
|
|
wr->mice[i].btn_m = g_mice[i].btn_m;
|
|
wr->mice[i].btn_r = g_mice[i].btn_r;
|
|
}
|
|
|
|
if (wr->joypad)
|
|
wr->joypad->poll();
|
|
}
|
|
|
|
static int16_t winraw_input_state(void *d,
|
|
rarch_joypad_info_t joypad_info,
|
|
const struct retro_keybind **binds,
|
|
unsigned port, unsigned device, unsigned index, unsigned id)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
switch (device)
|
|
{
|
|
case RETRO_DEVICE_KEYBOARD:
|
|
if (id < RETROK_LAST)
|
|
{
|
|
unsigned key = rarch_keysym_lut[(enum retro_key)id];
|
|
return wr->keyboard.keys[key];
|
|
}
|
|
break;
|
|
case RETRO_DEVICE_MOUSE:
|
|
if (port < g_mouse_cnt)
|
|
return winraw_mouse_state(&wr->mice[port], false, id);
|
|
break;
|
|
case RARCH_DEVICE_MOUSE_SCREEN:
|
|
if (port < g_mouse_cnt)
|
|
return winraw_mouse_state(&wr->mice[port], true, id);
|
|
break;
|
|
case RETRO_DEVICE_JOYPAD:
|
|
return winraw_joypad_state(wr, joypad_info, binds[port], port, id);
|
|
case RETRO_DEVICE_ANALOG:
|
|
if (binds[port])
|
|
return input_joypad_analog(wr->joypad, joypad_info,
|
|
port, index, id, binds[port]);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static bool winraw_meta_key_pressed(void *u1, int u2)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
static void winraw_free(void *d)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
WINRAW_LOG("Deinitializing input driver ...");
|
|
|
|
if (wr->joypad)
|
|
wr->joypad->destroy();
|
|
winraw_set_mouse_input(NULL, false);
|
|
winraw_set_keyboard_input(NULL);
|
|
winraw_destroy_window(wr->window);
|
|
free(g_mice);
|
|
free(g_keyboard);
|
|
free(wr->mice);
|
|
free(wr);
|
|
|
|
WINRAW_LOG("Input driver deinitialized.");
|
|
}
|
|
|
|
static uint64_t winraw_get_capabilities(void *u)
|
|
{
|
|
return (1 << RETRO_DEVICE_KEYBOARD) |
|
|
(1 << RETRO_DEVICE_MOUSE) |
|
|
(1 << RETRO_DEVICE_JOYPAD) |
|
|
(1 << RETRO_DEVICE_ANALOG);
|
|
}
|
|
|
|
static void winraw_grab_mouse(void *d, bool grab)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
bool r;
|
|
|
|
if (grab == wr->mouse_grab)
|
|
return;
|
|
|
|
r = winraw_set_mouse_input(wr->window, grab);
|
|
if (!r)
|
|
{
|
|
WINRAW_ERR("Mouse grab failed.");
|
|
return;
|
|
}
|
|
|
|
wr->mouse_grab = grab;
|
|
}
|
|
|
|
static bool winraw_set_rumble(void *d, unsigned port,
|
|
enum retro_rumble_effect effect, uint16_t strength)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
return input_joypad_set_rumble(wr->joypad, port, effect, strength);
|
|
}
|
|
|
|
static const input_device_driver_t *winraw_get_joypad_driver(void *d)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
return wr->joypad;
|
|
}
|
|
|
|
static bool winraw_keyboard_mapping_is_blocked(void *d)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
return wr->kbd_mapp_block;
|
|
}
|
|
|
|
static void winraw_keyboard_mapping_set_block(void *d, bool block)
|
|
{
|
|
winraw_input_t *wr = (winraw_input_t*)d;
|
|
|
|
wr->kbd_mapp_block = block;
|
|
}
|
|
|
|
input_driver_t input_winraw = {
|
|
winraw_init,
|
|
winraw_poll,
|
|
winraw_input_state,
|
|
winraw_meta_key_pressed,
|
|
winraw_free,
|
|
NULL,
|
|
NULL,
|
|
winraw_get_capabilities,
|
|
"raw",
|
|
winraw_grab_mouse,
|
|
NULL,
|
|
winraw_set_rumble,
|
|
winraw_get_joypad_driver,
|
|
NULL,
|
|
winraw_keyboard_mapping_is_blocked,
|
|
winraw_keyboard_mapping_set_block,
|
|
};
|