/* RetroArch - A frontend for libretro. * Copyright (C) 2010-2014 - Hans-Kristian Arntzen * Copyright (C) 2011-2016 - 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 #include #include #include #include #include #include "input_config.h" #include "input_autodetect.h" #include "../configuration.h" #include "../list_special.h" #include "../runloop.h" #include "../verbosity.h" /* Adds an index for devices with the same name, * so they can be identified in the GUI. */ static void input_reindex_devices(void) { unsigned i; settings_t *settings = config_get_ptr(); for(i = 0; i < settings->input.max_users; i++) settings->input.device_name_index[i]=0; for(i = 0; i < settings->input.max_users; i++) { unsigned j; const char *tmp = settings->input.device_names[i]; int k = 1; for(j = 0; j < settings->input.max_users; j++) { if(string_is_equal(tmp, settings->input.device_names[j]) && settings->input.device_name_index[i] == 0) settings->input.device_name_index[j]=k++; } } } 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_get_base(i), &binds[i]); input_config_parse_joy_axis(conf, "input", input_config_bind_map_get_base(i), &binds[i]); } } static int input_try_autoconfigure_joypad_from_conf(config_file_t *conf, autoconfig_params_t *params) { int tmp_int; char ident[PATH_MAX_LENGTH] = {0}; char input_driver[PATH_MAX_LENGTH] = {0}; int input_vid = 0; int input_pid = 0; int score = 0; if (!conf) return false; config_get_array(conf, "input_device", ident, sizeof(ident)); config_get_array(conf, "input_driver", input_driver, sizeof(input_driver)); if (config_get_int (conf, "input_vendor_id", &tmp_int)) input_vid = tmp_int; if (config_get_int (conf, "input_product_id", &tmp_int)) input_pid = tmp_int; /* Check for VID/PID */ if ( (params->vid == input_vid) && (params->pid == input_pid) && params->vid != 0 && params->pid != 0 && input_vid != 0 && input_pid != 0) { score += 3; #if 0 RARCH_LOG("Autodetect: VID/PID match score=%d\n", score); #endif } /* Check for name match */ if (string_is_equal(ident, params->name)) { score += 2; #if 0 RARCH_LOG("Autodetect: exact name match score=%d\n", score); #endif } else { if (!string_is_empty(ident) && !strncmp(params->name, ident, strlen(ident))) { score += 1; #if 0 RARCH_LOG("Autodetect: partial name match score=%d\n", score); #endif } } #if 0 RARCH_LOG("Autodetect: configuration file: %s score: %d\n", conf->path, score); #endif return score; } static void input_autoconfigure_joypad_add(config_file_t *conf, autoconfig_params_t *params) { bool block_osd_spam; static bool remote_is_bound = false; char msg[PATH_MAX_LENGTH] = {0}; char display_name[PATH_MAX_LENGTH] = {0}; char device_type[PATH_MAX_LENGTH] = {0}; settings_t *settings = config_get_ptr(); config_get_array(conf, "input_device_display_name", display_name, sizeof(display_name)); config_get_array(conf, "input_device_type", device_type, sizeof(device_type)); if (!settings) return; /* This will be the case if input driver is reinitialized. * No reason to spam autoconfigure messages every time. */ block_osd_spam = settings->input.autoconfigured[params->idx] && *params->name; settings->input.autoconfigured[params->idx] = true; input_autoconfigure_joypad_conf(conf, settings->input.autoconf_binds[params->idx]); if (string_is_equal(device_type,"remote")) { snprintf(msg, sizeof(msg), "%s configured", string_is_empty(display_name) ? params->name : display_name); if(!remote_is_bound) runloop_msg_queue_push(msg, 0, 60, false); remote_is_bound = true; } else { snprintf(msg, sizeof(msg), "%s configured in port #%u.", string_is_empty(display_name) ? params->name : display_name, params->idx); if (!block_osd_spam) runloop_msg_queue_push(msg, 0, 60, false); } input_reindex_devices(); #if 0 RARCH_LOG("Autodetect: %s\n", msg); #endif } #if defined(HAVE_BUILTIN_AUTOCONFIG) static int input_autoconfigure_joypad_from_conf( config_file_t *conf, autoconfig_params_t *params) { int ret = 0; if (!conf) return false; ret = input_try_autoconfigure_joypad_from_conf(conf, params); if (ret) input_autoconfigure_joypad_add(conf, params); config_file_free(conf); return ret; } #endif static bool input_autoconfigure_joypad_from_conf_dir( autoconfig_params_t *params) { size_t i; char path[PATH_MAX_LENGTH] = {0}; int ret = 0; int index = -1; int current_best = 0; config_file_t *conf = NULL; struct string_list *list = NULL; settings_t *settings = config_get_ptr(); if (!settings) return false; fill_pathname_join(path, settings->directory.autoconfig, settings->input.joypad_driver, sizeof(path)); list = dir_list_new_special(path, DIR_LIST_AUTOCONFIG, "cfg"); if (!list || !list->size) { if (list) string_list_free(list); list = dir_list_new_special(settings->directory.autoconfig, DIR_LIST_AUTOCONFIG, "cfg"); } if(!list) return false; RARCH_LOG("Autodetect: %d profiles found\n", list->size); for (i = 0; i < list->size; i++) { conf = config_file_new(list->elems[i].data); ret = input_try_autoconfigure_joypad_from_conf(conf, params); if(ret >= current_best) { index = i; current_best = ret; } config_file_free(conf); } if(index >= 0 && current_best > 0) { conf = config_file_new(list->elems[index].data); if (conf) { char conf_path[PATH_MAX_LENGTH]; config_get_config_path(conf, conf_path, sizeof(conf_path)); RARCH_LOG("Autodetect: selected configuration: %s\n", conf_path); input_autoconfigure_joypad_add(conf, params); config_file_free(conf); ret = 1; } } else ret = 0; string_list_free(list); if (ret == 0) return false; return true; } #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->directory.autoconfig) 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; } bool input_config_autoconfigure_joypad(autoconfig_params_t *params) { char msg[PATH_MAX_LENGTH]; if (!input_config_autoconfigure_joypad_init(params)) goto error; if (!*params->name) goto error; if (input_autoconfigure_joypad_from_conf_dir(params)) return true; #if defined(HAVE_BUILTIN_AUTOCONFIG) if (input_autoconfigure_joypad_from_conf_internal(params)) return true; #endif RARCH_LOG("Autodetect: no profiles found for %s (%d/%d)\n", params->name, params->vid, params->pid); snprintf(msg, sizeof(msg), "%s (%ld/%ld) not configured", params->name, (long)params->vid, (long)params->pid); runloop_msg_queue_push(msg, 0, 60, false); error: return false; } 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; } void input_config_autoconfigure_disconnect(unsigned i, const char *ident) { char msg[PATH_MAX_LENGTH]; snprintf(msg, sizeof(msg), "Device #%u (%s) disconnected.", i, ident); runloop_msg_queue_push(msg, 0, 60, false); RARCH_LOG("Autodetect: %s\n", msg); }