acp-loader.c 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. // SPDX-License-Identifier: (GPL-2.0-only OR BSD-3-Clause)
  2. //
  3. // This file is provided under a dual BSD/GPLv2 license. When using or
  4. // redistributing this file, you may do so under either license.
  5. //
  6. // Copyright(c) 2021 Advanced Micro Devices, Inc.
  7. //
  8. // Authors: Ajit Kumar Pandey <[email protected]>
  9. /*
  10. * Hardware interface for ACP DSP Firmware binaries loader
  11. */
  12. #include <linux/firmware.h>
  13. #include <linux/module.h>
  14. #include <linux/pci.h>
  15. #include "../ops.h"
  16. #include "acp-dsp-offset.h"
  17. #include "acp.h"
  18. #define FW_BIN 0
  19. #define FW_DATA_BIN 1
  20. #define FW_BIN_PTE_OFFSET 0x00
  21. #define FW_DATA_BIN_PTE_OFFSET 0x08
  22. #define ACP_DSP_RUN 0x00
  23. int acp_dsp_block_read(struct snd_sof_dev *sdev, enum snd_sof_fw_blk_type blk_type,
  24. u32 offset, void *dest, size_t size)
  25. {
  26. const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
  27. switch (blk_type) {
  28. case SOF_FW_BLK_TYPE_SRAM:
  29. offset = offset - desc->sram_pte_offset;
  30. memcpy_from_scratch(sdev, offset, dest, size);
  31. break;
  32. default:
  33. dev_err(sdev->dev, "bad blk type 0x%x\n", blk_type);
  34. return -EINVAL;
  35. }
  36. return 0;
  37. }
  38. EXPORT_SYMBOL_NS(acp_dsp_block_read, SND_SOC_SOF_AMD_COMMON);
  39. int acp_dsp_block_write(struct snd_sof_dev *sdev, enum snd_sof_fw_blk_type blk_type,
  40. u32 offset, void *src, size_t size)
  41. {
  42. struct snd_sof_pdata *plat_data = sdev->pdata;
  43. struct pci_dev *pci = to_pci_dev(sdev->dev);
  44. const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
  45. struct acp_dev_data *adata;
  46. void *dest;
  47. u32 dma_size, page_count;
  48. unsigned int size_fw;
  49. adata = sdev->pdata->hw_pdata;
  50. switch (blk_type) {
  51. case SOF_FW_BLK_TYPE_IRAM:
  52. if (!adata->bin_buf) {
  53. size_fw = plat_data->fw->size;
  54. page_count = PAGE_ALIGN(size_fw) >> PAGE_SHIFT;
  55. dma_size = page_count * ACP_PAGE_SIZE;
  56. adata->bin_buf = dma_alloc_coherent(&pci->dev, dma_size,
  57. &adata->sha_dma_addr,
  58. GFP_ATOMIC);
  59. if (!adata->bin_buf)
  60. return -ENOMEM;
  61. }
  62. adata->fw_bin_size = size + offset;
  63. dest = adata->bin_buf + offset;
  64. break;
  65. case SOF_FW_BLK_TYPE_DRAM:
  66. if (!adata->data_buf) {
  67. adata->data_buf = dma_alloc_coherent(&pci->dev,
  68. ACP_DEFAULT_DRAM_LENGTH,
  69. &adata->dma_addr,
  70. GFP_ATOMIC);
  71. if (!adata->data_buf)
  72. return -ENOMEM;
  73. }
  74. dest = adata->data_buf + offset;
  75. adata->fw_data_bin_size = size + offset;
  76. break;
  77. case SOF_FW_BLK_TYPE_SRAM:
  78. offset = offset - desc->sram_pte_offset;
  79. memcpy_to_scratch(sdev, offset, src, size);
  80. return 0;
  81. default:
  82. dev_err(sdev->dev, "bad blk type 0x%x\n", blk_type);
  83. return -EINVAL;
  84. }
  85. memcpy(dest, src, size);
  86. return 0;
  87. }
  88. EXPORT_SYMBOL_NS(acp_dsp_block_write, SND_SOC_SOF_AMD_COMMON);
  89. int acp_get_bar_index(struct snd_sof_dev *sdev, u32 type)
  90. {
  91. return type;
  92. }
  93. EXPORT_SYMBOL_NS(acp_get_bar_index, SND_SOC_SOF_AMD_COMMON);
  94. static void configure_pte_for_fw_loading(int type, int num_pages, struct acp_dev_data *adata)
  95. {
  96. struct snd_sof_dev *sdev = adata->dev;
  97. const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
  98. unsigned int low, high;
  99. dma_addr_t addr;
  100. u16 page_idx;
  101. u32 offset;
  102. switch (type) {
  103. case FW_BIN:
  104. offset = FW_BIN_PTE_OFFSET;
  105. addr = adata->sha_dma_addr;
  106. break;
  107. case FW_DATA_BIN:
  108. offset = adata->fw_bin_page_count * 8;
  109. addr = adata->dma_addr;
  110. break;
  111. default:
  112. dev_err(sdev->dev, "Invalid data type %x\n", type);
  113. return;
  114. }
  115. /* Group Enable */
  116. snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACPAXI2AXI_ATU_BASE_ADDR_GRP_1,
  117. desc->sram_pte_offset | BIT(31));
  118. snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACPAXI2AXI_ATU_PAGE_SIZE_GRP_1,
  119. PAGE_SIZE_4K_ENABLE);
  120. for (page_idx = 0; page_idx < num_pages; page_idx++) {
  121. low = lower_32_bits(addr);
  122. high = upper_32_bits(addr);
  123. snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_SCRATCH_REG_0 + offset, low);
  124. high |= BIT(31);
  125. snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_SCRATCH_REG_0 + offset + 4, high);
  126. offset += 8;
  127. addr += PAGE_SIZE;
  128. }
  129. /* Flush ATU Cache after PTE Update */
  130. snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACPAXI2AXI_ATU_CTRL, ACP_ATU_CACHE_INVALID);
  131. }
  132. /* pre fw run operations */
  133. int acp_dsp_pre_fw_run(struct snd_sof_dev *sdev)
  134. {
  135. struct pci_dev *pci = to_pci_dev(sdev->dev);
  136. struct snd_sof_pdata *plat_data = sdev->pdata;
  137. struct acp_dev_data *adata;
  138. unsigned int src_addr, size_fw;
  139. u32 page_count, dma_size;
  140. int ret;
  141. adata = sdev->pdata->hw_pdata;
  142. size_fw = adata->fw_bin_size;
  143. page_count = PAGE_ALIGN(size_fw) >> PAGE_SHIFT;
  144. adata->fw_bin_page_count = page_count;
  145. configure_pte_for_fw_loading(FW_BIN, page_count, adata);
  146. ret = configure_and_run_sha_dma(adata, adata->bin_buf, ACP_SYSTEM_MEMORY_WINDOW,
  147. ACP_IRAM_BASE_ADDRESS, size_fw);
  148. if (ret < 0) {
  149. dev_err(sdev->dev, "SHA DMA transfer failed status: %d\n", ret);
  150. return ret;
  151. }
  152. configure_pte_for_fw_loading(FW_DATA_BIN, ACP_DRAM_PAGE_COUNT, adata);
  153. src_addr = ACP_SYSTEM_MEMORY_WINDOW + page_count * ACP_PAGE_SIZE;
  154. ret = configure_and_run_dma(adata, src_addr, ACP_DATA_RAM_BASE_ADDRESS,
  155. adata->fw_data_bin_size);
  156. if (ret < 0) {
  157. dev_err(sdev->dev, "acp dma configuration failed: %d\n", ret);
  158. return ret;
  159. }
  160. ret = acp_dma_status(adata, 0);
  161. if (ret < 0)
  162. dev_err(sdev->dev, "acp dma transfer status: %d\n", ret);
  163. /* Free memory once DMA is complete */
  164. dma_size = (PAGE_ALIGN(plat_data->fw->size) >> PAGE_SHIFT) * ACP_PAGE_SIZE;
  165. dma_free_coherent(&pci->dev, dma_size, adata->bin_buf, adata->sha_dma_addr);
  166. dma_free_coherent(&pci->dev, ACP_DEFAULT_DRAM_LENGTH, adata->data_buf, adata->dma_addr);
  167. adata->bin_buf = NULL;
  168. adata->data_buf = NULL;
  169. return ret;
  170. }
  171. EXPORT_SYMBOL_NS(acp_dsp_pre_fw_run, SND_SOC_SOF_AMD_COMMON);
  172. int acp_sof_dsp_run(struct snd_sof_dev *sdev)
  173. {
  174. const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
  175. int val;
  176. snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_DSP0_RUNSTALL, ACP_DSP_RUN);
  177. val = snd_sof_dsp_read(sdev, ACP_DSP_BAR, ACP_DSP0_RUNSTALL);
  178. dev_dbg(sdev->dev, "ACP_DSP0_RUNSTALL : 0x%0x\n", val);
  179. /* Some platforms won't support fusion DSP,keep offset zero for no support */
  180. if (desc->fusion_dsp_offset) {
  181. snd_sof_dsp_write(sdev, ACP_DSP_BAR, desc->fusion_dsp_offset, ACP_DSP_RUN);
  182. val = snd_sof_dsp_read(sdev, ACP_DSP_BAR, desc->fusion_dsp_offset);
  183. dev_dbg(sdev->dev, "ACP_DSP0_FUSION_RUNSTALL : 0x%0x\n", val);
  184. }
  185. return 0;
  186. }
  187. EXPORT_SYMBOL_NS(acp_sof_dsp_run, SND_SOC_SOF_AMD_COMMON);