lima_l2_cache.c 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. // SPDX-License-Identifier: GPL-2.0 OR MIT
  2. /* Copyright 2017-2019 Qiang Yu <yuq825@gmail.com> */
  3. #include <linux/iopoll.h>
  4. #include <linux/device.h>
  5. #include "lima_device.h"
  6. #include "lima_l2_cache.h"
  7. #include "lima_regs.h"
  8. #define l2_cache_write(reg, data) writel(data, ip->iomem + reg)
  9. #define l2_cache_read(reg) readl(ip->iomem + reg)
  10. static int lima_l2_cache_wait_idle(struct lima_ip *ip)
  11. {
  12. struct lima_device *dev = ip->dev;
  13. int err;
  14. u32 v;
  15. err = readl_poll_timeout(ip->iomem + LIMA_L2_CACHE_STATUS, v,
  16. !(v & LIMA_L2_CACHE_STATUS_COMMAND_BUSY),
  17. 0, 1000);
  18. if (err) {
  19. dev_err(dev->dev, "%s wait command timeout\n",
  20. lima_ip_name(ip));
  21. return err;
  22. }
  23. return 0;
  24. }
  25. int lima_l2_cache_flush(struct lima_ip *ip)
  26. {
  27. int ret;
  28. spin_lock(&ip->data.lock);
  29. l2_cache_write(LIMA_L2_CACHE_COMMAND, LIMA_L2_CACHE_COMMAND_CLEAR_ALL);
  30. ret = lima_l2_cache_wait_idle(ip);
  31. spin_unlock(&ip->data.lock);
  32. return ret;
  33. }
  34. static int lima_l2_cache_hw_init(struct lima_ip *ip)
  35. {
  36. int err;
  37. err = lima_l2_cache_flush(ip);
  38. if (err)
  39. return err;
  40. l2_cache_write(LIMA_L2_CACHE_ENABLE,
  41. LIMA_L2_CACHE_ENABLE_ACCESS |
  42. LIMA_L2_CACHE_ENABLE_READ_ALLOCATE);
  43. l2_cache_write(LIMA_L2_CACHE_MAX_READS, 0x1c);
  44. return 0;
  45. }
  46. int lima_l2_cache_resume(struct lima_ip *ip)
  47. {
  48. return lima_l2_cache_hw_init(ip);
  49. }
  50. void lima_l2_cache_suspend(struct lima_ip *ip)
  51. {
  52. }
  53. int lima_l2_cache_init(struct lima_ip *ip)
  54. {
  55. int i;
  56. u32 size;
  57. struct lima_device *dev = ip->dev;
  58. /* l2_cache2 only exists when one of PP4-7 present */
  59. if (ip->id == lima_ip_l2_cache2) {
  60. for (i = lima_ip_pp4; i <= lima_ip_pp7; i++) {
  61. if (dev->ip[i].present)
  62. break;
  63. }
  64. if (i > lima_ip_pp7)
  65. return -ENODEV;
  66. }
  67. spin_lock_init(&ip->data.lock);
  68. size = l2_cache_read(LIMA_L2_CACHE_SIZE);
  69. dev_info(dev->dev, "%s %uK, %u-way, %ubyte cache line, %ubit external bus\n",
  70. lima_ip_name(ip),
  71. 1 << (((size >> 16) & 0xff) - 10),
  72. 1 << ((size >> 8) & 0xff),
  73. 1 << (size & 0xff),
  74. 1 << ((size >> 24) & 0xff));
  75. return lima_l2_cache_hw_init(ip);
  76. }
  77. void lima_l2_cache_fini(struct lima_ip *ip)
  78. {
  79. }