/* in_common.c general input driver Copyright (C) 1996-1997 Id Software, Inc. Copyright (C) 2000 Marcus Sundberg [mackan@stacken.kth.se] Copyright (C) 1999,2000 contributors of the QuakeForge project Please see the file "AUTHORS" for a list of contributors This program 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 Foundation; either version 2 of the License, or (at your option) any later version. This program 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 this program; if not, write to: Free Software Foundation, Inc. 59 Temple Place - Suite 330 Boston, MA 02111-1307, USA */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #ifdef HAVE_STRING_H # include #endif #ifdef HAVE_STRINGS_H # include #endif #ifdef HAVE_UNISTD_H # include #endif #define _BSD #include #include #include #include #include "QF/cvar.h" #include "QF/darray.h" #define IMPLEMENT_INPUT_Funcs #include "QF/input.h" #include "QF/joystick.h" #include "QF/keys.h" #include "QF/mathlib.h" #include "QF/plist.h" #include "QF/sys.h" #include "QF/vid.h" #include "QF/input/event.h" #include "qfselect.h" typedef struct { in_driver_t driver; void *data; } in_regdriver_t; static struct DARRAY_TYPE (in_regdriver_t) in_drivers = { .grow = 8 }; static struct DARRAY_TYPE (in_device_t) in_devices = { .grow = 8 }; int in_grab; static cvar_t in_grab_cvar = { .name = "in_grab", .description = "With this set to 1, quake will grab the mouse, preventing loss of " "input focus.", .default_value = "0", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_int, .value = &in_grab }, }; VISIBLE float in_amp; static cvar_t in_amp_cvar = { .name = "in_amp", .description = "global in_amp multiplier", .default_value = "1", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_float, .value = &in_amp }, }; VISIBLE float in_pre_amp; static cvar_t in_pre_amp_cvar = { .name = "in_pre_amp", .description = "global in_pre_amp multiplier", .default_value = "1", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_float, .value = &in_pre_amp }, }; int in_freelook; static cvar_t in_freelook_cvar = { .name = "freelook", .description = "force +mlook", .default_value = "0", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_int, .value = &in_freelook }, }; int in_mouse_filter; static cvar_t in_mouse_filter_cvar = { .name = "in_mouse_filter", .description = "Toggle mouse input filtering.", .default_value = "0", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_int, .value = &in_mouse_filter }, }; float in_mouse_amp; static cvar_t in_mouse_amp_cvar = { .name = "in_mouse_amp", .description = "mouse in_mouse_amp multiplier", .default_value = "15", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_float, .value = &in_mouse_amp }, }; float in_mouse_pre_amp; static cvar_t in_mouse_pre_amp_cvar = { .name = "in_mouse_pre_amp", .description = "mouse in_mouse_pre_amp multiplier", .default_value = "1", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_float, .value = &in_mouse_pre_amp }, }; int lookstrafe; static cvar_t lookstrafe_cvar = { .name = "lookstrafe", .description = "when mlook/klook on player will strafe", .default_value = "0", .flags = CVAR_ARCHIVE, .value = { .type = &cexpr_int, .value = &lookstrafe }, }; int64_t in_timeout = 10000;//10ms default timeout int IN_RegisterDriver (in_driver_t *driver, void *data) { in_regdriver_t rdriver = { *driver, data }; DARRAY_APPEND (&in_drivers, rdriver); return in_drivers.size - 1; } void IN_DriverData (int handle, void *data) { in_drivers.a[handle].data = data; } static int in_add_device (int driver, void *device, const char *name, const char *id) { size_t devid; in_device_t indev = { .driverid = driver, .device = device, .name = name, .id = id, }; for (devid = 0; devid < in_devices.size; devid++) { if (in_devices.a[devid].driverid == -1) { in_devices.a[devid] = indev; return devid; } } DARRAY_APPEND (&in_devices, indev); return devid; } int IN_AddDevice (int driver, void *device, const char *name, const char *id) { if ((size_t) driver >= in_drivers.size) { Sys_Error ("IN_AddDevice: invalid driver: %d", driver); } int devid = in_add_device (driver, device, name, id); IE_event_t event = { .type = ie_add_device, .when = Sys_LongTime (), .device = { .devid = devid, }, }; IE_Send_Event (&event); return devid; } void IN_RemoveDevice (int devid) { if ((size_t) devid >= in_devices.size) { Sys_Error ("IN_RemoveDevice: invalid devid: %d", devid); } IE_event_t event = { .type = ie_remove_device, .when = Sys_LongTime (), .device = { .devid = devid, }, }; IE_Send_Event (&event); in_devices.a[devid].device = 0; } void IN_SendConnectedDevices (void) { for (size_t devid = 0; devid < in_devices.size; devid++) { if (in_devices.a[devid].driverid >= 0 && in_devices.a[devid].device) { IE_event_t event = { .type = ie_add_device, .when = Sys_LongTime (),//FIXME actual time? .device = { .devid = devid, }, }; IE_Send_Event (&event); } } } int IN_FindDeviceId (const char *id) { for (size_t i = 0; i < in_devices.size; i++) { if (strcmp (id, in_devices.a[i].id) == 0) { return i; } } return -1; } const char * IN_GetDeviceName (int devid) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } return in_devices.a[devid].name; } const char * IN_GetDeviceId (int devid) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } return in_devices.a[devid].id; } void IN_SetDeviceEventData (int devid, void *data) { if ((size_t) devid >= in_devices.size) { return; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return; } in_regdriver_t *rd = &in_drivers.a[in_devices.a[devid].driverid]; rd->driver.set_device_event_data (in_devices.a[devid].device, data, rd->data); } void * IN_GetDeviceEventData (int devid) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } in_regdriver_t *rd = &in_drivers.a[in_devices.a[devid].driverid]; return rd->driver.get_device_event_data (in_devices.a[devid].device, rd->data); } int IN_AxisInfo (int devid, in_axisinfo_t *axes, int *numaxes) { if ((size_t) devid >= in_devices.size) { return -1; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return -1; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (!rd->driver.axis_info) { return -1; } rd->driver.axis_info (rd->data, in_devices.a[devid].device, axes, numaxes); if (axes) { for (int i = 0; i < *numaxes; i++) { axes[i].deviceid = devid; } } return 0; } int IN_ButtonInfo (int devid, in_buttoninfo_t *buttons, int *numbuttons) { if ((size_t) devid >= in_devices.size) { return -1; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return -1; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (!rd->driver.button_info) { return -1; } rd->driver.button_info (rd->data, in_devices.a[devid].device, buttons, numbuttons); if (buttons) { for (int i = 0; i < *numbuttons; i++) { buttons[i].deviceid = devid; } } return 0; } const char * IN_GetAxisName (int devid, int axis_num) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (!rd->driver.get_axis_name) { return 0; } return rd->driver.get_axis_name (rd->data, in_devices.a[devid].device, axis_num); } const char * IN_GetButtonName (int devid, int button_num) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (!rd->driver.get_button_name) { return 0; } return rd->driver.get_button_name (rd->data, in_devices.a[devid].device, button_num); } int IN_GetAxisNumber (int devid, const char *axis_name) { if ((size_t) devid >= in_devices.size) { return -1; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return -1; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (!rd->driver.get_axis_num) { return -1; } return rd->driver.get_axis_num (rd->data, in_devices.a[devid].device, axis_name); } int IN_GetButtonNumber (int devid, const char *button_name) { if ((size_t) devid >= in_devices.size) { return -1; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return -1; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (!rd->driver.get_button_num) { return -1; } return rd->driver.get_button_num (rd->data, in_devices.a[devid].device, button_name); } int IN_GetAxisInfo (int devid, int axis_num, in_axisinfo_t *info) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (rd->driver.get_axis_info (rd->data, in_devices.a[devid].device, axis_num, info)) { info->deviceid = devid; return 1; } return 0; } int IN_GetButtonInfo (int devid, int button_num, in_buttoninfo_t *info) { if ((size_t) devid >= in_devices.size) { return 0; } if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) { return 0; } int driver = in_devices.a[devid].driverid; in_regdriver_t *rd = &in_drivers.a[driver]; if (rd->driver.get_button_info (rd->data, in_devices.a[devid].device, button_num, info)) { info->deviceid = devid; return 1; } return 0; } void IN_UpdateGrab (int grab) { for (size_t i = 0; i < in_drivers.size; i++) { in_regdriver_t *rd = &in_drivers.a[i]; if (rd->driver.grab_input) { rd->driver.grab_input (rd->data, grab); } } } static void in_grab_f (void *data, const cvar_t *cvar) { IN_UpdateGrab (in_grab); } void IN_ProcessEvents (void) { qf_fd_set fdset; int maxfd = -1; int64_t timeout = in_timeout; QF_FD_ZERO (&fdset); for (size_t i = 0; i < in_drivers.size; i++) { in_regdriver_t *rd = &in_drivers.a[i]; if (rd->driver.add_select) { rd->driver.add_select (&fdset, &maxfd, rd->data); } if (rd->driver.process_events) { rd->driver.process_events (rd->data); // if a driver can't use select, then we can't block in select timeout = 0; } } if (maxfd >= 0 && Sys_Select (maxfd, &fdset, timeout) > 0) { for (size_t i = 0; i < in_drivers.size; i++) { in_regdriver_t *rd = &in_drivers.a[i]; if (rd->driver.check_select) { rd->driver.check_select (&fdset, rd->data); } } } } void IN_SaveConfig (plitem_t *config) { plitem_t *input_config = PL_NewDictionary (0); PL_D_AddObject (config, "input", input_config); IMT_SaveConfig (input_config); IN_Binding_SaveConfig (input_config); } void IN_LoadConfig (plitem_t *config) { plitem_t *input_config = PL_ObjectForKey (config, "input"); if (input_config) { IMT_LoadConfig (input_config); IN_Binding_LoadConfig (input_config); } } /* Called at shutdown */ static void IN_shutdown (void *data) { //JOY_Shutdown (); Sys_MaskPrintf (SYS_vid, "IN_Shutdown\n"); for (size_t i = in_drivers.size; i-- > 0; ) { in_regdriver_t *rd = &in_drivers.a[i]; if (rd->driver.shutdown) { rd->driver.shutdown (rd->data); } } IN_Binding_Shutdown (); IMT_Shutdown (); DARRAY_CLEAR (&in_drivers); DARRAY_CLEAR (&in_devices); } void IN_Init (void) { Sys_RegisterShutdown (IN_shutdown, 0); IMT_Init (); IN_Binding_Init (); for (size_t i = 0; i < in_drivers.size; i++) { in_regdriver_t *rd = &in_drivers.a[i]; rd->driver.init (rd->data); } } void IN_Init_Cvars (void) { Cvar_Register (&in_grab_cvar, in_grab_f, 0); Cvar_Register (&in_amp_cvar, 0, 0); Cvar_Register (&in_pre_amp_cvar, 0, 0); Cvar_Register (&in_freelook_cvar, 0, 0); Cvar_Register (&in_mouse_filter_cvar, 0, 0); Cvar_Register (&in_mouse_amp_cvar, 0, 0); Cvar_Register (&in_mouse_pre_amp_cvar, 0, 0); Cvar_Register (&lookstrafe_cvar, 0, 0); for (size_t i = 0; i < in_drivers.size; i++) { in_regdriver_t *rd = &in_drivers.a[i]; if (rd->driver.init_cvars) { rd->driver.init_cvars (rd->data); } } } void IN_ClearStates (void) { for (size_t i = 0; i < in_drivers.size; i++) { in_regdriver_t *rd = &in_drivers.a[i]; if (rd->driver.clear_states) { rd->driver.clear_states (rd->data); } } IN_AxisClearStates (); IN_ButtonClearStates (); } #ifdef HAVE_EVDEV extern int in_evdev_force_link; static __attribute__((used)) int *evdev_force_link = &in_evdev_force_link; #endif