moved suinput
This commit is contained in:
parent
ebfcbff2a5
commit
877d58a42f
225
jni/vnc/suinput/suinput.c
Executable file
225
jni/vnc/suinput/suinput.c
Executable file
@ -0,0 +1,225 @@
|
|||||||
|
/*
|
||||||
|
suinput - Simple C-API to the Linux uinput-system.
|
||||||
|
Copyright (C) 2009 Tuomas Räsänen <tuos@codegrove.org>
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 3 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library 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
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <linux/uinput.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include "suinput.h"
|
||||||
|
|
||||||
|
char* UINPUT_FILEPATHS[] = {
|
||||||
|
"/android/dev/uinput",
|
||||||
|
"/dev/uinput",
|
||||||
|
"/dev/input/uinput",
|
||||||
|
"/dev/misc/uinput",
|
||||||
|
};
|
||||||
|
#define UINPUT_FILEPATHS_COUNT (sizeof(UINPUT_FILEPATHS) / sizeof(char*))
|
||||||
|
|
||||||
|
int suinput_write(int uinput_fd,
|
||||||
|
uint16_t type, uint16_t code, int32_t value)
|
||||||
|
{
|
||||||
|
struct input_event event;
|
||||||
|
memset(&event, 0, sizeof(event));
|
||||||
|
gettimeofday(&event.time, 0); /* This should not be able to fail ever.. */
|
||||||
|
event.type = type;
|
||||||
|
event.code = code;
|
||||||
|
event.value = value;
|
||||||
|
if (write(uinput_fd, &event, sizeof(event)) != sizeof(event))
|
||||||
|
return -1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_write_syn(int uinput_fd,
|
||||||
|
uint16_t type, uint16_t code, int32_t value)
|
||||||
|
{
|
||||||
|
if (suinput_write(uinput_fd, type, code, value))
|
||||||
|
return -1;
|
||||||
|
return suinput_write(uinput_fd, EV_SYN, SYN_REPORT, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_open(const char* device_name, const struct input_id* id)
|
||||||
|
{
|
||||||
|
int original_errno = 0;
|
||||||
|
int uinput_fd = -1;
|
||||||
|
struct uinput_user_dev user_dev;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < UINPUT_FILEPATHS_COUNT; ++i) {
|
||||||
|
uinput_fd = open(UINPUT_FILEPATHS[i], O_WRONLY | O_NONBLOCK);
|
||||||
|
if (uinput_fd != -1)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (uinput_fd == -1)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Set device to handle following types of events: */
|
||||||
|
|
||||||
|
/* Key and button events */
|
||||||
|
if (ioctl(uinput_fd, UI_SET_EVBIT, EV_KEY) == -1)
|
||||||
|
goto err;
|
||||||
|
//
|
||||||
|
// /* Key and button repetition events */
|
||||||
|
if (ioctl(uinput_fd, UI_SET_EVBIT, EV_REP) == -1)
|
||||||
|
goto err;
|
||||||
|
//
|
||||||
|
// /* Relative pointer motions */
|
||||||
|
// if (ioctl(uinput_fd, UI_SET_EVBIT, EV_REL) == -1)
|
||||||
|
// goto err;
|
||||||
|
|
||||||
|
/* Absolute pointer motions */
|
||||||
|
|
||||||
|
if (ioctl(uinput_fd, UI_SET_EVBIT, EV_ABS) == -1)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
/* Synchronization events, this is probably set implicitely too. */
|
||||||
|
if (ioctl(uinput_fd, UI_SET_EVBIT, EV_SYN) == -1)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Configure device to handle relative x and y axis. */
|
||||||
|
// if (ioctl(uinput_fd, UI_SET_RELBIT, REL_X) == -1)
|
||||||
|
// goto err;
|
||||||
|
// if (ioctl(uinput_fd, UI_SET_RELBIT, REL_Y) == -1)
|
||||||
|
// goto err;
|
||||||
|
|
||||||
|
/* Configure device to handle absolute x and y axis. */
|
||||||
|
if (ioctl(uinput_fd, UI_SET_ABSBIT, ABS_X) == -1)
|
||||||
|
goto err;
|
||||||
|
if (ioctl(uinput_fd, UI_SET_ABSBIT, ABS_Y) == -1)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* Configure device to handle all keys, see linux/input.h. */
|
||||||
|
for (i = 0; i < KEY_MAX; i++) {
|
||||||
|
if (ioctl(uinput_fd, UI_SET_KEYBIT, i) == -1)
|
||||||
|
goto err;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Set device-specific information. */
|
||||||
|
memset(&user_dev, 0, sizeof(user_dev));
|
||||||
|
strncpy(user_dev.name, device_name, UINPUT_MAX_NAME_SIZE);
|
||||||
|
user_dev.id.bustype = id->bustype;
|
||||||
|
user_dev.id.vendor = id->vendor;
|
||||||
|
user_dev.id.product = id->product;
|
||||||
|
user_dev.id.version = id->version;
|
||||||
|
|
||||||
|
//minor tweak to support ABSolute events
|
||||||
|
user_dev.absmin[ABS_X] = -2047;
|
||||||
|
user_dev.absmax[ABS_X] = 2048;
|
||||||
|
user_dev.absfuzz[ABS_X] = 0;
|
||||||
|
user_dev.absflat[ABS_X] = 0;
|
||||||
|
|
||||||
|
user_dev.absmin[ABS_Y] = -2047;
|
||||||
|
user_dev.absmax[ABS_Y] = 2048;
|
||||||
|
user_dev.absfuzz[ABS_Y] = 0;
|
||||||
|
user_dev.absflat[ABS_Y] = 0;
|
||||||
|
|
||||||
|
if (write(uinput_fd, &user_dev, sizeof(user_dev)) != sizeof(user_dev))
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
|
||||||
|
if (ioctl(uinput_fd, UI_DEV_CREATE) == -1)
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
/*
|
||||||
|
The reason for generating a small delay is that creating succesfully
|
||||||
|
an uinput device does not guarantee that the device is ready to process
|
||||||
|
input events. It's probably due the asynchronous nature of the udev.
|
||||||
|
However, my experiments show that the device is not ready to process input
|
||||||
|
events even after a device creation event is received from udev.
|
||||||
|
*/
|
||||||
|
|
||||||
|
//sleep(2);
|
||||||
|
|
||||||
|
return uinput_fd;
|
||||||
|
|
||||||
|
err:
|
||||||
|
|
||||||
|
/*
|
||||||
|
At this point, errno is set for some reason. However, cleanup-actions
|
||||||
|
can also fail and reset errno, therefore we store the original one
|
||||||
|
and reset it before returning.
|
||||||
|
*/
|
||||||
|
original_errno = errno;
|
||||||
|
|
||||||
|
/* Cleanup. */
|
||||||
|
close(uinput_fd); /* Might fail, but we don't care anymore at this point. */
|
||||||
|
|
||||||
|
errno = original_errno;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_close(int uinput_fd)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
Sleep before destroying the device because there still can be some
|
||||||
|
unprocessed events. This is not the right way, but I am still
|
||||||
|
looking for better ways. The question is: how to know whether there
|
||||||
|
are any unprocessed uinput events?
|
||||||
|
*/
|
||||||
|
sleep(2);
|
||||||
|
|
||||||
|
if (ioctl(uinput_fd, UI_DEV_DESTROY) == -1) {
|
||||||
|
close(uinput_fd);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (close(uinput_fd) == -1)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_move_pointer(int uinput_fd, int32_t x, int32_t y)
|
||||||
|
{
|
||||||
|
if (suinput_write(uinput_fd, EV_REL, REL_X, x))
|
||||||
|
return -1;
|
||||||
|
return suinput_write_syn(uinput_fd, EV_REL, REL_Y, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_set_pointer(int uinput_fd, int32_t x, int32_t y)
|
||||||
|
{
|
||||||
|
if (suinput_write(uinput_fd, EV_ABS, ABS_X, x))
|
||||||
|
return -1;
|
||||||
|
return suinput_write_syn(uinput_fd, EV_ABS, ABS_Y, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_press(int uinput_fd, uint16_t code)
|
||||||
|
{
|
||||||
|
return suinput_write(uinput_fd, EV_KEY, code, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_release(int uinput_fd, uint16_t code)
|
||||||
|
{
|
||||||
|
return suinput_write(uinput_fd, EV_KEY, code, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int suinput_click(int uinput_fd, uint16_t code)
|
||||||
|
{
|
||||||
|
if (suinput_press(uinput_fd, code))
|
||||||
|
return -1;
|
||||||
|
return suinput_release(uinput_fd, code);
|
||||||
|
}
|
96
jni/vnc/suinput/suinput.h
Executable file
96
jni/vnc/suinput/suinput.h
Executable file
@ -0,0 +1,96 @@
|
|||||||
|
/*
|
||||||
|
suinput - Simple C-API to the Linux uinput-system.
|
||||||
|
Copyright (C) 2009 Tuomas Räsänen <tuos@codegrove.org>
|
||||||
|
|
||||||
|
This library is free software; you can redistribute it and/or
|
||||||
|
modify it under the terms of the GNU Lesser General Public
|
||||||
|
License as published by the Free Software Foundation; either
|
||||||
|
version 3 of the License, or (at your option) any later version.
|
||||||
|
|
||||||
|
This library 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
|
||||||
|
Lesser General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU Lesser General Public
|
||||||
|
License along with this library; if not, write to the Free Software
|
||||||
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef SUINPUT_H
|
||||||
|
#define SUINPUT_H
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#include <linux/input.h>
|
||||||
|
#include <linux/uinput.h>
|
||||||
|
|
||||||
|
int suinput_write(int uinput_fd,
|
||||||
|
uint16_t type, uint16_t code, int32_t value);
|
||||||
|
/*
|
||||||
|
Creates and opens a connection to the event device. Returns an uinput file
|
||||||
|
descriptor on success. On error, -1 is returned, and errno is set
|
||||||
|
appropriately.
|
||||||
|
*/
|
||||||
|
int suinput_open(const char* device_name, const struct input_id* id);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Destroys and closes a connection to the event device. Returns 0 on success.
|
||||||
|
On error, -1 is returned, and errno is set appropriately.
|
||||||
|
|
||||||
|
Behaviour is undefined when passed a file descriptor not returned by
|
||||||
|
suinput_open().
|
||||||
|
*/
|
||||||
|
int suinput_close(int uinput_fd);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Sends a relative pointer motion event to the event device. Values increase
|
||||||
|
towards right-bottom. Returns 0 on success. On error, -1 is returned, and
|
||||||
|
errno is set appropriately.
|
||||||
|
|
||||||
|
Behaviour is undefined when passed a file descriptor not returned by
|
||||||
|
suinput_open().
|
||||||
|
*/
|
||||||
|
int suinput_move_pointer(int uinput_fd, int32_t x, int32_t y);
|
||||||
|
int suinput_set_pointer(int uinput_fd, int32_t x, int32_t y);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Sends a press event to the event device. Event is repeated after
|
||||||
|
a short delay until a release event is sent. Returns 0 on success.
|
||||||
|
On error, -1 is returned, and errno is set appropriately.
|
||||||
|
|
||||||
|
Behaviour is undefined when passed a file descriptor not returned by
|
||||||
|
suinput_open().
|
||||||
|
|
||||||
|
All possible values of `code` are defined in linux/input.h prefixed
|
||||||
|
by KEY_ or BTN_.
|
||||||
|
*/
|
||||||
|
int suinput_press(int uinput_fd, uint16_t code);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Sends a release event to the event device. Returns 0 on success.
|
||||||
|
On error, -1 is returned, and errno is set appropriately.
|
||||||
|
|
||||||
|
Behaviour is undefined when passed a file descriptor not returned by
|
||||||
|
suinput_open().
|
||||||
|
|
||||||
|
All possible values of `code` are defined in linux/input.h prefixed
|
||||||
|
by KEY_ or BTN_.
|
||||||
|
*/
|
||||||
|
int suinput_release(int uinput_fd, uint16_t code);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Sends a press and release events to the event device. Returns 0 on
|
||||||
|
success. On error, -1 is returned, and errno is set appropriately.
|
||||||
|
|
||||||
|
Behaviour is undefined when passed a file descriptor not returned by
|
||||||
|
suinput_open().
|
||||||
|
|
||||||
|
All possible values of `code` are defined in linux/input.h prefixed
|
||||||
|
by KEY_ or BTN_.
|
||||||
|
|
||||||
|
This function is provided as a convenience and has effectively the
|
||||||
|
same result as calling suinput_press() and suinput_release() sequentially.
|
||||||
|
*/
|
||||||
|
int suinput_click(int uinput_fd, uint16_t code);
|
||||||
|
|
||||||
|
#endif /* SUINPUT_H */
|
Loading…
x
Reference in New Issue
Block a user