Merge remote-tracking branches 'asoc/topic/samsung' and 'asoc/topic/wm8750' into...
[deliverable/linux.git] / sound / soc / intel / sst / sst_loader.c
1 /*
2 * sst_dsp.c - Intel SST Driver for audio engine
3 *
4 * Copyright (C) 2008-14 Intel Corp
5 * Authors: Vinod Koul <vinod.koul@intel.com>
6 * Harsha Priya <priya.harsha@intel.com>
7 * Dharageswari R <dharageswari.r@intel.com>
8 * KP Jeeja <jeeja.kp@intel.com>
9 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
10 *
11 * This program is free software; you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation; version 2 of the License.
14 *
15 * This program is distributed in the hope that it will be useful, but
16 * WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * General Public License for more details.
19 *
20 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
21 *
22 * This file contains all dsp controlling functions like firmware download,
23 * setting/resetting dsp cores, etc
24 */
25 #include <linux/pci.h>
26 #include <linux/delay.h>
27 #include <linux/fs.h>
28 #include <linux/sched.h>
29 #include <linux/firmware.h>
30 #include <linux/dmaengine.h>
31 #include <linux/pm_runtime.h>
32 #include <linux/pm_qos.h>
33 #include <sound/core.h>
34 #include <sound/pcm.h>
35 #include <sound/soc.h>
36 #include <sound/compress_driver.h>
37 #include <asm/platform_sst_audio.h>
38 #include "../sst-mfld-platform.h"
39 #include "sst.h"
40 #include "../sst-dsp.h"
41
42 static inline void memcpy32_toio(void __iomem *dst, const void *src, int count)
43 {
44 /* __iowrite32_copy uses 32-bit count values so divide by 4 for
45 * right count in words
46 */
47 __iowrite32_copy(dst, src, count/4);
48 }
49
50 /**
51 * intel_sst_reset_dsp_mrfld - Resetting SST DSP
52 *
53 * This resets DSP in case of MRFLD platfroms
54 */
55 int intel_sst_reset_dsp_mrfld(struct intel_sst_drv *sst_drv_ctx)
56 {
57 union config_status_reg_mrfld csr;
58
59 dev_dbg(sst_drv_ctx->dev, "sst: Resetting the DSP in mrfld\n");
60 csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR);
61
62 dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full);
63
64 csr.full |= 0x7;
65 sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full);
66 csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR);
67
68 dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full);
69
70 csr.full &= ~(0x1);
71 sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full);
72
73 csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR);
74 dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full);
75 return 0;
76 }
77
78 /**
79 * sst_start_merrifield - Start the SST DSP processor
80 *
81 * This starts the DSP in MERRIFIELD platfroms
82 */
83 int sst_start_mrfld(struct intel_sst_drv *sst_drv_ctx)
84 {
85 union config_status_reg_mrfld csr;
86
87 dev_dbg(sst_drv_ctx->dev, "sst: Starting the DSP in mrfld LALALALA\n");
88 csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR);
89 dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full);
90
91 csr.full |= 0x7;
92 sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full);
93
94 csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR);
95 dev_dbg(sst_drv_ctx->dev, "value:0x%llx\n", csr.full);
96
97 csr.part.xt_snoop = 1;
98 csr.full &= ~(0x5);
99 sst_shim_write64(sst_drv_ctx->shim, SST_CSR, csr.full);
100
101 csr.full = sst_shim_read64(sst_drv_ctx->shim, SST_CSR);
102 dev_dbg(sst_drv_ctx->dev, "sst: Starting the DSP_merrifield:%llx\n",
103 csr.full);
104 return 0;
105 }
106
107 static int sst_validate_fw_image(struct intel_sst_drv *ctx, unsigned long size,
108 struct fw_module_header **module, u32 *num_modules)
109 {
110 struct sst_fw_header *header;
111 const void *sst_fw_in_mem = ctx->fw_in_mem;
112
113 dev_dbg(ctx->dev, "Enter\n");
114
115 /* Read the header information from the data pointer */
116 header = (struct sst_fw_header *)sst_fw_in_mem;
117 dev_dbg(ctx->dev,
118 "header sign=%s size=%x modules=%x fmt=%x size=%zx\n",
119 header->signature, header->file_size, header->modules,
120 header->file_format, sizeof(*header));
121
122 /* verify FW */
123 if ((strncmp(header->signature, SST_FW_SIGN, 4) != 0) ||
124 (size != header->file_size + sizeof(*header))) {
125 /* Invalid FW signature */
126 dev_err(ctx->dev, "InvalidFW sign/filesize mismatch\n");
127 return -EINVAL;
128 }
129 *num_modules = header->modules;
130 *module = (void *)sst_fw_in_mem + sizeof(*header);
131
132 return 0;
133 }
134
135 /*
136 * sst_fill_memcpy_list - Fill the memcpy list
137 *
138 * @memcpy_list: List to be filled
139 * @destn: Destination addr to be filled in the list
140 * @src: Source addr to be filled in the list
141 * @size: Size to be filled in the list
142 *
143 * Adds the node to the list after required fields
144 * are populated in the node
145 */
146 static int sst_fill_memcpy_list(struct list_head *memcpy_list,
147 void *destn, const void *src, u32 size, bool is_io)
148 {
149 struct sst_memcpy_list *listnode;
150
151 listnode = kzalloc(sizeof(*listnode), GFP_KERNEL);
152 if (listnode == NULL)
153 return -ENOMEM;
154 listnode->dstn = destn;
155 listnode->src = src;
156 listnode->size = size;
157 listnode->is_io = is_io;
158 list_add_tail(&listnode->memcpylist, memcpy_list);
159
160 return 0;
161 }
162
163 /**
164 * sst_parse_module_memcpy - Parse audio FW modules and populate the memcpy list
165 *
166 * @sst_drv_ctx : driver context
167 * @module : FW module header
168 * @memcpy_list : Pointer to the list to be populated
169 * Create the memcpy list as the number of block to be copied
170 * returns error or 0 if module sizes are proper
171 */
172 static int sst_parse_module_memcpy(struct intel_sst_drv *sst_drv_ctx,
173 struct fw_module_header *module, struct list_head *memcpy_list)
174 {
175 struct fw_block_info *block;
176 u32 count;
177 int ret_val = 0;
178 void __iomem *ram_iomem;
179
180 dev_dbg(sst_drv_ctx->dev, "module sign %s size %x blocks %x type %x\n",
181 module->signature, module->mod_size,
182 module->blocks, module->type);
183 dev_dbg(sst_drv_ctx->dev, "module entrypoint 0x%x\n", module->entry_point);
184
185 block = (void *)module + sizeof(*module);
186
187 for (count = 0; count < module->blocks; count++) {
188 if (block->size <= 0) {
189 dev_err(sst_drv_ctx->dev, "block size invalid\n");
190 return -EINVAL;
191 }
192 switch (block->type) {
193 case SST_IRAM:
194 ram_iomem = sst_drv_ctx->iram;
195 break;
196 case SST_DRAM:
197 ram_iomem = sst_drv_ctx->dram;
198 break;
199 case SST_DDR:
200 ram_iomem = sst_drv_ctx->ddr;
201 break;
202 case SST_CUSTOM_INFO:
203 block = (void *)block + sizeof(*block) + block->size;
204 continue;
205 default:
206 dev_err(sst_drv_ctx->dev, "wrong ram type0x%x in block0x%x\n",
207 block->type, count);
208 return -EINVAL;
209 }
210
211 ret_val = sst_fill_memcpy_list(memcpy_list,
212 ram_iomem + block->ram_offset,
213 (void *)block + sizeof(*block), block->size, 1);
214 if (ret_val)
215 return ret_val;
216
217 block = (void *)block + sizeof(*block) + block->size;
218 }
219 return 0;
220 }
221
222 /**
223 * sst_parse_fw_memcpy - parse the firmware image & populate the list for memcpy
224 *
225 * @ctx : pointer to drv context
226 * @size : size of the firmware
227 * @fw_list : pointer to list_head to be populated
228 * This function parses the FW image and saves the parsed image in the list
229 * for memcpy
230 */
231 static int sst_parse_fw_memcpy(struct intel_sst_drv *ctx, unsigned long size,
232 struct list_head *fw_list)
233 {
234 struct fw_module_header *module;
235 u32 count, num_modules;
236 int ret_val;
237
238 ret_val = sst_validate_fw_image(ctx, size, &module, &num_modules);
239 if (ret_val)
240 return ret_val;
241
242 for (count = 0; count < num_modules; count++) {
243 ret_val = sst_parse_module_memcpy(ctx, module, fw_list);
244 if (ret_val)
245 return ret_val;
246 module = (void *)module + sizeof(*module) + module->mod_size;
247 }
248
249 return 0;
250 }
251
252 /**
253 * sst_do_memcpy - function initiates the memcpy
254 *
255 * @memcpy_list: Pter to memcpy list on which the memcpy needs to be initiated
256 *
257 * Triggers the memcpy
258 */
259 static void sst_do_memcpy(struct list_head *memcpy_list)
260 {
261 struct sst_memcpy_list *listnode;
262
263 list_for_each_entry(listnode, memcpy_list, memcpylist) {
264 if (listnode->is_io == true)
265 memcpy32_toio((void __iomem *)listnode->dstn,
266 listnode->src, listnode->size);
267 else
268 memcpy(listnode->dstn, listnode->src, listnode->size);
269 }
270 }
271
272 void sst_memcpy_free_resources(struct intel_sst_drv *sst_drv_ctx)
273 {
274 struct sst_memcpy_list *listnode, *tmplistnode;
275
276 /* Free the list */
277 if (!list_empty(&sst_drv_ctx->memcpy_list)) {
278 list_for_each_entry_safe(listnode, tmplistnode,
279 &sst_drv_ctx->memcpy_list, memcpylist) {
280 list_del(&listnode->memcpylist);
281 kfree(listnode);
282 }
283 }
284 }
285
286 static int sst_cache_and_parse_fw(struct intel_sst_drv *sst,
287 const struct firmware *fw)
288 {
289 int retval = 0;
290
291 sst->fw_in_mem = kzalloc(fw->size, GFP_KERNEL);
292 if (!sst->fw_in_mem) {
293 retval = -ENOMEM;
294 goto end_release;
295 }
296 dev_dbg(sst->dev, "copied fw to %p", sst->fw_in_mem);
297 dev_dbg(sst->dev, "phys: %lx", (unsigned long)virt_to_phys(sst->fw_in_mem));
298 memcpy(sst->fw_in_mem, fw->data, fw->size);
299 retval = sst_parse_fw_memcpy(sst, fw->size, &sst->memcpy_list);
300 if (retval) {
301 dev_err(sst->dev, "Failed to parse fw\n");
302 kfree(sst->fw_in_mem);
303 sst->fw_in_mem = NULL;
304 }
305
306 end_release:
307 release_firmware(fw);
308 return retval;
309
310 }
311
312 void sst_firmware_load_cb(const struct firmware *fw, void *context)
313 {
314 struct intel_sst_drv *ctx = context;
315
316 dev_dbg(ctx->dev, "Enter\n");
317
318 if (fw == NULL) {
319 dev_err(ctx->dev, "request fw failed\n");
320 return;
321 }
322
323 mutex_lock(&ctx->sst_lock);
324
325 if (ctx->sst_state != SST_RESET ||
326 ctx->fw_in_mem != NULL) {
327 if (fw != NULL)
328 release_firmware(fw);
329 mutex_unlock(&ctx->sst_lock);
330 return;
331 }
332
333 dev_dbg(ctx->dev, "Request Fw completed\n");
334 sst_cache_and_parse_fw(ctx, fw);
335 mutex_unlock(&ctx->sst_lock);
336 }
337
338 /*
339 * sst_request_fw - requests audio fw from kernel and saves a copy
340 *
341 * This function requests the SST FW from the kernel, parses it and
342 * saves a copy in the driver context
343 */
344 static int sst_request_fw(struct intel_sst_drv *sst)
345 {
346 int retval = 0;
347 const struct firmware *fw;
348
349 retval = request_firmware(&fw, sst->firmware_name, sst->dev);
350 if (fw == NULL) {
351 dev_err(sst->dev, "fw is returning as null\n");
352 return -EINVAL;
353 }
354 if (retval) {
355 dev_err(sst->dev, "request fw failed %d\n", retval);
356 return retval;
357 }
358 mutex_lock(&sst->sst_lock);
359 retval = sst_cache_and_parse_fw(sst, fw);
360 mutex_unlock(&sst->sst_lock);
361
362 return retval;
363 }
364
365 /*
366 * Writing the DDR physical base to DCCM offset
367 * so that FW can use it to setup TLB
368 */
369 static void sst_dccm_config_write(void __iomem *dram_base,
370 unsigned int ddr_base)
371 {
372 void __iomem *addr;
373 u32 bss_reset = 0;
374
375 addr = (void __iomem *)(dram_base + MRFLD_FW_DDR_BASE_OFFSET);
376 memcpy32_toio(addr, (void *)&ddr_base, sizeof(u32));
377 bss_reset |= (1 << MRFLD_FW_BSS_RESET_BIT);
378 addr = (void __iomem *)(dram_base + MRFLD_FW_FEATURE_BASE_OFFSET);
379 memcpy32_toio(addr, &bss_reset, sizeof(u32));
380
381 }
382
383 void sst_post_download_mrfld(struct intel_sst_drv *ctx)
384 {
385 sst_dccm_config_write(ctx->dram, ctx->ddr_base);
386 dev_dbg(ctx->dev, "config written to DCCM\n");
387 }
388
389 /**
390 * sst_load_fw - function to load FW into DSP
391 * Transfers the FW to DSP using dma/memcpy
392 */
393 int sst_load_fw(struct intel_sst_drv *sst_drv_ctx)
394 {
395 int ret_val = 0;
396 struct sst_block *block;
397
398 dev_dbg(sst_drv_ctx->dev, "sst_load_fw\n");
399
400 if (sst_drv_ctx->sst_state != SST_RESET ||
401 sst_drv_ctx->sst_state == SST_SHUTDOWN)
402 return -EAGAIN;
403
404 if (!sst_drv_ctx->fw_in_mem) {
405 dev_dbg(sst_drv_ctx->dev, "sst: FW not in memory retry to download\n");
406 ret_val = sst_request_fw(sst_drv_ctx);
407 if (ret_val)
408 return ret_val;
409 }
410
411 BUG_ON(!sst_drv_ctx->fw_in_mem);
412 block = sst_create_block(sst_drv_ctx, 0, FW_DWNL_ID);
413 if (block == NULL)
414 return -ENOMEM;
415
416 /* Prevent C-states beyond C6 */
417 pm_qos_update_request(sst_drv_ctx->qos, 0);
418
419 sst_drv_ctx->sst_state = SST_FW_LOADING;
420
421 ret_val = sst_drv_ctx->ops->reset(sst_drv_ctx);
422 if (ret_val)
423 goto restore;
424
425 sst_do_memcpy(&sst_drv_ctx->memcpy_list);
426
427 /* Write the DRAM/DCCM config before enabling FW */
428 if (sst_drv_ctx->ops->post_download)
429 sst_drv_ctx->ops->post_download(sst_drv_ctx);
430
431 /* bring sst out of reset */
432 ret_val = sst_drv_ctx->ops->start(sst_drv_ctx);
433 if (ret_val)
434 goto restore;
435
436 ret_val = sst_wait_timeout(sst_drv_ctx, block);
437 if (ret_val) {
438 dev_err(sst_drv_ctx->dev, "fw download failed %d\n" , ret_val);
439 /* FW download failed due to timeout */
440 ret_val = -EBUSY;
441
442 }
443
444
445 restore:
446 /* Re-enable Deeper C-states beyond C6 */
447 pm_qos_update_request(sst_drv_ctx->qos, PM_QOS_DEFAULT_VALUE);
448 sst_free_block(sst_drv_ctx, block);
449 dev_dbg(sst_drv_ctx->dev, "fw load successful!!!\n");
450
451 if (sst_drv_ctx->ops->restore_dsp_context)
452 sst_drv_ctx->ops->restore_dsp_context();
453 sst_drv_ctx->sst_state = SST_FW_RUNNING;
454 return ret_val;
455 }
456
This page took 0.063156 seconds and 5 git commands to generate.