dsp_mailbox.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. /**
  2. * Copyright (c) 1997-2015, Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <device.h>
  8. #include <drivers/dsp.h>
  9. #include <soc_dsp.h>
  10. #include "dsp_inner.h"
  11. #include <init.h>
  12. #include "os_common_api.h"
  13. #define CONFIG_DSP_WORK_Q_STACK_SIZE (2048)
  14. #define CONFIG_DSP_WORK_Q_PRIORITY (-2)
  15. static struct k_work_q dsp_workq;
  16. static K_KERNEL_STACK_DEFINE(dsp_workq_stack, CONFIG_DSP_WORK_Q_STACK_SIZE);
  17. struct dsp_message pagemiss_message;
  18. struct device *pagemiss_dev = NULL;
  19. int dsp_acts_register_message_handler(struct device *dev, dsp_message_handler handler)
  20. {
  21. struct dsp_acts_data *dsp_data = dev->data;
  22. unsigned int irq_key;
  23. irq_key = irq_lock();
  24. dsp_data->msg_handler = handler;
  25. irq_unlock(irq_key);
  26. return 0;
  27. }
  28. int dsp_acts_unregister_message_handler(struct device *dev)
  29. {
  30. struct dsp_acts_data *dsp_data = dev->data;
  31. unsigned int irq_key;
  32. irq_key = irq_lock();
  33. dsp_data->msg_handler = NULL;
  34. irq_unlock(irq_key);
  35. return 0;
  36. }
  37. int dsp_acts_send_message_response(struct device *dev, struct dsp_message *message, int res)
  38. {
  39. const struct dsp_acts_config *dsp_cfg = dev->config;
  40. //struct dsp_acts_data *dsp_data = dev->data;
  41. struct dsp_protocol_mailbox *mailbox = dsp_cfg->cpu_mailbox;
  42. uint32_t status = MSG_STATUS(mailbox->msg);
  43. /* FIXME: force set result fail */
  44. if (res < 0)
  45. message->result = DSP_FAIL;
  46. /* confirm message received */
  47. status &= ~(MSG_FLAG_BUSY | MSG_FLAG_DONE | MSG_FLAG_RPLY | MSG_FLAG_FAIL);
  48. status |= MSG_FLAG_ACK;
  49. switch (message->result) {
  50. case DSP_REPLY:
  51. /* always copy back result */
  52. mailbox->param1 = message->param1;
  53. mailbox->param2 = message->param2;
  54. status |= MSG_FLAG_RPLY;
  55. case DSP_DONE:
  56. status |= MSG_FLAG_DONE;
  57. break;
  58. case DSP_FAIL:
  59. status |= MSG_FLAG_FAIL;
  60. break;
  61. default:
  62. break;
  63. }
  64. mailbox->msg = MAILBOX_MSG(message->id, status);
  65. printk("dsp_acts_send_message_response id %d status %x \n",message->id, status);
  66. return res;
  67. }
  68. static void dsp_page_miss_handle(struct k_work *work)
  69. {
  70. int res = dsp_acts_handle_image_pagemiss(pagemiss_dev, pagemiss_message.param1);
  71. dsp_acts_send_message_response(pagemiss_dev, &pagemiss_message, res);
  72. }
  73. K_WORK_DEFINE(dsp_page_miss_work, dsp_page_miss_handle);
  74. /* This function called in irq context */
  75. int dsp_acts_recv_message(struct device *dev)
  76. {
  77. const struct dsp_acts_config *dsp_cfg = dev->config;
  78. struct dsp_acts_data *dsp_data = dev->data;
  79. struct dsp_protocol_mailbox *mailbox = dsp_cfg->cpu_mailbox;
  80. struct dsp_message message = {
  81. .id = MSG_ID(mailbox->msg),
  82. .result = DSP_DONE,
  83. .owner = mailbox->owner,
  84. .param1 = mailbox->param1,
  85. .param2 = mailbox->param2,
  86. };
  87. uint32_t status = MSG_STATUS(mailbox->msg);
  88. int res = 0;
  89. if (dsp_data->pm_status == DSP_STATUS_POWEROFF){
  90. printk("%s: deaded\n", __func__);
  91. return -ENODEV;
  92. }
  93. if (!(status & MSG_FLAG_BUSY)) {
  94. printk("%s: busy flag of msg (%u:%u) not set\n", __func__,
  95. message.owner, message.id);
  96. return -EINVAL;
  97. }
  98. if (status & MSG_FLAG_ACK) {
  99. printk("%s: ack of msg (%u:%u) already set\n", __func__,
  100. message.owner, message.id);
  101. return -EINVAL;
  102. }
  103. #if 0
  104. printk("msg_id %d param %x %x\n", message.id, message.param1, message.param2);
  105. #endif
  106. switch (message.id) {
  107. case DSP_MSG_REQUEST_BOOTARGS:
  108. message.param1 = mcu_to_dsp_address((uint32_t)&dsp_data->bootargs, DATA_ADDR);
  109. message.result = DSP_REPLY;
  110. break;
  111. case DSP_MSG_STATE_CHANGED:
  112. switch (message.param1) {
  113. case DSP_TASK_STARTED:
  114. printk("%s: started param2 %d \n", __func__, message.param2);
  115. if (message.param2 == DSP_NEED_SYNC_CLOCK) {
  116. dsp_data->need_sync_clock = 'C';
  117. message.param1 = DSP_NEED_SYNC_CLOCK;
  118. message.param2 = 0;
  119. message.result = DSP_REPLY;
  120. } else {
  121. dsp_data->need_sync_clock = 0;
  122. message.param1 = 0;
  123. message.param2 = 0;
  124. message.result = DSP_REPLY;
  125. }
  126. break;
  127. case DSP_TASK_SUSPENDED:
  128. printk("%s: suspended\n", __func__);
  129. break;
  130. case DSP_TASK_RESUMED:
  131. printk("%s: resumed\n", __func__);
  132. break;
  133. default:
  134. break;
  135. }
  136. k_sem_give(&dsp_data->msg_sem);
  137. break;
  138. case DSP_MSG_PAGE_MISS:
  139. pagemiss_dev = dev;
  140. memcpy(&pagemiss_message, &message, sizeof(struct dsp_message));
  141. os_work_submit_to_queue(&dsp_workq, &dsp_page_miss_work);
  142. return res;
  143. case DSP_MSG_PAGE_FLUSH:
  144. res = dsp_acts_handle_image_pageflush(dev, message.param1);
  145. break;
  146. case DSP_MSG_NULL:
  147. break;
  148. case DSP_MSG_KICK:
  149. default:
  150. if (dsp_data->msg_handler) {
  151. res = dsp_data->msg_handler(&message);
  152. } else {
  153. printk("%s: unexpected msg %u\n", __func__, message.id);
  154. res = -ENOMSG;
  155. }
  156. break;
  157. }
  158. res = dsp_acts_send_message_response(dev, &message, res);
  159. return res;
  160. }
  161. static int wait_ack_timeout(struct dsp_protocol_mailbox *mailbox, int usec_to_wait)
  162. {
  163. do {
  164. uint32_t status = MSG_STATUS(mailbox->msg);
  165. if ((status & MSG_FLAG_ACK) || (usec_to_wait-- <= 0))
  166. break;
  167. k_busy_wait(1);
  168. } while (1);
  169. return usec_to_wait;
  170. }
  171. int dsp_acts_send_message(struct device *dev, struct dsp_message *message)
  172. {
  173. struct dsp_acts_data *dsp_data = dev->data;
  174. const struct dsp_acts_config *dsp_cfg = dev->config;
  175. struct dsp_protocol_mailbox *mailbox = dsp_cfg->dsp_mailbox;
  176. uint32_t status = MSG_STATUS(mailbox->msg);
  177. int ret = 0;
  178. if (dsp_data->pm_status == DSP_STATUS_POWEROFF)
  179. return -EFAULT;
  180. if (!(status & MSG_FLAG_ACK)) {
  181. printk("%s: ack of msg (%u:%u) not yet set by dsp\n", __func__,
  182. mailbox->owner, MSG_ID(mailbox->msg));
  183. return -EBUSY;
  184. }
  185. if (status & MSG_FLAG_BUSY) {
  186. printk("%s: busy flag of msg (%u:%u) not yet cleared by dsp\n",
  187. __func__, mailbox->owner, MSG_ID(mailbox->msg));
  188. return -EBUSY;
  189. }
  190. if (k_is_in_isr()) {
  191. printk("%s: send msg (%u:%u) in isr\n", __func__,
  192. message->owner, message->id);
  193. } else {
  194. k_mutex_lock(&dsp_data->msg_mutex, K_FOREVER);
  195. }
  196. mailbox->msg = MAILBOX_MSG(message->id, MSG_FLAG_BUSY);
  197. mailbox->owner = message->owner;
  198. mailbox->param1 = message->param1;
  199. mailbox->param2 = message->param2;
  200. /* trigger irq to dsp */
  201. mcu_trigger_irq_to_dsp();
  202. /* Wait for the ack bit: 1ms timeout (1us * 1000) */
  203. wait_ack_timeout(mailbox, 1000);
  204. /* de-trigger irq to dsp */
  205. mcu_untrigger_irq_to_dsp();
  206. status = MSG_STATUS(mailbox->msg);
  207. if (!(status & MSG_FLAG_ACK)) {
  208. printk("%s: ack of msg %u wait timeout\n", __func__, message->id);
  209. message->result = DSP_NOACK;
  210. ret = -ETIMEDOUT;
  211. goto EXIT;
  212. } else if (status & MSG_FLAG_FAIL) {
  213. message->result = DSP_FAIL;
  214. ret = -ENOTSUP;
  215. goto EXIT;
  216. } else if (status & MSG_FLAG_RPLY) {
  217. message->param1 = mailbox->param1;
  218. message->param2 = mailbox->param2;
  219. message->result = DSP_REPLY;
  220. } else if (status & MSG_FLAG_DONE) {
  221. message->result = DSP_DONE;
  222. } else {
  223. message->result = DSP_INPROGRESS;
  224. }
  225. EXIT:
  226. if (!k_is_in_isr())
  227. k_mutex_unlock(&dsp_data->msg_mutex);
  228. return ret;
  229. }
  230. static int dsp_work_q_init(const struct device *dev)
  231. {
  232. ARG_UNUSED(dev);
  233. struct k_work_queue_config cfg = {
  234. .name = "dspworkq",
  235. .no_yield = false,
  236. };
  237. k_work_queue_start(&dsp_workq,
  238. dsp_workq_stack,
  239. K_KERNEL_STACK_SIZEOF(dsp_workq_stack),
  240. CONFIG_DSP_WORK_Q_PRIORITY, &cfg);
  241. return 0;
  242. }
  243. SYS_INIT(dsp_work_q_init, POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);