diff options
Diffstat (limited to 'drivers/misc/mei/client.c')
| -rw-r--r-- | drivers/misc/mei/client.c | 91 | 
1 files changed, 57 insertions, 34 deletions
| diff --git a/drivers/misc/mei/client.c b/drivers/misc/mei/client.c index ebdcf0b450e2..1fc8ea0f519b 100644 --- a/drivers/misc/mei/client.c +++ b/drivers/misc/mei/client.c @@ -318,23 +318,6 @@ void mei_me_cl_rm_all(struct mei_device *dev)  }  /** - * mei_cl_cmp_id - tells if the clients are the same - * - * @cl1: host client 1 - * @cl2: host client 2 - * - * Return: true  - if the clients has same host and me ids - *         false - otherwise - */ -static inline bool mei_cl_cmp_id(const struct mei_cl *cl1, -				const struct mei_cl *cl2) -{ -	return cl1 && cl2 && -		(cl1->host_client_id == cl2->host_client_id) && -		(mei_cl_me_id(cl1) == mei_cl_me_id(cl2)); -} - -/**   * mei_io_cb_free - free mei_cb_private related memory   *   * @cb: mei callback struct @@ -418,7 +401,7 @@ static void mei_io_list_flush_cl(struct list_head *head,  	struct mei_cl_cb *cb, *next;  	list_for_each_entry_safe(cb, next, head, list) { -		if (mei_cl_cmp_id(cl, cb->cl)) +		if (cl == cb->cl)  			list_del_init(&cb->list);  	}  } @@ -435,7 +418,7 @@ static void mei_io_tx_list_free_cl(struct list_head *head,  	struct mei_cl_cb *cb, *next;  	list_for_each_entry_safe(cb, next, head, list) { -		if (mei_cl_cmp_id(cl, cb->cl)) +		if (cl == cb->cl)  			mei_tx_cb_dequeue(cb);  	}  } @@ -478,7 +461,7 @@ struct mei_cl_cb *mei_cl_alloc_cb(struct mei_cl *cl, size_t length,  	if (length == 0)  		return cb; -	cb->buf.data = kmalloc(length, GFP_KERNEL); +	cb->buf.data = kmalloc(roundup(length, MEI_SLOT_SIZE), GFP_KERNEL);  	if (!cb->buf.data) {  		mei_io_cb_free(cb);  		return NULL; @@ -1374,7 +1357,9 @@ int mei_cl_notify_request(struct mei_cl *cl,  	mutex_unlock(&dev->device_lock);  	wait_event_timeout(cl->wait, -			   cl->notify_en == request || !mei_cl_is_connected(cl), +			   cl->notify_en == request || +			   cl->status || +			   !mei_cl_is_connected(cl),  			   mei_secs_to_jiffies(MEI_CL_CONNECT_TIMEOUT));  	mutex_lock(&dev->device_lock); @@ -1573,10 +1558,13 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	struct mei_msg_hdr mei_hdr;  	size_t hdr_len = sizeof(mei_hdr);  	size_t len; -	size_t hbuf_len; +	size_t hbuf_len, dr_len;  	int hbuf_slots; +	u32 dr_slots; +	u32 dma_len;  	int rets;  	bool first_chunk; +	const void *data;  	if (WARN_ON(!cl || !cl->dev))  		return -ENODEV; @@ -1597,6 +1585,7 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	}  	len = buf->size - cb->buf_idx; +	data = buf->data + cb->buf_idx;  	hbuf_slots = mei_hbuf_empty_slots(dev);  	if (hbuf_slots < 0) {  		rets = -EOVERFLOW; @@ -1604,6 +1593,8 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	}  	hbuf_len = mei_slots2data(hbuf_slots); +	dr_slots = mei_dma_ring_empty_slots(dev); +	dr_len = mei_slots2data(dr_slots);  	mei_msg_hdr_init(&mei_hdr, cb); @@ -1614,23 +1605,33 @@ int mei_cl_irq_write(struct mei_cl *cl, struct mei_cl_cb *cb,  	if (len + hdr_len <= hbuf_len) {  		mei_hdr.length = len;  		mei_hdr.msg_complete = 1; +	} else if (dr_slots && hbuf_len >= hdr_len + sizeof(dma_len)) { +		mei_hdr.dma_ring = 1; +		if (len > dr_len) +			len = dr_len; +		else +			mei_hdr.msg_complete = 1; + +		mei_hdr.length = sizeof(dma_len); +		dma_len = len; +		data = &dma_len;  	} else if ((u32)hbuf_slots == mei_hbuf_depth(dev)) { -		mei_hdr.length = hbuf_len - hdr_len; +		len = hbuf_len - hdr_len; +		mei_hdr.length = len;  	} else {  		return 0;  	} -	cl_dbg(dev, cl, "buf: size = %zu idx = %zu\n", -			cb->buf.size, cb->buf_idx); +	if (mei_hdr.dma_ring) +		mei_dma_ring_write(dev, buf->data + cb->buf_idx, len); -	rets = mei_write_message(dev, &mei_hdr, hdr_len, -				 buf->data + cb->buf_idx, mei_hdr.length); +	rets = mei_write_message(dev, &mei_hdr, hdr_len, data, mei_hdr.length);  	if (rets)  		goto err;  	cl->status = 0;  	cl->writing_state = MEI_WRITING; -	cb->buf_idx += mei_hdr.length; +	cb->buf_idx += len;  	if (first_chunk) {  		if (mei_cl_tx_flow_ctrl_creds_reduce(cl)) { @@ -1665,11 +1666,13 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  	struct mei_msg_data *buf;  	struct mei_msg_hdr mei_hdr;  	size_t hdr_len = sizeof(mei_hdr); -	size_t len; -	size_t hbuf_len; +	size_t len, hbuf_len, dr_len;  	int hbuf_slots; +	u32 dr_slots; +	u32 dma_len;  	ssize_t rets;  	bool blocking; +	const void *data;  	if (WARN_ON(!cl || !cl->dev))  		return -ENODEV; @@ -1681,10 +1684,12 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  	buf = &cb->buf;  	len = buf->size; -	blocking = cb->blocking;  	cl_dbg(dev, cl, "len=%zd\n", len); +	blocking = cb->blocking; +	data = buf->data; +  	rets = pm_runtime_get(dev->dev);  	if (rets < 0 && rets != -EINPROGRESS) {  		pm_runtime_put_noidle(dev->dev); @@ -1721,16 +1726,32 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  	}  	hbuf_len = mei_slots2data(hbuf_slots); +	dr_slots = mei_dma_ring_empty_slots(dev); +	dr_len =  mei_slots2data(dr_slots);  	if (len + hdr_len <= hbuf_len) {  		mei_hdr.length = len;  		mei_hdr.msg_complete = 1; +	} else if (dr_slots && hbuf_len >= hdr_len + sizeof(dma_len)) { +		mei_hdr.dma_ring = 1; +		if (len > dr_len) +			len = dr_len; +		else +			mei_hdr.msg_complete = 1; + +		mei_hdr.length = sizeof(dma_len); +		dma_len = len; +		data = &dma_len;  	} else { -		mei_hdr.length = hbuf_len - hdr_len; +		len = hbuf_len - hdr_len; +		mei_hdr.length = len;  	} +	if (mei_hdr.dma_ring) +		mei_dma_ring_write(dev, buf->data, len); +  	rets = mei_write_message(dev, &mei_hdr, hdr_len, -				 buf->data, mei_hdr.length); +				 data, mei_hdr.length);  	if (rets)  		goto err; @@ -1739,7 +1760,9 @@ ssize_t mei_cl_write(struct mei_cl *cl, struct mei_cl_cb *cb)  		goto err;  	cl->writing_state = MEI_WRITING; -	cb->buf_idx = mei_hdr.length; +	cb->buf_idx = len; +	/* restore return value */ +	len = buf->size;  out:  	if (mei_hdr.msg_complete) | 
