diff options
author | David Herrmann <dh.herrmann@googlemail.com> | 2012-06-10 15:16:15 +0200 |
---|---|---|
committer | Humberto Borba <humberos@gmail.com> | 2013-01-12 21:04:15 -0200 |
commit | 402b6a44f1428cf89fb173929bb7895c58c231fa (patch) | |
tree | b6f9f32aff297b104ac61dfe576a4199ee7bdf78 /drivers/hid | |
parent | 6b48868f97c76007769e6f935ef636182ff3c23a (diff) | |
download | kernel_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.c | 7 |
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; } |