summaryrefslogtreecommitdiff
path: root/driver/xf86-input-ws/src/ws.c
diff options
context:
space:
mode:
authorMatthieu Herrb <matthieu@cvs.openbsd.org>2009-11-22 19:13:11 +0000
committerMatthieu Herrb <matthieu@cvs.openbsd.org>2009-11-22 19:13:11 +0000
commit778a6945b231e33c40179cd16d55d04a0493fdca (patch)
treeafc55485dd8f1396765b0da9f1ced9d52b277ac5 /driver/xf86-input-ws/src/ws.c
parent6f197f5c70d67120f835ca7a9af895218b8ca7c3 (diff)
Re-import the wscons pointer input driver using the correct name.
Diffstat (limited to 'driver/xf86-input-ws/src/ws.c')
-rw-r--r--driver/xf86-input-ws/src/ws.c699
1 files changed, 699 insertions, 0 deletions
diff --git a/driver/xf86-input-ws/src/ws.c b/driver/xf86-input-ws/src/ws.c
new file mode 100644
index 000000000..a8bd718a5
--- /dev/null
+++ b/driver/xf86-input-ws/src/ws.c
@@ -0,0 +1,699 @@
+/*
+ * Copyright (c) 2005 Matthieu Herrb
+ *
+ * Permission to use, copy, modify, and distribute this software for any
+ * purpose with or without fee is hereby granted, provided that the above
+ * copyright notice and this permission notice appear in all copies.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+ * ANY SPECIAL, DIRECT, 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.
+ */
+/* $OpenBSD: ws.c,v 1.1 2009/11/22 19:13:10 matthieu Exp $ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <unistd.h>
+#include <errno.h>
+#include <sys/time.h>
+#include <dev/wscons/wsconsio.h>
+
+#include <xf86.h>
+
+#include <xf86_OSproc.h>
+#include <X11/extensions/XI.h>
+#include <X11/extensions/XIproto.h>
+#include <xf86Xinput.h>
+#include <xisb.h>
+#include <mipointer.h>
+#include <extinit.h>
+
+#ifdef XFree86LOADER
+#include "xf86Module.h"
+#endif
+
+#define NAXES 2 /* X and Y axes only */
+#define NBUTTONS 32 /* max theoretical buttons */
+#define DFLTBUTTONS 3 /* default number of buttons */
+#define NUMEVENTS 16 /* max # of ws events to read at once */
+
+typedef struct WSDevice {
+ char *devName; /* device name */
+ unsigned int buttons; /* # of buttons */
+ unsigned int lastButtons; /* last state of buttons */
+ int x, y; /* current abs coordinates */
+ int min_x, max_x, min_y, max_y; /* coord space */
+ int swap_axes;
+ int inv_x, inv_y;
+ int screen_width, screen_height;
+ int screen_no;
+ int num, den, threshold; /* relative accel params */
+ pointer buffer;
+ int negativeZ, positiveZ; /* mappings for Z axis */
+ int negativeW, positiveW; /* mappings for W axis */
+} WSDeviceRec, *WSDevicePtr;
+
+#ifdef XFree86LOADER
+static MODULESETUPPROTO(SetupProc);
+static void TearDownProc(pointer);
+static const OptionInfoRec *wsAvailableOptions(void *);
+#endif
+
+static InputInfoPtr wsPreInit(InputDriverPtr, IDevPtr, int);
+static int wsProc(DeviceIntPtr, int);
+static void wsReadInput(InputInfoPtr);
+static void wsSendButtons(InputInfoPtr, int);
+static int wsChangeControl(InputInfoPtr, xDeviceCtl *);
+static int wsSwitchMode(ClientPtr, DeviceIntPtr, int);
+static Bool wsOpen(InputInfoPtr);
+static void wsClose(InputInfoPtr);
+static Bool wsConvert(InputInfoPtr, int, int, int, int, int, int, int, int,
+ int *, int *);
+static void wsControlProc(DeviceIntPtr , PtrCtrl *);
+
+
+static XF86ModuleVersionInfo VersionRec = {
+ "ws",
+ MODULEVENDORSTRING,
+ MODINFOSTRING1,
+ MODINFOSTRING2,
+ XORG_VERSION_CURRENT,
+ 1, 0, 0,
+ ABI_CLASS_XINPUT,
+ ABI_XINPUT_VERSION,
+ MOD_CLASS_XINPUT,
+ {0, 0, 0, 0}
+};
+
+typedef enum {
+ WSOPT_DEVICE,
+ WSOPT_DEBUG_LEVEL,
+ WSOPT_MINX,
+ WSOPT_MAXX,
+ WSOPT_MINY,
+ WSOPT_MAXY,
+ WSOPT_ROTATE,
+ WSOPT_SWAPXY,
+ WSOPT_SCREENNO,
+ WSOPT_ZAXIS_MAPPING,
+} WSOpts;
+
+#define WS_NOZMAP 0
+
+static const OptionInfoRec WSOptions[] = {
+ { WSOPT_DEVICE, "device", OPTV_STRING, {0}, FALSE },
+ { WSOPT_DEBUG_LEVEL, "debugLevel", OPTV_INTEGER, {0}, FALSE },
+ { WSOPT_MINX, "minX", OPTV_INTEGER, {0}, FALSE },
+ { WSOPT_MAXX, "maxX", OPTV_INTEGER, {0}, FALSE },
+ { WSOPT_MINY, "minY", OPTV_INTEGER, {0}, FALSE },
+ { WSOPT_MAXY, "maxY", OPTV_INTEGER, {0}, FALSE },
+ { WSOPT_ROTATE, "rotate", OPTV_STRING, {0}, FALSE },
+ { WSOPT_SWAPXY, "swapxy", OPTV_BOOLEAN, {0}, FALSE },
+ { WSOPT_SCREENNO, "ScreenNo", OPTV_INTEGER, {0}, FALSE },
+ { WSOPT_ZAXIS_MAPPING, "ZAxisMapping", OPTV_STRING, {0}, FALSE },
+ { -1, NULL, OPTV_NONE, {0}, FALSE }
+};
+
+XF86ModuleData wsModuleData = {&VersionRec,
+ SetupProc, TearDownProc };
+
+
+InputDriverRec WS = {
+ 1,
+ "ws",
+ NULL,
+ wsPreInit,
+ NULL,
+ NULL,
+ 0
+};
+
+/* #undef DEBUG */
+#define DEBUG
+#undef DBG
+static int debug_level = 0;
+#ifdef DEBUG
+# define DBG(lvl, f) { if ((lvl) <= debug_level) f;}
+#else
+# define DBG(lvl, f)
+#endif
+
+#ifdef XFree86LOADER
+static pointer
+SetupProc(pointer module, pointer options, int *errmaj, int *errmin)
+{
+ static Bool Initialised = FALSE;
+
+ if (!Initialised) {
+ xf86AddInputDriver(&WS, module, 0);
+ Initialised = TRUE;
+ }
+ return module;
+}
+
+static void
+TearDownProc(pointer p)
+{
+ DBG(1, ErrorF("WS TearDownProc called\n"));
+}
+
+static const OptionInfoRec *
+wsAvailableOptions(void *unused)
+{
+ return WSOptions;
+}
+#endif /* XFree86LOADER */
+
+static InputInfoPtr
+wsPreInit(InputDriverPtr drv, IDevPtr dev, int flags)
+{
+ InputInfoPtr pInfo = NULL;
+ WSDevicePtr priv;
+ MessageType buttons_from = X_CONFIG;
+ char *s;
+
+ pInfo = xf86AllocateInput(drv, 0);
+ if (pInfo == NULL) {
+ return NULL;
+ }
+ priv = (WSDevicePtr)xcalloc(1, sizeof(WSDeviceRec));
+ if (priv == NULL)
+ goto fail;
+ pInfo->flags = XI86_POINTER_CAPABLE | XI86_SEND_DRAG_EVENTS;
+ pInfo->conf_idev = dev;
+ pInfo->name = "ws";
+ pInfo->private = priv;
+
+ xf86CollectInputOptions(pInfo, NULL, NULL);
+ xf86ProcessCommonOptions(pInfo, pInfo->options);
+#ifdef DEBUG
+ debug_level = xf86SetIntOption(pInfo->options, "DebugLevel",
+ debug_level);
+ xf86Msg(X_INFO, "%s: debuglevel %d\n", dev->identifier, debug_level);
+#endif
+ priv->devName = xf86FindOptionValue(pInfo->options, "Device");
+ if (priv->devName == NULL) {
+ xf86Msg(X_ERROR, "%s: No Device specified.\n",
+ dev->identifier);
+ goto fail;
+ }
+ priv->buttons = xf86SetIntOption(pInfo->options, "Buttons", 0);
+ if (priv->buttons == 0) {
+ priv->buttons = DFLTBUTTONS;
+ buttons_from = X_DEFAULT;
+ }
+ priv->negativeZ = priv->positiveZ = WS_NOZMAP;
+ s = xf86SetStrOption(pInfo->options, "ZAxisMapping", NULL);
+ if (s) {
+ int b1, b2;
+
+ if (sscanf(s, "%d %d", &b1, &b2) == 2 &&
+ b1 > 0 && b1 <= NBUTTONS &&
+ b2 > 0 && b2 <= NBUTTONS) {
+ priv->negativeZ = b1;
+ priv->positiveZ = b2;
+ xf86Msg(X_CONFIG,
+ "%s: ZAxisMapping: buttons %d and %d\n",
+ pInfo->name, b1, b2);
+ } else {
+ xf86Msg(X_WARNING, "%s: invalid ZAxisMapping value: "
+ "\"%s\"\n", pInfo->name, s);
+ }
+ }
+ if (priv->negativeZ > priv->buttons) {
+ priv->buttons = priv->negativeZ;
+ buttons_from = X_CONFIG;
+ }
+ if (priv->positiveZ > priv->buttons) {
+ priv->buttons = priv->positiveZ;
+ buttons_from = X_CONFIG;
+ }
+ priv->negativeW = priv->positiveW = WS_NOZMAP;
+ s = xf86SetStrOption(pInfo->options, "WAxisMapping", NULL);
+ if (s) {
+ int b1, b2;
+
+ if (sscanf(s, "%d %d", &b1, &b2) == 2 &&
+ b1 > 0 && b1 <= NBUTTONS &&
+ b2 > 0 && b2 <= NBUTTONS) {
+ priv->negativeW = b1;
+ priv->positiveW = b2;
+ xf86Msg(X_CONFIG,
+ "%s: WAxisMapping: buttons %d and %d\n",
+ pInfo->name, b1, b2);
+ } else {
+ xf86Msg(X_WARNING, "%s: invalid WAxisMapping value: "
+ "\"%s\"\n", pInfo->name, s);
+ }
+ }
+ if (priv->negativeW > priv->buttons) {
+ priv->buttons = priv->negativeW;
+ buttons_from = X_CONFIG;
+ }
+ if (priv->positiveW > priv->buttons) {
+ priv->buttons = priv->positiveW;
+ buttons_from = X_CONFIG;
+ }
+
+ priv->screen_no = xf86SetIntOption(pInfo->options, "ScreenNo", 0);
+ xf86Msg(X_CONFIG, "%s associated screen: %d\n",
+ dev->identifier, priv->screen_no);
+ if (priv->screen_no >= screenInfo.numScreens ||
+ priv->screen_no < 0) {
+ priv->screen_no = 0;
+ }
+
+ priv->max_x = xf86SetIntOption(pInfo->options, "MaxX",
+ screenInfo.screens[priv->screen_no]->width - 1);
+ xf86Msg(X_INFO, "%s maximum x position: %d\n",
+ dev->identifier, priv->max_x);
+ priv->min_x = xf86SetIntOption(pInfo->options, "MinX", 0);
+ xf86Msg(X_INFO, "%s minimum x position: %d\n",
+ dev->identifier, priv->min_x);
+ priv->max_y = xf86SetIntOption(pInfo->options, "MaxY",
+ screenInfo.screens[priv->screen_no]->height - 1);
+ xf86Msg(X_INFO, "%s maximum y position: %d\n",
+ dev->identifier, priv->max_y);
+ priv->min_y = xf86SetIntOption(pInfo->options, "MinY", 0);
+ xf86Msg(X_INFO, "%s minimum y position: %d\n",
+ dev->identifier, priv->min_y);
+
+ priv->swap_axes = xf86SetBoolOption(pInfo->options, "SwapXY", 0);
+ if (priv->swap_axes) {
+ xf86Msg(X_CONFIG,
+ "%s device will work with X and Y axes swapped\n",
+ dev->identifier);
+ }
+ priv->inv_x = 0;
+ priv->inv_y = 0;
+ s = xf86FindOptionValue(pInfo->options, "Rotate");
+ if (s) {
+ if (xf86NameCmp(s, "CW") == 0) {
+ priv->inv_x = 1;
+ priv->inv_y = 0;
+ priv->swap_axes = 1;
+ } else if (xf86NameCmp(s, "CCW") == 0) {
+ priv->inv_x = 0;
+ priv->inv_y = 1;
+ priv->swap_axes = 1;
+ } else if (xf86NameCmp(s, "UD") == 0) {
+ priv->inv_x = 1;
+ priv->inv_y = 1;
+ } else {
+ xf86Msg(X_ERROR, "\"%s\" is not a valid value "
+ "for Option \"Rotate\"\n", s);
+ xf86Msg(X_ERROR, "Valid options are \"CW\", \"CCW\","
+ " or \"UD\"\n");
+ }
+ }
+
+ pInfo->name = dev->identifier;
+ pInfo->type_name = "wscons pointer";
+ pInfo->device_control = wsProc;
+ pInfo->read_input = wsReadInput;
+ pInfo->control_proc = wsChangeControl;
+ pInfo->switch_mode = wsSwitchMode;
+ pInfo->conversion_proc = wsConvert;
+ pInfo->reverse_conversion_proc = NULL;
+ pInfo->fd = -1;
+ pInfo->private = priv;
+ pInfo->old_x = -1;
+ pInfo->old_y = -1;
+ xf86Msg(buttons_from, "%s: Buttons: %d\n", pInfo->name, priv->buttons);
+
+ /* mark the device configured */
+ pInfo->flags |= XI86_CONFIGURED;
+ return pInfo;
+fail:
+ if (priv != NULL)
+ xfree(priv);
+ if (pInfo != NULL)
+ xfree(pInfo);
+ return NULL;
+}
+
+static int
+wsProc(DeviceIntPtr pWS, int what)
+{
+ InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate;
+ WSDevicePtr priv = (WSDevicePtr)XI_PRIVATE(pWS);
+ unsigned char map[NBUTTONS + 1];
+ int i;
+
+ switch (what) {
+ case DEVICE_INIT:
+ DBG(1, ErrorF("WS DEVICE_INIT\n"));
+
+ priv->screen_width =
+ screenInfo.screens[priv->screen_no]->width;
+ priv->screen_height =
+ screenInfo.screens[priv->screen_no]->height;
+
+ for (i = 0; i < NBUTTONS; i++)
+ map[i + 1] = i + 1;
+ InitPointerDeviceStruct((DevicePtr)pWS, map,
+ min(priv->buttons, NBUTTONS),
+#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) == 0
+ miPointerGetMotionEvents,
+#elif GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 3
+ GetMotionHistory,
+#endif
+ wsControlProc,
+#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) == 0
+ miPointerGetMotionBufferSize()
+#else
+ GetMotionHistorySize(), NAXES
+#endif
+ );
+ xf86InitValuatorAxisStruct(pWS, 0, 0, -1, 1, 0, 1);
+ xf86InitValuatorDefaults(pWS, 0);
+
+ xf86InitValuatorAxisStruct(pWS, 1, 0, -1, 1, 0, 1);
+ xf86InitValuatorDefaults(pWS, 1);
+ xf86MotionHistoryAllocate(pInfo);
+ AssignTypeAndName(pWS, pInfo->atom, pInfo->name);
+ pWS->public.on = FALSE;
+ /* This sould correspond to the center of the screen */
+ priv->x = (priv->max_x - priv->min_x) / 2;
+ priv->y = (priv->max_y - priv->min_y) / 2;
+ if (wsOpen(pInfo) != Success) {
+ return !Success;
+ }
+ break;
+
+ case DEVICE_ON:
+ DBG(1, ErrorF("WS DEVICE ON\n"));
+ if ((pInfo->fd < 0) && (wsOpen(pInfo) != Success)) {
+ xf86Msg(X_ERROR, "wsOpen failed %s\n",
+ strerror(errno));
+ return !Success;
+ }
+ priv->buffer = XisbNew(pInfo->fd,
+ sizeof(struct wscons_event) * NUMEVENTS);
+ if (priv->buffer == NULL) {
+ xf86Msg(X_ERROR, "cannot alloc xisb buffer\n");
+ wsClose(pInfo);
+ return !Success;
+ }
+ xf86AddEnabledDevice(pInfo);
+ pWS->public.on = TRUE;
+
+ break;
+
+ case DEVICE_OFF:
+ DBG(1, ErrorF("WS DEVICE OFF\n"));
+ if (pInfo->fd >= 0) {
+ xf86RemoveEnabledDevice(pInfo);
+ wsClose(pInfo);
+ }
+ if (priv->buffer) {
+ XisbFree(priv->buffer);
+ priv->buffer = NULL;
+ }
+ pWS->public.on = FALSE;
+ break;
+
+ case DEVICE_CLOSE:
+ DBG(1, ErrorF("WS DEVICE_CLOSE\n"));
+ wsClose(pInfo);
+ break;
+
+ default:
+ xf86Msg(X_ERROR, "WS: unknown command %d\n", what);
+ return !Success;
+ } /* switch */
+ return Success;
+} /* wsProc */
+
+static void
+wsReadInput(InputInfoPtr pInfo)
+{
+ WSDevicePtr priv;
+ static struct wscons_event eventList[NUMEVENTS];
+ int n, c;
+ struct wscons_event *event = eventList;
+ unsigned char *pBuf;
+ int ax, ay;
+
+ priv = pInfo->private;
+
+ XisbBlockDuration(priv->buffer, -1);
+ pBuf = (unsigned char *)eventList;
+ n = 0;
+ while (n < sizeof(eventList) && (c = XisbRead(priv->buffer)) >= 0) {
+ pBuf[n++] = (unsigned char)c;
+ }
+
+ if (n == 0)
+ return;
+
+ n /= sizeof(struct wscons_event);
+ while( n-- ) {
+ int buttons = priv->lastButtons;
+ int dx = 0, dy = 0, dz = 0, dw = 0;
+ int zbutton = 0, wbutton = 0;
+
+ ax = 0; ay = 0;
+ switch (event->type) {
+ case WSCONS_EVENT_MOUSE_UP:
+
+ buttons &= ~(1 << event->value);
+ DBG(4, ErrorF("Button %d up %x\n", event->value,
+ buttons));
+ break;
+ case WSCONS_EVENT_MOUSE_DOWN:
+ buttons |= (1 << event->value);
+ DBG(4, ErrorF("Button %d down %x\n", event->value,
+ buttons));
+ break;
+ case WSCONS_EVENT_MOUSE_DELTA_X:
+ dx = event->value;
+ DBG(4, ErrorF("Relative X %d\n", event->value));
+ break;
+ case WSCONS_EVENT_MOUSE_DELTA_Y:
+ dy = -event->value;
+ DBG(4, ErrorF("Relative Y %d\n", event->value));
+ break;
+ case WSCONS_EVENT_MOUSE_ABSOLUTE_X:
+ DBG(4, ErrorF("Absolute X %d\n", event->value));
+ if (event->value != 4095) {
+ ax = event->value;
+ if (priv->inv_x)
+ ax = priv->max_x - ax + priv->min_x;
+ }
+ break;
+ case WSCONS_EVENT_MOUSE_ABSOLUTE_Y:
+ DBG(4, ErrorF("Absolute Y %d\n", event->value));
+ ay = event->value;
+ if (priv->inv_y)
+ ay = priv->max_y - ay + priv->min_y;
+ break;
+#ifdef WSCONS_EVENT_MOUSE_DELTA_Z
+ case WSCONS_EVENT_MOUSE_DELTA_Z:
+ DBG(4, ErrorF("Relative Z %d\n", event->value));
+ dz = event->value;
+ break;
+#endif
+#ifdef WSCONS_EVENT_MOUSE_ABSOLUTE_Z
+ case WSCONS_EVENT_MOUSE_ABSOLUTE_Z:
+ /* ignore those */
+ ++event;
+ continue;
+ break;
+#endif
+#ifdef WSCONS_EVENT_MOUSE_DELTA_W
+ case WSCONS_EVENT_MOUSE_DELTA_W:
+ DBG(4, ErrorF("Relative W %d\n", event->value));
+ dw = event->value;
+ break;
+#endif
+ default:
+ xf86Msg(X_WARNING, "%s: bad wsmouse event type=%d\n",
+ pInfo->name, event->type);
+ ++event;
+ continue;
+ } /* case */
+
+ if (dx || dy) {
+ /* relative motion event */
+ DBG(3, ErrorF("postMotionEvent dX %d dY %d\n",
+ dx, dy));
+#if 0
+ priv->x += dx;
+ if (priv->x < priv->min_x) priv->x = priv->min_x;
+ if (priv->x > priv->max_x) priv->x = priv->max_x;
+ priv->y += dy;
+ if (priv->y < priv->min_y) priv->y = priv->min_y;
+ if (priv->y > priv->max_y) priv->y = priv->max_y;
+ xf86PostMotionEvent(pInfo->dev, 1, 0, 2,
+ priv->x, priv->y);
+#else
+ xf86PostMotionEvent(pInfo->dev, 0, 0, 2,
+ dx, dy);
+#endif
+ }
+ if (dz && priv->negativeZ != WS_NOZMAP
+ && priv->positiveZ != WS_NOZMAP) {
+ buttons &= ~(priv->negativeZ | priv->positiveZ);
+ if (dz < 0) {
+ DBG(4, ErrorF("Z -> button %d\n",
+ priv->negativeZ));
+ zbutton = 1 << (priv->negativeZ - 1);
+ } else {
+ DBG(4, ErrorF("Z -> button %d\n",
+ priv->positiveZ));
+ zbutton = 1 << (priv->positiveZ - 1);
+ }
+ buttons |= zbutton;
+ dz = 0;
+ }
+ if (dw && priv->negativeW != WS_NOZMAP
+ && priv->positiveW != WS_NOZMAP) {
+ buttons &= ~(priv->negativeW | priv->positiveW);
+ if (dw < 0) {
+ DBG(4, ErrorF("W -> button %d\n",
+ priv->negativeW));
+ wbutton = 1 << (priv->negativeW - 1);
+ } else {
+ DBG(4, ErrorF("W -> button %d\n",
+ priv->positiveW));
+ wbutton = 1 << (priv->positiveW - 1);
+ }
+ buttons |= wbutton;
+ dw = 0;
+ }
+ if (priv->lastButtons != buttons) {
+ /* button event */
+ wsSendButtons(pInfo, buttons);
+ }
+ if (zbutton != 0) {
+ /* generate a button up event */
+ buttons &= ~zbutton;
+ wsSendButtons(pInfo, buttons);
+ }
+ if (ax) {
+ /* absolute position event */
+ DBG(3, ErrorF("postMotionEvent X %d %d\n",
+ ax, priv->y));
+ xf86PostMotionEvent(pInfo->dev, 1, 0, 2, ax, priv->y);
+ priv->x = ax;
+ }
+ if (ay) {
+ /* absolute position event */
+ DBG(3, ErrorF("postMotionEvent y %d %d\n",
+ priv->x, ay));
+ xf86PostMotionEvent(pInfo->dev, 1, 0, 2, priv->x, ay);
+ priv->y = ay;
+ }
+ ++event;
+ }
+ return;
+} /* wsReadInput */
+
+static void
+wsSendButtons(InputInfoPtr pInfo, int buttons)
+{
+ WSDevicePtr priv = (WSDevicePtr)pInfo->private;
+ int button, mask;
+
+ for (button = 1; button < NBUTTONS; button++) {
+ mask = 1 << (button - 1);
+ if ((mask & priv->lastButtons) != (mask & buttons)) {
+ xf86PostButtonEvent(pInfo->dev, TRUE,
+ button, (buttons & mask) != 0,
+ 0, 0); /*2, priv->x, priv->y);*/
+ DBG(3, ErrorF("post button event %d %d\n",
+ button, (buttons & mask) != 0))
+ }
+ } /* for */
+ priv->lastButtons = buttons;
+} /* wsSendButtons */
+
+
+static int
+wsChangeControl(InputInfoPtr pInfo, xDeviceCtl *control)
+{
+ return BadMatch;
+}
+
+static int
+wsSwitchMode(ClientPtr client, DeviceIntPtr dev, int mode)
+{
+ return BadMatch;
+}
+
+static Bool
+wsOpen(InputInfoPtr pInfo)
+{
+ WSDevicePtr priv = (WSDevicePtr)pInfo->private;
+
+ DBG(1, ErrorF("WS open %s\n", priv->devName));
+ pInfo->fd = xf86OpenSerial(pInfo->options);
+ if (pInfo->fd == -1) {
+ xf86Msg(X_ERROR, "%s: cannot open input device\n", pInfo->name);
+ return !Success;
+ }
+ return Success;
+}
+
+static void
+wsClose(InputInfoPtr pInfo)
+{
+ xf86CloseSerial(pInfo->fd);
+ pInfo->fd = -1;
+}
+
+static Bool
+wsConvert(InputInfoPtr pInfo, int first, int num,
+ int v0, int v1, int v2, int v3, int v4, int v5,
+ int *x, int *y)
+{
+ WSDevicePtr priv = (WSDevicePtr) pInfo->private;
+ if (first != 0 || num != 2) {
+ return FALSE;
+ }
+
+ DBG(3, ErrorF("WSConvert: v0(%d), v1(%d)\n", v0, v1));
+
+ if (priv->swap_axes != 0) {
+ *x = xf86ScaleAxis(v1, 0, priv->screen_width - 1,
+ priv->min_y, priv->max_y);
+ *y = xf86ScaleAxis(v0, 0, priv->screen_height - 1,
+ priv->min_x, priv->max_x);
+ } else {
+ *x = xf86ScaleAxis(v0, 0, priv->screen_width - 1,
+ priv->min_x, priv->max_x);
+ *y = xf86ScaleAxis(v1, 0, priv->screen_height - 1,
+ priv->min_y, priv->max_y);
+ }
+
+ /*
+ * Need to check if still on the correct screen.
+ * This call is here so that this work can be done after
+ * calib and before posting the event.
+ */
+ xf86XInputSetScreen(pInfo, priv->screen_no, *x, *y);
+
+ DBG(3, ErrorF("WSConvert: x(%d), y(%d)\n", *x, *y));
+
+ return TRUE;
+}
+
+static void
+wsControlProc(DeviceIntPtr device, PtrCtrl *ctrl)
+{
+ InputInfoPtr pInfo = device->public.devicePrivate;
+ WSDevicePtr priv = (WSDevicePtr)pInfo->private;
+
+ DBG(1, ErrorF("wsControlProc\n"));
+ priv->num = ctrl->num;
+ priv->den = ctrl->den;
+ priv->threshold = ctrl->threshold;
+}
+