2012-10-16 17:46:31 +00:00
|
|
|
/* RetroArch - A frontend for libretro.
|
2014-01-01 00:50:59 +00:00
|
|
|
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
|
2015-01-07 16:46:50 +00:00
|
|
|
* Copyright (C) 2011-2015 - Daniel De Matteis
|
|
|
|
* Copyright (C) 2012-2015 - Michael Lelli
|
2014-01-01 00:50:59 +00:00
|
|
|
* Copyright (C) 2013-2014 - Steven Crowe
|
2012-10-16 17:46:31 +00:00
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2012-10-28 21:20:22 +00:00
|
|
|
#include <android/keycodes.h>
|
2012-11-01 04:48:20 +00:00
|
|
|
#include <unistd.h>
|
2013-03-03 22:17:16 +00:00
|
|
|
#include <dlfcn.h>
|
2015-04-21 00:46:50 +00:00
|
|
|
|
|
|
|
#include <retro_inline.h>
|
|
|
|
|
2015-01-12 20:47:01 +00:00
|
|
|
#include "../../frontend/drivers/platform_android.h"
|
2015-01-12 05:16:52 +00:00
|
|
|
#include "../input_autodetect.h"
|
|
|
|
#include "../input_common.h"
|
|
|
|
#include "../input_joypad.h"
|
|
|
|
#include "../../performance.h"
|
|
|
|
#include "../../general.h"
|
|
|
|
#include "../../driver.h"
|
2012-11-17 20:36:22 +00:00
|
|
|
|
2014-06-10 20:09:44 +00:00
|
|
|
#define AKEY_EVENT_NO_ACTION 255
|
|
|
|
|
2013-03-04 01:19:11 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
AXIS_X = 0,
|
|
|
|
AXIS_Y = 1,
|
|
|
|
AXIS_Z = 11,
|
2013-07-12 02:11:08 +00:00
|
|
|
AXIS_RZ = 14,
|
|
|
|
AXIS_HAT_X = 15,
|
2013-07-31 17:04:28 +00:00
|
|
|
AXIS_HAT_Y = 16,
|
|
|
|
AXIS_LTRIGGER = 17,
|
|
|
|
AXIS_RTRIGGER = 18,
|
2013-12-16 16:08:06 +00:00
|
|
|
AXIS_GAS = 22,
|
|
|
|
AXIS_BRAKE = 23,
|
2013-03-04 01:19:11 +00:00
|
|
|
};
|
2013-03-03 22:17:16 +00:00
|
|
|
|
2015-04-29 02:17:31 +00:00
|
|
|
|
2013-11-01 15:33:32 +00:00
|
|
|
typedef struct android_input
|
|
|
|
{
|
2015-05-01 15:25:26 +00:00
|
|
|
android_input_state_t copy;
|
2015-04-14 14:37:59 +00:00
|
|
|
const input_device_driver_t *joypad;
|
2013-11-01 15:33:32 +00:00
|
|
|
} android_input_t;
|
|
|
|
|
2015-02-24 17:20:21 +00:00
|
|
|
static void frontend_android_get_version_sdk(int32_t *sdk);
|
2015-01-03 21:29:37 +00:00
|
|
|
|
2015-04-29 02:48:34 +00:00
|
|
|
void (*engine_handle_dpad)(android_input_state_t *state,
|
2015-04-29 02:17:31 +00:00
|
|
|
AInputEvent*, int, int);
|
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
static bool android_input_set_sensor_state(void *data, unsigned port,
|
|
|
|
enum retro_sensor_action action, unsigned event_rate);
|
2013-03-04 10:18:22 +00:00
|
|
|
|
|
|
|
extern float AMotionEvent_getAxisValue(const AInputEvent* motion_event,
|
2014-10-20 18:31:00 +00:00
|
|
|
int32_t axis, size_t pointer_idx);
|
2013-03-03 22:17:16 +00:00
|
|
|
|
|
|
|
static typeof(AMotionEvent_getAxisValue) *p_AMotionEvent_getAxisValue;
|
|
|
|
|
|
|
|
#define AMotionEvent_getAxisValue (*p_AMotionEvent_getAxisValue)
|
|
|
|
|
2015-04-29 02:48:34 +00:00
|
|
|
static void engine_handle_dpad_default(
|
2015-04-29 02:17:31 +00:00
|
|
|
android_input_state_t *state, AInputEvent *event, int port, int source)
|
2013-03-04 10:18:22 +00:00
|
|
|
{
|
2014-08-20 14:45:46 +00:00
|
|
|
size_t motion_pointer = AMotionEvent_getAction(event) >>
|
|
|
|
AMOTION_EVENT_ACTION_POINTER_INDEX_SHIFT;
|
2013-03-04 10:18:22 +00:00
|
|
|
float x = AMotionEvent_getX(event, motion_pointer);
|
|
|
|
float y = AMotionEvent_getY(event, motion_pointer);
|
|
|
|
|
2015-04-29 02:17:31 +00:00
|
|
|
state->analog_state[port][0] = (int16_t)(x * 32767.0f);
|
|
|
|
state->analog_state[port][1] = (int16_t)(y * 32767.0f);
|
2013-03-04 10:18:22 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 02:48:34 +00:00
|
|
|
static void engine_handle_dpad_getaxisvalue(
|
2015-04-29 02:17:31 +00:00
|
|
|
android_input_state_t *state,
|
2014-08-20 14:45:46 +00:00
|
|
|
AInputEvent *event, int port, int source)
|
2013-03-04 10:18:22 +00:00
|
|
|
{
|
2014-08-20 14:45:46 +00:00
|
|
|
size_t motion_pointer = AMotionEvent_getAction(event) >>
|
|
|
|
AMOTION_EVENT_ACTION_POINTER_INDEX_SHIFT;
|
2015-04-29 02:48:34 +00:00
|
|
|
float x = AMotionEvent_getAxisValue(event, AXIS_X, motion_pointer);
|
|
|
|
float y = AMotionEvent_getAxisValue(event, AXIS_Y, motion_pointer);
|
|
|
|
float z = AMotionEvent_getAxisValue(event, AXIS_Z, motion_pointer);
|
|
|
|
float rz = AMotionEvent_getAxisValue(event, AXIS_RZ, motion_pointer);
|
|
|
|
float hatx = AMotionEvent_getAxisValue(event, AXIS_HAT_X, motion_pointer);
|
|
|
|
float haty = AMotionEvent_getAxisValue(event, AXIS_HAT_Y, motion_pointer);
|
|
|
|
float ltrig = AMotionEvent_getAxisValue(event, AXIS_LTRIGGER, motion_pointer);
|
|
|
|
float rtrig = AMotionEvent_getAxisValue(event, AXIS_RTRIGGER, motion_pointer);
|
|
|
|
float brake = AMotionEvent_getAxisValue(event, AXIS_BRAKE, motion_pointer);
|
|
|
|
float gas = AMotionEvent_getAxisValue(event, AXIS_GAS, motion_pointer);
|
2014-06-07 03:41:02 +00:00
|
|
|
|
2015-04-29 02:17:31 +00:00
|
|
|
state->hat_state[port][0] = (int)hatx;
|
|
|
|
state->hat_state[port][1] = (int)haty;
|
2014-06-11 14:01:59 +00:00
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
/* XXX: this could be a loop instead, but do we really want to
|
|
|
|
* loop through every axis?
|
|
|
|
*/
|
2015-04-29 02:17:31 +00:00
|
|
|
state->analog_state[port][0] = (int16_t)(x * 32767.0f);
|
|
|
|
state->analog_state[port][1] = (int16_t)(y * 32767.0f);
|
|
|
|
state->analog_state[port][2] = (int16_t)(z * 32767.0f);
|
|
|
|
state->analog_state[port][3] = (int16_t)(rz * 32767.0f);
|
|
|
|
state->analog_state[port][6] = (int16_t)(ltrig * 32767.0f);
|
|
|
|
state->analog_state[port][7] = (int16_t)(rtrig * 32767.0f);
|
|
|
|
state->analog_state[port][8] = (int16_t)(brake * 32767.0f);
|
|
|
|
state->analog_state[port][9] = (int16_t)(gas * 32767.0f);
|
2013-03-04 10:18:22 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
static void engine_handle_cmd(void)
|
2015-04-29 00:48:17 +00:00
|
|
|
{
|
|
|
|
struct android_app *android_app = (struct android_app*)g_android;
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2015-04-29 00:48:17 +00:00
|
|
|
runloop_t *runloop = rarch_main_get_ptr();
|
|
|
|
global_t *global = global_get_ptr();
|
|
|
|
driver_t *driver = driver_get_ptr();
|
2015-05-01 21:59:23 +00:00
|
|
|
int8_t cmd = android_app_read_cmd(android_app);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
|
|
|
switch (cmd)
|
|
|
|
{
|
|
|
|
case APP_CMD_INPUT_CHANGED:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
if (android_app->inputQueue)
|
|
|
|
AInputQueue_detachLooper(android_app->inputQueue);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
android_app->inputQueue = android_app->pendingInputQueue;
|
2015-04-29 00:48:17 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
if (android_app->inputQueue)
|
|
|
|
{
|
|
|
|
RARCH_LOG("Attaching input queue to looper");
|
|
|
|
AInputQueue_attachLooper(android_app->inputQueue,
|
|
|
|
android_app->looper, LOOPER_ID_INPUT, NULL,
|
|
|
|
NULL);
|
2015-04-29 03:43:05 +00:00
|
|
|
}
|
2015-04-29 03:59:40 +00:00
|
|
|
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_INIT_WINDOW:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
android_app->window = android_app->pendingWindow;
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
|
|
|
if (runloop->is_paused)
|
|
|
|
event_command(EVENT_CMD_REINIT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_RESUME:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
android_app->activityState = cmd;
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_START:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
android_app->activityState = cmd;
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_PAUSE:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
android_app->activityState = cmd;
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
|
|
|
if (!global->system.shutdown)
|
|
|
|
{
|
|
|
|
RARCH_LOG("Pausing RetroArch.\n");
|
|
|
|
runloop->is_paused = true;
|
|
|
|
runloop->is_idle = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_STOP:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
android_app->activityState = cmd;
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_CONFIG_CHANGED:
|
|
|
|
break;
|
|
|
|
case APP_CMD_TERM_WINDOW:
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_mutex_lock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
|
|
|
|
/* The window is being hidden or closed, clean it up. */
|
|
|
|
/* terminate display/EGL context here */
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
RARCH_WARN("Window is terminated outside PAUSED state.\n");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
android_app->window = NULL;
|
2015-05-01 22:13:28 +00:00
|
|
|
pthread_cond_broadcast(&android_app->cond);
|
|
|
|
pthread_mutex_unlock(&android_app->mutex);
|
2015-04-29 00:48:17 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_GAINED_FOCUS:
|
|
|
|
runloop->is_paused = false;
|
|
|
|
runloop->is_idle = false;
|
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
if ((userdata->sensor_state_mask
|
2015-04-29 00:48:17 +00:00
|
|
|
& (1ULL << RETRO_SENSOR_ACCELEROMETER_ENABLE))
|
2015-05-01 18:21:52 +00:00
|
|
|
&& userdata->accelerometerSensor == NULL
|
2015-04-29 00:48:17 +00:00
|
|
|
&& driver->input_data)
|
|
|
|
android_input_set_sensor_state(driver->input_data, 0,
|
|
|
|
RETRO_SENSOR_ACCELEROMETER_ENABLE,
|
2015-05-01 18:21:52 +00:00
|
|
|
userdata->accelerometer_event_rate);
|
2015-04-29 00:48:17 +00:00
|
|
|
break;
|
|
|
|
case APP_CMD_LOST_FOCUS:
|
|
|
|
/* Avoid draining battery while app is not being used. */
|
2015-05-01 18:21:52 +00:00
|
|
|
if ((userdata->sensor_state_mask
|
2015-04-29 00:48:17 +00:00
|
|
|
& (1ULL << RETRO_SENSOR_ACCELEROMETER_ENABLE))
|
2015-05-01 18:21:52 +00:00
|
|
|
&& userdata->accelerometerSensor != NULL
|
2015-04-29 00:48:17 +00:00
|
|
|
&& driver->input_data)
|
|
|
|
android_input_set_sensor_state(driver->input_data, 0,
|
|
|
|
RETRO_SENSOR_ACCELEROMETER_DISABLE,
|
2015-05-01 18:21:52 +00:00
|
|
|
userdata->accelerometer_event_rate);
|
2015-04-29 00:48:17 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APP_CMD_DESTROY:
|
|
|
|
global->system.shutdown = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
static void *android_input_init(void)
|
2013-03-02 21:56:58 +00:00
|
|
|
{
|
2015-04-29 03:59:40 +00:00
|
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
android_input_t *android = (android_input_t*)calloc(1, sizeof(*android));
|
2015-01-03 21:29:37 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
if (!android)
|
|
|
|
return NULL;
|
2013-03-02 21:56:58 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
android->copy.pads_connected = 0;
|
|
|
|
android->joypad = input_joypad_init_driver(settings->input.joypad_driver);
|
|
|
|
|
|
|
|
return android;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int zeus_id = -1;
|
|
|
|
static int zeus_second_id = -1;
|
|
|
|
|
|
|
|
static INLINE int android_input_poll_event_type_motion(
|
|
|
|
android_input_state_t *android, AInputEvent *event,
|
|
|
|
int port, int source)
|
|
|
|
{
|
|
|
|
int getaction, action;
|
|
|
|
size_t motion_pointer;
|
|
|
|
bool keyup;
|
|
|
|
|
|
|
|
if (source & ~(AINPUT_SOURCE_TOUCHSCREEN | AINPUT_SOURCE_MOUSE))
|
|
|
|
return 1;
|
|
|
|
|
2015-05-01 16:24:06 +00:00
|
|
|
getaction = AMotionEvent_getAction(event);
|
|
|
|
action = getaction & AMOTION_EVENT_ACTION_MASK;
|
2015-04-29 03:59:40 +00:00
|
|
|
motion_pointer = getaction >> AMOTION_EVENT_ACTION_POINTER_INDEX_SHIFT;
|
2015-05-01 16:24:06 +00:00
|
|
|
keyup = (
|
2015-04-29 03:59:40 +00:00
|
|
|
action == AMOTION_EVENT_ACTION_UP ||
|
|
|
|
action == AMOTION_EVENT_ACTION_CANCEL ||
|
|
|
|
action == AMOTION_EVENT_ACTION_POINTER_UP) ||
|
|
|
|
(source == AINPUT_SOURCE_MOUSE &&
|
|
|
|
action != AMOTION_EVENT_ACTION_DOWN);
|
|
|
|
|
|
|
|
if (keyup && motion_pointer < MAX_TOUCH)
|
2014-06-11 06:23:00 +00:00
|
|
|
{
|
2015-04-29 03:59:40 +00:00
|
|
|
memmove(android->pointer + motion_pointer,
|
|
|
|
android->pointer + motion_pointer + 1,
|
|
|
|
(MAX_TOUCH - motion_pointer - 1) * sizeof(struct input_pointer));
|
|
|
|
if (android->pointer_count > 0)
|
|
|
|
android->pointer_count--;
|
2014-06-02 12:28:26 +00:00
|
|
|
}
|
2015-04-29 03:59:40 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
float x, y;
|
|
|
|
int pointer_max = min(AMotionEvent_getPointerCount(event), MAX_TOUCH);
|
|
|
|
|
|
|
|
for (motion_pointer = 0; motion_pointer < pointer_max; motion_pointer++)
|
|
|
|
{
|
|
|
|
x = AMotionEvent_getX(event, motion_pointer);
|
|
|
|
y = AMotionEvent_getY(event, motion_pointer);
|
|
|
|
|
|
|
|
input_translate_coord_viewport(x, y,
|
|
|
|
&android->pointer[motion_pointer].x,
|
|
|
|
&android->pointer[motion_pointer].y,
|
|
|
|
&android->pointer[motion_pointer].full_x,
|
|
|
|
&android->pointer[motion_pointer].full_y);
|
|
|
|
|
|
|
|
android->pointer_count = max(
|
|
|
|
android->pointer_count,
|
|
|
|
motion_pointer + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static INLINE void android_input_poll_event_type_key(
|
|
|
|
android_input_state_t *android, struct android_app *android_app,
|
|
|
|
AInputEvent *event, int port, int keycode, int source,
|
|
|
|
int type_event, int *handled)
|
|
|
|
{
|
|
|
|
uint8_t *buf = android->pad_state[port];
|
|
|
|
int action = AKeyEvent_getAction(event);
|
|
|
|
|
|
|
|
/* some controllers send both the up and down events at once
|
|
|
|
* when the button is released for "special" buttons, like menu buttons
|
|
|
|
* work around that by only using down events for meta keys (which get
|
|
|
|
* cleared every poll anyway)
|
|
|
|
*/
|
|
|
|
if (action == AKEY_EVENT_ACTION_UP)
|
|
|
|
BIT_CLEAR(buf, keycode);
|
|
|
|
else if (action == AKEY_EVENT_ACTION_DOWN)
|
|
|
|
BIT_SET(buf, keycode);
|
2014-06-02 12:28:26 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
if ((keycode == AKEYCODE_VOLUME_UP || keycode == AKEYCODE_VOLUME_DOWN))
|
|
|
|
*handled = 0;
|
2014-06-02 12:28:26 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 02:48:34 +00:00
|
|
|
static int android_input_get_id_port(android_input_state_t *android, int id,
|
2014-08-20 14:45:46 +00:00
|
|
|
int source)
|
2014-06-21 04:17:10 +00:00
|
|
|
{
|
|
|
|
unsigned i;
|
2014-08-20 14:45:46 +00:00
|
|
|
if (source & (AINPUT_SOURCE_TOUCHSCREEN | AINPUT_SOURCE_MOUSE |
|
|
|
|
AINPUT_SOURCE_TOUCHPAD))
|
2014-12-05 12:53:49 +00:00
|
|
|
return 0; /* touch overlay is always user 1 */
|
2014-06-21 04:17:10 +00:00
|
|
|
|
|
|
|
for (i = 0; i < android->pads_connected; i++)
|
2014-06-21 04:35:52 +00:00
|
|
|
if (android->pad_states[i].id == id)
|
2014-06-21 04:17:10 +00:00
|
|
|
return i;
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
/* Returns the index inside android->pad_state */
|
|
|
|
static int android_input_get_id_index_from_name(android_input_state_t *android,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < android->pads_connected; i++)
|
|
|
|
{
|
|
|
|
if (strcmp(name, android->pad_states[i].name) == 0)
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
2014-06-21 04:35:52 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 02:48:34 +00:00
|
|
|
static void handle_hotplug(android_input_state_t *android,
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata, unsigned *port, unsigned id,
|
2014-06-10 22:11:30 +00:00
|
|
|
int source)
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
2014-06-11 20:54:29 +00:00
|
|
|
char device_name[256], name_buf[256];
|
|
|
|
name_buf[0] = device_name[0] = 0;
|
2015-01-03 21:29:37 +00:00
|
|
|
int vendorId = 0, productId = 0;
|
2015-03-27 15:57:58 +00:00
|
|
|
autoconfig_params_t params = {{0}};
|
2015-03-20 21:32:09 +00:00
|
|
|
settings_t *settings = config_get_ptr();
|
2014-06-11 20:54:29 +00:00
|
|
|
|
2015-03-20 21:32:09 +00:00
|
|
|
if (!settings->input.autodetect_enable)
|
2014-06-11 20:54:29 +00:00
|
|
|
return;
|
2014-06-10 20:09:44 +00:00
|
|
|
|
2014-06-11 14:21:05 +00:00
|
|
|
if (*port > MAX_PADS)
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
2014-06-10 21:24:09 +00:00
|
|
|
RARCH_ERR("Max number of pads reached.\n");
|
2014-06-10 20:09:44 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-01-03 21:29:37 +00:00
|
|
|
if (!engine_lookup_name(device_name, &vendorId, &productId, sizeof(device_name), id))
|
2014-09-30 17:31:28 +00:00
|
|
|
{
|
|
|
|
RARCH_ERR("Could not look up device name or IDs.\n");
|
|
|
|
return;
|
|
|
|
}
|
2014-06-10 20:09:44 +00:00
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
/* FIXME: Ugly hack, see other FIXME note below. */
|
|
|
|
if (strstr(device_name, "keypad-game-zeus") ||
|
|
|
|
strstr(device_name, "keypad-zeus"))
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
|
|
|
if (zeus_id < 0)
|
|
|
|
{
|
2014-09-05 23:43:28 +00:00
|
|
|
RARCH_LOG("zeus_pad 1 detected: %u\n", id);
|
2014-06-10 20:09:44 +00:00
|
|
|
zeus_id = id;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-09-05 23:43:28 +00:00
|
|
|
RARCH_LOG("zeus_pad 2 detected: %u\n", id);
|
2014-06-10 20:09:44 +00:00
|
|
|
zeus_second_id = id;
|
|
|
|
}
|
2014-06-11 18:07:05 +00:00
|
|
|
strlcpy(name_buf, "Xperia Play", sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
}
|
2014-08-20 14:45:46 +00:00
|
|
|
/* followed by a 4 (hex) char HW id */
|
|
|
|
else if (strstr(device_name, "iControlPad-"))
|
2014-08-13 21:04:59 +00:00
|
|
|
strlcpy(name_buf, "iControlPad HID Joystick profile", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "TTT THT Arcade console 2P USB Play"))
|
2014-06-14 01:42:07 +00:00
|
|
|
{
|
|
|
|
//FIXME - need to do a similar thing here as we did for nVidia Shield
|
|
|
|
//and Xperia Play. We need to keep 'count' of the amount of similar (grouped)
|
|
|
|
//devices.
|
|
|
|
//
|
2014-12-05 12:53:49 +00:00
|
|
|
//For Xperia Play - count similar devices and bind them to the same 'user'
|
2014-06-14 01:42:07 +00:00
|
|
|
//port
|
|
|
|
//
|
|
|
|
//For nVidia Shield - see above
|
|
|
|
//
|
|
|
|
//For TTT HT - keep track of how many of these 'pads' are already
|
2015-01-05 01:03:17 +00:00
|
|
|
//connected, and based on that, assign one of them to be User 1 and
|
|
|
|
//the other to be User 2.
|
2014-06-14 01:42:07 +00:00
|
|
|
//
|
|
|
|
//If this is finally implemented right, then these port conditionals can go.
|
|
|
|
if (*port == 0)
|
2015-01-05 01:03:17 +00:00
|
|
|
strlcpy(name_buf, "TTT THT Arcade (User 1)", sizeof(name_buf));
|
2014-06-14 01:42:07 +00:00
|
|
|
else if (*port == 1)
|
2015-01-05 01:03:17 +00:00
|
|
|
strlcpy(name_buf, "TTT THT Arcade (User 2)", sizeof(name_buf));
|
2014-06-14 01:42:07 +00:00
|
|
|
}
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "Sun4i-keypad"))
|
2014-06-13 23:11:28 +00:00
|
|
|
strlcpy(name_buf, "iDroid x360", sizeof(name_buf));
|
2014-06-15 06:13:22 +00:00
|
|
|
else if (strstr(device_name, "360 Wireless"))
|
|
|
|
strlcpy(name_buf, "XBox 360 Wireless", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "Microsoft"))
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
2015-04-29 05:58:19 +00:00
|
|
|
if (strstr(device_name, "SideWinder"))
|
2014-06-11 19:44:44 +00:00
|
|
|
strlcpy(name_buf, "SideWinder Classic", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "X-Box 360")
|
|
|
|
|| strstr(device_name, "X-Box"))
|
2014-06-11 08:16:27 +00:00
|
|
|
strlcpy(name_buf, "XBox 360", sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
}
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "WiseGroup"))
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
2014-08-20 14:45:46 +00:00
|
|
|
if (
|
|
|
|
strstr(device_name, "TigerGame") ||
|
|
|
|
strstr(device_name, "Game Controller Adapter") ||
|
|
|
|
strstr(device_name, "JC-PS102U") ||
|
|
|
|
strstr(device_name, "Dual USB Joypad"))
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
2014-06-11 20:54:29 +00:00
|
|
|
if (strstr(device_name, "WiseGroup"))
|
2014-06-11 19:44:44 +00:00
|
|
|
strlcpy(name_buf, "PlayStation2 WiseGroup", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "JC-PS102U"))
|
2014-06-13 23:11:28 +00:00
|
|
|
strlcpy(name_buf, "PlayStation2 JCPS102", sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
else
|
2014-06-13 23:11:28 +00:00
|
|
|
strlcpy(name_buf, "PlayStation2 Generic", sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
}
|
|
|
|
}
|
2014-08-20 14:45:46 +00:00
|
|
|
else if (
|
|
|
|
strstr(device_name, "PLAYSTATION(R)3") ||
|
|
|
|
strstr(device_name, "Dualshock3") ||
|
|
|
|
strstr(device_name, "Sixaxis") ||
|
|
|
|
strstr(device_name, "Gasia,Co") ||
|
|
|
|
(strstr(device_name, "Gamepad 0") ||
|
|
|
|
strstr(device_name, "Gamepad 1") ||
|
|
|
|
strstr(device_name, "Gamepad 2") ||
|
|
|
|
strstr(device_name, "Gamepad 3"))
|
|
|
|
)
|
2014-06-11 07:39:13 +00:00
|
|
|
strlcpy(name_buf, "PlayStation3", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "MOGA"))
|
2014-08-19 08:26:07 +00:00
|
|
|
strlcpy(name_buf, "Moga IME", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "adc joystick"))
|
2014-06-12 02:23:32 +00:00
|
|
|
strlcpy(name_buf, "JXD S7300B", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "2-Axis, 8-Button"))
|
2014-08-13 20:50:22 +00:00
|
|
|
strlcpy(name_buf, "Genius Maxfire G08XU", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "USB,2-axis 8-button gamepad"))
|
2014-08-19 08:33:30 +00:00
|
|
|
strlcpy(name_buf, "USB 2 Axis 8 button", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "joy_key"))
|
2014-06-12 01:48:01 +00:00
|
|
|
strlcpy(name_buf, "Archos Gamepad", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "matrix_keyboard"))
|
2014-06-12 02:23:32 +00:00
|
|
|
strlcpy(name_buf, "JXD S5110B", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "tincore_adc_joystick"))
|
2014-06-12 02:23:32 +00:00
|
|
|
strlcpy(name_buf, "JXD S5110B (Skelrom)", sizeof(name_buf));
|
2014-08-20 14:45:46 +00:00
|
|
|
else if (strstr(device_name, "keypad-zeus") ||
|
|
|
|
(strstr(device_name, "keypad-game-zeus"))
|
|
|
|
)
|
2014-06-11 18:11:37 +00:00
|
|
|
strlcpy(name_buf, "Xperia Play", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "USB Gamepad"))
|
2014-08-19 08:33:30 +00:00
|
|
|
strlcpy(name_buf, "Thrust Predator", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "ADC joystick"))
|
2014-06-12 02:23:32 +00:00
|
|
|
strlcpy(name_buf, "JXD S7800B", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "2Axes 11Keys Game Pad"))
|
2014-08-13 20:50:22 +00:00
|
|
|
strlcpy(name_buf, "Tomee NES USB", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "USB Gamepad"))
|
2014-08-19 08:18:21 +00:00
|
|
|
strlcpy(name_buf, "Defender Game Racer Classic", sizeof(name_buf));
|
2014-06-11 20:54:29 +00:00
|
|
|
else if (strstr(device_name, "NVIDIA Controller"))
|
2014-06-10 20:09:44 +00:00
|
|
|
{
|
2014-12-05 12:53:49 +00:00
|
|
|
/* Shield is always user 1. FIXME: This is kinda ugly.
|
2014-08-20 14:45:46 +00:00
|
|
|
* We really need to find a way to detect useless input devices
|
|
|
|
* like gpio-keys in a general way.
|
|
|
|
*/
|
|
|
|
*port = 0;
|
2014-06-10 21:24:09 +00:00
|
|
|
strlcpy(name_buf, "NVIDIA Shield", sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
}
|
2014-08-13 20:02:31 +00:00
|
|
|
else if (device_name[0] != '\0')
|
|
|
|
strlcpy(name_buf, device_name, sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
if (strstr(userdata->current_ime, "net.obsidianx.android.mogaime"))
|
|
|
|
strlcpy(name_buf, userdata->current_ime, sizeof(name_buf));
|
|
|
|
else if (strstr(userdata->current_ime, "com.ccpcreations.android.WiiUseAndroid"))
|
|
|
|
strlcpy(name_buf, userdata->current_ime, sizeof(name_buf));
|
|
|
|
else if (strstr(userdata->current_ime, "com.hexad.bluezime"))
|
|
|
|
strlcpy(name_buf, userdata->current_ime, sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
|
2014-08-19 08:33:30 +00:00
|
|
|
if (source == AINPUT_SOURCE_KEYBOARD && strcmp(name_buf, "Xperia Play"))
|
2014-08-19 10:03:03 +00:00
|
|
|
strlcpy(name_buf, "RetroKeyboard", sizeof(name_buf));
|
2014-06-10 20:09:44 +00:00
|
|
|
|
2014-06-11 20:54:29 +00:00
|
|
|
if (name_buf[0] != '\0')
|
2014-06-10 21:24:09 +00:00
|
|
|
{
|
2015-03-20 21:32:09 +00:00
|
|
|
strlcpy(settings->input.device_names[*port],
|
|
|
|
name_buf, sizeof(settings->input.device_names[*port]));
|
2014-09-30 18:35:58 +00:00
|
|
|
|
2014-06-11 14:21:05 +00:00
|
|
|
RARCH_LOG("Port %d: %s.\n", *port, name_buf);
|
2015-03-27 15:57:58 +00:00
|
|
|
strlcpy(params.name, name_buf, sizeof(params.name));
|
2015-05-01 16:24:06 +00:00
|
|
|
params.idx = *port;
|
2015-03-27 15:57:58 +00:00
|
|
|
params.vid = vendorId;
|
|
|
|
params.pid = productId;
|
2015-05-01 16:24:06 +00:00
|
|
|
strlcpy(params.driver, "android", sizeof(params.driver));
|
2015-03-27 15:57:58 +00:00
|
|
|
input_config_autoconfigure_joypad(¶ms);
|
2014-06-10 21:24:09 +00:00
|
|
|
}
|
2014-06-21 04:35:52 +00:00
|
|
|
|
|
|
|
*port = android->pads_connected;
|
|
|
|
android->pad_states[android->pads_connected].id = id;
|
|
|
|
android->pad_states[android->pads_connected].port = *port;
|
|
|
|
strlcpy(android->pad_states[*port].name, name_buf,
|
|
|
|
sizeof(android->pad_states[*port].name));
|
|
|
|
|
|
|
|
android->pads_connected++;
|
2014-06-10 20:09:44 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
static int android_input_get_id(AInputEvent *event)
|
2015-04-29 03:43:05 +00:00
|
|
|
{
|
2015-04-29 03:59:40 +00:00
|
|
|
int id = AInputEvent_getDeviceId(event);
|
2015-04-29 03:43:05 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
/* Needs to be cleaned up */
|
|
|
|
if (id == zeus_second_id)
|
|
|
|
id = zeus_id;
|
2015-04-29 03:43:05 +00:00
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
return id;
|
2014-06-21 04:43:21 +00:00
|
|
|
}
|
2014-06-21 03:58:09 +00:00
|
|
|
|
2015-04-29 02:48:34 +00:00
|
|
|
static void android_input_handle_input(android_input_state_t *android)
|
2015-04-21 00:58:12 +00:00
|
|
|
{
|
|
|
|
AInputEvent *event = NULL;
|
|
|
|
struct android_app *android_app = (struct android_app*)g_android;
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2015-04-21 00:58:12 +00:00
|
|
|
|
|
|
|
/* Read all pending events. */
|
2015-04-29 04:21:18 +00:00
|
|
|
while (AInputQueue_getEvent(android_app->inputQueue, &event) >= 0)
|
2015-04-21 00:58:12 +00:00
|
|
|
{
|
2015-05-01 16:24:06 +00:00
|
|
|
int32_t handled = 1;
|
2015-04-29 04:21:18 +00:00
|
|
|
int predispatched = AInputQueue_preDispatchEvent(android_app->inputQueue, event);
|
2015-05-01 16:24:06 +00:00
|
|
|
int source = AInputEvent_getSource(event);
|
|
|
|
int type_event = AInputEvent_getType(event);
|
|
|
|
int id = android_input_get_id(event);
|
|
|
|
int port = android_input_get_id_port(android, id, source);
|
2015-04-29 04:21:18 +00:00
|
|
|
|
|
|
|
if (port < 0)
|
2015-05-01 18:21:52 +00:00
|
|
|
handle_hotplug(android, userdata,
|
2015-04-29 04:21:18 +00:00
|
|
|
&android->pads_connected, id, source);
|
|
|
|
|
|
|
|
switch (type_event)
|
2015-04-21 00:58:12 +00:00
|
|
|
{
|
2015-04-29 04:21:18 +00:00
|
|
|
case AINPUT_EVENT_TYPE_MOTION:
|
|
|
|
if (android_input_poll_event_type_motion(android, event,
|
|
|
|
port, source))
|
|
|
|
engine_handle_dpad(android, event, port, source);
|
|
|
|
break;
|
|
|
|
case AINPUT_EVENT_TYPE_KEY:
|
|
|
|
{
|
|
|
|
int keycode = AKeyEvent_getKeyCode(event);
|
|
|
|
android_input_poll_event_type_key(android, android_app,
|
|
|
|
event, port, keycode, source, type_event, &handled);
|
|
|
|
}
|
|
|
|
break;
|
2015-04-21 00:58:12 +00:00
|
|
|
}
|
2015-04-29 04:21:18 +00:00
|
|
|
|
|
|
|
if (!predispatched)
|
|
|
|
AInputQueue_finishEvent(android_app->inputQueue, event,
|
|
|
|
handled);
|
2015-04-21 00:58:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-01 16:11:34 +00:00
|
|
|
static void android_input_handle_user(android_input_state_t *state)
|
2015-04-21 00:58:12 +00:00
|
|
|
{
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2015-04-21 00:58:12 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
if ((userdata->sensor_state_mask & (1ULL <<
|
2015-04-21 00:58:12 +00:00
|
|
|
RETRO_SENSOR_ACCELEROMETER_ENABLE))
|
2015-05-01 18:21:52 +00:00
|
|
|
&& userdata->accelerometerSensor)
|
2015-04-21 00:58:12 +00:00
|
|
|
{
|
|
|
|
ASensorEvent event;
|
2015-05-01 18:21:52 +00:00
|
|
|
while (ASensorEventQueue_getEvents(userdata->sensorEventQueue, &event, 1) > 0)
|
2015-04-21 00:58:12 +00:00
|
|
|
{
|
2015-05-01 16:11:34 +00:00
|
|
|
state->accelerometer_state.x = event.acceleration.x;
|
|
|
|
state->accelerometer_state.y = event.acceleration.y;
|
|
|
|
state->accelerometer_state.z = event.acceleration.z;
|
2015-04-21 00:58:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
/* Handle all events. If our activity is in pause state,
|
|
|
|
* block until we're unpaused.
|
|
|
|
*/
|
2013-03-02 21:56:58 +00:00
|
|
|
static void android_input_poll(void *data)
|
|
|
|
{
|
2013-03-04 13:12:04 +00:00
|
|
|
int ident;
|
2015-04-29 02:48:34 +00:00
|
|
|
android_input_t *android = (android_input_t*)data;
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2013-03-08 20:32:27 +00:00
|
|
|
|
2014-10-05 16:08:22 +00:00
|
|
|
while ((ident =
|
2015-03-24 07:53:49 +00:00
|
|
|
ALooper_pollAll((input_driver_key_pressed(RARCH_PAUSE_TOGGLE))
|
2014-08-20 14:45:46 +00:00
|
|
|
? -1 : 0,
|
2013-03-04 13:12:04 +00:00
|
|
|
NULL, NULL, NULL)) >= 0)
|
|
|
|
{
|
2015-04-21 00:58:12 +00:00
|
|
|
switch (ident)
|
2013-11-02 23:27:58 +00:00
|
|
|
{
|
2015-04-29 03:59:40 +00:00
|
|
|
case LOOPER_ID_INPUT:
|
2015-05-01 18:21:52 +00:00
|
|
|
android_input_handle_input(&userdata->thread_state);
|
2015-04-29 03:59:40 +00:00
|
|
|
break;
|
|
|
|
case LOOPER_ID_USER:
|
2015-05-01 18:21:52 +00:00
|
|
|
android_input_handle_user(&userdata->thread_state);
|
2015-04-29 03:59:40 +00:00
|
|
|
break;
|
2015-04-21 00:58:12 +00:00
|
|
|
case LOOPER_ID_MAIN:
|
2015-04-29 03:59:40 +00:00
|
|
|
engine_handle_cmd();
|
2015-04-21 00:58:12 +00:00
|
|
|
break;
|
2013-11-02 23:27:58 +00:00
|
|
|
}
|
2013-03-04 13:12:04 +00:00
|
|
|
}
|
2015-04-29 02:57:24 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
memcpy(&android->copy, &userdata->thread_state, sizeof(android->copy));
|
2012-10-16 17:46:31 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 00:48:17 +00:00
|
|
|
bool android_run_events(void *data)
|
|
|
|
{
|
|
|
|
global_t *global = global_get_ptr();
|
|
|
|
int id = ALooper_pollOnce(-1, NULL, NULL, NULL);
|
|
|
|
|
|
|
|
if (id == LOOPER_ID_MAIN)
|
2015-04-29 03:59:40 +00:00
|
|
|
engine_handle_cmd();
|
2015-04-29 00:48:17 +00:00
|
|
|
|
|
|
|
/* Check if we are exiting. */
|
|
|
|
if (global->system.shutdown)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
static int16_t android_input_state(void *data,
|
|
|
|
const struct retro_keybind **binds, unsigned port, unsigned device,
|
2014-10-20 18:31:00 +00:00
|
|
|
unsigned idx, unsigned id)
|
2012-10-16 17:46:31 +00:00
|
|
|
{
|
2013-11-01 15:33:32 +00:00
|
|
|
android_input_t *android = (android_input_t*)data;
|
|
|
|
|
2012-12-17 19:53:36 +00:00
|
|
|
switch (device)
|
2012-10-28 21:20:22 +00:00
|
|
|
{
|
2012-12-17 19:53:36 +00:00
|
|
|
case RETRO_DEVICE_JOYPAD:
|
2014-06-10 19:59:18 +00:00
|
|
|
return input_joypad_pressed(android->joypad, port, binds[port], id);
|
2013-07-31 17:04:28 +00:00
|
|
|
case RETRO_DEVICE_ANALOG:
|
2014-10-20 18:31:00 +00:00
|
|
|
return input_joypad_analog(android->joypad, port, idx, id,
|
2014-08-20 14:45:46 +00:00
|
|
|
binds[port]);
|
2012-12-21 00:31:01 +00:00
|
|
|
case RETRO_DEVICE_POINTER:
|
2013-07-31 17:04:28 +00:00
|
|
|
switch (id)
|
2012-12-21 00:31:01 +00:00
|
|
|
{
|
|
|
|
case RETRO_DEVICE_ID_POINTER_X:
|
2015-04-29 02:48:34 +00:00
|
|
|
return android->copy.pointer[idx].x;
|
2012-12-21 00:31:01 +00:00
|
|
|
case RETRO_DEVICE_ID_POINTER_Y:
|
2015-04-29 02:48:34 +00:00
|
|
|
return android->copy.pointer[idx].y;
|
2012-12-21 00:31:01 +00:00
|
|
|
case RETRO_DEVICE_ID_POINTER_PRESSED:
|
2015-04-29 02:48:34 +00:00
|
|
|
return (idx < android->copy.pointer_count) &&
|
|
|
|
(android->copy.pointer[idx].x != -0x8000) &&
|
|
|
|
(android->copy.pointer[idx].y != -0x8000);
|
2015-03-28 21:35:18 +00:00
|
|
|
case RARCH_DEVICE_ID_POINTER_BACK:
|
2015-04-29 02:48:34 +00:00
|
|
|
return BIT_GET(android->copy.pad_state[0], AKEYCODE_BACK);
|
2013-01-11 15:32:49 +00:00
|
|
|
}
|
2014-06-02 11:42:24 +00:00
|
|
|
break;
|
2013-01-11 15:32:49 +00:00
|
|
|
case RARCH_DEVICE_POINTER_SCREEN:
|
2013-07-31 17:04:28 +00:00
|
|
|
switch (id)
|
2013-01-11 15:32:49 +00:00
|
|
|
{
|
|
|
|
case RETRO_DEVICE_ID_POINTER_X:
|
2015-04-29 02:48:34 +00:00
|
|
|
return android->copy.pointer[idx].full_x;
|
2013-01-11 15:32:49 +00:00
|
|
|
case RETRO_DEVICE_ID_POINTER_Y:
|
2015-04-29 02:48:34 +00:00
|
|
|
return android->copy.pointer[idx].full_y;
|
2013-01-11 15:32:49 +00:00
|
|
|
case RETRO_DEVICE_ID_POINTER_PRESSED:
|
2015-04-29 02:48:34 +00:00
|
|
|
return (idx < android->copy.pointer_count) &&
|
|
|
|
(android->copy.pointer[idx].full_x != -0x8000) &&
|
|
|
|
(android->copy.pointer[idx].full_y != -0x8000);
|
2015-03-28 21:35:18 +00:00
|
|
|
case RARCH_DEVICE_ID_POINTER_BACK:
|
2015-04-29 02:48:34 +00:00
|
|
|
return BIT_GET(android->copy.pad_state[0], AKEYCODE_BACK);
|
2012-12-21 00:31:01 +00:00
|
|
|
}
|
2014-06-02 11:42:24 +00:00
|
|
|
break;
|
2012-12-17 19:53:36 +00:00
|
|
|
}
|
2014-01-20 13:52:53 +00:00
|
|
|
|
|
|
|
return 0;
|
2012-10-16 17:46:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool android_input_key_pressed(void *data, int key)
|
|
|
|
{
|
2014-06-10 19:59:18 +00:00
|
|
|
android_input_t *android = (android_input_t*)data;
|
2015-03-21 04:42:49 +00:00
|
|
|
driver_t *driver = driver_get_ptr();
|
|
|
|
global_t *global = global_get_ptr();
|
|
|
|
settings_t *settings = config_get_ptr();
|
2015-03-20 21:32:09 +00:00
|
|
|
|
2014-06-10 19:59:18 +00:00
|
|
|
if (!android)
|
|
|
|
return false;
|
2015-03-21 04:42:49 +00:00
|
|
|
|
|
|
|
return ((global->lifecycle_state | driver->overlay_state.buttons)
|
2014-08-20 14:45:46 +00:00
|
|
|
& (1ULL << key)) || input_joypad_pressed(android->joypad,
|
2015-03-20 21:32:09 +00:00
|
|
|
0, settings->input.binds[0], key);
|
2012-10-16 17:46:31 +00:00
|
|
|
}
|
|
|
|
|
2012-11-26 23:50:29 +00:00
|
|
|
static void android_input_free_input(void *data)
|
2012-10-16 17:46:31 +00:00
|
|
|
{
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2014-06-02 12:34:31 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
if (userdata->sensorManager)
|
|
|
|
ASensorManager_destroyEventQueue(userdata->sensorManager,
|
|
|
|
userdata->sensorEventQueue);
|
2014-06-07 03:41:02 +00:00
|
|
|
|
2014-05-29 21:57:58 +00:00
|
|
|
free(data);
|
2012-10-16 17:46:31 +00:00
|
|
|
}
|
|
|
|
|
2013-11-02 20:50:38 +00:00
|
|
|
static uint64_t android_input_get_capabilities(void *data)
|
2013-11-02 20:16:57 +00:00
|
|
|
{
|
2014-10-05 16:08:22 +00:00
|
|
|
(void)data;
|
2013-11-02 20:16:57 +00:00
|
|
|
|
2014-10-05 16:08:22 +00:00
|
|
|
return
|
|
|
|
(1 << RETRO_DEVICE_JOYPAD) |
|
|
|
|
(1 << RETRO_DEVICE_POINTER) |
|
|
|
|
(1 << RETRO_DEVICE_ANALOG);
|
2013-11-02 20:16:57 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 03:59:40 +00:00
|
|
|
static void android_input_enable_sensor_manager(void *data)
|
2013-11-02 23:27:58 +00:00
|
|
|
{
|
|
|
|
struct android_app *android_app = (struct android_app*)g_android;
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2013-11-02 23:27:58 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
userdata->sensorManager = ASensorManager_getInstance();
|
|
|
|
userdata->accelerometerSensor =
|
|
|
|
ASensorManager_getDefaultSensor(userdata->sensorManager,
|
2013-11-02 23:27:58 +00:00
|
|
|
ASENSOR_TYPE_ACCELEROMETER);
|
2015-05-01 18:21:52 +00:00
|
|
|
userdata->sensorEventQueue =
|
|
|
|
ASensorManager_createEventQueue(userdata->sensorManager,
|
2013-11-02 23:27:58 +00:00
|
|
|
android_app->looper, LOOPER_ID_USER, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
static bool android_input_set_sensor_state(void *data, unsigned port,
|
|
|
|
enum retro_sensor_action action, unsigned event_rate)
|
2013-11-02 23:27:58 +00:00
|
|
|
{
|
2015-04-29 03:59:40 +00:00
|
|
|
android_input_t *android = (android_input_t*)data;
|
2015-05-01 18:21:52 +00:00
|
|
|
struct android_app_userdata *userdata = (struct android_app_userdata*)g_android_userdata;
|
2013-11-02 23:27:58 +00:00
|
|
|
|
|
|
|
if (event_rate == 0)
|
|
|
|
event_rate = 60;
|
|
|
|
|
|
|
|
switch (action)
|
|
|
|
{
|
|
|
|
case RETRO_SENSOR_ACCELEROMETER_ENABLE:
|
2015-05-01 18:21:52 +00:00
|
|
|
if (!userdata->accelerometerSensor)
|
2015-04-29 03:59:40 +00:00
|
|
|
android_input_enable_sensor_manager(android);
|
2013-11-02 23:27:58 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
if (userdata->accelerometerSensor)
|
|
|
|
ASensorEventQueue_enableSensor(userdata->sensorEventQueue,
|
|
|
|
userdata->accelerometerSensor);
|
2013-11-02 23:27:58 +00:00
|
|
|
|
|
|
|
// events per second (in us).
|
2015-05-01 18:21:52 +00:00
|
|
|
if (userdata->accelerometerSensor)
|
|
|
|
ASensorEventQueue_setEventRate(userdata->sensorEventQueue,
|
|
|
|
userdata->accelerometerSensor, (1000L / event_rate)
|
2014-08-20 14:45:46 +00:00
|
|
|
* 1000);
|
2013-11-02 23:27:58 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
userdata->sensor_state_mask &= ~(1ULL << RETRO_SENSOR_ACCELEROMETER_DISABLE);
|
|
|
|
userdata->sensor_state_mask |= (1ULL << RETRO_SENSOR_ACCELEROMETER_ENABLE);
|
2013-11-03 11:06:14 +00:00
|
|
|
return true;
|
|
|
|
|
2013-11-02 23:27:58 +00:00
|
|
|
case RETRO_SENSOR_ACCELEROMETER_DISABLE:
|
2015-05-01 18:21:52 +00:00
|
|
|
if (userdata->accelerometerSensor)
|
|
|
|
ASensorEventQueue_disableSensor(userdata->sensorEventQueue,
|
|
|
|
userdata->accelerometerSensor);
|
2013-11-02 23:27:58 +00:00
|
|
|
|
2015-05-01 18:21:52 +00:00
|
|
|
userdata->sensor_state_mask &= ~(1ULL << RETRO_SENSOR_ACCELEROMETER_ENABLE);
|
|
|
|
userdata->sensor_state_mask |= (1ULL << RETRO_SENSOR_ACCELEROMETER_DISABLE);
|
2013-11-03 11:06:14 +00:00
|
|
|
return true;
|
2014-01-20 14:01:40 +00:00
|
|
|
default:
|
2015-05-01 18:21:52 +00:00
|
|
|
break;
|
2014-01-20 13:52:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-08-20 14:45:46 +00:00
|
|
|
static float android_input_get_sensor_input(void *data,
|
|
|
|
unsigned port,unsigned id)
|
2014-01-20 13:52:53 +00:00
|
|
|
{
|
|
|
|
android_input_t *android = (android_input_t*)data;
|
2013-11-03 11:06:14 +00:00
|
|
|
|
2014-01-20 13:59:06 +00:00
|
|
|
switch (id)
|
2014-01-20 13:52:53 +00:00
|
|
|
{
|
2014-01-20 16:16:25 +00:00
|
|
|
case RETRO_SENSOR_ACCELEROMETER_X:
|
2015-05-01 16:11:34 +00:00
|
|
|
return android->copy.accelerometer_state.x;
|
2014-01-20 16:16:25 +00:00
|
|
|
case RETRO_SENSOR_ACCELEROMETER_Y:
|
2015-05-01 16:11:34 +00:00
|
|
|
return android->copy.accelerometer_state.y;
|
2014-01-20 16:16:25 +00:00
|
|
|
case RETRO_SENSOR_ACCELEROMETER_Z:
|
2015-05-01 16:11:34 +00:00
|
|
|
return android->copy.accelerometer_state.z;
|
2013-11-02 23:27:58 +00:00
|
|
|
}
|
2014-01-20 13:52:53 +00:00
|
|
|
|
|
|
|
return 0;
|
2013-11-02 23:27:58 +00:00
|
|
|
}
|
|
|
|
|
2015-04-14 14:37:59 +00:00
|
|
|
static const input_device_driver_t *android_input_get_joypad_driver(void *data)
|
2014-06-02 11:42:24 +00:00
|
|
|
{
|
2014-06-10 19:59:18 +00:00
|
|
|
android_input_t *android = (android_input_t*)data;
|
2015-01-10 23:07:59 +00:00
|
|
|
if (!android)
|
|
|
|
return NULL;
|
|
|
|
return android->joypad;
|
2014-06-02 11:42:24 +00:00
|
|
|
}
|
|
|
|
|
2015-03-24 06:51:50 +00:00
|
|
|
static void android_input_grab_mouse(void *data, bool state)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)state;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool android_input_set_rumble(void *data, unsigned port,
|
|
|
|
enum retro_rumble_effect effect, uint16_t strength)
|
|
|
|
{
|
|
|
|
(void)data;
|
|
|
|
(void)port;
|
|
|
|
(void)effect;
|
|
|
|
(void)strength;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-09-11 05:06:20 +00:00
|
|
|
input_driver_t input_android = {
|
2012-11-26 23:50:29 +00:00
|
|
|
android_input_init,
|
2012-10-16 17:46:31 +00:00
|
|
|
android_input_poll,
|
|
|
|
android_input_state,
|
|
|
|
android_input_key_pressed,
|
2012-11-26 23:50:29 +00:00
|
|
|
android_input_free_input,
|
2013-11-02 23:27:58 +00:00
|
|
|
android_input_set_sensor_state,
|
2014-01-20 13:52:53 +00:00
|
|
|
android_input_get_sensor_input,
|
2013-11-02 20:16:57 +00:00
|
|
|
android_input_get_capabilities,
|
2015-04-29 00:49:23 +00:00
|
|
|
"android",
|
2014-06-02 11:42:24 +00:00
|
|
|
|
2015-03-24 06:51:50 +00:00
|
|
|
android_input_grab_mouse,
|
|
|
|
android_input_set_rumble,
|
2014-06-02 11:42:24 +00:00
|
|
|
android_input_get_joypad_driver,
|
|
|
|
};
|