/* * Copyright © 2005-2009,2011 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.33 2011/07/16 17:51:30 matthieu Exp $ */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ws.h" #include #include "ws-properties.h" #include static MODULESETUPPROTO(SetupProc); static void TearDownProc(pointer); #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 static InputInfoPtr wsPreInit(InputDriverPtr, IDevPtr, int); #endif static int wsPreInit12(InputDriverPtr, InputInfoPtr, int); static int wsProc(DeviceIntPtr, int); static int wsDeviceInit(DeviceIntPtr); static int wsDeviceOn(DeviceIntPtr); static void wsDeviceOff(DeviceIntPtr); 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 void wsControlProc(DeviceIntPtr , PtrCtrl *); static void wsInitProperty(DeviceIntPtr); static int wsSetProperty(DeviceIntPtr, Atom, XIPropertyValuePtr, BOOL); static Atom prop_calibration = 0; static Atom prop_swap = 0; #ifdef DEBUG int ws_debug_level = 0; #endif static XF86ModuleVersionInfo VersionRec = { "ws", MODULEVENDORSTRING, MODINFOSTRING1, MODINFOSTRING2, XORG_VERSION_CURRENT, PACKAGE_VERSION_MAJOR, PACKAGE_VERSION_MINOR, PACKAGE_VERSION_PATCHLEVEL, ABI_CLASS_XINPUT, ABI_XINPUT_VERSION, MOD_CLASS_XINPUT, {0, 0, 0, 0} }; #define WS_NOZMAP 0 XF86ModuleData wsModuleData = {&VersionRec, SetupProc, TearDownProc }; InputDriverRec WS = { 1, "ws", NULL, #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 wsPreInit, #else wsPreInit12, #endif NULL, NULL, 0 }; 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 int wsPreInit12(InputDriverPtr drv, InputInfoPtr pInfo, int flags) { WSDevicePtr priv; MessageType buttons_from = X_CONFIG; char *s; const char *cs; int rc; priv = (WSDevicePtr)calloc(1, sizeof(WSDeviceRec)); if (priv == NULL) { rc = BadAlloc; goto fail; } pInfo->private = priv; #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 xf86CollectInputOptions(pInfo, NULL, NULL); xf86ProcessCommonOptions(pInfo, pInfo->options); #else xf86CollectInputOptions(pInfo, NULL); #endif #ifdef DEBUG ws_debug_level = xf86SetIntOption(pInfo->options, "DebugLevel", ws_debug_level); xf86Msg(X_INFO, "%s: debuglevel %d\n", pInfo->name, ws_debug_level); #endif priv->devName = xf86FindOptionValue(pInfo->options, "Device"); if (priv->devName == NULL) { xf86Msg(X_ERROR, "%s: No Device specified.\n", pInfo->name); rc = BadValue; 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", "4 5 6 7"); 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", pInfo->name, priv->screen_no); if (priv->screen_no >= screenInfo.numScreens || priv->screen_no < 0) { priv->screen_no = 0; } 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", pInfo->name); } priv->inv_x = 0; priv->inv_y = 0; cs = xf86FindOptionValue(pInfo->options, "Rotate"); if (cs) { if (xf86NameCmp(cs, "CW") == 0) { priv->inv_x = 1; priv->inv_y = 0; priv->swap_axes = 1; } else if (xf86NameCmp(cs, "CCW") == 0) { priv->inv_x = 0; priv->inv_y = 1; priv->swap_axes = 1; } else if (xf86NameCmp(cs, "UD") == 0) { priv->inv_x = 1; priv->inv_y = 1; } else { xf86Msg(X_ERROR, "\"%s\" is not a valid value " "for Option \"Rotate\"\n", cs); xf86Msg(X_ERROR, "Valid options are \"CW\", \"CCW\"," " or \"UD\"\n"); } } if (wsOpen(pInfo) != Success) { rc = BadValue; goto fail; } if (ioctl(pInfo->fd, WSMOUSEIO_GTYPE, &priv->type) != 0) { wsClose(pInfo); rc = BadValue; goto fail; } /* assume screen coordinate space until proven wrong */ priv->min_x = 0; priv->max_x = screenInfo.screens[priv->screen_no]->width - 1; priv->min_y = 0; priv->max_y = screenInfo.screens[priv->screen_no]->height - 1; priv->raw = 0; /* don't rely on the device type - we may be listening to a mux */ if (ioctl(pInfo->fd, WSMOUSEIO_GCALIBCOORDS, &priv->coords) != 0) { /* can't get absolute coordinate space - assume mouse */ pInfo->type_name = XI_MOUSE; } else if (priv->coords.samplelen == WSMOUSE_CALIBCOORDS_RESET) { /* * we're getting raw coordinates - update accordingly and hope * that there is no other absolute positioning device on the * same mux */ priv->min_x = priv->coords.minx; priv->max_x = priv->coords.maxx; priv->min_y = priv->coords.miny; priv->max_y = priv->coords.maxy; priv->raw = 1; pInfo->type_name = XI_TOUCHSCREEN; } else { /* * touchscreen not in raw mode, should send us screen * coordinates */ pInfo->type_name = XI_TOUCHSCREEN; } /* * Force TPANEL type for muxes have have calibration data. A mux * may have a mix of absolute and relative positioning devices, * and we need to ensure that the xinput layer translates raw * absolute position events for us. */ if (priv->raw && priv->type != WSMOUSE_TYPE_TPANEL) { xf86Msg(X_INFO, "%s detected calibration data in raw mode, " "using touch panel mode\n", pInfo->name); priv->type = WSMOUSE_TYPE_TPANEL; } if (priv->raw) { xf86Msg(X_CONFIG, "%s device will work in raw mode\n", pInfo->name); } /* Allow options to override this */ priv->min_x = xf86SetIntOption(pInfo->options, "MinX", priv->min_x); xf86Msg(X_INFO, "%s minimum x position: %d\n", pInfo->name, priv->min_x); priv->max_x = xf86SetIntOption(pInfo->options, "MaxX", priv->max_x); xf86Msg(X_INFO, "%s maximum x position: %d\n", pInfo->name, priv->max_x); priv->min_y = xf86SetIntOption(pInfo->options, "MinY", priv->min_y); xf86Msg(X_INFO, "%s minimum y position: %d\n", pInfo->name, priv->min_y); priv->max_y = xf86SetIntOption(pInfo->options, "MaxY", priv->max_y); xf86Msg(X_INFO, "%s maximum y position: %d\n", pInfo->name, priv->max_y); pInfo->device_control = wsProc; pInfo->read_input = wsReadInput; pInfo->control_proc = wsChangeControl; pInfo->switch_mode = wsSwitchMode; pInfo->private = priv; #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 pInfo->conversion_proc = NULL; pInfo->reverse_conversion_proc = NULL; pInfo->old_x = -1; pInfo->old_y = -1; #endif xf86Msg(buttons_from, "%s: Buttons: %d\n", pInfo->name, priv->buttons); wsClose(pInfo); wsmbEmuPreInit(pInfo); return Success; fail: if (priv != NULL) { free(priv); pInfo->private = NULL; } return rc; } #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 static InputInfoPtr wsPreInit(InputDriverPtr drv, IDevPtr dev, int flags) { InputInfoPtr pInfo = NULL; pInfo = xf86AllocateInput(drv, 0); if (pInfo == NULL) { return NULL; } pInfo->name = dev->identifier; pInfo->flags = XI86_POINTER_CAPABLE | XI86_SEND_DRAG_EVENTS; pInfo->conf_idev = dev; pInfo->close_proc = NULL; pInfo->private_flags = 0; pInfo->always_core_feedback = NULL; if (wsPreInit12(drv, pInfo, flags) != Success) { xf86DeleteInput(pInfo, 0); return NULL; } /* mark the device configured */ pInfo->flags |= XI86_CONFIGURED; return pInfo; } #endif static int wsProc(DeviceIntPtr pWS, int what) { InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; switch (what) { case DEVICE_INIT: return wsDeviceInit(pWS); case DEVICE_ON: return wsDeviceOn(pWS); case DEVICE_OFF: wsDeviceOff(pWS); 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 int wsDeviceInit(DeviceIntPtr pWS) { InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; WSDevicePtr priv = (WSDevicePtr)pInfo->private; unsigned char map[NBUTTONS + 1]; int i, xmin, xmax, ymin, ymax; Atom btn_labels[NBUTTONS] = {0}; Atom axes_labels[NAXES] = {0}; DBG(1, ErrorF("WS DEVICE_INIT\n")); btn_labels[0] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_LEFT); btn_labels[1] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_MIDDLE); btn_labels[2] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_RIGHT); for (i = 0; i < NBUTTONS; i++) map[i + 1] = i + 1; if (!InitButtonClassDeviceStruct(pWS, min(priv->buttons, NBUTTONS), btn_labels, map)) return !Success; if (priv->type == WSMOUSE_TYPE_TPANEL) { xmin = priv->min_x; xmax = priv->max_x; ymin = priv->min_y; ymax = priv->max_y; } else { xmin = -1; xmax = -1; ymin = -1; ymax = -1; } if (priv->swap_axes) { int tmp; tmp = xmin; xmin = ymin; ymin = tmp; tmp = xmax; xmax = ymax; ymax = tmp; } if ((priv->type == WSMOUSE_TYPE_TPANEL)) { axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_X); axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_Y); } else { axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_X); axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_Y); } #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 axes_labels[HSCROLL_AXIS] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_HSCROLL); axes_labels[VSCROLL_AXIS] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_VSCROLL); #endif if (!InitValuatorClassDeviceStruct(pWS, NAXES, axes_labels, GetMotionHistorySize(), priv->type == WSMOUSE_TYPE_TPANEL ? Absolute : Relative)) return !Success; if (!InitPtrFeedbackClassDeviceStruct(pWS, wsControlProc)) return !Success; xf86InitValuatorAxisStruct(pWS, 0, axes_labels[0], xmin, xmax, 1, 0, 1 #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 12 , priv->type == WSMOUSE_TYPE_TPANEL ? Absolute : Relative #endif ); xf86InitValuatorDefaults(pWS, 0); xf86InitValuatorAxisStruct(pWS, 1, axes_labels[1], ymin, ymax, 1, 0, 1 #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 12 , priv->type == WSMOUSE_TYPE_TPANEL ? Absolute : Relative #endif ); xf86InitValuatorDefaults(pWS, 1); #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 xf86InitValuatorAxisStruct(pWS, HSCROLL_AXIS, axes_labels[HSCROLL_AXIS], 0, -1, 0, 0, 0, Relative); xf86InitValuatorAxisStruct(pWS, VSCROLL_AXIS, axes_labels[VSCROLL_AXIS], 0, -1, 0, 0, 0, Relative); priv->scroll_mask = valuator_mask_new(MAX_VALUATORS); if (!priv->scroll_mask) { return !Success; } /* * The value of an HSCROLL or VSCROLL event is the fraction * motion_delta / scroll_distance * in [*.12] fixed-point format. The 'increment' attribute of the * scroll axes is constant: */ SetScrollValuator(pWS, HSCROLL_AXIS, SCROLL_TYPE_HORIZONTAL, 4096, 0); SetScrollValuator(pWS, VSCROLL_AXIS, SCROLL_TYPE_VERTICAL, 4096, 0); #endif #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 xf86MotionHistoryAllocate(pInfo); AssignTypeAndName(pWS, pInfo->atom, pInfo->name); #endif pWS->public.on = FALSE; if (wsOpen(pInfo) != Success) { return !Success; } wsInitProperty(pWS); XIRegisterPropertyHandler(pWS, wsSetProperty, NULL, NULL); wsmbEmuInitProperty(pWS); return Success; } static int wsDeviceOn(DeviceIntPtr pWS) { InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; WSDevicePtr priv = (WSDevicePtr)pInfo->private; #ifndef __NetBSD__ struct wsmouse_calibcoords coords; #endif 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; } #ifndef __NetBSD__ if (priv->type == WSMOUSE_TYPE_TPANEL) { /* get calibration values */ if (ioctl(pInfo->fd, WSMOUSEIO_GCALIBCOORDS, &coords) != 0) { xf86Msg(X_ERROR, "GCALIBCOORS failed %s\n", strerror(errno)); return !Success; } memcpy(&priv->coords, &coords, sizeof coords); /* set raw mode */ if (coords.samplelen != priv->raw) { coords.samplelen = priv->raw; if (ioctl(pInfo->fd, WSMOUSEIO_SCALIBCOORDS, &coords) != 0) { xf86Msg(X_ERROR, "SCALIBCOORS failed %s\n", strerror(errno)); return !Success; } } } #endif 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); wsmbEmuOn(pInfo); pWS->public.on = TRUE; return Success; } static void wsDeviceOff(DeviceIntPtr pWS) { InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; WSDevicePtr priv = pInfo->private; #ifndef __NetBSD__ struct wsmouse_calibcoords coords; #endif DBG(1, ErrorF("WS DEVICE OFF\n")); wsmbEmuFinalize(pInfo); #ifndef __NetBSD__ if (priv->type == WSMOUSE_TYPE_TPANEL) { /* Restore calibration data */ memcpy(&coords, &priv->coords, sizeof coords); if (ioctl(pInfo->fd, WSMOUSEIO_SCALIBCOORDS, &coords) != 0) { xf86Msg(X_ERROR, "SCALIBCOORS failed %s\n", strerror(errno)); } } #endif if (pInfo->fd >= 0) { xf86RemoveEnabledDevice(pInfo); wsClose(pInfo); } if (priv->buffer) { XisbFree(priv->buffer); priv->buffer = NULL; } pWS->public.on = FALSE; } 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; #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 int hscroll = 0, vscroll = 0; #endif 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) break; 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; case WSCONS_EVENT_MOUSE_DELTA_Z: DBG(4, ErrorF("Relative Z %d\n", event->value)); dz = event->value; break; case WSCONS_EVENT_MOUSE_ABSOLUTE_Z: /* ignore those */ ++event; continue; break; case WSCONS_EVENT_MOUSE_DELTA_W: DBG(4, ErrorF("Relative W %d\n", event->value)); dw = event->value; break; #if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 case WSCONS_EVENT_HSCROLL: hscroll = event->value; DBG(4, ErrorF("Horiz. Scrolling %d\n", event->value)); break; case WSCONS_EVENT_VSCROLL: vscroll = event->value; DBG(4, ErrorF("Vert. Scrolling %d\n", 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)); xf86PostMotionEvent(pInfo->dev, 0, 0, 2, dx, dy); } 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 GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 if (hscroll || vscroll) { xf86Msg(X_WARNING, "%s: hscroll=%d, vscroll=%d\n", pInfo->name, hscroll, vscroll); valuator_mask_zero(priv->scroll_mask); valuator_mask_set_double(priv->scroll_mask, HSCROLL_AXIS, (double) hscroll); valuator_mask_set_double(priv->scroll_mask, VSCROLL_AXIS, (double) vscroll); xf86PostMotionEventM(pInfo->dev, FALSE, priv->scroll_mask); } #endif if (priv->lastButtons != buttons) { /* button event */ wsSendButtons(pInfo, buttons); } if (zbutton != 0) { /* generate a button up event */ buttons &= ~zbutton; wsSendButtons(pInfo, buttons); } if (priv->swap_axes) { int tmp; tmp = ax; ax = ay; ay = tmp; } if (ax) { /* absolute position event */ DBG(3, ErrorF("postMotionEvent X %d\n", ax)); xf86PostMotionEvent(pInfo->dev, 1, 0, 1, ax); } if (ay) { /* absolute position event */ DBG(3, ErrorF("postMotionEvent y %d\n", ay)); xf86PostMotionEvent(pInfo->dev, 1, 1, 1, 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)) { if (!wsmbEmuFilterEvent(pInfo, button, (buttons & mask) != 0)) { xf86PostButtonEvent(pInfo->dev, TRUE, button, (buttons & mask) != 0, 0, 0); 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; #ifdef __NetBSD__ int version = WSMOUSE_EVENT_VERSION; #endif 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; } #ifdef __NetBSD__ if (ioctl(pInfo->fd, WSMOUSEIO_SETVERSION, &version) == -1) { xf86Msg(X_ERROR, "%s: cannot set wsmouse event version\n", pInfo->name); return !Success; } #endif return Success; } static void wsClose(InputInfoPtr pInfo) { xf86CloseSerial(pInfo->fd); pInfo->fd = -1; } 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; } static void wsInitProperty(DeviceIntPtr device) { InputInfoPtr pInfo = device->public.devicePrivate; WSDevicePtr priv = (WSDevicePtr)pInfo->private; int rc; DBG(1, ErrorF("wsInitProperty\n")); if (priv->type != WSMOUSE_TYPE_TPANEL) return; prop_calibration = MakeAtom(WS_PROP_CALIBRATION, strlen(WS_PROP_CALIBRATION), TRUE); rc = XIChangeDeviceProperty(device, prop_calibration, XA_INTEGER, 32, PropModeReplace, 4, &priv->min_x, FALSE); if (rc != Success) return; XISetDevicePropertyDeletable(device, prop_calibration, FALSE); prop_swap = MakeAtom(WS_PROP_SWAP_AXES, strlen(WS_PROP_SWAP_AXES), TRUE); rc = XIChangeDeviceProperty(device, prop_swap, XA_INTEGER, 8, PropModeReplace, 1, &priv->swap_axes, FALSE); if (rc != Success) return; return; } static int wsSetProperty(DeviceIntPtr device, Atom atom, XIPropertyValuePtr val, BOOL checkonly) { InputInfoPtr pInfo = device->public.devicePrivate; WSDevicePtr priv = (WSDevicePtr)pInfo->private; struct wsmouse_calibcoords coords; int need_update = 0; AxisInfoPtr ax = device->valuator->axes, ay = device->valuator->axes + 1; DBG(1, ErrorF("wsSetProperty %s\n", NameForAtom(atom))); /* Ignore non panel devices */ if (priv->type != WSMOUSE_TYPE_TPANEL) return Success; if (atom == prop_calibration) { if (val->format != 32 || val->type != XA_INTEGER) return BadMatch; if (val->size != 4 && val->size != 0) return BadMatch; if (!checkonly) { if (val->size == 0) { DBG(1, ErrorF(" uncalibrate\n")); priv->min_x = 0; priv->max_x = -1; priv->min_y = 0; priv->max_y = -1; } else { priv->min_x = ((int *)(val->data))[0]; priv->max_x = ((int *)(val->data))[1]; priv->min_y = ((int *)(val->data))[2]; priv->max_y = ((int *)(val->data))[3]; DBG(1, ErrorF(" calibrate %d %d %d %d\n", priv->min_x, priv->max_x, priv->min_y, priv->max_y)); need_update++; } /* Update axes descriptors */ if (!priv->swap_axes) { ax->min_value = priv->min_x; ax->max_value = priv->max_x; ay->min_value = priv->min_y; ay->max_value = priv->max_y; } else { ax->min_value = priv->min_y; ax->max_value = priv->max_y; ay->min_value = priv->min_x; ay->max_value = priv->max_x; } } } else if (atom == prop_swap) { if (val->format != 8 || val->type != XA_INTEGER || val->size != 1) return BadMatch; if (!checkonly) { priv->swap_axes = *((BOOL *)val->data); DBG(1, ErrorF("swap_axes %d\n", priv->swap_axes)); need_update++; } } if (need_update) { /* Update the saved values to be restored on device off */ priv->coords.minx = priv->min_x; priv->coords.maxx = priv->max_x; priv->coords.miny = priv->min_y; priv->coords.maxy = priv->max_y; #ifndef __NetBSD__ priv->coords.swapxy = priv->swap_axes; #endif /* Update the kernel calibration table */ coords.minx = priv->min_x; coords.maxx = priv->max_x; coords.miny = priv->min_y; coords.maxy = priv->max_y; #ifndef __NetBSD__ coords.swapxy = priv->swap_axes; #endif coords.samplelen = priv->raw; #ifndef __NetBSD__ coords.resx = priv->coords.resx; coords.resy = priv->coords.resy; #endif if (ioctl(pInfo->fd, WSMOUSEIO_SCALIBCOORDS, &coords) != 0) { xf86Msg(X_ERROR, "SCALIBCOORDS failed %s\n", strerror(errno)); } } return Success; }