Commit d758b1f0 authored by David Herrmann's avatar David Herrmann Committed by Jiri Kosina

HID: wiimote: wake up if output queue failed

Our output queue is asynchronous but synchronous reports may wait for a
response to their request. Therefore, wake them up unconditionally if an
output report couldn't be sent. But keep the report ID intact so we don't
incorrectly assume our request succeeded.

Note that the underlying connection is required to be reliable and does
retransmission itself. So it is safe to assume that if the transmission
fails, the device is in inconsistent state. Hence, we abort every request
if any output report fails. No need to verify which report failed.
Signed-off-by: default avatarDavid Herrmann <dh.herrmann@gmail.com>
Signed-off-by: default avatarJiri Kosina <jkosina@suse.cz>
parent 6b80bb94
...@@ -58,11 +58,11 @@ static enum power_supply_property wiimote_battery_props[] = { ...@@ -58,11 +58,11 @@ static enum power_supply_property wiimote_battery_props[] = {
/* output queue handling */ /* output queue handling */
static ssize_t wiimote_hid_send(struct hid_device *hdev, __u8 *buffer, static int wiimote_hid_send(struct hid_device *hdev, __u8 *buffer,
size_t count) size_t count)
{ {
__u8 *buf; __u8 *buf;
ssize_t ret; int ret;
if (!hdev->hid_output_raw_report) if (!hdev->hid_output_raw_report)
return -ENODEV; return -ENODEV;
...@@ -84,14 +84,20 @@ static void wiimote_queue_worker(struct work_struct *work) ...@@ -84,14 +84,20 @@ static void wiimote_queue_worker(struct work_struct *work)
struct wiimote_data *wdata = container_of(queue, struct wiimote_data, struct wiimote_data *wdata = container_of(queue, struct wiimote_data,
queue); queue);
unsigned long flags; unsigned long flags;
int ret;
spin_lock_irqsave(&wdata->queue.lock, flags); spin_lock_irqsave(&wdata->queue.lock, flags);
while (wdata->queue.head != wdata->queue.tail) { while (wdata->queue.head != wdata->queue.tail) {
spin_unlock_irqrestore(&wdata->queue.lock, flags); spin_unlock_irqrestore(&wdata->queue.lock, flags);
wiimote_hid_send(wdata->hdev, ret = wiimote_hid_send(wdata->hdev,
wdata->queue.outq[wdata->queue.tail].data, wdata->queue.outq[wdata->queue.tail].data,
wdata->queue.outq[wdata->queue.tail].size); wdata->queue.outq[wdata->queue.tail].size);
if (ret < 0) {
spin_lock_irqsave(&wdata->state.lock, flags);
wiimote_cmd_abort(wdata);
spin_unlock_irqrestore(&wdata->state.lock, flags);
}
spin_lock_irqsave(&wdata->queue.lock, flags); spin_lock_irqsave(&wdata->queue.lock, flags);
wdata->queue.tail = (wdata->queue.tail + 1) % WIIMOTE_BUFSIZE; wdata->queue.tail = (wdata->queue.tail + 1) % WIIMOTE_BUFSIZE;
...@@ -108,7 +114,9 @@ static void wiimote_queue(struct wiimote_data *wdata, const __u8 *buffer, ...@@ -108,7 +114,9 @@ static void wiimote_queue(struct wiimote_data *wdata, const __u8 *buffer,
if (count > HID_MAX_BUFFER_SIZE) { if (count > HID_MAX_BUFFER_SIZE) {
hid_warn(wdata->hdev, "Sending too large output report\n"); hid_warn(wdata->hdev, "Sending too large output report\n");
return;
spin_lock_irqsave(&wdata->queue.lock, flags);
goto out_error;
} }
/* /*
...@@ -134,8 +142,14 @@ static void wiimote_queue(struct wiimote_data *wdata, const __u8 *buffer, ...@@ -134,8 +142,14 @@ static void wiimote_queue(struct wiimote_data *wdata, const __u8 *buffer,
wdata->queue.head = newhead; wdata->queue.head = newhead;
} else { } else {
hid_warn(wdata->hdev, "Output queue is full"); hid_warn(wdata->hdev, "Output queue is full");
goto out_error;
} }
goto out_unlock;
out_error:
wiimote_cmd_abort(wdata);
out_unlock:
spin_unlock_irqrestore(&wdata->queue.lock, flags); spin_unlock_irqrestore(&wdata->queue.lock, flags);
} }
......
...@@ -195,6 +195,16 @@ static inline void wiimote_cmd_complete(struct wiimote_data *wdata) ...@@ -195,6 +195,16 @@ static inline void wiimote_cmd_complete(struct wiimote_data *wdata)
complete(&wdata->state.ready); complete(&wdata->state.ready);
} }
/* requires the state.lock spinlock to be held */
static inline void wiimote_cmd_abort(struct wiimote_data *wdata)
{
/* Abort synchronous request by waking up the sleeping caller. But
* reset the state.cmd field to an invalid value so no further event
* handlers will work with it. */
wdata->state.cmd = WIIPROTO_REQ_MAX;
complete(&wdata->state.ready);
}
static inline int wiimote_cmd_acquire(struct wiimote_data *wdata) static inline int wiimote_cmd_acquire(struct wiimote_data *wdata)
{ {
return mutex_lock_interruptible(&wdata->state.sync) ? -ERESTARTSYS : 0; return mutex_lock_interruptible(&wdata->state.sync) ? -ERESTARTSYS : 0;
...@@ -223,11 +233,17 @@ static inline int wiimote_cmd_wait(struct wiimote_data *wdata) ...@@ -223,11 +233,17 @@ static inline int wiimote_cmd_wait(struct wiimote_data *wdata)
{ {
int ret; int ret;
/* The completion acts as implicit memory barrier so we can safely
* assume that state.cmd is set on success/failure and isn't accessed
* by any other thread, anymore. */
ret = wait_for_completion_interruptible_timeout(&wdata->state.ready, HZ); ret = wait_for_completion_interruptible_timeout(&wdata->state.ready, HZ);
if (ret < 0) if (ret < 0)
return -ERESTARTSYS; return -ERESTARTSYS;
else if (ret == 0) else if (ret == 0)
return -EIO; return -EIO;
else if (wdata->state.cmd != WIIPROTO_REQ_NULL)
return -EIO;
else else
return 0; return 0;
} }
...@@ -236,9 +252,12 @@ static inline int wiimote_cmd_wait_noint(struct wiimote_data *wdata) ...@@ -236,9 +252,12 @@ static inline int wiimote_cmd_wait_noint(struct wiimote_data *wdata)
{ {
unsigned long ret; unsigned long ret;
/* no locking needed; see wiimote_cmd_wait() */
ret = wait_for_completion_timeout(&wdata->state.ready, HZ); ret = wait_for_completion_timeout(&wdata->state.ready, HZ);
if (!ret) if (!ret)
return -EIO; return -EIO;
else if (wdata->state.cmd != WIIPROTO_REQ_NULL)
return -EIO;
else else
return 0; return 0;
} }
......
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