dsp_dev.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572
  1. /*
  2. * Copyright (c) 1997-2015, Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <device.h>
  7. #include <soc.h>
  8. #include <soc_dsp.h>
  9. #include <drivers/dsp.h>
  10. #include "dsp_inner.h"
  11. #include <soc_regs.h>
  12. #include <board_cfg.h>
  13. #include <dvfs.h>
  14. static struct dsp_acts_data dsp_data __in_section_unique(DSP_SHARE_RAM);
  15. #include <acts_ringbuf.h>
  16. struct acts_ringbuf *dsp_dev_get_print_buffer(void);
  17. static void dsp_print_work(struct k_work *work);
  18. static struct acts_ringbuf debug_buff __in_section_unique(DSP_SHARE_RAM);
  19. static u8_t debug_data_buff[0x200] __in_section_unique(DSP_SHARE_RAM);
  20. K_DELAYED_WORK_DEFINE(print_work, dsp_print_work);
  21. #ifdef CONFIG_SOC_NO_PSRAM
  22. __in_section_unique(media.noinit.heap)
  23. #endif
  24. char hex_buf[0x200];
  25. int dsp_print_all_buff(void)
  26. {
  27. int i = 0;
  28. int length;
  29. struct acts_ringbuf *debug_buf = &debug_buff;
  30. if(!debug_buf)
  31. return 0;
  32. length = acts_ringbuf_length(debug_buf);
  33. if(length > acts_ringbuf_size(debug_buf)) {
  34. acts_ringbuf_drop_all(&debug_buff);
  35. length = 0;
  36. }
  37. if(length >= sizeof(hex_buf)){
  38. length = sizeof(hex_buf) - 2;
  39. }
  40. if(length){
  41. //int index = 0;
  42. acts_ringbuf_get(debug_buf, &hex_buf[0], length);
  43. for(i = 0; i < length / 2; i++){
  44. hex_buf[i] = hex_buf[2 * i];
  45. }
  46. hex_buf[i] = 0;
  47. #ifdef CONFIG_DSP_DEBUG_PRINT
  48. printk("%s\n",hex_buf);
  49. #endif
  50. return acts_ringbuf_length(debug_buf);
  51. }
  52. return 0;
  53. }
  54. static void dsp_print_work(struct k_work *work)
  55. {
  56. while(dsp_print_all_buff() > 0) {
  57. ;
  58. }
  59. if (dsp_data.pm_status != DSP_STATUS_POWEROFF){
  60. k_delayed_work_submit(&print_work, K_MSEC(2));
  61. }
  62. }
  63. static void dsp_acts_isr(void *arg);
  64. static void dsp_acts_irq_enable(void);
  65. static void dsp_acts_irq_disable(void);
  66. static void dsp_acts_irq1_disable(void);
  67. void dsp_acts_set_irq1_callback( dsp_acts_irq_callback cb, void *cb_data);
  68. DEVICE_DECLARE(dsp_acts);
  69. int wait_hw_idle_timeout(int usec_to_wait)
  70. {
  71. do {
  72. if (dsp_check_hw_idle())
  73. return 0;
  74. if (usec_to_wait-- <= 0)
  75. break;
  76. k_busy_wait(1);
  77. } while (1);
  78. return -EAGAIN;
  79. }
  80. static int dsp_acts_suspend(struct device *dev)
  81. {
  82. struct dsp_acts_data *dsp_data = dev->data;
  83. if (dsp_data->pm_status != DSP_STATUS_POWERON)
  84. return 0;
  85. /* already idle ? */
  86. if (!get_hw_idle()) {
  87. struct dsp_message message = { .id = DSP_MSG_SUSPEND, };
  88. k_sem_reset(&dsp_data->msg_sem);
  89. if (dsp_acts_send_message(dev, &message))
  90. return -EBUSY;
  91. /* wait message DSP_MSG_BOOTUP */
  92. if (k_sem_take(&dsp_data->msg_sem, SYS_TIMEOUT_MS(20))) {
  93. SYS_LOG_ERR("dsp image <%s> busy\n", dsp_data->images[DSP_IMAGE_MAIN].name);
  94. return -EBUSY;
  95. }
  96. /* make sure hardware is really idle */
  97. if (wait_hw_idle_timeout(10))
  98. SYS_LOG_ERR("DSP wait idle signal timeout\n");
  99. }
  100. dsp_do_wait();
  101. acts_clock_peripheral_disable(CLOCK_ID_DSP);
  102. acts_clock_peripheral_disable(CLOCK_ID_AUDDSPTIMER);
  103. dsp_data->pm_status = DSP_STATUS_SUSPENDED;
  104. SYS_LOG_ERR("DSP suspended\n");
  105. return 0;
  106. }
  107. static int dsp_acts_resume(struct device *dev)
  108. {
  109. struct dsp_acts_data *dsp_data = dev->data;
  110. struct dsp_message message = { .id = DSP_MSG_RESUME, };
  111. if (dsp_data->pm_status != DSP_STATUS_SUSPENDED)
  112. return 0;
  113. acts_clock_peripheral_enable(CLOCK_ID_DSP);
  114. acts_clock_peripheral_enable(CLOCK_ID_AUDDSPTIMER);
  115. dsp_undo_wait();
  116. if (dsp_acts_send_message(dev, &message)) {
  117. SYS_LOG_ERR("DSP resume failed\n");
  118. return -EFAULT;
  119. }
  120. dsp_data->pm_status = DSP_STATUS_POWERON;
  121. SYS_LOG_ERR("DSP resumed\n");
  122. return 0;
  123. }
  124. static int dsp_acts_kick(struct device *dev, uint32_t owner, uint32_t event, uint32_t params)
  125. {
  126. struct dsp_acts_data *dsp_data = dev->data;
  127. const struct dsp_acts_config *dsp_cfg = dev->config;
  128. struct dsp_protocol_userinfo *dsp_userinfo = dsp_cfg->dsp_userinfo;
  129. struct dsp_message message = {
  130. .id = DSP_MSG_KICK,
  131. .owner = owner,
  132. .param1 = event,
  133. .param2 = params,
  134. };
  135. if (dsp_data->pm_status != DSP_STATUS_POWERON)
  136. return -EACCES;
  137. if (dsp_userinfo->task_state != DSP_TASK_SUSPENDED)
  138. return -EINPROGRESS;
  139. return dsp_acts_send_message(dev, &message);
  140. }
  141. static int dsp_acts_request_mem(struct device *dev, int type)
  142. {
  143. return dsp_soc_request_mem(type);
  144. }
  145. static int dsp_acts_release_mem(struct device *dev, int type)
  146. {
  147. return dsp_soc_release_mem(type);
  148. }
  149. /* synchronize cpu and dsp clocks, both base hw timer */
  150. static int dsp_acts_synchronize_clock(void)
  151. {
  152. uint32_t flags;
  153. uint16_t sync_clock_status, sync_clock_status_last;
  154. uint32_t sync_clock_cycles_cpu, sync_clock_cycles_cpu_end;
  155. uint32_t sync_clock_cycles_cpu_diff, sync_clock_cycles_cpu_diff_last = 0xffff0000;
  156. uint32_t sync_clock_cycles_dsp;
  157. uint32_t sync_clock_cycles_offset;
  158. uint32_t try_count = 0, match_count = 0;
  159. uint32_t jitter_cycles_threshold = 16;
  160. printk("synchronize cpu and dsp clocks start\n");
  161. flags = irq_lock();
  162. sync_clock_cycles_cpu = k_cycle_get_32();
  163. *(volatile uint32_t *)(DSP_SYNC_CLOCK_CYCLES_BASE) = sync_clock_cycles_cpu;
  164. *(volatile uint16_t *)(DSP_SYNC_CLOCK_STATUS_BASE) = DSP_SYNC_CLOCK_START;
  165. sync_clock_status_last = DSP_SYNC_CLOCK_START;
  166. while (1) {
  167. sync_clock_status = *(volatile uint16_t *)(DSP_SYNC_CLOCK_STATUS_BASE);
  168. if (sync_clock_status_last == DSP_SYNC_CLOCK_START) {
  169. if (sync_clock_status == DSP_SYNC_CLOCK_REPLY) {
  170. sync_clock_cycles_dsp = *(volatile uint32_t *)(DSP_SYNC_CLOCK_CYCLES_BASE);
  171. sync_clock_cycles_cpu_end = k_cycle_get_32();
  172. sync_clock_cycles_cpu_diff = sync_clock_cycles_cpu_end - sync_clock_cycles_cpu;
  173. if ((sync_clock_cycles_cpu_diff >= sync_clock_cycles_cpu_diff_last &&
  174. sync_clock_cycles_cpu_diff <= sync_clock_cycles_cpu_diff_last + jitter_cycles_threshold) ||
  175. (sync_clock_cycles_cpu_diff <= sync_clock_cycles_cpu_diff_last &&
  176. sync_clock_cycles_cpu_diff + jitter_cycles_threshold >= sync_clock_cycles_cpu_diff_last)) {
  177. match_count++;
  178. } else {
  179. match_count = 0;
  180. }
  181. try_count++;
  182. sync_clock_cycles_cpu_diff_last = sync_clock_cycles_cpu_diff;
  183. if (match_count >= 5) {
  184. sync_clock_cycles_offset = sync_clock_cycles_cpu - sync_clock_cycles_dsp;
  185. *(volatile uint32_t *)(DSP_SYNC_CLOCK_CYCLES_BASE) = sync_clock_cycles_offset;
  186. *(volatile uint16_t *)(DSP_SYNC_CLOCK_STATUS_BASE) = DSP_SYNC_CLOCK_OFFSET;
  187. sync_clock_status_last = DSP_SYNC_CLOCK_OFFSET;
  188. } else {
  189. sync_clock_cycles_cpu = k_cycle_get_32();
  190. *(volatile uint32_t *)(DSP_SYNC_CLOCK_CYCLES_BASE) = sync_clock_cycles_cpu;
  191. *(volatile uint16_t *)(DSP_SYNC_CLOCK_STATUS_BASE) = DSP_SYNC_CLOCK_START;
  192. }
  193. if (try_count > 20) {
  194. try_count = 0;
  195. match_count = 0;
  196. jitter_cycles_threshold += 16;
  197. }
  198. }
  199. } else {
  200. if (sync_clock_status == DSP_SYNC_CLOCK_NULL) {
  201. break;
  202. }
  203. }
  204. }
  205. irq_unlock(flags);
  206. printk("sync clock : 0x%x,%d,%d\n", k_cycle_get_32(), sync_clock_cycles_cpu_diff_last, jitter_cycles_threshold);
  207. return 0;
  208. }
  209. static int dsp_acts_power_on(struct device *dev, void *cmdbuf)
  210. {
  211. struct dsp_acts_data *dsp_data = dev->data;
  212. const struct dsp_acts_config *dsp_cfg = dev->config;
  213. struct dsp_protocol_userinfo *dsp_userinfo = dsp_cfg->dsp_userinfo;
  214. int i;
  215. if (dsp_data->pm_status != DSP_STATUS_POWEROFF)
  216. return 0;
  217. if (dsp_data->images[DSP_IMAGE_MAIN].size == 0) {
  218. SYS_LOG_ERR("%s: no image loaded\n", __func__);
  219. return -EFAULT;
  220. }
  221. if (cmdbuf == NULL) {
  222. SYS_LOG_ERR("%s: must assign a command buffer\n", __func__);
  223. return -EINVAL;
  224. }
  225. memset((void *)DSP_MAILBOX_REG_BASE, 0, 0x80);
  226. dsp_init_clk();
  227. #ifdef CONFIG_DSP_DEBUG_PRINT
  228. acts_ringbuf_init(&debug_buff, debug_data_buff, sizeof(debug_data_buff));
  229. debug_buff.dsp_ptr = mcu_to_dsp_address(debug_buff.cpu_ptr, 0);
  230. k_delayed_work_submit(&print_work, K_MSEC(2));
  231. #endif
  232. /* enable dsp clock*/
  233. acts_clock_peripheral_enable(CLOCK_ID_DSP);
  234. acts_clock_peripheral_enable(CLOCK_ID_AUDDSPTIMER);
  235. /* assert reset */
  236. acts_reset_peripheral_assert(RESET_ID_DSP);
  237. /* set bootargs */
  238. dsp_data->bootargs.command_buffer = mcu_to_dsp_address((uint32_t)cmdbuf, DATA_ADDR);
  239. dsp_data->bootargs.sub_entry_point = dsp_data->images[1].entry_point;
  240. /* assert dsp wait signal */
  241. dsp_do_wait();
  242. dsp_acts_request_mem(dev, 0);
  243. /* intialize shared registers */
  244. dsp_cfg->dsp_mailbox->msg = MAILBOX_MSG(DSP_MSG_NULL, MSG_FLAG_ACK) ;
  245. dsp_cfg->dsp_mailbox->owner = 0;
  246. dsp_cfg->cpu_mailbox->msg = MAILBOX_MSG(DSP_MSG_NULL, MSG_FLAG_ACK) ;
  247. dsp_cfg->cpu_mailbox->owner = 0;
  248. dsp_userinfo->task_state = DSP_TASK_PRESTART;
  249. dsp_userinfo->error_code = DSP_NO_ERROR;
  250. /* set dsp vector_address */
  251. //set_dsp_vector_addr((unsigned int)DSP_ADDR);
  252. dsp_data->pm_status = DSP_STATUS_POWERON;
  253. /* clear all pending */
  254. clear_dsp_all_irq_pending();
  255. /* enable dsp irq */
  256. dsp_acts_irq_enable();
  257. /* deassert dsp wait signal */
  258. dsp_undo_wait();
  259. k_sem_reset(&dsp_data->msg_sem);
  260. /* deassert reset */
  261. acts_reset_peripheral_deassert(RESET_ID_DSP);
  262. /* wait message DSP_MSG_STATE_CHANGED.DSP_TASK_STARTED */
  263. if (k_sem_take(&dsp_data->msg_sem, SYS_TIMEOUT_MS(100))) {
  264. SYS_LOG_ERR("dsp image <%s> cannot boot up\n", dsp_data->images[DSP_IMAGE_MAIN].name);
  265. dsp_dump_info();
  266. goto cleanup;
  267. }
  268. if (dsp_data->need_sync_clock == 'C')
  269. {
  270. dsp_acts_synchronize_clock();
  271. }
  272. /* FIXME: convert userinfo address from dsp to cpu here, since
  273. * dsp will never touch it again.
  274. */
  275. dsp_userinfo->func_table = dsp_to_mcu_address(dsp_userinfo->func_table, DATA_ADDR);
  276. for (i = 0; i < dsp_userinfo->func_size; i++) {
  277. volatile uint32_t addr = dsp_userinfo->func_table + i * 4;
  278. if (addr > 0) /* NULL, no information provided */
  279. sys_write32(dsp_to_mcu_address(sys_read32(addr), DATA_ADDR), addr);
  280. }
  281. //SYS_LOG_ERR("DSP power on %d %p\n", dsp_userinfo->func_size, dsp_userinfo->func_table);
  282. return 0;
  283. cleanup:
  284. dsp_acts_irq_disable();
  285. acts_clock_peripheral_disable(CLOCK_ID_DSP);
  286. acts_clock_peripheral_disable(CLOCK_ID_AUDDSPTIMER);
  287. acts_reset_peripheral_assert(RESET_ID_DSP);
  288. return -ETIMEDOUT;
  289. }
  290. static int dsp_acts_power_off(struct device *dev)
  291. {
  292. const struct dsp_acts_config *dsp_cfg = dev->config;
  293. struct dsp_acts_data *dsp_data = dev->data;
  294. if (dsp_data->pm_status == DSP_STATUS_POWEROFF)
  295. return 0;
  296. /* assert dsp wait signal */
  297. dsp_do_wait();
  298. /* disable dsp irq */
  299. dsp_acts_irq_disable();
  300. /* disable dsp irq 1 */
  301. dsp_acts_irq1_disable();
  302. /* assert reset dsp module */
  303. acts_reset_peripheral_assert(RESET_ID_DSP);
  304. /* disable dsp clock*/
  305. acts_clock_peripheral_disable(CLOCK_ID_DSP);
  306. acts_clock_peripheral_disable(CLOCK_ID_AUDDSPTIMER);
  307. /* deassert dsp wait signal */
  308. dsp_undo_wait();
  309. /* clear page mapping */
  310. clear_dsp_pageaddr();
  311. dsp_cfg->dsp_userinfo->task_state = DSP_TASK_DEAD;
  312. dsp_data->pm_status = DSP_STATUS_POWEROFF;
  313. dsp_print_all_buff();
  314. acts_ringbuf_drop_all(&debug_buff);
  315. #ifdef CONFIG_DSP_DEBUG_PRINT
  316. k_delayed_work_cancel(&print_work);
  317. #endif
  318. SYS_LOG_INF("DSP power off\n");
  319. return 0;
  320. }
  321. static int acts_request_userinfo(struct device *dev, int request, void *info)
  322. {
  323. const struct dsp_acts_config *dsp_cfg = dev->config;
  324. struct dsp_protocol_userinfo *dsp_userinfo = dsp_cfg->dsp_userinfo;
  325. union {
  326. struct dsp_request_session *ssn;
  327. struct dsp_request_function *func;
  328. } req_info;
  329. switch (request) {
  330. case DSP_REQUEST_TASK_STATE:
  331. *(int *)info = dsp_userinfo->task_state;
  332. break;
  333. case DSP_REQUEST_ERROR_CODE:
  334. *(int *)info = dsp_userinfo->error_code;
  335. break;
  336. case DSP_REQUEST_SESSION_INFO:
  337. req_info.ssn = info;
  338. req_info.ssn->func_enabled = dsp_userinfo->func_enabled;
  339. req_info.ssn->func_runnable = dsp_userinfo->func_runnable;
  340. req_info.ssn->func_counter = dsp_userinfo->func_counter;
  341. req_info.ssn->info = (void *)dsp_to_mcu_address(dsp_userinfo->ssn_info, DATA_ADDR);
  342. break;
  343. case DSP_REQUEST_FUNCTION_INFO:
  344. req_info.func = info;
  345. if (req_info.func->id < dsp_userinfo->func_size)
  346. req_info.func->info = (void *)sys_read32(dsp_userinfo->func_table + req_info.func->id * 4);
  347. break;
  348. case DSP_REQUEST_USER_DEFINED:
  349. {
  350. uint32_t index = *(uint32_t*)info;
  351. uint32_t debug_info = *(volatile uint32_t*)(DSP_DEBUG_REGION_REGISTER_BASE + 4 * index);
  352. *(uint32_t*)info = debug_info;
  353. break;
  354. }
  355. default:
  356. return -EINVAL;
  357. }
  358. return 0;
  359. }
  360. static void dsp_acts_isr(void *arg)
  361. {
  362. /* clear irq pending bits */
  363. clear_dsp_irq_pending(IRQ_ID_DSP);
  364. dsp_acts_recv_message(arg);
  365. }
  366. static void dsp_acts_isr1(void *arg)
  367. {
  368. struct device *dev = (struct device *)arg;
  369. struct dsp_acts_data *dsp_data = dev->data;
  370. /* clear irq pending bits */
  371. clear_dsp_irq1_pending(IRQ_ID_DSP1);
  372. if (dsp_data->cb) {
  373. dsp_data->cb(dsp_data->cb_data, 1, NULL);
  374. }
  375. }
  376. void dsp_acts_set_irq1_callback(dsp_acts_irq_callback cb, void *cb_data)
  377. {
  378. struct device *dev = (struct device *)DEVICE_GET(dsp_acts);
  379. if (!dev) {
  380. SYS_LOG_ERR("dev handle NULL");
  381. return;
  382. }
  383. struct dsp_acts_data *dsp_data = dev->data;
  384. if (dsp_data) {
  385. dsp_data->cb = cb;
  386. dsp_data->cb_data = cb_data;
  387. }
  388. }
  389. static int dsp_acts_init(const struct device *dev)
  390. {
  391. const struct dsp_acts_config *dsp_cfg = dev->config;
  392. struct dsp_acts_data *dsp_data = dev->data;
  393. acts_ringbuf_init(&debug_buff, debug_data_buff, sizeof(debug_data_buff));
  394. debug_buff.dsp_ptr = mcu_to_dsp_address(debug_buff.cpu_ptr, 0);
  395. dsp_data->bootargs.debug_buf = mcu_to_dsp_address(POINTER_TO_UINT(&debug_buff), DATA_ADDR);
  396. dsp_cfg->dsp_userinfo->task_state = DSP_TASK_DEAD;
  397. k_sem_init(&dsp_data->msg_sem, 0, 1);
  398. k_mutex_init(&dsp_data->msg_mutex);
  399. dsp_data->pm_status = DSP_STATUS_POWEROFF;
  400. memset(dsp_cfg->dsp_mailbox, 0, sizeof(struct dsp_protocol_mailbox));
  401. memset(dsp_cfg->cpu_mailbox, 0, sizeof(struct dsp_protocol_mailbox));
  402. memset(dsp_cfg->dsp_userinfo, 0, sizeof(struct dsp_protocol_userinfo));
  403. return 0;
  404. }
  405. static const struct dsp_acts_config dsp_acts_config = {
  406. .dsp_mailbox = (struct dsp_protocol_mailbox *)DSP_M2D_MAILBOX_REGISTER_BASE,
  407. .cpu_mailbox = (struct dsp_protocol_mailbox *)DSP_D2M_MAILBOX_REGISTER_BASE,
  408. .dsp_userinfo = (struct dsp_protocol_userinfo *)DSP_USER_REGION_REGISTER_BASE,
  409. };
  410. const struct dsp_driver_api dsp_drv_api = {
  411. .poweron = dsp_acts_power_on,
  412. .poweroff = dsp_acts_power_off,
  413. .suspend = dsp_acts_suspend,
  414. .resume = dsp_acts_resume,
  415. .kick = dsp_acts_kick,
  416. .register_message_handler = dsp_acts_register_message_handler,
  417. .unregister_message_handler = dsp_acts_unregister_message_handler,
  418. .request_image = dsp_acts_request_image,
  419. .release_image = dsp_acts_release_image,
  420. .request_mem = dsp_acts_request_mem,
  421. .release_mem = dsp_acts_release_mem,
  422. .send_message = dsp_acts_send_message,
  423. .request_userinfo = acts_request_userinfo,
  424. };
  425. DEVICE_DEFINE(dsp_acts, CONFIG_DSP_NAME,
  426. dsp_acts_init, NULL, &dsp_data, &dsp_acts_config,
  427. POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE, &dsp_drv_api);
  428. static void dsp_acts_irq_enable(void)
  429. {
  430. IRQ_CONNECT(IRQ_ID_DSP, CONFIG_DSP_IRQ_PRI, dsp_acts_isr, DEVICE_GET(dsp_acts), 0);
  431. irq_enable(IRQ_ID_DSP);
  432. IRQ_CONNECT(IRQ_ID_DSP1, CONFIG_DSP_IRQ_PRI, dsp_acts_isr1, DEVICE_GET(dsp_acts), 0);
  433. irq_enable(IRQ_ID_DSP1);
  434. }
  435. static void dsp_acts_irq_disable(void)
  436. {
  437. irq_disable(IRQ_ID_DSP);
  438. }
  439. static void dsp_acts_irq1_disable(void)
  440. {
  441. irq_disable(IRQ_ID_DSP1);
  442. }
  443. #define DEV_DATA(dev) \
  444. ((struct dsp_acts_data * const)(dev)->data)
  445. struct acts_ringbuf *dsp_dev_get_print_buffer(void)
  446. {
  447. struct dsp_acts_data *data = DEV_DATA(DEVICE_GET(dsp_acts));
  448. if(!data->bootargs.debug_buf){
  449. return NULL;
  450. }
  451. return (struct acts_ringbuf *)dsp_to_mcu_address(data->bootargs.debug_buf,DATA_ADDR);
  452. }