aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/hid
diff options
context:
space:
mode:
authorDavid Herrmann <dh.herrmann@googlemail.com>2012-06-10 15:16:15 +0200
committerHumberto Borba <humberos@gmail.com>2013-01-12 21:04:15 -0200
commit402b6a44f1428cf89fb173929bb7895c58c231fa (patch)
treeb6f9f32aff297b104ac61dfe576a4199ee7bdf78 /drivers/hid
parent6b48868f97c76007769e6f935ef636182ff3c23a (diff)
downloadkernel_samsung_smdk4412-402b6a44f1428cf89fb173929bb7895c58c231fa.zip
kernel_samsung_smdk4412-402b6a44f1428cf89fb173929bb7895c58c231fa.tar.gz
kernel_samsung_smdk4412-402b6a44f1428cf89fb173929bb7895c58c231fa.tar.bz2
HID: uhid: allow poll()'ing on uhid devices
As long as the internal buffer is not empty, we return POLLIN to user-space. uhid->head and uhid->tail are no atomics so the comparison may return inexact results. However, this doesn't matter here as user-space would need to poll() in two threads simultaneously to trigger this. And in this case it doesn't matter if a cached result is returned or the exact new result as user-space does not know which thread returns first from poll() and the following read(). So it is safe to compare the values without locking. Signed-off-by: David Herrmann <dh.herrmann@googlemail.com> Signed-off-by: Jiri Kosina <jkosina@suse.cz>
Diffstat (limited to 'drivers/hid')
-rw-r--r--drivers/hid/uhid.c7
1 files changed, 7 insertions, 0 deletions
diff --git a/drivers/hid/uhid.c b/drivers/hid/uhid.c
index 05ef4b0..b1a477f 100644
--- a/drivers/hid/uhid.c
+++ b/drivers/hid/uhid.c
@@ -117,6 +117,13 @@ static ssize_t uhid_char_write(struct file *file, const char __user *buffer,
static unsigned int uhid_char_poll(struct file *file, poll_table *wait)
{
+ struct uhid_device *uhid = file->private_data;
+
+ poll_wait(file, &uhid->waitq, wait);
+
+ if (uhid->head != uhid->tail)
+ return POLLIN | POLLRDNORM;
+
return 0;
}