RetroArch/tasks/task_autodetect.c

558 lines
15 KiB
C
Raw Normal View History

/* RetroArch - A frontend for libretro.
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
2017-01-22 13:40:32 +01:00
* Copyright (C) 2011-2017 - Daniel De Matteis
2019-02-22 16:31:54 -05:00
* Copyright (C) 2016-2019 - Brad Parker
* Copyright (C) 2016-2019 - Andrés Suárez
*
* 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 <string.h>
#include <stdlib.h>
#include <ctype.h>
2015-07-10 09:15:55 +02:00
#include <compat/strl.h>
2015-07-09 23:12:35 -05:00
#include <file/file_path.h>
2015-12-26 07:54:17 +01:00
#include <string/stdstring.h>
2016-02-07 13:25:55 +01:00
#include "../configuration.h"
2016-06-11 21:51:28 +02:00
#include "../file_path_special.h"
#include "../list_special.h"
#include "../retroarch.h"
#include "tasks_internal.h"
#ifdef HAVE_BLISSBOX
#include "../input/include/blissbox.h"
#endif
2017-05-13 22:30:45 +02:00
typedef struct autoconfig_disconnect autoconfig_disconnect_t;
struct autoconfig_disconnect
2016-12-01 22:36:38 +01:00
{
unsigned idx;
2017-09-29 06:16:35 +02:00
char *msg;
2017-05-13 22:30:45 +02:00
};
2016-12-01 22:36:38 +01:00
static bool input_autoconfigured[MAX_USERS];
2017-12-05 22:03:56 +00:00
static unsigned input_device_name_index[MAX_INPUT_DEVICES];
static bool input_autoconfigure_swap_override;
2019-02-03 15:49:35 -08:00
/* TODO/FIXME - Not thread safe to access this
2018-11-28 11:02:36 +01:00
* on main thread as well in its current state -
* menu_input.c - menu_event calls this function
* right now, while the underlying variable can
* be modified by a task thread. */
bool input_autoconfigure_get_swap_override(void)
{
return input_autoconfigure_swap_override;
}
/* Adds an index for devices with the same name,
* so they can be identified in the GUI. */
2019-04-30 13:43:01 +02:00
void input_autoconfigure_joypad_reindex_devices(void)
{
2017-12-05 22:03:56 +00:00
unsigned i, j, k;
2015-11-25 19:27:33 +01:00
2017-12-05 22:03:56 +00:00
for(i = 0; i < MAX_INPUT_DEVICES; i++)
input_device_name_index[i] = 0;
2015-11-25 19:27:33 +01:00
2017-12-05 22:03:56 +00:00
for(i = 0; i < MAX_INPUT_DEVICES; i++)
{
2017-04-25 15:53:30 +02:00
const char *tmp = input_config_get_device_name(i);
2017-12-05 22:03:56 +00:00
if ( !tmp || input_device_name_index[i] )
continue;
2017-12-05 22:03:56 +00:00
k = 2; /*Additional devices start at two*/
for(j = i+1; j < MAX_INPUT_DEVICES; j++)
{
2017-12-05 22:03:56 +00:00
const char *other = input_config_get_device_name(j);
if (!other)
continue;
/*another device with the same name found, for the first time*/
if (string_is_equal(tmp, other) &&
2017-12-05 22:03:56 +00:00
input_device_name_index[j]==0 )
{
/*Mark the first device of the set*/
input_device_name_index[i] = 1;
/*count this additional device, from two up*/
input_device_name_index[j] = k++;
2017-12-05 22:03:56 +00:00
}
}
}
}
2016-12-01 18:52:34 +01:00
static int input_autoconfigure_joypad_try_from_conf(config_file_t *conf,
autoconfig_params_t *params)
{
char ident[256];
char input_driver[32];
int tmp_int = 0;
int input_vid = 0;
int input_pid = 0;
int score = 0;
bool check_pid = false;
ident[0] = input_driver[0] = '\0';
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;
#ifdef HAVE_BLISSBOX
if (params->vid == BLISSBOX_VID)
input_pid = BLISSBOX_PID;
#endif
check_pid =
(params->vid == input_vid)
&& (params->pid == input_pid)
&& (params->vid != 0)
&& (params->pid != 0);
#ifdef HAVE_BLISSBOX
check_pid = check_pid
&& (params->vid != BLISSBOX_VID)
&& (params->pid != BLISSBOX_PID);
#endif
/* Check for VID/PID */
if (check_pid)
score += 3;
/* Check for name match */
if (
!string_is_empty(params->name)
2017-09-30 01:18:33 -04:00
&& !string_is_empty(ident)
2017-09-29 06:16:35 +02:00
&& string_is_equal(ident, params->name))
score += 2;
2016-09-12 18:39:46 +02:00
return score;
2015-03-27 17:27:21 +01:00
}
2016-02-05 14:06:43 +01:00
static void input_autoconfigure_joypad_add(config_file_t *conf,
autoconfig_params_t *params, retro_task_t *task)
2015-03-27 17:27:21 +01:00
{
2017-04-26 18:48:28 +02:00
char msg[128], display_name[128], device_type[128];
/* This will be the case if input driver is reinitialized.
* No reason to spam autoconfigure messages every time. */
2017-12-05 22:03:56 +00:00
bool block_osd_spam =
#if defined(HAVE_LIBNX) && defined(HAVE_MENU_WIDGETS)
true;
#else
2017-04-26 18:48:28 +02:00
input_autoconfigured[params->idx]
2017-09-30 06:08:09 +02:00
&& !string_is_empty(params->name);
#endif
2015-03-27 17:34:09 +01:00
msg[0] = display_name[0] = device_type[0] = '\0';
2016-02-05 14:06:43 +01:00
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));
input_autoconfigured[params->idx] = true;
2017-04-25 15:31:32 +02:00
2015-03-27 17:27:21 +01:00
input_autoconfigure_joypad_conf(conf,
input_autoconf_binds[params->idx]);
if (string_is_equal(device_type, "remote"))
2015-07-31 23:09:25 -05:00
{
2017-04-26 18:48:28 +02:00
static bool remote_is_bound = false;
2019-09-17 06:34:00 +02:00
const char *autoconfig_str = (string_is_empty(display_name) &&
!string_is_empty(params->name)) ? params->name : (!string_is_empty(display_name) ? display_name : "N/A");
strlcpy(msg, autoconfig_str, sizeof(msg));
strlcat(msg, " configured.", sizeof(msg));
if (!remote_is_bound)
{
task_free_title(task);
task_set_title(task, strdup(msg));
}
2015-07-31 23:09:25 -05:00
remote_is_bound = true;
if (params->idx == 0)
input_autoconfigure_swap_override = true;
2015-07-31 23:09:25 -05:00
}
else
2015-07-31 23:09:25 -05:00
{
2019-09-17 06:34:00 +02:00
bool tmp = false;
const char *autoconfig_str = (string_is_empty(display_name) &&
!string_is_empty(params->name))
2019-09-17 06:34:00 +02:00
? params->name : (!string_is_empty(display_name) ? display_name : "N/A");
snprintf(msg, sizeof(msg), "%s %s #%u.",
autoconfig_str,
2016-10-22 04:57:46 +02:00
msg_hash_to_str(MSG_DEVICE_CONFIGURED_IN_PORT),
2016-01-20 04:07:24 +01:00
params->idx);
2019-02-03 15:49:35 -08:00
/* allow overriding the swap menu controls for player 1*/
if (params->idx == 0)
2017-04-25 15:31:32 +02:00
{
if (config_get_bool(conf, "input_swap_override", &tmp))
input_autoconfigure_swap_override = tmp;
else
input_autoconfigure_swap_override = false;
2017-04-25 15:31:32 +02:00
}
2015-08-01 00:07:28 -05:00
if (!block_osd_spam)
{
task_free_title(task);
task_set_title(task, strdup(msg));
}
2015-07-31 23:09:25 -05:00
}
2018-01-12 10:29:35 -05:00
if (!string_is_empty(display_name))
input_config_set_device_display_name(params->idx, display_name);
else
input_config_set_device_display_name(params->idx, params->name);
2018-01-12 10:29:35 -05:00
if (!string_is_empty(conf->path))
{
2018-01-12 10:29:35 -05:00
input_config_set_device_config_name(params->idx, path_basename(conf->path));
input_config_set_device_config_path(params->idx, conf->path);
}
2018-01-12 10:29:35 -05:00
else
{
2018-01-12 10:29:35 -05:00
input_config_set_device_config_name(params->idx, "N/A");
input_config_set_device_config_path(params->idx, "N/A");
}
2017-12-05 22:03:56 +00:00
input_autoconfigure_joypad_reindex_devices();
}
static int input_autoconfigure_joypad_from_conf(
config_file_t *conf, autoconfig_params_t *params, retro_task_t *task)
{
2016-12-01 18:52:34 +01:00
int ret = input_autoconfigure_joypad_try_from_conf(conf,
params);
if (ret)
input_autoconfigure_joypad_add(conf, params, task);
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, retro_task_t *task)
{
size_t i;
char path[PATH_MAX_LENGTH];
char best_path[PATH_MAX_LENGTH];
2015-09-29 17:35:28 +02:00
int ret = 0;
int index = -1;
int current_best = 0;
config_file_t *best_conf = NULL;
struct string_list *list = NULL;
2015-07-09 23:12:35 -05:00
best_path[0] = '\0';
path[0] = '\0';
2016-06-11 21:51:28 +02:00
fill_pathname_application_special(path, sizeof(path),
APPLICATION_SPECIAL_DIRECTORY_AUTOCONFIG);
2015-09-29 17:35:28 +02:00
list = dir_list_new_special(path, DIR_LIST_AUTOCONFIG, "cfg");
if (!list || !list->size)
2016-05-23 21:28:43 +02:00
{
if (list)
{
2016-05-23 21:28:43 +02:00
string_list_free(list);
list = NULL;
}
2017-09-30 08:24:01 +02:00
if (!string_is_empty(params->autoconfig_directory))
2017-09-29 06:16:35 +02:00
list = dir_list_new_special(params->autoconfig_directory,
DIR_LIST_AUTOCONFIG, "cfg");
2016-05-23 21:28:43 +02:00
}
if (!list)
2015-03-27 18:05:43 +01:00
return false;
for (i = 0; i < list->size; i++)
{
int res;
config_file_t *conf = config_file_new_from_path_to_string(list->elems[i].data);
if (!conf)
continue;
res = input_autoconfigure_joypad_try_from_conf(conf, params);
2016-09-12 18:39:46 +02:00
if (res >= current_best)
{
index = (int)i;
current_best = res;
if (best_conf)
config_file_free(best_conf);
strlcpy(best_path, list->elems[i].data, sizeof(best_path));
best_conf = NULL;
best_conf = conf;
}
else
config_file_free(conf);
}
if (index >= 0 && current_best > 0 && best_conf)
{
input_autoconfigure_joypad_add(best_conf, params, task);
ret = 1;
2015-07-09 22:52:52 -05:00
}
if (best_conf)
config_file_free(best_conf);
string_list_free(list);
2015-03-27 18:05:43 +01:00
if (ret == 0)
return false;
return true;
}
static bool input_autoconfigure_joypad_from_conf_internal(
autoconfig_params_t *params, retro_task_t *task)
{
size_t i;
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], NULL);
if (conf && input_autoconfigure_joypad_from_conf(conf, params, task))
return true;
}
2017-09-30 07:28:58 +02:00
if (string_is_empty(params->autoconfig_directory))
return true;
return false;
}
2017-09-29 06:16:35 +02:00
static void input_autoconfigure_params_free(autoconfig_params_t *params)
{
if (!params)
return;
2017-09-30 06:08:09 +02:00
if (!string_is_empty(params->name))
2017-09-29 06:16:35 +02:00
free(params->name);
2017-09-30 06:08:09 +02:00
if (!string_is_empty(params->autoconfig_directory))
2017-09-29 06:16:35 +02:00
free(params->autoconfig_directory);
params->name = NULL;
params->autoconfig_directory = NULL;
}
static void input_autoconfigure_connect_handler(retro_task_t *task)
2015-03-27 18:05:43 +01:00
{
2016-12-03 06:09:55 +01:00
autoconfig_params_t *params = (autoconfig_params_t*)task->state;
if (!params || string_is_empty(params->name))
2017-01-03 03:52:08 +01:00
goto end;
if ( !input_autoconfigure_joypad_from_conf_dir(params, task)
&& !input_autoconfigure_joypad_from_conf_internal(params, task))
2016-12-01 20:41:41 +01:00
{
char msg[255];
2016-12-01 20:41:41 +01:00
msg[0] = '\0';
2017-05-13 23:41:13 +02:00
#ifdef ANDROID
2017-09-30 06:38:53 +02:00
if (!string_is_empty(params->name))
free(params->name);
2017-09-29 06:16:35 +02:00
params->name = strdup("Android Gamepad");
if (input_autoconfigure_joypad_from_conf_internal(params, task))
2017-02-05 14:24:34 -05:00
{
snprintf(msg, sizeof(msg), "%s (%ld/%ld) %s.",
2017-09-30 06:29:07 +02:00
!string_is_empty(params->name) ? params->name : "N/A",
(long)params->vid, (long)params->pid,
2017-02-05 14:24:34 -05:00
msg_hash_to_str(MSG_DEVICE_NOT_CONFIGURED_FALLBACK));
}
2017-05-13 23:41:13 +02:00
#else
snprintf(msg, sizeof(msg), "%s (%ld/%ld) %s.",
2017-09-30 06:29:07 +02:00
!string_is_empty(params->name) ? params->name : "N/A",
(long)params->vid, (long)params->pid,
2017-05-13 23:41:13 +02:00
msg_hash_to_str(MSG_DEVICE_NOT_CONFIGURED));
2017-02-05 14:24:34 -05:00
#endif
task_free_title(task);
task_set_title(task, strdup(msg));
2016-12-01 20:41:41 +01:00
}
2017-01-03 03:52:08 +01:00
end:
2017-09-29 06:16:35 +02:00
if (params)
{
input_autoconfigure_params_free(params);
free(params);
}
task_set_finished(task, true);
2016-12-01 22:36:38 +01:00
}
2016-12-01 22:36:38 +01:00
static void input_autoconfigure_disconnect_handler(retro_task_t *task)
{
2016-12-03 06:09:55 +01:00
autoconfig_disconnect_t *params = (autoconfig_disconnect_t*)task->state;
2016-12-01 22:36:38 +01:00
task_set_title(task, strdup(params->msg));
task_set_finished(task, true);
2016-12-01 22:36:38 +01:00
2017-09-30 06:08:09 +02:00
if (!string_is_empty(params->msg))
2017-09-29 06:16:35 +02:00
free(params->msg);
2016-12-04 05:50:50 +01:00
free(params);
}
2016-12-01 22:36:38 +01:00
bool input_autoconfigure_disconnect(unsigned i, const char *ident)
{
2016-10-27 09:32:07 +02:00
char msg[255];
retro_task_t *task = task_init();
2017-09-30 17:11:51 +02:00
autoconfig_disconnect_t *state = (autoconfig_disconnect_t*)calloc(1, sizeof(*state));
2016-12-21 01:45:19 +01:00
if (!state || !task)
goto error;
msg[0] = '\0';
state->idx = i;
2017-12-05 22:03:56 +00:00
snprintf(msg, sizeof(msg), "%s #%u (%s).",
2016-10-22 04:57:46 +02:00
msg_hash_to_str(MSG_DEVICE_DISCONNECTED_FROM_PORT),
i, ident);
2016-12-01 22:36:38 +01:00
2017-09-29 06:16:35 +02:00
state->msg = strdup(msg);
2016-12-01 22:36:38 +01:00
2017-04-25 15:53:30 +02:00
input_config_clear_device_name(state->idx);
input_config_clear_device_display_name(state->idx);
input_config_clear_device_config_name(state->idx);
input_config_clear_device_config_path(state->idx);
2016-12-01 22:36:38 +01:00
task->state = state;
task->handler = input_autoconfigure_disconnect_handler;
2017-05-14 20:43:39 +02:00
task_queue_push(task);
2016-12-01 22:36:38 +01:00
return true;
error:
if (state)
2017-09-29 06:16:35 +02:00
{
2017-09-30 06:08:09 +02:00
if (!string_is_empty(state->msg))
2017-09-29 06:16:35 +02:00
free(state->msg);
2016-12-01 22:36:38 +01:00
free(state);
2017-09-29 06:16:35 +02:00
}
2016-12-01 22:36:38 +01:00
if (task)
free(task);
return false;
}
void input_autoconfigure_reset(void)
{
unsigned i, j;
for (i = 0; i < MAX_USERS; i++)
{
for (j = 0; j < RARCH_BIND_LIST_END; j++)
{
input_autoconf_binds[i][j].joykey = NO_BTN;
input_autoconf_binds[i][j].joyaxis = AXIS_NONE;
}
input_device_name_index[i] = 0;
input_autoconfigured[i] = 0;
}
input_autoconfigure_swap_override = false;
}
bool input_is_autoconfigured(unsigned i)
{
return input_autoconfigured[i];
}
unsigned input_autoconfigure_get_device_name_index(unsigned i)
{
return input_device_name_index[i];
}
void input_autoconfigure_connect(
2016-12-31 07:43:34 +01:00
const char *name,
const char *display_name,
const char *driver,
unsigned idx,
unsigned vid,
unsigned pid)
{
unsigned i;
retro_task_t *task = task_init();
2017-09-30 17:08:42 +02:00
autoconfig_params_t *state = (autoconfig_params_t*)calloc(1, sizeof(*state));
settings_t *settings = config_get_ptr();
2017-09-30 16:52:41 +02:00
const char *dir_autoconf = settings ? settings->paths.directory_autoconfig : NULL;
bool autodetect_enable = settings ? settings->bools.input_autodetect_enable : false;
2017-09-30 16:52:41 +02:00
if (!task || !state || !autodetect_enable)
{
if (state)
{
input_autoconfigure_params_free(state);
free(state);
}
if (task)
free(task);
input_config_set_device_name(idx, name);
return;
}
2016-12-31 07:43:34 +01:00
if (!string_is_empty(name))
2017-09-29 17:06:38 +02:00
state->name = strdup(name);
2016-12-31 07:43:34 +01:00
if (!string_is_empty(dir_autoconf))
2017-09-29 06:16:35 +02:00
state->autoconfig_directory = strdup(dir_autoconf);
2017-09-29 17:06:38 +02:00
state->idx = idx;
state->vid = vid;
state->pid = pid;
state->max_users = *(
input_driver_get_uint(INPUT_ACTION_MAX_USERS));
#ifdef HAVE_BLISSBOX
if (state->vid == BLISSBOX_VID)
input_autoconfigure_override_handler(state);
#endif
2017-09-30 06:38:53 +02:00
if (!string_is_empty(state->name))
input_config_set_device_name(state->idx, state->name);
input_config_set_pid(state->idx, state->pid);
input_config_set_vid(state->idx, state->vid);
for (i = 0; i < RARCH_BIND_LIST_END; i++)
{
input_autoconf_binds[state->idx][i].joykey = NO_BTN;
input_autoconf_binds[state->idx][i].joyaxis = AXIS_NONE;
2017-12-05 22:03:56 +00:00
if (
!string_is_empty(input_autoconf_binds[state->idx][i].joykey_label))
2017-09-27 20:08:37 +02:00
free(input_autoconf_binds[state->idx][i].joykey_label);
2017-12-05 22:03:56 +00:00
if (
!string_is_empty(input_autoconf_binds[state->idx][i].joyaxis_label))
2017-09-27 20:11:44 +02:00
free(input_autoconf_binds[state->idx][i].joyaxis_label);
input_autoconf_binds[state->idx][i].joykey_label = NULL;
input_autoconf_binds[state->idx][i].joyaxis_label = NULL;
}
2017-04-25 15:31:32 +02:00
input_autoconfigured[state->idx] = false;
2016-12-01 22:17:36 +01:00
2017-09-30 16:52:41 +02:00
task->state = state;
task->handler = input_autoconfigure_connect_handler;
2017-05-14 20:43:39 +02:00
task_queue_push(task);
}