236 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			236 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
 | |
| //
 | |
| // This file is provided under a dual BSD/GPLv2 license.  When using or
 | |
| // redistributing this file, you may do so under either license.
 | |
| //
 | |
| // Copyright(c) 2018 Intel Corporation
 | |
| //
 | |
| // Author: Liam Girdwood <liam.r.girdwood@linux.intel.com>
 | |
| //
 | |
| // Generic IPC layer that can work over MMIO and SPI/I2C. PHY layer provided
 | |
| // by platform driver code.
 | |
| //
 | |
| 
 | |
| #include <linux/mutex.h>
 | |
| #include <linux/types.h>
 | |
| 
 | |
| #include "sof-priv.h"
 | |
| #include "sof-audio.h"
 | |
| #include "ops.h"
 | |
| 
 | |
| /**
 | |
|  * sof_ipc_send_msg - generic function to prepare and send one IPC message
 | |
|  * @sdev:		pointer to SOF core device struct
 | |
|  * @msg_data:		pointer to a message to send
 | |
|  * @msg_bytes:		number of bytes in the message
 | |
|  * @reply_bytes:	number of bytes available for the reply.
 | |
|  *			The buffer for the reply data is not passed to this
 | |
|  *			function, the available size is an information for the
 | |
|  *			reply handling functions.
 | |
|  *
 | |
|  * On success the function returns 0, otherwise negative error number.
 | |
|  *
 | |
|  * Note: higher level sdev->ipc->tx_mutex must be held to make sure that
 | |
|  *	 transfers are synchronized.
 | |
|  */
 | |
| int sof_ipc_send_msg(struct snd_sof_dev *sdev, void *msg_data, size_t msg_bytes,
 | |
| 		     size_t reply_bytes)
 | |
| {
 | |
| 	struct snd_sof_ipc *ipc = sdev->ipc;
 | |
| 	struct snd_sof_ipc_msg *msg;
 | |
| 	int ret;
 | |
| 
 | |
| 	if (ipc->disable_ipc_tx || sdev->fw_state != SOF_FW_BOOT_COMPLETE)
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	/*
 | |
| 	 * The spin-lock is needed to protect message objects against other
 | |
| 	 * atomic contexts.
 | |
| 	 */
 | |
| 	spin_lock_irq(&sdev->ipc_lock);
 | |
| 
 | |
| 	/* initialise the message */
 | |
| 	msg = &ipc->msg;
 | |
| 
 | |
| 	/* attach message data */
 | |
| 	msg->msg_data = msg_data;
 | |
| 	msg->msg_size = msg_bytes;
 | |
| 
 | |
| 	msg->reply_size = reply_bytes;
 | |
| 	msg->reply_error = 0;
 | |
| 
 | |
| 	sdev->msg = msg;
 | |
| 
 | |
| 	ret = snd_sof_dsp_send_msg(sdev, msg);
 | |
| 	/* Next reply that we receive will be related to this message */
 | |
| 	if (!ret)
 | |
| 		msg->ipc_complete = false;
 | |
| 
 | |
| 	spin_unlock_irq(&sdev->ipc_lock);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| /* send IPC message from host to DSP */
 | |
| int sof_ipc_tx_message(struct snd_sof_ipc *ipc, void *msg_data, size_t msg_bytes,
 | |
| 		       void *reply_data, size_t reply_bytes)
 | |
| {
 | |
| 	if (msg_bytes > ipc->max_payload_size ||
 | |
| 	    reply_bytes > ipc->max_payload_size)
 | |
| 		return -ENOBUFS;
 | |
| 
 | |
| 	return ipc->ops->tx_msg(ipc->sdev, msg_data, msg_bytes, reply_data,
 | |
| 				reply_bytes, false);
 | |
| }
 | |
| EXPORT_SYMBOL(sof_ipc_tx_message);
 | |
| 
 | |
| /* IPC set or get data from host to DSP */
 | |
| int sof_ipc_set_get_data(struct snd_sof_ipc *ipc, void *msg_data,
 | |
| 			 size_t msg_bytes, bool set)
 | |
| {
 | |
| 	return ipc->ops->set_get_data(ipc->sdev, msg_data, msg_bytes, set);
 | |
| }
 | |
| EXPORT_SYMBOL(sof_ipc_set_get_data);
 | |
| 
 | |
| /*
 | |
|  * send IPC message from host to DSP without modifying the DSP state.
 | |
|  * This will be used for IPC's that can be handled by the DSP
 | |
|  * even in a low-power D0 substate.
 | |
|  */
 | |
| int sof_ipc_tx_message_no_pm(struct snd_sof_ipc *ipc, void *msg_data, size_t msg_bytes,
 | |
| 			     void *reply_data, size_t reply_bytes)
 | |
| {
 | |
| 	if (msg_bytes > ipc->max_payload_size ||
 | |
| 	    reply_bytes > ipc->max_payload_size)
 | |
| 		return -ENOBUFS;
 | |
| 
 | |
| 	return ipc->ops->tx_msg(ipc->sdev, msg_data, msg_bytes, reply_data,
 | |
| 				reply_bytes, true);
 | |
| }
 | |
| EXPORT_SYMBOL(sof_ipc_tx_message_no_pm);
 | |
| 
 | |
| /* Generic helper function to retrieve the reply */
 | |
| void snd_sof_ipc_get_reply(struct snd_sof_dev *sdev)
 | |
| {
 | |
| 	/*
 | |
| 	 * Sometimes, there is unexpected reply ipc arriving. The reply
 | |
| 	 * ipc belongs to none of the ipcs sent from driver.
 | |
| 	 * In this case, the driver must ignore the ipc.
 | |
| 	 */
 | |
| 	if (!sdev->msg) {
 | |
| 		dev_warn(sdev->dev, "unexpected ipc interrupt raised!\n");
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	sdev->msg->reply_error = sdev->ipc->ops->get_reply(sdev);
 | |
| }
 | |
| EXPORT_SYMBOL(snd_sof_ipc_get_reply);
 | |
| 
 | |
| /* handle reply message from DSP */
 | |
| void snd_sof_ipc_reply(struct snd_sof_dev *sdev, u32 msg_id)
 | |
| {
 | |
| 	struct snd_sof_ipc_msg *msg = &sdev->ipc->msg;
 | |
| 
 | |
| 	if (msg->ipc_complete) {
 | |
| 		dev_dbg(sdev->dev,
 | |
| 			"no reply expected, received 0x%x, will be ignored",
 | |
| 			msg_id);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	/* wake up and return the error if we have waiters on this message ? */
 | |
| 	msg->ipc_complete = true;
 | |
| 	wake_up(&msg->waitq);
 | |
| }
 | |
| EXPORT_SYMBOL(snd_sof_ipc_reply);
 | |
| 
 | |
| struct snd_sof_ipc *snd_sof_ipc_init(struct snd_sof_dev *sdev)
 | |
| {
 | |
| 	struct snd_sof_ipc *ipc;
 | |
| 	struct snd_sof_ipc_msg *msg;
 | |
| 	const struct sof_ipc_ops *ops;
 | |
| 
 | |
| 	ipc = devm_kzalloc(sdev->dev, sizeof(*ipc), GFP_KERNEL);
 | |
| 	if (!ipc)
 | |
| 		return NULL;
 | |
| 
 | |
| 	mutex_init(&ipc->tx_mutex);
 | |
| 	ipc->sdev = sdev;
 | |
| 	msg = &ipc->msg;
 | |
| 
 | |
| 	/* indicate that we aren't sending a message ATM */
 | |
| 	msg->ipc_complete = true;
 | |
| 
 | |
| 	init_waitqueue_head(&msg->waitq);
 | |
| 
 | |
| 	switch (sdev->pdata->ipc_type) {
 | |
| #if defined(CONFIG_SND_SOC_SOF_IPC3)
 | |
| 	case SOF_IPC_TYPE_3:
 | |
| 		ops = &ipc3_ops;
 | |
| 		break;
 | |
| #endif
 | |
| #if defined(CONFIG_SND_SOC_SOF_IPC4)
 | |
| 	case SOF_IPC_TYPE_4:
 | |
| 		ops = &ipc4_ops;
 | |
| 		break;
 | |
| #endif
 | |
| 	default:
 | |
| 		dev_err(sdev->dev, "Not supported IPC version: %d\n",
 | |
| 			sdev->pdata->ipc_type);
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	/* check for mandatory ops */
 | |
| 	if (!ops->tx_msg || !ops->rx_msg || !ops->set_get_data || !ops->get_reply) {
 | |
| 		dev_err(sdev->dev, "Missing IPC message handling ops\n");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (!ops->fw_loader || !ops->fw_loader->validate ||
 | |
| 	    !ops->fw_loader->parse_ext_manifest) {
 | |
| 		dev_err(sdev->dev, "Missing IPC firmware loading ops\n");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (!ops->pcm) {
 | |
| 		dev_err(sdev->dev, "Missing IPC PCM ops\n");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (!ops->tplg || !ops->tplg->widget || !ops->tplg->control) {
 | |
| 		dev_err(sdev->dev, "Missing IPC topology ops\n");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (ops->fw_tracing && (!ops->fw_tracing->init || !ops->fw_tracing->suspend ||
 | |
| 				!ops->fw_tracing->resume)) {
 | |
| 		dev_err(sdev->dev, "Missing firmware tracing ops\n");
 | |
| 		return NULL;
 | |
| 	}
 | |
| 
 | |
| 	if (ops->init && ops->init(sdev))
 | |
| 		return NULL;
 | |
| 
 | |
| 	ipc->ops = ops;
 | |
| 
 | |
| 	return ipc;
 | |
| }
 | |
| EXPORT_SYMBOL(snd_sof_ipc_init);
 | |
| 
 | |
| void snd_sof_ipc_free(struct snd_sof_dev *sdev)
 | |
| {
 | |
| 	struct snd_sof_ipc *ipc = sdev->ipc;
 | |
| 
 | |
| 	if (!ipc)
 | |
| 		return;
 | |
| 
 | |
| 	/* disable sending of ipc's */
 | |
| 	mutex_lock(&ipc->tx_mutex);
 | |
| 	ipc->disable_ipc_tx = true;
 | |
| 	mutex_unlock(&ipc->tx_mutex);
 | |
| 
 | |
| 	if (ipc->ops->exit)
 | |
| 		ipc->ops->exit(sdev);
 | |
| }
 | |
| EXPORT_SYMBOL(snd_sof_ipc_free);
 |