The touch sensor integrated with the display is a DST sensor from 3M that is connected via USB to a computer (in this case the BB) and is HID compliant (no need for special drivers). Before it is used it has to be calibrated to tell the touch controller what is the resolution of the screen so it can report the coordinates accordingly. I calibrated it in Windows (with a SW provided by 3M) for a720p resolution and then I was able to use it seamlessly in Ubuntu once I set the resolution of the screen to 720p. The good news, the HW works and is a SW problem...
When I connected the touch sensor to the BB, I learned the following from the kernel log and Android logcat:
usb 2-1.2: new full speed USB device using musb_hdrc and address 3
usb 2-1.2: device v0596 p0300 is not supported
input: 3M 3M MicroTouch USB controller as /devices/platform/musb_hdrc/usb2/2-1/2-1.2/2-1.2:1.0/input/input1
generic-usb 0003:0596:0300.0001: input: USB HID v1.10 Pointer [3M 3M MicroTouch USB controller] on usb-musb_hdrc-1.2
E/EventHub( 868): could not get driver version for /dev/input/mouse0, Not a typewriter
I/EventHub( 868): New device: path=/dev/input/event1 name=3M 3M MicroTouch USB controller id=0x10001 (of 0x2) index=2 fd=29 classes=0x0
I/KeyInputQueue( 868): Ignoring non-input device: id=0x10001, name=3M 3M MicroTouch USB controller
First I checked the E/EventHub error but I realized the same message was displayed when I connected a mouse to the BB and it worked fine so I ignored it. Then I realized that the last message was different for the mouse, instead of displaying "... Ignoring non-input device...", the log for the mouse was as follow:
I/KeyInputQueue( 868): Device added: id=0x10002, name=Genius NetScroll + Mini Traveler, classes=40
This is the snipet of code from where this print comes from: androidRoot/frameworks/base/services/java/com/android/server/KeyInputQueue.java:
if (ev.type == RawInputEvent.EV_DEVICE_ADDED) {
synchronized (mFirst) {
di = newInputDevice(ev.deviceId);
if (di.classes != 0 ) {
// If this device is some kind of input class,
// we care about it.
mDevices.put(ev.deviceId, di);
if ((di.classes & RawInputEvent.CLASS_TOUCHSCREEN)
!= 0) {
readVirtualKeys(di.name);
}
// The configuration may have changed because
// of this device.
configChanged = true;
} else {
// We won't do anything with this device.
mIgnoredDevices.put(ev.deviceId, di);
Slog.i(TAG, "Ignoring non-input device: id=0x"
+ Integer.toHexString(di.id)
+ ", name=" + di.name);
}
}
}
else if (ev.type == RawInputEvent.EV_DEVICE_REMOVED) {
/*handle device removals*/
}
else{
di = getInputDevice(ev.deviceId);
if (di == null) {
Slog.i(TAG, "getInputDevice() null device!");
// This may be some junk from an ignored device.
continue;
}
/*handle any other kind of events (mouse moved,
key pressed, etc.)*/
}
If the event type corresponds to RawInputEvent.EV_DEVICE_ADDED, a new input device is created and then the classes field is checked. If it is different to zero the new input device is put in a List otherwise is ignored. When a event type different to EV_DEVICE_ADDED or EV_DEVICE_REMOVED arrives (the else at the bottom) it is checked whether the device that generated the event is in that list, if it is not the event is just ignored.
Then I went to the file androidRoot/frameworks/base/libs/ui/EventHub.cpp to the function open_device (called when the input device is created). Here a file descriptor is opened using the argument passed to this function and then the capabilities of the device are determined using the ioctl EVIOCGBIT. Depending on the device capabilities is the value of the field di.classes, if the device does not have the capabilities that Android is looking for this field is never set and its value remains zero.
First thing I did was to determine what were the capabilities supported by the touch sensor. The types of features supported by the event interface are:
- EV_KEY: absolute binary results, such as keys and buttons.
- EV_REL: relative results, such as the axes on a mouse.
- EV_ABS: absolute integer results, such as the axes on a joystick or for a tablet.
- EV_MSC: miscellaneous uses that didn't fit anywhere else.
- EV_LED: LEDs and similar indications.
- EV_SND: sound output, such as buzzers.
- EV_REP: enables autorepeat of keys in the input core.
- EV_FF: sends force-feedback effects to a device.
- EV_FF_STATUS: device reporting of force-feedback effects back to the host.
- EV_PWR: power management events.
These were pretty much the same capabilities supported by the mouse I was using to compare configuration differences except for EV_ABS, instead of this feature the mouse has EV_REL.
In the open_device function the following logic takes place:
/*check whether the device has support for buttons. If it does, flags representing the buttons supported are set in the key_bitmask array.
The value of these flags is under androidRoot/frameworks/base/core/java/android/view/RawInputEvent.java*/
if (ioctl(fd, EVIOCGBIT(EV_KEY, sizeof(key_bitmask)), key_bitmask) >= 0){
... /*code to check if it is a keyboard*/
}
/*check if the flag that corresponds to the mouse button (left button) was set. If it did, it means it is a mouse kind device*/
if (test_bit(BTN_MOUSE, key_bitmask)) {
/*For the touch sensor only the mouse button was
present, for he mouse the right, middle and mouse
button were present.
However, inside this if it was checked only if the
device had support for EV_REL. Which the touch
sensor did not. */
if (ioctl(fd, EVIOCGBIT(EV_REL, sizeof(rel_bitmask)),
rel_bitmask) >= 0){
/*code to set mouse or trackball class*/
}
/*I had to add the logic to detect if the device had
support for EV_ABS*/
if (ioctl(fd, EVIOCGBIT(EV_ABS, sizeof(abs_bitmask)),
abs_bitmask) && (device->classes == 0) ){
if (test_bit(ABS_X, abs_bitmask) && test_bit(ABS_Y, abs_bitmask)) {
device->classes |=CLASS_MOUSE;
} }
}
After doing so the touch sensor was added to the list of devices. The mouse cursor was now displayed in the upper left corner but it did not respond to touches due to the handling routine was surrounded by an if statement to only handle events related to EV_REL.
A brief comment, the difference with EV_REL and EV_ABS is that EV_ABS returns the cordinate where the click or movement happened i.e (500x700) and the EV_REL returns the cordinate relative to the previous coordinate i.e (5,4).
In the event handling routine I added the following code to take care of movment events generated by the touch sensor.
else if (ev.type == RawInputEvent.EV_ABS &&
(classes&RawInputEvent.CLASS_MOUSE) != 0 ) {
if (ev.scancode == RawInputEvent.ABS_X) {
int xvalue = ev.value;
di.mAbs.changed = true;
xvalue = (xvalue*mDisplayWidth)/1023;
di.mAbs.mNextData[MotionEvent.SAMPLE_X] = xvalue;
Slog.i(TAG,"mDisplayWidth " + mDisplayWidth +
" xvalue " + xvalue);
} else if (ev.scancode == RawInputEvent.ABS_Y) {
int yvalue =ev.value;
di.mAbs.changed = true;
yvalue = (yvalue*mDisplayHeight)/1023;
Slog.i(TAG,"mDisplayHeight " + mDisplayHeight +
" yvalue " + yvalue);
di.mAbs.mNextData[MotionEvent.SAMPLE_Y] = yvalue;
}
}
The X and Y coordinate values were always in the range from 0 to 1023 no matter that I calibrated the touch screen so I had to do some maths to adjust the coordinates with the display resolution.
Click events were handled correctly with the original code so I did not have to modify anything there.
To give a better illusion of a touch screen I "removed" the mouse cursor displayed in the screen (I guess this cursor was displayed because the touch sensor was identified as some kind of mouse) by making the cursor transparent in the file:
androidRoot/frameworks/base/services/java/com/android/server/WindowManagerService.java
In the function performLayoutAndPlaceSurfacesLockedInner() set alpha blending to zero in the lines:
tPaint.setColor(0x00ffff); //black color for the cursor
mCanvas.drawColor(0x00000000); // white color for the
// border of the cursor
Some extra tips to build only the libraries you need in BeagleBoard/Android enviroment:
$ cd $path/to/androidRoot
$ source build/envsetup.sh
$ lunch beagleboard-eng //this is the target device for which I am building
$ cd path/to/the/file/modified
$ mm
...
Install: out/target/product/beagleboard/system/framework/services.jar
Once the build finished replace the output file (.so or .jar) in your file system and reboot.
$ sudo cp $androidRoot/out/target/product/beagleboard/system/framework/services.jar /media/linus_fs_sdcard/system/framework
Check out the following video of the touch screen working:
Gary, this is awesome and something I was just about to work on. Could I hire you for a project? jeremy.shubert@gmail.com
ReplyDeletecan we make it possible on unrooted phone , basically I am writing Bluetooth mouse driver app for android. I want to know is that possible to implement mouse cursor in same fashion as on desktop . What i have done so far is that i have connected my HID mouse to android device via Bluetooth . Mouse is sending reports to device and the Intent service receive it and parse it broadcast x and y coordinates . Now i want to display cursor with same functionality as on desktop . Is that possible if yes please help !! any link or code . Thank you
ReplyDelete