dsp_bridge.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281
  1. /*
  2. * Copyright (C) 2009-2010 Felipe Contreras
  3. * Copyright (C) 2009-2010 Nokia Corporation
  4. * Copyright (C) 2007 Texas Instruments, Incorporated
  5. *
  6. * Author: Felipe Contreras <felipe.contreras@gmail.com>
  7. *
  8. * This file may be used under the terms of the GNU Lesser General Public
  9. * License version 2.1, a copy of which is found in LICENSE included in the
  10. * packaging of this file.
  11. */
  12. #include "dsp_bridge.h"
  13. /* for open */
  14. #include <sys/types.h>
  15. #include <sys/stat.h>
  16. #include <fcntl.h>
  17. #include <unistd.h> /* for close */
  18. #include <sys/ioctl.h> /* for ioctl */
  19. #include <stdlib.h> /* for free */
  20. #include <malloc.h> /* for memalign */
  21. #include <string.h> /* for memset */
  22. #define ALLOCATE_SM
  23. #ifdef ALLOCATE_SM
  24. #include <malloc.h> /* for memalign */
  25. #include <sys/mman.h> /* for mmap */
  26. #endif
  27. #if DSP_API < 2
  28. #include <errno.h>
  29. #endif
  30. #define VALGRIND
  31. /*
  32. * Dspbridge ioctl numbering scheme
  33. *
  34. * 7 0
  35. * ---------------------------------
  36. * | Module | ioctl Number |
  37. * ---------------------------------
  38. * | x | x | x | 0 | 0 | 0 | 0 | 0 |
  39. * ---------------------------------
  40. */
  41. /* ioctl driver identifier */
  42. #define DB 0xDB
  43. /*
  44. * Following are used to distinguish between module ioctls, this is needed
  45. * in case new ioctls are introduced.
  46. */
  47. #define DB_MODULE_MASK 0xE0
  48. #define DB_IOC_MASK 0x1F
  49. #if DSP_API >= 1
  50. #include <linux/ioctl.h>
  51. /* ioctl module masks */
  52. #define DB_MGR 0x0
  53. #define DB_PROC 0x20
  54. #define DB_NODE 0x40
  55. #define DB_STRM 0x60
  56. #define DB_CMM 0x80
  57. /* Used to calculate the ioctl per dspbridge module */
  58. #define DB_IOC(module, num) \
  59. (((module) & DB_MODULE_MASK) | ((num) & DB_IOC_MASK))
  60. #else
  61. #define DB_MGR 1
  62. #define DB_PROC 7
  63. #define DB_NODE 24
  64. #define DB_STRM 39
  65. #define DB_CMM 50
  66. #define DB_IOC(module, num) ((module) + (num))
  67. #undef _IOR
  68. #undef _IOW
  69. #undef _IOWR
  70. #define _IOR(type, nr, size) (nr)
  71. #define _IOW(type, nr, size) (nr)
  72. #define _IOWR(type, nr, size) (nr)
  73. #endif /* DSP_API */
  74. /* MGR Module */
  75. #define MGR_WAIT _IOWR(DB, DB_IOC(DB_MGR, 4), unsigned long)
  76. #define MGR_ENUMNODE_INFO _IOWR(DB, DB_IOC(DB_MGR, 0), unsigned long)
  77. #define MGR_REGISTEROBJECT _IOWR(DB, DB_IOC(DB_MGR, 2), unsigned long)
  78. #define MGR_UNREGISTEROBJECT _IOWR(DB, DB_IOC(DB_MGR, 3), unsigned long)
  79. /* PROC Module */
  80. #define PROC_ATTACH _IOWR(DB, DB_IOC(DB_PROC, 0), unsigned long)
  81. /* PROC_DETACH Deprecated */
  82. #define PROC_DETACH _IOR(DB, DB_IOC(DB_PROC, 2), unsigned long)
  83. #define PROC_REGISTERNOTIFY _IOWR(DB, DB_IOC(DB_PROC, 8), unsigned long)
  84. #define PROC_RSVMEM _IOWR(DB, DB_IOC(DB_PROC, 10), unsigned long)
  85. #define PROC_UNRSVMEM _IOW(DB, DB_IOC(DB_PROC, 11), unsigned long)
  86. #define PROC_MAPMEM _IOWR(DB, DB_IOC(DB_PROC, 12), unsigned long)
  87. #define PROC_UNMAPMEM _IOR(DB, DB_IOC(DB_PROC, 13), unsigned long)
  88. #define PROC_FLUSHMEMORY _IOW(DB, DB_IOC(DB_PROC, 14), unsigned long)
  89. #define PROC_INVALIDATEMEMORY _IOW(DB, DB_IOC(DB_PROC, 16), unsigned long)
  90. #define PROC_GET_STATE _IOWR(DB, DB_IOC(DB_PROC, 5), unsigned long)
  91. #define PROC_ENUMRESOURCES _IOWR(DB, DB_IOC(DB_PROC, 4), unsigned long)
  92. #define PROC_ENUMNODE _IOWR(DB, DB_IOC(DB_PROC, 3), unsigned long)
  93. #define PROC_STOP _IOWR(DB, DB_IOC(DB_PROC, 15), unsigned long)
  94. #define PROC_LOAD _IOW(DB, DB_IOC(DB_PROC, 7), unsigned long)
  95. #define PROC_START _IOW(DB, DB_IOC(DB_PROC, 9), unsigned long)
  96. #define PROC_BEGINDMA _IOW(DB, DB_IOC(DB_PROC, 17), unsigned long)
  97. #define PROC_ENDDMA _IOW(DB, DB_IOC(DB_PROC, 18), unsigned long)
  98. /* NODE Module */
  99. #define NODE_REGISTERNOTIFY _IOWR(DB, DB_IOC(DB_NODE, 11), unsigned long)
  100. #define NODE_CREATE _IOW(DB, DB_IOC(DB_NODE, 4), unsigned long)
  101. #define NODE_RUN _IOW(DB, DB_IOC(DB_NODE, 12), unsigned long)
  102. #define NODE_TERMINATE _IOWR(DB, DB_IOC(DB_NODE, 13), unsigned long)
  103. #define NODE_PUTMESSAGE _IOW(DB, DB_IOC(DB_NODE, 10), unsigned long)
  104. #define NODE_GETMESSAGE _IOWR(DB, DB_IOC(DB_NODE, 8), unsigned long)
  105. #define NODE_DELETE _IOW(DB, DB_IOC(DB_NODE, 5), unsigned long)
  106. #define NODE_GETATTR _IOWR(DB, DB_IOC(DB_NODE, 7), unsigned long)
  107. #define NODE_ALLOCMSGBUF _IOWR(DB, DB_IOC(DB_NODE, 1), unsigned long)
  108. #define NODE_GETUUIDPROPS _IOWR(DB, DB_IOC(DB_NODE, 14), unsigned long)
  109. #define NODE_ALLOCATE _IOWR(DB, DB_IOC(DB_NODE, 0), unsigned long)
  110. #define NODE_CONNECT _IOW(DB, DB_IOC(DB_NODE, 3), unsigned long)
  111. /* CMM Module */
  112. #define CMM_GETHANDLE _IOR(DB, DB_IOC(DB_CMM, 2), unsigned long)
  113. #define CMM_GETINFO _IOR(DB, DB_IOC(DB_CMM, 3), unsigned long)
  114. /* STRM Module */
  115. #define STRM_OPEN _IOWR(DB, DB_IOC(DB_STRM, 7), unsigned long)
  116. #define STRM_CLOSE _IOW(DB, DB_IOC(DB_STRM, 1), unsigned long)
  117. #define STRM_GETINFO _IOWR(DB, DB_IOC(DB_STRM, 4), unsigned long)
  118. #define STRM_ALLOCATEBUFFER _IOWR(DB, DB_IOC(DB_STRM, 0), unsigned long)
  119. #define STRM_IDLE _IOW(DB, DB_IOC(DB_STRM, 5), unsigned long)
  120. #define STRM_RECLAIM _IOWR(DB, DB_IOC(DB_STRM, 8), unsigned long)
  121. #define STRM_FREEBUFFER _IOWR(DB, DB_IOC(DB_STRM, 2), unsigned long)
  122. #define STRM_ISSUE _IOW(DB, DB_IOC(DB_STRM, 6), unsigned long)
  123. #if DSP_API < 2
  124. static inline int real_ioctl(int fd, int r, void *arg)
  125. {
  126. return ioctl(fd, r, arg);
  127. }
  128. #endif
  129. /* will not be needed when tidspbridge uses proper error codes */
  130. #define ioctl(...) (ioctl(__VA_ARGS__) < 0)
  131. int dsp_open(void)
  132. {
  133. return open("/dev/DspBridge", O_RDWR);
  134. }
  135. int dsp_close(int handle)
  136. {
  137. return close(handle);
  138. }
  139. struct proc_attach {
  140. unsigned int num;
  141. const void *info; /* not used */
  142. void **ret_handle;
  143. };
  144. bool dsp_attach(int handle,
  145. unsigned int num,
  146. const void *info,
  147. void **ret_handle)
  148. {
  149. struct proc_attach arg = {
  150. .num = num,
  151. .info = info,
  152. .ret_handle = ret_handle,
  153. };
  154. return !ioctl(handle, PROC_ATTACH, &arg);
  155. }
  156. struct proc_detach {
  157. void *proc_handle;
  158. };
  159. bool dsp_detach(int handle,
  160. void *proc_handle)
  161. {
  162. struct proc_detach arg = {
  163. .proc_handle = proc_handle,
  164. };
  165. return !ioctl(handle, PROC_DETACH, &arg);
  166. }
  167. struct register_notify {
  168. void *proc_handle;
  169. unsigned int event_mask;
  170. unsigned int notify_type;
  171. struct dsp_notification *info;
  172. };
  173. bool dsp_register_notify(int handle,
  174. void *proc_handle,
  175. unsigned int event_mask,
  176. unsigned int notify_type,
  177. struct dsp_notification *info)
  178. {
  179. struct register_notify arg = {
  180. .proc_handle = proc_handle,
  181. .event_mask = event_mask,
  182. .notify_type = notify_type,
  183. .info = info,
  184. };
  185. return !ioctl(handle, PROC_REGISTERNOTIFY, &arg);
  186. }
  187. struct proc_start {
  188. void *proc_handle;
  189. };
  190. bool dsp_start(int handle,
  191. void *proc_handle)
  192. {
  193. struct proc_start arg = {
  194. .proc_handle = proc_handle,
  195. };
  196. return !ioctl(handle, PROC_START, &arg);
  197. }
  198. bool dsp_stop(int handle,
  199. void *proc_handle)
  200. {
  201. struct proc_start arg = {
  202. .proc_handle = proc_handle,
  203. };
  204. return !ioctl(handle, PROC_STOP, &arg);
  205. }
  206. struct proc_load {
  207. void *proc_handle;
  208. int argc;
  209. char **argv;
  210. char **env;
  211. };
  212. bool dsp_load(int handle,
  213. void *proc_handle,
  214. int argc, char **argv,
  215. char **env)
  216. {
  217. struct proc_load arg = {
  218. .proc_handle = proc_handle,
  219. .argc = argc,
  220. .argv = argv,
  221. .env = env,
  222. };
  223. return !ioctl(handle, PROC_LOAD, &arg);
  224. }
  225. struct node_register_notify {
  226. void *node_handle;
  227. unsigned int event_mask;
  228. unsigned int notify_type;
  229. struct dsp_notification *info;
  230. };
  231. bool dsp_node_register_notify(int handle,
  232. struct dsp_node *node,
  233. unsigned int event_mask,
  234. unsigned int notify_type,
  235. struct dsp_notification *info)
  236. {
  237. struct node_register_notify arg = {
  238. .node_handle = node->handle,
  239. .event_mask = event_mask,
  240. .notify_type = notify_type,
  241. .info = info,
  242. };
  243. return !ioctl(handle, NODE_REGISTERNOTIFY, &arg);
  244. }
  245. struct wait_for_events {
  246. struct dsp_notification **notifications;
  247. unsigned int count;
  248. unsigned int *ret_index;
  249. unsigned int timeout;
  250. };
  251. bool dsp_wait_for_events(int handle,
  252. struct dsp_notification **notifications,
  253. unsigned int count,
  254. unsigned int *ret_index,
  255. unsigned int timeout)
  256. {
  257. struct wait_for_events arg = {
  258. .notifications = notifications,
  259. .count = count,
  260. .ret_index = ret_index,
  261. .timeout = timeout,
  262. };
  263. #if DSP_API >= 2
  264. return !ioctl(handle, MGR_WAIT, &arg);
  265. #else
  266. /*
  267. * Temporary hack since libc only saves errors -1 to -4095; 0x80008017
  268. * is not stored.
  269. */
  270. int r;
  271. r = real_ioctl(handle, MGR_WAIT, &arg);
  272. if (r == (int)0x80008017)
  273. errno = ETIME;
  274. return r >= 0;
  275. #endif
  276. }
  277. struct enum_node {
  278. unsigned int num;
  279. struct dsp_ndb_props *info;
  280. unsigned int info_size;
  281. unsigned int *ret_num;
  282. };
  283. bool dsp_enum(int handle,
  284. unsigned int num,
  285. struct dsp_ndb_props *info,
  286. size_t info_size,
  287. unsigned int *ret_num)
  288. {
  289. struct enum_node arg = {
  290. .num = num,
  291. .info = info,
  292. .info_size = info_size,
  293. .ret_num = ret_num,
  294. };
  295. return !ioctl(handle, MGR_ENUMNODE_INFO, &arg);
  296. }
  297. struct register_object {
  298. const struct dsp_uuid *uuid;
  299. enum dsp_dcd_object_type type;
  300. const char *path;
  301. };
  302. bool dsp_register(int handle,
  303. const struct dsp_uuid *uuid,
  304. enum dsp_dcd_object_type type,
  305. const char *path)
  306. {
  307. struct register_object arg = {
  308. .uuid = uuid,
  309. .type = type,
  310. .path = path,
  311. };
  312. return !ioctl(handle, MGR_REGISTEROBJECT, &arg);
  313. }
  314. struct unregister_object {
  315. const struct dsp_uuid *uuid;
  316. enum dsp_dcd_object_type type;
  317. };
  318. bool dsp_unregister(int handle,
  319. const struct dsp_uuid *uuid,
  320. enum dsp_dcd_object_type type)
  321. {
  322. struct unregister_object arg = {
  323. .uuid = uuid,
  324. .type = type,
  325. };
  326. return !ioctl(handle, MGR_UNREGISTEROBJECT, &arg);
  327. }
  328. struct node_create {
  329. void *node_handle;
  330. };
  331. bool dsp_node_create(int handle,
  332. struct dsp_node *node)
  333. {
  334. struct node_create arg = {
  335. .node_handle = node->handle,
  336. };
  337. return !ioctl(handle, NODE_CREATE, &arg);
  338. }
  339. struct node_run {
  340. void *node_handle;
  341. };
  342. bool dsp_node_run(int handle,
  343. struct dsp_node *node)
  344. {
  345. struct node_run arg = {
  346. .node_handle = node->handle,
  347. };
  348. return !ioctl(handle, NODE_RUN, &arg);
  349. }
  350. struct node_terminate {
  351. void *node_handle;
  352. unsigned long *status;
  353. };
  354. bool dsp_node_terminate(int handle,
  355. struct dsp_node *node,
  356. unsigned long *status)
  357. {
  358. struct node_terminate arg = {
  359. .node_handle = node->handle,
  360. .status = status,
  361. };
  362. return !ioctl(handle, NODE_TERMINATE, &arg);
  363. }
  364. struct node_put_message {
  365. void *node_handle;
  366. const struct dsp_msg *message;
  367. unsigned int timeout;
  368. };
  369. bool dsp_node_put_message(int handle,
  370. struct dsp_node *node,
  371. const struct dsp_msg *message,
  372. unsigned int timeout)
  373. {
  374. struct node_put_message arg = {
  375. .node_handle = node->handle,
  376. .message = message,
  377. .timeout = timeout,
  378. };
  379. return !ioctl(handle, NODE_PUTMESSAGE, &arg);
  380. }
  381. struct node_get_message {
  382. void *node_handle;
  383. struct dsp_msg *message;
  384. unsigned int timeout;
  385. };
  386. bool dsp_node_get_message(int handle,
  387. struct dsp_node *node,
  388. struct dsp_msg *message,
  389. unsigned int timeout)
  390. {
  391. struct node_get_message arg = {
  392. .node_handle = node->handle,
  393. .message = message,
  394. .timeout = timeout,
  395. };
  396. #ifdef VALGRIND
  397. memset(message, 0, sizeof(*message));
  398. #endif
  399. return !ioctl(handle, NODE_GETMESSAGE, &arg);
  400. }
  401. struct node_delete {
  402. void *node_handle;
  403. };
  404. static inline bool dsp_node_delete(int handle,
  405. struct dsp_node *node)
  406. {
  407. struct node_delete arg = {
  408. .node_handle = node->handle,
  409. };
  410. return !ioctl(handle, NODE_DELETE, &arg);
  411. }
  412. #ifdef ALLOCATE_SM
  413. struct node_get_attr {
  414. void *node_handle;
  415. struct dsp_node_attr *attr;
  416. unsigned int attr_size;
  417. };
  418. bool dsp_node_get_attr(int handle,
  419. struct dsp_node *node,
  420. struct dsp_node_attr *attr,
  421. size_t attr_size)
  422. {
  423. struct node_get_attr arg = {
  424. .node_handle = node->handle,
  425. .attr = attr,
  426. .attr_size = attr_size,
  427. };
  428. return !ioctl(handle, NODE_GETATTR, &arg);
  429. }
  430. struct dsp_buffer_attr {
  431. unsigned long cb;
  432. unsigned int segment;
  433. unsigned int alignment;
  434. };
  435. struct node_alloc_buf {
  436. void *node_handle;
  437. unsigned int size;
  438. struct dsp_buffer_attr *attr;
  439. void **buffer;
  440. };
  441. static inline bool dsp_node_alloc_buf(int handle,
  442. struct dsp_node *node,
  443. size_t size,
  444. struct dsp_buffer_attr *attr,
  445. void **buffer)
  446. {
  447. struct node_alloc_buf arg = {
  448. .node_handle = node->handle,
  449. .size = size,
  450. .attr = attr,
  451. .buffer = buffer,
  452. };
  453. if (ioctl(handle, NODE_ALLOCMSGBUF, &arg)) {
  454. *buffer = NULL;
  455. return false;
  456. }
  457. return true;
  458. }
  459. struct dsp_cmm_seg_info {
  460. unsigned long base_pa;
  461. unsigned long size;
  462. unsigned long gpp_base_pa;
  463. unsigned long gpp_size;
  464. unsigned long dsp_base_va;
  465. unsigned long dsp_size;
  466. unsigned long use_count;
  467. unsigned long base_va;
  468. };
  469. struct dsp_cmm_info {
  470. unsigned long segments;
  471. unsigned long use_count;
  472. unsigned long min_block_size;
  473. struct dsp_cmm_seg_info info[1];
  474. };
  475. struct cmm_get_handle {
  476. void *proc_handle;
  477. struct cmm_object **cmm;
  478. };
  479. struct cmm_get_info {
  480. struct cmm_object *cmm;
  481. struct dsp_cmm_info *info;
  482. };
  483. static inline bool get_cmm_info(int handle,
  484. void *proc_handle,
  485. struct dsp_cmm_info *cmm_info)
  486. {
  487. struct cmm_object *cmm;
  488. struct cmm_get_handle cmm_arg = {
  489. .proc_handle = proc_handle,
  490. .cmm = &cmm,
  491. };
  492. struct cmm_get_info cmm_info_arg = {
  493. .info = cmm_info,
  494. };
  495. if (ioctl(handle, CMM_GETHANDLE, &cmm_arg))
  496. return false;
  497. cmm_info_arg.cmm = cmm;
  498. if (ioctl(handle, CMM_GETINFO, &cmm_info_arg))
  499. return false;
  500. return true;
  501. }
  502. static inline bool allocate_segments(int handle,
  503. void *proc_handle,
  504. struct dsp_node *node)
  505. {
  506. struct dsp_cmm_info cmm_info;
  507. struct dsp_node_attr attr;
  508. enum dsp_node_type node_type;
  509. #ifdef VALGRIND
  510. memset(&cmm_info, 0, sizeof(cmm_info));
  511. memset(&attr, 0, sizeof(attr));
  512. #endif
  513. if (!get_cmm_info(handle, proc_handle, &cmm_info))
  514. return false;
  515. if (!dsp_node_get_attr(handle, node, &attr, sizeof(attr)))
  516. return false;
  517. node_type = attr.info.props.ntype;
  518. if ((node_type != DSP_NODE_DEVICE) && (cmm_info.segments > 0)) {
  519. struct dsp_cmm_seg_info *seg;
  520. seg = &cmm_info.info[0];
  521. if (seg->base_pa != 0 && seg->size > 0) {
  522. void *base;
  523. struct dsp_buffer_attr buffer_attr;
  524. base = mmap(NULL, seg->size,
  525. PROT_READ | PROT_WRITE, MAP_SHARED | 0x2000 /* MAP_LOCKED */,
  526. handle, seg->base_pa);
  527. if (!base)
  528. return false;
  529. buffer_attr.alignment = 0;
  530. buffer_attr.segment = 1 | 0x10000000;
  531. buffer_attr.cb = 0;
  532. if (!dsp_node_alloc_buf(handle, node, seg->size,
  533. &buffer_attr, &base))
  534. {
  535. munmap(base, seg->size);
  536. return false;
  537. }
  538. node->msgbuf_addr = base;
  539. node->msgbuf_size = seg->size;
  540. }
  541. }
  542. return true;
  543. }
  544. #endif
  545. #ifdef ALLOCATE_HEAP
  546. struct get_uuid_props {
  547. void *proc_handle;
  548. const struct dsp_uuid *node_uuid;
  549. struct dsp_ndb_props *props;
  550. };
  551. static inline bool get_uuid_props(int handle,
  552. void *proc_handle,
  553. const struct dsp_uuid *node_uuid,
  554. struct dsp_ndb_props *props)
  555. {
  556. struct get_uuid_props arg = {
  557. .proc_handle = proc_handle,
  558. .node_uuid = node_uuid,
  559. .props = props,
  560. };
  561. return !ioctl(handle, NODE_GETUUIDPROPS, &arg);
  562. }
  563. #define PG_SIZE_4K 4096
  564. #define PG_MASK(pg_size) (~((pg_size)-1))
  565. #define PG_ALIGN_LOW(addr, pg_size) ((addr) & PG_MASK(pg_size))
  566. #define PG_ALIGN_HIGH(addr, pg_size) (((addr)+(pg_size)-1) & PG_MASK(pg_size))
  567. #endif
  568. struct node_allocate {
  569. void *proc_handle;
  570. const struct dsp_uuid *node_id;
  571. const void *cb_data;
  572. struct dsp_node_attr_in *attrs;
  573. void **ret_node;
  574. };
  575. bool dsp_node_allocate(int handle,
  576. void *proc_handle,
  577. const struct dsp_uuid *node_uuid,
  578. const void *cb_data,
  579. struct dsp_node_attr_in *attrs,
  580. struct dsp_node **ret_node)
  581. {
  582. struct dsp_node *node;
  583. void *node_handle = NULL;
  584. struct node_allocate arg = {
  585. .proc_handle = proc_handle,
  586. .node_id = node_uuid,
  587. .cb_data = cb_data,
  588. .attrs = attrs,
  589. .ret_node = &node_handle,
  590. };
  591. #ifdef ALLOCATE_HEAP
  592. if (attrs) {
  593. struct dsp_ndb_props props;
  594. #ifdef VALGRIND
  595. memset(&props, 0, sizeof(props));
  596. #endif
  597. if (!get_uuid_props(handle, proc_handle, node_uuid, &props)) {
  598. attrs->gpp_va = NULL;
  599. return false;
  600. }
  601. if (attrs->profile_id < props.count_profiles) {
  602. unsigned int heap_size = 0;
  603. heap_size = props.node_profiles[attrs->profile_id].heap_size;
  604. if (heap_size) {
  605. void *virtual = NULL;
  606. heap_size = PG_ALIGN_HIGH(heap_size, PG_SIZE_4K);
  607. virtual = memalign(128, heap_size);
  608. if (!virtual)
  609. return false;
  610. attrs->heap_size = heap_size;
  611. attrs->gpp_va = virtual;
  612. }
  613. }
  614. }
  615. #endif
  616. if (ioctl(handle, NODE_ALLOCATE, &arg)) {
  617. if (attrs) {
  618. free(attrs->gpp_va);
  619. attrs->gpp_va = NULL;
  620. }
  621. return false;
  622. }
  623. node = calloc(1, sizeof(*node));
  624. node->handle = node_handle;
  625. if (attrs)
  626. node->heap = attrs->gpp_va;
  627. #ifdef ALLOCATE_SM
  628. if (!allocate_segments(handle, proc_handle, node)) {
  629. dsp_node_delete(handle, node);
  630. free(node->heap);
  631. free(node);
  632. return false;
  633. }
  634. #endif
  635. *ret_node = node;
  636. return true;
  637. }
  638. struct node_connect {
  639. void *node_handle;
  640. unsigned int stream;
  641. void *other_node_handle;
  642. unsigned int other_stream;
  643. struct dsp_stream_attr *attrs;
  644. void *params;
  645. };
  646. bool dsp_node_connect(int handle,
  647. struct dsp_node *node,
  648. unsigned int stream,
  649. struct dsp_node *other_node,
  650. unsigned int other_stream,
  651. struct dsp_stream_attr *attrs,
  652. void *params)
  653. {
  654. struct node_connect arg = {
  655. .node_handle = node->handle,
  656. .stream = stream,
  657. .other_node_handle = other_node->handle,
  658. .other_stream = other_stream,
  659. .attrs = attrs,
  660. .params = params,
  661. };
  662. return !ioctl(handle, NODE_CONNECT, &arg);
  663. }
  664. bool dsp_node_free(int handle,
  665. struct dsp_node *node)
  666. {
  667. #ifdef ALLOCATE_SM
  668. munmap(node->msgbuf_addr, node->msgbuf_size);
  669. #endif
  670. dsp_node_delete(handle, node);
  671. free(node->heap);
  672. free(node);
  673. return true;
  674. }
  675. struct reserve_mem {
  676. void *proc_handle;
  677. unsigned long size;
  678. void **addr;
  679. };
  680. bool dsp_reserve(int handle,
  681. void *proc_handle,
  682. unsigned long size,
  683. void **addr)
  684. {
  685. struct reserve_mem arg = {
  686. .proc_handle = proc_handle,
  687. .size = size,
  688. .addr = addr,
  689. };
  690. return !ioctl(handle, PROC_RSVMEM, &arg);
  691. }
  692. struct unreserve_mem {
  693. void *proc_handle;
  694. unsigned long size;
  695. void *addr;
  696. };
  697. bool dsp_unreserve(int handle,
  698. void *proc_handle,
  699. void *addr)
  700. {
  701. struct unreserve_mem arg = {
  702. .proc_handle = proc_handle,
  703. .addr = addr,
  704. };
  705. return !ioctl(handle, PROC_UNRSVMEM, &arg);
  706. }
  707. struct map_mem {
  708. void *proc_handle;
  709. void *mpu_addr;
  710. unsigned long size;
  711. void *req_addr;
  712. void **ret_map_addr;
  713. unsigned long attr;
  714. };
  715. bool dsp_map(int handle,
  716. void *proc_handle,
  717. void *mpu_addr,
  718. unsigned long size,
  719. void *req_addr,
  720. void *ret_map_addr,
  721. unsigned long attr)
  722. {
  723. struct map_mem arg = {
  724. .proc_handle = proc_handle,
  725. .mpu_addr = mpu_addr,
  726. .size = size,
  727. .req_addr = req_addr,
  728. .ret_map_addr = ret_map_addr,
  729. .attr = attr,
  730. };
  731. return !ioctl(handle, PROC_MAPMEM, &arg);
  732. }
  733. struct unmap_mem {
  734. void *proc_handle;
  735. unsigned long size;
  736. void *map_addr;
  737. };
  738. bool dsp_unmap(int handle,
  739. void *proc_handle,
  740. void *map_addr)
  741. {
  742. struct unmap_mem arg = {
  743. .proc_handle = proc_handle,
  744. .map_addr = map_addr,
  745. };
  746. return !ioctl(handle, PROC_UNMAPMEM, &arg);
  747. }
  748. struct flush_mem {
  749. void *proc_handle;
  750. void *mpu_addr;
  751. unsigned long size;
  752. unsigned long flags;
  753. };
  754. bool dsp_flush(int handle,
  755. void *proc_handle,
  756. void *mpu_addr,
  757. unsigned long size,
  758. unsigned long flags)
  759. {
  760. struct flush_mem arg = {
  761. .proc_handle = proc_handle,
  762. .mpu_addr = mpu_addr,
  763. .size = size,
  764. .flags = flags,
  765. };
  766. return !ioctl(handle, PROC_FLUSHMEMORY, &arg);
  767. }
  768. struct invalidate_mem {
  769. void *proc_handle;
  770. void *mpu_addr;
  771. unsigned long size;
  772. };
  773. bool dsp_invalidate(int handle,
  774. void *proc_handle,
  775. void *mpu_addr,
  776. unsigned long size)
  777. {
  778. struct invalidate_mem arg = {
  779. .proc_handle = proc_handle,
  780. .mpu_addr = mpu_addr,
  781. .size = size,
  782. };
  783. return !ioctl(handle, PROC_INVALIDATEMEMORY, &arg);
  784. }
  785. struct dma_op {
  786. void *proc_handle;
  787. void *mpu_addr;
  788. unsigned long size;
  789. unsigned long dir;
  790. };
  791. bool dsp_begin_dma(int handle,
  792. void *proc_handle,
  793. void *mpu_addr,
  794. unsigned long size,
  795. unsigned long dir)
  796. {
  797. struct dma_op arg = {
  798. .proc_handle = proc_handle,
  799. .mpu_addr = mpu_addr,
  800. .size = size,
  801. .dir = dir,
  802. };
  803. return !ioctl(handle, PROC_BEGINDMA, &arg);
  804. }
  805. bool dsp_end_dma(int handle,
  806. void *proc_handle,
  807. void *mpu_addr,
  808. unsigned long size,
  809. unsigned long dir)
  810. {
  811. struct dma_op arg = {
  812. .proc_handle = proc_handle,
  813. .mpu_addr = mpu_addr,
  814. .size = size,
  815. .dir = dir,
  816. };
  817. return !ioctl(handle, PROC_ENDDMA, &arg);
  818. }
  819. struct proc_get_info {
  820. void *proc_handle;
  821. unsigned type;
  822. struct dsp_info *info;
  823. unsigned size;
  824. };
  825. bool dsp_proc_get_info(int handle,
  826. void *proc_handle,
  827. unsigned type,
  828. struct dsp_info *info,
  829. unsigned size)
  830. {
  831. struct proc_get_info arg = {
  832. .proc_handle = proc_handle,
  833. .type = type,
  834. .info = info,
  835. .size = size,
  836. };
  837. return !ioctl(handle, PROC_ENUMRESOURCES, &arg);
  838. }
  839. struct enum_nodes {
  840. void *proc_handle;
  841. void **node_table;
  842. unsigned node_table_size;
  843. unsigned *num_nodes;
  844. unsigned *allocated;
  845. };
  846. bool dsp_enum_nodes(int handle,
  847. void *proc_handle,
  848. void **node_table,
  849. unsigned node_table_size,
  850. unsigned *num_nodes,
  851. unsigned *allocated)
  852. {
  853. struct enum_nodes arg = {
  854. .proc_handle = proc_handle,
  855. .node_table = node_table,
  856. .node_table_size = node_table_size,
  857. .num_nodes = num_nodes,
  858. .allocated = allocated,
  859. };
  860. return !ioctl(handle, PROC_ENUMNODE, &arg);
  861. }
  862. struct stream_attr {
  863. void *event;
  864. char *name;
  865. void *base;
  866. unsigned long size;
  867. struct dsp_stream_attr_in *attrin;
  868. };
  869. struct stream_open {
  870. void *node_handle;
  871. unsigned int direction;
  872. unsigned int index;
  873. struct stream_attr *attr;
  874. void *stream;
  875. };
  876. bool dsp_stream_open(int handle,
  877. struct dsp_node *node,
  878. unsigned int direction,
  879. unsigned int index,
  880. struct dsp_stream_attr_in *attrin,
  881. void *stream)
  882. {
  883. struct stream_attr strm_attr = {
  884. .attrin = attrin,
  885. };
  886. struct stream_open stream_arg = {
  887. .node_handle = node->handle,
  888. .direction = direction,
  889. .index = index,
  890. .attr = &strm_attr,
  891. .stream = stream,
  892. };
  893. if (attrin && (attrin->mode == STRMMODE_ZEROCOPY ||
  894. attrin->mode == STRMMODE_RDMA)) {
  895. struct dsp_cmm_info cmm_info;
  896. if (!get_cmm_info(handle, NULL, &cmm_info))
  897. return false;
  898. if (cmm_info.segments > 0) {
  899. void *base;
  900. struct dsp_cmm_seg_info *seg;
  901. seg = &cmm_info.info[0];
  902. base = mmap(NULL, seg->size,
  903. PROT_READ | PROT_WRITE,
  904. MAP_SHARED | 0x2000 /* MAP_LOCKED */,
  905. handle, seg->base_pa);
  906. if (!base)
  907. return false;
  908. strm_attr.base = base;
  909. strm_attr.size = seg->size;
  910. }
  911. }
  912. return !ioctl(handle, STRM_OPEN, &stream_arg);
  913. }
  914. struct stream_info {
  915. enum dsp_stream_mode mode;
  916. unsigned int segment;
  917. void *base;
  918. struct dsp_stream_info *info;
  919. };
  920. struct stream_get_info {
  921. void *stream;
  922. struct stream_info *info;
  923. unsigned int size;
  924. };
  925. static inline bool get_stream_info(int handle,
  926. void *stream,
  927. struct stream_info *info,
  928. unsigned int size)
  929. {
  930. struct stream_get_info arg = {
  931. .stream = stream,
  932. .info = info,
  933. .size = size,
  934. };
  935. return !ioctl(handle, STRM_GETINFO, &arg);
  936. }
  937. bool dsp_stream_close(int handle,
  938. void *stream)
  939. {
  940. struct stream_info info;
  941. if (!get_stream_info(handle, stream, &info, sizeof(struct stream_info)))
  942. return false;
  943. if (info.base) {
  944. struct dsp_cmm_info cmm_info;
  945. if (!get_cmm_info(handle, NULL, &cmm_info))
  946. return false;
  947. if (cmm_info.segments > 0) {
  948. struct dsp_cmm_seg_info *seg;
  949. seg = &cmm_info.info[0];
  950. if (munmap(info.base, seg->size))
  951. return false;
  952. }
  953. }
  954. return !ioctl(handle, STRM_CLOSE, &stream);
  955. }
  956. struct stream_idle {
  957. void *stream;
  958. bool flush;
  959. };
  960. bool dsp_stream_idle(int handle,
  961. void *stream,
  962. bool flush)
  963. {
  964. struct stream_idle arg = {
  965. .stream = stream,
  966. .flush = flush,
  967. };
  968. return !ioctl(handle, STRM_IDLE, &arg);
  969. }
  970. struct stream_reclaim {
  971. void *stream;
  972. unsigned char **buff;
  973. unsigned long *data_size;
  974. unsigned long *buff_size;
  975. unsigned long *flag;
  976. };
  977. bool dsp_stream_reclaim(int handle,
  978. void *stream,
  979. unsigned char **buff,
  980. unsigned long *data_size,
  981. unsigned long *buff_size,
  982. unsigned long *flag)
  983. {
  984. struct stream_reclaim arg = {
  985. .stream = stream,
  986. .buff = buff,
  987. .data_size = data_size,
  988. .buff_size = buff_size,
  989. .flag = flag,
  990. };
  991. return !ioctl(handle, STRM_RECLAIM, &arg);
  992. }
  993. struct stream_issue {
  994. void *stream;
  995. unsigned char *buff;
  996. unsigned long data_size;
  997. unsigned long buff_size;
  998. unsigned long flag;
  999. };
  1000. bool dsp_stream_issue(int handle,
  1001. void *stream,
  1002. unsigned char *buff,
  1003. unsigned long data_size,
  1004. unsigned long buff_size,
  1005. unsigned long flag)
  1006. {
  1007. struct stream_issue arg = {
  1008. .stream = stream,
  1009. .buff = buff,
  1010. .data_size = data_size,
  1011. .buff_size = buff_size,
  1012. .flag = flag,
  1013. };
  1014. return !ioctl(handle, STRM_ISSUE, &arg);
  1015. }
  1016. bool dsp_stream_get_info(int handle,
  1017. void *stream,
  1018. struct dsp_stream_info *info,
  1019. unsigned int size)
  1020. {
  1021. struct stream_info stream_info = {
  1022. .info = info
  1023. };
  1024. return get_stream_info(handle, stream, &stream_info, size);
  1025. }
  1026. struct stream_allocate_buffer {
  1027. void *stream;
  1028. unsigned int size;
  1029. unsigned char **buff;
  1030. unsigned int num_buf;
  1031. };
  1032. bool dsp_stream_allocate_buffers(int handle,
  1033. void *stream,
  1034. unsigned int size,
  1035. unsigned char **buff,
  1036. unsigned int num_buf)
  1037. {
  1038. unsigned int i;
  1039. struct stream_info info;
  1040. if (!get_stream_info(handle, stream, &info, sizeof(struct stream_info)))
  1041. return false;
  1042. if (info.segment > 0) {
  1043. struct stream_allocate_buffer arg = {
  1044. .stream = stream,
  1045. .size = size,
  1046. .buff = buff,
  1047. .num_buf = num_buf,
  1048. };
  1049. return !ioctl(handle, STRM_ALLOCATEBUFFER, &arg);
  1050. }
  1051. for (i = 0; i < num_buf; i++)
  1052. buff[i] = (unsigned char *) malloc(size);
  1053. return true;
  1054. }
  1055. struct stream_free_buffers {
  1056. void *stream;
  1057. unsigned char **buff;
  1058. unsigned int num_buf;
  1059. };
  1060. bool dsp_stream_free_buffers(int handle,
  1061. void *stream,
  1062. unsigned char **buff,
  1063. unsigned int num_buf)
  1064. {
  1065. unsigned int i;
  1066. struct stream_info info;
  1067. if (!get_stream_info(handle, stream, &info, sizeof(struct stream_info)))
  1068. return false;
  1069. if (info.segment) {
  1070. struct stream_free_buffers arg = {
  1071. .stream = stream,
  1072. .buff = buff,
  1073. .num_buf = num_buf,
  1074. };
  1075. return !ioctl(handle, STRM_FREEBUFFER, &arg);
  1076. }
  1077. for (i = 0; i < num_buf; i++) {
  1078. free(buff[i]);
  1079. buff[i] = NULL;
  1080. }
  1081. return true;
  1082. }