/* RetroArch - A frontend for libretro. * Copyright (C) 2013-2014 - Jason Fetters * 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 . */ #ifndef _JOYPAD_CONNECTION_H #define _JOYPAD_CONNECTION_H #include #include #include #include #include #include "../input_driver.h" /* Gekko (NGC/Wii) has PID/VID already swapped by USB_GetDescriptors from libogc, so skip bigendian byteswap */ #if defined(MSB_FIRST) && !defined(GEKKO) #define SWAP_IF_BIG(val) ((((val) & 0x00ff) << 8) | (((val) & 0xff00) >> 8)) #else #define SWAP_IF_BIG(val) (val) #endif #define VID_NONE 0x0000 #define VID_NINTENDO SWAP_IF_BIG(0x057e) #define VID_SONY SWAP_IF_BIG(0x054c) #define VID_MICRONTEK SWAP_IF_BIG(0x0079) #define VID_PCS SWAP_IF_BIG(0x0810) #define VID_PS3_CLONE SWAP_IF_BIG(0x0313) #define VID_SNES_CLONE SWAP_IF_BIG(0x081f) #define VID_RETRODE SWAP_IF_BIG(0x0403) #define VID_HORI_1 SWAP_IF_BIG(0x0f0d) #define VID_KADE SWAP_IF_BIG(0x10c4) #define VID_DRAGONRISE SWAP_IF_BIG(0x0079) #define PID_NONE 0x0000 #define PID_NINTENDO_PRO SWAP_IF_BIG(0x0330) #define PID_SONY_DS3 SWAP_IF_BIG(0x0268) #define PID_SONY_DS4 SWAP_IF_BIG(0x05c4) #define PID_SONY_DS4_R2 SWAP_IF_BIG(0x09cc) #define PID_DS3_CLONE SWAP_IF_BIG(0x20d6) #define PID_SNES_CLONE SWAP_IF_BIG(0xe401) #define PID_MICRONTEK_NES SWAP_IF_BIG(0x0011) #define PID_NINTENDO_GCA SWAP_IF_BIG(0x0337) #define PID_PCS_PS2PSX SWAP_IF_BIG(0x0001) #define PID_PCS_PSX2PS3 SWAP_IF_BIG(0x0003) #define PID_RETRODE SWAP_IF_BIG(0x97c1) #define PID_HORI_MINI_WIRED_PS4 SWAP_IF_BIG(0x00ee) #define PID_KADE SWAP_IF_BIG(0x82c0) #define PID_DRAGONRISE SWAP_IF_BIG(0x0006) struct joypad_connection { struct pad_connection_interface *iface; input_device_driver_t *input_driver; void* data; void* connection; bool connected; }; #define PAD_CONNECT_OFFLINE 0x00 /* the pad is offline and cannot be used */ #define PAD_CONNECT_READY 0x01 /* the pad is ready but is not bound to a RA slot */ #define PAD_CONNECT_BOUND 0x02 /* the pad is offline and is bound to a RA slot */ #define PAD_CONNECT_IN_USE 0x03 /* the pad is ready and is bound to a RA slot */ #define SLOT_AUTO -1 typedef struct pad_connection_interface { void* (*init)(void *data, uint32_t slot, hid_driver_t *driver); void (*deinit)(void* device); void (*packet_handler)(void* device, uint8_t *packet, uint16_t size); void (*set_rumble)(void* device, enum retro_rumble_effect effect, uint16_t strength); void (*get_buttons)(void *data, input_bits_t *state); int16_t (*get_axis)(void *data, unsigned axis); const char* (*get_name)(void *data); int32_t (*button)(void *data, uint16_t joykey); /* all fields/methods below this point are only required for multi-pad devices */ bool multi_pad; /* does the device provide multiple pads? */ int8_t max_pad; /* number of pads this device can provide */ void* (*pad_init)(void *data, int pad_index, joypad_connection_t *joyconn); void (*pad_deinit)(void *pad_data); /* pad_index is a number from 0 to max_pad-1 */ int8_t (*status)(void *data, int pad_index); /* returns a PAD_CONNECT_* state */ joypad_connection_t* (*joypad)(void *device_data, int pad_index); } pad_connection_interface_t; typedef struct joypad_connection_entry { const char* name; uint16_t vid; uint16_t pid; pad_connection_interface_t *iface; } joypad_connection_entry_t; extern pad_connection_interface_t pad_connection_wii; extern pad_connection_interface_t pad_connection_wiiupro; extern pad_connection_interface_t pad_connection_ps3; extern pad_connection_interface_t pad_connection_ps4; extern pad_connection_interface_t pad_connection_snesusb; extern pad_connection_interface_t pad_connection_nesusb; extern pad_connection_interface_t pad_connection_wiiugca; extern pad_connection_interface_t pad_connection_ps2adapter; extern pad_connection_interface_t pad_connection_psxadapter; extern pad_connection_interface_t pad_connection_retrode; extern pad_connection_interface_t pad_connection_ps4_hori_mini; extern pad_connection_interface_t pad_connection_kade; extern pad_connection_interface_t pad_connection_dragonrise; int32_t pad_connection_pad_init(joypad_connection_t *joyconn, const char* name, uint16_t vid, uint16_t pid, void *data, hid_driver_t *driver); joypad_connection_t *pad_connection_init(unsigned pads); void pad_connection_destroy(joypad_connection_t *joyconn); void pad_connection_pad_deinit(joypad_connection_t *joyconn, uint32_t idx); void pad_connection_packet(joypad_connection_t *joyconn, uint32_t idx, uint8_t* data, uint32_t length); void pad_connection_get_buttons(joypad_connection_t *joyconn, unsigned idx, input_bits_t* state); int16_t pad_connection_get_axis(joypad_connection_t *joyconn, unsigned idx, unsigned i); /* Determine if connected joypad is a hidpad backed device. * If false, pad_connection_packet cannot be used */ bool pad_connection_has_interface(joypad_connection_t *joyconn, unsigned idx); int pad_connection_find_vacant_pad(joypad_connection_t *joyconn); bool pad_connection_rumble(joypad_connection_t *s, unsigned pad, enum retro_rumble_effect effect, uint16_t strength); const char* pad_connection_get_name(joypad_connection_t *joyconn, unsigned idx); joypad_connection_entry_t *find_connection_entry(uint16_t vid, uint16_t pid, const char *name); int32_t pad_connection_pad_init_entry(joypad_connection_t *joyconn, joypad_connection_entry_t *entry, void *data, hid_driver_t *driver); void pad_connection_pad_register(joypad_connection_t *joyconn, pad_connection_interface_t *iface, void *pad_data, void *handle, input_device_driver_t *input_driver, int slot); void pad_connection_pad_deregister(joypad_connection_t *joyconn, pad_connection_interface_t *iface, void *pad_data); void pad_connection_pad_refresh(joypad_connection_t *joyconn, pad_connection_interface_t *iface, void *device_data, void *handle, input_device_driver_t *input_driver); #endif