root/sound/soc/intel/atom/sst/sst_ipc.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. sst_create_block
  2. sst_wake_up_block
  3. sst_free_block
  4. sst_post_message_mrfld
  5. intel_sst_clear_intr_mrfld
  6. process_fw_init
  7. process_fw_async_msg
  8. sst_process_reply_mrfld

   1 // SPDX-License-Identifier: GPL-2.0-only
   2 /*
   3  *  sst_ipc.c - Intel SST Driver for audio engine
   4  *
   5  *  Copyright (C) 2008-14 Intel Corporation
   6  *  Authors:    Vinod Koul <vinod.koul@intel.com>
   7  *              Harsha Priya <priya.harsha@intel.com>
   8  *              Dharageswari R <dharageswari.r@intel.com>
   9  *              KP Jeeja <jeeja.kp@intel.com>
  10  *  ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  11  *
  12  * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  13  */
  14 #include <linux/pci.h>
  15 #include <linux/firmware.h>
  16 #include <linux/sched.h>
  17 #include <linux/delay.h>
  18 #include <linux/pm_runtime.h>
  19 #include <sound/core.h>
  20 #include <sound/pcm.h>
  21 #include <sound/soc.h>
  22 #include <sound/compress_driver.h>
  23 #include <asm/intel-mid.h>
  24 #include <asm/platform_sst_audio.h>
  25 #include "../sst-mfld-platform.h"
  26 #include "sst.h"
  27 #include "../../common/sst-dsp.h"
  28 
  29 struct sst_block *sst_create_block(struct intel_sst_drv *ctx,
  30                                         u32 msg_id, u32 drv_id)
  31 {
  32         struct sst_block *msg = NULL;
  33 
  34         dev_dbg(ctx->dev, "Enter\n");
  35         msg = kzalloc(sizeof(*msg), GFP_KERNEL);
  36         if (!msg)
  37                 return NULL;
  38         msg->condition = false;
  39         msg->on = true;
  40         msg->msg_id = msg_id;
  41         msg->drv_id = drv_id;
  42         spin_lock_bh(&ctx->block_lock);
  43         list_add_tail(&msg->node, &ctx->block_list);
  44         spin_unlock_bh(&ctx->block_lock);
  45 
  46         return msg;
  47 }
  48 
  49 /*
  50  * while handling the interrupts, we need to check for message status and
  51  * then if we are blocking for a message
  52  *
  53  * here we are unblocking the blocked ones, this is based on id we have
  54  * passed and search that for block threads.
  55  * We will not find block in two cases
  56  *  a) when its small message and block in not there, so silently ignore
  57  *  them
  58  *  b) when we are actually not able to find the block (bug perhaps)
  59  *
  60  *  Since we have bit of small messages we can spam kernel log with err
  61  *  print on above so need to keep as debug prints which should be enabled
  62  *  via dynamic debug while debugging IPC issues
  63  */
  64 int sst_wake_up_block(struct intel_sst_drv *ctx, int result,
  65                 u32 drv_id, u32 ipc, void *data, u32 size)
  66 {
  67         struct sst_block *block = NULL;
  68 
  69         dev_dbg(ctx->dev, "Enter\n");
  70 
  71         spin_lock_bh(&ctx->block_lock);
  72         list_for_each_entry(block, &ctx->block_list, node) {
  73                 dev_dbg(ctx->dev, "Block ipc %d, drv_id %d\n", block->msg_id,
  74                                                         block->drv_id);
  75                 if (block->msg_id == ipc && block->drv_id == drv_id) {
  76                         dev_dbg(ctx->dev, "free up the block\n");
  77                         block->ret_code = result;
  78                         block->data = data;
  79                         block->size = size;
  80                         block->condition = true;
  81                         spin_unlock_bh(&ctx->block_lock);
  82                         wake_up(&ctx->wait_queue);
  83                         return 0;
  84                 }
  85         }
  86         spin_unlock_bh(&ctx->block_lock);
  87         dev_dbg(ctx->dev,
  88                 "Block not found or a response received for a short msg for ipc %d, drv_id %d\n",
  89                 ipc, drv_id);
  90         return -EINVAL;
  91 }
  92 
  93 int sst_free_block(struct intel_sst_drv *ctx, struct sst_block *freed)
  94 {
  95         struct sst_block *block = NULL, *__block;
  96 
  97         dev_dbg(ctx->dev, "Enter\n");
  98         spin_lock_bh(&ctx->block_lock);
  99         list_for_each_entry_safe(block, __block, &ctx->block_list, node) {
 100                 if (block == freed) {
 101                         pr_debug("pvt_id freed --> %d\n", freed->drv_id);
 102                         /* toggle the index position of pvt_id */
 103                         list_del(&freed->node);
 104                         spin_unlock_bh(&ctx->block_lock);
 105                         kfree(freed->data);
 106                         freed->data = NULL;
 107                         kfree(freed);
 108                         return 0;
 109                 }
 110         }
 111         spin_unlock_bh(&ctx->block_lock);
 112         dev_err(ctx->dev, "block is already freed!!!\n");
 113         return -EINVAL;
 114 }
 115 
 116 int sst_post_message_mrfld(struct intel_sst_drv *sst_drv_ctx,
 117                 struct ipc_post *ipc_msg, bool sync)
 118 {
 119         struct ipc_post *msg = ipc_msg;
 120         union ipc_header_mrfld header;
 121         unsigned int loop_count = 0;
 122         int retval = 0;
 123         unsigned long irq_flags;
 124 
 125         dev_dbg(sst_drv_ctx->dev, "Enter: sync: %d\n", sync);
 126         spin_lock_irqsave(&sst_drv_ctx->ipc_spin_lock, irq_flags);
 127         header.full = sst_shim_read64(sst_drv_ctx->shim, SST_IPCX);
 128         if (sync) {
 129                 while (header.p.header_high.part.busy) {
 130                         if (loop_count > 25) {
 131                                 dev_err(sst_drv_ctx->dev,
 132                                         "sst: Busy wait failed, cant send this msg\n");
 133                                 retval = -EBUSY;
 134                                 goto out;
 135                         }
 136                         cpu_relax();
 137                         loop_count++;
 138                         header.full = sst_shim_read64(sst_drv_ctx->shim, SST_IPCX);
 139                 }
 140         } else {
 141                 if (list_empty(&sst_drv_ctx->ipc_dispatch_list)) {
 142                         /* queue is empty, nothing to send */
 143                         spin_unlock_irqrestore(&sst_drv_ctx->ipc_spin_lock, irq_flags);
 144                         dev_dbg(sst_drv_ctx->dev,
 145                                         "Empty msg queue... NO Action\n");
 146                         return 0;
 147                 }
 148 
 149                 if (header.p.header_high.part.busy) {
 150                         spin_unlock_irqrestore(&sst_drv_ctx->ipc_spin_lock, irq_flags);
 151                         dev_dbg(sst_drv_ctx->dev, "Busy not free... post later\n");
 152                         return 0;
 153                 }
 154 
 155                 /* copy msg from list */
 156                 msg = list_entry(sst_drv_ctx->ipc_dispatch_list.next,
 157                                 struct ipc_post, node);
 158                 list_del(&msg->node);
 159         }
 160         dev_dbg(sst_drv_ctx->dev, "sst: Post message: header = %x\n",
 161                                 msg->mrfld_header.p.header_high.full);
 162         dev_dbg(sst_drv_ctx->dev, "sst: size = 0x%x\n",
 163                         msg->mrfld_header.p.header_low_payload);
 164 
 165         if (msg->mrfld_header.p.header_high.part.large)
 166                 memcpy_toio(sst_drv_ctx->mailbox + SST_MAILBOX_SEND,
 167                         msg->mailbox_data,
 168                         msg->mrfld_header.p.header_low_payload);
 169 
 170         sst_shim_write64(sst_drv_ctx->shim, SST_IPCX, msg->mrfld_header.full);
 171 
 172 out:
 173         spin_unlock_irqrestore(&sst_drv_ctx->ipc_spin_lock, irq_flags);
 174         kfree(msg->mailbox_data);
 175         kfree(msg);
 176         return retval;
 177 }
 178 
 179 void intel_sst_clear_intr_mrfld(struct intel_sst_drv *sst_drv_ctx)
 180 {
 181         union interrupt_reg_mrfld isr;
 182         union interrupt_reg_mrfld imr;
 183         union ipc_header_mrfld clear_ipc;
 184         unsigned long irq_flags;
 185 
 186         spin_lock_irqsave(&sst_drv_ctx->ipc_spin_lock, irq_flags);
 187         imr.full = sst_shim_read64(sst_drv_ctx->shim, SST_IMRX);
 188         isr.full = sst_shim_read64(sst_drv_ctx->shim, SST_ISRX);
 189 
 190         /* write 1 to clear*/
 191         isr.part.busy_interrupt = 1;
 192         sst_shim_write64(sst_drv_ctx->shim, SST_ISRX, isr.full);
 193 
 194         /* Set IA done bit */
 195         clear_ipc.full = sst_shim_read64(sst_drv_ctx->shim, SST_IPCD);
 196 
 197         clear_ipc.p.header_high.part.busy = 0;
 198         clear_ipc.p.header_high.part.done = 1;
 199         clear_ipc.p.header_low_payload = IPC_ACK_SUCCESS;
 200         sst_shim_write64(sst_drv_ctx->shim, SST_IPCD, clear_ipc.full);
 201         /* un mask busy interrupt */
 202         imr.part.busy_interrupt = 0;
 203         sst_shim_write64(sst_drv_ctx->shim, SST_IMRX, imr.full);
 204         spin_unlock_irqrestore(&sst_drv_ctx->ipc_spin_lock, irq_flags);
 205 }
 206 
 207 
 208 /*
 209  * process_fw_init - process the FW init msg
 210  *
 211  * @msg: IPC message mailbox data from FW
 212  *
 213  * This function processes the FW init msg from FW
 214  * marks FW state and prints debug info of loaded FW
 215  */
 216 static void process_fw_init(struct intel_sst_drv *sst_drv_ctx,
 217                         void *msg)
 218 {
 219         struct ipc_header_fw_init *init =
 220                 (struct ipc_header_fw_init *)msg;
 221         int retval = 0;
 222 
 223         dev_dbg(sst_drv_ctx->dev, "*** FW Init msg came***\n");
 224         if (init->result) {
 225                 sst_set_fw_state_locked(sst_drv_ctx, SST_RESET);
 226                 dev_err(sst_drv_ctx->dev, "FW Init failed, Error %x\n",
 227                                 init->result);
 228                 retval = init->result;
 229                 goto ret;
 230         }
 231         if (memcmp(&sst_drv_ctx->fw_version, &init->fw_version,
 232                    sizeof(init->fw_version)))
 233                 dev_info(sst_drv_ctx->dev, "FW Version %02x.%02x.%02x.%02x\n",
 234                         init->fw_version.type, init->fw_version.major,
 235                         init->fw_version.minor, init->fw_version.build);
 236         dev_dbg(sst_drv_ctx->dev, "Build date %s Time %s\n",
 237                         init->build_info.date, init->build_info.time);
 238 
 239         /* Save FW version */
 240         sst_drv_ctx->fw_version.type = init->fw_version.type;
 241         sst_drv_ctx->fw_version.major = init->fw_version.major;
 242         sst_drv_ctx->fw_version.minor = init->fw_version.minor;
 243         sst_drv_ctx->fw_version.build = init->fw_version.build;
 244 
 245 ret:
 246         sst_wake_up_block(sst_drv_ctx, retval, FW_DWNL_ID, 0 , NULL, 0);
 247 }
 248 
 249 static void process_fw_async_msg(struct intel_sst_drv *sst_drv_ctx,
 250                         struct ipc_post *msg)
 251 {
 252         u32 msg_id;
 253         int str_id;
 254         u32 data_size, i;
 255         void *data_offset;
 256         struct stream_info *stream;
 257         u32 msg_low, pipe_id;
 258 
 259         msg_low = msg->mrfld_header.p.header_low_payload;
 260         msg_id = ((struct ipc_dsp_hdr *)msg->mailbox_data)->cmd_id;
 261         data_offset = (msg->mailbox_data + sizeof(struct ipc_dsp_hdr));
 262         data_size =  msg_low - (sizeof(struct ipc_dsp_hdr));
 263 
 264         switch (msg_id) {
 265         case IPC_SST_PERIOD_ELAPSED_MRFLD:
 266                 pipe_id = ((struct ipc_dsp_hdr *)msg->mailbox_data)->pipe_id;
 267                 str_id = get_stream_id_mrfld(sst_drv_ctx, pipe_id);
 268                 if (str_id > 0) {
 269                         dev_dbg(sst_drv_ctx->dev,
 270                                 "Period elapsed rcvd for pipe id 0x%x\n",
 271                                 pipe_id);
 272                         stream = &sst_drv_ctx->streams[str_id];
 273                         /* If stream is dropped, skip processing this message*/
 274                         if (stream->status == STREAM_INIT)
 275                                 break;
 276                         if (stream->period_elapsed)
 277                                 stream->period_elapsed(stream->pcm_substream);
 278                         if (stream->compr_cb)
 279                                 stream->compr_cb(stream->compr_cb_param);
 280                 }
 281                 break;
 282 
 283         case IPC_IA_DRAIN_STREAM_MRFLD:
 284                 pipe_id = ((struct ipc_dsp_hdr *)msg->mailbox_data)->pipe_id;
 285                 str_id = get_stream_id_mrfld(sst_drv_ctx, pipe_id);
 286                 if (str_id > 0) {
 287                         stream = &sst_drv_ctx->streams[str_id];
 288                         if (stream->drain_notify)
 289                                 stream->drain_notify(stream->drain_cb_param);
 290                 }
 291                 break;
 292 
 293         case IPC_IA_FW_ASYNC_ERR_MRFLD:
 294                 dev_err(sst_drv_ctx->dev, "FW sent async error msg:\n");
 295                 for (i = 0; i < (data_size/4); i++)
 296                         print_hex_dump(KERN_DEBUG, NULL, DUMP_PREFIX_NONE,
 297                                         16, 4, data_offset, data_size, false);
 298                 break;
 299 
 300         case IPC_IA_FW_INIT_CMPLT_MRFLD:
 301                 process_fw_init(sst_drv_ctx, data_offset);
 302                 break;
 303 
 304         case IPC_IA_BUF_UNDER_RUN_MRFLD:
 305                 pipe_id = ((struct ipc_dsp_hdr *)msg->mailbox_data)->pipe_id;
 306                 str_id = get_stream_id_mrfld(sst_drv_ctx, pipe_id);
 307                 if (str_id > 0)
 308                         dev_err(sst_drv_ctx->dev,
 309                                 "Buffer under-run for pipe:%#x str_id:%d\n",
 310                                 pipe_id, str_id);
 311                 break;
 312 
 313         default:
 314                 dev_err(sst_drv_ctx->dev,
 315                         "Unrecognized async msg from FW msg_id %#x\n", msg_id);
 316         }
 317 }
 318 
 319 void sst_process_reply_mrfld(struct intel_sst_drv *sst_drv_ctx,
 320                 struct ipc_post *msg)
 321 {
 322         unsigned int drv_id;
 323         void *data;
 324         union ipc_header_high msg_high;
 325         u32 msg_low;
 326         struct ipc_dsp_hdr *dsp_hdr;
 327 
 328         msg_high = msg->mrfld_header.p.header_high;
 329         msg_low = msg->mrfld_header.p.header_low_payload;
 330 
 331         dev_dbg(sst_drv_ctx->dev, "IPC process message header %x payload %x\n",
 332                         msg->mrfld_header.p.header_high.full,
 333                         msg->mrfld_header.p.header_low_payload);
 334 
 335         drv_id = msg_high.part.drv_id;
 336 
 337         /* Check for async messages first */
 338         if (drv_id == SST_ASYNC_DRV_ID) {
 339                 /*FW sent async large message*/
 340                 process_fw_async_msg(sst_drv_ctx, msg);
 341                 return;
 342         }
 343 
 344         /* FW sent short error response for an IPC */
 345         if (msg_high.part.result && drv_id && !msg_high.part.large) {
 346                 /* 32-bit FW error code in msg_low */
 347                 dev_err(sst_drv_ctx->dev, "FW sent error response 0x%x", msg_low);
 348                 sst_wake_up_block(sst_drv_ctx, msg_high.part.result,
 349                         msg_high.part.drv_id,
 350                         msg_high.part.msg_id, NULL, 0);
 351                 return;
 352         }
 353 
 354         /*
 355          * Process all valid responses
 356          * if it is a large message, the payload contains the size to
 357          * copy from mailbox
 358          **/
 359         if (msg_high.part.large) {
 360                 data = kmemdup((void *)msg->mailbox_data, msg_low, GFP_KERNEL);
 361                 if (!data)
 362                         return;
 363                 /* Copy command id so that we can use to put sst to reset */
 364                 dsp_hdr = (struct ipc_dsp_hdr *)data;
 365                 dev_dbg(sst_drv_ctx->dev, "cmd_id %d\n", dsp_hdr->cmd_id);
 366                 if (sst_wake_up_block(sst_drv_ctx, msg_high.part.result,
 367                                 msg_high.part.drv_id,
 368                                 msg_high.part.msg_id, data, msg_low))
 369                         kfree(data);
 370         } else {
 371                 sst_wake_up_block(sst_drv_ctx, msg_high.part.result,
 372                                 msg_high.part.drv_id,
 373                                 msg_high.part.msg_id, NULL, 0);
 374         }
 375 
 376 }

/* [<][>][^][v][top][bottom][index][help] */