sunxi_mbus.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. // SPDX-License-Identifier: GPL-2.0
  2. /* Copyright (C) 2020 Maxime Ripard <[email protected]> */
  3. #include <linux/device.h>
  4. #include <linux/dma-map-ops.h>
  5. #include <linux/init.h>
  6. #include <linux/notifier.h>
  7. #include <linux/of.h>
  8. #include <linux/platform_device.h>
  9. static const char * const sunxi_mbus_devices[] = {
  10. /*
  11. * The display engine virtual devices are not strictly speaking
  12. * connected to the MBUS, but since DRM will perform all the
  13. * memory allocations and DMA operations through that device, we
  14. * need to have the quirk on those devices too.
  15. */
  16. "allwinner,sun4i-a10-display-engine",
  17. "allwinner,sun5i-a10s-display-engine",
  18. "allwinner,sun5i-a13-display-engine",
  19. "allwinner,sun6i-a31-display-engine",
  20. "allwinner,sun6i-a31s-display-engine",
  21. "allwinner,sun7i-a20-display-engine",
  22. "allwinner,sun8i-a23-display-engine",
  23. "allwinner,sun8i-a33-display-engine",
  24. "allwinner,sun9i-a80-display-engine",
  25. /*
  26. * And now we have the regular devices connected to the MBUS
  27. * (that we know of).
  28. */
  29. "allwinner,sun4i-a10-csi1",
  30. "allwinner,sun4i-a10-display-backend",
  31. "allwinner,sun4i-a10-display-frontend",
  32. "allwinner,sun4i-a10-video-engine",
  33. "allwinner,sun5i-a13-display-backend",
  34. "allwinner,sun5i-a13-video-engine",
  35. "allwinner,sun6i-a31-csi",
  36. "allwinner,sun6i-a31-display-backend",
  37. "allwinner,sun7i-a20-csi0",
  38. "allwinner,sun7i-a20-display-backend",
  39. "allwinner,sun7i-a20-display-frontend",
  40. "allwinner,sun7i-a20-video-engine",
  41. "allwinner,sun8i-a23-display-backend",
  42. "allwinner,sun8i-a23-display-frontend",
  43. "allwinner,sun8i-a33-display-backend",
  44. "allwinner,sun8i-a33-display-frontend",
  45. "allwinner,sun8i-a33-video-engine",
  46. "allwinner,sun8i-a83t-csi",
  47. "allwinner,sun8i-h3-csi",
  48. "allwinner,sun8i-h3-video-engine",
  49. "allwinner,sun8i-v3s-csi",
  50. "allwinner,sun9i-a80-display-backend",
  51. "allwinner,sun50i-a64-csi",
  52. "allwinner,sun50i-a64-video-engine",
  53. "allwinner,sun50i-h5-video-engine",
  54. NULL,
  55. };
  56. static int sunxi_mbus_notifier(struct notifier_block *nb,
  57. unsigned long event, void *__dev)
  58. {
  59. struct device *dev = __dev;
  60. int ret;
  61. if (event != BUS_NOTIFY_ADD_DEVICE)
  62. return NOTIFY_DONE;
  63. /*
  64. * Only the devices that need a large memory bandwidth do DMA
  65. * directly over the memory bus (called MBUS), instead of going
  66. * through the regular system bus.
  67. */
  68. if (!of_device_compatible_match(dev->of_node, sunxi_mbus_devices))
  69. return NOTIFY_DONE;
  70. /*
  71. * Devices with an interconnects property have the MBUS
  72. * relationship described in their DT and dealt with by
  73. * of_dma_configure, so we can just skip them.
  74. *
  75. * Older DTs or SoCs who are not clearly understood need to set
  76. * that DMA offset though.
  77. */
  78. if (of_find_property(dev->of_node, "interconnects", NULL))
  79. return NOTIFY_DONE;
  80. ret = dma_direct_set_offset(dev, PHYS_OFFSET, 0, SZ_4G);
  81. if (ret)
  82. dev_err(dev, "Couldn't setup our DMA offset: %d\n", ret);
  83. return NOTIFY_DONE;
  84. }
  85. static struct notifier_block sunxi_mbus_nb = {
  86. .notifier_call = sunxi_mbus_notifier,
  87. };
  88. static const char * const sunxi_mbus_platforms[] __initconst = {
  89. "allwinner,sun4i-a10",
  90. "allwinner,sun5i-a10s",
  91. "allwinner,sun5i-a13",
  92. "allwinner,sun6i-a31",
  93. "allwinner,sun7i-a20",
  94. "allwinner,sun8i-a23",
  95. "allwinner,sun8i-a33",
  96. "allwinner,sun8i-a83t",
  97. "allwinner,sun8i-h3",
  98. "allwinner,sun8i-r40",
  99. "allwinner,sun8i-v3",
  100. "allwinner,sun8i-v3s",
  101. "allwinner,sun9i-a80",
  102. "allwinner,sun50i-a64",
  103. "allwinner,sun50i-h5",
  104. "nextthing,gr8",
  105. NULL,
  106. };
  107. static int __init sunxi_mbus_init(void)
  108. {
  109. if (!of_device_compatible_match(of_root, sunxi_mbus_platforms))
  110. return 0;
  111. bus_register_notifier(&platform_bus_type, &sunxi_mbus_nb);
  112. return 0;
  113. }
  114. arch_initcall(sunxi_mbus_init);