/* RetroArch - A frontend for libretro. * Copyright (C) 2010-2014 - Hans-Kristian Arntzen * Copyright (C) 2011-2015 - 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 . */ #include "input_common.h" #include "input_autodetect.h" #include #include #include #include #include "../general.h" static void input_autoconfigure_joypad_conf(config_file_t *conf, struct retro_keybind *binds) { unsigned i; for (i = 0; i < RARCH_BIND_LIST_END; i++) { input_config_parse_joy_button(conf, "input", input_config_bind_map[i].base, &binds[i]); input_config_parse_joy_axis(conf, "input", input_config_bind_map[i].base, &binds[i]); } } enum { AUTODETECT_MATCH_NONE = 0, AUTODETECT_MATCH_VID, AUTODETECT_MATCH_PID, AUTODETECT_MATCH_IDENT, AUTODETECT_MATCH_NAME, }; static bool input_try_autoconfigure_joypad_from_conf(config_file_t *conf, autoconfig_params_t *params, unsigned *match) { char ident[PATH_MAX_LENGTH], ident_idx[PATH_MAX_LENGTH]; char input_driver[PATH_MAX_LENGTH]; int input_vid = 0, input_pid = 0; bool ret = false; if (!conf) return false; *ident = *input_driver = '\0'; config_get_array(conf, "input_device", ident, sizeof(ident)); config_get_array(conf, "input_driver", input_driver, sizeof(input_driver)); config_get_int (conf, "input_vendor_id", &input_vid); config_get_int (conf, "input_product_id", &input_pid); snprintf(ident_idx, sizeof(ident_idx), "%s_p%u", ident, params->idx); /* If Vendor ID and Product ID matches, we've found our * entry. */ if ( (params->vid == input_vid) && (params->pid == input_pid) && params->vid != 0 && params->pid != 0 && input_vid != 0 && input_pid != 0) { BIT32_SET(*match, AUTODETECT_MATCH_VID); BIT32_SET(*match, AUTODETECT_MATCH_PID); ret = true; } /* Check for name match. */ if (!strcmp(ident_idx, params->name)) { BIT32_SET(*match, AUTODETECT_MATCH_NAME); ret = true; } if (!strcmp(ident, params->name) && !strcmp(params->driver, input_driver)) { BIT32_SET(*match, AUTODETECT_MATCH_IDENT); ret = true; } return ret; } static void input_autoconfigure_joypad_add( config_file_t *conf, autoconfig_params_t *params) { char msg[PATH_MAX_LENGTH]; settings_t *settings = config_get_ptr(); /* This will be the case if input driver is reinitialized. * No reason to spam autoconfigure messages every time. */ bool block_osd_spam = settings && settings->input.autoconfigured[params->idx] && *params->name; if (!settings) return; settings->input.autoconfigured[params->idx] = true; input_autoconfigure_joypad_conf(conf, settings->input.autoconf_binds[params->idx]); snprintf(msg, sizeof(msg), "Joypad port #%u (%s) configured.", params->idx, params->name); if (!block_osd_spam) rarch_main_msg_queue_push(msg, 0, 60, false); RARCH_LOG("%s\n", msg); } static bool input_autoconfigure_joypad_from_conf( config_file_t *conf, autoconfig_params_t *params) { bool ret = false; uint32_t match = 0; if (!conf) return false; ret = input_try_autoconfigure_joypad_from_conf(conf, params, &match); if (ret) input_autoconfigure_joypad_add(conf, params); config_file_free(conf); return ret; } static bool input_autoconfigure_joypad_from_conf_dir( autoconfig_params_t *params) { size_t i; bool ret = false; settings_t *settings = config_get_ptr(); struct string_list *list = settings ? dir_list_new( settings->input.autoconfig_dir, "cfg", false) : NULL; if (!list) return false; for (i = 0; i < list->size; i++) { config_file_t *conf = config_file_new(list->elems[i].data); if ((ret = input_autoconfigure_joypad_from_conf(conf, params))) break; } string_list_free(list); return ret; } #if defined(HAVE_BUILTIN_AUTOCONFIG) static bool input_autoconfigure_joypad_from_conf_internal( autoconfig_params_t *params) { size_t i; settings_t *settings = config_get_ptr(); bool ret = false; /* Load internal autoconfig files */ for (i = 0; input_builtin_autoconfs[i]; i++) { config_file_t *conf = config_file_new_from_string( input_builtin_autoconfs[i]); if ((ret = input_autoconfigure_joypad_from_conf(conf, params))) break; } if (ret || !*settings->input.autoconfig_dir) return true; return false; } #endif static bool input_config_autoconfigure_joypad_init(autoconfig_params_t *params) { size_t i; settings_t *settings = config_get_ptr(); if (!settings || !settings->input.autodetect_enable) return false; for (i = 0; i < RARCH_BIND_LIST_END; i++) { settings->input.autoconf_binds[params->idx][i].joykey = NO_BTN; settings->input.autoconf_binds[params->idx][i].joyaxis = AXIS_NONE; settings->input.autoconf_binds[params->idx][i].joykey_label[0] = '\0'; settings->input.autoconf_binds[params->idx][i].joyaxis_label[0] = '\0'; } settings->input.autoconfigured[params->idx] = false; return true; } void input_config_autoconfigure_joypad(autoconfig_params_t *params) { bool ret = false; if (!input_config_autoconfigure_joypad_init(params)) return; if (!*params->name) return; #if defined(HAVE_BUILTIN_AUTOCONFIG) ret = input_autoconfigure_joypad_from_conf_internal(params); #endif if (!ret) ret = input_autoconfigure_joypad_from_conf_dir(params); } const struct retro_keybind *input_get_auto_bind(unsigned port, unsigned id) { settings_t *settings = config_get_ptr(); unsigned joy_idx = 0; if (settings) joy_idx = settings->input.joypad_map[port]; if (joy_idx < MAX_USERS) return &settings->input.autoconf_binds[joy_idx][id]; return NULL; }