Skip site navigation (1)Skip section navigation (2)
Date:      Sat, 8 Oct 2016 18:19:52 +0000 (UTC)
From:      Oleksandr Tymoshenko <gonzo@FreeBSD.org>
To:        src-committers@freebsd.org, svn-src-all@freebsd.org, svn-src-head@freebsd.org
Subject:   svn commit: r306860 - head/sys/arm/broadcom/bcm2835
Message-ID:  <201610081819.u98IJqur097213@repo.freebsd.org>

next in thread | raw e-mail | index | archive | help
Author: gonzo
Date: Sat Oct  8 18:19:52 2016
New Revision: 306860
URL: https://svnweb.freebsd.org/changeset/base/306860

Log:
  Add multitouch support for RPi's FT5406
  
  - Add multitouch support (protocol B)
  - Report physical size of the screen
  - Switch from using busy loop to callbacks
  - Enable callbacks only when there is active listener on /dev/input/eventX
  
  Submitted by:	Vladimir Kondratiev <wulf@cicgroup.ru>

Modified:
  head/sys/arm/broadcom/bcm2835/bcm2835_ft5406.c

Modified: head/sys/arm/broadcom/bcm2835/bcm2835_ft5406.c
==============================================================================
--- head/sys/arm/broadcom/bcm2835/bcm2835_ft5406.c	Sat Oct  8 18:16:18 2016	(r306859)
+++ head/sys/arm/broadcom/bcm2835/bcm2835_ft5406.c	Sat Oct  8 18:19:52 2016	(r306860)
@@ -43,7 +43,6 @@ __FBSDID("$FreeBSD$");
 #include <sys/poll.h>
 #include <sys/uio.h>
 #include <sys/conf.h>
-#include <sys/kthread.h>
 
 #include <vm/vm.h>
 #include <vm/pmap.h>
@@ -102,14 +101,20 @@ __FBSDID("$FreeBSD$");
 				    (buf[FT5406_POINT_YL(n)]))
 #define	GET_TOUCH_ID(buf, n)	((buf[FT5406_POINT_YH(n)] >> 4) & 0xf)
 
-#define	NO_POINTS	99
-#define	SCREEN_WIDTH	800
-#define	SCREEN_HEIGHT	480
+#define	NO_POINTS		99
+#define	SCREEN_WIDTH		800
+#define	SCREEN_HEIGHT		480
+#define	SCREEN_WIDTH_MM		155
+#define	SCREEN_HEIGHT_MM	86
+#define	SCREEN_RES_X	(SCREEN_WIDTH / SCREEN_WIDTH_MM)
+#define	SCREEN_RES_Y	(SCREEN_HEIGHT / SCREEN_HEIGHT_MM)
+#define	MAX_TOUCH_ID	(10 - 1)
 
 struct ft5406ts_softc {
 	device_t		sc_dev;
 	struct mtx		sc_mtx;
-	struct proc		*sc_worker;
+	int			sc_tick;
+	struct callout		sc_callout;
 
 	/* mbox buffer (mapped to KVA) */
 	uint8_t			*touch_buf;
@@ -118,86 +123,76 @@ struct ft5406ts_softc {
 	struct intr_config_hook	sc_init_hook;
 
 	struct evdev_dev	*sc_evdev;
-	int			sc_detaching;
+
+	uint8_t			sc_window[FT5406_WINDOW_SIZE];
+};
+
+static evdev_open_t ft5406ts_ev_open;
+static evdev_close_t ft5406ts_ev_close;
+
+static const struct evdev_methods ft5406ts_evdev_methods = {
+	.ev_open = &ft5406ts_ev_open,
+	.ev_close = &ft5406ts_ev_close,
 };
 
 static void
-ft5406ts_worker(void *data)
+ft5406ts_callout(void *data)
 {
 	struct ft5406ts_softc *sc = (struct ft5406ts_softc *)data;
 	int points;
-	int id, new_x, new_y, i, new_pen_down, updated;
-	int x, y, pen_down;
-	uint8_t window[FT5406_WINDOW_SIZE];
-	int tick;
+	int id, i, x, y;
 
-	/* 60Hz */
-	tick = hz*17/1000;
-	if (tick == 0)
-		tick = 1;
-
-	x = y = -1;
-	pen_down = 0;
-
-	FT5406_LOCK(sc);
-	while(1) {
-		msleep(sc, &sc->sc_mtx, PCATCH | PZERO, "ft5406ts", tick);
-
-		if (sc->sc_detaching)
-			break;
-
-		memcpy(window, sc->touch_buf, sizeof(window));
-		sc->touch_buf[FT5406_NUM_POINTS] = NO_POINTS;
-
-		points = GET_NUM_POINTS(window);
-		/*
-		 * No update from VC - do nothing
-		 */
-		if (points == NO_POINTS)
-			continue;
+	FT5406_LOCK_ASSERT(sc);
 
-		/* No points and pen is already up */
-		if ((points == 0) && !pen_down)
-			continue;
+	memcpy(sc->sc_window, sc->touch_buf, FT5406_WINDOW_SIZE);
+	sc->touch_buf[FT5406_NUM_POINTS] = NO_POINTS;
 
-		new_pen_down = 0;
-		for (i = 0; i < points; i++) {
-			id = GET_TOUCH_ID(window, 0);
-			/* For now consider only touch 0 */
-			if (id != 0)
-				continue;
-			new_pen_down = 1;
-			new_x = GET_X(window, 0);
-			new_y = GET_Y(window, 0);
+	points = GET_NUM_POINTS(sc->sc_window);
+	/*
+	 * No update from VC - do nothing.
+	 */
+	if (points == NO_POINTS)
+		goto out;
+
+	for (i = 0; i < points; i++) {
+		id = GET_TOUCH_ID(sc->sc_window, i);
+		x = GET_X(sc->sc_window, i);
+		y = GET_Y(sc->sc_window, i);
+
+		if (id > MAX_TOUCH_ID) {
+			device_printf(sc->sc_dev, "bad touch id: %d", id);
+			continue;
 		}
+		evdev_push_event(sc->sc_evdev, EV_ABS, ABS_MT_SLOT, id);
+		evdev_push_event(sc->sc_evdev, EV_ABS, ABS_MT_TRACKING_ID, id);
+		evdev_push_event(sc->sc_evdev, EV_ABS, ABS_MT_POSITION_X, x);
+		evdev_push_event(sc->sc_evdev, EV_ABS, ABS_MT_POSITION_Y, y);
+	}
+	evdev_sync(sc->sc_evdev);
+out:
+	callout_reset(&sc->sc_callout, sc->sc_tick, ft5406ts_callout, sc);
+}
 
-		updated = 0;
+static void
+ft5406ts_ev_close(struct evdev_dev *evdev, void *data)
+{
+	struct ft5406ts_softc *sc = (struct ft5406ts_softc *)data;
 
-		if (new_x != x) {
-			x = new_x;
-			updated = 1;
-		}
+	FT5406_LOCK_ASSERT(sc);
 
-		if (new_y != y) {
-			y = new_y;
-			updated = 1;
-		}
+	callout_stop(&sc->sc_callout);
+}
 
-		if (new_pen_down != pen_down) {
-			pen_down = new_pen_down;
-			updated = 1;
-		}
+static int
+ft5406ts_ev_open(struct evdev_dev *evdev, void *data)
+{
+	struct ft5406ts_softc *sc = (struct ft5406ts_softc *)data;
 
-		if (updated) {
-			evdev_push_event(sc->sc_evdev, EV_ABS, ABS_X, x);
-			evdev_push_event(sc->sc_evdev, EV_ABS, ABS_Y, y);
-			evdev_push_event(sc->sc_evdev, EV_KEY, BTN_TOUCH, pen_down);
-			evdev_sync(sc->sc_evdev);
-		}
-	}
-	FT5406_UNLOCK(sc);
+	FT5406_LOCK_ASSERT(sc);
 
-	kproc_exit(0);
+	callout_reset(&sc->sc_callout, sc->sc_tick, ft5406ts_callout, sc);
+
+	return (0);
 }
 
 static void
@@ -234,33 +229,40 @@ ft5406ts_init(void *arg)
 	touchbuf = VCBUS_TO_PHYS(msg.body.resp.address);
 	sc->touch_buf = (uint8_t*)pmap_mapdev(touchbuf, FT5406_WINDOW_SIZE);
 
+	/* 60Hz */
+	sc->sc_tick = hz * 17 / 1000;
+	if (sc->sc_tick == 0)
+		sc->sc_tick = 1;
+
 	sc->sc_evdev = evdev_alloc();
 	evdev_set_name(sc->sc_evdev, device_get_desc(sc->sc_dev));
 	evdev_set_phys(sc->sc_evdev, device_get_nameunit(sc->sc_dev));
-	evdev_set_id(sc->sc_evdev, BUS_VIRTUAL, 0, 0, 0);
+	evdev_set_id(sc->sc_evdev, BUS_HOST, 0, 0, 0);
+	evdev_set_methods(sc->sc_evdev, sc, &ft5406ts_evdev_methods);
+	evdev_set_flag(sc->sc_evdev, EVDEV_FLAG_MT_STCOMPAT);
+	evdev_set_flag(sc->sc_evdev, EVDEV_FLAG_MT_AUTOREL);
 	evdev_support_prop(sc->sc_evdev, INPUT_PROP_DIRECT);
 	evdev_support_event(sc->sc_evdev, EV_SYN);
 	evdev_support_event(sc->sc_evdev, EV_ABS);
-	evdev_support_event(sc->sc_evdev, EV_KEY);
 
-	evdev_support_abs(sc->sc_evdev, ABS_X, 0, 0,
-	    SCREEN_WIDTH, 0, 0, 0);
-	evdev_support_abs(sc->sc_evdev, ABS_Y, 0, 0,
-	    SCREEN_HEIGHT, 0, 0, 0);
+	evdev_support_abs(sc->sc_evdev, ABS_MT_SLOT, 0, 0,
+	    MAX_TOUCH_ID, 0, 0, 0);
+	evdev_support_abs(sc->sc_evdev, ABS_MT_TRACKING_ID, 0, -1,
+	    MAX_TOUCH_ID, 0, 0, 0);
+	evdev_support_abs(sc->sc_evdev, ABS_MT_POSITION_X, 0, 0,
+	    SCREEN_WIDTH, 0, 0, SCREEN_RES_X);
+	evdev_support_abs(sc->sc_evdev, ABS_MT_POSITION_Y, 0, 0,
+	    SCREEN_HEIGHT, 0, 0, SCREEN_RES_Y);
 
-	evdev_support_key(sc->sc_evdev, BTN_TOUCH);
-
-	err = evdev_register(sc->sc_evdev);
+	err = evdev_register_mtx(sc->sc_evdev, &sc->sc_mtx);
 	if (err) {
 		evdev_free(sc->sc_evdev);
+		sc->sc_evdev = NULL;	/* Avoid double free */
 		return;
 	}
 
 	sc->touch_buf[FT5406_NUM_POINTS] = NO_POINTS;
-	if (kproc_create(ft5406ts_worker, (void*)sc, &sc->sc_worker, 0, 0,
-	    "ft5406ts_worker") != 0) {
-		printf("failed to create ft5406ts_worker\n");
-	}
+	callout_init_mtx(&sc->sc_callout, &sc->sc_mtx, 0);
 }
 
 static int
@@ -292,6 +294,7 @@ ft5406ts_attach(device_t dev)
 
 	if (config_intrhook_establish(&sc->sc_init_hook) != 0) {
 		device_printf(dev, "config_intrhook_establish failed\n");
+		FT5406_LOCK_DESTROY(sc);
 		return (ENOMEM);
 	}
 
@@ -305,14 +308,7 @@ ft5406ts_detach(device_t dev)
 
 	sc = device_get_softc(dev);
 
-	FT5406_LOCK(sc);
-	if (sc->sc_worker)
-		sc->sc_detaching = 1;
-	wakeup(sc);
-	FT5406_UNLOCK(sc);
-
-	if (sc->sc_evdev)
-		evdev_free(sc->sc_evdev);
+	evdev_free(sc->sc_evdev);
 
 	FT5406_LOCK_DESTROY(sc);
 



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?201610081819.u98IJqur097213>