Commit 13970f80 authored by Christian Engelmayer's avatar Christian Engelmayer Committed by Greg Kroah-Hartman

Input: usbtouchscreen - separate report and transmit buffer size handling

commit 4ef38351 upstream.

This patch supports the separate handling of the USB transfer buffer length
and the length of the buffer used for multi packet support. For devices
supporting multiple report or diagnostic packets, the USB transfer size is now
limited to the USB endpoints wMaxPacketSize - otherwise it defaults to the
configured report packet size as before.

This fixes an issue where event reporting can be delayed for an arbitrary
time for multi packet devices. For instance the report size for eGalax devices
is defined to the 16 byte maximum diagnostic packet size as opposed to the 5
byte report packet size. In case the driver requests 16 byte from the USB
interrupt endpoint, the USB host controller driver needs to split up the
request into 2 accesses according to the endpoints wMaxPacketSize of 8 byte.
When the first transfer is answered by the eGalax device with not less than
the full 8 byte requested, the host controller has got no way of knowing
whether the touch controller has got additional data queued and will issue
the second transfer. If per example a liftoff event finishes at such a
wMaxPacketSize boundary, the data will not be available to the usbtouch driver
until a further event is triggered and transfered to the host. From user
perspective the BTN_TOUCH release event in this case is stuck until the next
touch down event.
Signed-off-by: default avatarChristian Engelmayer <christian.engelmayer@frequentis.com>
Signed-off-by: default avatarDmitry Torokhov <dmitry.torokhov@gmail.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 6905bdb1
...@@ -106,6 +106,7 @@ struct usbtouch_device_info { ...@@ -106,6 +106,7 @@ struct usbtouch_device_info {
struct usbtouch_usb { struct usbtouch_usb {
unsigned char *data; unsigned char *data;
dma_addr_t data_dma; dma_addr_t data_dma;
int data_size;
unsigned char *buffer; unsigned char *buffer;
int buf_len; int buf_len;
struct urb *irq; struct urb *irq;
...@@ -1474,7 +1475,7 @@ static int usbtouch_reset_resume(struct usb_interface *intf) ...@@ -1474,7 +1475,7 @@ static int usbtouch_reset_resume(struct usb_interface *intf)
static void usbtouch_free_buffers(struct usb_device *udev, static void usbtouch_free_buffers(struct usb_device *udev,
struct usbtouch_usb *usbtouch) struct usbtouch_usb *usbtouch)
{ {
usb_free_coherent(udev, usbtouch->type->rept_size, usb_free_coherent(udev, usbtouch->data_size,
usbtouch->data, usbtouch->data_dma); usbtouch->data, usbtouch->data_dma);
kfree(usbtouch->buffer); kfree(usbtouch->buffer);
} }
...@@ -1519,7 +1520,20 @@ static int usbtouch_probe(struct usb_interface *intf, ...@@ -1519,7 +1520,20 @@ static int usbtouch_probe(struct usb_interface *intf,
if (!type->process_pkt) if (!type->process_pkt)
type->process_pkt = usbtouch_process_pkt; type->process_pkt = usbtouch_process_pkt;
usbtouch->data = usb_alloc_coherent(udev, type->rept_size, usbtouch->data_size = type->rept_size;
if (type->get_pkt_len) {
/*
* When dealing with variable-length packets we should
* not request more than wMaxPacketSize bytes at once
* as we do not know if there is more data coming or
* we filled exactly wMaxPacketSize bytes and there is
* nothing else.
*/
usbtouch->data_size = min(usbtouch->data_size,
usb_endpoint_maxp(endpoint));
}
usbtouch->data = usb_alloc_coherent(udev, usbtouch->data_size,
GFP_KERNEL, &usbtouch->data_dma); GFP_KERNEL, &usbtouch->data_dma);
if (!usbtouch->data) if (!usbtouch->data)
goto out_free; goto out_free;
...@@ -1578,12 +1592,12 @@ static int usbtouch_probe(struct usb_interface *intf, ...@@ -1578,12 +1592,12 @@ static int usbtouch_probe(struct usb_interface *intf,
if (usb_endpoint_type(endpoint) == USB_ENDPOINT_XFER_INT) if (usb_endpoint_type(endpoint) == USB_ENDPOINT_XFER_INT)
usb_fill_int_urb(usbtouch->irq, udev, usb_fill_int_urb(usbtouch->irq, udev,
usb_rcvintpipe(udev, endpoint->bEndpointAddress), usb_rcvintpipe(udev, endpoint->bEndpointAddress),
usbtouch->data, type->rept_size, usbtouch->data, usbtouch->data_size,
usbtouch_irq, usbtouch, endpoint->bInterval); usbtouch_irq, usbtouch, endpoint->bInterval);
else else
usb_fill_bulk_urb(usbtouch->irq, udev, usb_fill_bulk_urb(usbtouch->irq, udev,
usb_rcvbulkpipe(udev, endpoint->bEndpointAddress), usb_rcvbulkpipe(udev, endpoint->bEndpointAddress),
usbtouch->data, type->rept_size, usbtouch->data, usbtouch->data_size,
usbtouch_irq, usbtouch); usbtouch_irq, usbtouch);
usbtouch->irq->dev = udev; usbtouch->irq->dev = udev;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment