mpu_v7r.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Cortex-R Memory Protection Unit specific code
  4. *
  5. * Copyright (C) 2018 Texas Instruments Incorporated - http://www.ti.com/
  6. * Lokesh Vutla <lokeshvutla@ti.com>
  7. */
  8. #include <common.h>
  9. #include <command.h>
  10. #include <asm/armv7.h>
  11. #include <asm/system.h>
  12. #include <asm/barriers.h>
  13. #include <linux/compiler.h>
  14. #include <asm/armv7_mpu.h>
  15. /* MPU Type register definitions */
  16. #define MPUIR_S_SHIFT 0
  17. #define MPUIR_S_MASK BIT(MPUIR_S_SHIFT)
  18. #define MPUIR_DREGION_SHIFT 8
  19. #define MPUIR_DREGION_MASK (0xff << 8)
  20. /**
  21. * Note:
  22. * The Memory Protection Unit(MPU) allows to partition memory into regions
  23. * and set individual protection attributes for each region. In absence
  24. * of MPU a default map[1] will take effect. make sure to run this code
  25. * from a region which has execution permissions by default.
  26. * [1] http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ddi0460d/I1002400.html
  27. */
  28. void disable_mpu(void)
  29. {
  30. u32 reg;
  31. reg = get_cr();
  32. reg &= ~CR_M;
  33. dsb();
  34. set_cr(reg);
  35. isb();
  36. }
  37. void enable_mpu(void)
  38. {
  39. u32 reg;
  40. reg = get_cr();
  41. reg |= CR_M;
  42. dsb();
  43. set_cr(reg);
  44. isb();
  45. }
  46. int mpu_enabled(void)
  47. {
  48. return get_cr() & CR_M;
  49. }
  50. void mpu_config(struct mpu_region_config *rgn)
  51. {
  52. u32 attr, val;
  53. attr = get_attr_encoding(rgn->mr_attr);
  54. /* MPU Region Number Register */
  55. asm volatile ("mcr p15, 0, %0, c6, c2, 0" : : "r" (rgn->region_no));
  56. /* MPU Region Base Address Register */
  57. asm volatile ("mcr p15, 0, %0, c6, c1, 0" : : "r" (rgn->start_addr));
  58. /* MPU Region Size and Enable Register */
  59. if (rgn->reg_size)
  60. val = (rgn->reg_size << REGION_SIZE_SHIFT) | ENABLE_REGION;
  61. else
  62. val = DISABLE_REGION;
  63. asm volatile ("mcr p15, 0, %0, c6, c1, 2" : : "r" (val));
  64. /* MPU Region Access Control Register */
  65. val = rgn->xn << XN_SHIFT | rgn->ap << AP_SHIFT | attr;
  66. asm volatile ("mcr p15, 0, %0, c6, c1, 4" : : "r" (val));
  67. }
  68. void setup_mpu_regions(struct mpu_region_config *rgns, u32 num_rgns)
  69. {
  70. u32 num, i;
  71. asm volatile ("mrc p15, 0, %0, c0, c0, 4" : "=r" (num));
  72. num = (num & MPUIR_DREGION_MASK) >> MPUIR_DREGION_SHIFT;
  73. /* Regions to be configured cannot be greater than available regions */
  74. if (num < num_rgns)
  75. num_rgns = num;
  76. /**
  77. * Assuming dcache might not be enabled at this point, disabling
  78. * and invalidating only icache.
  79. */
  80. icache_disable();
  81. invalidate_icache_all();
  82. disable_mpu();
  83. for (i = 0; i < num_rgns; i++)
  84. mpu_config(&rgns[i]);
  85. enable_mpu();
  86. icache_enable();
  87. }
  88. void enable_caches(void)
  89. {
  90. /*
  91. * setup_mpu_regions() might have enabled Icache. So add a check
  92. * before enabling Icache
  93. */
  94. if (!icache_status())
  95. icache_enable();
  96. dcache_enable();
  97. }