dsp_hal.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. /*
  2. * Copyright (c) 1997-2015, Wind River Systems, Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <assert.h>
  7. #include <errno.h>
  8. #include <devicetree.h>
  9. #include <string.h>
  10. #include <mem_manager.h>
  11. #include "dsp_inner.h"
  12. #include <sdfs.h>
  13. extern const struct dsp_imageinfo *dsp_create_image(const char *name);
  14. extern void dsp_free_image(const struct dsp_imageinfo *image);
  15. static struct dsp_session *global_session = NULL;
  16. static unsigned int global_uuid = 0;
  17. int dsp_session_get_state(struct dsp_session *session)
  18. {
  19. int state = 0;
  20. dsp_request_userinfo(session->dev, DSP_REQUEST_TASK_STATE, &state);
  21. return state;
  22. }
  23. int dsp_session_get_error(struct dsp_session *session)
  24. {
  25. int error = 0;
  26. dsp_request_userinfo(session->dev, DSP_REQUEST_ERROR_CODE, &error);
  27. return error;
  28. }
  29. uint32_t dsp_session_get_debug_info(struct dsp_session *session, uint32_t index)
  30. {
  31. dsp_request_userinfo(session->dev, DSP_REQUEST_USER_DEFINED, &index);
  32. return index;
  33. }
  34. void dsp_session_dump(struct dsp_session *session)
  35. {
  36. struct dsp_request_session request;
  37. if (!session) {
  38. /* implicitly point to global session */
  39. session = global_session;
  40. if (!session)
  41. return;
  42. }
  43. dsp_request_userinfo(session->dev, DSP_REQUEST_SESSION_INFO, &request);
  44. SYS_LOG_ERR("\n---------------------------dsp dump---------------------------\n");
  45. SYS_LOG_ERR("session (id=%u, uuid=%u):\n", session->id, session->uuid);
  46. SYS_LOG_ERR("\tstate=%d\n", dsp_session_get_state(session));
  47. SYS_LOG_ERR("\terror=0x%x\n", dsp_session_get_error(session));
  48. SYS_LOG_ERR("\tfunc_allowed=0x%x\n", session->func_allowed);
  49. SYS_LOG_ERR("\tfunc_enabled=0x%x\n", request.func_enabled);
  50. SYS_LOG_ERR("\tfunc_runnable=0x%x\n", request.func_runnable);
  51. SYS_LOG_ERR("\tfunc_counter=0x%x\n", request.func_counter);
  52. if (request.info)
  53. dsp_session_dump_info(session, request.info);
  54. if (request.func_enabled) {
  55. for (int i = 0; i < DSP_NUM_FUNCTIONS; i++) {
  56. if (request.func_enabled & DSP_FUNC_BIT(i))
  57. dsp_session_dump_function(session, i);
  58. }
  59. }
  60. SYS_LOG_ERR("--------------------------------------------------------------\n\n");
  61. }
  62. static int dsp_session_set_id(struct dsp_session *session)
  63. {
  64. struct dsp_session_id_set set = {
  65. .id = session->id,
  66. .uuid = session->uuid,
  67. .func_allowed = session->func_allowed,
  68. };
  69. struct dsp_command *command =
  70. dsp_command_alloc(DSP_CMD_SET_SESSION, NULL, sizeof(set), &set);
  71. if (command == NULL)
  72. return -ENOMEM;
  73. dsp_session_submit_command(session, command);
  74. dsp_command_free(command);
  75. return 0;
  76. }
  77. static inline int dsp_session_get(struct dsp_session *session)
  78. {
  79. return atomic_inc(&session->open_refcnt);
  80. }
  81. static inline int dsp_session_put(struct dsp_session *session)
  82. {
  83. return atomic_dec(&session->open_refcnt);
  84. }
  85. static void dsp_session_destroy(struct dsp_session *session);
  86. static struct acts_ringbuf cmd_buff __in_section_unique(DSP_SHARE_RAM);
  87. static u8_t dsp_cmd_buff[DSP_SESSION_COMMAND_BUFFER_SIZE] __in_section_unique(DSP_SHARE_RAM);
  88. static struct dsp_session static_globle_session __in_section_unique(DSP_SHARE_RAM);
  89. static struct dsp_session *dsp_session_create(struct dsp_session_info *info)
  90. {
  91. struct dsp_session *session = NULL;
  92. if (info == NULL || info->main_dsp == NULL)
  93. return NULL;
  94. session = &static_globle_session;
  95. memset(session, 0, sizeof(struct dsp_session));
  96. if (session == NULL) {
  97. SYS_LOG_ERR("malloc failed %d", sizeof(*session));
  98. return NULL;
  99. }
  100. memset(session, 0, sizeof(*session));
  101. /* find dsp device */
  102. session->dev = (struct device *)device_get_binding(CONFIG_DSP_NAME);
  103. if (session->dev == NULL) {
  104. SYS_LOG_ERR("cannot find device \"%s\"", CONFIG_DSP_ACTS_DEV_NAME);
  105. goto fail_exit;
  106. }
  107. /* find images */
  108. session->images[DSP_IMAGE_MAIN] = dsp_create_image(info->main_dsp);
  109. if (!session->images[DSP_IMAGE_MAIN]) {
  110. goto fail_exit;
  111. }
  112. if (info->sub_dsp) {
  113. session->images[DSP_IMAGE_SUB] = dsp_create_image(info->sub_dsp);
  114. if (!session->images[DSP_IMAGE_SUB]) {
  115. //goto fail_exit;
  116. }
  117. }
  118. /* allocate command buffer */
  119. acts_ringbuf_init(&cmd_buff, dsp_cmd_buff, DSP_SESSION_COMMAND_BUFFER_SIZE);
  120. cmd_buff.dsp_ptr = mcu_to_dsp_address(cmd_buff.cpu_ptr, 0);
  121. session->cmdbuf.cur_seq = 0;
  122. session->cmdbuf.alloc_seq = 1;
  123. session->cmdbuf.cpu_bufptr = (uint32_t)&cmd_buff;
  124. session->cmdbuf.dsp_bufptr = mcu_to_dsp_address(session->cmdbuf.cpu_bufptr, 0);
  125. k_sem_init(&session->sem, 0, 1);
  126. k_mutex_init(&session->mutex);
  127. atomic_set(&session->open_refcnt, 0);
  128. #ifdef ENABLE_SESSION_BEGIN_END_REFCNT
  129. atomic_set(&session->run_refcnt, 0);
  130. #endif
  131. session->id = info->type;
  132. session->uuid = ++global_uuid;
  133. SYS_LOG_INF("session %u created (uuid=%u)", session->id, session->uuid);
  134. return session;
  135. fail_exit:
  136. dsp_session_destroy(session);
  137. return NULL;
  138. }
  139. static void dsp_session_destroy(struct dsp_session *session)
  140. {
  141. if (atomic_get(&session->open_refcnt) != 0)
  142. SYS_LOG_ERR("session not closed yet (ref=%d)", atomic_get(&session->open_refcnt));
  143. for (int i = 0; i < ARRAY_SIZE(session->images); i++) {
  144. if (session->images[i])
  145. dsp_free_image(session->images[i]);
  146. }
  147. //static malloc, should no be free
  148. //if (session->cmdbuf.cpu_bufptr)
  149. // acts_ringbuf_free((struct acts_ringbuf *)(session->cmdbuf.cpu_bufptr));
  150. SYS_LOG_INF("session %u destroyed (uuid=%u)", session->id, session->uuid);
  151. //mem_free(session);
  152. }
  153. static int dsp_session_message_handler(struct dsp_message *message)
  154. {
  155. if (!global_session || message->id != DSP_MSG_KICK)
  156. return -ENOTTY;
  157. switch (message->param1) {
  158. case DSP_EVENT_FENCE_SYNC:
  159. if (message->param2){
  160. if(k_sem_count_get((struct k_sem *)message->param2) > 1){
  161. SYS_LOG_ERR("invalid ksem %p", (struct k_sem *)message->param2);
  162. }else{
  163. k_sem_give((struct k_sem *)message->param2);
  164. }
  165. }
  166. break;
  167. case DSP_EVENT_NEW_CMD:
  168. case DSP_EVENT_NEW_DATA:
  169. k_sem_give(&global_session->sem);
  170. break;
  171. default:
  172. return -ENOTTY;
  173. }
  174. return 0;
  175. }
  176. #ifdef CONFIG_DSP_LOAD_ROM
  177. #define DSP_PROM_ADDR 0x30010000
  178. #define DSP_SDKPROM_ADDR (0x30010000 + 0x12E00)
  179. #define DSP_DROM_ADDR 0x30000000
  180. #define DSP_PROM_NAME "DPROM.bin"
  181. #define DSP_DROM_NAME "DDROM.bin"
  182. #define DSP_SDKDROM_NAME "sdk_rom.bin"
  183. static int rom_image_loaded = 0;
  184. static int dsp_load_rom_image(const char *name, unsigned int addr)
  185. {
  186. void *ptr;
  187. int size;
  188. SYS_LOG_ERR("dsp image \"%s\"\n", name);
  189. if (sd_fmap(name, (void **)&ptr, (int *)&size)) {
  190. SYS_LOG_ERR("cannot find dsp image \"%s\"\n", name);
  191. return -1;
  192. }
  193. memcpy((void *)addr, ptr, size);
  194. return 0;
  195. }
  196. #endif
  197. static int dsp_session_open(struct dsp_session *session, struct dsp_session_info *info)
  198. {
  199. int res = 0;
  200. if (dsp_session_get(session) != 0) {
  201. //SYS_LOG_ERR("session is opened already! uuid: 0x%x\n", session->uuid);
  202. return -EALREADY;
  203. }
  204. if (info == NULL) {
  205. res = -EINVAL;
  206. goto fail_out;
  207. }
  208. /* load dsp rom image */
  209. #ifdef CONFIG_DSP_LOAD_ROM
  210. if( !rom_image_loaded ) {
  211. if(dsp_load_rom_image(DSP_PROM_NAME, DSP_PROM_ADDR) < 0) {
  212. res = -1;
  213. goto fail_out;
  214. }
  215. if(dsp_load_rom_image(DSP_DROM_NAME, DSP_DROM_ADDR) < 0) {
  216. res = -1;
  217. goto fail_out;
  218. }
  219. #if 0
  220. if(dsp_load_rom_image(DSP_SDKDROM_NAME, DSP_SDKPROM_ADDR) < 0) {
  221. res = -1;
  222. goto fail_out;
  223. }
  224. #endif
  225. rom_image_loaded = 1;
  226. }
  227. #endif
  228. /* load main dsp image */
  229. res = dsp_request_image(session->dev, session->images[DSP_IMAGE_MAIN], DSP_IMAGE_MAIN);
  230. if (res) {
  231. SYS_LOG_ERR("failed to load main image \"%s\"", session->images[DSP_IMAGE_MAIN]->name);
  232. goto fail_out;
  233. }
  234. /* load sub dsp image */
  235. if (session->images[DSP_IMAGE_SUB]) {
  236. res = dsp_request_image(session->dev, session->images[DSP_IMAGE_SUB], DSP_IMAGE_SUB);
  237. if (res) {
  238. SYS_LOG_ERR("failed to load sub image \"%s\"", session->images[DSP_IMAGE_SUB]->name);
  239. goto fail_release_main;
  240. }
  241. }
  242. /* reset and register command buffer */
  243. session->cmdbuf.cur_seq = 0;
  244. session->cmdbuf.alloc_seq = 1;
  245. acts_ringbuf_reset((struct acts_ringbuf *)(session->cmdbuf.cpu_bufptr));
  246. /* power on */
  247. res = dsp_poweron(session->dev, &session->cmdbuf);
  248. if (res) {
  249. SYS_LOG_ERR("dsp_poweron failed");
  250. goto fail_release_sub;
  251. }
  252. /* configure session id to dsp */
  253. session->func_allowed = info->func_allowed;
  254. res = dsp_session_set_id(session);
  255. if (res) {
  256. SYS_LOG_ERR("dsp_session_set_id failed");
  257. dsp_poweroff(session->dev);
  258. goto fail_release_sub;
  259. }
  260. /* register message handler only after everything OK */
  261. k_sem_reset(&session->sem);
  262. dsp_register_message_handler(session->dev, dsp_session_message_handler);
  263. SYS_LOG_INF("session %u opened (uuid=%u, allowed=0x%x)",
  264. session->id, session->uuid, info->func_allowed);
  265. return 0;
  266. fail_release_sub:
  267. if (session->images[DSP_IMAGE_SUB])
  268. dsp_release_image(session->dev, DSP_IMAGE_SUB);
  269. fail_release_main:
  270. dsp_release_image(session->dev, DSP_IMAGE_MAIN);
  271. fail_out:
  272. dsp_session_put(session);
  273. return res;
  274. }
  275. static int dsp_session_close(struct dsp_session *session)
  276. {
  277. if (dsp_session_put(session) != 1)
  278. return -EBUSY;
  279. if (session->cmdbuf.cur_seq + 1 != session->cmdbuf.alloc_seq)
  280. SYS_LOG_WRN("session %u command not finished %d %d ", session->id,session->cmdbuf.cur_seq,session->cmdbuf.alloc_seq);
  281. dsp_unregister_message_handler(session->dev);
  282. dsp_poweroff(session->dev);
  283. for (int i = 0; i < ARRAY_SIZE(session->images); i++) {
  284. if (session->images[i])
  285. dsp_release_image(session->dev, i);
  286. }
  287. dsp_release_mem(session->dev, 0);
  288. SYS_LOG_INF("session %u closed (uuid=%u)", session->id, session->uuid);
  289. return 0;
  290. }
  291. struct dsp_session *dsp_open_global_session(struct dsp_session_info *info)
  292. {
  293. int res;
  294. if (global_session == NULL) {
  295. if(info){
  296. global_session = dsp_session_create(info);
  297. if (global_session == NULL) {
  298. SYS_LOG_ERR("failed to create global session");
  299. return NULL;
  300. }
  301. }else{
  302. return NULL;
  303. }
  304. }
  305. if(info){
  306. res = dsp_session_open(global_session, info);
  307. if (res < 0 && res != -EALREADY) {
  308. SYS_LOG_ERR("failed to open global session");
  309. dsp_session_destroy(global_session);
  310. global_session = NULL;
  311. }
  312. }else{
  313. res = dsp_session_open(global_session, info);
  314. if (res < 0 && res != -EALREADY) {
  315. SYS_LOG_ERR("failed to open global session");
  316. return NULL;
  317. }
  318. }
  319. return global_session;
  320. }
  321. void dsp_close_global_session(struct dsp_session *session)
  322. {
  323. //assert(session == global_session);
  324. if (global_session) {
  325. if (!dsp_session_close(global_session)) {
  326. dsp_session_destroy(global_session);
  327. global_session = NULL;
  328. }
  329. }
  330. }
  331. int dsp_session_wait(struct dsp_session *session, int timeout)
  332. {
  333. return k_sem_take(&session->sem, SYS_TIMEOUT_MS(timeout));
  334. }
  335. int dsp_session_kick(struct dsp_session *session)
  336. {
  337. return dsp_kick(session->dev, session->uuid, DSP_EVENT_NEW_DATA, 0);
  338. }
  339. int dsp_session_submit_command(struct dsp_session *session, struct dsp_command *command)
  340. {
  341. struct acts_ringbuf *buf = (struct acts_ringbuf *)(session->cmdbuf.cpu_bufptr);
  342. unsigned int size = sizeof_dsp_command(command);
  343. unsigned int space;
  344. unsigned int wait_time = 500;
  345. int res = -ENOMEM;
  346. if (!global_session){
  347. return -ENODEV;
  348. }
  349. k_mutex_lock(&session->mutex, K_FOREVER);
  350. #ifdef ENABLE_SESSION_BEGIN_END_REFCNT
  351. if ((command->id == DSP_CMD_SESSION_BEGIN && atomic_inc(&session->run_refcnt) != 0) ||
  352. (command->id == DSP_CMD_SESSION_END && atomic_dec(&session->run_refcnt) != 1)) {
  353. if (command->sem) {
  354. SYS_LOG_DBG("ignore command %d", command->id);
  355. k_sem_give((struct k_sem *)command->sem);
  356. }
  357. res = 0;
  358. goto out_unlock;
  359. }
  360. #endif
  361. while(wait_time){
  362. space = acts_ringbuf_space(buf);
  363. if (space < ACTS_RINGBUF_NELEM(size)) {
  364. os_sleep(10);
  365. wait_time -= 10;
  366. SYS_LOG_WRN("wait cmd space %d %d %d", size, acts_ringbuf_head_offset(buf), acts_ringbuf_tail_offset(buf));
  367. }else{
  368. break;
  369. }
  370. }
  371. if (wait_time == 0){
  372. SYS_LOG_ERR("No enough space (%u) for command (id=0x%04x, size=%u)",
  373. space, command->id, size);
  374. #ifdef ENABLE_SESSION_BEGIN_END_REFCNT
  375. switch (command->id) {
  376. case DSP_CMD_SESSION_BEGIN:
  377. atomic_dec(&session->run_refcnt);
  378. break;
  379. case DSP_CMD_SESSION_END:
  380. atomic_inc(&session->run_refcnt);
  381. break;
  382. default:
  383. break;
  384. }
  385. #endif
  386. goto out_unlock;
  387. }
  388. /* alloc sequence number */
  389. command->seq = session->cmdbuf.alloc_seq++;
  390. /* insert command */
  391. res = acts_ringbuf_put(buf, command, ACTS_RINGBUF_NELEM(size));
  392. if(res == 0){
  393. SYS_LOG_ERR("send cmd failed\n");
  394. }
  395. SYS_LOG_INF("submit cmd %d %d %d %d seq %d totalsize %d buffer %d \n", command->id, command->data[0], command->data[1],command->size,command->seq,ACTS_RINGBUF_NELEM(size),acts_ringbuf_length(buf));
  396. /* kick dsp to process command */
  397. res = dsp_kick(session->dev, session->uuid, DSP_EVENT_NEW_CMD, 0);
  398. res = 0;
  399. out_unlock:
  400. k_mutex_unlock(&session->mutex);
  401. return res;
  402. }
  403. int dsp_console_set_cmd(uint16_t id, uint32_t param1, uint32_t param2)
  404. {
  405. struct dsp_command *command;
  406. int ret = 0;
  407. command = mem_malloc(sizeof(*command) + 8);
  408. if (command) {
  409. command->id = id;
  410. command->sem = 0;
  411. command->size = 8;
  412. command->data[0] = param1;
  413. command->data[1] = param2;
  414. } else {
  415. return -ENOMEM;
  416. }
  417. if (!global_session) {
  418. printk("[WRN] dsp global session is null\n");
  419. ret = -ENODEV;
  420. } else {
  421. dsp_session_submit_command(global_session, command);
  422. }
  423. mem_free((void *)command);
  424. return ret;
  425. }