msm_audio_ion_vm.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2013-2020, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/init.h>
  6. #include <linux/kernel.h>
  7. #include <linux/module.h>
  8. #include <linux/err.h>
  9. #include <linux/delay.h>
  10. #include <linux/slab.h>
  11. #include <linux/mutex.h>
  12. #include <linux/list.h>
  13. #include <linux/dma-mapping.h>
  14. #include <linux/dma-contiguous.h>
  15. #include <linux/dma-buf.h>
  16. #include <linux/iommu.h>
  17. #include <linux/platform_device.h>
  18. #include <linux/of_device.h>
  19. #include <linux/export.h>
  20. #include <linux/ion_kernel.h>
  21. #include <ipc/apr.h>
  22. #include <asm/dma-iommu.h>
  23. #include <dsp/msm_audio_ion.h>
  24. #include <soc/qcom/secure_buffer.h>
  25. #include <linux/habmm.h>
  26. #define MSM_AUDIO_ION_PROBED (1 << 0)
  27. #define MSM_AUDIO_ION_PHYS_ADDR(alloc_data) \
  28. alloc_data->table->sgl->dma_address
  29. #define MSM_AUDIO_SMMU_VM_CMD_MAP 0x00000001
  30. #define MSM_AUDIO_SMMU_VM_CMD_UNMAP 0x00000002
  31. #define MSM_AUDIO_SMMU_VM_HAB_MINOR_ID 1
  32. enum msm_audio_mem_type{
  33. MSM_AUDIO_MEM_TYPE_ION,
  34. MSM_AUDIO_MEM_TYPE_DMA,
  35. };
  36. struct msm_audio_ion_private {
  37. bool smmu_enabled;
  38. struct device *cb_dev;
  39. u8 device_status;
  40. struct list_head alloc_list;
  41. struct mutex list_mutex;
  42. };
  43. struct msm_audio_alloc_data {
  44. size_t len;
  45. void *vaddr;
  46. void *handle;
  47. struct dma_buf_attachment *attach;
  48. struct sg_table *table;
  49. struct list_head list;
  50. dma_addr_t *paddr;
  51. enum msm_audio_mem_type type;
  52. u32 export_id;
  53. };
  54. struct msm_audio_smmu_vm_map_cmd {
  55. int cmd_id;
  56. u32 export_id;
  57. u32 buf_size;
  58. };
  59. struct msm_audio_smmu_vm_map_cmd_rsp {
  60. int status;
  61. u64 addr;
  62. };
  63. struct msm_audio_smmu_vm_unmap_cmd {
  64. int cmd_id;
  65. u32 export_id;
  66. };
  67. struct msm_audio_smmu_vm_unmap_cmd_rsp {
  68. int status;
  69. };
  70. static struct msm_audio_ion_private msm_audio_ion_data = {0,};
  71. static u32 msm_audio_ion_hab_handle;
  72. static void msm_audio_ion_add_allocation(
  73. struct msm_audio_ion_private *msm_audio_ion_data,
  74. struct msm_audio_alloc_data *alloc_data)
  75. {
  76. /*
  77. * Since these APIs can be invoked by multiple
  78. * clients, there is need to make sure the list
  79. * of allocations is always protected
  80. */
  81. mutex_lock(&(msm_audio_ion_data->list_mutex));
  82. list_add_tail(&(alloc_data->list),
  83. &(msm_audio_ion_data->alloc_list));
  84. mutex_unlock(&(msm_audio_ion_data->list_mutex));
  85. }
  86. static int msm_audio_dma_buf_map(void *handle, void *vaddr,
  87. dma_addr_t *paddr,
  88. size_t *len)
  89. {
  90. struct msm_audio_alloc_data *alloc_data;
  91. /* Data required per buffer mapping */
  92. alloc_data = kzalloc(sizeof(*alloc_data), GFP_KERNEL);
  93. if (!alloc_data)
  94. return -ENOMEM;
  95. alloc_data->handle = handle;
  96. alloc_data->len = *len;
  97. alloc_data->vaddr = vaddr;
  98. alloc_data->paddr = paddr;
  99. alloc_data->type = MSM_AUDIO_MEM_TYPE_DMA;
  100. msm_audio_ion_add_allocation(&msm_audio_ion_data,
  101. alloc_data);
  102. return 0;
  103. }
  104. static int msm_audio_ion_dma_buf_map(struct dma_buf *dma_buf,
  105. dma_addr_t *addr, size_t *len)
  106. {
  107. struct msm_audio_alloc_data *alloc_data;
  108. struct device *cb_dev;
  109. unsigned long ionflag = 0;
  110. int rc = 0;
  111. cb_dev = msm_audio_ion_data.cb_dev;
  112. /* Data required per buffer mapping */
  113. alloc_data = kzalloc(sizeof(*alloc_data), GFP_KERNEL);
  114. if (!alloc_data)
  115. return -ENOMEM;
  116. alloc_data->handle = (void*)dma_buf;
  117. alloc_data->len = dma_buf->size;
  118. alloc_data->type = MSM_AUDIO_MEM_TYPE_ION;
  119. *len = dma_buf->size;
  120. /* Attach the dma_buf to context bank device */
  121. alloc_data->attach = dma_buf_attach(dma_buf, cb_dev);
  122. if (IS_ERR(alloc_data->attach)) {
  123. rc = PTR_ERR(alloc_data->attach);
  124. dev_err(cb_dev,
  125. "%s: Fail to attach dma_buf to CB, rc = %d\n",
  126. __func__, rc);
  127. goto free_alloc_data;
  128. }
  129. /* For uncached buffers, avoid cache maintanance */
  130. rc = dma_buf_get_flags(dma_buf, &ionflag);
  131. if (rc) {
  132. dev_err(cb_dev, "%s: dma_buf_get_flags failed: %d\n",
  133. __func__, rc);
  134. goto detach_dma_buf;
  135. }
  136. if (!(ionflag & ION_FLAG_CACHED))
  137. alloc_data->attach->dma_map_attrs |= DMA_ATTR_SKIP_CPU_SYNC;
  138. /*
  139. * Get the scatter-gather list.
  140. * There is no info as this is a write buffer or
  141. * read buffer, hence the request is bi-directional
  142. * to accommodate both read and write mappings.
  143. */
  144. alloc_data->table = dma_buf_map_attachment(alloc_data->attach,
  145. DMA_BIDIRECTIONAL);
  146. if (IS_ERR(alloc_data->table)) {
  147. rc = PTR_ERR(alloc_data->table);
  148. dev_err(cb_dev,
  149. "%s: Fail to map attachment, rc = %d\n",
  150. __func__, rc);
  151. goto detach_dma_buf;
  152. }
  153. /* physical address from mapping */
  154. *addr = MSM_AUDIO_ION_PHYS_ADDR(alloc_data);
  155. alloc_data->paddr = addr;
  156. msm_audio_ion_add_allocation(&msm_audio_ion_data,
  157. alloc_data);
  158. return rc;
  159. detach_dma_buf:
  160. dma_buf_detach(dma_buf, alloc_data->attach);
  161. free_alloc_data:
  162. kfree(alloc_data);
  163. return rc;
  164. }
  165. static int msm_audio_ion_unmap_kernel(void *vaddr, void *handle)
  166. {
  167. int rc = 0;
  168. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  169. if (!vaddr) {
  170. dev_err(cb_dev,
  171. "%s: cannot find allocation for handle %pK\n",
  172. __func__, handle);
  173. rc = -EINVAL;
  174. goto err;
  175. }
  176. dma_buf_vunmap((struct dma_buf*)handle, vaddr);
  177. rc = dma_buf_end_cpu_access((struct dma_buf*)handle, DMA_BIDIRECTIONAL);
  178. if (rc) {
  179. dev_err(cb_dev, "%s: kmap dma_buf_end_cpu_access fail\n",
  180. __func__);
  181. goto err;
  182. }
  183. err:
  184. return rc;
  185. }
  186. static int msm_audio_dma_buf_unmap(void *handle)
  187. {
  188. int rc = 0;
  189. struct msm_audio_alloc_data *alloc_data = NULL;
  190. struct list_head *ptr, *next;
  191. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  192. bool found = false;
  193. /*
  194. * Though list_for_each_safe is delete safe, lock
  195. * should be explicitly acquired to avoid race condition
  196. * on adding elements to the list.
  197. */
  198. mutex_lock(&(msm_audio_ion_data.list_mutex));
  199. list_for_each_safe(ptr, next, &(msm_audio_ion_data.alloc_list)) {
  200. alloc_data = list_entry(ptr, struct msm_audio_alloc_data, list);
  201. if(alloc_data->type == MSM_AUDIO_MEM_TYPE_ION) {
  202. if (alloc_data->handle == handle) {
  203. rc = msm_audio_ion_unmap_kernel(
  204. alloc_data->vaddr,
  205. handle);
  206. if(rc) {
  207. pr_err("%s: Unable to unmap ion mem rc: %d\n",
  208. __func__, rc);
  209. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  210. return rc;
  211. }
  212. found = true;
  213. dma_buf_unmap_attachment(alloc_data->attach,
  214. alloc_data->table,
  215. DMA_BIDIRECTIONAL);
  216. dma_buf_detach((struct dma_buf*)
  217. alloc_data->handle,
  218. alloc_data->attach);
  219. dma_buf_put((struct dma_buf*)
  220. alloc_data->handle);
  221. list_del(&(alloc_data->list));
  222. kfree(alloc_data);
  223. break;
  224. }
  225. } else {
  226. alloc_data = list_entry(ptr,
  227. struct msm_audio_alloc_data,
  228. list);
  229. if (alloc_data->handle == handle) {
  230. found = true;
  231. dma_free_coherent(cb_dev, alloc_data->len,
  232. alloc_data->vaddr,
  233. *(alloc_data->paddr));
  234. list_del(&(alloc_data->list));
  235. kfree(alloc_data);
  236. break;
  237. }
  238. }
  239. }
  240. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  241. if (!found) {
  242. dev_err(cb_dev,
  243. "%s: cannot find allocation, handle %pK",
  244. __func__, handle);
  245. rc = -EINVAL;
  246. }
  247. return rc;
  248. }
  249. static int msm_audio_ion_smmu_map(void *handle,
  250. dma_addr_t *paddr, size_t *len)
  251. {
  252. int rc;
  253. u32 export_id;
  254. u32 cmd_rsp_size;
  255. bool found = false;
  256. bool exported = false;
  257. struct msm_audio_smmu_vm_map_cmd smmu_map_cmd;
  258. struct msm_audio_smmu_vm_map_cmd_rsp cmd_rsp;
  259. struct msm_audio_alloc_data *alloc_data = NULL;
  260. unsigned long delay = jiffies + (HZ / 2);
  261. *len = ((struct dma_buf*)handle)->size;
  262. mutex_lock(&(msm_audio_ion_data.list_mutex));
  263. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  264. list) {
  265. if (alloc_data->handle == handle) {
  266. found = true;
  267. /* Export the buffer to physical VM */
  268. rc = habmm_export(msm_audio_ion_hab_handle, handle, *len,
  269. &export_id, HABMM_EXPIMP_FLAGS_DMABUF);
  270. if (rc) {
  271. pr_err("%s: habmm_export failed handle = %pK, len = %zd, rc = %d\n",
  272. __func__, handle, *len, rc);
  273. goto err;
  274. }
  275. exported = true;
  276. smmu_map_cmd.cmd_id = MSM_AUDIO_SMMU_VM_CMD_MAP;
  277. smmu_map_cmd.export_id = export_id;
  278. smmu_map_cmd.buf_size = *len;
  279. rc = habmm_socket_send(msm_audio_ion_hab_handle,
  280. (void *)&smmu_map_cmd, sizeof(smmu_map_cmd), 0);
  281. if (rc) {
  282. pr_err("%s: habmm_socket_send failed %d\n",
  283. __func__, rc);
  284. goto err;
  285. }
  286. do {
  287. cmd_rsp_size = sizeof(cmd_rsp);
  288. rc = habmm_socket_recv(msm_audio_ion_hab_handle,
  289. (void *)&cmd_rsp,
  290. &cmd_rsp_size,
  291. 0xFFFFFFFF,
  292. 0);
  293. } while (time_before(jiffies, delay) && (rc == -EINTR) &&
  294. (cmd_rsp_size == 0));
  295. if (rc) {
  296. pr_err("%s: habmm_socket_recv failed %d\n",
  297. __func__, rc);
  298. goto err;
  299. }
  300. if (cmd_rsp_size != sizeof(cmd_rsp)) {
  301. pr_err("%s: invalid size for cmd rsp %u, expected %zu\n",
  302. __func__, cmd_rsp_size, sizeof(cmd_rsp));
  303. rc = -EIO;
  304. goto err;
  305. }
  306. if (cmd_rsp.status) {
  307. pr_err("%s: SMMU map command failed %d\n",
  308. __func__, cmd_rsp.status);
  309. rc = cmd_rsp.status;
  310. goto err;
  311. }
  312. *paddr = (dma_addr_t)cmd_rsp.addr;
  313. alloc_data->export_id = export_id;
  314. break;
  315. }
  316. }
  317. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  318. if (!found) {
  319. pr_err("%s: cannot find allocation, handle %pK\n", __func__, handle);
  320. return -EINVAL;
  321. }
  322. return 0;
  323. err:
  324. if (exported)
  325. (void)habmm_unexport(msm_audio_ion_hab_handle, export_id, 0);
  326. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  327. return rc;
  328. }
  329. static int msm_audio_ion_smmu_unmap(void *handle)
  330. {
  331. int rc;
  332. bool found = false;
  333. u32 cmd_rsp_size;
  334. struct msm_audio_smmu_vm_unmap_cmd smmu_unmap_cmd;
  335. struct msm_audio_smmu_vm_unmap_cmd_rsp cmd_rsp;
  336. struct msm_audio_alloc_data *alloc_data, *next;
  337. unsigned long delay = jiffies + (HZ / 2);
  338. /*
  339. * Though list_for_each_entry_safe is delete safe, lock
  340. * should be explicitly acquired to avoid race condition
  341. * on adding elements to the list.
  342. */
  343. mutex_lock(&(msm_audio_ion_data.list_mutex));
  344. list_for_each_entry_safe(alloc_data, next,
  345. &(msm_audio_ion_data.alloc_list), list) {
  346. if (alloc_data->handle == handle) {
  347. found = true;
  348. smmu_unmap_cmd.cmd_id = MSM_AUDIO_SMMU_VM_CMD_UNMAP;
  349. smmu_unmap_cmd.export_id = alloc_data->export_id;
  350. rc = habmm_socket_send(msm_audio_ion_hab_handle,
  351. (void *)&smmu_unmap_cmd,
  352. sizeof(smmu_unmap_cmd), 0);
  353. if (rc) {
  354. pr_err("%s: habmm_socket_send failed %d\n",
  355. __func__, rc);
  356. goto err;
  357. }
  358. do {
  359. cmd_rsp_size = sizeof(cmd_rsp);
  360. rc = habmm_socket_recv(msm_audio_ion_hab_handle,
  361. (void *)&cmd_rsp,
  362. &cmd_rsp_size,
  363. 0xFFFFFFFF,
  364. 0);
  365. } while (time_before(jiffies, delay) &&
  366. (rc == -EINTR) && (cmd_rsp_size == 0));
  367. if (rc) {
  368. pr_err("%s: habmm_socket_recv failed %d\n",
  369. __func__, rc);
  370. goto err;
  371. }
  372. if (cmd_rsp_size != sizeof(cmd_rsp)) {
  373. pr_err("%s: invalid size for cmd rsp %u\n",
  374. __func__, cmd_rsp_size);
  375. rc = -EIO;
  376. goto err;
  377. }
  378. if (cmd_rsp.status) {
  379. pr_err("%s: SMMU unmap command failed %d\n",
  380. __func__, cmd_rsp.status);
  381. rc = cmd_rsp.status;
  382. goto err;
  383. }
  384. rc = habmm_unexport(msm_audio_ion_hab_handle,
  385. alloc_data->export_id, 0xFFFFFFFF);
  386. if (rc) {
  387. pr_err("%s: habmm_unexport failed export_id = %d, rc = %d\n",
  388. __func__, alloc_data->export_id, rc);
  389. }
  390. break;
  391. }
  392. }
  393. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  394. if (!found) {
  395. pr_err("%s: cannot find allocation, handle %pK\n", __func__, handle);
  396. rc = -EINVAL;
  397. }
  398. return rc;
  399. err:
  400. if (found) {
  401. (void)habmm_unexport(msm_audio_ion_hab_handle,
  402. alloc_data->export_id, 0xFFFFFFFF);
  403. list_del(&(alloc_data->list));
  404. kfree(alloc_data);
  405. }
  406. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  407. return rc;
  408. }
  409. static int msm_audio_ion_get_phys(struct dma_buf *dma_buf,
  410. dma_addr_t *addr, size_t *len)
  411. {
  412. int rc = 0;
  413. rc = msm_audio_ion_dma_buf_map(dma_buf, addr, len);
  414. if (rc) {
  415. pr_err("%s: failed to map DMA buf, err = %d\n",
  416. __func__, rc);
  417. goto err;
  418. }
  419. pr_debug("phys=%pK, len=%zd, rc=%d\n", addr, *len, rc);
  420. err:
  421. return rc;
  422. }
  423. static void *msm_audio_ion_map_kernel(void *handle)
  424. {
  425. int rc = 0;
  426. void *addr = NULL;
  427. struct msm_audio_alloc_data *alloc_data = NULL;
  428. rc = dma_buf_begin_cpu_access((struct dma_buf*)handle,
  429. DMA_BIDIRECTIONAL);
  430. if (rc) {
  431. pr_err("%s: kmap dma_buf_begin_cpu_access fail\n", __func__);
  432. goto exit;
  433. }
  434. addr = dma_buf_vmap((struct dma_buf*)handle);
  435. if (!addr) {
  436. pr_err("%s: kernel mapping of dma_buf failed\n",
  437. __func__);
  438. goto exit;
  439. }
  440. /*
  441. * TBD: remove the below section once new API
  442. * for mapping kernel virtual address is available.
  443. */
  444. mutex_lock(&(msm_audio_ion_data.list_mutex));
  445. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  446. list) {
  447. if (alloc_data->handle == handle) {
  448. alloc_data->vaddr = addr;
  449. break;
  450. }
  451. }
  452. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  453. exit:
  454. return addr;
  455. }
  456. static int msm_audio_ion_map_buf(void *handle, dma_addr_t *paddr,
  457. size_t *plen, void **vaddr)
  458. {
  459. int rc = 0;
  460. rc = msm_audio_ion_get_phys((struct dma_buf*) handle, paddr, plen);
  461. if (rc) {
  462. pr_err("%s: ION Get Physical for AUDIO failed, rc = %d\n",
  463. __func__, rc);
  464. dma_buf_put(dma_buf);
  465. goto err;
  466. }
  467. *vaddr = msm_audio_ion_map_kernel(handle);
  468. if (IS_ERR_OR_NULL(*vaddr)) {
  469. pr_err("%s: ION memory mapping for AUDIO failed\n", __func__);
  470. rc = -ENOMEM;
  471. msm_audio_dma_buf_unmap(dma_buf);
  472. goto err;
  473. }
  474. if (msm_audio_ion_data.smmu_enabled) {
  475. rc = msm_audio_ion_smmu_map(handle, paddr, plen);
  476. if (rc) {
  477. pr_err("%s: failed to do smmu map, err = %d\n",
  478. __func__, rc);
  479. msm_audio_dma_buf_unmap((struct dma_buf *) handle);
  480. goto err;
  481. }
  482. }
  483. err:
  484. return rc;
  485. }
  486. /**
  487. * msm_audio_ion_alloc -
  488. * Allocs ION memory for given client name
  489. *
  490. * @handle: generic handle to the memory allocation
  491. * dma_buf for the system heap memory. vaddr for audio heap memory.
  492. * @bufsz: buffer size
  493. * @paddr: Physical address to be assigned with allocated region
  494. * @plen: length of allocated region to be assigned
  495. * vaddr: virtual address to be assigned
  496. *
  497. * Returns 0 on success or error on failure
  498. */
  499. int msm_audio_ion_alloc(void **handle, size_t bufsz,
  500. dma_addr_t *paddr, size_t *plen, void **vaddr)
  501. {
  502. int rc = -EINVAL;
  503. unsigned long err_ion_ptr = 0;
  504. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  505. pr_debug("%s:probe is not done, deferred\n", __func__);
  506. return -EPROBE_DEFER;
  507. }
  508. if (!handle || !paddr || !vaddr || !bufsz || !plen) {
  509. pr_err("%s: Invalid params\n", __func__);
  510. return -EINVAL;
  511. }
  512. if (msm_audio_ion_data.smmu_enabled == true) {
  513. pr_debug("%s: system heap is used\n", __func__);
  514. *handle = ion_alloc(bufsz, ION_HEAP(ION_SYSTEM_HEAP_ID), 0);
  515. } else {
  516. pr_debug("%s: audio heap is used\n", __func__);
  517. *vaddr = *handle = dma_alloc_coherent(
  518. msm_audio_ion_data.cb_dev,
  519. bufsz, paddr, GFP_KERNEL);
  520. if(*vaddr != NULL) {
  521. pr_err("%s: vaddr = %pK, size=%zd\n", __func__, *vaddr,
  522. bufsz);
  523. rc = 0;
  524. }
  525. }
  526. if (IS_ERR_OR_NULL((void *)(*handle))) {
  527. if (IS_ERR((void *)(*handle)))
  528. err_ion_ptr = PTR_ERR((int *)(*handle));
  529. pr_err("%s: ION alloc fail err ptr=%ld, smmu_enabled=%d\n",
  530. __func__, err_ion_ptr, msm_audio_ion_data.smmu_enabled);
  531. rc = -ENOMEM;
  532. goto err;
  533. }
  534. if (msm_audio_ion_data.smmu_enabled) {
  535. rc = msm_audio_ion_map_buf(*handle, paddr, plen, vaddr);
  536. if (rc) {
  537. pr_err("%s: failed to map ION buf, rc = %d\n", __func__,
  538. rc);
  539. }
  540. } else {
  541. rc = msm_audio_dma_buf_map(*handle, *vaddr, paddr,
  542. &bufsz);
  543. if (rc) {
  544. pr_err("%s: failed to map ION buf, rc = %d\n", __func__,
  545. rc);
  546. dma_free_coherent(msm_audio_ion_data.cb_dev,
  547. bufsz, vaddr, *paddr);
  548. }
  549. }
  550. pr_debug("%s: mapped address = %pK, size=%zd\n", __func__,
  551. *vaddr, bufsz);
  552. memset(*vaddr, 0, bufsz);
  553. err:
  554. return rc;
  555. }
  556. EXPORT_SYMBOL(msm_audio_ion_alloc);
  557. int msm_audio_ion_phys_free(void *handle,
  558. dma_addr_t *paddr,
  559. size_t *pa_len,
  560. u8 assign_type,
  561. int id,
  562. int key)
  563. {
  564. handle = NULL;
  565. return 0;
  566. }
  567. EXPORT_SYMBOL(msm_audio_ion_phys_free);
  568. int msm_audio_ion_phys_assign(void **handle, int fd,
  569. dma_addr_t *paddr, size_t *pa_len, u8 assign_type, int id)
  570. {
  571. *handle = NULL;
  572. return 0;
  573. }
  574. EXPORT_SYMBOL(msm_audio_ion_phys_assign);
  575. bool msm_audio_is_hypervisor_supported(void)
  576. {
  577. return false;
  578. }
  579. EXPORT_SYMBOL(msm_audio_is_hypervisor_supported);
  580. /**
  581. * msm_audio_ion_import-
  582. * Import ION buffer with given file descriptor
  583. *
  584. * @handle: generic handle to the memory allocation
  585. * dma_buf for the system heap memory. vaddr for audio heap memory.
  586. * @fd: file descriptor for the ION memory
  587. * @ionflag: flags associated with ION buffer
  588. * @bufsz: buffer size
  589. * @paddr: Physical address to be assigned with allocated region
  590. * @plen: length of allocated region to be assigned
  591. * vaddr: virtual address to be assigned
  592. *
  593. * Returns 0 on success or error on failure
  594. */
  595. int msm_audio_ion_import(void **handle, int fd,
  596. unsigned long *ionflag, size_t bufsz,
  597. dma_addr_t *paddr, size_t *plen, void **vaddr)
  598. {
  599. int rc = 0;
  600. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  601. pr_debug("%s: probe is not done, deferred\n", __func__);
  602. return -EPROBE_DEFER;
  603. }
  604. if (!handle || !paddr || !vaddr || !plen) {
  605. pr_err("%s: Invalid params\n", __func__);
  606. return -EINVAL;
  607. }
  608. /* bufsz should be 0 and fd shouldn't be 0 as of now */
  609. *handle = dma_buf_get(fd);
  610. pr_debug("%s: handle =%pK, fd=%d\n", __func__, *handle, fd);
  611. if (IS_ERR_OR_NULL((void *)(*handle))) {
  612. pr_err("%s: dma_buf_get failed\n", __func__);
  613. rc = -EINVAL;
  614. goto err;
  615. }
  616. if (ionflag != NULL) {
  617. rc = dma_buf_get_flags((struct dma_buf*)*handle, ionflag);
  618. if (rc) {
  619. pr_err("%s: could not get flags for the dma_buf\n",
  620. __func__);
  621. goto err_ion_flag;
  622. }
  623. }
  624. rc = msm_audio_ion_map_buf(*handle, paddr, plen, vaddr);
  625. if (rc) {
  626. pr_err("%s: failed to map ION buf, rc = %d\n", __func__, rc);
  627. goto err;
  628. }
  629. pr_debug("%s: mapped address = %pK, size=%zd\n", __func__,
  630. *vaddr, bufsz);
  631. return 0;
  632. err_ion_flag:
  633. dma_buf_put((struct dma_buf*) *handle);
  634. err:
  635. *handle = NULL;
  636. return rc;
  637. }
  638. EXPORT_SYMBOL(msm_audio_ion_import);
  639. /**
  640. * msm_audio_ion_free -
  641. * fress ION memory for given client and handle
  642. *
  643. * @handle: generic handle to the memory allocation
  644. * dma_buf for the system heap memory. vaddr for audio heap memory.
  645. *
  646. * Returns 0 on success or error on failure
  647. */
  648. int msm_audio_ion_free(void *handle)
  649. {
  650. int ret = 0;
  651. if (!handle) {
  652. pr_err("%s: handle invalid\n", __func__);
  653. return -EINVAL;
  654. }
  655. if (msm_audio_ion_data.smmu_enabled) {
  656. ret = msm_audio_ion_smmu_unmap(handle);
  657. if (ret)
  658. pr_err("%s: smmu unmap failed with ret %d\n",
  659. __func__, ret);
  660. }
  661. msm_audio_dma_buf_unmap(handle);
  662. return 0;
  663. }
  664. EXPORT_SYMBOL(msm_audio_ion_free);
  665. /**
  666. * msm_audio_ion_mmap -
  667. * Audio ION memory map
  668. *
  669. * @abuff: audio buf pointer
  670. * @vma: virtual mem area
  671. *
  672. * Returns 0 on success or error on failure
  673. */
  674. int msm_audio_ion_mmap(struct audio_buffer *abuff,
  675. struct vm_area_struct *vma)
  676. {
  677. struct msm_audio_alloc_data *alloc_data = NULL;
  678. struct sg_table *table;
  679. unsigned long addr = vma->vm_start;
  680. unsigned long offset = vma->vm_pgoff * PAGE_SIZE;
  681. struct scatterlist *sg;
  682. unsigned int i;
  683. struct page *page;
  684. int ret = 0;
  685. bool found = false;
  686. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  687. mutex_lock(&(msm_audio_ion_data.list_mutex));
  688. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  689. list) {
  690. if (alloc_data->handle == abuff->mem_handle) {
  691. found = true;
  692. table = alloc_data->table;
  693. break;
  694. }
  695. }
  696. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  697. if (!found) {
  698. dev_err(cb_dev,
  699. "%s: cannot find allocation, dma_buf %pK",
  700. __func__, abuff->mem_handle);
  701. return -EINVAL;
  702. }
  703. /* uncached */
  704. vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
  705. /* We need to check if a page is associated with this sg list because:
  706. * If the allocation came from a carveout we currently don't have
  707. * pages associated with carved out memory. This might change in the
  708. * future and we can remove this check and the else statement.
  709. */
  710. page = sg_page(table->sgl);
  711. if (page) {
  712. pr_debug("%s: page is NOT null\n", __func__);
  713. for_each_sg(table->sgl, sg, table->nents, i) {
  714. unsigned long remainder = vma->vm_end - addr;
  715. unsigned long len = sg->length;
  716. page = sg_page(sg);
  717. if (offset >= len) {
  718. offset -= len;
  719. continue;
  720. } else if (offset) {
  721. page += offset / PAGE_SIZE;
  722. len -= offset;
  723. offset = 0;
  724. }
  725. len = min(len, remainder);
  726. pr_debug("vma=%pK, addr=%x len=%ld vm_start=%x vm_end=%x vm_page_prot=%lu\n",
  727. vma, (unsigned int)addr, len,
  728. (unsigned int)vma->vm_start,
  729. (unsigned int)vma->vm_end,
  730. (unsigned long)pgprot_val(vma->vm_page_prot));
  731. remap_pfn_range(vma, addr, page_to_pfn(page), len,
  732. vma->vm_page_prot);
  733. addr += len;
  734. if (addr >= vma->vm_end)
  735. return 0;
  736. }
  737. } else {
  738. pr_debug("%s: page is NULL\n", __func__);
  739. ret = -EINVAL;
  740. }
  741. return ret;
  742. }
  743. EXPORT_SYMBOL(msm_audio_ion_mmap);
  744. /**
  745. * msm_audio_populate_upper_32_bits -
  746. * retrieve upper 32bits of 64bit address
  747. *
  748. * @pa: 64bit physical address
  749. *
  750. */
  751. u32 msm_audio_populate_upper_32_bits(dma_addr_t pa)
  752. {
  753. return upper_32_bits(pa);
  754. }
  755. EXPORT_SYMBOL(msm_audio_populate_upper_32_bits);
  756. static const struct of_device_id msm_audio_ion_dt_match[] = {
  757. { .compatible = "qcom,msm-audio-ion" },
  758. { }
  759. };
  760. MODULE_DEVICE_TABLE(of, msm_audio_ion_dt_match);
  761. static int msm_audio_ion_probe(struct platform_device *pdev)
  762. {
  763. int rc = 0;
  764. const char *msm_audio_ion_dt = "qcom,smmu-enabled";
  765. bool smmu_enabled;
  766. struct device *dev = &pdev->dev;
  767. if (dev->of_node == NULL) {
  768. dev_err(dev,
  769. "%s: device tree is not found\n",
  770. __func__);
  771. msm_audio_ion_data.smmu_enabled = 0;
  772. return 0;
  773. }
  774. smmu_enabled = of_property_read_bool(dev->of_node,
  775. msm_audio_ion_dt);
  776. msm_audio_ion_data.smmu_enabled = smmu_enabled;
  777. if (!smmu_enabled) {
  778. dev_dbg(dev, "%s: SMMU is Disabled\n", __func__);
  779. goto exit;
  780. }
  781. rc = habmm_socket_open(&msm_audio_ion_hab_handle,
  782. HAB_MMID_CREATE(MM_AUD_3,
  783. MSM_AUDIO_SMMU_VM_HAB_MINOR_ID),
  784. 0xFFFFFFFF,
  785. HABMM_SOCKET_OPEN_FLAGS_SINGLE_BE_SINGLE_FE);
  786. if (rc) {
  787. dev_err(dev, "%s: habmm_socket_open failed %d\n",
  788. __func__, rc);
  789. return rc;
  790. }
  791. dev_info(dev, "%s: msm_audio_ion_hab_handle %x\n",
  792. __func__, msm_audio_ion_hab_handle);
  793. exit:
  794. if (!rc)
  795. msm_audio_ion_data.device_status |= MSM_AUDIO_ION_PROBED;
  796. msm_audio_ion_data.cb_dev = dev;
  797. INIT_LIST_HEAD(&msm_audio_ion_data.alloc_list);
  798. mutex_init(&(msm_audio_ion_data.list_mutex));
  799. return rc;
  800. }
  801. static int msm_audio_ion_remove(struct platform_device *pdev)
  802. {
  803. if (msm_audio_ion_data.smmu_enabled) {
  804. if (msm_audio_ion_hab_handle)
  805. habmm_socket_close(msm_audio_ion_hab_handle);
  806. }
  807. msm_audio_ion_data.smmu_enabled = 0;
  808. msm_audio_ion_data.device_status = 0;
  809. mutex_destroy(&(msm_audio_ion_data.list_mutex));
  810. return 0;
  811. }
  812. static struct platform_driver msm_audio_ion_driver = {
  813. .driver = {
  814. .name = "msm-audio-ion",
  815. .owner = THIS_MODULE,
  816. .of_match_table = msm_audio_ion_dt_match,
  817. },
  818. .probe = msm_audio_ion_probe,
  819. .remove = msm_audio_ion_remove,
  820. };
  821. int __init msm_audio_ion_init(void)
  822. {
  823. return platform_driver_register(&msm_audio_ion_driver);
  824. }
  825. void msm_audio_ion_exit(void)
  826. {
  827. platform_driver_unregister(&msm_audio_ion_driver);
  828. }
  829. MODULE_DESCRIPTION("MSM Audio ION VM module");
  830. MODULE_LICENSE("GPL v2");