[PATCH 05/15] drm/dsi: Implement generic read and write commands
Andrzej Hajda
a.hajda at samsung.com
Tue Oct 14 01:59:26 PDT 2014
On 10/13/2014 12:16 PM, Thierry Reding wrote:
> From: Thierry Reding <treding at nvidia.com>
>
> Implement generic read and write commands. Selection of the proper data
> type for packets is done automatically based on the number of parameters
> or payload length.
>
> Signed-off-by: Thierry Reding <treding at nvidia.com>
> ---
> drivers/gpu/drm/drm_mipi_dsi.c | 115 +++++++++++++++++++++++++++++++++++++++++
> include/drm/drm_mipi_dsi.h | 6 +++
> 2 files changed, 121 insertions(+)
>
> diff --git a/drivers/gpu/drm/drm_mipi_dsi.c b/drivers/gpu/drm/drm_mipi_dsi.c
> index 27fc6dac5e4a..7cd69273dbad 100644
> --- a/drivers/gpu/drm/drm_mipi_dsi.c
> +++ b/drivers/gpu/drm/drm_mipi_dsi.c
> @@ -229,6 +229,121 @@ int mipi_dsi_set_maximum_return_packet_size(struct mipi_dsi_device *dsi,
> EXPORT_SYMBOL(mipi_dsi_set_maximum_return_packet_size);
>
> /**
> + * mipi_dsi_generic_write() - transmit data using a generic write packet
> + * @dsi: DSI peripheral device
> + * @payload: buffer containing the payload
> + * @size: size of payload buffer
> + *
> + * This function will automatically choose the right data type depending on
> + * the payload length.
> + *
> + * Return: The number of bytes transmitted on success or a negative error code
> + * on failure.
> + */
> +ssize_t mipi_dsi_generic_write(struct mipi_dsi_device *dsi, const void *payload,
> + size_t size)
> +{
> + struct mipi_dsi_msg msg;
> + ssize_t err;
> + u8 *tx;
> +
> + memset(&msg, 0, sizeof(msg));
> + msg.channel = dsi->channel;
Again, maybe initializer would be better.
> + msg.flags = MIPI_DSI_MSG_USE_LPM | MIPI_DSI_MSG_REQ_ACK;
You should not hardcode flags here, these flags have nothing to do with
dsi generic write. But there is general problem of encoding flags in
helpers. Possible solutions I see:
1. Translate DSI flags to msg flags as in case of MIPI_DSI_MODE_LPM ->
MIPI_DSI_MSG_USE_LPM.
2. Copy dsi.mode_flags to msg.flags, or to dedicated field of msg.
3. Call transfer callback with dsi pointer instead of host, this
way it will have access to mode_flags.
> +
> + if (size > 2) {
> + tx = kmalloc(2 + size, GFP_KERNEL);
> + if (!tx)
> + return -ENOMEM;
> +
> + tx[0] = (size >> 0) & 0xff;
> + tx[1] = (size >> 8) & 0xff;
> +
> + memcpy(tx + 2, payload, size);
> +
> + msg.tx_len = 2 + size;
> + msg.tx_buf = tx;
> + } else {
> + msg.tx_buf = payload;
> + msg.tx_len = size;
> + }
Another API breakage, commented already in response to 1st patch.
> +
> + switch (size) {
> + case 0:
> + msg.type = MIPI_DSI_GENERIC_SHORT_WRITE_0_PARAM;
> + break;
> +
> + case 1:
> + msg.type = MIPI_DSI_GENERIC_SHORT_WRITE_1_PARAM;
> + break;
> +
> + case 2:
> + msg.type = MIPI_DSI_GENERIC_SHORT_WRITE_2_PARAM;
> + break;
> +
> + default:
> + msg.type = MIPI_DSI_GENERIC_LONG_WRITE;
> + break;
> + }
> +
> + err = dsi->host->ops->transfer(dsi->host, &msg);
> +
> + if (size > 2)
> + kfree(tx);
> +
> + return err;
> +}
> +EXPORT_SYMBOL(mipi_dsi_generic_write);
> +
> +/**
> + * mipi_dsi_generic_read() - receive data using a generic read packet
> + * @dsi: DSI peripheral device
> + * @params: buffer containing the request parameters
> + * @num_params: number of request parameters
> + * @data: buffer in which to return the received data
> + * @size: size of receive buffer
> + *
> + * This function will automatically choose the right data type depending on
> + * the number of parameters passed in.
> + *
> + * Return: The number of bytes successfully read or a negative error code on
> + * failure.
> + */
> +ssize_t mipi_dsi_generic_read(struct mipi_dsi_device *dsi, const void *params,
> + size_t num_params, void *data, size_t size)
> +{
> + struct mipi_dsi_msg msg;
> +
> + memset(&msg, 0, sizeof(msg));
> + msg.channel = dsi->channel;
> + msg.flags = MIPI_DSI_MSG_USE_LPM;
> + msg.tx_len = num_params;
> + msg.tx_buf = params;
> + msg.rx_len = size;
> + msg.rx_buf = data;
> +
> + switch (num_params) {
> + case 0:
> + msg.type = MIPI_DSI_GENERIC_READ_REQUEST_0_PARAM;
> + break;
> +
> + case 1:
> + msg.type = MIPI_DSI_GENERIC_READ_REQUEST_1_PARAM;
> + break;
> +
> + case 2:
> + msg.type = MIPI_DSI_GENERIC_READ_REQUEST_2_PARAM;
> + break;
> +
> + default:
> + return -EINVAL;
> + }
> +
> + return dsi->host->ops->transfer(dsi->host, &msg);
> +}
> +EXPORT_SYMBOL(mipi_dsi_generic_read);
> +
> +/**
> * mipi_dsi_dcs_write_buffer() - transmit a DCS command with payload
> * @dsi: DSI peripheral device
> * @data: buffer containing data to be transmitted
> diff --git a/include/drm/drm_mipi_dsi.h b/include/drm/drm_mipi_dsi.h
> index ef50b5d0de57..92ca66306702 100644
> --- a/include/drm/drm_mipi_dsi.h
> +++ b/include/drm/drm_mipi_dsi.h
> @@ -134,6 +134,12 @@ int mipi_dsi_attach(struct mipi_dsi_device *dsi);
> int mipi_dsi_detach(struct mipi_dsi_device *dsi);
> int mipi_dsi_set_maximum_return_packet_size(struct mipi_dsi_device *dsi,
> u16 value);
> +
> +ssize_t mipi_dsi_generic_write(struct mipi_dsi_device *dsi, const void *payload,
> + size_t size);
> +ssize_t mipi_dsi_generic_read(struct mipi_dsi_device *dsi, const void *params,
> + size_t num_params, void *data, size_t size);
> +
> ssize_t mipi_dsi_dcs_write_buffer(struct mipi_dsi_device *dsi,
> const void *data, size_t len);
> ssize_t mipi_dsi_dcs_write(struct mipi_dsi_device *dsi, u8 cmd,
>
More information about the dri-devel
mailing list