msm_audio_ion_vm.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2013-2019, 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. void *vaddr;
  262. *len = ((struct dma_buf*)handle)->size;
  263. mutex_lock(&(msm_audio_ion_data.list_mutex));
  264. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  265. list) {
  266. if (alloc_data->handle == handle) {
  267. found = true;
  268. vaddr = alloc_data->vaddr;
  269. /* Export the buffer to physical VM */
  270. rc = habmm_export(msm_audio_ion_hab_handle, vaddr, *len,
  271. &export_id, 0);
  272. if (rc) {
  273. pr_err("%s: habmm_export failed vaddr = %pK, len = %zd, rc = %d\n",
  274. __func__, vaddr, *len, rc);
  275. goto err;
  276. }
  277. exported = true;
  278. smmu_map_cmd.cmd_id = MSM_AUDIO_SMMU_VM_CMD_MAP;
  279. smmu_map_cmd.export_id = export_id;
  280. smmu_map_cmd.buf_size = *len;
  281. rc = habmm_socket_send(msm_audio_ion_hab_handle,
  282. (void *)&smmu_map_cmd, sizeof(smmu_map_cmd), 0);
  283. if (rc) {
  284. pr_err("%s: habmm_socket_send failed %d\n",
  285. __func__, rc);
  286. goto err;
  287. }
  288. do {
  289. cmd_rsp_size = sizeof(cmd_rsp);
  290. rc = habmm_socket_recv(msm_audio_ion_hab_handle,
  291. (void *)&cmd_rsp,
  292. &cmd_rsp_size,
  293. 0xFFFFFFFF,
  294. 0);
  295. } while (time_before(jiffies, delay) && (rc == -EINTR) &&
  296. (cmd_rsp_size == 0));
  297. if (rc) {
  298. pr_err("%s: habmm_socket_recv failed %d\n",
  299. __func__, rc);
  300. goto err;
  301. }
  302. if (cmd_rsp_size != sizeof(cmd_rsp)) {
  303. pr_err("%s: invalid size for cmd rsp %u, expected %zu\n",
  304. __func__, cmd_rsp_size, sizeof(cmd_rsp));
  305. rc = -EIO;
  306. goto err;
  307. }
  308. if (cmd_rsp.status) {
  309. pr_err("%s: SMMU map command failed %d\n",
  310. __func__, cmd_rsp.status);
  311. rc = cmd_rsp.status;
  312. goto err;
  313. }
  314. *paddr = (dma_addr_t)cmd_rsp.addr;
  315. alloc_data->export_id = export_id;
  316. break;
  317. }
  318. }
  319. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  320. if (!found) {
  321. pr_err("%s: cannot find allocation, handle %pK\n", __func__, handle);
  322. return -EINVAL;
  323. }
  324. return 0;
  325. err:
  326. if (exported)
  327. (void)habmm_unexport(msm_audio_ion_hab_handle, export_id, 0);
  328. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  329. return rc;
  330. }
  331. static int msm_audio_ion_smmu_unmap(void *handle)
  332. {
  333. int rc;
  334. bool found = false;
  335. u32 cmd_rsp_size;
  336. struct msm_audio_smmu_vm_unmap_cmd smmu_unmap_cmd;
  337. struct msm_audio_smmu_vm_unmap_cmd_rsp cmd_rsp;
  338. struct msm_audio_alloc_data *alloc_data, *next;
  339. unsigned long delay = jiffies + (HZ / 2);
  340. /*
  341. * Though list_for_each_entry_safe is delete safe, lock
  342. * should be explicitly acquired to avoid race condition
  343. * on adding elements to the list.
  344. */
  345. mutex_lock(&(msm_audio_ion_data.list_mutex));
  346. list_for_each_entry_safe(alloc_data, next,
  347. &(msm_audio_ion_data.alloc_list), list) {
  348. if (alloc_data->handle == handle) {
  349. found = true;
  350. smmu_unmap_cmd.cmd_id = MSM_AUDIO_SMMU_VM_CMD_UNMAP;
  351. smmu_unmap_cmd.export_id = alloc_data->export_id;
  352. rc = habmm_socket_send(msm_audio_ion_hab_handle,
  353. (void *)&smmu_unmap_cmd,
  354. sizeof(smmu_unmap_cmd), 0);
  355. if (rc) {
  356. pr_err("%s: habmm_socket_send failed %d\n",
  357. __func__, rc);
  358. goto err;
  359. }
  360. do {
  361. cmd_rsp_size = sizeof(cmd_rsp);
  362. rc = habmm_socket_recv(msm_audio_ion_hab_handle,
  363. (void *)&cmd_rsp,
  364. &cmd_rsp_size,
  365. 0xFFFFFFFF,
  366. 0);
  367. } while (time_before(jiffies, delay) &&
  368. (rc == -EINTR) && (cmd_rsp_size == 0));
  369. if (rc) {
  370. pr_err("%s: habmm_socket_recv failed %d\n",
  371. __func__, rc);
  372. goto err;
  373. }
  374. if (cmd_rsp_size != sizeof(cmd_rsp)) {
  375. pr_err("%s: invalid size for cmd rsp %u\n",
  376. __func__, cmd_rsp_size);
  377. rc = -EIO;
  378. goto err;
  379. }
  380. if (cmd_rsp.status) {
  381. pr_err("%s: SMMU unmap command failed %d\n",
  382. __func__, cmd_rsp.status);
  383. rc = cmd_rsp.status;
  384. goto err;
  385. }
  386. rc = habmm_unexport(msm_audio_ion_hab_handle,
  387. alloc_data->export_id, 0xFFFFFFFF);
  388. if (rc) {
  389. pr_err("%s: habmm_unexport failed export_id = %d, rc = %d\n",
  390. __func__, alloc_data->export_id, rc);
  391. }
  392. break;
  393. }
  394. }
  395. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  396. if (!found) {
  397. pr_err("%s: cannot find allocation, handle %pK\n", __func__, handle);
  398. rc = -EINVAL;
  399. }
  400. return rc;
  401. err:
  402. if (found) {
  403. (void)habmm_unexport(msm_audio_ion_hab_handle,
  404. alloc_data->export_id, 0xFFFFFFFF);
  405. list_del(&(alloc_data->list));
  406. kfree(alloc_data);
  407. }
  408. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  409. return rc;
  410. }
  411. static int msm_audio_ion_get_phys(struct dma_buf *dma_buf,
  412. dma_addr_t *addr, size_t *len)
  413. {
  414. int rc = 0;
  415. rc = msm_audio_ion_dma_buf_map(dma_buf, addr, len);
  416. if (rc) {
  417. pr_err("%s: failed to map DMA buf, err = %d\n",
  418. __func__, rc);
  419. goto err;
  420. }
  421. pr_debug("phys=%pK, len=%zd, rc=%d\n", addr, *len, rc);
  422. err:
  423. return rc;
  424. }
  425. static void *msm_audio_ion_map_kernel(void *handle)
  426. {
  427. int rc = 0;
  428. void *addr = NULL;
  429. struct msm_audio_alloc_data *alloc_data = NULL;
  430. rc = dma_buf_begin_cpu_access((struct dma_buf*)handle,
  431. DMA_BIDIRECTIONAL);
  432. if (rc) {
  433. pr_err("%s: kmap dma_buf_begin_cpu_access fail\n", __func__);
  434. goto exit;
  435. }
  436. addr = dma_buf_vmap((struct dma_buf*)handle);
  437. if (!addr) {
  438. pr_err("%s: kernel mapping of dma_buf failed\n",
  439. __func__);
  440. goto exit;
  441. }
  442. /*
  443. * TBD: remove the below section once new API
  444. * for mapping kernel virtual address is available.
  445. */
  446. mutex_lock(&(msm_audio_ion_data.list_mutex));
  447. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  448. list) {
  449. if (alloc_data->handle == handle) {
  450. alloc_data->vaddr = addr;
  451. break;
  452. }
  453. }
  454. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  455. exit:
  456. return addr;
  457. }
  458. static int msm_audio_ion_map_buf(void *handle, dma_addr_t *paddr,
  459. size_t *plen, void **vaddr)
  460. {
  461. int rc = 0;
  462. rc = msm_audio_ion_get_phys((struct dma_buf*) handle, paddr, plen);
  463. if (rc) {
  464. pr_err("%s: ION Get Physical for AUDIO failed, rc = %d\n",
  465. __func__, rc);
  466. dma_buf_put(dma_buf);
  467. goto err;
  468. }
  469. *vaddr = msm_audio_ion_map_kernel(handle);
  470. if (IS_ERR_OR_NULL(*vaddr)) {
  471. pr_err("%s: ION memory mapping for AUDIO failed\n", __func__);
  472. rc = -ENOMEM;
  473. msm_audio_dma_buf_unmap(dma_buf);
  474. goto err;
  475. }
  476. if (msm_audio_ion_data.smmu_enabled) {
  477. rc = msm_audio_ion_smmu_map(handle, paddr, plen);
  478. if (rc) {
  479. pr_err("%s: failed to do smmu map, err = %d\n",
  480. __func__, rc);
  481. goto err;
  482. }
  483. }
  484. err:
  485. return rc;
  486. }
  487. /**
  488. * msm_audio_ion_alloc -
  489. * Allocs ION memory for given client name
  490. *
  491. * @handle: generic handle to the memory allocation
  492. * dma_buf for the system heap memory. vaddr for audio heap memory.
  493. * @bufsz: buffer size
  494. * @paddr: Physical address to be assigned with allocated region
  495. * @plen: length of allocated region to be assigned
  496. * vaddr: virtual address to be assigned
  497. *
  498. * Returns 0 on success or error on failure
  499. */
  500. int msm_audio_ion_alloc(void **handle, size_t bufsz,
  501. dma_addr_t *paddr, size_t *plen, void **vaddr)
  502. {
  503. int rc = -EINVAL;
  504. unsigned long err_ion_ptr = 0;
  505. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  506. pr_debug("%s:probe is not done, deferred\n", __func__);
  507. return -EPROBE_DEFER;
  508. }
  509. if (!handle || !paddr || !vaddr || !bufsz || !plen) {
  510. pr_err("%s: Invalid params\n", __func__);
  511. return -EINVAL;
  512. }
  513. if (msm_audio_ion_data.smmu_enabled == true) {
  514. pr_debug("%s: system heap is used\n", __func__);
  515. *handle = ion_alloc(bufsz, ION_HEAP(ION_SYSTEM_HEAP_ID), 0);
  516. } else {
  517. pr_debug("%s: audio heap is used\n", __func__);
  518. *vaddr = *handle = dma_alloc_coherent(
  519. msm_audio_ion_data.cb_dev,
  520. bufsz, paddr, GFP_KERNEL);
  521. if(*vaddr != NULL) {
  522. pr_err("%s: vaddr = %pK, size=%zd\n", __func__, *vaddr,
  523. bufsz);
  524. rc = 0;
  525. }
  526. }
  527. if (IS_ERR_OR_NULL((void *)(*handle))) {
  528. if (IS_ERR((void *)(*handle)))
  529. err_ion_ptr = PTR_ERR((int *)(*handle));
  530. pr_err("%s: ION alloc fail err ptr=%ld, smmu_enabled=%d\n",
  531. __func__, err_ion_ptr, msm_audio_ion_data.smmu_enabled);
  532. rc = -ENOMEM;
  533. goto err;
  534. }
  535. if (msm_audio_ion_data.smmu_enabled) {
  536. rc = msm_audio_ion_map_buf(*handle, paddr, plen, vaddr);
  537. if (rc) {
  538. pr_err("%s: failed to map ION buf, rc = %d\n", __func__,
  539. rc);
  540. }
  541. } else {
  542. rc = msm_audio_dma_buf_map(*handle, *vaddr, paddr,
  543. &bufsz);
  544. if (rc) {
  545. pr_err("%s: failed to map ION buf, rc = %d\n", __func__,
  546. rc);
  547. dma_free_coherent(msm_audio_ion_data.cb_dev,
  548. bufsz, vaddr, *paddr);
  549. }
  550. }
  551. pr_debug("%s: mapped address = %pK, size=%zd\n", __func__,
  552. *vaddr, bufsz);
  553. memset(*vaddr, 0, bufsz);
  554. err:
  555. return rc;
  556. }
  557. EXPORT_SYMBOL(msm_audio_ion_alloc);
  558. /**
  559. * msm_audio_ion_import-
  560. * Import ION buffer with given file descriptor
  561. *
  562. * @handle: generic handle to the memory allocation
  563. * dma_buf for the system heap memory. vaddr for audio heap memory.
  564. * @fd: file descriptor for the ION memory
  565. * @ionflag: flags associated with ION buffer
  566. * @bufsz: buffer size
  567. * @paddr: Physical address to be assigned with allocated region
  568. * @plen: length of allocated region to be assigned
  569. * vaddr: virtual address to be assigned
  570. *
  571. * Returns 0 on success or error on failure
  572. */
  573. int msm_audio_ion_import(void **handle, int fd,
  574. unsigned long *ionflag, size_t bufsz,
  575. dma_addr_t *paddr, size_t *plen, void **vaddr)
  576. {
  577. int rc = 0;
  578. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  579. pr_debug("%s: probe is not done, deferred\n", __func__);
  580. return -EPROBE_DEFER;
  581. }
  582. if (!handle || !paddr || !vaddr || !plen) {
  583. pr_err("%s: Invalid params\n", __func__);
  584. return -EINVAL;
  585. }
  586. /* bufsz should be 0 and fd shouldn't be 0 as of now */
  587. *handle = dma_buf_get(fd);
  588. pr_debug("%s: handle =%pK, fd=%d\n", __func__, *handle, fd);
  589. if (IS_ERR_OR_NULL((void *)(*handle))) {
  590. pr_err("%s: dma_buf_get failed\n", __func__);
  591. rc = -EINVAL;
  592. goto err;
  593. }
  594. if (ionflag != NULL) {
  595. rc = dma_buf_get_flags((struct dma_buf*)*handle, ionflag);
  596. if (rc) {
  597. pr_err("%s: could not get flags for the dma_buf\n",
  598. __func__);
  599. goto err_ion_flag;
  600. }
  601. }
  602. rc = msm_audio_ion_map_buf(*handle, paddr, plen, vaddr);
  603. if (rc) {
  604. pr_err("%s: failed to map ION buf, rc = %d\n", __func__, rc);
  605. goto err;
  606. }
  607. pr_debug("%s: mapped address = %pK, size=%zd\n", __func__,
  608. *vaddr, bufsz);
  609. return 0;
  610. err_ion_flag:
  611. dma_buf_put((struct dma_buf*) *handle);
  612. err:
  613. *handle = NULL;
  614. return rc;
  615. }
  616. EXPORT_SYMBOL(msm_audio_ion_import);
  617. /**
  618. * msm_audio_ion_free -
  619. * fress ION memory for given client and handle
  620. *
  621. * @handle: generic handle to the memory allocation
  622. * dma_buf for the system heap memory. vaddr for audio heap memory.
  623. *
  624. * Returns 0 on success or error on failure
  625. */
  626. int msm_audio_ion_free(void *handle)
  627. {
  628. int ret = 0;
  629. if (!handle) {
  630. pr_err("%s: handle invalid\n", __func__);
  631. return -EINVAL;
  632. }
  633. if (msm_audio_ion_data.smmu_enabled) {
  634. ret = msm_audio_ion_smmu_unmap(handle);
  635. if (ret)
  636. pr_err("%s: smmu unmap failed with ret %d\n",
  637. __func__, ret);
  638. }
  639. msm_audio_dma_buf_unmap(handle);
  640. return 0;
  641. }
  642. EXPORT_SYMBOL(msm_audio_ion_free);
  643. /**
  644. * msm_audio_ion_mmap -
  645. * Audio ION memory map
  646. *
  647. * @abuff: audio buf pointer
  648. * @vma: virtual mem area
  649. *
  650. * Returns 0 on success or error on failure
  651. */
  652. int msm_audio_ion_mmap(struct audio_buffer *abuff,
  653. struct vm_area_struct *vma)
  654. {
  655. struct msm_audio_alloc_data *alloc_data = NULL;
  656. struct sg_table *table;
  657. unsigned long addr = vma->vm_start;
  658. unsigned long offset = vma->vm_pgoff * PAGE_SIZE;
  659. struct scatterlist *sg;
  660. unsigned int i;
  661. struct page *page;
  662. int ret = 0;
  663. bool found = false;
  664. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  665. mutex_lock(&(msm_audio_ion_data.list_mutex));
  666. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  667. list) {
  668. if (alloc_data->handle == abuff->mem_handle) {
  669. found = true;
  670. table = alloc_data->table;
  671. break;
  672. }
  673. }
  674. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  675. if (!found) {
  676. dev_err(cb_dev,
  677. "%s: cannot find allocation, dma_buf %pK",
  678. __func__, abuff->mem_handle);
  679. return -EINVAL;
  680. }
  681. /* uncached */
  682. vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
  683. /* We need to check if a page is associated with this sg list because:
  684. * If the allocation came from a carveout we currently don't have
  685. * pages associated with carved out memory. This might change in the
  686. * future and we can remove this check and the else statement.
  687. */
  688. page = sg_page(table->sgl);
  689. if (page) {
  690. pr_debug("%s: page is NOT null\n", __func__);
  691. for_each_sg(table->sgl, sg, table->nents, i) {
  692. unsigned long remainder = vma->vm_end - addr;
  693. unsigned long len = sg->length;
  694. page = sg_page(sg);
  695. if (offset >= len) {
  696. offset -= len;
  697. continue;
  698. } else if (offset) {
  699. page += offset / PAGE_SIZE;
  700. len -= offset;
  701. offset = 0;
  702. }
  703. len = min(len, remainder);
  704. pr_debug("vma=%pK, addr=%x len=%ld vm_start=%x vm_end=%x vm_page_prot=%lu\n",
  705. vma, (unsigned int)addr, len,
  706. (unsigned int)vma->vm_start,
  707. (unsigned int)vma->vm_end,
  708. (unsigned long)pgprot_val(vma->vm_page_prot));
  709. remap_pfn_range(vma, addr, page_to_pfn(page), len,
  710. vma->vm_page_prot);
  711. addr += len;
  712. if (addr >= vma->vm_end)
  713. return 0;
  714. }
  715. } else {
  716. pr_debug("%s: page is NULL\n", __func__);
  717. ret = -EINVAL;
  718. }
  719. return ret;
  720. }
  721. EXPORT_SYMBOL(msm_audio_ion_mmap);
  722. /**
  723. * msm_audio_populate_upper_32_bits -
  724. * retrieve upper 32bits of 64bit address
  725. *
  726. * @pa: 64bit physical address
  727. *
  728. */
  729. u32 msm_audio_populate_upper_32_bits(dma_addr_t pa)
  730. {
  731. return upper_32_bits(pa);
  732. }
  733. EXPORT_SYMBOL(msm_audio_populate_upper_32_bits);
  734. static const struct of_device_id msm_audio_ion_dt_match[] = {
  735. { .compatible = "qcom,msm-audio-ion" },
  736. { }
  737. };
  738. MODULE_DEVICE_TABLE(of, msm_audio_ion_dt_match);
  739. static int msm_audio_ion_probe(struct platform_device *pdev)
  740. {
  741. int rc = 0;
  742. const char *msm_audio_ion_dt = "qcom,smmu-enabled";
  743. bool smmu_enabled;
  744. struct device *dev = &pdev->dev;
  745. if (dev->of_node == NULL) {
  746. dev_err(dev,
  747. "%s: device tree is not found\n",
  748. __func__);
  749. msm_audio_ion_data.smmu_enabled = 0;
  750. return 0;
  751. }
  752. smmu_enabled = of_property_read_bool(dev->of_node,
  753. msm_audio_ion_dt);
  754. msm_audio_ion_data.smmu_enabled = smmu_enabled;
  755. if (!smmu_enabled) {
  756. dev_dbg(dev, "%s: SMMU is Disabled\n", __func__);
  757. goto exit;
  758. }
  759. rc = habmm_socket_open(&msm_audio_ion_hab_handle,
  760. HAB_MMID_CREATE(MM_AUD_3,
  761. MSM_AUDIO_SMMU_VM_HAB_MINOR_ID),
  762. 0xFFFFFFFF,
  763. HABMM_SOCKET_OPEN_FLAGS_SINGLE_BE_SINGLE_FE);
  764. if (rc) {
  765. dev_err(dev, "%s: habmm_socket_open failed %d\n",
  766. __func__, rc);
  767. return rc;
  768. }
  769. dev_info(dev, "%s: msm_audio_ion_hab_handle %x\n",
  770. __func__, msm_audio_ion_hab_handle);
  771. exit:
  772. if (!rc)
  773. msm_audio_ion_data.device_status |= MSM_AUDIO_ION_PROBED;
  774. msm_audio_ion_data.cb_dev = dev;
  775. INIT_LIST_HEAD(&msm_audio_ion_data.alloc_list);
  776. mutex_init(&(msm_audio_ion_data.list_mutex));
  777. return rc;
  778. }
  779. static int msm_audio_ion_remove(struct platform_device *pdev)
  780. {
  781. if (msm_audio_ion_data.smmu_enabled) {
  782. if (msm_audio_ion_hab_handle)
  783. habmm_socket_close(msm_audio_ion_hab_handle);
  784. }
  785. msm_audio_ion_data.smmu_enabled = 0;
  786. msm_audio_ion_data.device_status = 0;
  787. mutex_destroy(&(msm_audio_ion_data.list_mutex));
  788. return 0;
  789. }
  790. static struct platform_driver msm_audio_ion_driver = {
  791. .driver = {
  792. .name = "msm-audio-ion",
  793. .owner = THIS_MODULE,
  794. .of_match_table = msm_audio_ion_dt_match,
  795. },
  796. .probe = msm_audio_ion_probe,
  797. .remove = msm_audio_ion_remove,
  798. };
  799. int __init msm_audio_ion_init(void)
  800. {
  801. return platform_driver_register(&msm_audio_ion_driver);
  802. }
  803. void msm_audio_ion_exit(void)
  804. {
  805. platform_driver_unregister(&msm_audio_ion_driver);
  806. }
  807. MODULE_DESCRIPTION("MSM Audio ION VM module");
  808. MODULE_LICENSE("GPL v2");