kgsl_regmap.c 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2020-2021, The Linux Foundation. All rights reserved.
  4. * Copyright (c) 2022-2023, Qualcomm Innovation Center, Inc. All rights reserved.
  5. */
  6. #include <linux/io.h>
  7. #include <linux/platform_device.h>
  8. #include "kgsl_regmap.h"
  9. #include "kgsl_trace.h"
  10. #define region_addr(region, _offset) \
  11. ((region)->virt + (((_offset) - (region)->offset) << 2))
  12. static int kgsl_regmap_init_region(struct kgsl_regmap *regmap,
  13. struct platform_device *pdev,
  14. struct kgsl_regmap_region *region,
  15. struct resource *res, const struct kgsl_regmap_ops *ops,
  16. void *priv)
  17. {
  18. void __iomem *ptr;
  19. ptr = devm_ioremap(&pdev->dev, res->start, resource_size(res));
  20. if (!ptr)
  21. return -ENOMEM;
  22. region->virt = ptr;
  23. region->offset = (res->start - regmap->base->start) >> 2;
  24. region->size = resource_size(res) >> 2;
  25. region->ops = ops;
  26. region->priv = priv;
  27. return 0;
  28. }
  29. /* Initialize the regmap with the base region. All added regions will be offset
  30. * from this base
  31. */
  32. int kgsl_regmap_init(struct platform_device *pdev, struct kgsl_regmap *regmap,
  33. const char *name, const struct kgsl_regmap_ops *ops,
  34. void *priv)
  35. {
  36. struct kgsl_regmap_region *region;
  37. struct resource *res;
  38. int ret;
  39. res = platform_get_resource_byname(pdev, IORESOURCE_MEM, name);
  40. if (!res)
  41. return -ENODEV;
  42. regmap->base = res;
  43. region = &regmap->region[0];
  44. ret = kgsl_regmap_init_region(regmap, pdev, region, res, ops, priv);
  45. if (!ret)
  46. regmap->count = 1;
  47. return ret;
  48. }
  49. /* Add a new region to the regmap */
  50. int kgsl_regmap_add_region(struct kgsl_regmap *regmap, struct platform_device *pdev,
  51. const char *name, const struct kgsl_regmap_ops *ops, void *priv)
  52. {
  53. struct kgsl_regmap_region *region;
  54. struct resource *res;
  55. int ret;
  56. res = platform_get_resource_byname(pdev, IORESOURCE_MEM, name);
  57. if (!res)
  58. return -ENODEV;
  59. if (WARN_ON(regmap->count >= ARRAY_SIZE(regmap->region)))
  60. return -ENODEV;
  61. region = &regmap->region[regmap->count];
  62. ret = kgsl_regmap_init_region(regmap, pdev, region, res, ops, priv);
  63. if (!ret)
  64. regmap->count++;
  65. return ret;
  66. }
  67. #define in_range(a, base, len) \
  68. (((a) >= (base)) && ((a) < ((base) + (len))))
  69. struct kgsl_regmap_region *kgsl_regmap_get_region(struct kgsl_regmap *regmap,
  70. u32 offset)
  71. {
  72. int i;
  73. for (i = 0; i < regmap->count; i++) {
  74. struct kgsl_regmap_region *region = &regmap->region[i];
  75. if (in_range(offset, region->offset, region->size))
  76. return region;
  77. }
  78. return NULL;
  79. }
  80. bool kgsl_regmap_valid_offset(struct kgsl_regmap *regmap, u32 offset)
  81. {
  82. if (kgsl_regmap_get_region(regmap, offset))
  83. return true;
  84. return false;
  85. }
  86. u32 kgsl_regmap_read(struct kgsl_regmap *regmap, u32 offset)
  87. {
  88. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, offset);
  89. u32 val;
  90. if (WARN(!region, "Out of bounds register read offset: 0x%x\n", offset))
  91. return 0;
  92. if (region->ops && region->ops->preaccess)
  93. region->ops->preaccess(region);
  94. val = readl_relaxed(region_addr(region, offset));
  95. /* Allow previous read to post before returning the value */
  96. rmb();
  97. return val;
  98. }
  99. void kgsl_regmap_write(struct kgsl_regmap *regmap, u32 value, u32 offset)
  100. {
  101. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, offset);
  102. if (WARN(!region, "Out of bounds register write offset: 0x%x\n", offset))
  103. return;
  104. if (region->ops && region->ops->preaccess)
  105. region->ops->preaccess(region);
  106. /* Make sure all pending writes have posted first */
  107. wmb();
  108. writel_relaxed(value, region_addr(region, offset));
  109. trace_kgsl_regwrite(offset, value);
  110. }
  111. void kgsl_regmap_multi_write(struct kgsl_regmap *regmap,
  112. const struct kgsl_regmap_list *list, int count)
  113. {
  114. struct kgsl_regmap_region *region, *prev = NULL;
  115. int i;
  116. /*
  117. * do one write barrier to ensure all previous writes are done before
  118. * starting the list
  119. */
  120. wmb();
  121. for (i = 0; i < count; i++) {
  122. region = kgsl_regmap_get_region(regmap, list[i].offset);
  123. if (WARN(!region, "Out of bounds register write offset: 0x%x\n",
  124. list[i].offset))
  125. continue;
  126. /*
  127. * The registers might be in different regions. If a region has
  128. * a preaccess function we need to call it at least once before
  129. * writing registers but we don't want to call it every time if
  130. * we can avoid it. "cache" the current region and don't call
  131. * pre-access if it is the same region from the previous access.
  132. * This isn't perfect but it should cut down on some unneeded
  133. * cpu cycles
  134. */
  135. if (region != prev && region->ops && region->ops->preaccess)
  136. region->ops->preaccess(region);
  137. prev = region;
  138. writel_relaxed(list[i].val, region_addr(region, list[i].offset));
  139. trace_kgsl_regwrite(list[i].offset, list[i].val);
  140. }
  141. }
  142. void kgsl_regmap_rmw(struct kgsl_regmap *regmap, u32 offset, u32 mask,
  143. u32 or)
  144. {
  145. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, offset);
  146. u32 val;
  147. if (WARN(!region, "Out of bounds register read-modify-write offset: 0x%x\n",
  148. offset))
  149. return;
  150. if (region->ops && region->ops->preaccess)
  151. region->ops->preaccess(region);
  152. val = readl_relaxed(region_addr(region, offset));
  153. /* Make sure the read posted and all pending writes are done */
  154. mb();
  155. writel_relaxed((val & ~mask) | or, region_addr(region, offset));
  156. trace_kgsl_regwrite(offset, (val & ~mask) | or);
  157. }
  158. void kgsl_regmap_bulk_write(struct kgsl_regmap *regmap, u32 offset,
  159. const void *data, int dwords)
  160. {
  161. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, offset);
  162. if (WARN(!region, "Out of bounds register bulk write offset: 0x%x\n", offset))
  163. return;
  164. if (region->ops && region->ops->preaccess)
  165. region->ops->preaccess(region);
  166. /*
  167. * A bulk write operation can only be in one region - it cannot
  168. * cross boundaries
  169. */
  170. if (WARN((offset - region->offset) + dwords > region->size,
  171. "OUt of bounds bulk write size: 0x%x\n", offset + dwords))
  172. return;
  173. /* Make sure all pending write are done first */
  174. wmb();
  175. memcpy_toio(region_addr(region, offset), data, dwords << 2);
  176. }
  177. void kgsl_regmap_bulk_read(struct kgsl_regmap *regmap, u32 offset,
  178. const void *data, int dwords)
  179. {
  180. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, offset);
  181. if (WARN(!region, "Out of bounds register bulk read offset: 0x%x\n", offset))
  182. return;
  183. if (region->ops && region->ops->preaccess)
  184. region->ops->preaccess(region);
  185. /*
  186. * A bulk read operation can only be in one region - it cannot
  187. * cross boundaries
  188. */
  189. if (WARN((offset - region->offset) + dwords > region->size,
  190. "Out of bounds bulk read size: 0x%x\n", offset + dwords))
  191. return;
  192. memcpy_fromio(region_addr(region, offset), data, dwords << 2);
  193. /* Make sure the copy is finished before moving on */
  194. rmb();
  195. }
  196. void __iomem *kgsl_regmap_virt(struct kgsl_regmap *regmap, u32 offset)
  197. {
  198. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, offset);
  199. if (region)
  200. return region_addr(region, offset);
  201. return NULL;
  202. }
  203. void kgsl_regmap_read_indexed(struct kgsl_regmap *regmap, u32 addr,
  204. u32 data, u32 *dest, int count)
  205. {
  206. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, addr);
  207. int i;
  208. if (!region)
  209. return;
  210. /* Make sure the offset is in the same region */
  211. if (kgsl_regmap_get_region(regmap, data) != region)
  212. return;
  213. if (region->ops && region->ops->preaccess)
  214. region->ops->preaccess(region);
  215. /* Write the address register */
  216. writel_relaxed(0, region_addr(region, addr));
  217. /* Make sure the write finishes */
  218. wmb();
  219. for (i = 0; i < count; i++)
  220. dest[i] = readl_relaxed(region_addr(region, data));
  221. /* Do one barrier at the end to make sure all the data is posted */
  222. rmb();
  223. }
  224. void kgsl_regmap_read_indexed_interleaved(struct kgsl_regmap *regmap, u32 addr,
  225. u32 data, u32 *dest, u32 start, int count)
  226. {
  227. struct kgsl_regmap_region *region = kgsl_regmap_get_region(regmap, addr);
  228. int i;
  229. if (!region)
  230. return;
  231. /* Make sure the offset is in the same region */
  232. if (kgsl_regmap_get_region(regmap, data) != region)
  233. return;
  234. if (region->ops && region->ops->preaccess)
  235. region->ops->preaccess(region);
  236. for (i = 0; i < count; i++) {
  237. /* Write the address register */
  238. writel_relaxed(start + i, region_addr(region, addr));
  239. /* Make sure the write finishes */
  240. wmb();
  241. dest[i] = readl_relaxed(region_addr(region, data));
  242. /* Make sure the read finishes */
  243. rmb();
  244. }
  245. }
  246. /* A special helper function to work with read_poll_timeout */
  247. int kgsl_regmap_poll_read(struct kgsl_regmap_region *region, u32 offset,
  248. u32 *val)
  249. {
  250. /* FIXME: WARN on !region? */
  251. if (WARN(!region, "Out of bounds poll read: 0x%x\n", offset))
  252. return -ENODEV;
  253. *val = readl_relaxed(region_addr(region, offset));
  254. /* Make sure the read is finished before moving on */
  255. rmb();
  256. return 0;
  257. }