msm_mdf.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732
  1. /*
  2. * Copyright (c) 2018, The Linux Foundation. All rights reserved.
  3. *
  4. * This program is free software; you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License version 2 and
  6. * only version 2 as published by the Free Software Foundation.
  7. *
  8. * This program is distributed in the hope that it will be useful,
  9. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. * GNU General Public License for more details.
  12. *
  13. */
  14. #include <linux/init.h>
  15. #include <linux/kernel.h>
  16. #include <linux/module.h>
  17. #include <linux/err.h>
  18. #include <linux/clk.h>
  19. #include <linux/delay.h>
  20. #include <linux/slab.h>
  21. #include <linux/list.h>
  22. #include <linux/scatterlist.h>
  23. #include <linux/dma-mapping.h>
  24. #include <linux/dma-buf.h>
  25. #include <linux/iommu.h>
  26. #include <linux/platform_device.h>
  27. #include <linux/of_device.h>
  28. #include <linux/ion_kernel.h>
  29. #include <linux/msm_ion.h>
  30. #include <dsp/msm_audio_ion.h>
  31. #include <ipc/apr.h>
  32. #include <dsp/msm_mdf.h>
  33. #include <asm/dma-iommu.h>
  34. #include <soc/qcom/secure_buffer.h>
  35. #include <soc/qcom/subsystem_notif.h>
  36. #include <soc/qcom/subsystem_restart.h>
  37. #include <soc/qcom/scm.h>
  38. #include <dsp/q6audio-v2.h>
  39. #include <dsp/q6core.h>
  40. #include <asm/cacheflush.h>
  41. #define VMID_SSC_Q6 5
  42. #define VMID_LPASS 6
  43. #define VMID_MSS_MSA 15
  44. #define VMID_CDSP 30
  45. #define MSM_MDF_PROBED (1 << 0)
  46. #define MSM_MDF_INITIALIZED (1 << 1)
  47. #define MSM_MDF_MEM_ALLOCATED (1 << 2)
  48. #define MSM_MDF_MEM_MAPPED (1 << 3)
  49. #define MSM_MDF_MEM_PERMISSION (1 << 4) /* 0 - HLOS, 1 - Subsys */
  50. /* TODO: Update IOVA range for subsys SMMUs */
  51. #define MSM_MDF_IOVA_START 0x80000000
  52. #define MSM_MDF_IOVA_LEN 0x800000
  53. #define MSM_MDF_SMMU_SID_OFFSET 32
  54. #define ADSP_STATE_READY_TIMEOUT_MS 3000
  55. /* mem protection defines */
  56. #define TZ_MPU_LOCK_NS_REGION 0x00000025
  57. #define MEM_PROTECT_AC_PERM_READ 0x4
  58. #define MEM_PROTECT_AC_PERM_WRITE 0x2
  59. #define MSM_AUDIO_SMMU_SID_OFFSET 32
  60. enum {
  61. SUBSYS_ADSP, /* Audio DSP must have index 0 */
  62. SUBSYS_SCC, /* Sensor DSP */
  63. SUBSYS_MSS, /* Modem DSP */
  64. SUBSYS_CDSP, /* Compute DSP */
  65. SUBSYS_MAX,
  66. };
  67. struct msm_mdf_dest_vm_and_perm_info {
  68. uint32_t dst_vm;
  69. /* Destination VM defined by ACVirtualMachineId. */
  70. uint32_t dst_vm_perm;
  71. /* Permissions of the IPA to be mapped to VM, bitwise OR of AC_PERM. */
  72. uint64_t ctx;
  73. /* Destination of the VM-specific context information. */
  74. uint32_t ctx_size;
  75. /* Size of context buffer in bytes. */
  76. };
  77. struct msm_mdf_protect_mem {
  78. uint64_t dma_start_address;
  79. uint64_t dma_end_address;
  80. struct msm_mdf_dest_vm_and_perm_info dest_info[SUBSYS_MAX];
  81. uint32_t dest_info_size;
  82. };
  83. struct msm_mdf_mem {
  84. struct device *dev;
  85. uint8_t device_status;
  86. uint32_t map_handle;
  87. struct dma_buf *dma_buf;
  88. dma_addr_t dma_addr;
  89. size_t size;
  90. void *va;
  91. };
  92. static struct msm_mdf_mem mdf_mem_data = {NULL,};
  93. struct msm_mdf_smmu {
  94. bool enabled;
  95. char *subsys;
  96. int vmid;
  97. uint32_t proc_id;
  98. struct device *cb_dev;
  99. uint8_t device_status;
  100. uint64_t sid;
  101. struct dma_iommu_mapping *mapping;
  102. u64 pa;
  103. size_t pa_len;
  104. };
  105. static struct msm_mdf_smmu mdf_smmu_data[SUBSYS_MAX] = {
  106. {
  107. .subsys = "adsp",
  108. .vmid = VMID_LPASS,
  109. },
  110. {
  111. .subsys = "dsps",
  112. .vmid = VMID_SSC_Q6,
  113. .proc_id = AVS_MDF_SSC_PROC_ID,
  114. },
  115. {
  116. .subsys = "modem",
  117. .vmid = VMID_MSS_MSA,
  118. .proc_id = AVS_MDF_MDSP_PROC_ID,
  119. },
  120. {
  121. .subsys = "cdsp",
  122. .vmid = VMID_CDSP,
  123. .proc_id = AVS_MDF_CDSP_PROC_ID,
  124. },
  125. };
  126. static void *ssr_handle;
  127. static inline uint64_t buf_page_start(uint64_t buf)
  128. {
  129. uint64_t start = (uint64_t) buf & PAGE_MASK;
  130. return start;
  131. }
  132. static inline uint64_t buf_page_offset(uint64_t buf)
  133. {
  134. uint64_t offset = (uint64_t) buf & (PAGE_SIZE - 1);
  135. return offset;
  136. }
  137. static inline int buf_num_pages(uint64_t buf, ssize_t len)
  138. {
  139. uint64_t start = buf_page_start(buf) >> PAGE_SHIFT;
  140. uint64_t end = (((uint64_t) buf + len - 1) & PAGE_MASK) >> PAGE_SHIFT;
  141. int nPages = end - start + 1;
  142. return nPages;
  143. }
  144. static inline uint64_t buf_page_size(uint32_t size)
  145. {
  146. uint64_t sz = (size + (PAGE_SIZE - 1)) & PAGE_MASK;
  147. return sz > PAGE_SIZE ? sz : PAGE_SIZE;
  148. }
  149. static inline void *uint64_to_ptr(uint64_t addr)
  150. {
  151. void *ptr = (void *)((uintptr_t)addr);
  152. return ptr;
  153. }
  154. static inline uint64_t ptr_to_uint64(void *ptr)
  155. {
  156. uint64_t addr = (uint64_t)((uintptr_t)ptr);
  157. return addr;
  158. }
  159. static int msm_mdf_dma_buf_map(struct msm_mdf_mem *mem,
  160. struct msm_mdf_smmu *smmu)
  161. {
  162. int rc = 0;
  163. dma_addr_t pa = 0;
  164. smmu->pa = 0;
  165. if (!smmu)
  166. return -EINVAL;
  167. if (smmu->device_status & MSM_MDF_MEM_MAPPED)
  168. return 0;
  169. if (smmu->enabled) {
  170. if (smmu->cb_dev == NULL) {
  171. pr_err("%s: cb device is not initialized\n",
  172. __func__);
  173. /* Retry if LPASS cb device is not ready
  174. * from audio ION during probing.
  175. */
  176. if (!strcmp("adsp", smmu->subsys)) {
  177. rc = msm_audio_ion_get_smmu_info(&smmu->cb_dev,
  178. &smmu->sid);
  179. if (rc) {
  180. pr_err("%s: msm_audio_ion_get_smmu_info failed, rc = %d\n",
  181. __func__, rc);
  182. goto err;
  183. }
  184. } else
  185. return -ENODEV;
  186. }
  187. pa = dma_map_single_attrs(smmu->cb_dev, mem->va,
  188. mem->size, DMA_BIDIRECTIONAL, DMA_ATTR_SKIP_CPU_SYNC);
  189. if (dma_mapping_error(smmu->cb_dev, smmu->pa)) {
  190. rc = -ENOMEM;
  191. pr_err("%s: failed to map single, rc = %d\n",
  192. __func__, rc);
  193. goto err;
  194. }
  195. smmu->pa |= pa;
  196. smmu->pa_len = mem->size;
  197. /* Append the SMMU SID information to the IOVA address */
  198. if (smmu->sid)
  199. smmu->pa |= smmu->sid;
  200. } else {
  201. smmu->pa |= mem->dma_addr;
  202. smmu->pa_len = mem->size;
  203. }
  204. pr_err("%s: pa=%pa, pa_len=%zd\n", __func__,
  205. &smmu->pa, smmu->pa_len);
  206. smmu->device_status |= MSM_MDF_MEM_MAPPED;
  207. return 0;
  208. err:
  209. return rc;
  210. }
  211. static int msm_mdf_alloc_dma_buf(struct msm_mdf_mem *mem)
  212. {
  213. int rc = 0;
  214. if (!mem)
  215. return -EINVAL;
  216. if (mem->device_status & MSM_MDF_MEM_ALLOCATED)
  217. return 0;
  218. if (mem->dev == NULL) {
  219. pr_err("%s: device is not initialized\n",
  220. __func__);
  221. return -ENODEV;
  222. }
  223. mem->va = dma_alloc_attrs(mem->dev, mem->size,
  224. &mem->dma_addr, GFP_KERNEL, DMA_ATTR_NO_KERNEL_MAPPING);
  225. if (IS_ERR_OR_NULL(mem->va)) {
  226. pr_err("%s: failed to allocate dma memory, rc = %d\n",
  227. __func__, rc);
  228. return -ENOMEM;
  229. }
  230. mem->va = phys_to_virt(mem->dma_addr);
  231. mem->device_status |= MSM_MDF_MEM_ALLOCATED;
  232. return rc;
  233. }
  234. static int msm_mdf_free_dma_buf(struct msm_mdf_mem *mem)
  235. {
  236. if (!mem)
  237. return -EINVAL;
  238. if (mem->dev == NULL) {
  239. pr_err("%s: device is not initialized\n",
  240. __func__);
  241. return -ENODEV;
  242. }
  243. //dma_free_coherent(mem->dev, mem->size, mem->va,
  244. // mem->dma_addr);
  245. mem->device_status &= ~MSM_MDF_MEM_ALLOCATED;
  246. return 0;
  247. }
  248. static int msm_mdf_dma_buf_unmap(struct msm_mdf_mem *mem,
  249. struct msm_mdf_smmu *smmu)
  250. {
  251. if (!smmu)
  252. return -EINVAL;
  253. if (smmu->enabled) {
  254. if (smmu->cb_dev == NULL) {
  255. pr_err("%s: cb device is not initialized\n",
  256. __func__);
  257. return -ENODEV;
  258. }
  259. //if (smmu->pa && mem->size)
  260. //dma_unmap_single(smmu->cb_dev, smmu->pa,
  261. // mem->size, DMA_BIDIRECTIONAL);
  262. }
  263. smmu->device_status &= ~MSM_MDF_MEM_MAPPED;
  264. return 0;
  265. }
  266. static int msm_mdf_map_memory_to_subsys(struct msm_mdf_mem *mem,
  267. struct msm_mdf_smmu *smmu)
  268. {
  269. int rc = 0;
  270. if (!mem || !smmu)
  271. return -EINVAL;
  272. /* Map mdf shared memory to ADSP */
  273. if (!strcmp("adsp", smmu->subsys)) {
  274. rc = q6core_map_mdf_memory_regions((uint64_t *)&smmu->pa,
  275. ADSP_MEMORY_MAP_MDF_SHMEM_4K_POOL,
  276. (uint32_t *)&smmu->pa_len, 1, &mem->map_handle);
  277. if (rc) {
  278. pr_err("%s: q6core_map_memory_regions failed, rc = %d\n",
  279. __func__, rc);
  280. }
  281. } else {
  282. if (mem->map_handle) {
  283. /* Map mdf shared memory to remote DSPs */
  284. rc = q6core_map_mdf_shared_memory(mem->map_handle,
  285. (uint64_t *)&smmu->pa, smmu->proc_id,
  286. (uint32_t *)&smmu->pa_len, 1);
  287. if (rc) {
  288. pr_err("%s: q6core_map_mdf_shared_memory failed, rc = %d\n",
  289. __func__, rc);
  290. }
  291. }
  292. }
  293. return rc;
  294. }
  295. static void msm_mdf_unmap_memory_to_subsys(struct msm_mdf_mem *mem,
  296. struct msm_mdf_smmu *smmu)
  297. {
  298. if (!mem || !smmu)
  299. return;
  300. if (!strcmp("adsp", smmu->subsys)) {
  301. if (mem->map_handle)
  302. q6core_memory_unmap_regions(mem->map_handle);
  303. }
  304. }
  305. static int msm_mdf_assign_memory_to_subsys(struct msm_mdf_mem *mem)
  306. {
  307. int ret = 0, i;
  308. struct scm_desc desc = {0};
  309. struct msm_mdf_protect_mem *scm_buffer;
  310. uint32_t fnid;
  311. scm_buffer = kzalloc(sizeof(struct msm_mdf_protect_mem), GFP_KERNEL);
  312. if (!scm_buffer)
  313. return -ENOMEM;
  314. scm_buffer->dma_start_address = mem->dma_addr;
  315. scm_buffer->dma_end_address = mem->dma_addr + buf_page_size(mem->size);
  316. for (i = 0; i < SUBSYS_MAX; i++) {
  317. scm_buffer->dest_info[i].dst_vm = mdf_smmu_data[i].vmid;
  318. scm_buffer->dest_info[i].dst_vm_perm =
  319. MEM_PROTECT_AC_PERM_READ | MEM_PROTECT_AC_PERM_WRITE;
  320. scm_buffer->dest_info[i].ctx = 0;
  321. scm_buffer->dest_info[i].ctx_size = 0;
  322. }
  323. scm_buffer->dest_info_size =
  324. sizeof(struct msm_mdf_dest_vm_and_perm_info) * SUBSYS_MAX;
  325. /* flush cache required by scm_call2 */
  326. dmac_flush_range(scm_buffer, ((void *)scm_buffer) +
  327. sizeof(struct msm_mdf_protect_mem));
  328. desc.args[0] = scm_buffer->dma_start_address;
  329. desc.args[1] = scm_buffer->dma_end_address;
  330. desc.args[2] = virt_to_phys(&(scm_buffer->dest_info[0]));
  331. desc.args[3] = scm_buffer->dest_info_size;
  332. desc.arginfo = SCM_ARGS(4, SCM_VAL, SCM_VAL, SCM_RO, SCM_VAL);
  333. fnid = SCM_SIP_FNID(SCM_SVC_MP, TZ_MPU_LOCK_NS_REGION);
  334. ret = scm_call2(fnid, &desc);
  335. if (ret < 0) {
  336. pr_err("%s: SCM call2 failed, ret %d scm_resp %llu\n",
  337. __func__, ret, desc.ret[0]);
  338. }
  339. /* No More need for scm_buffer, freeing the same */
  340. kfree(scm_buffer);
  341. return ret;
  342. }
  343. /**
  344. * msm_mdf_mem_init - Initializes MDF memory pool and
  345. * map memory to subsystem
  346. *
  347. * Returns 0 on success or ret on failure.
  348. */
  349. int msm_mdf_mem_init(void)
  350. {
  351. int rc = 0, i, j;
  352. struct msm_mdf_mem *mem = &mdf_mem_data;
  353. struct msm_mdf_smmu *smmu;
  354. unsigned long timeout = jiffies +
  355. msecs_to_jiffies(ADSP_STATE_READY_TIMEOUT_MS);
  356. int adsp_ready = 0;
  357. if (!(mdf_mem_data.device_status & MSM_MDF_PROBED))
  358. return -ENODEV;
  359. if (mdf_mem_data.device_status & MSM_MDF_INITIALIZED)
  360. return 0;
  361. /* TODO: pulling may not be needed as Q6 Core state should be
  362. * checked during machine driver probing.
  363. */
  364. do {
  365. if (!q6core_is_adsp_ready()) {
  366. pr_err("%s: ADSP Audio NOT Ready\n",
  367. __func__);
  368. /* ADSP will be coming up after subsystem restart and
  369. * it might not be fully up when the control reaches
  370. * here. So, wait for 50msec before checking ADSP state
  371. */
  372. msleep(50);
  373. } else {
  374. pr_debug("%s: ADSP Audio Ready\n",
  375. __func__);
  376. adsp_ready = 1;
  377. break;
  378. }
  379. } while (time_after(timeout, jiffies));
  380. if (!adsp_ready) {
  381. pr_err("%s: timed out waiting for ADSP Audio\n",
  382. __func__);
  383. return -ETIMEDOUT;
  384. }
  385. if (mem->device_status & MSM_MDF_MEM_ALLOCATED) {
  386. for (i = 0; i < SUBSYS_MAX; i++) {
  387. smmu = &mdf_smmu_data[i];
  388. rc = msm_mdf_dma_buf_map(mem, smmu);
  389. if (rc) {
  390. pr_err("%s: msm_mdf_dma_buf_map failed, rc = %d\n",
  391. __func__, rc);
  392. goto err;
  393. }
  394. }
  395. rc = msm_mdf_assign_memory_to_subsys(mem);
  396. if (rc) {
  397. pr_err("%s: msm_mdf_assign_memory_to_subsys failed\n",
  398. __func__);
  399. goto err;
  400. }
  401. for (j = 0; j < SUBSYS_MAX; j++) {
  402. smmu = &mdf_smmu_data[j];
  403. rc = msm_mdf_map_memory_to_subsys(mem, smmu);
  404. if (rc) {
  405. pr_err("%s: msm_mdf_map_memory_to_subsys failed\n",
  406. __func__);
  407. goto err;
  408. }
  409. }
  410. mdf_mem_data.device_status |= MSM_MDF_INITIALIZED;
  411. }
  412. return 0;
  413. err:
  414. return rc;
  415. }
  416. EXPORT_SYMBOL(msm_mdf_mem_init);
  417. int msm_mdf_mem_deinit(void)
  418. {
  419. int rc = 0, i;
  420. struct msm_mdf_mem *mem = &mdf_mem_data;
  421. struct msm_mdf_smmu *smmu;
  422. if (!(mdf_mem_data.device_status & MSM_MDF_INITIALIZED))
  423. return -ENODEV;
  424. for (i = SUBSYS_MAX - 1; i >= 0; i--) {
  425. smmu = &mdf_smmu_data[i];
  426. msm_mdf_unmap_memory_to_subsys(mem, smmu);
  427. }
  428. if (!rc) {
  429. for (i = SUBSYS_MAX - 1; i >= 0; i--) {
  430. smmu = &mdf_smmu_data[i];
  431. msm_mdf_dma_buf_unmap(mem, smmu);
  432. }
  433. msm_mdf_free_dma_buf(mem);
  434. mem->device_status &= ~MSM_MDF_MEM_ALLOCATED;
  435. }
  436. mdf_mem_data.device_status &= ~MSM_MDF_INITIALIZED;
  437. return 0;
  438. }
  439. EXPORT_SYMBOL(msm_mdf_mem_deinit);
  440. static int msm_mdf_restart_notifier_cb(struct notifier_block *this,
  441. unsigned long code,
  442. void *_cmd)
  443. {
  444. static int boot_count = 3;
  445. /* During LPASS boot, HLOS receives events:
  446. * SUBSYS_BEFORE_POWERUP
  447. * SUBSYS_PROXY_VOTE
  448. * SUBSYS_AFTER_POWERUP - need skip
  449. * SUBSYS_PROXY_UNVOTE
  450. */
  451. if (boot_count) {
  452. boot_count--;
  453. return NOTIFY_OK;
  454. }
  455. switch (code) {
  456. case SUBSYS_BEFORE_SHUTDOWN:
  457. pr_debug("Subsys Notify: Shutdown Started\n");
  458. /* Unmap and free memory upon restart event. */
  459. msm_mdf_mem_deinit();
  460. break;
  461. case SUBSYS_AFTER_SHUTDOWN:
  462. pr_debug("Subsys Notify: Shutdown Completed\n");
  463. break;
  464. case SUBSYS_BEFORE_POWERUP:
  465. pr_debug("Subsys Notify: Bootup Started\n");
  466. break;
  467. case SUBSYS_AFTER_POWERUP:
  468. pr_debug("Subsys Notify: Bootup Completed\n");
  469. /* Allocate and map memory after restart complete. */
  470. if (msm_mdf_mem_init())
  471. pr_err("msm_mdf_mem_init failed\n");
  472. break;
  473. default:
  474. pr_err("Subsys Notify: Generel: %lu\n", code);
  475. break;
  476. }
  477. return NOTIFY_DONE;
  478. }
  479. static const struct of_device_id msm_mdf_match_table[] = {
  480. { .compatible = "qcom,msm-mdf", },
  481. { .compatible = "qcom,msm-mdf-mem-region", },
  482. { .compatible = "qcom,msm-mdf-cb", },
  483. {}
  484. };
  485. MODULE_DEVICE_TABLE(of, msm_mdf_match_table);
  486. static int msm_mdf_cb_probe(struct device *dev)
  487. {
  488. struct msm_mdf_smmu *smmu;
  489. u64 smmu_sid = 0;
  490. u64 smmu_sid_mask = 0;
  491. struct of_phandle_args iommuspec;
  492. const char *subsys;
  493. int rc = 0, i;
  494. subsys = of_get_property(dev->of_node, "label", NULL);
  495. if (!subsys) {
  496. dev_err(dev, "%s: could not get label\n",
  497. __func__);
  498. return -EINVAL;
  499. }
  500. for (i = 0; i < SUBSYS_MAX; i++) {
  501. if (!mdf_smmu_data[i].subsys)
  502. continue;
  503. if (!strcmp(subsys, mdf_smmu_data[i].subsys))
  504. break;
  505. }
  506. if (i >= SUBSYS_MAX) {
  507. dev_err(dev, "%s: subsys %s not supported\n",
  508. __func__, subsys);
  509. return -EINVAL;
  510. }
  511. smmu = &mdf_smmu_data[i];
  512. smmu->enabled = of_property_read_bool(dev->of_node,
  513. "qcom,smmu-enabled");
  514. dev_info(dev, "%s: SMMU is %s for %s\n", __func__,
  515. (smmu->enabled) ? "enabled" : "disabled",
  516. smmu->subsys);
  517. if (smmu->enabled) {
  518. /* Get SMMU SID information from Devicetree */
  519. rc = of_property_read_u64(dev->of_node,
  520. "qcom,smmu-sid-mask",
  521. &smmu_sid_mask);
  522. if (rc) {
  523. dev_err(dev,
  524. "%s: qcom,smmu-sid-mask missing in DT node, using default\n",
  525. __func__);
  526. smmu_sid_mask = 0xF;
  527. }
  528. rc = of_parse_phandle_with_args(dev->of_node, "iommus",
  529. "#iommu-cells", 0, &iommuspec);
  530. if (rc)
  531. dev_err(dev, "%s: could not get smmu SID, ret = %d\n",
  532. __func__, rc);
  533. else
  534. smmu_sid = (iommuspec.args[0] & smmu_sid_mask);
  535. smmu->sid =
  536. smmu_sid << MSM_AUDIO_SMMU_SID_OFFSET;
  537. smmu->cb_dev = dev;
  538. }
  539. return 0;
  540. }
  541. static int msm_mdf_remove(struct platform_device *pdev)
  542. {
  543. int rc = 0, i;
  544. for (i = 0; i < SUBSYS_MAX; i++) {
  545. if (!IS_ERR_OR_NULL(mdf_smmu_data[i].cb_dev))
  546. arm_iommu_detach_device(mdf_smmu_data[i].cb_dev);
  547. if (!IS_ERR_OR_NULL(mdf_smmu_data[i].mapping))
  548. arm_iommu_release_mapping(mdf_smmu_data[i].mapping);
  549. mdf_smmu_data[i].enabled = 0;
  550. }
  551. mdf_mem_data.device_status = 0;
  552. return rc;
  553. }
  554. static int msm_mdf_probe(struct platform_device *pdev)
  555. {
  556. int rc = 0;
  557. enum apr_subsys_state q6_state;
  558. struct device *dev = &pdev->dev;
  559. uint32_t mdf_mem_data_size = 0;
  560. /* TODO: MDF probing should have no dependency
  561. * on ADSP Q6 state.
  562. */
  563. q6_state = apr_get_q6_state();
  564. if (q6_state == APR_SUBSYS_DOWN) {
  565. dev_dbg(dev, "defering %s, adsp_state %d\n",
  566. __func__, q6_state);
  567. rc = -EPROBE_DEFER;
  568. goto err;
  569. } else
  570. dev_dbg(dev, "%s: adsp is ready\n", __func__);
  571. if (of_device_is_compatible(dev->of_node,
  572. "qcom,msm-mdf-cb"))
  573. return msm_mdf_cb_probe(dev);
  574. if (of_device_is_compatible(dev->of_node,
  575. "qcom,msm-mdf-mem-region")) {
  576. mdf_mem_data.dev = dev;
  577. rc = of_property_read_u32(dev->of_node,
  578. "qcom,msm-mdf-mem-data-size",
  579. &mdf_mem_data_size);
  580. if (rc) {
  581. dev_dbg(&pdev->dev, "MDF mem data size entry not found\n");
  582. goto err;
  583. }
  584. mdf_mem_data.size = mdf_mem_data_size;
  585. dev_info(dev, "%s: mem region size %zd\n",
  586. __func__, mdf_mem_data.size);
  587. msm_mdf_alloc_dma_buf(&mdf_mem_data);
  588. return 0;
  589. }
  590. rc = of_platform_populate(pdev->dev.of_node,
  591. msm_mdf_match_table,
  592. NULL, &pdev->dev);
  593. if (rc) {
  594. dev_err(&pdev->dev, "%s: failed to populate child nodes",
  595. __func__);
  596. goto err;
  597. }
  598. mdf_mem_data.device_status |= MSM_MDF_PROBED;
  599. err:
  600. return rc;
  601. }
  602. static struct platform_driver msm_mdf_driver = {
  603. .probe = msm_mdf_probe,
  604. .remove = msm_mdf_remove,
  605. .driver = {
  606. .name = "msm-mdf",
  607. .owner = THIS_MODULE,
  608. .of_match_table = msm_mdf_match_table,
  609. },
  610. };
  611. static struct notifier_block nb = {
  612. .priority = 0,
  613. .notifier_call = msm_mdf_restart_notifier_cb,
  614. };
  615. int __init msm_mdf_init(void)
  616. {
  617. /* Only need to monitor SSR from ADSP, which
  618. * is the master DSP managing MDF memory.
  619. */
  620. ssr_handle = subsys_notif_register_notifier("adsp", &nb);
  621. return platform_driver_register(&msm_mdf_driver);
  622. }
  623. void __exit msm_mdf_exit(void)
  624. {
  625. platform_driver_unregister(&msm_mdf_driver);
  626. if (ssr_handle)
  627. subsys_notif_unregister_notifier(ssr_handle, &nb);
  628. }
  629. MODULE_DESCRIPTION("MSM MDF Module");
  630. MODULE_LICENSE("GPL v2");