~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/sound/soc/sof/amd/acp-loader.c

Version: ~ [ linux-6.11.5 ] ~ [ linux-6.10.14 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.58 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.114 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.169 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.228 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.284 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.322 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.9 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  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, 2023 Advanced Micro Devices, Inc.
  7 //
  8 // Authors: Ajit Kumar Pandey <AjitKumar.Pandey@amd.com>
  9 
 10 /*
 11  * Hardware interface for ACP DSP Firmware binaries loader
 12  */
 13 
 14 #include <linux/firmware.h>
 15 #include <linux/module.h>
 16 #include <linux/pci.h>
 17 
 18 #include "../ops.h"
 19 #include "acp-dsp-offset.h"
 20 #include "acp.h"
 21 
 22 #define FW_BIN                  0
 23 #define FW_DATA_BIN             1
 24 #define FW_SRAM_DATA_BIN        2
 25 
 26 #define FW_BIN_PTE_OFFSET       0x00
 27 #define FW_DATA_BIN_PTE_OFFSET  0x08
 28 
 29 #define ACP_DSP_RUN     0x00
 30 
 31 int acp_dsp_block_read(struct snd_sof_dev *sdev, enum snd_sof_fw_blk_type blk_type,
 32                        u32 offset, void *dest, size_t size)
 33 {
 34         const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
 35         switch (blk_type) {
 36         case SOF_FW_BLK_TYPE_SRAM:
 37                 offset = offset - desc->sram_pte_offset;
 38                 memcpy_from_scratch(sdev, offset, dest, size);
 39                 break;
 40         default:
 41                 dev_err(sdev->dev, "bad blk type 0x%x\n", blk_type);
 42                 return -EINVAL;
 43         }
 44 
 45         return 0;
 46 }
 47 EXPORT_SYMBOL_NS(acp_dsp_block_read, SND_SOC_SOF_AMD_COMMON);
 48 
 49 int acp_dsp_block_write(struct snd_sof_dev *sdev, enum snd_sof_fw_blk_type blk_type,
 50                         u32 offset, void *src, size_t size)
 51 {
 52         struct pci_dev *pci = to_pci_dev(sdev->dev);
 53         struct acp_dev_data *adata;
 54         void *dest;
 55         u32 dma_size, page_count;
 56         unsigned int size_fw;
 57 
 58         adata = sdev->pdata->hw_pdata;
 59 
 60         switch (blk_type) {
 61         case SOF_FW_BLK_TYPE_IRAM:
 62                 if (!adata->bin_buf) {
 63                         size_fw = sdev->basefw.fw->size;
 64                         page_count = PAGE_ALIGN(size_fw) >> PAGE_SHIFT;
 65                         dma_size = page_count * ACP_PAGE_SIZE;
 66                         adata->bin_buf = dma_alloc_coherent(&pci->dev, dma_size,
 67                                                             &adata->sha_dma_addr,
 68                                                             GFP_ATOMIC);
 69                         if (!adata->bin_buf)
 70                                 return -ENOMEM;
 71                 }
 72                 adata->fw_bin_size = size + offset;
 73                 dest = adata->bin_buf + offset;
 74                 break;
 75         case SOF_FW_BLK_TYPE_DRAM:
 76                 if (!adata->data_buf) {
 77                         adata->data_buf = dma_alloc_coherent(&pci->dev,
 78                                                              ACP_DEFAULT_DRAM_LENGTH,
 79                                                              &adata->dma_addr,
 80                                                              GFP_ATOMIC);
 81                         if (!adata->data_buf)
 82                                 return -ENOMEM;
 83                 }
 84                 dest = adata->data_buf + offset;
 85                 adata->fw_data_bin_size = size + offset;
 86                 adata->is_dram_in_use = true;
 87                 break;
 88         case SOF_FW_BLK_TYPE_SRAM:
 89                 if (!adata->sram_data_buf) {
 90                         adata->sram_data_buf = dma_alloc_coherent(&pci->dev,
 91                                                                   ACP_DEFAULT_SRAM_LENGTH,
 92                                                                   &adata->sram_dma_addr,
 93                                                                   GFP_ATOMIC);
 94                         if (!adata->sram_data_buf)
 95                                 return -ENOMEM;
 96                 }
 97                 adata->fw_sram_data_bin_size = size + offset;
 98                 dest = adata->sram_data_buf + offset;
 99                 adata->is_sram_in_use = true;
100                 break;
101         default:
102                 dev_err(sdev->dev, "bad blk type 0x%x\n", blk_type);
103                 return -EINVAL;
104         }
105 
106         memcpy(dest, src, size);
107         return 0;
108 }
109 EXPORT_SYMBOL_NS(acp_dsp_block_write, SND_SOC_SOF_AMD_COMMON);
110 
111 int acp_get_bar_index(struct snd_sof_dev *sdev, u32 type)
112 {
113         return type;
114 }
115 EXPORT_SYMBOL_NS(acp_get_bar_index, SND_SOC_SOF_AMD_COMMON);
116 
117 static void configure_pte_for_fw_loading(int type, int num_pages, struct acp_dev_data *adata)
118 {
119         struct snd_sof_dev *sdev = adata->dev;
120         const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
121         unsigned int low, high;
122         dma_addr_t addr;
123         u16 page_idx;
124         u32 offset;
125 
126         switch (type) {
127         case FW_BIN:
128                 offset = FW_BIN_PTE_OFFSET;
129                 addr = adata->sha_dma_addr;
130                 break;
131         case FW_DATA_BIN:
132                 offset = adata->fw_bin_page_count * 8;
133                 addr = adata->dma_addr;
134                 break;
135         case FW_SRAM_DATA_BIN:
136                 offset = (adata->fw_bin_page_count + ACP_DRAM_PAGE_COUNT) * 8;
137                 addr = adata->sram_dma_addr;
138                 break;
139         default:
140                 dev_err(sdev->dev, "Invalid data type %x\n", type);
141                 return;
142         }
143 
144         /* Group Enable */
145         snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACPAXI2AXI_ATU_BASE_ADDR_GRP_1,
146                           desc->sram_pte_offset | BIT(31));
147         snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACPAXI2AXI_ATU_PAGE_SIZE_GRP_1,
148                           PAGE_SIZE_4K_ENABLE);
149 
150         for (page_idx = 0; page_idx < num_pages; page_idx++) {
151                 low = lower_32_bits(addr);
152                 high = upper_32_bits(addr);
153                 snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_SCRATCH_REG_0 + offset, low);
154                 high |= BIT(31);
155                 snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_SCRATCH_REG_0 + offset + 4, high);
156                 offset += 8;
157                 addr += PAGE_SIZE;
158         }
159 
160         /* Flush ATU Cache after PTE Update */
161         snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACPAXI2AXI_ATU_CTRL, ACP_ATU_CACHE_INVALID);
162 }
163 
164 /* pre fw run operations */
165 int acp_dsp_pre_fw_run(struct snd_sof_dev *sdev)
166 {
167         struct pci_dev *pci = to_pci_dev(sdev->dev);
168         const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
169         struct acp_dev_data *adata;
170         unsigned int src_addr, size_fw, dest_addr;
171         u32 page_count, dma_size;
172         int ret;
173 
174         adata = sdev->pdata->hw_pdata;
175 
176         if (adata->quirks && adata->quirks->signed_fw_image)
177                 size_fw = adata->fw_bin_size - ACP_FIRMWARE_SIGNATURE;
178         else
179                 size_fw = adata->fw_bin_size;
180 
181         page_count = PAGE_ALIGN(size_fw) >> PAGE_SHIFT;
182         adata->fw_bin_page_count = page_count;
183 
184         configure_pte_for_fw_loading(FW_BIN, page_count, adata);
185         ret = configure_and_run_sha_dma(adata, adata->bin_buf, ACP_SYSTEM_MEMORY_WINDOW,
186                                         ACP_IRAM_BASE_ADDRESS, size_fw);
187         if (ret < 0) {
188                 dev_err(sdev->dev, "SHA DMA transfer failed status: %d\n", ret);
189                 return ret;
190         }
191         if (adata->is_dram_in_use) {
192                 configure_pte_for_fw_loading(FW_DATA_BIN, ACP_DRAM_PAGE_COUNT, adata);
193                 src_addr = ACP_SYSTEM_MEMORY_WINDOW + (page_count * ACP_PAGE_SIZE);
194                 dest_addr = ACP_DRAM_BASE_ADDRESS;
195 
196                 ret = configure_and_run_dma(adata, src_addr, dest_addr, adata->fw_data_bin_size);
197                 if (ret < 0) {
198                         dev_err(sdev->dev, "acp dma configuration failed: %d\n", ret);
199                         return ret;
200                 }
201                 ret = acp_dma_status(adata, 0);
202                 if (ret < 0)
203                         dev_err(sdev->dev, "acp dma transfer status: %d\n", ret);
204         }
205         if (adata->is_sram_in_use) {
206                 configure_pte_for_fw_loading(FW_SRAM_DATA_BIN, ACP_SRAM_PAGE_COUNT, adata);
207                 src_addr = ACP_SYSTEM_MEMORY_WINDOW + ACP_DEFAULT_SRAM_LENGTH +
208                            (page_count * ACP_PAGE_SIZE);
209                 dest_addr = ACP_SRAM_BASE_ADDRESS;
210 
211                 ret = configure_and_run_dma(adata, src_addr, dest_addr,
212                                             adata->fw_sram_data_bin_size);
213                 if (ret < 0) {
214                         dev_err(sdev->dev, "acp dma configuration failed: %d\n", ret);
215                         return ret;
216                 }
217                 ret = acp_dma_status(adata, 0);
218                 if (ret < 0)
219                         dev_err(sdev->dev, "acp dma transfer status: %d\n", ret);
220         }
221 
222         if (desc->rev > 3) {
223                 /* Cache Window enable */
224                 snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_DSP0_CACHE_OFFSET0, desc->sram_pte_offset);
225                 snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_DSP0_CACHE_SIZE0, SRAM1_SIZE | BIT(31));
226         }
227 
228         /* Free memory once DMA is complete */
229         dma_size =  (PAGE_ALIGN(sdev->basefw.fw->size) >> PAGE_SHIFT) * ACP_PAGE_SIZE;
230         dma_free_coherent(&pci->dev, dma_size, adata->bin_buf, adata->sha_dma_addr);
231         adata->bin_buf = NULL;
232         if (adata->is_dram_in_use) {
233                 dma_free_coherent(&pci->dev, ACP_DEFAULT_DRAM_LENGTH, adata->data_buf,
234                                   adata->dma_addr);
235                 adata->data_buf = NULL;
236         }
237         if (adata->is_sram_in_use) {
238                 dma_free_coherent(&pci->dev, ACP_DEFAULT_SRAM_LENGTH, adata->sram_data_buf,
239                                   adata->sram_dma_addr);
240                 adata->sram_data_buf = NULL;
241         }
242         return ret;
243 }
244 EXPORT_SYMBOL_NS(acp_dsp_pre_fw_run, SND_SOC_SOF_AMD_COMMON);
245 
246 int acp_sof_dsp_run(struct snd_sof_dev *sdev)
247 {
248         struct acp_dev_data *adata = sdev->pdata->hw_pdata;
249         const struct sof_amd_acp_desc *desc = get_chip_info(sdev->pdata);
250         int val;
251 
252         snd_sof_dsp_write(sdev, ACP_DSP_BAR, ACP_DSP0_RUNSTALL, ACP_DSP_RUN);
253         val = snd_sof_dsp_read(sdev, ACP_DSP_BAR, ACP_DSP0_RUNSTALL);
254         dev_dbg(sdev->dev, "ACP_DSP0_RUNSTALL : 0x%0x\n", val);
255 
256         /* Some platforms won't support fusion DSP,keep offset zero for no support */
257         if (desc->fusion_dsp_offset && adata->enable_fw_debug) {
258                 snd_sof_dsp_write(sdev, ACP_DSP_BAR, desc->fusion_dsp_offset, ACP_DSP_RUN);
259                 val = snd_sof_dsp_read(sdev, ACP_DSP_BAR, desc->fusion_dsp_offset);
260                 dev_dbg(sdev->dev, "ACP_DSP0_FUSION_RUNSTALL : 0x%0x\n", val);
261         }
262         return 0;
263 }
264 EXPORT_SYMBOL_NS(acp_sof_dsp_run, SND_SOC_SOF_AMD_COMMON);
265 
266 int acp_sof_load_signed_firmware(struct snd_sof_dev *sdev)
267 {
268         struct snd_sof_pdata *plat_data = sdev->pdata;
269         struct acp_dev_data *adata = plat_data->hw_pdata;
270         const char *fw_filename;
271         int ret;
272 
273         fw_filename = kasprintf(GFP_KERNEL, "%s/%s",
274                                 plat_data->fw_filename_prefix,
275                                 adata->fw_code_bin);
276         if (!fw_filename)
277                 return -ENOMEM;
278 
279         ret = request_firmware(&sdev->basefw.fw, fw_filename, sdev->dev);
280         if (ret < 0) {
281                 kfree(fw_filename);
282                 dev_err(sdev->dev, "sof signed firmware code bin is missing\n");
283                 return ret;
284         } else {
285                 dev_dbg(sdev->dev, "request_firmware %s successful\n", fw_filename);
286         }
287         kfree(fw_filename);
288 
289         ret = snd_sof_dsp_block_write(sdev, SOF_FW_BLK_TYPE_IRAM, 0,
290                                       (void *)sdev->basefw.fw->data,
291                                       sdev->basefw.fw->size);
292         if (ret < 0)
293                 return ret;
294 
295         fw_filename = kasprintf(GFP_KERNEL, "%s/%s",
296                                 plat_data->fw_filename_prefix,
297                                 adata->fw_data_bin);
298         if (!fw_filename)
299                 return -ENOMEM;
300 
301         ret = request_firmware(&adata->fw_dbin, fw_filename, sdev->dev);
302         if (ret < 0) {
303                 kfree(fw_filename);
304                 dev_err(sdev->dev, "sof signed firmware data bin is missing\n");
305                 return ret;
306 
307         } else {
308                 dev_dbg(sdev->dev, "request_firmware %s successful\n", fw_filename);
309         }
310         kfree(fw_filename);
311 
312         ret = snd_sof_dsp_block_write(sdev, SOF_FW_BLK_TYPE_DRAM, 0,
313                                       (void *)adata->fw_dbin->data,
314                                       adata->fw_dbin->size);
315         return ret;
316 }
317 EXPORT_SYMBOL_NS(acp_sof_load_signed_firmware, SND_SOC_SOF_AMD_COMMON);
318 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php