| /* |
| * sst_dsp.c - Intel SST Driver for audio engine |
| * |
| * Copyright (C) 2008-14 Intel Corp |
| * Authors: Vinod Koul <vinod.koul@intel.com> |
| * Harsha Priya <priya.harsha@intel.com> |
| * Dharageswari R <dharageswari.r@intel.com> |
| * KP Jeeja <jeeja.kp@intel.com> |
| * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; version 2 of the License. |
| * |
| * This program is distributed in the hope that it will be useful, but |
| * WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * General Public License for more details. |
| * |
| * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |
| * |
| * This file contains all dsp controlling functions like firmware download, |
| * setting/resetting dsp cores, etc |
| */ |
| #include <linux/pci.h> |
| #include <linux/delay.h> |
| #include <linux/fs.h> |
| #include <linux/sched.h> |
| #include <linux/firmware.h> |
| #include <linux/dmaengine.h> |
| #include <linux/pm_runtime.h> |
| #include <linux/pm_qos.h> |
| #include <sound/core.h> |
| #include <sound/pcm.h> |
| #include <sound/soc.h> |
| #include <sound/compress_driver.h> |
| #include <asm/platform_sst_audio.h> |
| #include "../sst-mfld-platform.h" |
| #include "sst.h" |
| #include "../../common/sst-dsp.h" |
| |
| void memcpy32_toio(void __iomem *dst, const void *src, int count) |
| { |
| /* __iowrite32_copy uses 32-bit count values so divide by 4 for |
| * right count in words |
| */ |
| __iowrite32_copy(dst, src, count / 4); |
| } |
| |
| void memcpy32_fromio(void *dst, const void __iomem *src, int count) |
| { |
| /* __ioread32_copy uses 32-bit count values so divide by 4 for |
| * right count in words |
| */ |
| __ioread32_copy(dst, src, count / 4); |
| } |
| |
| /** |
| * intel_sst_reset_dsp_mrfld - Resetting SST DSP |
| * |
| * This resets DSP in case of MRFLD platfroms |
| */ |
| int intel_sst_reset_dsp_mrfld(struct intel_sst_drv *sst_drv_ctx) |
| { |
| union config_status_reg_mrfld csr; |
| |
| dev_dbg(sst_drv_ctx->dev, "sst: Resetting the DSP in mrfld\n"); |
| csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR); |
| |
| dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full); |
| |
| csr.full |= 0x7; |
| sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full); |
| csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR); |
| |
| dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full); |
| |
| csr.full &= ~(0x1); |
| sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full); |
| |
| csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR); |
| dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full); |
| return 0; |
| } |
| |
| /** |
| * sst_start_merrifield - Start the SST DSP processor |
| * |
| * This starts the DSP in MERRIFIELD platfroms |
| */ |
| int sst_start_mrfld(struct intel_sst_drv *sst_drv_ctx) |
| { |
| union config_status_reg_mrfld csr; |
| |
| dev_dbg(sst_drv_ctx->dev, "sst: Starting the DSP in mrfld LALALALA\n"); |
| csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR); |
| dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full); |
| |
| csr.full |= 0x7; |
| sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full); |
| |
| csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR); |
| dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full); |
| |
| csr.part.xt_snoop = 1; |
| csr.full &= ~(0x5); |
| sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full); |
| |
| csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR); |
| dev_dbg(sst_drv_ctx->dev, "sst: Starting the DSP_merrifield:%llx\n", |
| csr.full); |
| return 0; |
| } |
| |
| static int sst_validate_fw_image(struct intel_sst_drv *ctx, unsigned long size, |
| struct fw_module_header **module, u32 *num_modules) |
| { |
| struct sst_fw_header *header; |
| const void *sst_fw_in_mem = ctx->fw_in_mem; |
| |
| dev_dbg(ctx->dev, "Enter\n"); |
| |
| /* Read the header information from the data pointer */ |
| header = (struct sst_fw_header *)sst_fw_in_mem; |
| dev_dbg(ctx->dev, |
| "header sign=%s size=%x modules=%x fmt=%x size=%zx\n", |
| header->signature, header->file_size, header->modules, |
| header->file_format, sizeof(*header)); |
| |
| /* verify FW */ |
| if ((strncmp(header->signature, SST_FW_SIGN, 4) != 0) || |
| (size != header->file_size + sizeof(*header))) { |
| /* Invalid FW signature */ |
| dev_err(ctx->dev, "InvalidFW sign/filesize mismatch\n"); |
| return -EINVAL; |
| } |
| *num_modules = header->modules; |
| *module = (void *)sst_fw_in_mem + sizeof(*header); |
| |
| return 0; |
| } |
| |
| /* |
| * sst_fill_memcpy_list - Fill the memcpy list |
| * |
| * @memcpy_list: List to be filled |
| * @destn: Destination addr to be filled in the list |
| * @src: Source addr to be filled in the list |
| * @size: Size to be filled in the list |
| * |
| * Adds the node to the list after required fields |
| * are populated in the node |
| */ |
| static int sst_fill_memcpy_list(struct list_head *memcpy_list, |
| void *destn, const void *src, u32 size, bool is_io) |
| { |
| struct sst_memcpy_list *listnode; |
| |
| listnode = kzalloc(sizeof(*listnode), GFP_KERNEL); |
| if (listnode == NULL) |
| return -ENOMEM; |
| listnode->dstn = destn; |
| listnode->src = src; |
| listnode->size = size; |
| listnode->is_io = is_io; |
| list_add_tail(&listnode->memcpylist, memcpy_list); |
| |
| return 0; |
| } |
| |
| /** |
| * sst_parse_module_memcpy - Parse audio FW modules and populate the memcpy list |
| * |
| * @sst_drv_ctx : driver context |
| * @module : FW module header |
| * @memcpy_list : Pointer to the list to be populated |
| * Create the memcpy list as the number of block to be copied |
| * returns error or 0 if module sizes are proper |
| */ |
| static int sst_parse_module_memcpy(struct intel_sst_drv *sst_drv_ctx, |
| struct fw_module_header *module, struct list_head *memcpy_list) |
| { |
| struct fw_block_info *block; |
| u32 count; |
| int ret_val = 0; |
| void __iomem *ram_iomem; |
| |
| dev_dbg(sst_drv_ctx->dev, "module sign %s size %x blocks %x type %x\n", |
| module->signature, module->mod_size, |
| module->blocks, module->type); |
| dev_dbg(sst_drv_ctx->dev, "module entrypoint 0x%x\n", module->entry_point); |
| |
| block = (void *)module + sizeof(*module); |
| |
| for (count = 0; count < module->blocks; count++) { |
| if (block->size <= 0) { |
| dev_err(sst_drv_ctx->dev, "block size invalid\n"); |
| return -EINVAL; |
| } |
| switch (block->type) { |
| case SST_IRAM: |
| ram_iomem = sst_drv_ctx->iram; |
| break; |
| case SST_DRAM: |
| ram_iomem = sst_drv_ctx->dram; |
| break; |
| case SST_DDR: |
| ram_iomem = sst_drv_ctx->ddr; |
| break; |
| case SST_CUSTOM_INFO: |
| block = (void *)block + sizeof(*block) + block->size; |
| continue; |
| default: |
| dev_err(sst_drv_ctx->dev, "wrong ram type0x%x in block0x%x\n", |
| block->type, count); |
| return -EINVAL; |
| } |
| |
| ret_val = sst_fill_memcpy_list(memcpy_list, |
| ram_iomem + block->ram_offset, |
| (void *)block + sizeof(*block), block->size, 1); |
| if (ret_val) |
| return ret_val; |
| |
| block = (void *)block + sizeof(*block) + block->size; |
| } |
| return 0; |
| } |
| |
| /** |
| * sst_parse_fw_memcpy - parse the firmware image & populate the list for memcpy |
| * |
| * @ctx : pointer to drv context |
| * @size : size of the firmware |
| * @fw_list : pointer to list_head to be populated |
| * This function parses the FW image and saves the parsed image in the list |
| * for memcpy |
| */ |
| static int sst_parse_fw_memcpy(struct intel_sst_drv *ctx, unsigned long size, |
| struct list_head *fw_list) |
| { |
| struct fw_module_header *module; |
| u32 count, num_modules; |
| int ret_val; |
| |
| ret_val = sst_validate_fw_image(ctx, size, &module, &num_modules); |
| if (ret_val) |
| return ret_val; |
| |
| for (count = 0; count < num_modules; count++) { |
| ret_val = sst_parse_module_memcpy(ctx, module, fw_list); |
| if (ret_val) |
| return ret_val; |
| module = (void *)module + sizeof(*module) + module->mod_size; |
| } |
| |
| return 0; |
| } |
| |
| /** |
| * sst_do_memcpy - function initiates the memcpy |
| * |
| * @memcpy_list: Pter to memcpy list on which the memcpy needs to be initiated |
| * |
| * Triggers the memcpy |
| */ |
| static void sst_do_memcpy(struct list_head *memcpy_list) |
| { |
| struct sst_memcpy_list *listnode; |
| |
| list_for_each_entry(listnode, memcpy_list, memcpylist) { |
| if (listnode->is_io) |
| memcpy32_toio((void __iomem *)listnode->dstn, |
| listnode->src, listnode->size); |
| else |
| memcpy(listnode->dstn, listnode->src, listnode->size); |
| } |
| } |
| |
| void sst_memcpy_free_resources(struct intel_sst_drv *sst_drv_ctx) |
| { |
| struct sst_memcpy_list *listnode, *tmplistnode; |
| |
| /* Free the list */ |
| if (!list_empty(&sst_drv_ctx->memcpy_list)) { |
| list_for_each_entry_safe(listnode, tmplistnode, |
| &sst_drv_ctx->memcpy_list, memcpylist) { |
| list_del(&listnode->memcpylist); |
| kfree(listnode); |
| } |
| } |
| } |
| |
| static int sst_cache_and_parse_fw(struct intel_sst_drv *sst, |
| const struct firmware *fw) |
| { |
| int retval = 0; |
| |
| sst->fw_in_mem = kzalloc(fw->size, GFP_KERNEL); |
| if (!sst->fw_in_mem) { |
| retval = -ENOMEM; |
| goto end_release; |
| } |
| dev_dbg(sst->dev, "copied fw to %p", sst->fw_in_mem); |
| dev_dbg(sst->dev, "phys: %lx", (unsigned long)virt_to_phys(sst->fw_in_mem)); |
| memcpy(sst->fw_in_mem, fw->data, fw->size); |
| retval = sst_parse_fw_memcpy(sst, fw->size, &sst->memcpy_list); |
| if (retval) { |
| dev_err(sst->dev, "Failed to parse fw\n"); |
| kfree(sst->fw_in_mem); |
| sst->fw_in_mem = NULL; |
| } |
| |
| end_release: |
| release_firmware(fw); |
| return retval; |
| |
| } |
| |
| void sst_firmware_load_cb(const struct firmware *fw, void *context) |
| { |
| struct intel_sst_drv *ctx = context; |
| |
| dev_dbg(ctx->dev, "Enter\n"); |
| |
| if (fw == NULL) { |
| dev_err(ctx->dev, "request fw failed\n"); |
| return; |
| } |
| |
| mutex_lock(&ctx->sst_lock); |
| |
| if (ctx->sst_state != SST_RESET || |
| ctx->fw_in_mem != NULL) { |
| release_firmware(fw); |
| mutex_unlock(&ctx->sst_lock); |
| return; |
| } |
| |
| dev_dbg(ctx->dev, "Request Fw completed\n"); |
| sst_cache_and_parse_fw(ctx, fw); |
| mutex_unlock(&ctx->sst_lock); |
| } |
| |
| /* |
| * sst_request_fw - requests audio fw from kernel and saves a copy |
| * |
| * This function requests the SST FW from the kernel, parses it and |
| * saves a copy in the driver context |
| */ |
| static int sst_request_fw(struct intel_sst_drv *sst) |
| { |
| int retval = 0; |
| const struct firmware *fw; |
| |
| retval = request_firmware(&fw, sst->firmware_name, sst->dev); |
| if (retval) { |
| dev_err(sst->dev, "request fw failed %d\n", retval); |
| return retval; |
| } |
| if (fw == NULL) { |
| dev_err(sst->dev, "fw is returning as null\n"); |
| return -EINVAL; |
| } |
| mutex_lock(&sst->sst_lock); |
| retval = sst_cache_and_parse_fw(sst, fw); |
| mutex_unlock(&sst->sst_lock); |
| |
| return retval; |
| } |
| |
| /* |
| * Writing the DDR physical base to DCCM offset |
| * so that FW can use it to setup TLB |
| */ |
| static void sst_dccm_config_write(void __iomem *dram_base, |
| unsigned int ddr_base) |
| { |
| void __iomem *addr; |
| u32 bss_reset = 0; |
| |
| addr = (void __iomem *)(dram_base + MRFLD_FW_DDR_BASE_OFFSET); |
| memcpy32_toio(addr, (void *)&ddr_base, sizeof(u32)); |
| bss_reset |= (1 << MRFLD_FW_BSS_RESET_BIT); |
| addr = (void __iomem *)(dram_base + MRFLD_FW_FEATURE_BASE_OFFSET); |
| memcpy32_toio(addr, &bss_reset, sizeof(u32)); |
| |
| } |
| |
| void sst_post_download_mrfld(struct intel_sst_drv *ctx) |
| { |
| sst_dccm_config_write(ctx->dram, ctx->ddr_base); |
| dev_dbg(ctx->dev, "config written to DCCM\n"); |
| } |
| |
| /** |
| * sst_load_fw - function to load FW into DSP |
| * Transfers the FW to DSP using dma/memcpy |
| */ |
| int sst_load_fw(struct intel_sst_drv *sst_drv_ctx) |
| { |
| int ret_val = 0; |
| struct sst_block *block; |
| |
| dev_dbg(sst_drv_ctx->dev, "sst_load_fw\n"); |
| |
| if (sst_drv_ctx->sst_state != SST_RESET || |
| sst_drv_ctx->sst_state == SST_SHUTDOWN) |
| return -EAGAIN; |
| |
| if (!sst_drv_ctx->fw_in_mem) { |
| dev_dbg(sst_drv_ctx->dev, "sst: FW not in memory retry to download\n"); |
| ret_val = sst_request_fw(sst_drv_ctx); |
| if (ret_val) |
| return ret_val; |
| } |
| |
| block = sst_create_block(sst_drv_ctx, 0, FW_DWNL_ID); |
| if (block == NULL) |
| return -ENOMEM; |
| |
| /* Prevent C-states beyond C6 */ |
| pm_qos_update_request(sst_drv_ctx->qos, 0); |
| |
| sst_drv_ctx->sst_state = SST_FW_LOADING; |
| |
| ret_val = sst_drv_ctx->ops->reset(sst_drv_ctx); |
| if (ret_val) |
| goto restore; |
| |
| sst_do_memcpy(&sst_drv_ctx->memcpy_list); |
| |
| /* Write the DRAM/DCCM config before enabling FW */ |
| if (sst_drv_ctx->ops->post_download) |
| sst_drv_ctx->ops->post_download(sst_drv_ctx); |
| |
| /* bring sst out of reset */ |
| ret_val = sst_drv_ctx->ops->start(sst_drv_ctx); |
| if (ret_val) |
| goto restore; |
| |
| ret_val = sst_wait_timeout(sst_drv_ctx, block); |
| if (ret_val) { |
| dev_err(sst_drv_ctx->dev, "fw download failed %d\n" , ret_val); |
| /* FW download failed due to timeout */ |
| ret_val = -EBUSY; |
| |
| } |
| |
| |
| restore: |
| /* Re-enable Deeper C-states beyond C6 */ |
| pm_qos_update_request(sst_drv_ctx->qos, PM_QOS_DEFAULT_VALUE); |
| sst_free_block(sst_drv_ctx, block); |
| dev_dbg(sst_drv_ctx->dev, "fw load successful!!!\n"); |
| |
| if (sst_drv_ctx->ops->restore_dsp_context) |
| sst_drv_ctx->ops->restore_dsp_context(); |
| sst_drv_ctx->sst_state = SST_FW_RUNNING; |
| return ret_val; |
| } |
| |