RetroArch/input/input_autodetect.c

268 lines
7.1 KiB
C
Raw Normal View History

/* RetroArch - A frontend for libretro.
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2015-01-07 17:46:50 +01:00
* 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 <http://www.gnu.org/licenses/>.
*/
#include "input_common.h"
#include "input_autodetect.h"
2014-10-22 01:13:05 +02:00
#include <file/dir_list.h>
#include <string.h>
#include <stdlib.h>
#include <ctype.h>
#include "../general.h"
/*enum
2015-04-03 01:22:55 +02:00
{
AUTODETECT_MATCH_NONE = 0,
AUTODETECT_MATCH_VID,
AUTODETECT_MATCH_PID,
AUTODETECT_MATCH_IDENT,
AUTODETECT_MATCH_DRIVER,
2015-06-26 18:35:35 +02:00
AUTODETECT_MATCH_NAME
};*/
2015-04-03 01:22:55 +02:00
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]);
}
}
static int input_try_autoconfigure_joypad_from_conf(config_file_t *conf,
autoconfig_params_t *params)
{
2015-06-12 23:52:52 +02:00
char ident[PATH_MAX_LENGTH] = {0};
//char ident_idx[PATH_MAX_LENGTH] = {0};
2015-06-12 23:52:52 +02:00
char input_driver[PATH_MAX_LENGTH] = {0};
int input_vid = 0;
int input_pid = 0;
int score = 0;
2015-01-09 18:04:29 +01:00
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));
2015-03-27 17:27:21 +01:00
config_get_int (conf, "input_vendor_id", &input_vid);
config_get_int (conf, "input_product_id", &input_pid);
/* Check for VID/PID */
2015-03-27 17:27:21 +01:00
if ( (params->vid == input_vid)
&& (params->pid == input_pid)
&& params->vid != 0
&& params->pid != 0
&& input_vid != 0
&& input_pid != 0)
{
score += 3;
RARCH_LOG("Autoconf: VID/PID match score=%d\n", score);
}
/* Check for name match */
if (!strcmp(ident, params->name))
{
score += 2;
RARCH_LOG("Autoconf: exact name match score=%d\n", score);
}
else
{
if (ident[0] != '\0' && !strncmp(params->name, ident, strlen(ident)))
{
score += 1;
RARCH_LOG("Autoconf: partial name match score=%d\n", score);
}
}
RARCH_LOG("Autoconf: configuration score=%d\n", score);
return score;
2015-03-27 17:27:21 +01:00
}
static void input_autoconfigure_joypad_add(
config_file_t *conf,
2015-03-27 17:31:59 +01:00
autoconfig_params_t *params)
2015-03-27 17:27:21 +01:00
{
2015-06-12 23:52:52 +02:00
char msg[PATH_MAX_LENGTH] = {0};
settings_t *settings = config_get_ptr();
2015-03-27 17:34:09 +01:00
/* This will be the case if input driver is reinitialized.
* No reason to spam autoconfigure messages every time. */
bool block_osd_spam = settings &&
2015-03-28 22:20:05 +01:00
settings->input.autoconfigured[params->idx] && *params->name;
2015-03-27 17:27:21 +01:00
if (!settings)
return;
2015-03-27 16:57:58 +01:00
settings->input.autoconfigured[params->idx] = true;
2015-03-27 17:27:21 +01:00
input_autoconfigure_joypad_conf(conf,
settings->input.autoconf_binds[params->idx]);
snprintf(msg, sizeof(msg), "Device port #%u (%s) configured.",
2015-03-27 16:57:58 +01:00
params->idx, params->name);
if (!block_osd_spam)
rarch_main_msg_queue_push(msg, 0, 60, false);
RARCH_LOG("%s\n", msg);
}
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;
}
2015-03-27 18:05:43 +01:00
static bool input_autoconfigure_joypad_from_conf_dir(
autoconfig_params_t *params)
{
size_t i;
int ret = 0;
2015-03-20 21:22:38 +01:00
settings_t *settings = config_get_ptr();
struct string_list *list = settings ? dir_list_new(
settings->input.autoconfig_dir, "cfg", false) : NULL;
if (!list)
2015-03-27 18:05:43 +01:00
return false;
int current_best = 0;
int index = 0;
config_file_t *conf;
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);
}
conf = config_file_new(list->elems[index].data);
input_autoconfigure_joypad_add(conf, params);
config_file_free(conf);
string_list_free(list);
2015-03-27 18:05:43 +01:00
return ret !=0 ? true : false;
}
#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;
2015-03-27 17:55:00 +01:00
/* Load internal autoconfig files */
for (i = 0; input_builtin_autoconfs[i]; i++)
{
2015-03-27 17:47:15 +01:00
config_file_t *conf = config_file_new_from_string(
input_builtin_autoconfs[i]);
if ((ret = input_autoconfigure_joypad_from_conf(conf, params)))
break;
}
2015-03-27 17:47:15 +01:00
if (ret || !*settings->input.autoconfig_dir)
return true;
return false;
}
#endif
2015-03-27 18:05:43 +01:00
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)
2015-03-27 18:05:43 +01:00
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;
2015-03-27 18:05:43 +01:00
return true;
}
bool input_config_autoconfigure_joypad(autoconfig_params_t *params)
2015-03-27 18:05:43 +01:00
{
bool ret = false;
2015-03-27 18:05:43 +01:00
if (!input_config_autoconfigure_joypad_init(params))
return ret;
2015-03-27 18:05:43 +01:00
2015-03-28 22:20:05 +01:00
if (!*params->name)
return ret;
#if defined(HAVE_BUILTIN_AUTOCONFIG)
ret = input_autoconfigure_joypad_from_conf_internal(params);
#endif
if (!ret)
2015-03-27 18:05:43 +01:00
ret = input_autoconfigure_joypad_from_conf_dir(params);
return ret;
}
const struct retro_keybind *input_get_auto_bind(unsigned port, unsigned id)
{
2015-03-20 21:22:38 +01:00
settings_t *settings = config_get_ptr();
2015-03-27 17:34:09 +01:00
unsigned joy_idx = 0;
2015-03-27 17:34:09 +01:00
if (settings)
joy_idx = settings->input.joypad_map[port];
2015-01-05 01:58:00 +01:00
if (joy_idx < MAX_USERS)
2015-03-20 21:22:38 +01:00
return &settings->input.autoconf_binds[joy_idx][id];
return NULL;
}
void input_config_autoconfigure_disconnect(unsigned i, const char *ident)
{
2015-06-12 23:52:52 +02:00
char msg[PATH_MAX_LENGTH] = {0};
snprintf(msg, sizeof(msg), "Device #%u (%s) disconnected.", i, ident);
rarch_main_msg_queue_push(msg, 0, 60, false);
RARCH_LOG("%s\n", msg);
}