xserver-multidpi/hw/xfree86/drivers/inputtest/xf86-input-inputtest.c
Povilas Kanapickas 8c0afc9eb2 xfree86: Implement a test input driver
Signed-off-by: Povilas Kanapickas <povilas@radix.lt>
2020-12-14 01:12:10 +00:00

1028 lines
31 KiB
C

/*
* Copyright © 2013-2017 Red Hat, Inc.
* Copyright © 2020 Povilas Kanapickas <povilas@radix.lt>
*
* Permission to use, copy, modify, distribute, and sell this software
* and its documentation for any purpose is hereby granted without
* fee, provided that the above copyright notice appear in all copies
* and that both that copyright notice and this permission notice
* appear in supporting documentation, and that the name of Red Hat
* not be used in advertising or publicity pertaining to distribution
* of the software without specific, written prior permission. Red
* Hat makes no representations about the suitability of this software
* for any purpose. It is provided "as is" without express or implied
* warranty.
*
* THE AUTHORS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN
* NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
* OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
* NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
#ifdef HAVE_XORG_CONFIG_H
#include <xorg-config.h>
#endif
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#include <exevents.h>
#include <input.h>
#include <xkbsrv.h>
#include <xf86.h>
#include <xf86Xinput.h>
#include "xorgVersion.h"
#include <xserver-properties.h>
#include <os.h>
#include <X11/Xatom.h>
#include <linux/input.h>
#include <sys/socket.h>
#include <sys/stat.h>
#include <sys/un.h>
#include <stdbool.h>
#include "xf86-input-inputtest-protocol.h"
#define POINTER_NUM_AXES 4 /* x, y, hscroll, vscroll */
#define TOUCH_NUM_AXES 4 /* x, y, hscroll, vscroll */
#define TOUCH_MAX_SLOTS 15
#define TOUCH_AXIS_MAX 0xffff
#define TABLET_PRESSURE_AXIS_MAX 2047
#define EVENT_BUFFER_SIZE 4096
enum xf86ITDeviceType {
DEVICE_KEYBOARD = 1,
DEVICE_POINTER,
DEVICE_POINTER_ABS,
DEVICE_POINTER_ABS_PROXIMITY,
DEVICE_TOUCH,
};
enum xf86ITClientState {
CLIENT_STATE_NOT_CONNECTED = 0,
/* connection_fd is valid */
CLIENT_STATE_NEW,
/* connection_fd is valid and client_protocol.{major,minor} are set */
CLIENT_STATE_READY,
};
typedef struct {
InputInfoPtr pInfo;
int socket_fd; /* for accepting new clients */
int connection_fd; /* current client connection */
char *socket_path;
enum xf86ITClientState client_state;
struct {
int major, minor;
} client_protocol;
struct {
char data[EVENT_BUFFER_SIZE];
int valid_length;
} buffer;
uint32_t device_type;
/* last_processed_event_num == last_event_num and waiting_for_drain != 0 must never be true
both at the same time. This would mean that we are waiting for the input queue to be
processed, yet all events have already been processed, i.e. a deadlock.
waiting_for_drain_mutex protects concurrent access to waiting_for_drain variable which
may be modified from multiple threads.
*/
pthread_mutex_t waiting_for_drain_mutex;
bool waiting_for_drain;
int last_processed_event_num;
int last_event_num;
ValuatorMask *valuators;
ValuatorMask *valuators_unaccelerated;
} xf86ITDevice, *xf86ITDevicePtr;
static void
read_input_from_connection(InputInfoPtr pInfo);
static Bool
notify_sync_finished(ClientPtr ptr, void *closure)
{
int fd = (int)(intptr_t) closure;
xf86ITResponseSyncFinished response;
response.header.length = sizeof(response);
response.header.type = XF86IT_RESPONSE_SYNC_FINISHED;
input_lock();
/* we don't really care whether the write succeeds. It may fail if the device is
already shut down and the descriptor is closed.
*/
if (write(fd, &response, response.header.length) != response.header.length) {
LogMessageVerbSigSafe(X_ERROR, 0,
"inputtest: Failed to write sync response: %s\n",
strerror(errno));
}
input_unlock();
return TRUE;
}
static void
input_drain_callback(CallbackListPtr *callback, void *data, void *call_data)
{
void *drain_write_closure;
InputInfoPtr pInfo = data;
xf86ITDevicePtr driver_data = pInfo->private;
bool notify_synchronization = false;
pthread_mutex_lock(&driver_data->waiting_for_drain_mutex);
driver_data->last_processed_event_num = driver_data->last_event_num;
if (driver_data->waiting_for_drain) {
driver_data->waiting_for_drain = false;
notify_synchronization = true;
}
pthread_mutex_unlock(&driver_data->waiting_for_drain_mutex);
if (notify_synchronization) {
drain_write_closure = (void*)(intptr_t) driver_data->connection_fd;
/* One input event may result in additional sets of events being submitted to the
input queue from the input processing code itself. This results in
input_drain_callback being called multiple times.
We therefore schedule a WorkProc (to be run when the server is no longer busy)
to notify the client when all current events have been processed.
*/
xf86IDrvMsg(pInfo, X_DEBUG, "Synchronization finished\n");
QueueWorkProc(notify_sync_finished, NULL, drain_write_closure);
}
}
static void
read_events(int fd, int ready, void *data)
{
DeviceIntPtr dev = (DeviceIntPtr) data;
InputInfoPtr pInfo = dev->public.devicePrivate;
read_input_from_connection(pInfo);
}
static void
try_accept_connection(int fd, int ready, void *data)
{
DeviceIntPtr dev = (DeviceIntPtr) data;
InputInfoPtr pInfo = dev->public.devicePrivate;
xf86ITDevicePtr driver_data = pInfo->private;
int connection_fd;
int flags;
if (driver_data->connection_fd >= 0)
return;
connection_fd = accept(driver_data->socket_fd, NULL, NULL);
if (connection_fd < 0) {
if (errno == EAGAIN || errno == EWOULDBLOCK)
return;
xf86IDrvMsg(pInfo, X_ERROR, "Failed to accept a connection\n");
return;
}
xf86IDrvMsg(pInfo, X_DEBUG, "Accepted input control connection\n");
flags = fcntl(connection_fd, F_GETFL, 0);
fcntl(connection_fd, F_SETFL, flags | O_NONBLOCK);
driver_data->connection_fd = connection_fd;
xf86AddInputEventDrainCallback(input_drain_callback, pInfo);
SetNotifyFd(driver_data->connection_fd, read_events, X_NOTIFY_READ, dev);
driver_data->client_state = CLIENT_STATE_NEW;
}
static int
device_on(DeviceIntPtr dev)
{
InputInfoPtr pInfo = dev->public.devicePrivate;
xf86ITDevicePtr driver_data = pInfo->private;
xf86IDrvMsg(pInfo, X_DEBUG, "Device turned on\n");
xf86AddEnabledDevice(pInfo);
dev->public.on = TRUE;
driver_data->buffer.valid_length = 0;
try_accept_connection(-1, 0, dev);
if (driver_data->connection_fd < 0)
SetNotifyFd(driver_data->socket_fd, try_accept_connection, X_NOTIFY_READ, dev);
return Success;
}
static void
teardown_client_connection(InputInfoPtr pInfo)
{
xf86ITDevicePtr driver_data = pInfo->private;
if (driver_data->client_state != CLIENT_STATE_NOT_CONNECTED) {
RemoveNotifyFd(driver_data->connection_fd);
xf86RemoveInputEventDrainCallback(input_drain_callback, pInfo);
close(driver_data->connection_fd);
driver_data->connection_fd = -1;
}
RemoveNotifyFd(driver_data->socket_fd);
driver_data->client_state = CLIENT_STATE_NOT_CONNECTED;
}
static int
device_off(DeviceIntPtr dev)
{
InputInfoPtr pInfo = dev->public.devicePrivate;
xf86IDrvMsg(pInfo, X_DEBUG, "Device turned off\n");
if (dev->public.on) {
teardown_client_connection(pInfo);
xf86RemoveEnabledDevice(pInfo);
}
dev->public.on = FALSE;
return Success;
}
static void
ptr_ctl(DeviceIntPtr dev, PtrCtrl *ctl)
{
}
static void
init_button_map(unsigned char *btnmap, size_t size)
{
int i;
memset(btnmap, 0, size);
for (i = 0; i < size; i++)
btnmap[i] = i;
}
static void
init_button_labels(Atom *labels, size_t size)
{
assert(size > 10);
memset(labels, 0, size * sizeof(Atom));
labels[0] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_LEFT);
labels[1] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_MIDDLE);
labels[2] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_RIGHT);
labels[3] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_WHEEL_UP);
labels[4] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_WHEEL_DOWN);
labels[5] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_HWHEEL_LEFT);
labels[6] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_HWHEEL_RIGHT);
labels[7] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_SIDE);
labels[8] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_EXTRA);
labels[9] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_FORWARD);
labels[10] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_BACK);
}
static void
init_axis_labels(Atom *labels, size_t size)
{
memset(labels, 0, size * sizeof(Atom));
labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_X);
labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_Y);
labels[2] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_HSCROLL);
labels[3] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_VSCROLL);
}
static void
init_pointer(InputInfoPtr pInfo)
{
DeviceIntPtr dev= pInfo->dev;
int min, max, res;
int nbuttons = 7;
unsigned char btnmap[MAX_BUTTONS + 1];
Atom btnlabels[MAX_BUTTONS];
Atom axislabels[POINTER_NUM_AXES];
nbuttons = xf86SetIntOption(pInfo->options, "PointerButtonCount", 7);
init_button_map(btnmap, ARRAY_SIZE(btnmap));
init_button_labels(btnlabels, ARRAY_SIZE(btnlabels));
init_axis_labels(axislabels, ARRAY_SIZE(axislabels));
InitPointerDeviceStruct((DevicePtr)dev,
btnmap,
nbuttons,
btnlabels,
ptr_ctl,
GetMotionHistorySize(),
POINTER_NUM_AXES,
axislabels);
min = -1;
max = -1;
res = 0;
xf86InitValuatorAxisStruct(dev, 0, XIGetKnownProperty(AXIS_LABEL_PROP_REL_X),
min, max, res * 1000, 0, res * 1000, Relative);
xf86InitValuatorAxisStruct(dev, 1, XIGetKnownProperty(AXIS_LABEL_PROP_REL_Y),
min, max, res * 1000, 0, res * 1000, Relative);
SetScrollValuator(dev, 2, SCROLL_TYPE_HORIZONTAL, 15, 0);
SetScrollValuator(dev, 3, SCROLL_TYPE_VERTICAL, 15, 0);
}
static void
init_pointer_absolute(InputInfoPtr pInfo)
{
DeviceIntPtr dev = pInfo->dev;
int min, max, res;
int nbuttons = 7;
unsigned char btnmap[MAX_BUTTONS + 1];
Atom btnlabels[MAX_BUTTONS];
Atom axislabels[POINTER_NUM_AXES];
nbuttons = xf86SetIntOption(pInfo->options, "PointerButtonCount", 7);
init_button_map(btnmap, ARRAY_SIZE(btnmap));
init_button_labels(btnlabels, ARRAY_SIZE(btnlabels));
init_axis_labels(axislabels, ARRAY_SIZE(axislabels));
InitPointerDeviceStruct((DevicePtr)dev,
btnmap,
nbuttons,
btnlabels,
ptr_ctl,
GetMotionHistorySize(),
POINTER_NUM_AXES,
axislabels);
min = 0;
max = TOUCH_AXIS_MAX;
res = 0;
xf86InitValuatorAxisStruct(dev, 0, XIGetKnownProperty(AXIS_LABEL_PROP_ABS_X),
min, max, res * 1000, 0, res * 1000, Absolute);
xf86InitValuatorAxisStruct(dev, 1, XIGetKnownProperty(AXIS_LABEL_PROP_ABS_Y),
min, max, res * 1000, 0, res * 1000, Absolute);
SetScrollValuator(dev, 2, SCROLL_TYPE_HORIZONTAL, 15, 0);
SetScrollValuator(dev, 3, SCROLL_TYPE_VERTICAL, 15, 0);
}
static void
init_proximity(InputInfoPtr pInfo)
{
DeviceIntPtr dev = pInfo->dev;
InitProximityClassDeviceStruct(dev);
}
static void
init_keyboard(InputInfoPtr pInfo)
{
DeviceIntPtr dev= pInfo->dev;
XkbRMLVOSet rmlvo = {0};
XkbRMLVOSet defaults = {0};
XkbGetRulesDflts(&defaults);
rmlvo.rules = xf86SetStrOption(pInfo->options, "xkb_rules", defaults.rules);
rmlvo.model = xf86SetStrOption(pInfo->options, "xkb_model", defaults.model);
rmlvo.layout = xf86SetStrOption(pInfo->options, "xkb_layout", defaults.layout);
rmlvo.variant = xf86SetStrOption(pInfo->options, "xkb_variant", defaults.variant);
rmlvo.options = xf86SetStrOption(pInfo->options, "xkb_options", defaults.options);
InitKeyboardDeviceStruct(dev, &rmlvo, NULL, NULL);
XkbFreeRMLVOSet(&rmlvo, FALSE);
XkbFreeRMLVOSet(&defaults, FALSE);
}
static void
init_touch(InputInfoPtr pInfo)
{
DeviceIntPtr dev = pInfo->dev;
int min, max, res;
unsigned char btnmap[MAX_BUTTONS + 1];
Atom btnlabels[MAX_BUTTONS];
Atom axislabels[TOUCH_NUM_AXES];
int nbuttons = 7;
int ntouches = TOUCH_MAX_SLOTS;
init_button_map(btnmap, ARRAY_SIZE(btnmap));
init_button_labels(btnlabels, ARRAY_SIZE(btnlabels));
init_axis_labels(axislabels, ARRAY_SIZE(axislabels));
InitPointerDeviceStruct((DevicePtr)dev,
btnmap,
nbuttons,
btnlabels,
ptr_ctl,
GetMotionHistorySize(),
TOUCH_NUM_AXES,
axislabels);
min = 0;
max = TOUCH_AXIS_MAX;
res = 0;
xf86InitValuatorAxisStruct(dev, 0,
XIGetKnownProperty(AXIS_LABEL_PROP_ABS_MT_POSITION_X),
min, max, res * 1000, 0, res * 1000, Absolute);
xf86InitValuatorAxisStruct(dev, 1,
XIGetKnownProperty(AXIS_LABEL_PROP_ABS_MT_POSITION_Y),
min, max, res * 1000, 0, res * 1000, Absolute);
xf86InitValuatorAxisStruct(dev, 2,
XIGetKnownProperty(AXIS_LABEL_PROP_ABS_MT_PRESSURE),
min, TABLET_PRESSURE_AXIS_MAX, res * 1000, 0, res * 1000, Absolute);
ntouches = xf86SetIntOption(pInfo->options, "TouchCount", TOUCH_MAX_SLOTS);
if (ntouches == 0) /* unknown */
ntouches = TOUCH_MAX_SLOTS;
InitTouchClassDeviceStruct(dev, ntouches, XIDirectTouch, 2);
}
static void
device_init(DeviceIntPtr dev)
{
InputInfoPtr pInfo = dev->public.devicePrivate;
xf86ITDevicePtr driver_data = pInfo->private;
dev->public.on = FALSE;
switch (driver_data->device_type) {
case DEVICE_KEYBOARD:
init_keyboard(pInfo);
break;
case DEVICE_POINTER:
init_pointer(pInfo);
break;
case DEVICE_POINTER_ABS:
init_pointer_absolute(pInfo);
break;
case DEVICE_POINTER_ABS_PROXIMITY:
init_pointer_absolute(pInfo);
init_proximity(pInfo);
break;
case DEVICE_TOUCH:
init_touch(pInfo);
break;
}
}
static void
device_destroy(DeviceIntPtr dev)
{
InputInfoPtr pInfo = dev->public.devicePrivate;
xf86IDrvMsg(pInfo, X_INFO, "Close\n");
}
static int
device_control(DeviceIntPtr dev, int mode)
{
switch (mode) {
case DEVICE_INIT:
device_init(dev);
break;
case DEVICE_ON:
device_on(dev);
break;
case DEVICE_OFF:
device_off(dev);
break;
case DEVICE_CLOSE:
device_destroy(dev);
break;
}
return Success;
}
static void
convert_to_valuator_mask(xf86ITValuatorData *event, ValuatorMask *mask)
{
valuator_mask_zero(mask);
for (int i = 0; i < min(XF86IT_MAX_VALUATORS, MAX_VALUATORS); ++i) {
if (BitIsOn(event->mask, i)) {
if (event->has_unaccelerated) {
valuator_mask_set_unaccelerated(mask, i, event->valuators[i],
event->unaccelerated[i]);
} else {
valuator_mask_set_double(mask, i, event->valuators[i]);
}
}
}
}
static void
handle_client_version(InputInfoPtr pInfo, xf86ITEventClientVersion *event)
{
xf86ITDevicePtr driver_data = pInfo->private;
xf86ITResponseServerVersion response;
response.header.length = sizeof(response);
response.header.type = XF86IT_RESPONSE_SERVER_VERSION;
response.major = XF86IT_PROTOCOL_VERSION_MAJOR;
response.minor = XF86IT_PROTOCOL_VERSION_MINOR;
if (write(driver_data->connection_fd, &response, response.header.length) != response.header.length) {
xf86IDrvMsg(pInfo, X_ERROR, "Error writing driver version: %s\n", strerror(errno));
teardown_client_connection(pInfo);
return;
}
if (event->major != XF86IT_PROTOCOL_VERSION_MAJOR ||
event->minor > XF86IT_PROTOCOL_VERSION_MINOR)
{
xf86IDrvMsg(pInfo, X_ERROR, "Unsupported protocol version: %d.%d (current %d.%d)\n",
event->major, event->minor,
XF86IT_PROTOCOL_VERSION_MAJOR,
XF86IT_PROTOCOL_VERSION_MINOR);
teardown_client_connection(pInfo);
return;
}
driver_data->client_protocol.major = event->major;
driver_data->client_protocol.minor = event->minor;
driver_data->client_state = CLIENT_STATE_READY;
}
static void
handle_wait_for_sync(InputInfoPtr pInfo)
{
xf86ITDevicePtr driver_data = pInfo->private;
bool notify_synchronization = false;
void *drain_write_closure;
xf86IDrvMsg(pInfo, X_DEBUG, "Handling sync event\n");
pthread_mutex_lock(&driver_data->waiting_for_drain_mutex);
if (driver_data->last_processed_event_num == driver_data->last_event_num) {
notify_synchronization = true;
} else {
driver_data->waiting_for_drain = true;
}
pthread_mutex_unlock(&driver_data->waiting_for_drain_mutex);
if (notify_synchronization) {
drain_write_closure = (void*)(intptr_t) driver_data->connection_fd;
xf86IDrvMsg(pInfo, X_DEBUG, "Synchronization finished\n");
notify_sync_finished(NULL, drain_write_closure);
}
}
static void
handle_motion(InputInfoPtr pInfo, xf86ITEventMotion *event)
{
DeviceIntPtr dev = pInfo->dev;
xf86ITDevicePtr driver_data = pInfo->private;
ValuatorMask *mask = driver_data->valuators;
xf86IDrvMsg(pInfo, X_DEBUG, "Handling motion event\n");
driver_data->last_event_num++;
convert_to_valuator_mask(&event->valuators, mask);
xf86PostMotionEventM(dev, event->is_absolute ? Absolute : Relative, mask);
}
static void
handle_proximity(InputInfoPtr pInfo, xf86ITEventProximity *event)
{
DeviceIntPtr dev = pInfo->dev;
xf86ITDevicePtr driver_data = pInfo->private;
ValuatorMask *mask = driver_data->valuators;
xf86IDrvMsg(pInfo, X_DEBUG, "Handling proximity event\n");
driver_data->last_event_num++;
convert_to_valuator_mask(&event->valuators, mask);
xf86PostProximityEventM(dev, event->is_prox_in, mask);
}
static void
handle_button(InputInfoPtr pInfo, xf86ITEventButton *event)
{
DeviceIntPtr dev = pInfo->dev;
xf86ITDevicePtr driver_data = pInfo->private;
ValuatorMask *mask = driver_data->valuators;
xf86IDrvMsg(pInfo, X_DEBUG, "Handling button event\n");
driver_data->last_event_num++;
convert_to_valuator_mask(&event->valuators, mask);
xf86PostButtonEventM(dev, event->is_absolute ? Absolute : Relative, event->button,
event->is_press, mask);
}
static void
handle_key(InputInfoPtr pInfo, xf86ITEventKey *event)
{
DeviceIntPtr dev = pInfo->dev;
xf86ITDevicePtr driver_data = pInfo->private;
xf86IDrvMsg(pInfo, X_DEBUG, "Handling key event\n");
driver_data->last_event_num++;
xf86PostKeyboardEvent(dev, event->key_code, event->is_press);
}
static void
handle_touch(InputInfoPtr pInfo, xf86ITEventTouch *event)
{
DeviceIntPtr dev = pInfo->dev;
xf86ITDevicePtr driver_data = pInfo->private;
ValuatorMask *mask = driver_data->valuators;
xf86IDrvMsg(pInfo, X_DEBUG, "Handling touch event\n");
driver_data->last_event_num++;
convert_to_valuator_mask(&event->valuators, mask);
xf86PostTouchEvent(dev, event->touchid, event->touch_type, 0, mask);
}
static void
client_new_handle_event(InputInfoPtr pInfo, xf86ITEventAny *event)
{
switch (event->header.type) {
case XF86IT_EVENT_CLIENT_VERSION:
handle_client_version(pInfo, &event->version);
break;
default:
xf86IDrvMsg(pInfo, X_ERROR, "Event before client is ready: event type %d\n",
event->header.type);
teardown_client_connection(pInfo);
break;
}
}
static void
client_ready_handle_event(InputInfoPtr pInfo, xf86ITEventAny *event)
{
switch (event->header.type) {
case XF86IT_EVENT_WAIT_FOR_SYNC:
handle_wait_for_sync(pInfo);
break;
case XF86IT_EVENT_MOTION:
handle_motion(pInfo, &event->motion);
break;
case XF86IT_EVENT_PROXIMITY:
handle_proximity(pInfo, &event->proximity);
break;
case XF86IT_EVENT_BUTTON:
handle_button(pInfo, &event->button);
break;
case XF86IT_EVENT_KEY:
handle_key(pInfo, &event->key);
break;
case XF86IT_EVENT_TOUCH:
handle_touch(pInfo, &event->touch);
break;
case XF86IT_EVENT_CLIENT_VERSION:
xf86IDrvMsg(pInfo, X_ERROR, "Only single ClientVersion event is allowed\n");
teardown_client_connection(pInfo);
break;
default:
xf86IDrvMsg(pInfo, X_ERROR, "Invalid event when client is ready %d\n",
event->header.type);
teardown_client_connection(pInfo);
break;
}
}
static void
handle_event(InputInfoPtr pInfo, xf86ITEventAny *event)
{
xf86ITDevicePtr driver_data = pInfo->private;
if (!pInfo->dev->public.on)
return;
switch (driver_data->client_state) {
case CLIENT_STATE_NOT_CONNECTED:
xf86IDrvMsg(pInfo, X_ERROR, "Got event when client is not connected\n");
break;
case CLIENT_STATE_NEW:
client_new_handle_event(pInfo, event);
break;
case CLIENT_STATE_READY:
client_ready_handle_event(pInfo, event);
break;
}
}
static bool
is_supported_event(enum xf86ITEventType type)
{
switch (type) {
case XF86IT_EVENT_CLIENT_VERSION:
case XF86IT_EVENT_WAIT_FOR_SYNC:
case XF86IT_EVENT_MOTION:
case XF86IT_EVENT_PROXIMITY:
case XF86IT_EVENT_BUTTON:
case XF86IT_EVENT_KEY:
case XF86IT_EVENT_TOUCH:
return true;
}
return false;
}
static int
get_event_size(enum xf86ITEventType type)
{
switch (type) {
case XF86IT_EVENT_CLIENT_VERSION: return sizeof(xf86ITEventClientVersion);
case XF86IT_EVENT_WAIT_FOR_SYNC: return sizeof(xf86ITEventWaitForSync);
case XF86IT_EVENT_MOTION: return sizeof(xf86ITEventMotion);
case XF86IT_EVENT_PROXIMITY: return sizeof(xf86ITEventProximity);
case XF86IT_EVENT_BUTTON: return sizeof(xf86ITEventButton);
case XF86IT_EVENT_KEY: return sizeof(xf86ITEventKey);
case XF86IT_EVENT_TOUCH: return sizeof(xf86ITEventTouch);
}
abort();
}
static void
read_input_from_connection(InputInfoPtr pInfo)
{
xf86ITDevicePtr driver_data = pInfo->private;
while (1) {
int processed_size = 0;
int read_size = read(driver_data->connection_fd,
driver_data->buffer.data + driver_data->buffer.valid_length,
EVENT_BUFFER_SIZE - driver_data->buffer.valid_length);
if (read_size < 0) {
if (errno == EAGAIN || errno == EWOULDBLOCK)
return;
xf86IDrvMsg(pInfo, X_ERROR, "Error reading events: %s\n", strerror(errno));
teardown_client_connection(pInfo);
return;
}
driver_data->buffer.valid_length += read_size;
while (1) {
xf86ITEventHeader *event_header;
char *event_begin = driver_data->buffer.data + processed_size;
if (driver_data->buffer.valid_length - processed_size < sizeof(xf86ITEventHeader))
break;
event_header = (xf86ITEventHeader*) event_begin;
if (event_header->length >= EVENT_BUFFER_SIZE) {
xf86IDrvMsg(pInfo, X_ERROR, "Received event with too long length: %d\n",
event_header->length);
teardown_client_connection(pInfo);
return;
}
if (driver_data->buffer.valid_length - processed_size < event_header->length)
break;
if (is_supported_event(event_header->type)) {
int expected_event_size = get_event_size(event_header->type);
if (event_header->length != expected_event_size) {
xf86IDrvMsg(pInfo, X_ERROR, "Unexpected event length: was %d bytes, "
"expected %d (event type: %d)\n",
event_header->length, expected_event_size,
(int) event_header->type);
teardown_client_connection(pInfo);
return;
}
handle_event(pInfo, (xf86ITEventAny*) event_begin);
}
processed_size += event_header->length;
}
if (processed_size > 0) {
memmove(driver_data->buffer.data,
driver_data->buffer.data + processed_size,
driver_data->buffer.valid_length - processed_size);
driver_data->buffer.valid_length -= processed_size;
}
if (read_size == 0)
break;
}
}
static void
read_input(InputInfoPtr pInfo)
{
/* The test input driver does not set up the pInfo->fd and use the regular
read_input callback because we want to only accept the connection to
the controlling socket after the device is turned on.
*/
}
static const char*
get_type_name(InputInfoPtr pInfo, xf86ITDevicePtr driver_data)
{
switch (driver_data->device_type) {
case DEVICE_TOUCH: return XI_TOUCHSCREEN;
case DEVICE_POINTER: return XI_MOUSE;
case DEVICE_POINTER_ABS: return XI_MOUSE;
case DEVICE_POINTER_ABS_PROXIMITY: return XI_TABLET;
case DEVICE_KEYBOARD: return XI_KEYBOARD;
}
xf86IDrvMsg(pInfo, X_ERROR, "Unexpected device type %d\n",
driver_data->device_type);
return XI_KEYBOARD;
}
static xf86ITDevicePtr
device_alloc(void)
{
xf86ITDevicePtr driver_data = calloc(sizeof(xf86ITDevice), 1);
if (!driver_data)
return NULL;
driver_data->socket_fd = -1;
driver_data->connection_fd = -1;
return driver_data;
}
static void
free_driver_data(xf86ITDevicePtr driver_data)
{
if (driver_data) {
close(driver_data->connection_fd);
close(driver_data->socket_fd);
if (driver_data->socket_path)
unlink(driver_data->socket_path);
free(driver_data->socket_path);
pthread_mutex_destroy(&driver_data->waiting_for_drain_mutex);
if (driver_data->valuators)
valuator_mask_free(&driver_data->valuators);
if (driver_data->valuators_unaccelerated)
valuator_mask_free(&driver_data->valuators_unaccelerated);
}
free(driver_data);
}
static int
pre_init(InputDriverPtr drv, InputInfoPtr pInfo, int flags)
{
xf86ITDevicePtr driver_data = NULL;
char *device_type_option;
struct sockaddr_un addr;
pInfo->type_name = 0;
pInfo->device_control = device_control;
pInfo->read_input = read_input;
pInfo->control_proc = NULL;
pInfo->switch_mode = NULL;
driver_data = device_alloc();
if (!driver_data)
goto fail;
driver_data->client_state = CLIENT_STATE_NOT_CONNECTED;
driver_data->last_event_num = 1;
driver_data->last_processed_event_num = 0;
driver_data->waiting_for_drain = false;
pthread_mutex_init(&driver_data->waiting_for_drain_mutex, NULL);
driver_data->valuators = valuator_mask_new(6);
if (!driver_data->valuators)
goto fail;
driver_data->valuators_unaccelerated = valuator_mask_new(2);
if (!driver_data->valuators_unaccelerated)
goto fail;
driver_data->socket_path = xf86SetStrOption(pInfo->options, "SocketPath", NULL);
if (!driver_data->socket_path){
xf86IDrvMsg(pInfo, X_ERROR, "SocketPath must be specified\n");
goto fail;
}
if (strlen(driver_data->socket_path) >= sizeof(addr.sun_path)) {
xf86IDrvMsg(pInfo, X_ERROR, "SocketPath is too long\n");
goto fail;
}
unlink(driver_data->socket_path);
driver_data->socket_fd = socket(PF_UNIX, SOCK_STREAM | SOCK_NONBLOCK, 0);
if (driver_data->socket_fd < 0) {
xf86IDrvMsg(pInfo, X_ERROR, "Failed to create a socket for communication: %s\n",
strerror(errno));
goto fail;
}
memset(&addr, 0, sizeof(addr));
addr.sun_family = AF_UNIX;
strncpy(addr.sun_path, driver_data->socket_path, sizeof(addr.sun_path) - 1);
if (bind(driver_data->socket_fd, (struct sockaddr*) &addr, sizeof(addr)) < 0) {
xf86IDrvMsg(pInfo, X_ERROR, "Failed to assign address to the socket\n");
goto fail;
}
if (chmod(driver_data->socket_path, 0777) != 0) {
xf86IDrvMsg(pInfo, X_ERROR, "Failed to chmod the socket path\n");
goto fail;
}
if (listen(driver_data->socket_fd, 1) != 0) {
xf86IDrvMsg(pInfo, X_ERROR, "Failed to listen on the socket\n");
goto fail;
}
device_type_option = xf86SetStrOption(pInfo->options, "DeviceType", NULL);
if (device_type_option == NULL) {
xf86IDrvMsg(pInfo, X_ERROR, "DeviceType option must be specified\n");
goto fail;
}
if (strcmp(device_type_option, "Keyboard") == 0) {
driver_data->device_type = DEVICE_KEYBOARD;
} else if (strcmp(device_type_option, "Pointer") == 0) {
driver_data->device_type = DEVICE_POINTER;
} else if (strcmp(device_type_option, "PointerAbsolute") == 0) {
driver_data->device_type = DEVICE_POINTER_ABS;
} else if (strcmp(device_type_option, "PointerAbsoluteProximity") == 0) {
driver_data->device_type = DEVICE_POINTER_ABS_PROXIMITY;
} else if (strcmp(device_type_option, "Touch") == 0) {
driver_data->device_type = DEVICE_TOUCH;
} else {
xf86IDrvMsg(pInfo, X_ERROR, "Unsupported DeviceType option.\n");
goto fail;
}
free(device_type_option);
pInfo->private = driver_data;
driver_data->pInfo = pInfo;
pInfo->type_name = get_type_name(pInfo, driver_data);
return Success;
fail:
free_driver_data(driver_data);
return BadValue;
}
static void
uninit(InputDriverPtr drv, InputInfoPtr pInfo, int flags)
{
xf86ITDevicePtr driver_data = pInfo->private;
free_driver_data(driver_data);
pInfo->private = NULL;
xf86DeleteInput(pInfo, flags);
}
InputDriverRec driver = {
.driverVersion = 1,
.driverName = "inputtest",
.PreInit = pre_init,
.UnInit = uninit,
.module = NULL,
.default_options = NULL,
.capabilities = 0
};
static XF86ModuleVersionInfo version_info = {
"inputtest",
MODULEVENDORSTRING,
MODINFOSTRING1,
MODINFOSTRING2,
XORG_VERSION_CURRENT,
XORG_VERSION_MAJOR,
XORG_VERSION_MINOR,
XORG_VERSION_PATCH,
ABI_CLASS_XINPUT,
ABI_XINPUT_VERSION,
MOD_CLASS_XINPUT,
{0, 0, 0, 0}
};
static void*
setup_proc(void *module, void *options, int *errmaj, int *errmin)
{
xf86AddInputDriver(&driver, module, 0);
return module;
}
_X_EXPORT XF86ModuleData inputtestModuleData = {
.vers = &version_info,
.setup = &setup_proc,
.teardown = NULL
};