/* * Copyright 1995, 1999 by Patrick Lecoanet, France. * Copyright 2002, Red Hat * * This driver is a merge of the Elographics driver (from Patrick Lecoanet) and * the driver for Fujitsu Pen Computers from Rob Tsuk and John Apfelbaum. * * Modified for Stylistic 3400 passive pen support by David Clay * Fixed processing of all packets * Detangled and simplified if-statement logic * Fixed hover-mode pointer movement * Added Passive parameter for passive displays * Added switch 3 for "right" mouse button * I might have broken active pen support. I can't test it. * January 2005 * * Stylistic 500, 1000, 1200, 2300 Support fixed by John Apfelbaum * June 2001 * * Richard Miller-Smith * * Fixed up for XFree86 4.2, cleaned up the junk, cured crash on pointer * setting. * * 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 Patrick Lecoanet not be used in * advertising or publicity pertaining to distribution of the software without * specific, written prior permission. The Authors make no * representations about the suitability of this software for any purpose. It * is provided "as is" without express or implied warranty. * * PATRICK LECOANET DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO * EVENT SHALL PATRICK LECOANET 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. * * RED HAT DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO * EVENT SHALL RED HAT 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_CONFIG_H #include "config.h" #endif #include #include #include #include # include # include # include # include # include # include # include /* *************************************************************************** * * Default constants. * *************************************************************************** */ #define FPIT_PORT "/dev/ttyS3" #define FPIT_MAX_X 4100 #define FPIT_MIN_X 0 #define FPIT_MAX_Y 4100 #define FPIT_MIN_Y 0 /* *************************************************************************** * * Protocol constants. * *************************************************************************** */ #define FPIT_PACKET_SIZE 5 #define BUFFER_SIZE (FPIT_PACKET_SIZE*20) /* *************************************************************************** * * Screen orientation descriptors. * *************************************************************************** */ typedef enum { FPIT_INVERT_X = 0x01, FPIT_INVERT_Y = 0x02, FPIT_THEN_SWAP_XY = 0x04 } FpitOrientation; /* *************************************************************************** * * Device private records. * *************************************************************************** */ typedef struct { char *fpitDev; /* device file name */ int screen_width; int screen_height; Rotation screen_rotation; int screen_no; int fpitOldX; /* previous X position */ int fpitOldY; /* previous Y position */ int fpitOldProximity; /* previous proximity */ int fpitOldButtons; /* previous buttons state */ int fpitMinX; /* min X value */ int fpitMinY; /* min Y value */ int fpitMaxX; /* max X value */ int fpitMaxY; /* max Y value */ int fpitIndex; /* number of bytes read */ unsigned char fpitData[BUFFER_SIZE]; /* data read on the device */ FpitOrientation fpitBaseOrientation; /* read from X config */ FpitOrientation fpitTotalOrientation; /* above + RandR */ int fpitPassive; /* translate passive buttons */ int fpitTrackRandR; /* check for, react to screen rotate/resize */ /* XXX when this last option is unset, we provide "compatibly stupid" * behavior. */ } FpitPrivateRec, *FpitPrivatePtr; /* *************************************************************************** * * xf86FpitConvert -- * Convert extended valuators to x and y suitable for core motion * events. Return True if ok and False if the requested conversion * can't be done for the specified valuators. * *************************************************************************** */ static Bool xf86FpitConvert(InputInfoPtr pInfo, int first, int num, int v0, int v1, int v2, int v3, int v4, int v5, int *x, int *y) { FpitPrivatePtr priv = (FpitPrivatePtr) pInfo->private; AxisInfoPtr axes = pInfo->dev->valuator->axes; if (first != 0 || num != 2) { return FALSE; } *x = xf86ScaleAxis(v0, 0, priv->screen_width, axes[0].min_value, axes[0].max_value); *y = xf86ScaleAxis(v1, 0, priv->screen_height, axes[1].min_value, axes[1].max_value); return TRUE; } /* *************************************************************************** * * xf86FpitSetUpAxes -- * Based on current screen resolution and, if RandR support is enabled, * current rotation state, set up the XInput axes and orientation info. * *************************************************************************** */ static void xf86FpitSetUpAxes(DeviceIntPtr dev, FpitPrivatePtr priv) { /* * Device reports motions on 2 axes in absolute coordinates. * Axes min and max values are reported in raw coordinates. * Resolution is computed roughly by the difference between * max and min values scaled from the approximate size of the * screen to fit one meter. */ int quarter_turns; Atom axis_labels[2] = { 0, 0 }; priv->screen_width = screenInfo.screens[priv->screen_no]->width; priv->screen_height = screenInfo.screens[priv->screen_no]->height; priv->fpitTotalOrientation = priv->fpitBaseOrientation; if (!priv->fpitTrackRandR) return; /* now apply transforms specified by RandR: * slightly complicated because invertX/Y and swapXY don't commute. */ priv->screen_rotation = RRGetRotation(screenInfo.screens[priv->screen_no]); quarter_turns = ( (priv->screen_rotation & RR_Rotate_90 ? 1 : 0) + (priv->screen_rotation & RR_Rotate_180 ? 2 : 0) + (priv->screen_rotation & RR_Rotate_270 ? 3 : 0) ) % 4; if (quarter_turns / 2 != 0) priv->fpitTotalOrientation ^= FPIT_INVERT_X | FPIT_INVERT_Y; if (quarter_turns % 2 != 0) { priv->fpitTotalOrientation ^= (priv->fpitTotalOrientation & FPIT_THEN_SWAP_XY ? FPIT_INVERT_X : FPIT_INVERT_Y) | FPIT_THEN_SWAP_XY; } if (priv->fpitTotalOrientation & FPIT_THEN_SWAP_XY) { InitValuatorAxisStruct(dev, 1, axis_labels[1], priv->fpitMinX, priv->fpitMaxX, 9500, 0 /* min_res */ , 9500 /* max_res */, Absolute); InitValuatorAxisStruct(dev, 0, axis_labels[0], priv->fpitMinY, priv->fpitMaxY, 10500, 0 /* min_res */ , 10500 /* max_res */, Absolute); } else { InitValuatorAxisStruct(dev, 0, axis_labels[0], priv->fpitMinX, priv->fpitMaxX, 9500, 0 /* min_res */ , 9500 /* max_res */, Absolute); InitValuatorAxisStruct(dev, 1, axis_labels[1], priv->fpitMinY, priv->fpitMaxY, 10500, 0 /* min_res */ , 10500 /* max_res */, Absolute); } } /* ** xf86FpitReadInput ** Reads from the Fpit and posts any new events to the server. */ static void xf86FpitReadInput(InputInfoPtr pInfo) { FpitPrivatePtr priv = (FpitPrivatePtr) pInfo->private; int len, loop; int is_core_pointer = 0; int x, y, buttons, prox; DeviceIntPtr device; int conv_x, conv_y; if (priv->fpitTrackRandR && ( priv->screen_width != screenInfo.screens[priv->screen_no]->width || priv->screen_height != screenInfo.screens[priv->screen_no]->height || priv->screen_rotation != RRGetRotation(screenInfo.screens[priv->screen_no]) )) xf86FpitSetUpAxes(pInfo->dev, priv); do { /* keep reading blocks until there are no more */ /* Read data into buffer */ len = xf86ReadSerial(pInfo->fd, priv->fpitData+priv->fpitIndex, BUFFER_SIZE-priv->fpitIndex); if (len <= 0) { Error("error reading FPIT device"); priv->fpitIndex = 0; return; } /* Since the Fujitsu only delivers data in an absolute mode, we can look through the data backwards to find the last full and valid position. (This may make cursor movement a bit faster) */ /* DMC: We want to process ALL packets! This way, all points will come through and drawing curves are smoother. Also we won't miss any button events. */ priv->fpitIndex += len; #define PHASING_BIT 0x80 #define PROXIMITY_BIT 0x20 #define BUTTON_BITS 0x07 #define SW1 0x01 #define SW2 0x02 #define SW3 0x04 /* process each packet in this block */ /* Format of 5 bytes data packet for Fpit Tablets Byte 1 bit 7 (0x80) Phasing bit always 1 bit 6 (0x40) Switch status change bit 5 (0x20) Proximity bit 4 (0x10) Always 0 bit 3 (0x08) Test data bits 2-0 (0x07) Buttons: bit 2 (0x04) Sw3 (2nd side sw) bit 1 (0x02) Sw2 (1st side sw) bit 0 (0x01) Sw1 (Pen tip sw) Byte 2 bit 7 (0x80) Always 0 bits 6-0 (0x7f) X6 - X0 Byte 3 bit 7 (0x80) Always 0 bits 6-0 (0x7f) X13 - X7 Byte 4 bit 7 (0x80) Always 0 bits 6-0 (0x7f) Y6 - Y0 Byte 5 bit 7 (0x80) Always 0 bits 6-0 (0x7f) Y13 - Y7 */ for (loop=0;loop+FPIT_PACKET_SIZE<=priv->fpitIndex;loop++) { if (!(priv->fpitData[loop] & PHASING_BIT)) continue; /* we don't have a start bit yet */ x = (int) (priv->fpitData[loop + 1] & 0x7f) + ((int) (priv->fpitData[loop + 2] & 0x7f) << 7); y = (int) (priv->fpitData[loop + 3] & 0x7f) + ((int) (priv->fpitData[loop + 4] & 0x7f) << 7); /* Adjust to orientation */ if (priv->fpitTotalOrientation & FPIT_INVERT_X) x = priv->fpitMaxX - x + priv->fpitMinX; if (priv->fpitTotalOrientation & FPIT_INVERT_Y) y = priv->fpitMaxY - y + priv->fpitMinY; if (priv->fpitTotalOrientation & FPIT_THEN_SWAP_XY) { int z = x; x = y; y = z; } prox = (priv->fpitData[loop] & PROXIMITY_BIT) ? 0 : 1; buttons = (priv->fpitData[loop] & BUTTON_BITS); device = pInfo->dev; xf86FpitConvert(pInfo, 0, 2, x, y, 0, 0, 0, 0, &conv_x, &conv_y); xf86XInputSetScreen(pInfo, priv->screen_no, conv_x, conv_y); /* coordinates are ready we can send events */ if (prox!=priv->fpitOldProximity) /* proximity changed */ if (!is_core_pointer) xf86PostProximityEvent(device, prox, 0, 2, x, y); if (priv->fpitOldX != x || priv->fpitOldY != y) /* position changed */ xf86PostMotionEvent(device, 1, 0, 2, x, y); if (priv->fpitPassive) { /* For passive pen (Stylistic 3400, et al.): sw1 = 1 if pen is moving sw2 = 1 if pen was lifted from the pad / isn't in contact sw3 = 1 if right mouse-button icon was chosen */ /* convert the pen button bits to actual mouse buttons */ if (buttons & SW2) buttons=0; /* the pen was lifted, so no buttons are pressed */ else if (buttons & SW3) buttons=SW3; /* the "right mouse" button was pressed, so send down event */ else if (prox) buttons=SW1; /* the "left mouse" button was pressed and we are not hovering, so send down event */ else buttons=0; /* We are in hover mode, so not left-clicking. */ } /* the active pen's buttons map directly to the mouse buttons. Right-click may happen even in hover mode. */ /* DBG(2, ErrorF("%02d/%02d Prox=%d SW:%x Buttons:%x->%x (%d, %d)\n", loop,priv->fpitIndex,prox,priv->fpitData[loop]&BUTTON_BITS,priv->fpitOldButtons,buttons,x,y));*/ if (priv->fpitOldButtons != buttons) { int delta; delta = buttons ^ priv->fpitOldButtons; /* set delta to the bits that have changed */ while (delta) { int id; id = ffs(delta); delta &= ~(1 << (id - 1)); xf86PostButtonEvent(device, 1, id, (buttons & (1 << (id - 1))), 0, 2, x, y); /* DBG(1, ErrorF("Button %d %s\n",id,(buttons & (1 << (id - 1)))?"DOWN":"UP"));*/ } priv->fpitOldButtons = buttons; } priv->fpitOldX = x; priv->fpitOldY = y; priv->fpitOldProximity = prox; loop+=FPIT_PACKET_SIZE-1; /* advance to the next packet */ } /* for each packet */ /* remove from the data buffer all that we have processed */ if (loopfpitIndex) memmove(priv->fpitData, priv->fpitData+loop,priv->fpitIndex-loop); priv->fpitIndex-=loop; /* DMC: My system did not read the pen-up event until another event was posted, the result was the button sticking down even though I had lifted the pen. So I am checking the device for more data and then retrieving it. This fixed it for me. I don't know if this is just my system. */ } while (xf86WaitForInput(pInfo->fd,0)>0); /* go back and check for more data (we don't want to block for I/O!) */ return; } static void xf86FpitPtrCtrl(DeviceIntPtr device, PtrCtrl *ctrl) { /* I have no clue what this does, except that registering it stops the X server segfaulting in ProcGetPointerMapping() Ho Hum. */ } /* *************************************************************************** * * xf86FpitControl -- * *************************************************************************** */ static Bool xf86FpitControl(DeviceIntPtr dev, int mode) { InputInfoPtr pInfo = (InputInfoPtr) dev->public.devicePrivate; FpitPrivatePtr priv = (FpitPrivatePtr) (pInfo->private); unsigned char map[] = { 0, 1, 2, 3 /* DMC: changed this so we can use all three buttons */ }; Atom btn_labels[3] = { 0 }; Atom axis_labels[2] = { 0, 0 }; switch (mode) { case DEVICE_INIT: { if (priv->screen_no >= screenInfo.numScreens || priv->screen_no < 0) { priv->screen_no = 0; } /* * Device reports button press for up to 3 buttons. */ if (InitButtonClassDeviceStruct(dev, 3, btn_labels, map) == FALSE) { ErrorF("Unable to allocate Fpit touchscreen ButtonClassDeviceStruct\n"); return !Success; } if (InitFocusClassDeviceStruct(dev) == FALSE) { ErrorF("Unable to allocate Fpit touchscreen FocusClassDeviceStruct\n"); return !Success; } if(InitPtrFeedbackClassDeviceStruct(dev, xf86FpitPtrCtrl) == FALSE) { ErrorF("Unable to allocate PtrFeedBackClassDeviceStruct\n"); } if (InitValuatorClassDeviceStruct(dev, 2, axis_labels, GetMotionHistorySize(), Absolute) == FALSE) { ErrorF("Unable to allocate Fpit touchscreen ValuatorClassDeviceStruct\n"); return !Success; } xf86FpitSetUpAxes(dev, priv); if (InitFocusClassDeviceStruct(dev) == FALSE) { ErrorF("Unable to allocate Fpit touchscreen FocusClassDeviceStruct\n"); } /* * Allocate the motion events buffer. */ xf86MotionHistoryAllocate(pInfo); /* * This once has caused the server to crash after doing a malloc & strcpy ?? */ return Success; } case DEVICE_ON: if (pInfo->fd < 0) { pInfo->fd = xf86OpenSerial(pInfo->options); if (pInfo->fd < 0) { Error("Unable to open Fpit touchscreen device"); return !Success; } xf86AddEnabledDevice(pInfo); dev->public.on = TRUE; } return Success; /* * Deactivate the device. After this, the device will not emit * events until a subsequent DEVICE_ON. Thus, we can momentarily * close the port. */ case DEVICE_OFF: dev->public.on = FALSE; if (pInfo->fd >= 0) { xf86RemoveEnabledDevice(pInfo); } xf86CloseSerial(pInfo->fd); pInfo->fd = -1; return Success; /* * Final close before server exit. This is used during server shutdown. * Close the port and free all the resources. */ case DEVICE_CLOSE: dev->public.on = FALSE; if (pInfo->fd >= 0) { RemoveEnabledDevice(pInfo->fd); } xf86CloseSerial(pInfo->fd); pInfo->fd = -1; return Success; default: ErrorF("unsupported mode=%d\n", mode); return !Success; } } /* *************************************************************************** * * xf86FpitAllocate -- * *************************************************************************** */ static int xf86FpitAllocate(InputDriverPtr drv, InputInfoPtr pInfo) { FpitPrivatePtr priv; priv = malloc(sizeof(FpitPrivateRec)); if (!priv) return BadAlloc; priv->fpitDev = strdup(FPIT_PORT); priv->screen_no = 0; priv->screen_width = -1; priv->screen_height = -1; priv->screen_rotation = RR_Rotate_0; priv->fpitMinX = FPIT_MIN_X; priv->fpitMaxX = FPIT_MAX_X; priv->fpitMinY = FPIT_MIN_Y; priv->fpitMaxY = FPIT_MAX_Y; priv->fpitOldX = priv->fpitOldY = -1; priv->fpitOldButtons = 0; priv->fpitOldProximity = 0; priv->fpitIndex = 0; priv->fpitPassive = 0; pInfo->name = XI_TOUCHSCREEN; pInfo->flags = 0 /* XI86_NO_OPEN_ON_INIT */ ; pInfo->device_control = xf86FpitControl; pInfo->read_input = xf86FpitReadInput; pInfo->control_proc = NULL; pInfo->switch_mode = NULL; pInfo->fd = -1; pInfo->dev = NULL; pInfo->private = priv; pInfo->type_name = "Fujitsu Stylistic"; return Success; } static void xf86FpitUninit(InputDriverPtr drv, InputInfoPtr pInfo, int flags) { FpitPrivatePtr priv = (FpitPrivatePtr) pInfo->private; xf86FpitControl(pInfo->dev, DEVICE_OFF); free(priv->fpitDev); free(priv); pInfo->private = NULL; xf86DeleteInput(pInfo, 0); } static char *default_options[] = { "BaudRate", "19200", "StopBits", "0", "DataBits", "8", "Parity", "None", "Vmin", "10", "Vtime", "1", "FlowControl", "None", NULL }; static int xf86FpitInit(InputDriverPtr drv, InputInfoPtr pInfo, int flags) { FpitPrivatePtr priv = NULL; char *str; int rc; rc = xf86FpitAllocate(drv, pInfo); if (rc != Success) return rc; priv = pInfo->private; str = xf86FindOptionValue(pInfo->options, "Device"); if (!str) { xf86Msg(X_ERROR, "%s: No Device specified in FPIT module config.\n", pInfo->name); if (priv) { if (priv->fpitDev) { free(priv->fpitDev); } free(priv); } return BadValue; } priv->fpitDev = strdup(str); pInfo->name = xf86SetStrOption(pInfo->options, "DeviceName", XI_TOUCHSCREEN); xf86Msg(X_CONFIG, "FPIT device name: %s\n", pInfo->name); priv->screen_no = xf86SetIntOption(pInfo->options, "ScreenNo", 0); xf86Msg(X_CONFIG, "Fpit associated screen: %d\n", priv->screen_no); priv->fpitMaxX = xf86SetIntOption(pInfo->options, "MaximumXPosition", FPIT_MAX_X); xf86Msg(X_CONFIG, "FPIT maximum x position: %d\n", priv->fpitMaxX); priv->fpitMinX = xf86SetIntOption(pInfo->options, "MinimumXPosition", FPIT_MIN_X); xf86Msg(X_CONFIG, "FPIT minimum x position: %d\n", priv->fpitMinX); priv->fpitMaxY = xf86SetIntOption(pInfo->options, "MaximumYPosition", FPIT_MAX_Y); xf86Msg(X_CONFIG, "FPIT maximum y position: %d\n", priv->fpitMaxY); priv->fpitMinY = xf86SetIntOption(pInfo->options, "MinimumYPosition", FPIT_MIN_Y); xf86Msg(X_CONFIG, "FPIT minimum y position: %d\n", priv->fpitMinY); priv->fpitBaseOrientation = 0; if (xf86SetBoolOption(pInfo->options, "InvertX", 0)) priv->fpitBaseOrientation |= FPIT_INVERT_X; if (xf86SetBoolOption(pInfo->options, "InvertY", 0)) priv->fpitBaseOrientation |= FPIT_INVERT_Y; if (xf86SetBoolOption(pInfo->options, "SwapXY", 0)) priv->fpitBaseOrientation |= FPIT_THEN_SWAP_XY; priv->fpitPassive = xf86SetBoolOption(pInfo->options, "Passive", 0); priv->fpitTrackRandR = xf86SetBoolOption(pInfo->options, "TrackRandR", 0); /* XXX "Rotate" option provides compatibly stupid behavior. JEB. */ str = xf86SetStrOption(pInfo->options, "Rotate", 0); if (!xf86NameCmp(str, "CW")) priv->fpitBaseOrientation |= FPIT_INVERT_X | FPIT_INVERT_Y | FPIT_THEN_SWAP_XY; else if (!xf86NameCmp(str, "CCW")) priv->fpitBaseOrientation |= FPIT_THEN_SWAP_XY; xf86Msg(X_CONFIG, "FPIT invert X axis: %s\n", priv->fpitBaseOrientation & FPIT_INVERT_X ? "Yes" : "No"); xf86Msg(X_CONFIG, "FPIT invert Y axis: %s\n", priv->fpitBaseOrientation & FPIT_INVERT_Y ? "Yes" : "No"); xf86Msg(X_CONFIG, "FPIT swap X and Y axis: %s\n", priv->fpitBaseOrientation & FPIT_THEN_SWAP_XY ? "Yes" : "No"); xf86Msg(X_CONFIG, "FPIT Passive button mode: %s\n", priv->fpitPassive ? "Yes" : "No"); xf86Msg(X_CONFIG, "FPIT RandR tracking: %s\n", priv->fpitTrackRandR ? "Yes" : "No"); return Success; } _X_EXPORT InputDriverRec FPIT = { 1, /* driver version */ "fpit", /* driver name */ NULL, /* identify */ xf86FpitInit, /* pre-init */ xf86FpitUninit, /* un-init */ NULL, /* module */ default_options }; static pointer Plug(pointer module, pointer options, int *errmaj, int *errmin) { xf86AddInputDriver(&FPIT, module, 0); return module; } static void Unplug(pointer p) { } static XF86ModuleVersionInfo version_rec = { "fpit", 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} }; /* * This is the entry point in the module. The name * is setup after the pattern ModuleData. * Do not change it. */ _X_EXPORT XF86ModuleData fpitModuleData = { &version_rec, Plug, Unplug };