Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 | // 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_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); |