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