1240a9a23Smbalmer/* 2f3ff32a7Smrg * Copyright © 2005-2009,2011 Matthieu Herrb 3240a9a23Smbalmer * 4240a9a23Smbalmer * Permission to use, copy, modify, and distribute this software for any 5240a9a23Smbalmer * purpose with or without fee is hereby granted, provided that the above 6240a9a23Smbalmer * copyright notice and this permission notice appear in all copies. 7240a9a23Smbalmer * 8240a9a23Smbalmer * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 9240a9a23Smbalmer * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 10240a9a23Smbalmer * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 11240a9a23Smbalmer * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 12240a9a23Smbalmer * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 13240a9a23Smbalmer * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 14240a9a23Smbalmer * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 15240a9a23Smbalmer */ 16f3ff32a7Smrg/* $OpenBSD: ws.c,v 1.33 2011/07/16 17:51:30 matthieu Exp $ */ 17240a9a23Smbalmer 18240a9a23Smbalmer#ifdef HAVE_CONFIG_H 19240a9a23Smbalmer#include "config.h" 20240a9a23Smbalmer#endif 21240a9a23Smbalmer 22240a9a23Smbalmer#include <unistd.h> 23240a9a23Smbalmer#include <errno.h> 24240a9a23Smbalmer#include <sys/ioctl.h> 25240a9a23Smbalmer#include <sys/time.h> 26240a9a23Smbalmer#include <dev/wscons/wsconsio.h> 27240a9a23Smbalmer 28f3ff32a7Smrg#include <xorg-server.h> 29240a9a23Smbalmer#include <xf86.h> 30240a9a23Smbalmer#include <xf86_OSproc.h> 31240a9a23Smbalmer#include <X11/extensions/XI.h> 32240a9a23Smbalmer#include <X11/extensions/XIproto.h> 33240a9a23Smbalmer#include <xf86Xinput.h> 34240a9a23Smbalmer#include <exevents.h> 35240a9a23Smbalmer#include <xisb.h> 36240a9a23Smbalmer#include <mipointer.h> 37240a9a23Smbalmer#include <extinit.h> 38240a9a23Smbalmer 39240a9a23Smbalmer#include "ws.h" 40240a9a23Smbalmer 41240a9a23Smbalmer#include <X11/Xatom.h> 42240a9a23Smbalmer#include "ws-properties.h" 43240a9a23Smbalmer#include <xserver-properties.h> 44240a9a23Smbalmer 45240a9a23Smbalmer 46240a9a23Smbalmerstatic MODULESETUPPROTO(SetupProc); 47240a9a23Smbalmerstatic void TearDownProc(pointer); 48240a9a23Smbalmer 49f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 50240a9a23Smbalmerstatic InputInfoPtr wsPreInit(InputDriverPtr, IDevPtr, int); 51f3ff32a7Smrg#endif 52f3ff32a7Smrgstatic int wsPreInit12(InputDriverPtr, InputInfoPtr, int); 53240a9a23Smbalmerstatic int wsProc(DeviceIntPtr, int); 54240a9a23Smbalmerstatic int wsDeviceInit(DeviceIntPtr); 55240a9a23Smbalmerstatic int wsDeviceOn(DeviceIntPtr); 56240a9a23Smbalmerstatic void wsDeviceOff(DeviceIntPtr); 57240a9a23Smbalmerstatic void wsReadInput(InputInfoPtr); 58240a9a23Smbalmerstatic void wsSendButtons(InputInfoPtr, int); 59240a9a23Smbalmerstatic int wsChangeControl(InputInfoPtr, xDeviceCtl *); 60240a9a23Smbalmerstatic int wsSwitchMode(ClientPtr, DeviceIntPtr, int); 61240a9a23Smbalmerstatic Bool wsOpen(InputInfoPtr); 62240a9a23Smbalmerstatic void wsClose(InputInfoPtr); 63240a9a23Smbalmerstatic void wsControlProc(DeviceIntPtr , PtrCtrl *); 64240a9a23Smbalmer 65240a9a23Smbalmerstatic void wsInitProperty(DeviceIntPtr); 66240a9a23Smbalmerstatic int wsSetProperty(DeviceIntPtr, Atom, XIPropertyValuePtr, BOOL); 67240a9a23Smbalmer 68240a9a23Smbalmerstatic Atom prop_calibration = 0; 69240a9a23Smbalmerstatic Atom prop_swap = 0; 70240a9a23Smbalmer 71240a9a23Smbalmer#ifdef DEBUG 72240a9a23Smbalmerint ws_debug_level = 0; 73240a9a23Smbalmer#endif 74240a9a23Smbalmer 75240a9a23Smbalmerstatic XF86ModuleVersionInfo VersionRec = { 76240a9a23Smbalmer "ws", 77240a9a23Smbalmer MODULEVENDORSTRING, 78240a9a23Smbalmer MODINFOSTRING1, 79240a9a23Smbalmer MODINFOSTRING2, 80240a9a23Smbalmer XORG_VERSION_CURRENT, 81240a9a23Smbalmer PACKAGE_VERSION_MAJOR, 82240a9a23Smbalmer PACKAGE_VERSION_MINOR, 83240a9a23Smbalmer PACKAGE_VERSION_PATCHLEVEL, 84240a9a23Smbalmer ABI_CLASS_XINPUT, 85240a9a23Smbalmer ABI_XINPUT_VERSION, 86240a9a23Smbalmer MOD_CLASS_XINPUT, 87240a9a23Smbalmer {0, 0, 0, 0} 88240a9a23Smbalmer}; 89240a9a23Smbalmer 90240a9a23Smbalmer#define WS_NOZMAP 0 91240a9a23Smbalmer 92240a9a23SmbalmerXF86ModuleData wsModuleData = {&VersionRec, 93240a9a23Smbalmer SetupProc, TearDownProc }; 94240a9a23Smbalmer 95240a9a23Smbalmer 96240a9a23SmbalmerInputDriverRec WS = { 97240a9a23Smbalmer 1, 98240a9a23Smbalmer "ws", 99240a9a23Smbalmer NULL, 100f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 101240a9a23Smbalmer wsPreInit, 102f3ff32a7Smrg#else 103f3ff32a7Smrg wsPreInit12, 104f3ff32a7Smrg#endif 105240a9a23Smbalmer NULL, 106240a9a23Smbalmer NULL, 107240a9a23Smbalmer 0 108240a9a23Smbalmer}; 109240a9a23Smbalmer 110240a9a23Smbalmerstatic pointer 111240a9a23SmbalmerSetupProc(pointer module, pointer options, int *errmaj, int *errmin) 112240a9a23Smbalmer{ 113240a9a23Smbalmer static Bool Initialised = FALSE; 114240a9a23Smbalmer 115240a9a23Smbalmer if (!Initialised) { 116240a9a23Smbalmer xf86AddInputDriver(&WS, module, 0); 117240a9a23Smbalmer Initialised = TRUE; 118240a9a23Smbalmer } 119240a9a23Smbalmer return module; 120240a9a23Smbalmer} 121240a9a23Smbalmer 122240a9a23Smbalmerstatic void 123240a9a23SmbalmerTearDownProc(pointer p) 124240a9a23Smbalmer{ 125240a9a23Smbalmer DBG(1, ErrorF("WS TearDownProc called\n")); 126240a9a23Smbalmer} 127240a9a23Smbalmer 128f3ff32a7Smrg 129f3ff32a7Smrgstatic int 130f3ff32a7SmrgwsPreInit12(InputDriverPtr drv, InputInfoPtr pInfo, int flags) 131240a9a23Smbalmer{ 132240a9a23Smbalmer WSDevicePtr priv; 133240a9a23Smbalmer MessageType buttons_from = X_CONFIG; 134240a9a23Smbalmer char *s; 1353cd2e234Smrg const char *cs; 136f3ff32a7Smrg int rc; 137240a9a23Smbalmer 138f3ff32a7Smrg priv = (WSDevicePtr)calloc(1, sizeof(WSDeviceRec)); 139f3ff32a7Smrg if (priv == NULL) { 140f3ff32a7Smrg rc = BadAlloc; 141240a9a23Smbalmer goto fail; 142f3ff32a7Smrg } 143240a9a23Smbalmer pInfo->private = priv; 144240a9a23Smbalmer 145f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 146240a9a23Smbalmer xf86CollectInputOptions(pInfo, NULL, NULL); 147240a9a23Smbalmer xf86ProcessCommonOptions(pInfo, pInfo->options); 148f3ff32a7Smrg#else 149f3ff32a7Smrg xf86CollectInputOptions(pInfo, NULL); 150f3ff32a7Smrg#endif 151240a9a23Smbalmer#ifdef DEBUG 152240a9a23Smbalmer ws_debug_level = xf86SetIntOption(pInfo->options, "DebugLevel", 153240a9a23Smbalmer ws_debug_level); 154f3ff32a7Smrg xf86Msg(X_INFO, "%s: debuglevel %d\n", pInfo->name, 155240a9a23Smbalmer ws_debug_level); 156240a9a23Smbalmer#endif 157240a9a23Smbalmer priv->devName = xf86FindOptionValue(pInfo->options, "Device"); 158240a9a23Smbalmer if (priv->devName == NULL) { 159240a9a23Smbalmer xf86Msg(X_ERROR, "%s: No Device specified.\n", 160f3ff32a7Smrg pInfo->name); 161f3ff32a7Smrg rc = BadValue; 162240a9a23Smbalmer goto fail; 163240a9a23Smbalmer } 164240a9a23Smbalmer priv->buttons = xf86SetIntOption(pInfo->options, "Buttons", 0); 165240a9a23Smbalmer if (priv->buttons == 0) { 166240a9a23Smbalmer priv->buttons = DFLTBUTTONS; 167240a9a23Smbalmer buttons_from = X_DEFAULT; 168240a9a23Smbalmer } 169240a9a23Smbalmer priv->negativeZ = priv->positiveZ = WS_NOZMAP; 170f3ff32a7Smrg s = xf86SetStrOption(pInfo->options, "ZAxisMapping", "4 5 6 7"); 171240a9a23Smbalmer if (s) { 172240a9a23Smbalmer int b1, b2; 173240a9a23Smbalmer 174240a9a23Smbalmer if (sscanf(s, "%d %d", &b1, &b2) == 2 && 175240a9a23Smbalmer b1 > 0 && b1 <= NBUTTONS && 176240a9a23Smbalmer b2 > 0 && b2 <= NBUTTONS) { 177240a9a23Smbalmer priv->negativeZ = b1; 178240a9a23Smbalmer priv->positiveZ = b2; 179240a9a23Smbalmer xf86Msg(X_CONFIG, 180240a9a23Smbalmer "%s: ZAxisMapping: buttons %d and %d\n", 181240a9a23Smbalmer pInfo->name, b1, b2); 182240a9a23Smbalmer } else { 183240a9a23Smbalmer xf86Msg(X_WARNING, "%s: invalid ZAxisMapping value: " 184240a9a23Smbalmer "\"%s\"\n", pInfo->name, s); 185240a9a23Smbalmer } 186240a9a23Smbalmer } 187240a9a23Smbalmer if (priv->negativeZ > priv->buttons) { 188240a9a23Smbalmer priv->buttons = priv->negativeZ; 189240a9a23Smbalmer buttons_from = X_CONFIG; 190240a9a23Smbalmer } 191240a9a23Smbalmer if (priv->positiveZ > priv->buttons) { 192240a9a23Smbalmer priv->buttons = priv->positiveZ; 193240a9a23Smbalmer buttons_from = X_CONFIG; 194240a9a23Smbalmer } 195240a9a23Smbalmer priv->negativeW = priv->positiveW = WS_NOZMAP; 196240a9a23Smbalmer s = xf86SetStrOption(pInfo->options, "WAxisMapping", NULL); 197240a9a23Smbalmer if (s) { 198240a9a23Smbalmer int b1, b2; 199240a9a23Smbalmer 200240a9a23Smbalmer if (sscanf(s, "%d %d", &b1, &b2) == 2 && 201240a9a23Smbalmer b1 > 0 && b1 <= NBUTTONS && 202240a9a23Smbalmer b2 > 0 && b2 <= NBUTTONS) { 203240a9a23Smbalmer priv->negativeW = b1; 204240a9a23Smbalmer priv->positiveW = b2; 205240a9a23Smbalmer xf86Msg(X_CONFIG, 206240a9a23Smbalmer "%s: WAxisMapping: buttons %d and %d\n", 207240a9a23Smbalmer pInfo->name, b1, b2); 208240a9a23Smbalmer } else { 209240a9a23Smbalmer xf86Msg(X_WARNING, "%s: invalid WAxisMapping value: " 210240a9a23Smbalmer "\"%s\"\n", pInfo->name, s); 211240a9a23Smbalmer } 212240a9a23Smbalmer } 213240a9a23Smbalmer if (priv->negativeW > priv->buttons) { 214240a9a23Smbalmer priv->buttons = priv->negativeW; 215240a9a23Smbalmer buttons_from = X_CONFIG; 216240a9a23Smbalmer } 217240a9a23Smbalmer if (priv->positiveW > priv->buttons) { 218240a9a23Smbalmer priv->buttons = priv->positiveW; 219240a9a23Smbalmer buttons_from = X_CONFIG; 220240a9a23Smbalmer } 221240a9a23Smbalmer 222240a9a23Smbalmer priv->screen_no = xf86SetIntOption(pInfo->options, "ScreenNo", 0); 223240a9a23Smbalmer xf86Msg(X_CONFIG, "%s associated screen: %d\n", 224f3ff32a7Smrg pInfo->name, priv->screen_no); 225240a9a23Smbalmer if (priv->screen_no >= screenInfo.numScreens || 226240a9a23Smbalmer priv->screen_no < 0) { 227240a9a23Smbalmer priv->screen_no = 0; 228240a9a23Smbalmer } 229240a9a23Smbalmer 230240a9a23Smbalmer 231240a9a23Smbalmer priv->swap_axes = xf86SetBoolOption(pInfo->options, "SwapXY", 0); 232240a9a23Smbalmer if (priv->swap_axes) { 233240a9a23Smbalmer xf86Msg(X_CONFIG, 234240a9a23Smbalmer "%s device will work with X and Y axes swapped\n", 235f3ff32a7Smrg pInfo->name); 236240a9a23Smbalmer } 237240a9a23Smbalmer priv->inv_x = 0; 238240a9a23Smbalmer priv->inv_y = 0; 2393cd2e234Smrg cs = xf86FindOptionValue(pInfo->options, "Rotate"); 2403cd2e234Smrg if (cs) { 2413cd2e234Smrg if (xf86NameCmp(cs, "CW") == 0) { 242240a9a23Smbalmer priv->inv_x = 1; 243240a9a23Smbalmer priv->inv_y = 0; 244240a9a23Smbalmer priv->swap_axes = 1; 2453cd2e234Smrg } else if (xf86NameCmp(cs, "CCW") == 0) { 246240a9a23Smbalmer priv->inv_x = 0; 247240a9a23Smbalmer priv->inv_y = 1; 248240a9a23Smbalmer priv->swap_axes = 1; 2493cd2e234Smrg } else if (xf86NameCmp(cs, "UD") == 0) { 250240a9a23Smbalmer priv->inv_x = 1; 251240a9a23Smbalmer priv->inv_y = 1; 252240a9a23Smbalmer } else { 253240a9a23Smbalmer xf86Msg(X_ERROR, "\"%s\" is not a valid value " 2543cd2e234Smrg "for Option \"Rotate\"\n", cs); 255240a9a23Smbalmer xf86Msg(X_ERROR, "Valid options are \"CW\", \"CCW\"," 256240a9a23Smbalmer " or \"UD\"\n"); 257240a9a23Smbalmer } 258240a9a23Smbalmer } 259240a9a23Smbalmer if (wsOpen(pInfo) != Success) { 260f3ff32a7Smrg rc = BadValue; 261240a9a23Smbalmer goto fail; 262240a9a23Smbalmer } 263240a9a23Smbalmer if (ioctl(pInfo->fd, WSMOUSEIO_GTYPE, &priv->type) != 0) { 264240a9a23Smbalmer wsClose(pInfo); 265f3ff32a7Smrg rc = BadValue; 266240a9a23Smbalmer goto fail; 267240a9a23Smbalmer } 268293de341Smacallan 269293de341Smacallan /* assume screen coordinate space until proven wrong */ 270293de341Smacallan priv->min_x = 0; 271293de341Smacallan priv->max_x = screenInfo.screens[priv->screen_no]->width - 1; 272293de341Smacallan priv->min_y = 0; 273293de341Smacallan priv->max_y = screenInfo.screens[priv->screen_no]->height - 1; 274293de341Smacallan priv->raw = 0; 275293de341Smacallan 276293de341Smacallan /* don't rely on the device type - we may be listening to a mux */ 277293de341Smacallan if (ioctl(pInfo->fd, WSMOUSEIO_GCALIBCOORDS, 278293de341Smacallan &priv->coords) != 0) { 279293de341Smacallan /* can't get absolute coordinate space - assume mouse */ 280293de341Smacallan pInfo->type_name = XI_MOUSE; 281293de341Smacallan } else if (priv->coords.samplelen == WSMOUSE_CALIBCOORDS_RESET) { 282293de341Smacallan /* 283293de341Smacallan * we're getting raw coordinates - update accordingly and hope 284293de341Smacallan * that there is no other absolute positioning device on the 285293de341Smacallan * same mux 286293de341Smacallan */ 287293de341Smacallan priv->min_x = priv->coords.minx; 288293de341Smacallan priv->max_x = priv->coords.maxx; 289293de341Smacallan priv->min_y = priv->coords.miny; 290293de341Smacallan priv->max_y = priv->coords.maxy; 291293de341Smacallan priv->raw = 1; 292240a9a23Smbalmer pInfo->type_name = XI_TOUCHSCREEN; 293240a9a23Smbalmer } else { 294293de341Smacallan /* 295293de341Smacallan * touchscreen not in raw mode, should send us screen 296293de341Smacallan * coordinates 297293de341Smacallan */ 298293de341Smacallan pInfo->type_name = XI_TOUCHSCREEN; 299240a9a23Smbalmer } 300293de341Smacallan 301136eb1c6Sjmcneill /* 302136eb1c6Sjmcneill * Force TPANEL type for muxes have have calibration data. A mux 303136eb1c6Sjmcneill * may have a mix of absolute and relative positioning devices, 304136eb1c6Sjmcneill * and we need to ensure that the xinput layer translates raw 305136eb1c6Sjmcneill * absolute position events for us. 306136eb1c6Sjmcneill */ 307136eb1c6Sjmcneill if (priv->raw && priv->type != WSMOUSE_TYPE_TPANEL) { 308136eb1c6Sjmcneill xf86Msg(X_INFO, "%s detected calibration data in raw mode, " 309136eb1c6Sjmcneill "using touch panel mode\n", pInfo->name); 310136eb1c6Sjmcneill priv->type = WSMOUSE_TYPE_TPANEL; 311136eb1c6Sjmcneill } 312136eb1c6Sjmcneill 313240a9a23Smbalmer if (priv->raw) { 314240a9a23Smbalmer xf86Msg(X_CONFIG, 315240a9a23Smbalmer "%s device will work in raw mode\n", 316f3ff32a7Smrg pInfo->name); 317240a9a23Smbalmer } 318240a9a23Smbalmer 319240a9a23Smbalmer /* Allow options to override this */ 320240a9a23Smbalmer priv->min_x = xf86SetIntOption(pInfo->options, "MinX", priv->min_x); 321240a9a23Smbalmer xf86Msg(X_INFO, "%s minimum x position: %d\n", 322f3ff32a7Smrg pInfo->name, priv->min_x); 323240a9a23Smbalmer priv->max_x = xf86SetIntOption(pInfo->options, "MaxX", priv->max_x); 324240a9a23Smbalmer xf86Msg(X_INFO, "%s maximum x position: %d\n", 325f3ff32a7Smrg pInfo->name, priv->max_x); 326240a9a23Smbalmer priv->min_y = xf86SetIntOption(pInfo->options, "MinY", priv->min_y); 327240a9a23Smbalmer xf86Msg(X_INFO, "%s minimum y position: %d\n", 328f3ff32a7Smrg pInfo->name, priv->min_y); 329240a9a23Smbalmer priv->max_y = xf86SetIntOption(pInfo->options, "MaxY", priv->max_y); 330240a9a23Smbalmer xf86Msg(X_INFO, "%s maximum y position: %d\n", 331f3ff32a7Smrg pInfo->name, priv->max_y); 332240a9a23Smbalmer 333240a9a23Smbalmer pInfo->device_control = wsProc; 334240a9a23Smbalmer pInfo->read_input = wsReadInput; 335240a9a23Smbalmer pInfo->control_proc = wsChangeControl; 336240a9a23Smbalmer pInfo->switch_mode = wsSwitchMode; 337f3ff32a7Smrg pInfo->private = priv; 338f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 339240a9a23Smbalmer pInfo->conversion_proc = NULL; 340240a9a23Smbalmer pInfo->reverse_conversion_proc = NULL; 341240a9a23Smbalmer pInfo->old_x = -1; 342240a9a23Smbalmer pInfo->old_y = -1; 343f3ff32a7Smrg#endif 344240a9a23Smbalmer xf86Msg(buttons_from, "%s: Buttons: %d\n", pInfo->name, priv->buttons); 345240a9a23Smbalmer 346240a9a23Smbalmer wsClose(pInfo); 347240a9a23Smbalmer 348240a9a23Smbalmer wsmbEmuPreInit(pInfo); 349f3ff32a7Smrg return Success; 350240a9a23Smbalmer 351240a9a23Smbalmerfail: 35253cb6c62Smbalmer if (priv != NULL) { 353f3ff32a7Smrg free(priv); 35453cb6c62Smbalmer pInfo->private = NULL; 35553cb6c62Smbalmer } 356f3ff32a7Smrg return rc; 357f3ff32a7Smrg} 358f3ff32a7Smrg 359f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 360f3ff32a7Smrgstatic InputInfoPtr 361f3ff32a7SmrgwsPreInit(InputDriverPtr drv, IDevPtr dev, int flags) 362f3ff32a7Smrg{ 363f3ff32a7Smrg InputInfoPtr pInfo = NULL; 364f3ff32a7Smrg 365f3ff32a7Smrg pInfo = xf86AllocateInput(drv, 0); 366f3ff32a7Smrg if (pInfo == NULL) { 367f3ff32a7Smrg return NULL; 368f3ff32a7Smrg } 369f3ff32a7Smrg pInfo->name = dev->identifier; 370f3ff32a7Smrg pInfo->flags = XI86_POINTER_CAPABLE | XI86_SEND_DRAG_EVENTS; 371f3ff32a7Smrg pInfo->conf_idev = dev; 372f3ff32a7Smrg pInfo->close_proc = NULL; 373f3ff32a7Smrg pInfo->private_flags = 0; 374f3ff32a7Smrg pInfo->always_core_feedback = NULL; 375f3ff32a7Smrg 376f3ff32a7Smrg if (wsPreInit12(drv, pInfo, flags) != Success) { 37753cb6c62Smbalmer xf86DeleteInput(pInfo, 0); 378f3ff32a7Smrg return NULL; 37953cb6c62Smbalmer } 380f3ff32a7Smrg /* mark the device configured */ 381f3ff32a7Smrg pInfo->flags |= XI86_CONFIGURED; 382f3ff32a7Smrg return pInfo; 383240a9a23Smbalmer} 384f3ff32a7Smrg#endif 385240a9a23Smbalmer 386240a9a23Smbalmerstatic int 387240a9a23SmbalmerwsProc(DeviceIntPtr pWS, int what) 388240a9a23Smbalmer{ 389240a9a23Smbalmer InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; 390240a9a23Smbalmer 391240a9a23Smbalmer switch (what) { 392240a9a23Smbalmer case DEVICE_INIT: 393240a9a23Smbalmer return wsDeviceInit(pWS); 394240a9a23Smbalmer 395240a9a23Smbalmer case DEVICE_ON: 396240a9a23Smbalmer return wsDeviceOn(pWS); 397240a9a23Smbalmer 398240a9a23Smbalmer case DEVICE_OFF: 399240a9a23Smbalmer wsDeviceOff(pWS); 400240a9a23Smbalmer break; 401240a9a23Smbalmer 402240a9a23Smbalmer case DEVICE_CLOSE: 403240a9a23Smbalmer DBG(1, ErrorF("WS DEVICE_CLOSE\n")); 404240a9a23Smbalmer wsClose(pInfo); 405240a9a23Smbalmer break; 406240a9a23Smbalmer 407240a9a23Smbalmer default: 408240a9a23Smbalmer xf86Msg(X_ERROR, "WS: unknown command %d\n", what); 409240a9a23Smbalmer return !Success; 410240a9a23Smbalmer } /* switch */ 411240a9a23Smbalmer return Success; 412240a9a23Smbalmer} /* wsProc */ 413240a9a23Smbalmer 414240a9a23Smbalmerstatic int 415240a9a23SmbalmerwsDeviceInit(DeviceIntPtr pWS) 416240a9a23Smbalmer{ 417240a9a23Smbalmer InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; 418f3ff32a7Smrg WSDevicePtr priv = (WSDevicePtr)pInfo->private; 419240a9a23Smbalmer unsigned char map[NBUTTONS + 1]; 420240a9a23Smbalmer int i, xmin, xmax, ymin, ymax; 421240a9a23Smbalmer Atom btn_labels[NBUTTONS] = {0}; 422240a9a23Smbalmer Atom axes_labels[NAXES] = {0}; 423240a9a23Smbalmer 424240a9a23Smbalmer DBG(1, ErrorF("WS DEVICE_INIT\n")); 425240a9a23Smbalmer 426240a9a23Smbalmer btn_labels[0] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_LEFT); 427240a9a23Smbalmer btn_labels[1] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_MIDDLE); 428240a9a23Smbalmer btn_labels[2] = XIGetKnownProperty(BTN_LABEL_PROP_BTN_RIGHT); 429240a9a23Smbalmer for (i = 0; i < NBUTTONS; i++) 430240a9a23Smbalmer map[i + 1] = i + 1; 431240a9a23Smbalmer if (!InitButtonClassDeviceStruct(pWS, 432240a9a23Smbalmer min(priv->buttons, NBUTTONS), 433240a9a23Smbalmer btn_labels, 434240a9a23Smbalmer map)) 435240a9a23Smbalmer return !Success; 436240a9a23Smbalmer 437240a9a23Smbalmer if (priv->type == WSMOUSE_TYPE_TPANEL) { 438240a9a23Smbalmer xmin = priv->min_x; 439240a9a23Smbalmer xmax = priv->max_x; 440240a9a23Smbalmer ymin = priv->min_y; 441240a9a23Smbalmer ymax = priv->max_y; 442240a9a23Smbalmer } else { 443240a9a23Smbalmer xmin = -1; 444240a9a23Smbalmer xmax = -1; 445240a9a23Smbalmer ymin = -1; 446240a9a23Smbalmer ymax = -1; 447240a9a23Smbalmer } 448240a9a23Smbalmer 449e0e11d4dSmbalmer if (priv->swap_axes) { 450e0e11d4dSmbalmer int tmp; 451e0e11d4dSmbalmer tmp = xmin; 452e0e11d4dSmbalmer xmin = ymin; 453e0e11d4dSmbalmer ymin = tmp; 454e0e11d4dSmbalmer tmp = xmax; 455e0e11d4dSmbalmer xmax = ymax; 456e0e11d4dSmbalmer ymax = tmp; 457e0e11d4dSmbalmer } 458240a9a23Smbalmer if ((priv->type == WSMOUSE_TYPE_TPANEL)) { 459240a9a23Smbalmer axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_X); 460240a9a23Smbalmer axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_ABS_Y); 461240a9a23Smbalmer } else { 462240a9a23Smbalmer axes_labels[0] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_X); 463240a9a23Smbalmer axes_labels[1] = XIGetKnownProperty(AXIS_LABEL_PROP_REL_Y); 464240a9a23Smbalmer } 46530bf185bSnia#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 4665eb33880Snia axes_labels[HSCROLL_AXIS] = 4675eb33880Snia XIGetKnownProperty(AXIS_LABEL_PROP_REL_HSCROLL); 4685eb33880Snia axes_labels[VSCROLL_AXIS] = 4695eb33880Snia XIGetKnownProperty(AXIS_LABEL_PROP_REL_VSCROLL); 47030bf185bSnia#endif 471240a9a23Smbalmer if (!InitValuatorClassDeviceStruct(pWS, 472240a9a23Smbalmer NAXES, 473240a9a23Smbalmer axes_labels, 474240a9a23Smbalmer GetMotionHistorySize(), 475240a9a23Smbalmer priv->type == WSMOUSE_TYPE_TPANEL ? 476240a9a23Smbalmer Absolute : Relative)) 477240a9a23Smbalmer return !Success; 478240a9a23Smbalmer if (!InitPtrFeedbackClassDeviceStruct(pWS, wsControlProc)) 479240a9a23Smbalmer return !Success; 480240a9a23Smbalmer 481240a9a23Smbalmer xf86InitValuatorAxisStruct(pWS, 0, 482240a9a23Smbalmer axes_labels[0], 483f3ff32a7Smrg xmin, xmax, 1, 0, 1 484f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 12 485f3ff32a7Smrg , priv->type == WSMOUSE_TYPE_TPANEL ? Absolute : Relative 486240a9a23Smbalmer#endif 487f3ff32a7Smrg ); 488240a9a23Smbalmer xf86InitValuatorDefaults(pWS, 0); 489240a9a23Smbalmer 490240a9a23Smbalmer xf86InitValuatorAxisStruct(pWS, 1, 491240a9a23Smbalmer axes_labels[1], 492f3ff32a7Smrg ymin, ymax, 1, 0, 1 493f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 12 494f3ff32a7Smrg , priv->type == WSMOUSE_TYPE_TPANEL ? Absolute : Relative 495240a9a23Smbalmer#endif 496f3ff32a7Smrg ); 497240a9a23Smbalmer xf86InitValuatorDefaults(pWS, 1); 498e0e11d4dSmbalmer 4995eb33880Snia 50030bf185bSnia#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 5015eb33880Snia xf86InitValuatorAxisStruct(pWS, HSCROLL_AXIS, 5025eb33880Snia axes_labels[HSCROLL_AXIS], 0, -1, 0, 0, 0, Relative); 5035eb33880Snia xf86InitValuatorAxisStruct(pWS, VSCROLL_AXIS, 5045eb33880Snia axes_labels[VSCROLL_AXIS], 0, -1, 0, 0, 0, Relative); 5055eb33880Snia priv->scroll_mask = valuator_mask_new(MAX_VALUATORS); 5065eb33880Snia if (!priv->scroll_mask) { 5075eb33880Snia return !Success; 5085eb33880Snia } 5095eb33880Snia 5105eb33880Snia /* 5115eb33880Snia * The value of an HSCROLL or VSCROLL event is the fraction 5125eb33880Snia * motion_delta / scroll_distance 5135eb33880Snia * in [*.12] fixed-point format. The 'increment' attribute of the 5145eb33880Snia * scroll axes is constant: 5155eb33880Snia */ 5165eb33880Snia SetScrollValuator(pWS, HSCROLL_AXIS, SCROLL_TYPE_HORIZONTAL, 4096, 0); 5175eb33880Snia SetScrollValuator(pWS, VSCROLL_AXIS, SCROLL_TYPE_VERTICAL, 4096, 0); 51830bf185bSnia#endif 5195eb33880Snia 520f3ff32a7Smrg#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) < 12 521240a9a23Smbalmer xf86MotionHistoryAllocate(pInfo); 522240a9a23Smbalmer AssignTypeAndName(pWS, pInfo->atom, pInfo->name); 523f3ff32a7Smrg#endif 524240a9a23Smbalmer pWS->public.on = FALSE; 525240a9a23Smbalmer if (wsOpen(pInfo) != Success) { 526240a9a23Smbalmer return !Success; 527240a9a23Smbalmer } 528240a9a23Smbalmer wsInitProperty(pWS); 529240a9a23Smbalmer XIRegisterPropertyHandler(pWS, wsSetProperty, NULL, NULL); 530240a9a23Smbalmer wsmbEmuInitProperty(pWS); 531240a9a23Smbalmer return Success; 532240a9a23Smbalmer} 533240a9a23Smbalmer 534240a9a23Smbalmerstatic int 535240a9a23SmbalmerwsDeviceOn(DeviceIntPtr pWS) 536240a9a23Smbalmer{ 537240a9a23Smbalmer InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; 538f3ff32a7Smrg WSDevicePtr priv = (WSDevicePtr)pInfo->private; 5393fc0ac56Stsutsui#ifndef __NetBSD__ 540240a9a23Smbalmer struct wsmouse_calibcoords coords; 5413fc0ac56Stsutsui#endif 542240a9a23Smbalmer 543240a9a23Smbalmer DBG(1, ErrorF("WS DEVICE ON\n")); 544240a9a23Smbalmer if ((pInfo->fd < 0) && (wsOpen(pInfo) != Success)) { 545240a9a23Smbalmer xf86Msg(X_ERROR, "wsOpen failed %s\n", 546240a9a23Smbalmer strerror(errno)); 547240a9a23Smbalmer return !Success; 548240a9a23Smbalmer } 549240a9a23Smbalmer 5504071cc8bStsutsui#ifndef __NetBSD__ 551240a9a23Smbalmer if (priv->type == WSMOUSE_TYPE_TPANEL) { 552f3ff32a7Smrg /* get calibration values */ 553240a9a23Smbalmer if (ioctl(pInfo->fd, WSMOUSEIO_GCALIBCOORDS, &coords) != 0) { 554240a9a23Smbalmer xf86Msg(X_ERROR, "GCALIBCOORS failed %s\n", 555240a9a23Smbalmer strerror(errno)); 556240a9a23Smbalmer return !Success; 557240a9a23Smbalmer } 558240a9a23Smbalmer memcpy(&priv->coords, &coords, sizeof coords); 559240a9a23Smbalmer /* set raw mode */ 560240a9a23Smbalmer if (coords.samplelen != priv->raw) { 561240a9a23Smbalmer coords.samplelen = priv->raw; 562240a9a23Smbalmer if (ioctl(pInfo->fd, WSMOUSEIO_SCALIBCOORDS, 563240a9a23Smbalmer &coords) != 0) { 564f3ff32a7Smrg xf86Msg(X_ERROR, "SCALIBCOORS failed %s\n", 565240a9a23Smbalmer strerror(errno)); 566240a9a23Smbalmer return !Success; 567240a9a23Smbalmer } 568240a9a23Smbalmer } 569240a9a23Smbalmer } 5704071cc8bStsutsui#endif 571240a9a23Smbalmer priv->buffer = XisbNew(pInfo->fd, 572240a9a23Smbalmer sizeof(struct wscons_event) * NUMEVENTS); 573240a9a23Smbalmer if (priv->buffer == NULL) { 574240a9a23Smbalmer xf86Msg(X_ERROR, "cannot alloc xisb buffer\n"); 575240a9a23Smbalmer wsClose(pInfo); 576240a9a23Smbalmer return !Success; 577240a9a23Smbalmer } 578240a9a23Smbalmer xf86AddEnabledDevice(pInfo); 579240a9a23Smbalmer wsmbEmuOn(pInfo); 580240a9a23Smbalmer pWS->public.on = TRUE; 581240a9a23Smbalmer return Success; 582240a9a23Smbalmer} 583240a9a23Smbalmer 584240a9a23Smbalmerstatic void 585240a9a23SmbalmerwsDeviceOff(DeviceIntPtr pWS) 586240a9a23Smbalmer{ 587240a9a23Smbalmer InputInfoPtr pInfo = (InputInfoPtr)pWS->public.devicePrivate; 588f3ff32a7Smrg WSDevicePtr priv = pInfo->private; 5893fc0ac56Stsutsui#ifndef __NetBSD__ 590240a9a23Smbalmer struct wsmouse_calibcoords coords; 5913fc0ac56Stsutsui#endif 592240a9a23Smbalmer 593240a9a23Smbalmer DBG(1, ErrorF("WS DEVICE OFF\n")); 594240a9a23Smbalmer wsmbEmuFinalize(pInfo); 5954071cc8bStsutsui#ifndef __NetBSD__ 596240a9a23Smbalmer if (priv->type == WSMOUSE_TYPE_TPANEL) { 597240a9a23Smbalmer /* Restore calibration data */ 598240a9a23Smbalmer memcpy(&coords, &priv->coords, sizeof coords); 599240a9a23Smbalmer if (ioctl(pInfo->fd, WSMOUSEIO_SCALIBCOORDS, &coords) != 0) { 600240a9a23Smbalmer xf86Msg(X_ERROR, "SCALIBCOORS failed %s\n", 601240a9a23Smbalmer strerror(errno)); 602240a9a23Smbalmer } 603240a9a23Smbalmer } 6044071cc8bStsutsui#endif 605240a9a23Smbalmer if (pInfo->fd >= 0) { 606240a9a23Smbalmer xf86RemoveEnabledDevice(pInfo); 607240a9a23Smbalmer wsClose(pInfo); 608240a9a23Smbalmer } 609240a9a23Smbalmer if (priv->buffer) { 610240a9a23Smbalmer XisbFree(priv->buffer); 611240a9a23Smbalmer priv->buffer = NULL; 612240a9a23Smbalmer } 613240a9a23Smbalmer pWS->public.on = FALSE; 614240a9a23Smbalmer} 615240a9a23Smbalmer 616240a9a23Smbalmerstatic void 617240a9a23SmbalmerwsReadInput(InputInfoPtr pInfo) 618240a9a23Smbalmer{ 619240a9a23Smbalmer WSDevicePtr priv; 620240a9a23Smbalmer static struct wscons_event eventList[NUMEVENTS]; 621240a9a23Smbalmer int n, c; 622240a9a23Smbalmer struct wscons_event *event = eventList; 623240a9a23Smbalmer unsigned char *pBuf; 624240a9a23Smbalmer int ax, ay; 625240a9a23Smbalmer 626240a9a23Smbalmer priv = pInfo->private; 627240a9a23Smbalmer 628240a9a23Smbalmer XisbBlockDuration(priv->buffer, -1); 629240a9a23Smbalmer pBuf = (unsigned char *)eventList; 630240a9a23Smbalmer n = 0; 631240a9a23Smbalmer while (n < sizeof(eventList) && (c = XisbRead(priv->buffer)) >= 0) { 632240a9a23Smbalmer pBuf[n++] = (unsigned char)c; 633240a9a23Smbalmer } 634240a9a23Smbalmer 635240a9a23Smbalmer if (n == 0) 636240a9a23Smbalmer return; 637240a9a23Smbalmer 638240a9a23Smbalmer n /= sizeof(struct wscons_event); 639240a9a23Smbalmer while( n-- ) { 640240a9a23Smbalmer int buttons = priv->lastButtons; 641240a9a23Smbalmer int dx = 0, dy = 0, dz = 0, dw = 0; 642240a9a23Smbalmer int zbutton = 0, wbutton = 0; 64330bf185bSnia#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 6445eb33880Snia int hscroll = 0, vscroll = 0; 64530bf185bSnia#endif 646240a9a23Smbalmer 647240a9a23Smbalmer ax = 0; ay = 0; 648240a9a23Smbalmer switch (event->type) { 649240a9a23Smbalmer case WSCONS_EVENT_MOUSE_UP: 650240a9a23Smbalmer 651240a9a23Smbalmer buttons &= ~(1 << event->value); 652240a9a23Smbalmer DBG(4, ErrorF("Button %d up %x\n", event->value, 653240a9a23Smbalmer buttons)); 654240a9a23Smbalmer break; 655240a9a23Smbalmer case WSCONS_EVENT_MOUSE_DOWN: 656240a9a23Smbalmer buttons |= (1 << event->value); 657240a9a23Smbalmer DBG(4, ErrorF("Button %d down %x\n", event->value, 658240a9a23Smbalmer buttons)); 659240a9a23Smbalmer break; 660240a9a23Smbalmer case WSCONS_EVENT_MOUSE_DELTA_X: 661240a9a23Smbalmer dx = event->value; 662240a9a23Smbalmer DBG(4, ErrorF("Relative X %d\n", event->value)); 663240a9a23Smbalmer break; 664240a9a23Smbalmer case WSCONS_EVENT_MOUSE_DELTA_Y: 665240a9a23Smbalmer dy = -event->value; 666240a9a23Smbalmer DBG(4, ErrorF("Relative Y %d\n", event->value)); 667240a9a23Smbalmer break; 668240a9a23Smbalmer case WSCONS_EVENT_MOUSE_ABSOLUTE_X: 669240a9a23Smbalmer DBG(4, ErrorF("Absolute X %d\n", event->value)); 670e0e11d4dSmbalmer if (event->value == 4095) 671e0e11d4dSmbalmer break; 672e0e11d4dSmbalmer ax = event->value; 673e0e11d4dSmbalmer if (priv->inv_x) 674e0e11d4dSmbalmer ax = priv->max_x - ax + priv->min_x; 675240a9a23Smbalmer break; 676240a9a23Smbalmer case WSCONS_EVENT_MOUSE_ABSOLUTE_Y: 677240a9a23Smbalmer DBG(4, ErrorF("Absolute Y %d\n", event->value)); 678240a9a23Smbalmer ay = event->value; 679240a9a23Smbalmer if (priv->inv_y) 680240a9a23Smbalmer ay = priv->max_y - ay + priv->min_y; 681240a9a23Smbalmer break; 682240a9a23Smbalmer case WSCONS_EVENT_MOUSE_DELTA_Z: 683240a9a23Smbalmer DBG(4, ErrorF("Relative Z %d\n", event->value)); 684240a9a23Smbalmer dz = event->value; 685240a9a23Smbalmer break; 686240a9a23Smbalmer case WSCONS_EVENT_MOUSE_ABSOLUTE_Z: 687240a9a23Smbalmer /* ignore those */ 688240a9a23Smbalmer ++event; 689240a9a23Smbalmer continue; 690240a9a23Smbalmer break; 691240a9a23Smbalmer case WSCONS_EVENT_MOUSE_DELTA_W: 692240a9a23Smbalmer DBG(4, ErrorF("Relative W %d\n", event->value)); 693240a9a23Smbalmer dw = event->value; 694240a9a23Smbalmer break; 69530bf185bSnia#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 6965eb33880Snia case WSCONS_EVENT_HSCROLL: 6975eb33880Snia hscroll = event->value; 6985eb33880Snia DBG(4, ErrorF("Horiz. Scrolling %d\n", event->value)); 6995eb33880Snia break; 7005eb33880Snia case WSCONS_EVENT_VSCROLL: 7015eb33880Snia vscroll = event->value; 7025eb33880Snia DBG(4, ErrorF("Vert. Scrolling %d\n", event->value)); 7035eb33880Snia break; 70430bf185bSnia#endif 705240a9a23Smbalmer default: 706240a9a23Smbalmer xf86Msg(X_WARNING, "%s: bad wsmouse event type=%d\n", 707240a9a23Smbalmer pInfo->name, event->type); 708240a9a23Smbalmer ++event; 709240a9a23Smbalmer continue; 710240a9a23Smbalmer } /* case */ 711240a9a23Smbalmer 712240a9a23Smbalmer if (dx || dy) { 713240a9a23Smbalmer /* relative motion event */ 714240a9a23Smbalmer DBG(3, ErrorF("postMotionEvent dX %d dY %d\n", 715240a9a23Smbalmer dx, dy)); 716240a9a23Smbalmer xf86PostMotionEvent(pInfo->dev, 0, 0, 2, 717240a9a23Smbalmer dx, dy); 718240a9a23Smbalmer } 719240a9a23Smbalmer if (dz && priv->negativeZ != WS_NOZMAP 720240a9a23Smbalmer && priv->positiveZ != WS_NOZMAP) { 721240a9a23Smbalmer buttons &= ~(priv->negativeZ | priv->positiveZ); 722240a9a23Smbalmer if (dz < 0) { 723240a9a23Smbalmer DBG(4, ErrorF("Z -> button %d\n", 724240a9a23Smbalmer priv->negativeZ)); 725240a9a23Smbalmer zbutton = 1 << (priv->negativeZ - 1); 726240a9a23Smbalmer } else { 727240a9a23Smbalmer DBG(4, ErrorF("Z -> button %d\n", 728240a9a23Smbalmer priv->positiveZ)); 729240a9a23Smbalmer zbutton = 1 << (priv->positiveZ - 1); 730240a9a23Smbalmer } 731240a9a23Smbalmer buttons |= zbutton; 732240a9a23Smbalmer dz = 0; 733240a9a23Smbalmer } 734240a9a23Smbalmer if (dw && priv->negativeW != WS_NOZMAP 735240a9a23Smbalmer && priv->positiveW != WS_NOZMAP) { 736240a9a23Smbalmer buttons &= ~(priv->negativeW | priv->positiveW); 737240a9a23Smbalmer if (dw < 0) { 738240a9a23Smbalmer DBG(4, ErrorF("W -> button %d\n", 739240a9a23Smbalmer priv->negativeW)); 740240a9a23Smbalmer wbutton = 1 << (priv->negativeW - 1); 741240a9a23Smbalmer } else { 742240a9a23Smbalmer DBG(4, ErrorF("W -> button %d\n", 743240a9a23Smbalmer priv->positiveW)); 744240a9a23Smbalmer wbutton = 1 << (priv->positiveW - 1); 745240a9a23Smbalmer } 746240a9a23Smbalmer buttons |= wbutton; 747240a9a23Smbalmer dw = 0; 748240a9a23Smbalmer } 74930bf185bSnia#if GET_ABI_MAJOR(ABI_XINPUT_VERSION) >= 14 750d9ebab69Smlelstv if (hscroll || vscroll) { 751d9ebab69Smlelstv static int warned = 0; 752d9ebab69Smlelstv if (!warned) { 753d9ebab69Smlelstv warned = 1; 754d9ebab69Smlelstv xf86Msg(X_WARNING, "%s: hscroll=%d, vscroll=%d\n", 755d9ebab69Smlelstv pInfo->name, hscroll, vscroll); 756d9ebab69Smlelstv } 7575eb33880Snia valuator_mask_zero(priv->scroll_mask); 7585eb33880Snia valuator_mask_set_double(priv->scroll_mask, 7595eb33880Snia HSCROLL_AXIS, (double) hscroll); 7605eb33880Snia valuator_mask_set_double(priv->scroll_mask, 7615eb33880Snia VSCROLL_AXIS, (double) vscroll); 7625eb33880Snia xf86PostMotionEventM(pInfo->dev, FALSE, priv->scroll_mask); 7635eb33880Snia } 76430bf185bSnia#endif 765240a9a23Smbalmer if (priv->lastButtons != buttons) { 766240a9a23Smbalmer /* button event */ 767240a9a23Smbalmer wsSendButtons(pInfo, buttons); 768240a9a23Smbalmer } 769240a9a23Smbalmer if (zbutton != 0) { 770240a9a23Smbalmer /* generate a button up event */ 771240a9a23Smbalmer buttons &= ~zbutton; 772240a9a23Smbalmer wsSendButtons(pInfo, buttons); 773240a9a23Smbalmer } 774e0e11d4dSmbalmer if (priv->swap_axes) { 775e0e11d4dSmbalmer int tmp; 776e0e11d4dSmbalmer 777e0e11d4dSmbalmer tmp = ax; 778e0e11d4dSmbalmer ax = ay; 779e0e11d4dSmbalmer ay = tmp; 780e0e11d4dSmbalmer } 781240a9a23Smbalmer if (ax) { 782240a9a23Smbalmer /* absolute position event */ 783240a9a23Smbalmer DBG(3, ErrorF("postMotionEvent X %d\n", ax)); 784240a9a23Smbalmer xf86PostMotionEvent(pInfo->dev, 1, 0, 1, ax); 785240a9a23Smbalmer } 786240a9a23Smbalmer if (ay) { 787240a9a23Smbalmer /* absolute position event */ 788240a9a23Smbalmer DBG(3, ErrorF("postMotionEvent y %d\n", ay)); 789240a9a23Smbalmer xf86PostMotionEvent(pInfo->dev, 1, 1, 1, ay); 790240a9a23Smbalmer } 791240a9a23Smbalmer ++event; 792240a9a23Smbalmer } 793240a9a23Smbalmer return; 794240a9a23Smbalmer} /* wsReadInput */ 795240a9a23Smbalmer 796240a9a23Smbalmerstatic void 797240a9a23SmbalmerwsSendButtons(InputInfoPtr pInfo, int buttons) 798240a9a23Smbalmer{ 799240a9a23Smbalmer WSDevicePtr priv = (WSDevicePtr)pInfo->private; 800240a9a23Smbalmer int button, mask; 801240a9a23Smbalmer 802240a9a23Smbalmer for (button = 1; button < NBUTTONS; button++) { 803240a9a23Smbalmer mask = 1 << (button - 1); 804240a9a23Smbalmer if ((mask & priv->lastButtons) != (mask & buttons)) { 805240a9a23Smbalmer if (!wsmbEmuFilterEvent(pInfo, button, 806240a9a23Smbalmer (buttons & mask) != 0)) { 807240a9a23Smbalmer xf86PostButtonEvent(pInfo->dev, TRUE, 808240a9a23Smbalmer button, (buttons & mask) != 0, 809240a9a23Smbalmer 0, 0); 810240a9a23Smbalmer DBG(3, ErrorF("post button event %d %d\n", 811240a9a23Smbalmer button, (buttons & mask) != 0)) 812240a9a23Smbalmer } 813240a9a23Smbalmer } 814240a9a23Smbalmer } /* for */ 815240a9a23Smbalmer priv->lastButtons = buttons; 816240a9a23Smbalmer} /* wsSendButtons */ 817240a9a23Smbalmer 818240a9a23Smbalmer 819240a9a23Smbalmerstatic int 820240a9a23SmbalmerwsChangeControl(InputInfoPtr pInfo, xDeviceCtl *control) 821240a9a23Smbalmer{ 822240a9a23Smbalmer return BadMatch; 823240a9a23Smbalmer} 824240a9a23Smbalmer 825240a9a23Smbalmerstatic int 826240a9a23SmbalmerwsSwitchMode(ClientPtr client, DeviceIntPtr dev, int mode) 827240a9a23Smbalmer{ 828240a9a23Smbalmer return BadMatch; 829240a9a23Smbalmer} 830240a9a23Smbalmer 831240a9a23Smbalmerstatic Bool 832240a9a23SmbalmerwsOpen(InputInfoPtr pInfo) 833240a9a23Smbalmer{ 834240a9a23Smbalmer WSDevicePtr priv = (WSDevicePtr)pInfo->private; 835240a9a23Smbalmer#ifdef __NetBSD__ 836240a9a23Smbalmer int version = WSMOUSE_EVENT_VERSION; 837240a9a23Smbalmer#endif 838240a9a23Smbalmer 839240a9a23Smbalmer DBG(1, ErrorF("WS open %s\n", priv->devName)); 840240a9a23Smbalmer pInfo->fd = xf86OpenSerial(pInfo->options); 841240a9a23Smbalmer if (pInfo->fd == -1) { 842240a9a23Smbalmer xf86Msg(X_ERROR, "%s: cannot open input device\n", pInfo->name); 843240a9a23Smbalmer return !Success; 844240a9a23Smbalmer } 845240a9a23Smbalmer#ifdef __NetBSD__ 846240a9a23Smbalmer if (ioctl(pInfo->fd, WSMOUSEIO_SETVERSION, &version) == -1) { 847240a9a23Smbalmer xf86Msg(X_ERROR, "%s: cannot set wsmouse event version\n", 848240a9a23Smbalmer pInfo->name); 849240a9a23Smbalmer return !Success; 850240a9a23Smbalmer } 851240a9a23Smbalmer#endif 852240a9a23Smbalmer return Success; 853240a9a23Smbalmer} 854240a9a23Smbalmer 855240a9a23Smbalmerstatic void 856240a9a23SmbalmerwsClose(InputInfoPtr pInfo) 857240a9a23Smbalmer{ 858240a9a23Smbalmer xf86CloseSerial(pInfo->fd); 859240a9a23Smbalmer pInfo->fd = -1; 860240a9a23Smbalmer} 861240a9a23Smbalmer 862240a9a23Smbalmerstatic void 863240a9a23SmbalmerwsControlProc(DeviceIntPtr device, PtrCtrl *ctrl) 864240a9a23Smbalmer{ 865240a9a23Smbalmer InputInfoPtr pInfo = device->public.devicePrivate; 866240a9a23Smbalmer WSDevicePtr priv = (WSDevicePtr)pInfo->private; 867240a9a23Smbalmer 868240a9a23Smbalmer DBG(1, ErrorF("wsControlProc\n")); 869240a9a23Smbalmer priv->num = ctrl->num; 870240a9a23Smbalmer priv->den = ctrl->den; 871240a9a23Smbalmer priv->threshold = ctrl->threshold; 872240a9a23Smbalmer} 873240a9a23Smbalmer 874240a9a23Smbalmerstatic void 875240a9a23SmbalmerwsInitProperty(DeviceIntPtr device) 876240a9a23Smbalmer{ 877240a9a23Smbalmer InputInfoPtr pInfo = device->public.devicePrivate; 878240a9a23Smbalmer WSDevicePtr priv = (WSDevicePtr)pInfo->private; 879240a9a23Smbalmer int rc; 880240a9a23Smbalmer 881240a9a23Smbalmer DBG(1, ErrorF("wsInitProperty\n")); 882240a9a23Smbalmer if (priv->type != WSMOUSE_TYPE_TPANEL) 883240a9a23Smbalmer return; 884240a9a23Smbalmer 885240a9a23Smbalmer prop_calibration = MakeAtom(WS_PROP_CALIBRATION, 886240a9a23Smbalmer strlen(WS_PROP_CALIBRATION), TRUE); 887240a9a23Smbalmer rc = XIChangeDeviceProperty(device, prop_calibration, XA_INTEGER, 32, 888240a9a23Smbalmer PropModeReplace, 4, &priv->min_x, FALSE); 889240a9a23Smbalmer if (rc != Success) 890240a9a23Smbalmer return; 891240a9a23Smbalmer 892240a9a23Smbalmer XISetDevicePropertyDeletable(device, prop_calibration, FALSE); 893240a9a23Smbalmer 894240a9a23Smbalmer prop_swap = MakeAtom(WS_PROP_SWAP_AXES, 895240a9a23Smbalmer strlen(WS_PROP_SWAP_AXES), TRUE); 896240a9a23Smbalmer rc = XIChangeDeviceProperty(device, prop_swap, XA_INTEGER, 8, 897240a9a23Smbalmer PropModeReplace, 1, &priv->swap_axes, FALSE); 898240a9a23Smbalmer if (rc != Success) 899240a9a23Smbalmer return; 900240a9a23Smbalmer return; 901240a9a23Smbalmer} 902240a9a23Smbalmer 903240a9a23Smbalmerstatic int 904240a9a23SmbalmerwsSetProperty(DeviceIntPtr device, Atom atom, XIPropertyValuePtr val, 905240a9a23Smbalmer BOOL checkonly) 906240a9a23Smbalmer{ 907240a9a23Smbalmer InputInfoPtr pInfo = device->public.devicePrivate; 908240a9a23Smbalmer WSDevicePtr priv = (WSDevicePtr)pInfo->private; 909240a9a23Smbalmer struct wsmouse_calibcoords coords; 910240a9a23Smbalmer int need_update = 0; 911240a9a23Smbalmer AxisInfoPtr ax = device->valuator->axes, 912240a9a23Smbalmer ay = device->valuator->axes + 1; 913240a9a23Smbalmer 914240a9a23Smbalmer DBG(1, ErrorF("wsSetProperty %s\n", NameForAtom(atom))); 915240a9a23Smbalmer 916240a9a23Smbalmer /* Ignore non panel devices */ 917240a9a23Smbalmer if (priv->type != WSMOUSE_TYPE_TPANEL) 918240a9a23Smbalmer return Success; 919240a9a23Smbalmer 920240a9a23Smbalmer if (atom == prop_calibration) { 921240a9a23Smbalmer if (val->format != 32 || val->type != XA_INTEGER) 922240a9a23Smbalmer return BadMatch; 923240a9a23Smbalmer if (val->size != 4 && val->size != 0) 924240a9a23Smbalmer return BadMatch; 925240a9a23Smbalmer if (!checkonly) { 926240a9a23Smbalmer if (val->size == 0) { 927240a9a23Smbalmer DBG(1, ErrorF(" uncalibrate\n")); 928240a9a23Smbalmer priv->min_x = 0; 929240a9a23Smbalmer priv->max_x = -1; 930240a9a23Smbalmer priv->min_y = 0; 931240a9a23Smbalmer priv->max_y = -1; 932240a9a23Smbalmer } else { 933240a9a23Smbalmer priv->min_x = ((int *)(val->data))[0]; 934240a9a23Smbalmer priv->max_x = ((int *)(val->data))[1]; 935240a9a23Smbalmer priv->min_y = ((int *)(val->data))[2]; 936240a9a23Smbalmer priv->max_y = ((int *)(val->data))[3]; 937240a9a23Smbalmer DBG(1, ErrorF(" calibrate %d %d %d %d\n", 938240a9a23Smbalmer priv->min_x, priv->max_x, 939240a9a23Smbalmer priv->min_y, priv->max_y)); 940240a9a23Smbalmer need_update++; 941240a9a23Smbalmer } 942240a9a23Smbalmer /* Update axes descriptors */ 943e0e11d4dSmbalmer if (!priv->swap_axes) { 944e0e11d4dSmbalmer ax->min_value = priv->min_x; 945e0e11d4dSmbalmer ax->max_value = priv->max_x; 946e0e11d4dSmbalmer ay->min_value = priv->min_y; 947e0e11d4dSmbalmer ay->max_value = priv->max_y; 948e0e11d4dSmbalmer } else { 949e0e11d4dSmbalmer ax->min_value = priv->min_y; 950e0e11d4dSmbalmer ax->max_value = priv->max_y; 951e0e11d4dSmbalmer ay->min_value = priv->min_x; 952e0e11d4dSmbalmer ay->max_value = priv->max_x; 953e0e11d4dSmbalmer } 954240a9a23Smbalmer } 955240a9a23Smbalmer } else if (atom == prop_swap) { 956240a9a23Smbalmer if (val->format != 8 || val->type != XA_INTEGER || 957240a9a23Smbalmer val->size != 1) 958240a9a23Smbalmer return BadMatch; 959240a9a23Smbalmer if (!checkonly) { 960240a9a23Smbalmer priv->swap_axes = *((BOOL *)val->data); 961240a9a23Smbalmer DBG(1, ErrorF("swap_axes %d\n", priv->swap_axes)); 962240a9a23Smbalmer need_update++; 963240a9a23Smbalmer } 964240a9a23Smbalmer } 965240a9a23Smbalmer if (need_update) { 966240a9a23Smbalmer /* Update the saved values to be restored on device off */ 967240a9a23Smbalmer priv->coords.minx = priv->min_x; 968240a9a23Smbalmer priv->coords.maxx = priv->max_x; 969240a9a23Smbalmer priv->coords.miny = priv->min_y; 970240a9a23Smbalmer priv->coords.maxy = priv->max_y; 971d232d9e0Stsutsui#ifndef __NetBSD__ 972240a9a23Smbalmer priv->coords.swapxy = priv->swap_axes; 973d6a0ccf1Smrg#endif 974f3ff32a7Smrg 975240a9a23Smbalmer /* Update the kernel calibration table */ 976240a9a23Smbalmer coords.minx = priv->min_x; 977240a9a23Smbalmer coords.maxx = priv->max_x; 978240a9a23Smbalmer coords.miny = priv->min_y; 979240a9a23Smbalmer coords.maxy = priv->max_y; 980d232d9e0Stsutsui#ifndef __NetBSD__ 981240a9a23Smbalmer coords.swapxy = priv->swap_axes; 982d6a0ccf1Smrg#endif 983f3ff32a7Smrg coords.samplelen = priv->raw; 984d232d9e0Stsutsui#ifndef __NetBSD__ 985240a9a23Smbalmer coords.resx = priv->coords.resx; 986240a9a23Smbalmer coords.resy = priv->coords.resy; 987d6a0ccf1Smrg#endif 988240a9a23Smbalmer if (ioctl(pInfo->fd, WSMOUSEIO_SCALIBCOORDS, &coords) != 0) { 989240a9a23Smbalmer xf86Msg(X_ERROR, "SCALIBCOORDS failed %s\n", 990240a9a23Smbalmer strerror(errno)); 991240a9a23Smbalmer } 992240a9a23Smbalmer } 993240a9a23Smbalmer return Success; 994240a9a23Smbalmer} 995