qcom_q6v5_wcss.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) 2016-2018 Linaro Ltd.
  4. * Copyright (C) 2014 Sony Mobile Communications AB
  5. * Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
  6. */
  7. #include <linux/iopoll.h>
  8. #include <linux/kernel.h>
  9. #include <linux/mfd/syscon.h>
  10. #include <linux/module.h>
  11. #include <linux/of_reserved_mem.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/regmap.h>
  14. #include <linux/reset.h>
  15. #include <linux/soc/qcom/mdt_loader.h>
  16. #include "qcom_common.h"
  17. #include "qcom_q6v5.h"
  18. #define WCSS_CRASH_REASON 421
  19. /* Q6SS Register Offsets */
  20. #define Q6SS_RESET_REG 0x014
  21. #define Q6SS_GFMUX_CTL_REG 0x020
  22. #define Q6SS_PWR_CTL_REG 0x030
  23. #define Q6SS_MEM_PWR_CTL 0x0B0
  24. /* AXI Halt Register Offsets */
  25. #define AXI_HALTREQ_REG 0x0
  26. #define AXI_HALTACK_REG 0x4
  27. #define AXI_IDLE_REG 0x8
  28. #define HALT_ACK_TIMEOUT_MS 100
  29. /* Q6SS_RESET */
  30. #define Q6SS_STOP_CORE BIT(0)
  31. #define Q6SS_CORE_ARES BIT(1)
  32. #define Q6SS_BUS_ARES_ENABLE BIT(2)
  33. /* Q6SS_GFMUX_CTL */
  34. #define Q6SS_CLK_ENABLE BIT(1)
  35. /* Q6SS_PWR_CTL */
  36. #define Q6SS_L2DATA_STBY_N BIT(18)
  37. #define Q6SS_SLP_RET_N BIT(19)
  38. #define Q6SS_CLAMP_IO BIT(20)
  39. #define QDSS_BHS_ON BIT(21)
  40. /* Q6SS parameters */
  41. #define Q6SS_LDO_BYP BIT(25)
  42. #define Q6SS_BHS_ON BIT(24)
  43. #define Q6SS_CLAMP_WL BIT(21)
  44. #define Q6SS_CLAMP_QMC_MEM BIT(22)
  45. #define HALT_CHECK_MAX_LOOPS 200
  46. #define Q6SS_XO_CBCR GENMASK(5, 3)
  47. /* Q6SS config/status registers */
  48. #define TCSR_GLOBAL_CFG0 0x0
  49. #define TCSR_GLOBAL_CFG1 0x4
  50. #define SSCAON_CONFIG 0x8
  51. #define SSCAON_STATUS 0xc
  52. #define Q6SS_BHS_STATUS 0x78
  53. #define Q6SS_RST_EVB 0x10
  54. #define BHS_EN_REST_ACK BIT(0)
  55. #define SSCAON_ENABLE BIT(13)
  56. #define SSCAON_BUS_EN BIT(15)
  57. #define SSCAON_BUS_MUX_MASK GENMASK(18, 16)
  58. #define MEM_BANKS 19
  59. #define TCSR_WCSS_CLK_MASK 0x1F
  60. #define TCSR_WCSS_CLK_ENABLE 0x14
  61. struct q6v5_wcss {
  62. struct device *dev;
  63. void __iomem *reg_base;
  64. void __iomem *rmb_base;
  65. struct regmap *halt_map;
  66. u32 halt_q6;
  67. u32 halt_wcss;
  68. u32 halt_nc;
  69. struct reset_control *wcss_aon_reset;
  70. struct reset_control *wcss_reset;
  71. struct reset_control *wcss_q6_reset;
  72. struct qcom_q6v5 q6v5;
  73. phys_addr_t mem_phys;
  74. phys_addr_t mem_reloc;
  75. void *mem_region;
  76. size_t mem_size;
  77. };
  78. static int q6v5_wcss_reset(struct q6v5_wcss *wcss)
  79. {
  80. int ret;
  81. u32 val;
  82. int i;
  83. /* Assert resets, stop core */
  84. val = readl(wcss->reg_base + Q6SS_RESET_REG);
  85. val |= Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENABLE | Q6SS_STOP_CORE;
  86. writel(val, wcss->reg_base + Q6SS_RESET_REG);
  87. /* BHS require xo cbcr to be enabled */
  88. val = readl(wcss->reg_base + Q6SS_XO_CBCR);
  89. val |= 0x1;
  90. writel(val, wcss->reg_base + Q6SS_XO_CBCR);
  91. /* Read CLKOFF bit to go low indicating CLK is enabled */
  92. ret = readl_poll_timeout(wcss->reg_base + Q6SS_XO_CBCR,
  93. val, !(val & BIT(31)), 1,
  94. HALT_CHECK_MAX_LOOPS);
  95. if (ret) {
  96. dev_err(wcss->dev,
  97. "xo cbcr enabling timed out (rc:%d)\n", ret);
  98. return ret;
  99. }
  100. /* Enable power block headswitch and wait for it to stabilize */
  101. val = readl(wcss->reg_base + Q6SS_PWR_CTL_REG);
  102. val |= Q6SS_BHS_ON;
  103. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  104. udelay(1);
  105. /* Put LDO in bypass mode */
  106. val |= Q6SS_LDO_BYP;
  107. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  108. /* Deassert Q6 compiler memory clamp */
  109. val = readl(wcss->reg_base + Q6SS_PWR_CTL_REG);
  110. val &= ~Q6SS_CLAMP_QMC_MEM;
  111. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  112. /* Deassert memory peripheral sleep and L2 memory standby */
  113. val |= Q6SS_L2DATA_STBY_N | Q6SS_SLP_RET_N;
  114. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  115. /* Turn on L1, L2, ETB and JU memories 1 at a time */
  116. val = readl(wcss->reg_base + Q6SS_MEM_PWR_CTL);
  117. for (i = MEM_BANKS; i >= 0; i--) {
  118. val |= BIT(i);
  119. writel(val, wcss->reg_base + Q6SS_MEM_PWR_CTL);
  120. /*
  121. * Read back value to ensure the write is done then
  122. * wait for 1us for both memory peripheral and data
  123. * array to turn on.
  124. */
  125. val |= readl(wcss->reg_base + Q6SS_MEM_PWR_CTL);
  126. udelay(1);
  127. }
  128. /* Remove word line clamp */
  129. val = readl(wcss->reg_base + Q6SS_PWR_CTL_REG);
  130. val &= ~Q6SS_CLAMP_WL;
  131. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  132. /* Remove IO clamp */
  133. val &= ~Q6SS_CLAMP_IO;
  134. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  135. /* Bring core out of reset */
  136. val = readl(wcss->reg_base + Q6SS_RESET_REG);
  137. val &= ~Q6SS_CORE_ARES;
  138. writel(val, wcss->reg_base + Q6SS_RESET_REG);
  139. /* Turn on core clock */
  140. val = readl(wcss->reg_base + Q6SS_GFMUX_CTL_REG);
  141. val |= Q6SS_CLK_ENABLE;
  142. writel(val, wcss->reg_base + Q6SS_GFMUX_CTL_REG);
  143. /* Start core execution */
  144. val = readl(wcss->reg_base + Q6SS_RESET_REG);
  145. val &= ~Q6SS_STOP_CORE;
  146. writel(val, wcss->reg_base + Q6SS_RESET_REG);
  147. return 0;
  148. }
  149. static int q6v5_wcss_start(struct rproc *rproc)
  150. {
  151. struct q6v5_wcss *wcss = rproc->priv;
  152. int ret;
  153. qcom_q6v5_prepare(&wcss->q6v5);
  154. /* Release Q6 and WCSS reset */
  155. ret = reset_control_deassert(wcss->wcss_reset);
  156. if (ret) {
  157. dev_err(wcss->dev, "wcss_reset failed\n");
  158. return ret;
  159. }
  160. ret = reset_control_deassert(wcss->wcss_q6_reset);
  161. if (ret) {
  162. dev_err(wcss->dev, "wcss_q6_reset failed\n");
  163. goto wcss_reset;
  164. }
  165. /* Lithium configuration - clock gating and bus arbitration */
  166. ret = regmap_update_bits(wcss->halt_map,
  167. wcss->halt_nc + TCSR_GLOBAL_CFG0,
  168. TCSR_WCSS_CLK_MASK,
  169. TCSR_WCSS_CLK_ENABLE);
  170. if (ret)
  171. goto wcss_q6_reset;
  172. ret = regmap_update_bits(wcss->halt_map,
  173. wcss->halt_nc + TCSR_GLOBAL_CFG1,
  174. 1, 0);
  175. if (ret)
  176. goto wcss_q6_reset;
  177. /* Write bootaddr to EVB so that Q6WCSS will jump there after reset */
  178. writel(rproc->bootaddr >> 4, wcss->reg_base + Q6SS_RST_EVB);
  179. ret = q6v5_wcss_reset(wcss);
  180. if (ret)
  181. goto wcss_q6_reset;
  182. ret = qcom_q6v5_wait_for_start(&wcss->q6v5, 5 * HZ);
  183. if (ret == -ETIMEDOUT)
  184. dev_err(wcss->dev, "start timed out\n");
  185. return ret;
  186. wcss_q6_reset:
  187. reset_control_assert(wcss->wcss_q6_reset);
  188. wcss_reset:
  189. reset_control_assert(wcss->wcss_reset);
  190. return ret;
  191. }
  192. static void q6v5_wcss_halt_axi_port(struct q6v5_wcss *wcss,
  193. struct regmap *halt_map,
  194. u32 offset)
  195. {
  196. unsigned long timeout;
  197. unsigned int val;
  198. int ret;
  199. /* Check if we're already idle */
  200. ret = regmap_read(halt_map, offset + AXI_IDLE_REG, &val);
  201. if (!ret && val)
  202. return;
  203. /* Assert halt request */
  204. regmap_write(halt_map, offset + AXI_HALTREQ_REG, 1);
  205. /* Wait for halt */
  206. timeout = jiffies + msecs_to_jiffies(HALT_ACK_TIMEOUT_MS);
  207. for (;;) {
  208. ret = regmap_read(halt_map, offset + AXI_HALTACK_REG, &val);
  209. if (ret || val || time_after(jiffies, timeout))
  210. break;
  211. msleep(1);
  212. }
  213. ret = regmap_read(halt_map, offset + AXI_IDLE_REG, &val);
  214. if (ret || !val)
  215. dev_err(wcss->dev, "port failed halt\n");
  216. /* Clear halt request (port will remain halted until reset) */
  217. regmap_write(halt_map, offset + AXI_HALTREQ_REG, 0);
  218. }
  219. static int q6v5_wcss_powerdown(struct q6v5_wcss *wcss)
  220. {
  221. int ret;
  222. u32 val;
  223. /* 1 - Assert WCSS/Q6 HALTREQ */
  224. q6v5_wcss_halt_axi_port(wcss, wcss->halt_map, wcss->halt_wcss);
  225. /* 2 - Enable WCSSAON_CONFIG */
  226. val = readl(wcss->rmb_base + SSCAON_CONFIG);
  227. val |= SSCAON_ENABLE;
  228. writel(val, wcss->rmb_base + SSCAON_CONFIG);
  229. /* 3 - Set SSCAON_CONFIG */
  230. val |= SSCAON_BUS_EN;
  231. val &= ~SSCAON_BUS_MUX_MASK;
  232. writel(val, wcss->rmb_base + SSCAON_CONFIG);
  233. /* 4 - SSCAON_CONFIG 1 */
  234. val |= BIT(1);
  235. writel(val, wcss->rmb_base + SSCAON_CONFIG);
  236. /* 5 - wait for SSCAON_STATUS */
  237. ret = readl_poll_timeout(wcss->rmb_base + SSCAON_STATUS,
  238. val, (val & 0xffff) == 0x400, 1000,
  239. HALT_CHECK_MAX_LOOPS);
  240. if (ret) {
  241. dev_err(wcss->dev,
  242. "can't get SSCAON_STATUS rc:%d)\n", ret);
  243. return ret;
  244. }
  245. /* 6 - De-assert WCSS_AON reset */
  246. reset_control_assert(wcss->wcss_aon_reset);
  247. /* 7 - Disable WCSSAON_CONFIG 13 */
  248. val = readl(wcss->rmb_base + SSCAON_CONFIG);
  249. val &= ~SSCAON_ENABLE;
  250. writel(val, wcss->rmb_base + SSCAON_CONFIG);
  251. /* 8 - De-assert WCSS/Q6 HALTREQ */
  252. reset_control_assert(wcss->wcss_reset);
  253. return 0;
  254. }
  255. static int q6v5_q6_powerdown(struct q6v5_wcss *wcss)
  256. {
  257. int ret;
  258. u32 val;
  259. int i;
  260. /* 1 - Halt Q6 bus interface */
  261. q6v5_wcss_halt_axi_port(wcss, wcss->halt_map, wcss->halt_q6);
  262. /* 2 - Disable Q6 Core clock */
  263. val = readl(wcss->reg_base + Q6SS_GFMUX_CTL_REG);
  264. val &= ~Q6SS_CLK_ENABLE;
  265. writel(val, wcss->reg_base + Q6SS_GFMUX_CTL_REG);
  266. /* 3 - Clamp I/O */
  267. val = readl(wcss->reg_base + Q6SS_PWR_CTL_REG);
  268. val |= Q6SS_CLAMP_IO;
  269. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  270. /* 4 - Clamp WL */
  271. val |= QDSS_BHS_ON;
  272. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  273. /* 5 - Clear Erase standby */
  274. val &= ~Q6SS_L2DATA_STBY_N;
  275. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  276. /* 6 - Clear Sleep RTN */
  277. val &= ~Q6SS_SLP_RET_N;
  278. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  279. /* 7 - turn off Q6 memory foot/head switch one bank at a time */
  280. for (i = 0; i < 20; i++) {
  281. val = readl(wcss->reg_base + Q6SS_MEM_PWR_CTL);
  282. val &= ~BIT(i);
  283. writel(val, wcss->reg_base + Q6SS_MEM_PWR_CTL);
  284. mdelay(1);
  285. }
  286. /* 8 - Assert QMC memory RTN */
  287. val = readl(wcss->reg_base + Q6SS_PWR_CTL_REG);
  288. val |= Q6SS_CLAMP_QMC_MEM;
  289. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  290. /* 9 - Turn off BHS */
  291. val &= ~Q6SS_BHS_ON;
  292. writel(val, wcss->reg_base + Q6SS_PWR_CTL_REG);
  293. udelay(1);
  294. /* 10 - Wait till BHS Reset is done */
  295. ret = readl_poll_timeout(wcss->reg_base + Q6SS_BHS_STATUS,
  296. val, !(val & BHS_EN_REST_ACK), 1000,
  297. HALT_CHECK_MAX_LOOPS);
  298. if (ret) {
  299. dev_err(wcss->dev, "BHS_STATUS not OFF (rc:%d)\n", ret);
  300. return ret;
  301. }
  302. /* 11 - Assert WCSS reset */
  303. reset_control_assert(wcss->wcss_reset);
  304. /* 12 - Assert Q6 reset */
  305. reset_control_assert(wcss->wcss_q6_reset);
  306. return 0;
  307. }
  308. static int q6v5_wcss_stop(struct rproc *rproc)
  309. {
  310. struct q6v5_wcss *wcss = rproc->priv;
  311. int ret;
  312. /* WCSS powerdown */
  313. ret = qcom_q6v5_request_stop(&wcss->q6v5);
  314. if (ret == -ETIMEDOUT) {
  315. dev_err(wcss->dev, "timed out on wait\n");
  316. return ret;
  317. }
  318. ret = q6v5_wcss_powerdown(wcss);
  319. if (ret)
  320. return ret;
  321. /* Q6 Power down */
  322. ret = q6v5_q6_powerdown(wcss);
  323. if (ret)
  324. return ret;
  325. qcom_q6v5_unprepare(&wcss->q6v5);
  326. return 0;
  327. }
  328. static void *q6v5_wcss_da_to_va(struct rproc *rproc, u64 da, int len)
  329. {
  330. struct q6v5_wcss *wcss = rproc->priv;
  331. int offset;
  332. offset = da - wcss->mem_reloc;
  333. if (offset < 0 || offset + len > wcss->mem_size)
  334. return NULL;
  335. return wcss->mem_region + offset;
  336. }
  337. static int q6v5_wcss_load(struct rproc *rproc, const struct firmware *fw)
  338. {
  339. struct q6v5_wcss *wcss = rproc->priv;
  340. return qcom_mdt_load_no_init(wcss->dev, fw, rproc->firmware,
  341. 0, wcss->mem_region, wcss->mem_phys,
  342. wcss->mem_size, &wcss->mem_reloc);
  343. }
  344. static const struct rproc_ops q6v5_wcss_ops = {
  345. .start = q6v5_wcss_start,
  346. .stop = q6v5_wcss_stop,
  347. .da_to_va = q6v5_wcss_da_to_va,
  348. .load = q6v5_wcss_load,
  349. .get_boot_addr = rproc_elf_get_boot_addr,
  350. };
  351. static int q6v5_wcss_init_reset(struct q6v5_wcss *wcss)
  352. {
  353. struct device *dev = wcss->dev;
  354. wcss->wcss_aon_reset = devm_reset_control_get(dev, "wcss_aon_reset");
  355. if (IS_ERR(wcss->wcss_aon_reset)) {
  356. dev_err(wcss->dev, "unable to acquire wcss_aon_reset\n");
  357. return PTR_ERR(wcss->wcss_aon_reset);
  358. }
  359. wcss->wcss_reset = devm_reset_control_get(dev, "wcss_reset");
  360. if (IS_ERR(wcss->wcss_reset)) {
  361. dev_err(wcss->dev, "unable to acquire wcss_reset\n");
  362. return PTR_ERR(wcss->wcss_reset);
  363. }
  364. wcss->wcss_q6_reset = devm_reset_control_get(dev, "wcss_q6_reset");
  365. if (IS_ERR(wcss->wcss_q6_reset)) {
  366. dev_err(wcss->dev, "unable to acquire wcss_q6_reset\n");
  367. return PTR_ERR(wcss->wcss_q6_reset);
  368. }
  369. return 0;
  370. }
  371. static int q6v5_wcss_init_mmio(struct q6v5_wcss *wcss,
  372. struct platform_device *pdev)
  373. {
  374. struct of_phandle_args args;
  375. struct resource *res;
  376. int ret;
  377. res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "qdsp6");
  378. wcss->reg_base = devm_ioremap_resource(&pdev->dev, res);
  379. if (IS_ERR(wcss->reg_base))
  380. return PTR_ERR(wcss->reg_base);
  381. res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "rmb");
  382. wcss->rmb_base = devm_ioremap_resource(&pdev->dev, res);
  383. if (IS_ERR(wcss->rmb_base))
  384. return PTR_ERR(wcss->rmb_base);
  385. ret = of_parse_phandle_with_fixed_args(pdev->dev.of_node,
  386. "qcom,halt-regs", 3, 0, &args);
  387. if (ret < 0) {
  388. dev_err(&pdev->dev, "failed to parse qcom,halt-regs\n");
  389. return -EINVAL;
  390. }
  391. wcss->halt_map = syscon_node_to_regmap(args.np);
  392. of_node_put(args.np);
  393. if (IS_ERR(wcss->halt_map))
  394. return PTR_ERR(wcss->halt_map);
  395. wcss->halt_q6 = args.args[0];
  396. wcss->halt_wcss = args.args[1];
  397. wcss->halt_nc = args.args[2];
  398. return 0;
  399. }
  400. static int q6v5_alloc_memory_region(struct q6v5_wcss *wcss)
  401. {
  402. struct reserved_mem *rmem = NULL;
  403. struct device_node *node;
  404. struct device *dev = wcss->dev;
  405. node = of_parse_phandle(dev->of_node, "memory-region", 0);
  406. if (node)
  407. rmem = of_reserved_mem_lookup(node);
  408. of_node_put(node);
  409. if (!rmem) {
  410. dev_err(dev, "unable to acquire memory-region\n");
  411. return -EINVAL;
  412. }
  413. wcss->mem_phys = rmem->base;
  414. wcss->mem_reloc = rmem->base;
  415. wcss->mem_size = rmem->size;
  416. wcss->mem_region = devm_ioremap_wc(dev, wcss->mem_phys, wcss->mem_size);
  417. if (!wcss->mem_region) {
  418. dev_err(dev, "unable to map memory region: %pa+%pa\n",
  419. &rmem->base, &rmem->size);
  420. return -EBUSY;
  421. }
  422. return 0;
  423. }
  424. static int q6v5_wcss_probe(struct platform_device *pdev)
  425. {
  426. struct q6v5_wcss *wcss;
  427. struct rproc *rproc;
  428. int ret;
  429. rproc = rproc_alloc(&pdev->dev, pdev->name, &q6v5_wcss_ops,
  430. "IPQ8074/q6_fw.mdt", sizeof(*wcss));
  431. if (!rproc) {
  432. dev_err(&pdev->dev, "failed to allocate rproc\n");
  433. return -ENOMEM;
  434. }
  435. wcss = rproc->priv;
  436. wcss->dev = &pdev->dev;
  437. ret = q6v5_wcss_init_mmio(wcss, pdev);
  438. if (ret)
  439. goto free_rproc;
  440. ret = q6v5_alloc_memory_region(wcss);
  441. if (ret)
  442. goto free_rproc;
  443. ret = q6v5_wcss_init_reset(wcss);
  444. if (ret)
  445. goto free_rproc;
  446. ret = qcom_q6v5_init(&wcss->q6v5, pdev, rproc, WCSS_CRASH_REASON, NULL);
  447. if (ret)
  448. goto free_rproc;
  449. ret = rproc_add(rproc);
  450. if (ret)
  451. goto free_rproc;
  452. platform_set_drvdata(pdev, rproc);
  453. return 0;
  454. free_rproc:
  455. rproc_free(rproc);
  456. return ret;
  457. }
  458. static int q6v5_wcss_remove(struct platform_device *pdev)
  459. {
  460. struct rproc *rproc = platform_get_drvdata(pdev);
  461. rproc_del(rproc);
  462. rproc_free(rproc);
  463. return 0;
  464. }
  465. static const struct of_device_id q6v5_wcss_of_match[] = {
  466. { .compatible = "qcom,ipq8074-wcss-pil" },
  467. { },
  468. };
  469. MODULE_DEVICE_TABLE(of, q6v5_wcss_of_match);
  470. static struct platform_driver q6v5_wcss_driver = {
  471. .probe = q6v5_wcss_probe,
  472. .remove = q6v5_wcss_remove,
  473. .driver = {
  474. .name = "qcom-q6v5-wcss-pil",
  475. .of_match_table = q6v5_wcss_of_match,
  476. },
  477. };
  478. module_platform_driver(q6v5_wcss_driver);
  479. MODULE_DESCRIPTION("Hexagon WCSS Peripheral Image Loader");
  480. MODULE_LICENSE("GPL v2");