blob: a6d46aef9724da9a709587486e83a83b3b38504b [file] [log] [blame]
Asish Bhattacharya8e2277f2017-07-20 18:31:55 +05301/*
2 * Copyright (c) 2016-2017, The Linux Foundation. All rights reserved.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 */
13#include <linux/module.h>
14#include <linux/platform_device.h>
15#include <linux/slab.h>
16#include <linux/stringify.h>
17#include <linux/of.h>
18#include <linux/debugfs.h>
19#include <linux/component.h>
20#include <linux/dma-mapping.h>
21#include <soc/qcom/ramdump.h>
22#include <sound/wcd-dsp-mgr.h>
23#include "wcd-dsp-utils.h"
24
25/* Forward declarations */
26static char *wdsp_get_cmpnt_type_string(enum wdsp_cmpnt_type);
27
28/* Component related macros */
Asish Bhattacharya84f7f732017-07-25 16:29:27 +053029#define WDSP_GET_COMPONENT(wdsp, x) ((x >= WDSP_CMPNT_TYPE_MAX || x < 0) ? \
30 NULL : (&(wdsp->cmpnts[x])))
Asish Bhattacharya8e2277f2017-07-20 18:31:55 +053031#define WDSP_GET_CMPNT_TYPE_STR(x) wdsp_get_cmpnt_type_string(x)
32
33/*
34 * These #defines indicate the bit number in status field
35 * for each of the status. If bit is set, it indicates
36 * the status as done, else if bit is not set, it indicates
37 * the status is either failed or not done.
38 */
39#define WDSP_STATUS_INITIALIZED BIT(0)
40#define WDSP_STATUS_CODE_DLOADED BIT(1)
41#define WDSP_STATUS_DATA_DLOADED BIT(2)
42#define WDSP_STATUS_BOOTED BIT(3)
43
44/* Helper macros for printing wdsp messages */
45#define WDSP_ERR(wdsp, fmt, ...) \
46 dev_err(wdsp->mdev, "%s: " fmt "\n", __func__, ##__VA_ARGS__)
47#define WDSP_DBG(wdsp, fmt, ...) \
48 dev_dbg(wdsp->mdev, "%s: " fmt "\n", __func__, ##__VA_ARGS__)
49
50/* Helper macros for locking */
51#define WDSP_MGR_MUTEX_LOCK(wdsp, lock) \
52{ \
53 WDSP_DBG(wdsp, "mutex_lock(%s)", \
54 __stringify_1(lock)); \
55 mutex_lock(&lock); \
56}
57
58#define WDSP_MGR_MUTEX_UNLOCK(wdsp, lock) \
59{ \
60 WDSP_DBG(wdsp, "mutex_unlock(%s)", \
61 __stringify_1(lock)); \
62 mutex_unlock(&lock); \
63}
64
65/* Helper macros for using status mask */
66#define WDSP_SET_STATUS(wdsp, state) \
67{ \
68 wdsp->status |= state; \
69 WDSP_DBG(wdsp, "set 0x%lx, new_state = 0x%x", \
70 state, wdsp->status); \
71}
72
73#define WDSP_CLEAR_STATUS(wdsp, state) \
74{ \
75 wdsp->status &= (~state); \
76 WDSP_DBG(wdsp, "clear 0x%lx, new_state = 0x%x", \
77 state, wdsp->status); \
78}
79
80#define WDSP_STATUS_IS_SET(wdsp, state) (wdsp->status & state)
81
82/* SSR relate status macros */
83#define WDSP_SSR_STATUS_WDSP_READY BIT(0)
84#define WDSP_SSR_STATUS_CDC_READY BIT(1)
85#define WDSP_SSR_STATUS_READY \
86 (WDSP_SSR_STATUS_WDSP_READY | WDSP_SSR_STATUS_CDC_READY)
87#define WDSP_SSR_READY_WAIT_TIMEOUT (10 * HZ)
88
89enum wdsp_ssr_type {
90
91 /* Init value, indicates there is no SSR in progress */
92 WDSP_SSR_TYPE_NO_SSR = 0,
93
94 /*
95 * Indicates WDSP crashed. The manager driver internally
96 * decides when to perform WDSP restart based on the
97 * users of wdsp. Hence there is no explicit WDSP_UP.
98 */
99 WDSP_SSR_TYPE_WDSP_DOWN,
100
101 /* Indicates codec hardware is down */
102 WDSP_SSR_TYPE_CDC_DOWN,
103
104 /* Indicates codec hardware is up, trigger to restart WDSP */
105 WDSP_SSR_TYPE_CDC_UP,
106};
107
108struct wdsp_cmpnt {
109
110 /* OF node of the phandle */
111 struct device_node *np;
112
113 /*
114 * Child component's dev_name, should be set in DT for the child's
115 * phandle if child's dev->of_node does not match the phandle->of_node
116 */
117 const char *cdev_name;
118
119 /* Child component's device node */
120 struct device *cdev;
121
122 /* Private data that component may want back on callbacks */
123 void *priv_data;
124
125 /* Child ops */
126 struct wdsp_cmpnt_ops *ops;
127};
128
129struct wdsp_ramdump_data {
130
131 /* Ramdump device */
132 void *rd_dev;
133
134 /* DMA address of the dump */
135 dma_addr_t rd_addr;
136
137 /* Virtual address of the dump */
138 void *rd_v_addr;
139
140 /* Data provided through error interrupt */
141 struct wdsp_err_signal_arg err_data;
142};
143
144struct wdsp_mgr_priv {
145
146 /* Manager driver's struct device pointer */
147 struct device *mdev;
148
149 /* Match struct for component framework */
150 struct component_match *match;
151
152 /* Manager's ops/function callbacks */
153 struct wdsp_mgr_ops *ops;
154
155 /* Array to store information for all expected components */
156 struct wdsp_cmpnt cmpnts[WDSP_CMPNT_TYPE_MAX];
157
158 /* The filename of image to be downloaded */
159 const char *img_fname;
160
161 /* Keeps track of current state of manager driver */
162 u32 status;
163
164 /* Work to load the firmware image after component binding */
165 struct work_struct load_fw_work;
166
167 /* List of segments in image to be downloaded */
168 struct list_head *seg_list;
169
170 /* Base address of the image in memory */
171 u32 base_addr;
172
173 /* Instances using dsp */
174 int dsp_users;
175
176 /* Lock for serializing ops called by components */
177 struct mutex api_mutex;
178
179 struct wdsp_ramdump_data dump_data;
180
181 /* SSR related */
182 enum wdsp_ssr_type ssr_type;
183 struct mutex ssr_mutex;
184 struct work_struct ssr_work;
185 u16 ready_status;
186 struct completion ready_compl;
187
188 /* Debugfs related */
189 struct dentry *entry;
190 bool panic_on_error;
191};
192
193static char *wdsp_get_ssr_type_string(enum wdsp_ssr_type type)
194{
195 switch (type) {
196 case WDSP_SSR_TYPE_NO_SSR:
197 return "NO_SSR";
198 case WDSP_SSR_TYPE_WDSP_DOWN:
199 return "WDSP_DOWN";
200 case WDSP_SSR_TYPE_CDC_DOWN:
201 return "CDC_DOWN";
202 case WDSP_SSR_TYPE_CDC_UP:
203 return "CDC_UP";
204 default:
205 pr_err("%s: Invalid ssr_type %d\n",
206 __func__, type);
207 return "Invalid";
208 }
209}
210
211static char *wdsp_get_cmpnt_type_string(enum wdsp_cmpnt_type type)
212{
213 switch (type) {
214 case WDSP_CMPNT_CONTROL:
215 return "control";
216 case WDSP_CMPNT_IPC:
217 return "ipc";
218 case WDSP_CMPNT_TRANSPORT:
219 return "transport";
220 default:
221 pr_err("%s: Invalid component type %d\n",
222 __func__, type);
223 return "Invalid";
224 }
225}
226
227static void __wdsp_clr_ready_locked(struct wdsp_mgr_priv *wdsp,
228 u16 value)
229{
230 wdsp->ready_status &= ~(value);
231 WDSP_DBG(wdsp, "ready_status = 0x%x", wdsp->ready_status);
232}
233
234static void __wdsp_set_ready_locked(struct wdsp_mgr_priv *wdsp,
235 u16 value, bool mark_complete)
236{
237 wdsp->ready_status |= value;
238 WDSP_DBG(wdsp, "ready_status = 0x%x", wdsp->ready_status);
239
240 if (mark_complete &&
241 wdsp->ready_status == WDSP_SSR_STATUS_READY) {
242 WDSP_DBG(wdsp, "marking ready completion");
243 complete(&wdsp->ready_compl);
244 }
245}
246
247static void wdsp_broadcast_event_upseq(struct wdsp_mgr_priv *wdsp,
248 enum wdsp_event_type event,
249 void *data)
250{
251 struct wdsp_cmpnt *cmpnt;
252 int i;
253
254 for (i = 0; i < WDSP_CMPNT_TYPE_MAX; i++) {
255 cmpnt = WDSP_GET_COMPONENT(wdsp, i);
256 if (cmpnt && cmpnt->ops && cmpnt->ops->event_handler)
257 cmpnt->ops->event_handler(cmpnt->cdev, cmpnt->priv_data,
258 event, data);
259 }
260}
261
262static void wdsp_broadcast_event_downseq(struct wdsp_mgr_priv *wdsp,
263 enum wdsp_event_type event,
264 void *data)
265{
266 struct wdsp_cmpnt *cmpnt;
267 int i;
268
269 for (i = WDSP_CMPNT_TYPE_MAX - 1; i >= 0; i--) {
270 cmpnt = WDSP_GET_COMPONENT(wdsp, i);
271 if (cmpnt && cmpnt->ops && cmpnt->ops->event_handler)
272 cmpnt->ops->event_handler(cmpnt->cdev, cmpnt->priv_data,
273 event, data);
274 }
275}
276
277static int wdsp_unicast_event(struct wdsp_mgr_priv *wdsp,
278 enum wdsp_cmpnt_type type,
279 enum wdsp_event_type event,
280 void *data)
281{
282 struct wdsp_cmpnt *cmpnt;
283 int ret;
284
285 cmpnt = WDSP_GET_COMPONENT(wdsp, type);
286 if (cmpnt && cmpnt->ops && cmpnt->ops->event_handler) {
287 ret = cmpnt->ops->event_handler(cmpnt->cdev, cmpnt->priv_data,
288 event, data);
289 } else {
290 WDSP_ERR(wdsp, "not valid event_handler for %s",
291 WDSP_GET_CMPNT_TYPE_STR(type));
292 ret = -EINVAL;
293 }
294
295 return ret;
296}
297
298static void wdsp_deinit_components(struct wdsp_mgr_priv *wdsp)
299{
300 struct wdsp_cmpnt *cmpnt;
301 int i;
302
303 for (i = WDSP_CMPNT_TYPE_MAX - 1; i >= 0; i--) {
304 cmpnt = WDSP_GET_COMPONENT(wdsp, i);
305 if (cmpnt && cmpnt->ops && cmpnt->ops->deinit)
306 cmpnt->ops->deinit(cmpnt->cdev, cmpnt->priv_data);
307 }
308}
309
310static int wdsp_init_components(struct wdsp_mgr_priv *wdsp)
311{
312 struct wdsp_cmpnt *cmpnt;
313 int fail_idx = WDSP_CMPNT_TYPE_MAX;
314 int i, ret = 0;
315
316 for (i = 0; i < WDSP_CMPNT_TYPE_MAX; i++) {
317
318 cmpnt = WDSP_GET_COMPONENT(wdsp, i);
319
320 /* Init is allowed to be NULL */
321 if (!cmpnt->ops || !cmpnt->ops->init)
322 continue;
323 ret = cmpnt->ops->init(cmpnt->cdev, cmpnt->priv_data);
324 if (ret) {
325 WDSP_ERR(wdsp, "Init failed (%d) for component %s",
326 ret, WDSP_GET_CMPNT_TYPE_STR(i));
327 fail_idx = i;
328 break;
329 }
330 }
331
332 if (fail_idx < WDSP_CMPNT_TYPE_MAX) {
333 /* Undo init for already initialized components */
334 for (i = fail_idx - 1; i >= 0; i--) {
335 struct wdsp_cmpnt *cmpnt = WDSP_GET_COMPONENT(wdsp, i);
336
337 if (cmpnt->ops && cmpnt->ops->deinit)
338 cmpnt->ops->deinit(cmpnt->cdev,
339 cmpnt->priv_data);
340 }
341 } else {
342 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_POST_INIT, NULL);
343 }
344
345 return ret;
346}
347
348static int wdsp_load_each_segment(struct wdsp_mgr_priv *wdsp,
349 struct wdsp_img_segment *seg)
350{
351 struct wdsp_img_section img_section;
352 int ret;
353
354 WDSP_DBG(wdsp,
355 "base_addr 0x%x, split_fname %s, load_addr 0x%x, size 0x%zx",
356 wdsp->base_addr, seg->split_fname, seg->load_addr, seg->size);
357
358 if (seg->load_addr < wdsp->base_addr) {
359 WDSP_ERR(wdsp, "Invalid addr 0x%x, base_addr = 0x%x",
360 seg->load_addr, wdsp->base_addr);
361 return -EINVAL;
362 }
363
364 img_section.addr = seg->load_addr - wdsp->base_addr;
365 img_section.size = seg->size;
366 img_section.data = seg->data;
367
368 ret = wdsp_unicast_event(wdsp, WDSP_CMPNT_TRANSPORT,
369 WDSP_EVENT_DLOAD_SECTION,
370 &img_section);
371 if (ret < 0)
372 WDSP_ERR(wdsp,
373 "Failed, err = %d for base_addr = 0x%x split_fname = %s, load_addr = 0x%x, size = 0x%zx",
374 ret, wdsp->base_addr, seg->split_fname,
375 seg->load_addr, seg->size);
376 return ret;
377}
378
379static int wdsp_download_segments(struct wdsp_mgr_priv *wdsp,
380 unsigned int type)
381{
382 struct wdsp_cmpnt *ctl;
383 struct wdsp_img_segment *seg = NULL;
384 enum wdsp_event_type pre, post;
385 long status;
386 int ret;
387
388 ctl = WDSP_GET_COMPONENT(wdsp, WDSP_CMPNT_CONTROL);
389
390 if (type == WDSP_ELF_FLAG_RE) {
391 pre = WDSP_EVENT_PRE_DLOAD_CODE;
392 post = WDSP_EVENT_POST_DLOAD_CODE;
393 status = WDSP_STATUS_CODE_DLOADED;
394 } else if (type == WDSP_ELF_FLAG_WRITE) {
395 pre = WDSP_EVENT_PRE_DLOAD_DATA;
396 post = WDSP_EVENT_POST_DLOAD_DATA;
397 status = WDSP_STATUS_DATA_DLOADED;
398 } else {
399 WDSP_ERR(wdsp, "Invalid type %u", type);
400 return -EINVAL;
401 }
402
403 ret = wdsp_get_segment_list(ctl->cdev, wdsp->img_fname,
404 type, wdsp->seg_list, &wdsp->base_addr);
405 if (ret < 0 ||
406 list_empty(wdsp->seg_list)) {
407 WDSP_ERR(wdsp, "Error %d to get image segments for type %d",
408 ret, type);
409 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_DLOAD_FAILED,
410 NULL);
411 goto done;
412 }
413
414 /* Notify all components that image is about to be downloaded */
415 wdsp_broadcast_event_upseq(wdsp, pre, NULL);
416
417 /* Go through the list of segments and download one by one */
418 list_for_each_entry(seg, wdsp->seg_list, list) {
419 ret = wdsp_load_each_segment(wdsp, seg);
420 if (ret < 0) {
421 wdsp_broadcast_event_downseq(wdsp,
422 WDSP_EVENT_DLOAD_FAILED,
423 NULL);
424 goto dload_error;
425 }
426 }
427
428 WDSP_SET_STATUS(wdsp, status);
429
430 /* Notify all components that image is downloaded */
431 wdsp_broadcast_event_downseq(wdsp, post, NULL);
432
433dload_error:
434 wdsp_flush_segment_list(wdsp->seg_list);
435done:
436 return ret;
437}
438
439static int wdsp_init_and_dload_code_sections(struct wdsp_mgr_priv *wdsp)
440{
441 int ret;
442 bool is_initialized;
443
444 is_initialized = WDSP_STATUS_IS_SET(wdsp, WDSP_STATUS_INITIALIZED);
445
446 if (!is_initialized) {
447 /* Components are not initialized yet, initialize them */
448 ret = wdsp_init_components(wdsp);
449 if (ret < 0) {
450 WDSP_ERR(wdsp, "INIT failed, err = %d", ret);
451 goto done;
452 }
453 WDSP_SET_STATUS(wdsp, WDSP_STATUS_INITIALIZED);
454 }
455
456 /* Download the read-execute sections of image */
457 ret = wdsp_download_segments(wdsp, WDSP_ELF_FLAG_RE);
458 if (ret < 0) {
459 WDSP_ERR(wdsp, "Error %d to download code sections", ret);
460 goto done;
461 }
462done:
463 return ret;
464}
465
466static void wdsp_load_fw_image(struct work_struct *work)
467{
468 struct wdsp_mgr_priv *wdsp;
469 int ret;
470
471 wdsp = container_of(work, struct wdsp_mgr_priv, load_fw_work);
472 if (!wdsp) {
473 pr_err("%s: Invalid private_data\n", __func__);
474 return;
475 }
476
477 ret = wdsp_init_and_dload_code_sections(wdsp);
478 if (ret < 0)
479 WDSP_ERR(wdsp, "dload code sections failed, err = %d", ret);
480}
481
482static int wdsp_enable_dsp(struct wdsp_mgr_priv *wdsp)
483{
484 int ret;
485
486 /* Make sure wdsp is in good state */
487 if (!WDSP_STATUS_IS_SET(wdsp, WDSP_STATUS_CODE_DLOADED)) {
488 WDSP_ERR(wdsp, "WDSP in invalid state 0x%x", wdsp->status);
489 ret = -EINVAL;
490 goto done;
491 }
492
493 /* Download the read-write sections of image */
494 ret = wdsp_download_segments(wdsp, WDSP_ELF_FLAG_WRITE);
495 if (ret < 0) {
496 WDSP_ERR(wdsp, "Data section download failed, err = %d", ret);
497 goto done;
498 }
499
500 wdsp_broadcast_event_upseq(wdsp, WDSP_EVENT_PRE_BOOTUP, NULL);
501
502 ret = wdsp_unicast_event(wdsp, WDSP_CMPNT_CONTROL,
503 WDSP_EVENT_DO_BOOT, NULL);
504 if (ret < 0) {
505 WDSP_ERR(wdsp, "Failed to boot dsp, err = %d", ret);
506 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_DATA_DLOADED);
507 goto done;
508 }
509
510 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_POST_BOOTUP, NULL);
511 WDSP_SET_STATUS(wdsp, WDSP_STATUS_BOOTED);
512done:
513 return ret;
514}
515
516static int wdsp_disable_dsp(struct wdsp_mgr_priv *wdsp)
517{
518 int ret;
519
520 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->ssr_mutex);
521
522 /*
523 * If Disable happened while SSR is in progress, then set the SSR
524 * ready status indicating WDSP is now ready. Ignore the disable
525 * event here and let the SSR handler go through shutdown.
526 */
527 if (wdsp->ssr_type != WDSP_SSR_TYPE_NO_SSR) {
528 __wdsp_set_ready_locked(wdsp, WDSP_SSR_STATUS_WDSP_READY, true);
529 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->ssr_mutex);
530 return 0;
531 }
532
533 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->ssr_mutex);
534
535 /* Make sure wdsp is in good state */
536 if (!WDSP_STATUS_IS_SET(wdsp, WDSP_STATUS_BOOTED)) {
537 WDSP_ERR(wdsp, "wdsp in invalid state 0x%x", wdsp->status);
538 ret = -EINVAL;
539 goto done;
540 }
541
542 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_PRE_SHUTDOWN, NULL);
543 ret = wdsp_unicast_event(wdsp, WDSP_CMPNT_CONTROL,
544 WDSP_EVENT_DO_SHUTDOWN, NULL);
545 if (ret < 0) {
546 WDSP_ERR(wdsp, "Failed to shutdown dsp, err = %d", ret);
547 goto done;
548 }
549
550 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_POST_SHUTDOWN, NULL);
551 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_BOOTED);
552
553 /* Data sections are to be downloaded per boot */
554 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_DATA_DLOADED);
555done:
556 return ret;
557}
558
559static int wdsp_register_cmpnt_ops(struct device *wdsp_dev,
560 struct device *cdev,
561 void *priv_data,
562 struct wdsp_cmpnt_ops *ops)
563{
564 struct wdsp_mgr_priv *wdsp;
565 struct wdsp_cmpnt *cmpnt;
566 int i, ret;
567
568 if (!wdsp_dev || !cdev || !ops)
569 return -EINVAL;
570
571 wdsp = dev_get_drvdata(wdsp_dev);
572
573 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->api_mutex);
574
575 for (i = 0; i < WDSP_CMPNT_TYPE_MAX; i++) {
576 cmpnt = WDSP_GET_COMPONENT(wdsp, i);
577 if ((cdev->of_node && cdev->of_node == cmpnt->np) ||
578 (cmpnt->cdev_name &&
579 !strcmp(dev_name(cdev), cmpnt->cdev_name))) {
580 break;
581 }
582 }
583
584 if (i == WDSP_CMPNT_TYPE_MAX) {
585 WDSP_ERR(wdsp, "Failed to register component dev %s",
586 dev_name(cdev));
587 ret = -EINVAL;
588 goto done;
589 }
590
591 cmpnt->cdev = cdev;
592 cmpnt->ops = ops;
593 cmpnt->priv_data = priv_data;
594done:
595 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->api_mutex);
596 return 0;
597}
598
599static struct device *wdsp_get_dev_for_cmpnt(struct device *wdsp_dev,
600 enum wdsp_cmpnt_type type)
601{
602 struct wdsp_mgr_priv *wdsp;
603 struct wdsp_cmpnt *cmpnt;
604
605 if (!wdsp_dev || type >= WDSP_CMPNT_TYPE_MAX)
606 return NULL;
607
608 wdsp = dev_get_drvdata(wdsp_dev);
609 cmpnt = WDSP_GET_COMPONENT(wdsp, type);
610
611 return cmpnt->cdev;
612}
613
614static int wdsp_get_devops_for_cmpnt(struct device *wdsp_dev,
615 enum wdsp_cmpnt_type type,
616 void *data)
617{
618 struct wdsp_mgr_priv *wdsp;
619 int ret = 0;
620
621 if (!wdsp_dev || type >= WDSP_CMPNT_TYPE_MAX)
622 return -EINVAL;
623
624 wdsp = dev_get_drvdata(wdsp_dev);
625 ret = wdsp_unicast_event(wdsp, type,
626 WDSP_EVENT_GET_DEVOPS, data);
627 if (ret)
628 WDSP_ERR(wdsp, "get_dev_ops failed for cmpnt type %d",
629 type);
630 return ret;
631}
632
633static void wdsp_collect_ramdumps(struct wdsp_mgr_priv *wdsp)
634{
635 struct wdsp_img_section img_section;
636 struct wdsp_err_signal_arg *data = &wdsp->dump_data.err_data;
637 struct ramdump_segment rd_seg;
638 int ret = 0;
639
640 if (wdsp->ssr_type != WDSP_SSR_TYPE_WDSP_DOWN ||
641 !data->mem_dumps_enabled) {
642 WDSP_DBG(wdsp, "cannot dump memory, ssr_type %s, dumps %s",
643 wdsp_get_ssr_type_string(wdsp->ssr_type),
644 !(data->mem_dumps_enabled) ? "disabled" : "enabled");
645 goto done;
646 }
647
648 if (data->dump_size == 0 ||
649 data->remote_start_addr < wdsp->base_addr) {
650 WDSP_ERR(wdsp, "Invalid start addr 0x%x or dump_size 0x%zx",
651 data->remote_start_addr, data->dump_size);
652 goto done;
653 }
654
655 if (!wdsp->dump_data.rd_dev) {
656 WDSP_ERR(wdsp, "Ramdump device is not setup");
657 goto done;
658 }
659
660 WDSP_DBG(wdsp, "base_addr 0x%x, dump_start_addr 0x%x, dump_size 0x%zx",
661 wdsp->base_addr, data->remote_start_addr, data->dump_size);
662
663 /* Allocate memory for dumps */
664 wdsp->dump_data.rd_v_addr = dma_alloc_coherent(wdsp->mdev,
665 data->dump_size,
666 &wdsp->dump_data.rd_addr,
667 GFP_KERNEL);
668 if (!wdsp->dump_data.rd_v_addr)
669 goto done;
670
671 img_section.addr = data->remote_start_addr - wdsp->base_addr;
672 img_section.size = data->dump_size;
673 img_section.data = wdsp->dump_data.rd_v_addr;
674
675 ret = wdsp_unicast_event(wdsp, WDSP_CMPNT_TRANSPORT,
676 WDSP_EVENT_READ_SECTION,
677 &img_section);
678 if (ret < 0) {
679 WDSP_ERR(wdsp, "Failed to read dumps, size 0x%zx at addr 0x%x",
680 img_section.size, img_section.addr);
681 goto err_read_dumps;
682 }
683
684 /*
685 * If panic_on_error flag is explicitly set through the debugfs,
686 * then cause a BUG here to aid debugging.
687 */
688 BUG_ON(wdsp->panic_on_error);
689
690 rd_seg.address = (unsigned long) wdsp->dump_data.rd_v_addr;
691 rd_seg.size = img_section.size;
692 rd_seg.v_address = wdsp->dump_data.rd_v_addr;
693
694 ret = do_ramdump(wdsp->dump_data.rd_dev, &rd_seg, 1);
695 if (ret < 0)
696 WDSP_ERR(wdsp, "do_ramdump failed with error %d", ret);
697
698err_read_dumps:
699 dma_free_coherent(wdsp->mdev, data->dump_size,
700 wdsp->dump_data.rd_v_addr, wdsp->dump_data.rd_addr);
701done:
702 return;
703}
704
705static void wdsp_ssr_work_fn(struct work_struct *work)
706{
707 struct wdsp_mgr_priv *wdsp;
708 int ret;
709
710 wdsp = container_of(work, struct wdsp_mgr_priv, ssr_work);
711 if (!wdsp) {
712 pr_err("%s: Invalid private_data\n", __func__);
713 return;
714 }
715
716 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->ssr_mutex);
717
718 /* Issue ramdumps and shutdown only if DSP is currently booted */
719 if (WDSP_STATUS_IS_SET(wdsp, WDSP_STATUS_BOOTED)) {
720 wdsp_collect_ramdumps(wdsp);
721 ret = wdsp_unicast_event(wdsp, WDSP_CMPNT_CONTROL,
722 WDSP_EVENT_DO_SHUTDOWN, NULL);
723 if (ret < 0)
724 WDSP_ERR(wdsp, "Failed WDSP shutdown, err = %d", ret);
725
726 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_POST_SHUTDOWN,
727 NULL);
728 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_BOOTED);
729 }
730
731 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->ssr_mutex);
732 ret = wait_for_completion_timeout(&wdsp->ready_compl,
733 WDSP_SSR_READY_WAIT_TIMEOUT);
734 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->ssr_mutex);
735 if (ret == 0) {
736 WDSP_ERR(wdsp, "wait_for_ready timed out, status = 0x%x",
737 wdsp->ready_status);
738 goto done;
739 }
740
741 /* Data sections are to downloaded per WDSP boot */
742 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_DATA_DLOADED);
743
744 /*
745 * Even though code section could possible be retained on DSP
746 * crash, go ahead and still re-download just to avoid any
747 * memory corruption from previous crash.
748 */
749 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_CODE_DLOADED);
750
751 /* If codec restarted, then all components must be re-initialized */
752 if (wdsp->ssr_type == WDSP_SSR_TYPE_CDC_UP) {
753 wdsp_deinit_components(wdsp);
754 WDSP_CLEAR_STATUS(wdsp, WDSP_STATUS_INITIALIZED);
755 }
756
757 ret = wdsp_init_and_dload_code_sections(wdsp);
758 if (ret < 0) {
759 WDSP_ERR(wdsp, "Failed to dload code sections err = %d",
760 ret);
761 goto done;
762 }
763
764 /* SSR handling is finished, mark SSR type as NO_SSR */
765 wdsp->ssr_type = WDSP_SSR_TYPE_NO_SSR;
766done:
767 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->ssr_mutex);
768}
769
770static int wdsp_ssr_handler(struct wdsp_mgr_priv *wdsp, void *arg,
771 enum wdsp_ssr_type ssr_type)
772{
773 enum wdsp_ssr_type current_ssr_type;
774 struct wdsp_err_signal_arg *err_data;
775
776 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->ssr_mutex);
777
778 current_ssr_type = wdsp->ssr_type;
779 WDSP_DBG(wdsp, "Current ssr_type %s, handling ssr_type %s",
780 wdsp_get_ssr_type_string(current_ssr_type),
781 wdsp_get_ssr_type_string(ssr_type));
782 wdsp->ssr_type = ssr_type;
783
784 if (arg) {
785 err_data = (struct wdsp_err_signal_arg *) arg;
786 memcpy(&wdsp->dump_data.err_data, err_data,
787 sizeof(*err_data));
788 } else {
789 memset(&wdsp->dump_data.err_data, 0,
790 sizeof(wdsp->dump_data.err_data));
791 }
792
793 switch (ssr_type) {
794
795 case WDSP_SSR_TYPE_WDSP_DOWN:
796 __wdsp_clr_ready_locked(wdsp, WDSP_SSR_STATUS_WDSP_READY);
797 wdsp_broadcast_event_downseq(wdsp, WDSP_EVENT_PRE_SHUTDOWN,
798 NULL);
799 schedule_work(&wdsp->ssr_work);
800 break;
801
802 case WDSP_SSR_TYPE_CDC_DOWN:
803 __wdsp_clr_ready_locked(wdsp, WDSP_SSR_STATUS_CDC_READY);
804 /*
805 * If DSP is booted when CDC_DOWN is received, it needs
806 * to be shutdown.
807 */
808 if (WDSP_STATUS_IS_SET(wdsp, WDSP_STATUS_BOOTED)) {
809 __wdsp_clr_ready_locked(wdsp,
810 WDSP_SSR_STATUS_WDSP_READY);
811 wdsp_broadcast_event_downseq(wdsp,
812 WDSP_EVENT_PRE_SHUTDOWN,
813 NULL);
814 }
815
816 schedule_work(&wdsp->ssr_work);
817 break;
818
819 case WDSP_SSR_TYPE_CDC_UP:
820 __wdsp_set_ready_locked(wdsp, WDSP_SSR_STATUS_CDC_READY, true);
821 break;
822
823 default:
824 WDSP_ERR(wdsp, "undefined ssr_type %d\n", ssr_type);
825 /* Revert back the ssr_type for undefined events */
826 wdsp->ssr_type = current_ssr_type;
827 break;
828 }
829
830 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->ssr_mutex);
831
832 return 0;
833}
834
835static int wdsp_signal_handler(struct device *wdsp_dev,
836 enum wdsp_signal signal, void *arg)
837{
838 struct wdsp_mgr_priv *wdsp;
839 int ret;
840
841 if (!wdsp_dev)
842 return -EINVAL;
843
844 wdsp = dev_get_drvdata(wdsp_dev);
845 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->api_mutex);
846
847 WDSP_DBG(wdsp, "Raised signal %d", signal);
848
849 switch (signal) {
850 case WDSP_IPC1_INTR:
851 ret = wdsp_unicast_event(wdsp, WDSP_CMPNT_IPC,
852 WDSP_EVENT_IPC1_INTR, NULL);
853 break;
854 case WDSP_ERR_INTR:
855 ret = wdsp_ssr_handler(wdsp, arg, WDSP_SSR_TYPE_WDSP_DOWN);
856 break;
857 case WDSP_CDC_DOWN_SIGNAL:
858 ret = wdsp_ssr_handler(wdsp, arg, WDSP_SSR_TYPE_CDC_DOWN);
859 break;
860 case WDSP_CDC_UP_SIGNAL:
861 ret = wdsp_ssr_handler(wdsp, arg, WDSP_SSR_TYPE_CDC_UP);
862 break;
863 default:
864 ret = -EINVAL;
865 break;
866 }
867
868 if (ret < 0)
869 WDSP_ERR(wdsp, "handling signal %d failed with error %d",
870 signal, ret);
871 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->api_mutex);
872
873 return ret;
874}
875
876static int wdsp_vote_for_dsp(struct device *wdsp_dev,
877 bool vote)
878{
879 struct wdsp_mgr_priv *wdsp;
880 int ret = 0;
881
882 if (!wdsp_dev)
883 return -EINVAL;
884
885 wdsp = dev_get_drvdata(wdsp_dev);
886
887 WDSP_MGR_MUTEX_LOCK(wdsp, wdsp->api_mutex);
888 WDSP_DBG(wdsp, "request %s, current users = %d",
889 vote ? "enable" : "disable", wdsp->dsp_users);
890
891 if (vote) {
892 wdsp->dsp_users++;
893 if (wdsp->dsp_users == 1)
894 ret = wdsp_enable_dsp(wdsp);
895 } else {
896 if (wdsp->dsp_users == 0)
897 goto done;
898
899 wdsp->dsp_users--;
900 if (wdsp->dsp_users == 0)
901 ret = wdsp_disable_dsp(wdsp);
902 }
903
904 if (ret < 0)
905 WDSP_DBG(wdsp, "wdsp %s failed, err = %d",
906 vote ? "enable" : "disable", ret);
907
908done:
909 WDSP_MGR_MUTEX_UNLOCK(wdsp, wdsp->api_mutex);
910 return ret;
911}
912
913static int wdsp_suspend(struct device *wdsp_dev)
914{
915 struct wdsp_mgr_priv *wdsp;
916 int rc = 0, i;
917
918 if (!wdsp_dev) {
919 pr_err("%s: Invalid handle to device\n", __func__);
920 return -EINVAL;
921 }
922
923 wdsp = dev_get_drvdata(wdsp_dev);
924
925 for (i = WDSP_CMPNT_TYPE_MAX - 1; i >= 0; i--) {
926 rc = wdsp_unicast_event(wdsp, i, WDSP_EVENT_SUSPEND, NULL);
927 if (rc < 0) {
928 WDSP_ERR(wdsp, "component %s failed to suspend\n",
929 WDSP_GET_CMPNT_TYPE_STR(i));
930 break;
931 }
932 }
933
934 return rc;
935}
936
937static int wdsp_resume(struct device *wdsp_dev)
938{
939 struct wdsp_mgr_priv *wdsp;
940 int rc = 0, i;
941
942 if (!wdsp_dev) {
943 pr_err("%s: Invalid handle to device\n", __func__);
944 return -EINVAL;
945 }
946
947 wdsp = dev_get_drvdata(wdsp_dev);
948
949 for (i = 0; i < WDSP_CMPNT_TYPE_MAX; i++) {
950 rc = wdsp_unicast_event(wdsp, i, WDSP_EVENT_RESUME, NULL);
951 if (rc < 0) {
952 WDSP_ERR(wdsp, "component %s failed to resume\n",
953 WDSP_GET_CMPNT_TYPE_STR(i));
954 break;
955 }
956 }
957
958 return rc;
959}
960
961static struct wdsp_mgr_ops wdsp_ops = {
962 .register_cmpnt_ops = wdsp_register_cmpnt_ops,
963 .get_dev_for_cmpnt = wdsp_get_dev_for_cmpnt,
964 .get_devops_for_cmpnt = wdsp_get_devops_for_cmpnt,
965 .signal_handler = wdsp_signal_handler,
966 .vote_for_dsp = wdsp_vote_for_dsp,
967 .suspend = wdsp_suspend,
968 .resume = wdsp_resume,
969};
970
971static int wdsp_mgr_compare_of(struct device *dev, void *data)
972{
973 struct wdsp_cmpnt *cmpnt = data;
974
975 /*
976 * First try to match based on of_node, if of_node is not
977 * present, try to match on the dev_name
978 */
979 return ((dev->of_node && dev->of_node == cmpnt->np) ||
980 (cmpnt->cdev_name &&
981 !strcmp(dev_name(dev), cmpnt->cdev_name)));
982}
983
984static void wdsp_mgr_debugfs_init(struct wdsp_mgr_priv *wdsp)
985{
986 wdsp->entry = debugfs_create_dir("wdsp_mgr", NULL);
987 if (IS_ERR_OR_NULL(wdsp->entry))
988 return;
989
990 debugfs_create_bool("panic_on_error", 0644,
991 wdsp->entry, &wdsp->panic_on_error);
992}
993
994static void wdsp_mgr_debugfs_remove(struct wdsp_mgr_priv *wdsp)
995{
996 debugfs_remove_recursive(wdsp->entry);
997 wdsp->entry = NULL;
998}
999
1000static int wdsp_mgr_bind(struct device *dev)
1001{
1002 struct wdsp_mgr_priv *wdsp = dev_get_drvdata(dev);
1003 struct wdsp_cmpnt *cmpnt;
1004 int ret, idx;
1005
1006 wdsp->ops = &wdsp_ops;
1007
1008 /* Setup ramdump device */
1009 wdsp->dump_data.rd_dev = create_ramdump_device("wdsp", dev);
1010 if (!wdsp->dump_data.rd_dev)
1011 dev_info(dev, "%s: create_ramdump_device failed\n", __func__);
1012
1013 ret = component_bind_all(dev, wdsp->ops);
1014 if (ret < 0)
1015 WDSP_ERR(wdsp, "component_bind_all failed %d\n", ret);
1016
1017 /* Make sure all components registered ops */
1018 for (idx = 0; idx < WDSP_CMPNT_TYPE_MAX; idx++) {
1019 cmpnt = WDSP_GET_COMPONENT(wdsp, idx);
1020 if (!cmpnt->cdev || !cmpnt->ops) {
1021 WDSP_ERR(wdsp, "%s did not register ops\n",
1022 WDSP_GET_CMPNT_TYPE_STR(idx));
1023 ret = -EINVAL;
1024 component_unbind_all(dev, wdsp->ops);
1025 break;
1026 }
1027 }
1028
1029 wdsp_mgr_debugfs_init(wdsp);
1030
1031 /* Schedule the work to download image if binding was successful. */
1032 if (!ret)
1033 schedule_work(&wdsp->load_fw_work);
1034
1035 return ret;
1036}
1037
1038static void wdsp_mgr_unbind(struct device *dev)
1039{
1040 struct wdsp_mgr_priv *wdsp = dev_get_drvdata(dev);
1041 struct wdsp_cmpnt *cmpnt;
1042 int idx;
1043
1044 component_unbind_all(dev, wdsp->ops);
1045
1046 wdsp_mgr_debugfs_remove(wdsp);
1047
1048 if (wdsp->dump_data.rd_dev) {
1049 destroy_ramdump_device(wdsp->dump_data.rd_dev);
1050 wdsp->dump_data.rd_dev = NULL;
1051 }
1052
1053 /* Clear all status bits */
1054 wdsp->status = 0x00;
1055
1056 /* clean up the components */
1057 for (idx = 0; idx < WDSP_CMPNT_TYPE_MAX; idx++) {
1058 cmpnt = WDSP_GET_COMPONENT(wdsp, idx);
1059 cmpnt->cdev = NULL;
1060 cmpnt->ops = NULL;
1061 cmpnt->priv_data = NULL;
1062 }
1063}
1064
1065static const struct component_master_ops wdsp_master_ops = {
1066 .bind = wdsp_mgr_bind,
1067 .unbind = wdsp_mgr_unbind,
1068};
1069
1070static void *wdsp_mgr_parse_phandle(struct wdsp_mgr_priv *wdsp,
1071 int index)
1072{
1073 struct device *mdev = wdsp->mdev;
1074 struct device_node *np;
1075 struct wdsp_cmpnt *cmpnt = NULL;
1076 struct of_phandle_args pargs;
1077 u32 value;
1078 int ret;
1079
1080 ret = of_parse_phandle_with_fixed_args(mdev->of_node,
1081 "qcom,wdsp-components", 1,
1082 index, &pargs);
1083 if (ret) {
1084 WDSP_ERR(wdsp, "parse_phandle at index %d failed %d",
1085 index, ret);
1086 return NULL;
1087 }
1088
1089 np = pargs.np;
1090 value = pargs.args[0];
1091
1092 if (value >= WDSP_CMPNT_TYPE_MAX) {
1093 WDSP_ERR(wdsp, "invalid phandle_arg to of_node %s", np->name);
1094 goto done;
1095 }
1096
1097 cmpnt = WDSP_GET_COMPONENT(wdsp, value);
1098 if (cmpnt->np || cmpnt->cdev_name) {
1099 WDSP_ERR(wdsp, "cmpnt %d already added", value);
1100 cmpnt = NULL;
1101 goto done;
1102 }
1103
1104 cmpnt->np = np;
1105 of_property_read_string(np, "qcom,wdsp-cmpnt-dev-name",
1106 &cmpnt->cdev_name);
1107done:
1108 of_node_put(np);
1109 return cmpnt;
1110}
1111
1112static int wdsp_mgr_parse_dt_entries(struct wdsp_mgr_priv *wdsp)
1113{
1114 struct device *dev = wdsp->mdev;
1115 void *match_data;
1116 int ph_idx, ret;
1117
1118 ret = of_property_read_string(dev->of_node, "qcom,img-filename",
1119 &wdsp->img_fname);
1120 if (ret < 0) {
1121 WDSP_ERR(wdsp, "Reading property %s failed, error = %d",
1122 "qcom,img-filename", ret);
1123 return ret;
1124 }
1125
1126 ret = of_count_phandle_with_args(dev->of_node,
1127 "qcom,wdsp-components",
1128 NULL);
1129 if (ret == -ENOENT) {
1130 WDSP_ERR(wdsp, "Property %s not defined in DT",
1131 "qcom,wdsp-components");
1132 goto done;
1133 } else if (ret != WDSP_CMPNT_TYPE_MAX * 2) {
1134 WDSP_ERR(wdsp, "Invalid phandle + arg count %d, expected %d",
1135 ret, WDSP_CMPNT_TYPE_MAX * 2);
1136 ret = -EINVAL;
1137 goto done;
1138 }
1139
1140 ret = 0;
1141
1142 for (ph_idx = 0; ph_idx < WDSP_CMPNT_TYPE_MAX; ph_idx++) {
1143
1144 match_data = wdsp_mgr_parse_phandle(wdsp, ph_idx);
1145 if (!match_data) {
1146 WDSP_ERR(wdsp, "component not found at idx %d", ph_idx);
1147 ret = -EINVAL;
1148 goto done;
1149 }
1150
1151 component_match_add(dev, &wdsp->match,
1152 wdsp_mgr_compare_of, match_data);
1153 }
1154
1155done:
1156 return ret;
1157}
1158
1159static int wdsp_mgr_probe(struct platform_device *pdev)
1160{
1161 struct wdsp_mgr_priv *wdsp;
1162 struct device *mdev = &pdev->dev;
1163 int ret;
1164
1165 wdsp = devm_kzalloc(mdev, sizeof(*wdsp), GFP_KERNEL);
1166 if (!wdsp)
1167 return -ENOMEM;
1168 wdsp->mdev = mdev;
1169 wdsp->seg_list = devm_kzalloc(mdev, sizeof(struct list_head),
1170 GFP_KERNEL);
1171 if (!wdsp->seg_list) {
1172 devm_kfree(mdev, wdsp);
1173 return -ENOMEM;
1174 }
1175
1176 ret = wdsp_mgr_parse_dt_entries(wdsp);
1177 if (ret)
1178 goto err_dt_parse;
1179
1180 INIT_WORK(&wdsp->load_fw_work, wdsp_load_fw_image);
1181 INIT_LIST_HEAD(wdsp->seg_list);
1182 mutex_init(&wdsp->api_mutex);
1183 mutex_init(&wdsp->ssr_mutex);
1184 wdsp->ssr_type = WDSP_SSR_TYPE_NO_SSR;
1185 wdsp->ready_status = WDSP_SSR_STATUS_READY;
1186 INIT_WORK(&wdsp->ssr_work, wdsp_ssr_work_fn);
1187 init_completion(&wdsp->ready_compl);
1188 arch_setup_dma_ops(wdsp->mdev, 0, 0, NULL, 0);
1189 dev_set_drvdata(mdev, wdsp);
1190
1191 ret = component_master_add_with_match(mdev, &wdsp_master_ops,
1192 wdsp->match);
1193 if (ret < 0) {
1194 WDSP_ERR(wdsp, "Failed to add master, err = %d", ret);
1195 goto err_master_add;
1196 }
1197
1198 return 0;
1199
1200err_master_add:
1201 mutex_destroy(&wdsp->api_mutex);
1202 mutex_destroy(&wdsp->ssr_mutex);
1203err_dt_parse:
1204 devm_kfree(mdev, wdsp->seg_list);
1205 devm_kfree(mdev, wdsp);
1206 dev_set_drvdata(mdev, NULL);
1207
1208 return ret;
1209}
1210
1211static int wdsp_mgr_remove(struct platform_device *pdev)
1212{
1213 struct device *mdev = &pdev->dev;
1214 struct wdsp_mgr_priv *wdsp = dev_get_drvdata(mdev);
1215
1216 component_master_del(mdev, &wdsp_master_ops);
1217
1218 mutex_destroy(&wdsp->api_mutex);
1219 mutex_destroy(&wdsp->ssr_mutex);
1220 devm_kfree(mdev, wdsp->seg_list);
1221 devm_kfree(mdev, wdsp);
1222 dev_set_drvdata(mdev, NULL);
1223
1224 return 0;
1225};
1226
1227static const struct of_device_id wdsp_mgr_dt_match[] = {
1228 {.compatible = "qcom,wcd-dsp-mgr" },
1229 { }
1230};
1231
1232static struct platform_driver wdsp_mgr_driver = {
1233 .driver = {
1234 .name = "wcd-dsp-mgr",
1235 .owner = THIS_MODULE,
1236 .of_match_table = of_match_ptr(wdsp_mgr_dt_match),
1237 },
1238 .probe = wdsp_mgr_probe,
1239 .remove = wdsp_mgr_remove,
1240};
1241
1242int wcd_dsp_mgr_init(void)
1243{
1244 return platform_driver_register(&wdsp_mgr_driver);
1245}
1246
1247void wcd_dsp_mgr_exit(void)
1248{
1249 platform_driver_unregister(&wdsp_mgr_driver);
1250}
1251
1252MODULE_DESCRIPTION("WCD DSP manager driver");
1253MODULE_DEVICE_TABLE(of, wdsp_mgr_dt_match);
1254MODULE_LICENSE("GPL v2");