qcom-apcs-ipc-mailbox.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2017, Linaro Ltd
  4. */
  5. #include <linux/kernel.h>
  6. #include <linux/module.h>
  7. #include <linux/io.h>
  8. #include <linux/slab.h>
  9. #include <linux/of.h>
  10. #include <linux/of_platform.h>
  11. #include <linux/platform_device.h>
  12. #include <linux/regmap.h>
  13. #include <linux/mailbox_controller.h>
  14. #define QCOM_APCS_IPC_BITS 32
  15. struct qcom_apcs_ipc {
  16. struct mbox_controller mbox;
  17. struct mbox_chan mbox_chans[QCOM_APCS_IPC_BITS];
  18. struct regmap *regmap;
  19. unsigned long offset;
  20. struct platform_device *clk;
  21. };
  22. struct qcom_apcs_ipc_data {
  23. int offset;
  24. char *clk_name;
  25. };
  26. static const struct qcom_apcs_ipc_data ipq6018_apcs_data = {
  27. .offset = 8, .clk_name = "qcom,apss-ipq6018-clk"
  28. };
  29. static const struct qcom_apcs_ipc_data msm8916_apcs_data = {
  30. .offset = 8, .clk_name = "qcom-apcs-msm8916-clk"
  31. };
  32. static const struct qcom_apcs_ipc_data msm8994_apcs_data = {
  33. .offset = 8, .clk_name = NULL
  34. };
  35. static const struct qcom_apcs_ipc_data msm8996_apcs_data = {
  36. .offset = 16, .clk_name = NULL
  37. };
  38. static const struct qcom_apcs_ipc_data apps_shared_apcs_data = {
  39. .offset = 12, .clk_name = NULL
  40. };
  41. static const struct qcom_apcs_ipc_data sdx55_apcs_data = {
  42. .offset = 0x1008, .clk_name = "qcom-sdx55-acps-clk"
  43. };
  44. static const struct regmap_config apcs_regmap_config = {
  45. .reg_bits = 32,
  46. .reg_stride = 4,
  47. .val_bits = 32,
  48. .max_register = 0x1008,
  49. .fast_io = true,
  50. };
  51. static int qcom_apcs_ipc_send_data(struct mbox_chan *chan, void *data)
  52. {
  53. struct qcom_apcs_ipc *apcs = container_of(chan->mbox,
  54. struct qcom_apcs_ipc, mbox);
  55. unsigned long idx = (unsigned long)chan->con_priv;
  56. return regmap_write(apcs->regmap, apcs->offset, BIT(idx));
  57. }
  58. static const struct mbox_chan_ops qcom_apcs_ipc_ops = {
  59. .send_data = qcom_apcs_ipc_send_data,
  60. };
  61. static int qcom_apcs_ipc_probe(struct platform_device *pdev)
  62. {
  63. struct qcom_apcs_ipc *apcs;
  64. const struct qcom_apcs_ipc_data *apcs_data;
  65. struct regmap *regmap;
  66. void __iomem *base;
  67. unsigned long i;
  68. int ret;
  69. apcs = devm_kzalloc(&pdev->dev, sizeof(*apcs), GFP_KERNEL);
  70. if (!apcs)
  71. return -ENOMEM;
  72. base = devm_platform_ioremap_resource(pdev, 0);
  73. if (IS_ERR(base))
  74. return PTR_ERR(base);
  75. regmap = devm_regmap_init_mmio(&pdev->dev, base, &apcs_regmap_config);
  76. if (IS_ERR(regmap))
  77. return PTR_ERR(regmap);
  78. apcs_data = of_device_get_match_data(&pdev->dev);
  79. apcs->regmap = regmap;
  80. apcs->offset = apcs_data->offset;
  81. /* Initialize channel identifiers */
  82. for (i = 0; i < ARRAY_SIZE(apcs->mbox_chans); i++)
  83. apcs->mbox_chans[i].con_priv = (void *)i;
  84. apcs->mbox.dev = &pdev->dev;
  85. apcs->mbox.ops = &qcom_apcs_ipc_ops;
  86. apcs->mbox.chans = apcs->mbox_chans;
  87. apcs->mbox.num_chans = ARRAY_SIZE(apcs->mbox_chans);
  88. ret = devm_mbox_controller_register(&pdev->dev, &apcs->mbox);
  89. if (ret) {
  90. dev_err(&pdev->dev, "failed to register APCS IPC controller\n");
  91. return ret;
  92. }
  93. if (apcs_data->clk_name) {
  94. apcs->clk = platform_device_register_data(&pdev->dev,
  95. apcs_data->clk_name,
  96. PLATFORM_DEVID_AUTO,
  97. NULL, 0);
  98. if (IS_ERR(apcs->clk))
  99. dev_err(&pdev->dev, "failed to register APCS clk\n");
  100. }
  101. platform_set_drvdata(pdev, apcs);
  102. return 0;
  103. }
  104. static int qcom_apcs_ipc_remove(struct platform_device *pdev)
  105. {
  106. struct qcom_apcs_ipc *apcs = platform_get_drvdata(pdev);
  107. struct platform_device *clk = apcs->clk;
  108. platform_device_unregister(clk);
  109. return 0;
  110. }
  111. /* .data is the offset of the ipc register within the global block */
  112. static const struct of_device_id qcom_apcs_ipc_of_match[] = {
  113. { .compatible = "qcom,ipq6018-apcs-apps-global", .data = &ipq6018_apcs_data },
  114. { .compatible = "qcom,ipq8074-apcs-apps-global", .data = &ipq6018_apcs_data },
  115. { .compatible = "qcom,msm8916-apcs-kpss-global", .data = &msm8916_apcs_data },
  116. { .compatible = "qcom,msm8939-apcs-kpss-global", .data = &msm8916_apcs_data },
  117. { .compatible = "qcom,msm8953-apcs-kpss-global", .data = &msm8994_apcs_data },
  118. { .compatible = "qcom,msm8976-apcs-kpss-global", .data = &msm8994_apcs_data },
  119. { .compatible = "qcom,msm8994-apcs-kpss-global", .data = &msm8994_apcs_data },
  120. { .compatible = "qcom,msm8996-apcs-hmss-global", .data = &msm8996_apcs_data },
  121. { .compatible = "qcom,msm8998-apcs-hmss-global", .data = &msm8994_apcs_data },
  122. { .compatible = "qcom,qcm2290-apcs-hmss-global", .data = &msm8994_apcs_data },
  123. { .compatible = "qcom,qcs404-apcs-apps-global", .data = &msm8916_apcs_data },
  124. { .compatible = "qcom,sc7180-apss-shared", .data = &apps_shared_apcs_data },
  125. { .compatible = "qcom,sc8180x-apss-shared", .data = &apps_shared_apcs_data },
  126. { .compatible = "qcom,sdm660-apcs-hmss-global", .data = &msm8994_apcs_data },
  127. { .compatible = "qcom,sdm845-apss-shared", .data = &apps_shared_apcs_data },
  128. { .compatible = "qcom,sm6125-apcs-hmss-global", .data = &msm8994_apcs_data },
  129. { .compatible = "qcom,sm8150-apss-shared", .data = &apps_shared_apcs_data },
  130. { .compatible = "qcom,sm6115-apcs-hmss-global", .data = &msm8994_apcs_data },
  131. { .compatible = "qcom,sdx55-apcs-gcc", .data = &sdx55_apcs_data },
  132. {}
  133. };
  134. MODULE_DEVICE_TABLE(of, qcom_apcs_ipc_of_match);
  135. static struct platform_driver qcom_apcs_ipc_driver = {
  136. .probe = qcom_apcs_ipc_probe,
  137. .remove = qcom_apcs_ipc_remove,
  138. .driver = {
  139. .name = "qcom_apcs_ipc",
  140. .of_match_table = qcom_apcs_ipc_of_match,
  141. },
  142. };
  143. static int __init qcom_apcs_ipc_init(void)
  144. {
  145. return platform_driver_register(&qcom_apcs_ipc_driver);
  146. }
  147. postcore_initcall(qcom_apcs_ipc_init);
  148. static void __exit qcom_apcs_ipc_exit(void)
  149. {
  150. platform_driver_unregister(&qcom_apcs_ipc_driver);
  151. }
  152. module_exit(qcom_apcs_ipc_exit);
  153. MODULE_LICENSE("GPL v2");
  154. MODULE_DESCRIPTION("Qualcomm APCS IPC driver");