IMPS/2 HID devices w/ fbcon patch

Ok, it’s fixed. :slight_smile:

Apparently, with /dev/input/mice, the thing defaults to the PS/2 protocol
until you explicitly tell it to use IMPS/2. Always trying to set this
seems like a good idea to me, because it gives you access to the
mousewheel, and shouldn’t hurt anything if it can’t set it.

Thanks Ryan, your fix is in CVS. :slight_smile:

See ya!
-Sam Lantinga, Lead Programmer, Loki Software, Inc.

Ok, it’s fixed. :slight_smile:

Apparently, with /dev/input/mice, the thing defaults to the PS/2 protocol
until you explicitly tell it to use IMPS/2. Always trying to set this
seems like a good idea to me, because it gives you access to the
mousewheel, and shouldn’t hurt anything if it can’t set it.

So attached is a patch to SDL_fbevents.c against the latest CVS. It
explicitly checks for IMPS/2 support when opening /dev/input/mice
(which is now opened Read/Write first, and readonly failing that), always
tries to explicitly set IMPS/2 mode before the check, even if you use the
envr variable SDL_MOUSEDEV_IMPS2, and, if the check fails, flags the
mouse as a standard PS/2 protocol mouse. A couple other fixes are in there
too; a sizeof() where a number was hardcoded, a Uint8 where an unsigned
char was used, etc.

It seems to work well here; please try it and see if it helps.

–ryan.

-------------- next part --------------
--- SDL_fbevents.c_original	Mon Jun 11 09:03:40 2001
+++ SDL_fbevents.c	Mon Jun 11 08:59:33 2001
@@ -397,21 +397,64 @@
 	return available;
 }+
+/* rcg06112001 Set up IMPS/2 mode, if possible. This gives
+ *  us access to the mousewheel, etc. Returns zero if
+ *  writes to device failed, but you still need to query the
+ *  device to see which mode it's actually in.
+ */
+static int set_imps2_mode(int fd)
+{
+	/* If you wanted to control the mouse mode (and we do :)  ) ...
+		Set IMPS/2 protocol:
+			{0xf3,200,0xf3,100,0xf3,80}
+		Reset mouse device:
+			{0xFF}
+	*/
+	Uint8 set_imps2[] = {0xf3, 200, 0xf3, 100, 0xf3, 80};
+	Uint8 reset = 0xff;
+	fd_set fdset;
+	struct timeval tv;
+	int retval = 0;
+
+	if ( write(fd, &set_imps2, sizeof(set_imps2)) == sizeof(set_imps2) ) {
+		if (write(fd, &reset, sizeof (reset)) == sizeof (reset) ) {
+			retval = 1;
+		}
+	}
+
+	/* Get rid of any chatter from the above */
+	FD_ZERO(&fdset);
+	FD_SET(fd, &fdset);
+	tv.tv_sec = 0;
+	tv.tv_usec = 0;
+	while ( select(fd+1, &fdset, 0, 0, &tv) > 0 ) {
+		char temp[32];
+		read(fd, temp, sizeof(temp));
+	}
+
+	return retval;
+}
+
+
 /* Returns true if the mouse uses the IMPS/2 protocol */
 static int detect_imps2(int fd)
 {
 	int imps2;
 
 	imps2 = 0;
+
+	/* rcg06112001 Attempt to set IMPS/2 mode first, even with envr var... */
+	set_imps2_mode(fd);
+
 	if ( getenv("SDL_MOUSEDEV_IMPS2") ) {
 		imps2 = 1;
 	}
 	if ( ! imps2 ) {
-		unsigned char query_ps2 = 0xF2;
+		Uint8 query_ps2 = 0xF2;
 		fd_set fdset;
 		struct timeval tv;
 
-
 		/* Get rid of any mouse motion noise */
 		FD_ZERO(&fdset);
 		FD_SET(fd, &fdset);
@@ -422,16 +465,10 @@
 			read(fd, temp, sizeof(temp));
 		}
 
-		/* Query for the type of mouse protocol */
-		if ( write(fd, &query_ps2, 1) == 1 ) {
-			unsigned char ch = 0;
-
-			/* If you wanted to control the mouse mode:
-			   Set IMPS/2 protocol:
-				{0xf3,200,0xf3,100,0xf3,80}
-			   Reset mouse device:
-				{0xFF}
-			*/
+   		/* Query for the type of mouse protocol */
+   		if ( write(fd, &query_ps2, sizeof (query_ps2)) == sizeof (query_ps2)) {
+   			Uint8 ch = 0;
+
 			/* Get the mouse protocol response */
 			do {
 				FD_ZERO(&fdset);
@@ -441,7 +478,7 @@
 				if ( select(fd+1, &fdset, 0, 0, &tv) < 1 ) {
 					break;
 				}
-			} while ( (read(fd, &ch, 1) == 1) &&
+			} while ( (read(fd, &ch, sizeof (ch)) == sizeof (ch)) &&
 			          ((ch == 0xFA) || (ch == 0xAA)) );
 
 			/* Experimental values (Logitech wheelmouse) */
@@ -502,9 +539,16 @@
 		}
 		/* Now try to use the new HID unified mouse device */
 		if ( mouse_fd < 0 ) {
-			mouse_fd = open("/dev/input/mice", O_RDONLY, 0);
-			if ( mouse_fd >= 0 ) {
-				mouse_drv = MOUSE_IMPS2;
+			mouse_fd = open("/dev/input/mice", O_RDWR, 0);
+			if (mouse_fd < 0) {
+				mouse_fd = open("/dev/input/mice", O_RDONLY, 0);
+			}
+			if (mouse_fd >= 0) {
+				if (detect_imps2(mouse_fd)) {
+					mouse_drv = MOUSE_IMPS2;
+				} else {
+					mouse_drv = MOUSE_PS2;
+				}
 			}
 		}
 		/* Now try to use a modern PS/2 port mouse */

Ok, it’s fixed. :slight_smile:

Apparently, with /dev/input/mice, the thing defaults to the PS/2 protocol
until you explicitly tell it to use IMPS/2. Always trying to set this
seems like a good idea to me, because it gives you access to the
mousewheel, and shouldn’t hurt anything if it can’t set it.

Thanks Ryan, your fix is in CVS. :slight_smile:

Confirmation from here - problem solved :slight_smile:
Thank you very much Ryan!

Works fine, and all events are now 4 bytes so that’s fine.

G’day, eh? :slight_smile:
- Winterlion, back later when internet service doesn’t require a
long distance call for me…On Mon, 11 Jun 2001, Sam Lantinga wrote: