msm_audio_ion.c 31 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262
  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-buf.h>
  15. #include <linux/platform_device.h>
  16. #include <linux/of_device.h>
  17. #include <linux/export.h>
  18. #include <linux/ion.h>
  19. #include <linux/ioctl.h>
  20. #include <linux/cdev.h>
  21. #include <linux/fs.h>
  22. #include <linux/device.h>
  23. #ifndef CONFIG_SPF_CORE
  24. #include <ipc/apr.h>
  25. #endif
  26. #include <dsp/msm_audio_ion.h>
  27. #include <linux/msm_audio.h>
  28. #define MSM_AUDIO_ION_PROBED (1 << 0)
  29. #define MSM_AUDIO_ION_PHYS_ADDR(alloc_data) \
  30. alloc_data->table->sgl->dma_address
  31. #define MSM_AUDIO_SMMU_SID_OFFSET 32
  32. #define TZ_PIL_PROTECT_MEM_SUBSYS_ID 0x0C
  33. #define TZ_PIL_CLEAR_PROTECT_MEM_SUBSYS_ID 0x0D
  34. #define MSM_AUDIO_ION_DRIVER_NAME "msm_audio_ion"
  35. #define MINOR_NUMBER_COUNT 1
  36. struct msm_audio_ion_private {
  37. bool smmu_enabled;
  38. struct device *cb_dev;
  39. struct device *cb_cma_dev;
  40. u8 device_status;
  41. struct list_head alloc_list;
  42. struct mutex list_mutex;
  43. u64 smmu_sid_bits;
  44. u32 smmu_version;
  45. bool is_non_hypervisor;
  46. /*list to store fd, phy. addr and handle data */
  47. struct list_head fd_list;
  48. /*char dev related data */
  49. dev_t ion_major;
  50. struct class *ion_class;
  51. struct device *chardev;
  52. struct cdev cdev;
  53. };
  54. struct msm_audio_alloc_data {
  55. size_t len;
  56. void *vaddr;
  57. struct dma_buf *dma_buf;
  58. struct dma_buf_attachment *attach;
  59. struct sg_table *table;
  60. struct list_head list;
  61. };
  62. static struct msm_audio_ion_private msm_audio_ion_data = {0,};
  63. struct msm_audio_fd_data {
  64. int fd;
  65. void *handle;
  66. dma_addr_t paddr;
  67. struct list_head list;
  68. };
  69. static void msm_audio_ion_add_allocation(
  70. struct msm_audio_ion_private *msm_audio_ion_data,
  71. struct msm_audio_alloc_data *alloc_data)
  72. {
  73. /*
  74. * Since these APIs can be invoked by multiple
  75. * clients, there is need to make sure the list
  76. * of allocations is always protected
  77. */
  78. mutex_lock(&(msm_audio_ion_data->list_mutex));
  79. list_add_tail(&(alloc_data->list),
  80. &(msm_audio_ion_data->alloc_list));
  81. mutex_unlock(&(msm_audio_ion_data->list_mutex));
  82. }
  83. static void *msm_audio_ion_map_kernel(struct dma_buf *dma_buf)
  84. {
  85. int rc = 0;
  86. void *addr = NULL;
  87. struct msm_audio_alloc_data *alloc_data = NULL;
  88. rc = dma_buf_begin_cpu_access(dma_buf, DMA_BIDIRECTIONAL);
  89. if (rc) {
  90. pr_err("%s: kmap dma_buf_begin_cpu_access fail\n", __func__);
  91. goto exit;
  92. }
  93. addr = dma_buf_vmap(dma_buf);
  94. if (!addr) {
  95. pr_err("%s: kernel mapping of dma_buf failed\n",
  96. __func__);
  97. goto exit;
  98. }
  99. /*
  100. * TBD: remove the below section once new API
  101. * for mapping kernel virtual address is available.
  102. */
  103. mutex_lock(&(msm_audio_ion_data.list_mutex));
  104. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  105. list) {
  106. if (alloc_data->dma_buf == dma_buf) {
  107. alloc_data->vaddr = addr;
  108. break;
  109. }
  110. }
  111. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  112. exit:
  113. return addr;
  114. }
  115. static int msm_audio_dma_buf_map(struct dma_buf *dma_buf,
  116. dma_addr_t *addr, size_t *len, bool is_iova,
  117. bool cma_mem)
  118. {
  119. struct msm_audio_alloc_data *alloc_data;
  120. struct device *cb_dev;
  121. unsigned long ionflag = 0;
  122. int rc = 0;
  123. void *vaddr = NULL;
  124. if (cma_mem)
  125. cb_dev = msm_audio_ion_data.cb_cma_dev;
  126. else
  127. cb_dev = msm_audio_ion_data.cb_dev;
  128. /* Data required per buffer mapping */
  129. alloc_data = kzalloc(sizeof(*alloc_data), GFP_KERNEL);
  130. if (!alloc_data)
  131. return -ENOMEM;
  132. alloc_data->dma_buf = dma_buf;
  133. alloc_data->len = dma_buf->size;
  134. *len = dma_buf->size;
  135. /* Attach the dma_buf to context bank device */
  136. alloc_data->attach = dma_buf_attach(alloc_data->dma_buf,
  137. cb_dev);
  138. if (IS_ERR(alloc_data->attach)) {
  139. rc = PTR_ERR(alloc_data->attach);
  140. dev_err(cb_dev,
  141. "%s: Fail to attach dma_buf to CB, rc = %d\n",
  142. __func__, rc);
  143. goto free_alloc_data;
  144. }
  145. /* For uncached buffers, avoid cache maintanance */
  146. rc = dma_buf_get_flags(alloc_data->dma_buf, &ionflag);
  147. if (rc) {
  148. dev_err(cb_dev, "%s: dma_buf_get_flags failed: %d\n",
  149. __func__, rc);
  150. goto detach_dma_buf;
  151. }
  152. if (!(ionflag & ION_FLAG_CACHED))
  153. alloc_data->attach->dma_map_attrs |= DMA_ATTR_SKIP_CPU_SYNC;
  154. /*
  155. * Get the scatter-gather list.
  156. * There is no info as this is a write buffer or
  157. * read buffer, hence the request is bi-directional
  158. * to accommodate both read and write mappings.
  159. */
  160. alloc_data->table = dma_buf_map_attachment(alloc_data->attach,
  161. DMA_BIDIRECTIONAL);
  162. if (IS_ERR(alloc_data->table)) {
  163. rc = PTR_ERR(alloc_data->table);
  164. dev_err(cb_dev,
  165. "%s: Fail to map attachment, rc = %d\n",
  166. __func__, rc);
  167. goto detach_dma_buf;
  168. }
  169. /* physical address from mapping */
  170. if (!is_iova) {
  171. *addr = sg_phys(alloc_data->table->sgl);
  172. vaddr = msm_audio_ion_map_kernel((void *)dma_buf);
  173. if (IS_ERR_OR_NULL(vaddr)) {
  174. pr_err("%s: ION memory mapping for AUDIO failed\n",
  175. __func__);
  176. rc = -ENOMEM;
  177. goto detach_dma_buf;
  178. }
  179. alloc_data->vaddr = vaddr;
  180. } else {
  181. *addr = MSM_AUDIO_ION_PHYS_ADDR(alloc_data);
  182. }
  183. msm_audio_ion_add_allocation(&msm_audio_ion_data,
  184. alloc_data);
  185. return rc;
  186. detach_dma_buf:
  187. dma_buf_detach(alloc_data->dma_buf,
  188. alloc_data->attach);
  189. free_alloc_data:
  190. kfree(alloc_data);
  191. return rc;
  192. }
  193. static int msm_audio_dma_buf_unmap(struct dma_buf *dma_buf, bool cma_mem)
  194. {
  195. int rc = 0;
  196. struct msm_audio_alloc_data *alloc_data = NULL;
  197. struct list_head *ptr, *next;
  198. struct device *cb_dev;
  199. bool found = false;
  200. if (cma_mem)
  201. cb_dev = msm_audio_ion_data.cb_cma_dev;
  202. else
  203. cb_dev = msm_audio_ion_data.cb_dev;
  204. /*
  205. * Though list_for_each_safe is delete safe, lock
  206. * should be explicitly acquired to avoid race condition
  207. * on adding elements to the list.
  208. */
  209. mutex_lock(&(msm_audio_ion_data.list_mutex));
  210. list_for_each_safe(ptr, next,
  211. &(msm_audio_ion_data.alloc_list)) {
  212. alloc_data = list_entry(ptr, struct msm_audio_alloc_data,
  213. list);
  214. if (alloc_data->dma_buf == dma_buf) {
  215. found = true;
  216. dma_buf_unmap_attachment(alloc_data->attach,
  217. alloc_data->table,
  218. DMA_BIDIRECTIONAL);
  219. dma_buf_detach(alloc_data->dma_buf,
  220. alloc_data->attach);
  221. dma_buf_put(alloc_data->dma_buf);
  222. list_del(&(alloc_data->list));
  223. kfree(alloc_data);
  224. break;
  225. }
  226. }
  227. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  228. if (!found) {
  229. dev_err(cb_dev,
  230. "%s: cannot find allocation, dma_buf %pK",
  231. __func__, dma_buf);
  232. rc = -EINVAL;
  233. }
  234. return rc;
  235. }
  236. static int msm_audio_ion_get_phys(struct dma_buf *dma_buf,
  237. dma_addr_t *addr, size_t *len, bool is_iova)
  238. {
  239. int rc = 0;
  240. rc = msm_audio_dma_buf_map(dma_buf, addr, len, is_iova, false);
  241. if (rc) {
  242. pr_err("%s: failed to map DMA buf, err = %d\n",
  243. __func__, rc);
  244. goto err;
  245. }
  246. if (msm_audio_ion_data.smmu_enabled && is_iova) {
  247. /* Append the SMMU SID information to the IOVA address */
  248. *addr |= msm_audio_ion_data.smmu_sid_bits;
  249. }
  250. pr_debug("phys=%pK, len=%zd, rc=%d\n", &(*addr), *len, rc);
  251. err:
  252. return rc;
  253. }
  254. int msm_audio_ion_get_smmu_info(struct device **cb_dev,
  255. u64 *smmu_sid)
  256. {
  257. if (!cb_dev || !smmu_sid) {
  258. pr_err("%s: Invalid params\n",
  259. __func__);
  260. return -EINVAL;
  261. }
  262. if (!msm_audio_ion_data.cb_dev ||
  263. !msm_audio_ion_data.smmu_sid_bits) {
  264. pr_err("%s: Params not initialized\n",
  265. __func__);
  266. return -EINVAL;
  267. }
  268. *cb_dev = msm_audio_ion_data.cb_dev;
  269. *smmu_sid = msm_audio_ion_data.smmu_sid_bits;
  270. return 0;
  271. }
  272. static int msm_audio_ion_unmap_kernel(struct dma_buf *dma_buf)
  273. {
  274. int rc = 0;
  275. void *vaddr = NULL;
  276. struct msm_audio_alloc_data *alloc_data = NULL;
  277. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  278. /*
  279. * TBD: remove the below section once new API
  280. * for unmapping kernel virtual address is available.
  281. */
  282. mutex_lock(&(msm_audio_ion_data.list_mutex));
  283. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  284. list) {
  285. if (alloc_data->dma_buf == dma_buf) {
  286. vaddr = alloc_data->vaddr;
  287. break;
  288. }
  289. }
  290. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  291. if (!vaddr) {
  292. dev_err(cb_dev,
  293. "%s: cannot find allocation for dma_buf %pK",
  294. __func__, dma_buf);
  295. rc = -EINVAL;
  296. goto err;
  297. }
  298. dma_buf_vunmap(dma_buf, vaddr);
  299. rc = dma_buf_end_cpu_access(dma_buf, DMA_BIDIRECTIONAL);
  300. if (rc) {
  301. dev_err(cb_dev, "%s: kmap dma_buf_end_cpu_access fail\n",
  302. __func__);
  303. goto err;
  304. }
  305. err:
  306. return rc;
  307. }
  308. static int msm_audio_ion_map_buf(struct dma_buf *dma_buf, dma_addr_t *paddr,
  309. size_t *plen, void **vaddr)
  310. {
  311. int rc = 0;
  312. bool is_iova = true;
  313. rc = msm_audio_ion_get_phys(dma_buf, paddr, plen, is_iova);
  314. if (rc) {
  315. pr_err("%s: ION Get Physical for AUDIO failed, rc = %d\n",
  316. __func__, rc);
  317. dma_buf_put(dma_buf);
  318. goto err;
  319. }
  320. *vaddr = msm_audio_ion_map_kernel(dma_buf);
  321. if (IS_ERR_OR_NULL(*vaddr)) {
  322. pr_err("%s: ION memory mapping for AUDIO failed\n", __func__);
  323. rc = -ENOMEM;
  324. msm_audio_dma_buf_unmap(dma_buf, false);
  325. goto err;
  326. }
  327. err:
  328. return rc;
  329. }
  330. static u32 msm_audio_ion_get_smmu_sid_mode32(void)
  331. {
  332. if (msm_audio_ion_data.smmu_enabled)
  333. return upper_32_bits(msm_audio_ion_data.smmu_sid_bits);
  334. else
  335. return 0;
  336. }
  337. /**
  338. * msm_audio_ion_alloc -
  339. * Allocs ION memory for given client name
  340. *
  341. * @dma_buf: dma_buf for the ION memory
  342. * @bufsz: buffer size
  343. * @paddr: Physical address to be assigned with allocated region
  344. * @plen: length of allocated region to be assigned
  345. * vaddr: virtual address to be assigned
  346. *
  347. * Returns 0 on success or error on failure
  348. */
  349. int msm_audio_ion_alloc(struct dma_buf **dma_buf, size_t bufsz,
  350. dma_addr_t *paddr, size_t *plen, void **vaddr)
  351. {
  352. int rc = -EINVAL;
  353. unsigned long err_ion_ptr = 0;
  354. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  355. pr_debug("%s:probe is not done, deferred\n", __func__);
  356. return -EPROBE_DEFER;
  357. }
  358. if (!dma_buf || !paddr || !vaddr || !bufsz || !plen) {
  359. pr_err("%s: Invalid params\n", __func__);
  360. return -EINVAL;
  361. }
  362. if (msm_audio_ion_data.smmu_enabled == true) {
  363. pr_debug("%s: system heap is used\n", __func__);
  364. *dma_buf = ion_alloc(bufsz, ION_HEAP(ION_SYSTEM_HEAP_ID), 0);
  365. } else {
  366. pr_debug("%s: audio heap is used\n", __func__);
  367. *dma_buf = ion_alloc(bufsz, ION_HEAP(ION_AUDIO_HEAP_ID), 0);
  368. }
  369. if (IS_ERR_OR_NULL((void *)(*dma_buf))) {
  370. if (IS_ERR((void *)(*dma_buf)))
  371. err_ion_ptr = PTR_ERR((int *)(*dma_buf));
  372. pr_err("%s: ION alloc fail err ptr=%ld, smmu_enabled=%d\n",
  373. __func__, err_ion_ptr, msm_audio_ion_data.smmu_enabled);
  374. rc = -ENOMEM;
  375. goto err;
  376. }
  377. rc = msm_audio_ion_map_buf(*dma_buf, paddr, plen, vaddr);
  378. if (rc) {
  379. pr_err("%s: failed to map ION buf, rc = %d\n", __func__, rc);
  380. goto err;
  381. }
  382. pr_debug("%s: mapped address = %pK, size=%zd\n", __func__,
  383. *vaddr, bufsz);
  384. memset(*vaddr, 0, bufsz);
  385. // Cleanp dmabuf if error?
  386. err:
  387. return rc;
  388. }
  389. EXPORT_SYMBOL(msm_audio_ion_alloc);
  390. int msm_audio_ion_phys_free(void *handle,
  391. dma_addr_t *paddr,
  392. size_t *pa_len,
  393. u8 assign_type,
  394. int id,
  395. int key)
  396. {
  397. handle = NULL;
  398. return 0;
  399. }
  400. EXPORT_SYMBOL(msm_audio_ion_phys_free);
  401. int msm_audio_ion_phys_assign(void **handle, int fd,
  402. dma_addr_t *paddr, size_t *pa_len, u8 assign_type, int id)
  403. {
  404. *handle = NULL;
  405. return 0;
  406. }
  407. EXPORT_SYMBOL(msm_audio_ion_phys_assign);
  408. bool msm_audio_is_hypervisor_supported(void)
  409. {
  410. return !(msm_audio_ion_data.is_non_hypervisor);
  411. }
  412. EXPORT_SYMBOL(msm_audio_is_hypervisor_supported);
  413. /**
  414. * msm_audio_ion_dma_map -
  415. * Memory maps for a given DMA buffer
  416. *
  417. * @phys_addr: Physical address of DMA buffer to be mapped
  418. * @iova_base: IOVA address of memory mapped DMA buffer
  419. * @size: buffer size
  420. * @dir: DMA direction
  421. * Returns 0 on success or error on failure
  422. */
  423. int msm_audio_ion_dma_map(dma_addr_t *phys_addr, dma_addr_t *iova_base,
  424. u32 size, enum dma_data_direction dir)
  425. {
  426. dma_addr_t iova;
  427. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  428. if (!phys_addr || !iova_base || !size)
  429. return -EINVAL;
  430. iova = dma_map_resource(cb_dev, *phys_addr, size,
  431. dir, 0);
  432. if (dma_mapping_error(cb_dev, iova)) {
  433. pr_err("%s: dma_mapping_error\n", __func__);
  434. return -EIO;
  435. }
  436. pr_debug("%s: dma_mapping_success iova:0x%lx\n", __func__,
  437. (unsigned long)iova);
  438. if (msm_audio_ion_data.smmu_enabled)
  439. /* Append the SMMU SID information to the IOVA address */
  440. iova |= msm_audio_ion_data.smmu_sid_bits;
  441. *iova_base = iova;
  442. return 0;
  443. }
  444. EXPORT_SYMBOL(msm_audio_ion_dma_map);
  445. void msm_audio_fd_list_debug(void)
  446. {
  447. struct msm_audio_fd_data *msm_audio_fd_data = NULL;
  448. list_for_each_entry(msm_audio_fd_data,
  449. &msm_audio_ion_data.fd_list, list) {
  450. pr_debug("%s fd %d handle %pK phy. addr %pK\n", __func__,
  451. msm_audio_fd_data->fd, msm_audio_fd_data->handle,
  452. (void *)msm_audio_fd_data->paddr);
  453. }
  454. }
  455. void msm_audio_update_fd_list(struct msm_audio_fd_data *msm_audio_fd_data)
  456. {
  457. struct msm_audio_fd_data *msm_audio_fd_data1 = NULL;
  458. mutex_lock(&(msm_audio_ion_data.list_mutex));
  459. list_for_each_entry(msm_audio_fd_data1,
  460. &msm_audio_ion_data.fd_list, list) {
  461. if (msm_audio_fd_data1->fd == msm_audio_fd_data->fd) {
  462. pr_err("%s fd already present, not updating the list",
  463. __func__);
  464. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  465. return;
  466. }
  467. }
  468. list_add_tail(&msm_audio_fd_data->list, &msm_audio_ion_data.fd_list);
  469. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  470. }
  471. void msm_audio_delete_fd_entry(void *handle)
  472. {
  473. struct msm_audio_fd_data *msm_audio_fd_data = NULL;
  474. struct list_head *ptr, *next;
  475. mutex_lock(&(msm_audio_ion_data.list_mutex));
  476. list_for_each_safe(ptr, next,
  477. &msm_audio_ion_data.fd_list) {
  478. msm_audio_fd_data = list_entry(ptr, struct msm_audio_fd_data,
  479. list);
  480. if (msm_audio_fd_data->handle == handle) {
  481. pr_debug("%s deleting handle %pK entry from list\n",
  482. __func__, handle);
  483. list_del(&(msm_audio_fd_data->list));
  484. kfree(msm_audio_fd_data);
  485. break;
  486. }
  487. }
  488. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  489. }
  490. int msm_audio_get_phy_addr(int fd, dma_addr_t *paddr)
  491. {
  492. struct msm_audio_fd_data *msm_audio_fd_data = NULL;
  493. int status = -EINVAL;
  494. if (!paddr) {
  495. pr_err("%s Invalid paddr param status %d\n", __func__, status);
  496. return status;
  497. }
  498. pr_debug("%s, fd %d\n", __func__, fd);
  499. mutex_lock(&(msm_audio_ion_data.list_mutex));
  500. list_for_each_entry(msm_audio_fd_data,
  501. &msm_audio_ion_data.fd_list, list) {
  502. if (msm_audio_fd_data->fd == fd) {
  503. *paddr = msm_audio_fd_data->paddr;
  504. status = 0;
  505. pr_debug("%s Found fd %d paddr %pK\n",
  506. __func__, fd, paddr);
  507. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  508. return status;
  509. }
  510. }
  511. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  512. return status;
  513. }
  514. EXPORT_SYMBOL(msm_audio_get_phy_addr);
  515. void msm_audio_get_handle(int fd, void **handle)
  516. {
  517. struct msm_audio_fd_data *msm_audio_fd_data = NULL;
  518. pr_debug("%s fd %d\n", __func__, fd);
  519. mutex_lock(&(msm_audio_ion_data.list_mutex));
  520. list_for_each_entry(msm_audio_fd_data,
  521. &msm_audio_ion_data.fd_list, list) {
  522. if (msm_audio_fd_data->fd == fd) {
  523. *handle = (struct dma_buf *)msm_audio_fd_data->handle;
  524. pr_debug("%s handle %pK\n", __func__, *handle);
  525. break;
  526. }
  527. }
  528. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  529. }
  530. /**
  531. * msm_audio_ion_import-
  532. * Import ION buffer with given file descriptor
  533. *
  534. * @dma_buf: dma_buf for the ION memory
  535. * @fd: file descriptor for the ION memory
  536. * @ionflag: flags associated with ION buffer
  537. * @bufsz: buffer size
  538. * @paddr: Physical address to be assigned with allocated region
  539. * @plen: length of allocated region to be assigned
  540. * @vaddr: virtual address to be assigned
  541. *
  542. * Returns 0 on success or error on failure
  543. */
  544. int msm_audio_ion_import(struct dma_buf **dma_buf, int fd,
  545. unsigned long *ionflag, size_t bufsz,
  546. dma_addr_t *paddr, size_t *plen, void **vaddr)
  547. {
  548. int rc = 0;
  549. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  550. pr_debug("%s: probe is not done, deferred\n", __func__);
  551. return -EPROBE_DEFER;
  552. }
  553. if (!dma_buf || !paddr || !vaddr || !plen) {
  554. pr_err("%s: Invalid params\n", __func__);
  555. return -EINVAL;
  556. }
  557. /* bufsz should be 0 and fd shouldn't be 0 as of now */
  558. *dma_buf = dma_buf_get(fd);
  559. pr_debug("%s: dma_buf =%pK, fd=%d\n", __func__, *dma_buf, fd);
  560. if (IS_ERR_OR_NULL((void *)(*dma_buf))) {
  561. pr_err("%s: dma_buf_get failed\n", __func__);
  562. rc = -EINVAL;
  563. goto err;
  564. }
  565. if (ionflag != NULL) {
  566. rc = dma_buf_get_flags(*dma_buf, ionflag);
  567. if (rc) {
  568. pr_err("%s: could not get flags for the dma_buf\n",
  569. __func__);
  570. goto err_ion_flag;
  571. }
  572. }
  573. rc = msm_audio_ion_map_buf(*dma_buf, paddr, plen, vaddr);
  574. if (rc) {
  575. pr_err("%s: failed to map ION buf, rc = %d\n", __func__, rc);
  576. goto err;
  577. }
  578. pr_debug("%s: mapped address = %pK, size=%zd\n", __func__,
  579. *vaddr, bufsz);
  580. return 0;
  581. err_ion_flag:
  582. dma_buf_put(*dma_buf);
  583. err:
  584. *dma_buf = NULL;
  585. return rc;
  586. }
  587. EXPORT_SYMBOL(msm_audio_ion_import);
  588. /**
  589. * msm_audio_ion_import_cma-
  590. * Import ION buffer with given file descriptor
  591. *
  592. * @dma_buf: dma_buf for the ION memory
  593. * @fd: file descriptor for the ION memory
  594. * @ionflag: flags associated with ION buffer
  595. * @bufsz: buffer size
  596. * @paddr: Physical address to be assigned with allocated region
  597. * @plen: length of allocated region to be assigned
  598. * vaddr: virtual address to be assigned
  599. *
  600. * Returns 0 on success or error on failure
  601. */
  602. int msm_audio_ion_import_cma(struct dma_buf **dma_buf, int fd,
  603. unsigned long *ionflag, size_t bufsz,
  604. dma_addr_t *paddr, size_t *plen, void **vaddr)
  605. {
  606. int rc = 0;
  607. if (!(msm_audio_ion_data.device_status & MSM_AUDIO_ION_PROBED)) {
  608. pr_debug("%s: probe is not done, deferred\n", __func__);
  609. return -EPROBE_DEFER;
  610. }
  611. if (!dma_buf || !paddr || !vaddr || !plen ||
  612. !msm_audio_ion_data.cb_cma_dev) {
  613. pr_err("%s: Invalid params\n", __func__);
  614. return -EINVAL;
  615. }
  616. /* bufsz should be 0 and fd shouldn't be 0 as of now */
  617. *dma_buf = dma_buf_get(fd);
  618. pr_debug("%s: dma_buf =%pK, fd=%d\n", __func__, *dma_buf, fd);
  619. if (IS_ERR_OR_NULL((void *)(*dma_buf))) {
  620. pr_err("%s: dma_buf_get failed\n", __func__);
  621. rc = -EINVAL;
  622. goto err;
  623. }
  624. if (ionflag != NULL) {
  625. rc = dma_buf_get_flags(*dma_buf, ionflag);
  626. if (rc) {
  627. pr_err("%s: could not get flags for the dma_buf\n",
  628. __func__);
  629. goto err_ion_flag;
  630. }
  631. }
  632. msm_audio_dma_buf_map(*dma_buf, paddr, plen, true, true);
  633. return 0;
  634. err_ion_flag:
  635. dma_buf_put(*dma_buf);
  636. err:
  637. *dma_buf = NULL;
  638. return rc;
  639. }
  640. EXPORT_SYMBOL(msm_audio_ion_import_cma);
  641. /**
  642. * msm_audio_ion_free -
  643. * fress ION memory for given client and handle
  644. *
  645. * @dma_buf: dma_buf for the ION memory
  646. *
  647. * Returns 0 on success or error on failure
  648. */
  649. int msm_audio_ion_free(struct dma_buf *dma_buf)
  650. {
  651. int ret = 0;
  652. if (!dma_buf) {
  653. pr_err("%s: dma_buf invalid\n", __func__);
  654. return -EINVAL;
  655. }
  656. ret = msm_audio_ion_unmap_kernel(dma_buf);
  657. if (ret)
  658. return ret;
  659. msm_audio_dma_buf_unmap(dma_buf, false);
  660. return 0;
  661. }
  662. EXPORT_SYMBOL(msm_audio_ion_free);
  663. /**
  664. * msm_audio_ion_crash_handler -
  665. * handles cleanup after userspace crashes.
  666. *
  667. * To be called from machine driver.
  668. */
  669. void msm_audio_ion_crash_handler(void)
  670. {
  671. struct msm_audio_fd_data *msm_audio_fd_data = NULL;
  672. struct list_head *ptr, *next;
  673. void *handle = NULL;
  674. pr_debug("Inside %s\n", __func__);
  675. list_for_each_entry(msm_audio_fd_data,
  676. &msm_audio_ion_data.fd_list, list) {
  677. handle = msm_audio_fd_data->handle;
  678. msm_audio_ion_free(handle);
  679. }
  680. list_for_each_safe(ptr, next,
  681. &msm_audio_ion_data.fd_list) {
  682. msm_audio_fd_data = list_entry(ptr, struct msm_audio_fd_data,
  683. list);
  684. list_del(&(msm_audio_fd_data->list));
  685. kfree(msm_audio_fd_data);
  686. }
  687. }
  688. EXPORT_SYMBOL(msm_audio_ion_crash_handler);
  689. /**
  690. * msm_audio_ion_free_cma -
  691. * fress ION memory for given client and handle
  692. *
  693. * @dma_buf: dma_buf for the ION memory
  694. *
  695. * Returns 0 on success or error on failure
  696. */
  697. int msm_audio_ion_free_cma(struct dma_buf *dma_buf)
  698. {
  699. if (!dma_buf) {
  700. pr_err("%s: dma_buf invalid\n", __func__);
  701. return -EINVAL;
  702. }
  703. msm_audio_dma_buf_unmap(dma_buf, true);
  704. return 0;
  705. }
  706. EXPORT_SYMBOL(msm_audio_ion_free_cma);
  707. /**
  708. * msm_audio_ion_mmap -
  709. * Audio ION memory map
  710. *
  711. * @abuff: audio buf pointer
  712. * @vma: virtual mem area
  713. *
  714. * Returns 0 on success or error on failure
  715. */
  716. int msm_audio_ion_mmap(struct audio_buffer *abuff,
  717. struct vm_area_struct *vma)
  718. {
  719. struct msm_audio_alloc_data *alloc_data = NULL;
  720. struct sg_table *table;
  721. unsigned long addr = vma->vm_start;
  722. unsigned long offset = vma->vm_pgoff * PAGE_SIZE;
  723. struct scatterlist *sg;
  724. unsigned int i;
  725. struct page *page;
  726. int ret = 0;
  727. bool found = false;
  728. struct device *cb_dev = msm_audio_ion_data.cb_dev;
  729. mutex_lock(&(msm_audio_ion_data.list_mutex));
  730. list_for_each_entry(alloc_data, &(msm_audio_ion_data.alloc_list),
  731. list) {
  732. if (alloc_data->dma_buf == abuff->dma_buf) {
  733. found = true;
  734. table = alloc_data->table;
  735. break;
  736. }
  737. }
  738. mutex_unlock(&(msm_audio_ion_data.list_mutex));
  739. if (!found) {
  740. dev_err(cb_dev,
  741. "%s: cannot find allocation, dma_buf %pK",
  742. __func__, abuff->dma_buf);
  743. return -EINVAL;
  744. }
  745. /* uncached */
  746. vma->vm_page_prot = pgprot_writecombine(vma->vm_page_prot);
  747. /* We need to check if a page is associated with this sg list because:
  748. * If the allocation came from a carveout we currently don't have
  749. * pages associated with carved out memory. This might change in the
  750. * future and we can remove this check and the else statement.
  751. */
  752. page = sg_page(table->sgl);
  753. if (page) {
  754. pr_debug("%s: page is NOT null\n", __func__);
  755. for_each_sg(table->sgl, sg, table->nents, i) {
  756. unsigned long remainder = vma->vm_end - addr;
  757. unsigned long len = sg->length;
  758. page = sg_page(sg);
  759. if (offset >= len) {
  760. offset -= len;
  761. continue;
  762. } else if (offset) {
  763. page += offset / PAGE_SIZE;
  764. len -= offset;
  765. offset = 0;
  766. }
  767. len = min(len, remainder);
  768. pr_debug("vma=%pK, addr=%x len=%ld vm_start=%x vm_end=%x vm_page_prot=%lu\n",
  769. vma, (unsigned int)addr, len,
  770. (unsigned int)vma->vm_start,
  771. (unsigned int)vma->vm_end,
  772. (unsigned long)pgprot_val(vma->vm_page_prot));
  773. remap_pfn_range(vma, addr, page_to_pfn(page), len,
  774. vma->vm_page_prot);
  775. addr += len;
  776. if (addr >= vma->vm_end)
  777. return 0;
  778. }
  779. } else {
  780. pr_debug("%s: page is NULL\n", __func__);
  781. ret = -EINVAL;
  782. }
  783. return ret;
  784. }
  785. EXPORT_SYMBOL(msm_audio_ion_mmap);
  786. /**
  787. * msm_audio_ion_cache_operations-
  788. * Cache operations on cached Audio ION buffers
  789. *
  790. * @abuff: audio buf pointer
  791. * @cache_op: cache operation to be performed
  792. *
  793. * Returns 0 on success or error on failure
  794. */
  795. int msm_audio_ion_cache_operations(struct audio_buffer *abuff, int cache_op)
  796. {
  797. unsigned long ionflag = 0;
  798. int rc = 0;
  799. if (!abuff) {
  800. pr_err("%s: Invalid params: %pK\n", __func__, abuff);
  801. return -EINVAL;
  802. }
  803. rc = dma_buf_get_flags(abuff->dma_buf, &ionflag);
  804. if (rc) {
  805. pr_err("%s: dma_buf_get_flags failed: %d\n", __func__, rc);
  806. goto cache_op_failed;
  807. }
  808. /* Has to be CACHED */
  809. if (ionflag & ION_FLAG_CACHED) {
  810. /* MSM_AUDIO_ION_INV_CACHES or MSM_AUDIO_ION_CLEAN_CACHES */
  811. switch (cache_op) {
  812. case MSM_AUDIO_ION_INV_CACHES:
  813. case MSM_AUDIO_ION_CLEAN_CACHES:
  814. dma_buf_begin_cpu_access(abuff->dma_buf,
  815. DMA_BIDIRECTIONAL);
  816. dma_buf_end_cpu_access(abuff->dma_buf,
  817. DMA_BIDIRECTIONAL);
  818. break;
  819. default:
  820. pr_err("%s: Invalid cache operation %d\n",
  821. __func__, cache_op);
  822. }
  823. } else {
  824. pr_err("%s: Cache ops called on uncached buffer: %pK\n",
  825. __func__, abuff->dma_buf);
  826. rc = -EINVAL;
  827. }
  828. cache_op_failed:
  829. return rc;
  830. }
  831. EXPORT_SYMBOL(msm_audio_ion_cache_operations);
  832. /**
  833. * msm_audio_populate_upper_32_bits -
  834. * retrieve upper 32bits of 64bit address
  835. *
  836. * @pa: 64bit physical address
  837. *
  838. */
  839. u32 msm_audio_populate_upper_32_bits(dma_addr_t pa)
  840. {
  841. if (sizeof(dma_addr_t) == sizeof(u32))
  842. return msm_audio_ion_get_smmu_sid_mode32();
  843. else
  844. return upper_32_bits(pa);
  845. }
  846. EXPORT_SYMBOL(msm_audio_populate_upper_32_bits);
  847. static int msm_audio_ion_open(struct inode *inode, struct file *file)
  848. {
  849. int ret = 0;
  850. struct msm_audio_ion_private *ion_data = container_of(inode->i_cdev,
  851. struct msm_audio_ion_private,
  852. cdev);
  853. struct device *dev = ion_data->chardev;
  854. pr_debug("Inside %s\n", __func__);
  855. get_device(dev);
  856. return ret;
  857. }
  858. static int msm_audio_ion_release(struct inode *inode, struct file *file)
  859. {
  860. struct msm_audio_ion_private *ion_data = container_of(inode->i_cdev,
  861. struct msm_audio_ion_private,
  862. cdev);
  863. struct device *dev = ion_data->chardev;
  864. pr_debug("Inside %s\n", __func__);
  865. put_device(dev);
  866. return 0;
  867. }
  868. static long msm_audio_ion_ioctl(struct file *file, unsigned int ioctl_num,
  869. unsigned long __user ioctl_param)
  870. {
  871. void *mem_handle;
  872. dma_addr_t paddr;
  873. size_t pa_len = 0;
  874. void *vaddr;
  875. int ret = 0;
  876. struct msm_audio_fd_data *msm_audio_fd_data = NULL;
  877. pr_debug("%s ioctl num %u\n", __func__, ioctl_num);
  878. switch (ioctl_num) {
  879. case IOCTL_MAP_PHYS_ADDR:
  880. msm_audio_fd_data = kzalloc((sizeof(struct msm_audio_fd_data)),
  881. GFP_KERNEL);
  882. if (!msm_audio_fd_data) {
  883. ret = -ENOMEM;
  884. pr_err("%s : Out of memory ret %d\n", __func__, ret);
  885. return ret;
  886. }
  887. ret = msm_audio_ion_import((struct dma_buf **)&mem_handle, (int)ioctl_param,
  888. NULL, 0, &paddr, &pa_len, &vaddr);
  889. if (ret < 0) {
  890. pr_err("%s Memory map Failed %d\n", __func__, ret);
  891. kfree(msm_audio_fd_data);
  892. return ret;
  893. }
  894. msm_audio_fd_data->fd = (int)ioctl_param;
  895. msm_audio_fd_data->handle = mem_handle;
  896. msm_audio_fd_data->paddr = paddr;
  897. msm_audio_update_fd_list(msm_audio_fd_data);
  898. break;
  899. case IOCTL_UNMAP_PHYS_ADDR:
  900. msm_audio_get_handle((int)ioctl_param, &mem_handle);
  901. ret = msm_audio_ion_free(mem_handle);
  902. if (ret < 0) {
  903. pr_err("%s Ion free failed %d\n", __func__, ret);
  904. return ret;
  905. }
  906. msm_audio_delete_fd_entry(mem_handle);
  907. break;
  908. default:
  909. pr_err("%s Entered default. Invalid ioctl num %u",
  910. __func__, ioctl_num);
  911. ret = -EINVAL;
  912. break;
  913. }
  914. return ret;
  915. }
  916. static int msm_audio_smmu_init(struct device *dev)
  917. {
  918. INIT_LIST_HEAD(&msm_audio_ion_data.alloc_list);
  919. mutex_init(&(msm_audio_ion_data.list_mutex));
  920. return 0;
  921. }
  922. static const struct of_device_id msm_audio_ion_dt_match[] = {
  923. { .compatible = "qcom,msm-audio-ion" },
  924. { .compatible = "qcom,msm-audio-ion-cma"},
  925. { }
  926. };
  927. MODULE_DEVICE_TABLE(of, msm_audio_ion_dt_match);
  928. static const struct file_operations msm_audio_ion_fops = {
  929. .owner = THIS_MODULE,
  930. .open = msm_audio_ion_open,
  931. .release = msm_audio_ion_release,
  932. .unlocked_ioctl = msm_audio_ion_ioctl,
  933. };
  934. static int msm_audio_ion_reg_chrdev(struct msm_audio_ion_private *ion_data)
  935. {
  936. int ret = 0;
  937. ret = alloc_chrdev_region(&ion_data->ion_major, 0,
  938. MINOR_NUMBER_COUNT, MSM_AUDIO_ION_DRIVER_NAME);
  939. if (ret < 0) {
  940. pr_err("%s alloc_chr_dev_region failed ret : %d\n",
  941. __func__, ret);
  942. return ret;
  943. }
  944. pr_debug("%s major number %d", __func__, MAJOR(ion_data->ion_major));
  945. ion_data->ion_class = class_create(THIS_MODULE,
  946. MSM_AUDIO_ION_DRIVER_NAME);
  947. if (IS_ERR(ion_data->ion_class)) {
  948. ret = PTR_ERR(ion_data->ion_class);
  949. pr_err("%s class create failed. ret : %d", __func__, ret);
  950. goto err_class;
  951. }
  952. ion_data->chardev = device_create(ion_data->ion_class, NULL,
  953. ion_data->ion_major, NULL,
  954. MSM_AUDIO_ION_DRIVER_NAME);
  955. if (IS_ERR(ion_data->chardev)) {
  956. ret = PTR_ERR(ion_data->chardev);
  957. pr_err("%s device create failed ret : %d\n", __func__, ret);
  958. goto err_device;
  959. }
  960. cdev_init(&ion_data->cdev, &msm_audio_ion_fops);
  961. ret = cdev_add(&ion_data->cdev, ion_data->ion_major, 1);
  962. if (ret) {
  963. pr_err("%s cdev add failed, ret : %d\n", __func__, ret);
  964. goto err_cdev;
  965. }
  966. return ret;
  967. err_cdev:
  968. device_destroy(ion_data->ion_class, ion_data->ion_major);
  969. err_device:
  970. class_destroy(ion_data->ion_class);
  971. err_class:
  972. unregister_chrdev_region(0, MINOR_NUMBER_COUNT);
  973. return ret;
  974. }
  975. static int msm_audio_ion_unreg_chrdev(struct msm_audio_ion_private *ion_data)
  976. {
  977. cdev_del(&ion_data->cdev);
  978. device_destroy(ion_data->ion_class, ion_data->ion_major);
  979. class_destroy(ion_data->ion_class);
  980. unregister_chrdev_region(0, MINOR_NUMBER_COUNT);
  981. return 0;
  982. }
  983. static int msm_audio_ion_probe(struct platform_device *pdev)
  984. {
  985. int rc = 0;
  986. u64 smmu_sid = 0;
  987. u64 smmu_sid_mask = 0;
  988. const char *msm_audio_ion_dt = "qcom,smmu-enabled";
  989. const char *msm_audio_ion_non_hyp = "qcom,non-hyp-assign";
  990. const char *msm_audio_ion_smmu = "qcom,smmu-version";
  991. const char *msm_audio_ion_smmu_sid_mask = "qcom,smmu-sid-mask";
  992. bool smmu_enabled;
  993. bool is_non_hypervisor_en;
  994. struct device *dev = &pdev->dev;
  995. struct of_phandle_args iommuspec;
  996. #ifndef CONFIG_SPF_CORE
  997. enum apr_subsys_state q6_state;
  998. #endif
  999. dev_err(dev, "%s: msm_audio_ion_probe\n", __func__);
  1000. if (dev->of_node == NULL) {
  1001. dev_err(dev,
  1002. "%s: device tree is not found\n",
  1003. __func__);
  1004. msm_audio_ion_data.smmu_enabled = 0;
  1005. return 0;
  1006. }
  1007. is_non_hypervisor_en = of_property_read_bool(dev->of_node,
  1008. msm_audio_ion_non_hyp);
  1009. msm_audio_ion_data.is_non_hypervisor = is_non_hypervisor_en;
  1010. if (of_device_is_compatible(dev->of_node, "qcom,msm-audio-ion-cma")) {
  1011. msm_audio_ion_data.cb_cma_dev = dev;
  1012. return 0;
  1013. }
  1014. smmu_enabled = of_property_read_bool(dev->of_node,
  1015. msm_audio_ion_dt);
  1016. msm_audio_ion_data.smmu_enabled = smmu_enabled;
  1017. if (!smmu_enabled) {
  1018. dev_dbg(dev, "%s: SMMU is Disabled\n", __func__);
  1019. goto exit;
  1020. }
  1021. #ifndef CONFIG_SPF_CORE
  1022. q6_state = apr_get_q6_state();
  1023. if (q6_state == APR_SUBSYS_DOWN) {
  1024. dev_info(dev,
  1025. "defering %s, adsp_state %d\n",
  1026. __func__, q6_state);
  1027. return -EPROBE_DEFER;
  1028. }
  1029. #endif
  1030. dev_dbg(dev, "%s: adsp is ready\n", __func__);
  1031. rc = of_property_read_u32(dev->of_node,
  1032. msm_audio_ion_smmu,
  1033. &msm_audio_ion_data.smmu_version);
  1034. if (rc) {
  1035. dev_err(dev,
  1036. "%s: qcom,smmu_version missing in DT node\n",
  1037. __func__);
  1038. return rc;
  1039. }
  1040. dev_dbg(dev, "%s: SMMU is Enabled. SMMU version is (%d)",
  1041. __func__, msm_audio_ion_data.smmu_version);
  1042. /* Get SMMU SID information from Devicetree */
  1043. rc = of_property_read_u64(dev->of_node,
  1044. msm_audio_ion_smmu_sid_mask,
  1045. &smmu_sid_mask);
  1046. if (rc) {
  1047. dev_err(dev,
  1048. "%s: qcom,smmu-sid-mask missing in DT node, using default\n",
  1049. __func__);
  1050. smmu_sid_mask = 0xFFFFFFFFFFFFFFFF;
  1051. }
  1052. rc = of_parse_phandle_with_args(dev->of_node, "iommus",
  1053. "#iommu-cells", 0, &iommuspec);
  1054. if (rc)
  1055. dev_err(dev, "%s: could not get smmu SID, ret = %d\n",
  1056. __func__, rc);
  1057. else
  1058. smmu_sid = (iommuspec.args[0] & smmu_sid_mask);
  1059. msm_audio_ion_data.smmu_sid_bits =
  1060. smmu_sid << MSM_AUDIO_SMMU_SID_OFFSET;
  1061. if (msm_audio_ion_data.smmu_version == 0x2) {
  1062. rc = msm_audio_smmu_init(dev);
  1063. } else {
  1064. dev_err(dev, "%s: smmu version invalid %d\n",
  1065. __func__, msm_audio_ion_data.smmu_version);
  1066. rc = -EINVAL;
  1067. }
  1068. if (rc)
  1069. dev_err(dev, "%s: smmu init failed, err = %d\n",
  1070. __func__, rc);
  1071. exit:
  1072. if (!rc)
  1073. msm_audio_ion_data.device_status |= MSM_AUDIO_ION_PROBED;
  1074. msm_audio_ion_data.cb_dev = dev;
  1075. INIT_LIST_HEAD(&msm_audio_ion_data.fd_list);
  1076. rc = msm_audio_ion_reg_chrdev(&msm_audio_ion_data);
  1077. if (rc) {
  1078. pr_err("%s register char dev failed, rc : %d", __func__, rc);
  1079. return rc;
  1080. }
  1081. return rc;
  1082. }
  1083. static int msm_audio_ion_remove(struct platform_device *pdev)
  1084. {
  1085. struct device *audio_cb_dev;
  1086. audio_cb_dev = msm_audio_ion_data.cb_dev;
  1087. msm_audio_ion_data.smmu_enabled = 0;
  1088. msm_audio_ion_data.device_status = 0;
  1089. msm_audio_ion_unreg_chrdev(&msm_audio_ion_data);
  1090. return 0;
  1091. }
  1092. static struct platform_driver msm_audio_ion_driver = {
  1093. .driver = {
  1094. .name = "msm-audio-ion",
  1095. .owner = THIS_MODULE,
  1096. .of_match_table = msm_audio_ion_dt_match,
  1097. .suppress_bind_attrs = true,
  1098. },
  1099. .probe = msm_audio_ion_probe,
  1100. .remove = msm_audio_ion_remove,
  1101. };
  1102. int __init msm_audio_ion_init(void)
  1103. {
  1104. pr_debug("%s: msm_audio_ion_init called \n",__func__);
  1105. return platform_driver_register(&msm_audio_ion_driver);
  1106. }
  1107. void msm_audio_ion_exit(void)
  1108. {
  1109. platform_driver_unregister(&msm_audio_ion_driver);
  1110. }
  1111. module_init(msm_audio_ion_init);
  1112. module_exit(msm_audio_ion_exit);
  1113. MODULE_DESCRIPTION("MSM Audio ION module");
  1114. MODULE_LICENSE("GPL v2");