Define a truly synchronous API for the bus Tx path by putting all pending request to the write list and wait for the interrupt tx handler to wake us up. The ___mei_send() out path is also slightly reworked to make it look more like main.c:mei_write().
Signed-off-by: Samuel Ortiz <sa...@linux.intel.com> Signed-off-by: Tomas Winkler <tomas.wink...@intel.com> --- drivers/misc/mei/bus.c | 38 ++++++++++++++++++++++++++++---------- drivers/misc/mei/mei_dev.h | 1 + 2 files changed, 29 insertions(+), 10 deletions(-) diff --git a/drivers/misc/mei/bus.c b/drivers/misc/mei/bus.c index 4565e9a..85ca872 100644 --- a/drivers/misc/mei/bus.c +++ b/drivers/misc/mei/bus.c @@ -202,7 +202,7 @@ void mei_driver_unregister(struct mei_driver *driver) } EXPORT_SYMBOL_GPL(mei_driver_unregister); -int __mei_send(struct mei_cl *cl, u8 *buf, size_t length) +static int ___mei_send(struct mei_cl *cl, u8 *buf, size_t length, bool blocking) { struct mei_host *dev; struct mei_msg_hdr mei_hdr; @@ -253,11 +253,8 @@ int __mei_send(struct mei_cl *cl, u8 *buf, size_t length) cb->buf_idx = 0; mei_hdr.msg_complete = 0; cl->writing_state = MEI_WRITING; - list_add_tail(&cb->list, &dev->write_list.list); - - mutex_unlock(&dev->device_lock); - return length; + goto out; } dev->hbuf_is_ready = false; @@ -283,19 +280,30 @@ int __mei_send(struct mei_cl *cl, u8 *buf, size_t length) cl->writing_state = MEI_WRITING; cb->buf_idx = mei_hdr.length; - if (!mei_hdr.msg_complete) { - list_add_tail(&cb->list, &dev->write_list.list); - } else { +out: + if (mei_hdr.msg_complete) { if (mei_cl_flow_ctrl_reduce(cl)) { - err = -EIO; + err = -ENODEV; goto out_err; } - list_add_tail(&cb->list, &dev->write_waiting_list.list); + } else { + list_add_tail(&cb->list, &dev->write_list.list); } mutex_unlock(&dev->device_lock); + if (blocking && cl->writing_state != MEI_WRITE_COMPLETE) { + if (wait_event_interruptible(cl->tx_wait, + cl->writing_state == MEI_WRITE_COMPLETE)) { + if (signal_pending(current)) + err = -EINTR; + err = -ERESTARTSYS; + mutex_lock(&dev->device_lock); + goto out_err; + } + } + return mei_hdr.length; out_err: @@ -362,6 +370,16 @@ out: return r_length; } +inline int __mei_async_send(struct mei_cl *cl, u8 *buf, size_t length) +{ + return ___mei_send(cl, buf, length, 0); +} + +inline int __mei_send(struct mei_cl *cl, u8 *buf, size_t length) +{ + return ___mei_send(cl, buf, length, 1); +} + int mei_send(struct mei_device *device, u8 *buf, size_t length) { struct mei_cl *cl = device->cl; diff --git a/drivers/misc/mei/mei_dev.h b/drivers/misc/mei/mei_dev.h index 546cd87..fc485d4 100644 --- a/drivers/misc/mei/mei_dev.h +++ b/drivers/misc/mei/mei_dev.h @@ -273,6 +273,7 @@ struct mei_hw_ops { struct mei_device *mei_add_device(struct mei_host *mei_host, uuid_le uuid, char *name); void mei_remove_device(struct mei_device *device); +int __mei_async_send(struct mei_cl *cl, u8 *buf, size_t length); int __mei_send(struct mei_cl *cl, u8 *buf, size_t length); int __mei_recv(struct mei_cl *cl, u8 *buf, size_t length); -- 1.7.10.4 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/