| // 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. All rights reserved. |
| // |
| // 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: |
| ops = &ipc3_ops; |
| break; |
| #endif |
| #if defined(CONFIG_SND_SOC_SOF_INTEL_IPC4) |
| case SOF_INTEL_IPC4: |
| 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); |