msm_mdf.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728
  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. dma_addr_t 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. if (!smmu)
  164. return -EINVAL;
  165. if (smmu->device_status & MSM_MDF_MEM_MAPPED)
  166. return 0;
  167. if (smmu->enabled) {
  168. if (smmu->cb_dev == NULL) {
  169. pr_err("%s: cb device is not initialized\n",
  170. __func__);
  171. /* Retry if LPASS cb device is not ready
  172. * from audio ION during probing.
  173. */
  174. if (!strcmp("adsp", smmu->subsys)) {
  175. rc = msm_audio_ion_get_smmu_info(&smmu->cb_dev,
  176. &smmu->sid);
  177. if (rc) {
  178. pr_err("%s: msm_audio_ion_get_smmu_info failed, rc = %d\n",
  179. __func__, rc);
  180. goto err;
  181. }
  182. } else
  183. return -ENODEV;
  184. }
  185. smmu->pa = dma_map_single_attrs(smmu->cb_dev, mem->va,
  186. mem->size, DMA_BIDIRECTIONAL, DMA_ATTR_SKIP_CPU_SYNC);
  187. if (dma_mapping_error(smmu->cb_dev, smmu->pa)) {
  188. rc = -ENOMEM;
  189. pr_err("%s: failed to map single, rc = %d\n",
  190. __func__, rc);
  191. goto err;
  192. }
  193. smmu->pa_len = mem->size;
  194. /* Append the SMMU SID information to the IOVA address */
  195. if (smmu->sid)
  196. smmu->pa |= smmu->sid;
  197. } else {
  198. smmu->pa = mem->dma_addr;
  199. smmu->pa_len = mem->size;
  200. }
  201. pr_err("%s: pa=%pa, pa_len=%zd\n", __func__,
  202. &smmu->pa, smmu->pa_len);
  203. smmu->device_status |= MSM_MDF_MEM_MAPPED;
  204. return 0;
  205. err:
  206. return rc;
  207. }
  208. static int msm_mdf_alloc_dma_buf(struct msm_mdf_mem *mem)
  209. {
  210. int rc = 0;
  211. if (!mem)
  212. return -EINVAL;
  213. if (mem->device_status & MSM_MDF_MEM_ALLOCATED)
  214. return 0;
  215. if (mem->dev == NULL) {
  216. pr_err("%s: device is not initialized\n",
  217. __func__);
  218. return -ENODEV;
  219. }
  220. mem->va = dma_alloc_attrs(mem->dev, mem->size,
  221. &mem->dma_addr, GFP_KERNEL, DMA_ATTR_NO_KERNEL_MAPPING);
  222. if (IS_ERR_OR_NULL(mem->va)) {
  223. pr_err("%s: failed to allocate dma memory, rc = %d\n",
  224. __func__, rc);
  225. return -ENOMEM;
  226. }
  227. mem->va = phys_to_virt(mem->dma_addr);
  228. mem->device_status |= MSM_MDF_MEM_ALLOCATED;
  229. return rc;
  230. }
  231. static int msm_mdf_free_dma_buf(struct msm_mdf_mem *mem)
  232. {
  233. if (!mem)
  234. return -EINVAL;
  235. if (mem->dev == NULL) {
  236. pr_err("%s: device is not initialized\n",
  237. __func__);
  238. return -ENODEV;
  239. }
  240. //dma_free_coherent(mem->dev, mem->size, mem->va,
  241. // mem->dma_addr);
  242. mem->device_status &= ~MSM_MDF_MEM_ALLOCATED;
  243. return 0;
  244. }
  245. static int msm_mdf_dma_buf_unmap(struct msm_mdf_mem *mem,
  246. struct msm_mdf_smmu *smmu)
  247. {
  248. if (!smmu)
  249. return -EINVAL;
  250. if (smmu->enabled) {
  251. if (smmu->cb_dev == NULL) {
  252. pr_err("%s: cb device is not initialized\n",
  253. __func__);
  254. return -ENODEV;
  255. }
  256. //if (smmu->pa && mem->size)
  257. //dma_unmap_single(smmu->cb_dev, smmu->pa,
  258. // mem->size, DMA_BIDIRECTIONAL);
  259. }
  260. smmu->device_status &= ~MSM_MDF_MEM_MAPPED;
  261. return 0;
  262. }
  263. static int msm_mdf_map_memory_to_subsys(struct msm_mdf_mem *mem,
  264. struct msm_mdf_smmu *smmu)
  265. {
  266. int rc = 0;
  267. if (!mem || !smmu)
  268. return -EINVAL;
  269. /* Map mdf shared memory to ADSP */
  270. if (!strcmp("adsp", smmu->subsys)) {
  271. rc = q6core_map_memory_regions((phys_addr_t *)&smmu->pa,
  272. ADSP_MEMORY_MAP_MDF_SHMEM_4K_POOL,
  273. (uint32_t *)&smmu->pa_len, 1, &mem->map_handle);
  274. if (rc) {
  275. pr_err("%s: q6core_map_memory_regions failed, rc = %d\n",
  276. __func__, rc);
  277. }
  278. } else {
  279. if (mem->map_handle) {
  280. /* Map mdf shared memory to remote DSPs */
  281. rc = q6core_map_mdf_shared_memory(mem->map_handle,
  282. (phys_addr_t *)&smmu->pa, smmu->proc_id,
  283. (uint32_t *)&smmu->pa_len, 1);
  284. if (rc) {
  285. pr_err("%s: q6core_map_mdf_shared_memory failed, rc = %d\n",
  286. __func__, rc);
  287. }
  288. }
  289. }
  290. return rc;
  291. }
  292. static void msm_mdf_unmap_memory_to_subsys(struct msm_mdf_mem *mem,
  293. struct msm_mdf_smmu *smmu)
  294. {
  295. if (!mem || !smmu)
  296. return;
  297. if (!strcmp("adsp", smmu->subsys)) {
  298. if (mem->map_handle)
  299. q6core_memory_unmap_regions(mem->map_handle);
  300. }
  301. }
  302. static int msm_mdf_assign_memory_to_subsys(struct msm_mdf_mem *mem)
  303. {
  304. int ret = 0, i;
  305. struct scm_desc desc = {0};
  306. struct msm_mdf_protect_mem *scm_buffer;
  307. uint32_t fnid;
  308. scm_buffer = kzalloc(sizeof(struct msm_mdf_protect_mem), GFP_KERNEL);
  309. if (!scm_buffer)
  310. return -ENOMEM;
  311. scm_buffer->dma_start_address = mem->dma_addr;
  312. scm_buffer->dma_end_address = mem->dma_addr + buf_page_size(mem->size);
  313. for (i = 0; i < SUBSYS_MAX; i++) {
  314. scm_buffer->dest_info[i].dst_vm = mdf_smmu_data[i].vmid;
  315. scm_buffer->dest_info[i].dst_vm_perm =
  316. MEM_PROTECT_AC_PERM_READ | MEM_PROTECT_AC_PERM_WRITE;
  317. scm_buffer->dest_info[i].ctx = 0;
  318. scm_buffer->dest_info[i].ctx_size = 0;
  319. }
  320. scm_buffer->dest_info_size =
  321. sizeof(struct msm_mdf_dest_vm_and_perm_info) * SUBSYS_MAX;
  322. /* flush cache required by scm_call2 */
  323. dmac_flush_range(scm_buffer, ((void *)scm_buffer) +
  324. sizeof(struct msm_mdf_protect_mem));
  325. desc.args[0] = scm_buffer->dma_start_address;
  326. desc.args[1] = scm_buffer->dma_end_address;
  327. desc.args[2] = virt_to_phys(&(scm_buffer->dest_info[0]));
  328. desc.args[3] = scm_buffer->dest_info_size;
  329. desc.arginfo = SCM_ARGS(4, SCM_VAL, SCM_VAL, SCM_RO, SCM_VAL);
  330. fnid = SCM_SIP_FNID(SCM_SVC_MP, TZ_MPU_LOCK_NS_REGION);
  331. ret = scm_call2(fnid, &desc);
  332. if (ret < 0) {
  333. pr_err("%s: SCM call2 failed, ret %d scm_resp %llu\n",
  334. __func__, ret, desc.ret[0]);
  335. }
  336. /* No More need for scm_buffer, freeing the same */
  337. kfree(scm_buffer);
  338. return ret;
  339. }
  340. /**
  341. * msm_mdf_mem_init - Initializes MDF memory pool and
  342. * map memory to subsystem
  343. *
  344. * Returns 0 on success or ret on failure.
  345. */
  346. int msm_mdf_mem_init(void)
  347. {
  348. int rc = 0, i, j;
  349. struct msm_mdf_mem *mem = &mdf_mem_data;
  350. struct msm_mdf_smmu *smmu;
  351. unsigned long timeout = jiffies +
  352. msecs_to_jiffies(ADSP_STATE_READY_TIMEOUT_MS);
  353. int adsp_ready = 0;
  354. if (!(mdf_mem_data.device_status & MSM_MDF_PROBED))
  355. return -ENODEV;
  356. if (mdf_mem_data.device_status & MSM_MDF_INITIALIZED)
  357. return 0;
  358. /* TODO: pulling may not be needed as Q6 Core state should be
  359. * checked during machine driver probing.
  360. */
  361. do {
  362. if (!q6core_is_adsp_ready()) {
  363. pr_err("%s: ADSP Audio NOT Ready\n",
  364. __func__);
  365. /* ADSP will be coming up after subsystem restart and
  366. * it might not be fully up when the control reaches
  367. * here. So, wait for 50msec before checking ADSP state
  368. */
  369. msleep(50);
  370. } else {
  371. pr_debug("%s: ADSP Audio Ready\n",
  372. __func__);
  373. adsp_ready = 1;
  374. break;
  375. }
  376. } while (time_after(timeout, jiffies));
  377. if (!adsp_ready) {
  378. pr_err("%s: timed out waiting for ADSP Audio\n",
  379. __func__);
  380. return -ETIMEDOUT;
  381. }
  382. if (mem->device_status & MSM_MDF_MEM_ALLOCATED) {
  383. for (i = 0; i < SUBSYS_MAX; i++) {
  384. smmu = &mdf_smmu_data[i];
  385. rc = msm_mdf_dma_buf_map(mem, smmu);
  386. if (rc) {
  387. pr_err("%s: msm_mdf_dma_buf_map failed, rc = %d\n",
  388. __func__, rc);
  389. goto err;
  390. }
  391. }
  392. rc = msm_mdf_assign_memory_to_subsys(mem);
  393. if (rc) {
  394. pr_err("%s: msm_mdf_assign_memory_to_subsys failed\n",
  395. __func__);
  396. goto err;
  397. }
  398. for (j = 0; j < SUBSYS_MAX; j++) {
  399. smmu = &mdf_smmu_data[j];
  400. rc = msm_mdf_map_memory_to_subsys(mem, smmu);
  401. if (rc) {
  402. pr_err("%s: msm_mdf_map_memory_to_subsys failed\n",
  403. __func__);
  404. goto err;
  405. }
  406. }
  407. mdf_mem_data.device_status |= MSM_MDF_INITIALIZED;
  408. }
  409. return 0;
  410. err:
  411. return rc;
  412. }
  413. EXPORT_SYMBOL(msm_mdf_mem_init);
  414. int msm_mdf_mem_deinit(void)
  415. {
  416. int rc = 0, i;
  417. struct msm_mdf_mem *mem = &mdf_mem_data;
  418. struct msm_mdf_smmu *smmu;
  419. if (!(mdf_mem_data.device_status & MSM_MDF_INITIALIZED))
  420. return -ENODEV;
  421. for (i = SUBSYS_MAX - 1; i >= 0; i--) {
  422. smmu = &mdf_smmu_data[i];
  423. msm_mdf_unmap_memory_to_subsys(mem, smmu);
  424. }
  425. if (!rc) {
  426. for (i = SUBSYS_MAX - 1; i >= 0; i--) {
  427. smmu = &mdf_smmu_data[i];
  428. msm_mdf_dma_buf_unmap(mem, smmu);
  429. }
  430. msm_mdf_free_dma_buf(mem);
  431. mem->device_status &= ~MSM_MDF_MEM_ALLOCATED;
  432. }
  433. mdf_mem_data.device_status &= ~MSM_MDF_INITIALIZED;
  434. return 0;
  435. }
  436. EXPORT_SYMBOL(msm_mdf_mem_deinit);
  437. static int msm_mdf_restart_notifier_cb(struct notifier_block *this,
  438. unsigned long code,
  439. void *_cmd)
  440. {
  441. static int boot_count = 3;
  442. /* During LPASS boot, HLOS receives events:
  443. * SUBSYS_BEFORE_POWERUP
  444. * SUBSYS_PROXY_VOTE
  445. * SUBSYS_AFTER_POWERUP - need skip
  446. * SUBSYS_PROXY_UNVOTE
  447. */
  448. if (boot_count) {
  449. boot_count--;
  450. return NOTIFY_OK;
  451. }
  452. switch (code) {
  453. case SUBSYS_BEFORE_SHUTDOWN:
  454. pr_debug("Subsys Notify: Shutdown Started\n");
  455. /* Unmap and free memory upon restart event. */
  456. msm_mdf_mem_deinit();
  457. break;
  458. case SUBSYS_AFTER_SHUTDOWN:
  459. pr_debug("Subsys Notify: Shutdown Completed\n");
  460. break;
  461. case SUBSYS_BEFORE_POWERUP:
  462. pr_debug("Subsys Notify: Bootup Started\n");
  463. break;
  464. case SUBSYS_AFTER_POWERUP:
  465. pr_debug("Subsys Notify: Bootup Completed\n");
  466. /* Allocate and map memory after restart complete. */
  467. if (msm_mdf_mem_init())
  468. pr_err("msm_mdf_mem_init failed\n");
  469. break;
  470. default:
  471. pr_err("Subsys Notify: Generel: %lu\n", code);
  472. break;
  473. }
  474. return NOTIFY_DONE;
  475. }
  476. static const struct of_device_id msm_mdf_match_table[] = {
  477. { .compatible = "qcom,msm-mdf", },
  478. { .compatible = "qcom,msm-mdf-mem-region", },
  479. { .compatible = "qcom,msm-mdf-cb", },
  480. {}
  481. };
  482. MODULE_DEVICE_TABLE(of, msm_mdf_match_table);
  483. static int msm_mdf_cb_probe(struct device *dev)
  484. {
  485. struct msm_mdf_smmu *smmu;
  486. u64 smmu_sid = 0;
  487. u64 smmu_sid_mask = 0;
  488. struct of_phandle_args iommuspec;
  489. const char *subsys;
  490. int rc = 0, i;
  491. subsys = of_get_property(dev->of_node, "label", NULL);
  492. if (!subsys) {
  493. dev_err(dev, "%s: could not get label\n",
  494. __func__);
  495. return -EINVAL;
  496. }
  497. for (i = 0; i < SUBSYS_MAX; i++) {
  498. if (!mdf_smmu_data[i].subsys)
  499. continue;
  500. if (!strcmp(subsys, mdf_smmu_data[i].subsys))
  501. break;
  502. }
  503. if (i >= SUBSYS_MAX) {
  504. dev_err(dev, "%s: subsys %s not supported\n",
  505. __func__, subsys);
  506. return -EINVAL;
  507. }
  508. smmu = &mdf_smmu_data[i];
  509. smmu->enabled = of_property_read_bool(dev->of_node,
  510. "qcom,smmu-enabled");
  511. dev_info(dev, "%s: SMMU is %s for %s\n", __func__,
  512. (smmu->enabled) ? "enabled" : "disabled",
  513. smmu->subsys);
  514. if (smmu->enabled) {
  515. /* Get SMMU SID information from Devicetree */
  516. rc = of_property_read_u64(dev->of_node,
  517. "qcom,smmu-sid-mask",
  518. &smmu_sid_mask);
  519. if (rc) {
  520. dev_err(dev,
  521. "%s: qcom,smmu-sid-mask missing in DT node, using default\n",
  522. __func__);
  523. smmu_sid_mask = 0xFFFFFFFFFFFFFFFF;
  524. }
  525. rc = of_parse_phandle_with_args(dev->of_node, "iommus",
  526. "#iommu-cells", 0, &iommuspec);
  527. if (rc)
  528. dev_err(dev, "%s: could not get smmu SID, ret = %d\n",
  529. __func__, rc);
  530. else
  531. smmu_sid = (iommuspec.args[0] & smmu_sid_mask);
  532. smmu->sid =
  533. smmu_sid << MSM_AUDIO_SMMU_SID_OFFSET;
  534. smmu->cb_dev = dev;
  535. }
  536. return 0;
  537. }
  538. static int msm_mdf_remove(struct platform_device *pdev)
  539. {
  540. int rc = 0, i;
  541. for (i = 0; i < SUBSYS_MAX; i++) {
  542. if (!IS_ERR_OR_NULL(mdf_smmu_data[i].cb_dev))
  543. arm_iommu_detach_device(mdf_smmu_data[i].cb_dev);
  544. if (!IS_ERR_OR_NULL(mdf_smmu_data[i].mapping))
  545. arm_iommu_release_mapping(mdf_smmu_data[i].mapping);
  546. mdf_smmu_data[i].enabled = 0;
  547. }
  548. mdf_mem_data.device_status = 0;
  549. return rc;
  550. }
  551. static int msm_mdf_probe(struct platform_device *pdev)
  552. {
  553. int rc = 0;
  554. enum apr_subsys_state q6_state;
  555. struct device *dev = &pdev->dev;
  556. uint32_t mdf_mem_data_size = 0;
  557. /* TODO: MDF probing should have no dependency
  558. * on ADSP Q6 state.
  559. */
  560. q6_state = apr_get_q6_state();
  561. if (q6_state == APR_SUBSYS_DOWN) {
  562. dev_dbg(dev, "defering %s, adsp_state %d\n",
  563. __func__, q6_state);
  564. rc = -EPROBE_DEFER;
  565. goto err;
  566. } else
  567. dev_dbg(dev, "%s: adsp is ready\n", __func__);
  568. if (of_device_is_compatible(dev->of_node,
  569. "qcom,msm-mdf-cb"))
  570. return msm_mdf_cb_probe(dev);
  571. if (of_device_is_compatible(dev->of_node,
  572. "qcom,msm-mdf-mem-region")) {
  573. mdf_mem_data.dev = dev;
  574. rc = of_property_read_u32(dev->of_node,
  575. "qcom,msm-mdf-mem-data-size",
  576. &mdf_mem_data_size);
  577. if (rc) {
  578. dev_dbg(&pdev->dev, "MDF mem data size entry not found\n");
  579. goto err;
  580. }
  581. mdf_mem_data.size = mdf_mem_data_size;
  582. dev_info(dev, "%s: mem region size %zd\n",
  583. __func__, mdf_mem_data.size);
  584. msm_mdf_alloc_dma_buf(&mdf_mem_data);
  585. return 0;
  586. }
  587. rc = of_platform_populate(pdev->dev.of_node,
  588. msm_mdf_match_table,
  589. NULL, &pdev->dev);
  590. if (rc) {
  591. dev_err(&pdev->dev, "%s: failed to populate child nodes",
  592. __func__);
  593. goto err;
  594. }
  595. mdf_mem_data.device_status |= MSM_MDF_PROBED;
  596. err:
  597. return rc;
  598. }
  599. static struct platform_driver msm_mdf_driver = {
  600. .probe = msm_mdf_probe,
  601. .remove = msm_mdf_remove,
  602. .driver = {
  603. .name = "msm-mdf",
  604. .owner = THIS_MODULE,
  605. .of_match_table = msm_mdf_match_table,
  606. },
  607. };
  608. static struct notifier_block nb = {
  609. .priority = 0,
  610. .notifier_call = msm_mdf_restart_notifier_cb,
  611. };
  612. int __init msm_mdf_init(void)
  613. {
  614. /* Only need to monitor SSR from ADSP, which
  615. * is the master DSP managing MDF memory.
  616. */
  617. ssr_handle = subsys_notif_register_notifier("adsp", &nb);
  618. return platform_driver_register(&msm_mdf_driver);
  619. }
  620. void __exit msm_mdf_exit(void)
  621. {
  622. platform_driver_unregister(&msm_mdf_driver);
  623. if (ssr_handle)
  624. subsys_notif_unregister_notifier(ssr_handle, &nb);
  625. }
  626. MODULE_DESCRIPTION("MSM MDF Module");
  627. MODULE_LICENSE("GPL v2");