msm_mdf.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  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 <dsp/q6audio-v2.h>
  38. #include <dsp/q6core.h>
  39. #define VMID_SSC_Q6 5
  40. #define VMID_LPASS 6
  41. #define VMID_MSS_MSA 15
  42. #define VMID_CDSP 8
  43. #define MSM_MDF_PROBED (1 << 0)
  44. #define MSM_MDF_INITIALIZED (1 << 1)
  45. #define MSM_MDF_MEM_ALLOCATED (1 << 2)
  46. #define MSM_MDF_MEM_MAPPED (1 << 3)
  47. #define MSM_MDF_MEM_PERMISSION (1 << 4) /* 0 - HLOS, 1 - Subsys */
  48. /* TODO: Update IOVA range for subsys SMMUs */
  49. #define MSM_MDF_IOVA_START 0x80000000
  50. #define MSM_MDF_IOVA_LEN 0x800000
  51. #define MSM_MDF_SMMU_SID_OFFSET 32
  52. #define ADSP_STATE_READY_TIMEOUT_MS 3000
  53. enum {
  54. SUBSYS_ADSP, /* Audio DSP must have index 0 */
  55. SUBSYS_SCC, /* Sensor DSP */
  56. SUBSYS_MSS, /* Modem DSP */
  57. SUBSYS_CDSP, /* Compute DSP */
  58. SUBSYS_MAX,
  59. };
  60. struct msm_mdf_mem {
  61. struct device *dev;
  62. uint8_t device_status;
  63. uint32_t map_handle;
  64. struct dma_buf *dma_buf;
  65. dma_addr_t dma_addr;
  66. size_t size;
  67. void *va;
  68. };
  69. static struct msm_mdf_mem mdf_mem_data = {NULL,};
  70. struct msm_mdf_smmu {
  71. bool enabled;
  72. char *subsys;
  73. int vmid;
  74. uint32_t proc_id;
  75. struct device *cb_dev;
  76. uint8_t device_status;
  77. uint64_t sid;
  78. struct dma_iommu_mapping *mapping;
  79. dma_addr_t pa;
  80. size_t pa_len;
  81. };
  82. static struct msm_mdf_smmu mdf_smmu_data[SUBSYS_MAX] = {
  83. {
  84. .subsys = "adsp",
  85. .vmid = VMID_LPASS,
  86. },
  87. {
  88. .subsys = "dsps",
  89. .vmid = VMID_SSC_Q6,
  90. .proc_id = AVS_MDF_SSC_PROC_ID,
  91. },
  92. {
  93. .subsys = "modem",
  94. .vmid = VMID_MSS_MSA,
  95. .proc_id = AVS_MDF_MDSP_PROC_ID,
  96. },
  97. {
  98. .subsys = "cdsp",
  99. .vmid = VMID_CDSP,
  100. .proc_id = AVS_MDF_CDSP_PROC_ID,
  101. },
  102. };
  103. static void *ssr_handle;
  104. static inline uint64_t buf_page_start(uint64_t buf)
  105. {
  106. uint64_t start = (uint64_t) buf & PAGE_MASK;
  107. return start;
  108. }
  109. static inline uint64_t buf_page_offset(uint64_t buf)
  110. {
  111. uint64_t offset = (uint64_t) buf & (PAGE_SIZE - 1);
  112. return offset;
  113. }
  114. static inline int buf_num_pages(uint64_t buf, ssize_t len)
  115. {
  116. uint64_t start = buf_page_start(buf) >> PAGE_SHIFT;
  117. uint64_t end = (((uint64_t) buf + len - 1) & PAGE_MASK) >> PAGE_SHIFT;
  118. int nPages = end - start + 1;
  119. return nPages;
  120. }
  121. static inline uint64_t buf_page_size(uint32_t size)
  122. {
  123. uint64_t sz = (size + (PAGE_SIZE - 1)) & PAGE_MASK;
  124. return sz > PAGE_SIZE ? sz : PAGE_SIZE;
  125. }
  126. static inline void *uint64_to_ptr(uint64_t addr)
  127. {
  128. void *ptr = (void *)((uintptr_t)addr);
  129. return ptr;
  130. }
  131. static inline uint64_t ptr_to_uint64(void *ptr)
  132. {
  133. uint64_t addr = (uint64_t)((uintptr_t)ptr);
  134. return addr;
  135. }
  136. static int msm_mdf_dma_buf_map(struct msm_mdf_mem *mem,
  137. struct msm_mdf_smmu *smmu)
  138. {
  139. int rc = 0;
  140. if (!smmu)
  141. return -EINVAL;
  142. if (smmu->device_status & MSM_MDF_MEM_MAPPED)
  143. return 0;
  144. if (smmu->enabled) {
  145. if (smmu->cb_dev == NULL) {
  146. pr_err("%s: cb device is not initialized\n",
  147. __func__);
  148. /* Retry if LPASS cb device is not ready
  149. * from audio ION during probing.
  150. */
  151. if (!strcmp("adsp", smmu->subsys)) {
  152. rc = msm_audio_ion_get_smmu_info(&smmu->cb_dev,
  153. &smmu->sid);
  154. if (rc) {
  155. pr_err("%s: msm_audio_ion_get_smmu_info failed, rc = %d\n",
  156. __func__, rc);
  157. goto err;
  158. }
  159. } else
  160. return -ENODEV;
  161. }
  162. smmu->pa = dma_map_single(smmu->cb_dev, mem->va,
  163. mem->size, DMA_BIDIRECTIONAL);
  164. if (dma_mapping_error(smmu->cb_dev, smmu->pa)) {
  165. rc = -ENOMEM;
  166. pr_err("%s: failed to map single, rc = %d\n",
  167. __func__, rc);
  168. goto err;
  169. }
  170. smmu->pa_len = mem->size;
  171. /* Append the SMMU SID information to the IOVA address */
  172. if (smmu->sid)
  173. smmu->pa |= smmu->sid;
  174. } else {
  175. smmu->pa = mem->dma_addr;
  176. smmu->pa_len = mem->size;
  177. }
  178. pr_err("%s: pa=%pa, pa_len=%zd\n", __func__,
  179. &smmu->pa, smmu->pa_len);
  180. smmu->device_status |= MSM_MDF_MEM_MAPPED;
  181. return 0;
  182. err:
  183. return rc;
  184. }
  185. static int msm_mdf_alloc_dma_buf(struct msm_mdf_mem *mem)
  186. {
  187. int rc = 0;
  188. if (!mem)
  189. return -EINVAL;
  190. if (mem->device_status & MSM_MDF_MEM_ALLOCATED)
  191. return 0;
  192. if (mem->dev == NULL) {
  193. pr_err("%s: device is not initialized\n",
  194. __func__);
  195. return -ENODEV;
  196. }
  197. mem->va = dma_alloc_coherent(mem->dev, mem->size,
  198. &mem->dma_addr, GFP_KERNEL);
  199. if (IS_ERR_OR_NULL(mem->va)) {
  200. pr_err("%s: failed to allocate dma memory, rc = %d\n",
  201. __func__, rc);
  202. return -ENOMEM;
  203. }
  204. mem->va = phys_to_virt(mem->dma_addr);
  205. mem->device_status |= MSM_MDF_MEM_ALLOCATED;
  206. return rc;
  207. }
  208. static int msm_mdf_free_dma_buf(struct msm_mdf_mem *mem)
  209. {
  210. if (!mem)
  211. return -EINVAL;
  212. if (mem->dev == NULL) {
  213. pr_err("%s: device is not initialized\n",
  214. __func__);
  215. return -ENODEV;
  216. }
  217. //dma_free_coherent(mem->dev, mem->size, mem->va,
  218. // mem->dma_addr);
  219. mem->device_status &= ~MSM_MDF_MEM_ALLOCATED;
  220. return 0;
  221. }
  222. static int msm_mdf_dma_buf_unmap(struct msm_mdf_mem *mem,
  223. struct msm_mdf_smmu *smmu)
  224. {
  225. if (!smmu)
  226. return -EINVAL;
  227. if (smmu->enabled) {
  228. if (smmu->cb_dev == NULL) {
  229. pr_err("%s: cb device is not initialized\n",
  230. __func__);
  231. return -ENODEV;
  232. }
  233. //if (smmu->pa && mem->size)
  234. //dma_unmap_single(smmu->cb_dev, smmu->pa,
  235. // mem->size, DMA_BIDIRECTIONAL);
  236. }
  237. smmu->device_status &= ~MSM_MDF_MEM_MAPPED;
  238. return 0;
  239. }
  240. static int msm_mdf_map_memory_to_subsys(struct msm_mdf_mem *mem,
  241. struct msm_mdf_smmu *smmu)
  242. {
  243. int rc = 0;
  244. if (!mem || !smmu)
  245. return -EINVAL;
  246. /* Map mdf shared memory to ADSP */
  247. if (!strcmp("adsp", smmu->subsys)) {
  248. rc = q6core_map_memory_regions((phys_addr_t *)&smmu->pa,
  249. ADSP_MEMORY_MAP_MDF_SHMEM_4K_POOL,
  250. (uint32_t *)&smmu->pa_len, 1, &mem->map_handle);
  251. if (rc) {
  252. pr_err("%s: q6core_map_memory_regions failed, rc = %d\n",
  253. __func__, rc);
  254. }
  255. } else {
  256. if (mem->map_handle) {
  257. /* Map mdf shared memory to remote DSPs */
  258. rc = q6core_map_mdf_shared_memory(mem->map_handle,
  259. (phys_addr_t *)&smmu->pa, smmu->proc_id,
  260. (uint32_t *)&smmu->pa_len, 1);
  261. if (rc) {
  262. pr_err("%s: q6core_map_mdf_shared_memory failed, rc = %d\n",
  263. __func__, rc);
  264. }
  265. }
  266. }
  267. return rc;
  268. }
  269. static void msm_mdf_unmap_memory_to_subsys(struct msm_mdf_mem *mem,
  270. struct msm_mdf_smmu *smmu)
  271. {
  272. if (!mem || !smmu)
  273. return;
  274. if (!strcmp("adsp", smmu->subsys)) {
  275. if (mem->map_handle)
  276. q6core_memory_unmap_regions(mem->map_handle);
  277. }
  278. }
  279. /**
  280. * msm_mdf_mem_init - Initializes MDF memory pool and
  281. * map memory to subsystem
  282. *
  283. * Returns 0 on success or ret on failure.
  284. */
  285. int msm_mdf_mem_init(void)
  286. {
  287. int rc = 0, i, j;
  288. struct msm_mdf_mem *mem = &mdf_mem_data;
  289. struct msm_mdf_smmu *smmu;
  290. unsigned long timeout = jiffies +
  291. msecs_to_jiffies(ADSP_STATE_READY_TIMEOUT_MS);
  292. int adsp_ready = 0;
  293. if (!(mdf_mem_data.device_status & MSM_MDF_PROBED))
  294. return -ENODEV;
  295. if (mdf_mem_data.device_status & MSM_MDF_INITIALIZED)
  296. return 0;
  297. /* TODO: pulling may not be needed as Q6 Core state should be
  298. * checked during machine driver probing.
  299. */
  300. do {
  301. if (!q6core_is_adsp_ready()) {
  302. pr_err("%s: ADSP Audio NOT Ready\n",
  303. __func__);
  304. /* ADSP will be coming up after subsystem restart and
  305. * it might not be fully up when the control reaches
  306. * here. So, wait for 50msec before checking ADSP state
  307. */
  308. msleep(50);
  309. } else {
  310. pr_debug("%s: ADSP Audio Ready\n",
  311. __func__);
  312. adsp_ready = 1;
  313. break;
  314. }
  315. } while (time_after(timeout, jiffies));
  316. if (!adsp_ready) {
  317. pr_err("%s: timed out waiting for ADSP Audio\n",
  318. __func__);
  319. return -ETIMEDOUT;
  320. }
  321. if (mem->device_status & MSM_MDF_MEM_ALLOCATED) {
  322. for (i = 0; i < SUBSYS_MAX; i++) {
  323. smmu = &mdf_smmu_data[i];
  324. rc = msm_mdf_dma_buf_map(mem, smmu);
  325. if (rc) {
  326. pr_err("%s: msm_mdf_dma_buf_map failed, rc = %d\n",
  327. __func__, rc);
  328. goto err;
  329. }
  330. }
  331. for (j = 0; j < SUBSYS_MAX; j++) {
  332. smmu = &mdf_smmu_data[j];
  333. rc = msm_mdf_map_memory_to_subsys(mem, smmu);
  334. if (rc) {
  335. pr_err("%s: msm_mdf_map_memory_to_subsys failed\n",
  336. __func__);
  337. goto err;
  338. }
  339. }
  340. mdf_mem_data.device_status |= MSM_MDF_INITIALIZED;
  341. }
  342. return 0;
  343. err:
  344. return rc;
  345. }
  346. EXPORT_SYMBOL(msm_mdf_mem_init);
  347. int msm_mdf_mem_deinit(void)
  348. {
  349. int rc = 0, i;
  350. struct msm_mdf_mem *mem = &mdf_mem_data;
  351. struct msm_mdf_smmu *smmu;
  352. if (!(mdf_mem_data.device_status & MSM_MDF_INITIALIZED))
  353. return -ENODEV;
  354. for (i = SUBSYS_MAX - 1; i >= 0; i--) {
  355. smmu = &mdf_smmu_data[i];
  356. msm_mdf_unmap_memory_to_subsys(mem, smmu);
  357. }
  358. if (!rc) {
  359. for (i = SUBSYS_MAX - 1; i >= 0; i--) {
  360. smmu = &mdf_smmu_data[i];
  361. msm_mdf_dma_buf_unmap(mem, smmu);
  362. }
  363. msm_mdf_free_dma_buf(mem);
  364. mem->device_status &= ~MSM_MDF_MEM_ALLOCATED;
  365. }
  366. mdf_mem_data.device_status &= ~MSM_MDF_INITIALIZED;
  367. return 0;
  368. }
  369. EXPORT_SYMBOL(msm_mdf_mem_deinit);
  370. static int msm_mdf_restart_notifier_cb(struct notifier_block *this,
  371. unsigned long code,
  372. void *_cmd)
  373. {
  374. static int boot_count = 3;
  375. /* During LPASS boot, HLOS receives events:
  376. * SUBSYS_BEFORE_POWERUP
  377. * SUBSYS_PROXY_VOTE
  378. * SUBSYS_AFTER_POWERUP - need skip
  379. * SUBSYS_PROXY_UNVOTE
  380. */
  381. if (boot_count) {
  382. boot_count--;
  383. return NOTIFY_OK;
  384. }
  385. switch (code) {
  386. case SUBSYS_BEFORE_SHUTDOWN:
  387. pr_debug("Subsys Notify: Shutdown Started\n");
  388. /* Unmap and free memory upon restart event. */
  389. msm_mdf_mem_deinit();
  390. break;
  391. case SUBSYS_AFTER_SHUTDOWN:
  392. pr_debug("Subsys Notify: Shutdown Completed\n");
  393. break;
  394. case SUBSYS_BEFORE_POWERUP:
  395. pr_debug("Subsys Notify: Bootup Started\n");
  396. break;
  397. case SUBSYS_AFTER_POWERUP:
  398. pr_debug("Subsys Notify: Bootup Completed\n");
  399. /* Allocate and map memory after restart complete. */
  400. if (msm_mdf_mem_init())
  401. pr_err("msm_mdf_mem_init failed\n");
  402. break;
  403. default:
  404. pr_err("Subsys Notify: Generel: %lu\n", code);
  405. break;
  406. }
  407. return NOTIFY_DONE;
  408. }
  409. static const struct of_device_id msm_mdf_match_table[] = {
  410. { .compatible = "qcom,msm-mdf", },
  411. { .compatible = "qcom,msm-mdf-mem-region", },
  412. { .compatible = "qcom,msm-mdf-cb", },
  413. {}
  414. };
  415. MODULE_DEVICE_TABLE(of, msm_mdf_match_table);
  416. static int msm_mdf_cb_probe(struct device *dev)
  417. {
  418. struct msm_mdf_smmu *smmu;
  419. const char *subsys;
  420. int rc = 0, i;
  421. subsys = of_get_property(dev->of_node, "label", NULL);
  422. if (!subsys) {
  423. dev_err(dev, "%s: could not get label\n",
  424. __func__);
  425. return -EINVAL;
  426. }
  427. for (i = 0; i < SUBSYS_MAX; i++) {
  428. if (!mdf_smmu_data[i].subsys)
  429. continue;
  430. if (!strcmp(subsys, mdf_smmu_data[i].subsys))
  431. break;
  432. }
  433. if (i >= SUBSYS_MAX) {
  434. dev_err(dev, "%s: subsys %s not supported\n",
  435. __func__, subsys);
  436. return -EINVAL;
  437. }
  438. smmu = &mdf_smmu_data[i];
  439. smmu->enabled = of_property_read_bool(dev->of_node,
  440. "qcom,smmu-enabled");
  441. dev_info(dev, "%s: SMMU is %s for %s\n", __func__,
  442. (smmu->enabled) ? "enabled" : "disabled",
  443. smmu->subsys);
  444. if (smmu->enabled) {
  445. if (!strcmp("adsp", smmu->subsys)) {
  446. /* Get SMMU info from audio ION */
  447. rc = msm_audio_ion_get_smmu_info(&smmu->cb_dev,
  448. &smmu->sid);
  449. if (rc) {
  450. dev_err(dev, "%s: msm_audio_ion_get_smmu_info failed, rc = %d\n",
  451. __func__, rc);
  452. goto err;
  453. }
  454. }
  455. } else {
  456. /* Setup SMMU CB if enabled for subsys other than ADSP */
  457. }
  458. return 0;
  459. err:
  460. return rc;
  461. }
  462. static int msm_mdf_remove(struct platform_device *pdev)
  463. {
  464. int rc = 0, i;
  465. for (i = 0; i < SUBSYS_MAX; i++) {
  466. if (!IS_ERR_OR_NULL(mdf_smmu_data[i].cb_dev))
  467. arm_iommu_detach_device(mdf_smmu_data[i].cb_dev);
  468. if (!IS_ERR_OR_NULL(mdf_smmu_data[i].mapping))
  469. arm_iommu_release_mapping(mdf_smmu_data[i].mapping);
  470. mdf_smmu_data[i].enabled = 0;
  471. }
  472. mdf_mem_data.device_status = 0;
  473. return rc;
  474. }
  475. static int msm_mdf_probe(struct platform_device *pdev)
  476. {
  477. int rc = 0;
  478. enum apr_subsys_state q6_state;
  479. struct device *dev = &pdev->dev;
  480. uint32_t mdf_mem_data_size = 0;
  481. /* TODO: MDF probing should have no dependency
  482. * on ADSP Q6 state.
  483. */
  484. q6_state = apr_get_q6_state();
  485. if (q6_state == APR_SUBSYS_DOWN) {
  486. dev_dbg(dev, "defering %s, adsp_state %d\n",
  487. __func__, q6_state);
  488. rc = -EPROBE_DEFER;
  489. goto err;
  490. } else
  491. dev_dbg(dev, "%s: adsp is ready\n", __func__);
  492. if (of_device_is_compatible(dev->of_node,
  493. "qcom,msm-mdf-cb"))
  494. return msm_mdf_cb_probe(dev);
  495. if (of_device_is_compatible(dev->of_node,
  496. "qcom,msm-mdf-mem-region")) {
  497. mdf_mem_data.dev = dev;
  498. rc = of_property_read_u32(dev->of_node,
  499. "qcom,msm-mdf-mem-data-size",
  500. &mdf_mem_data_size);
  501. if (rc) {
  502. dev_dbg(&pdev->dev, "MDF mem data size entry not found\n");
  503. goto err;
  504. }
  505. mdf_mem_data.size = mdf_mem_data_size;
  506. dev_info(dev, "%s: mem region size %zd\n",
  507. __func__, mdf_mem_data.size);
  508. msm_mdf_alloc_dma_buf(&mdf_mem_data);
  509. return 0;
  510. }
  511. rc = of_platform_populate(pdev->dev.of_node,
  512. msm_mdf_match_table,
  513. NULL, &pdev->dev);
  514. if (rc) {
  515. dev_err(&pdev->dev, "%s: failed to populate child nodes",
  516. __func__);
  517. goto err;
  518. }
  519. mdf_mem_data.device_status |= MSM_MDF_PROBED;
  520. err:
  521. return rc;
  522. }
  523. static struct platform_driver msm_mdf_driver = {
  524. .probe = msm_mdf_probe,
  525. .remove = msm_mdf_remove,
  526. .driver = {
  527. .name = "msm-mdf",
  528. .owner = THIS_MODULE,
  529. .of_match_table = msm_mdf_match_table,
  530. },
  531. };
  532. static struct notifier_block nb = {
  533. .priority = 0,
  534. .notifier_call = msm_mdf_restart_notifier_cb,
  535. };
  536. int __init msm_mdf_init(void)
  537. {
  538. /* Only need to monitor SSR from ADSP, which
  539. * is the master DSP managing MDF memory.
  540. */
  541. ssr_handle = subsys_notif_register_notifier("adsp", &nb);
  542. return platform_driver_register(&msm_mdf_driver);
  543. }
  544. void __exit msm_mdf_exit(void)
  545. {
  546. platform_driver_unregister(&msm_mdf_driver);
  547. if (ssr_handle)
  548. subsys_notif_unregister_notifier(ssr_handle, &nb);
  549. }
  550. MODULE_DESCRIPTION("MSM MDF Module");
  551. MODULE_LICENSE("GPL v2");