The Memory Protection Unit(MPU) allows to partition memory into regions and set individual protection attributes for each region. In absence of MPU a default map[1] will take effect. Add support for configuring MPU on Cortex-R, by reusing the existing support for Cortex-M processor. [1] http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ddi0460d/I1002400.html Tested-by: Michal Simek <michal.simek@xilinx.com> Signed-off-by: Lokesh Vutla <lokeshvutla@ti.com> Reviewed-by: Tom Rini <trini@konsulko.com>lime2-spi
parent
4bbd6b1d94
commit
f2ef204312
@ -0,0 +1,108 @@ |
||||
// SPDX-License-Identifier: GPL-2.0+
|
||||
/*
|
||||
* Cortex-R Memory Protection Unit specific code |
||||
* |
||||
* Copyright (C) 2018 Texas Instruments Incorporated - http://www.ti.com/
|
||||
* Lokesh Vutla <lokeshvutla@ti.com> |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <command.h> |
||||
#include <asm/armv7.h> |
||||
#include <asm/system.h> |
||||
#include <asm/barriers.h> |
||||
#include <linux/compiler.h> |
||||
|
||||
#include <asm/armv7_mpu.h> |
||||
|
||||
/* MPU Type register definitions */ |
||||
#define MPUIR_S_SHIFT 0 |
||||
#define MPUIR_S_MASK BIT(MPUIR_S_SHIFT) |
||||
#define MPUIR_DREGION_SHIFT 8 |
||||
#define MPUIR_DREGION_MASK (0xff << 8) |
||||
|
||||
/**
|
||||
* Note: |
||||
* The Memory Protection Unit(MPU) allows to partition memory into regions |
||||
* and set individual protection attributes for each region. In absence |
||||
* of MPU a default map[1] will take effect. make sure to run this code |
||||
* from a region which has execution permissions by default. |
||||
* [1] http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.ddi0460d/I1002400.html
|
||||
*/ |
||||
|
||||
void disable_mpu(void) |
||||
{ |
||||
u32 reg; |
||||
|
||||
reg = get_cr(); |
||||
reg &= ~CR_M; |
||||
dsb(); |
||||
set_cr(reg); |
||||
isb(); |
||||
} |
||||
|
||||
void enable_mpu(void) |
||||
{ |
||||
u32 reg; |
||||
|
||||
reg = get_cr(); |
||||
reg |= CR_M; |
||||
dsb(); |
||||
set_cr(reg); |
||||
isb(); |
||||
} |
||||
|
||||
int mpu_enabled(void) |
||||
{ |
||||
return get_cr() & CR_M; |
||||
} |
||||
|
||||
void mpu_config(struct mpu_region_config *rgn) |
||||
{ |
||||
u32 attr, val; |
||||
|
||||
attr = get_attr_encoding(rgn->mr_attr); |
||||
|
||||
/* MPU Region Number Register */ |
||||
asm volatile ("mcr p15, 0, %0, c6, c2, 0" : : "r" (rgn->region_no)); |
||||
|
||||
/* MPU Region Base Address Register */ |
||||
asm volatile ("mcr p15, 0, %0, c6, c1, 0" : : "r" (rgn->start_addr)); |
||||
|
||||
/* MPU Region Size and Enable Register */ |
||||
if (rgn->reg_size) |
||||
val = (rgn->reg_size << REGION_SIZE_SHIFT) | ENABLE_REGION; |
||||
else |
||||
val = DISABLE_REGION; |
||||
asm volatile ("mcr p15, 0, %0, c6, c1, 2" : : "r" (val)); |
||||
|
||||
/* MPU Region Access Control Register */ |
||||
val = rgn->xn << XN_SHIFT | rgn->ap << AP_SHIFT | attr; |
||||
asm volatile ("mcr p15, 0, %0, c6, c1, 4" : : "r" (val)); |
||||
} |
||||
|
||||
void setup_mpu_regions(struct mpu_region_config *rgns, u32 num_rgns) |
||||
{ |
||||
u32 num, i; |
||||
|
||||
asm volatile ("mrc p15, 0, %0, c0, c0, 4" : "=r" (num)); |
||||
num = (num & MPUIR_DREGION_MASK) >> MPUIR_DREGION_SHIFT; |
||||
/* Regions to be configured cannot be greater than available regions */ |
||||
if (num < num_rgns) |
||||
num_rgns = num; |
||||
/**
|
||||
* Assuming dcache might not be enabled at this point, disabling |
||||
* and invalidating only icache. |
||||
*/ |
||||
icache_disable(); |
||||
invalidate_icache_all(); |
||||
|
||||
disable_mpu(); |
||||
|
||||
for (i = 0; i < num_rgns; i++) |
||||
mpu_config(&rgns[i]); |
||||
|
||||
enable_mpu(); |
||||
|
||||
icache_enable(); |
||||
} |
@ -0,0 +1,130 @@ |
||||
/* SPDX-License-Identifier: GPL-2.0+ */ |
||||
/*
|
||||
* Copyright (C) 2017, STMicroelectronics - All Rights Reserved |
||||
* Author(s): Vikas Manocha, <vikas.manocha@st.com> for STMicroelectronics. |
||||
*/ |
||||
|
||||
#ifndef _ASM_ARMV7_MPU_H |
||||
#define _ASM_ARMV7_MPU_H |
||||
|
||||
#ifdef CONFIG_CPU_V7M |
||||
#define AP_SHIFT 24 |
||||
#define XN_SHIFT 28 |
||||
#define TEX_SHIFT 19 |
||||
#define S_SHIFT 18 |
||||
#define C_SHIFT 17 |
||||
#define B_SHIFT 16 |
||||
#else /* CONFIG_CPU_V7R */ |
||||
#define XN_SHIFT 12 |
||||
#define AP_SHIFT 8 |
||||
#define TEX_SHIFT 3 |
||||
#define S_SHIFT 2 |
||||
#define C_SHIFT 1 |
||||
#define B_SHIFT 0 |
||||
#endif /* CONFIG_CPU_V7R */ |
||||
|
||||
#define CACHEABLE BIT(C_SHIFT) |
||||
#define BUFFERABLE BIT(B_SHIFT) |
||||
#define SHAREABLE BIT(S_SHIFT) |
||||
#define REGION_SIZE_SHIFT 1 |
||||
#define ENABLE_REGION BIT(0) |
||||
#define DISABLE_REGION 0 |
||||
|
||||
enum region_number { |
||||
REGION_0 = 0, |
||||
REGION_1, |
||||
REGION_2, |
||||
REGION_3, |
||||
REGION_4, |
||||
REGION_5, |
||||
REGION_6, |
||||
REGION_7, |
||||
}; |
||||
|
||||
enum ap { |
||||
NO_ACCESS = 0, |
||||
PRIV_RW_USR_NO, |
||||
PRIV_RW_USR_RO, |
||||
PRIV_RW_USR_RW, |
||||
UNPREDICTABLE, |
||||
PRIV_RO_USR_NO, |
||||
PRIV_RO_USR_RO, |
||||
}; |
||||
|
||||
enum mr_attr { |
||||
STRONG_ORDER = 0, |
||||
SHARED_WRITE_BUFFERED, |
||||
O_I_WT_NO_WR_ALLOC, |
||||
O_I_WB_NO_WR_ALLOC, |
||||
O_I_NON_CACHEABLE, |
||||
O_I_WB_RD_WR_ALLOC, |
||||
DEVICE_NON_SHARED, |
||||
}; |
||||
enum size { |
||||
REGION_8MB = 22, |
||||
REGION_16MB, |
||||
REGION_32MB, |
||||
REGION_64MB, |
||||
REGION_128MB, |
||||
REGION_256MB, |
||||
REGION_512MB, |
||||
REGION_1GB, |
||||
REGION_2GB, |
||||
REGION_4GB, |
||||
}; |
||||
|
||||
enum xn { |
||||
XN_DIS = 0, |
||||
XN_EN, |
||||
}; |
||||
|
||||
struct mpu_region_config { |
||||
uint32_t start_addr; |
||||
enum region_number region_no; |
||||
enum xn xn; |
||||
enum ap ap; |
||||
enum mr_attr mr_attr; |
||||
enum size reg_size; |
||||
}; |
||||
|
||||
void disable_mpu(void); |
||||
void enable_mpu(void); |
||||
int mpu_enabled(void); |
||||
void mpu_config(struct mpu_region_config *reg_config); |
||||
void setup_mpu_regions(struct mpu_region_config *rgns, u32 num_rgns); |
||||
|
||||
static inline u32 get_attr_encoding(u32 mr_attr) |
||||
{ |
||||
u32 attr; |
||||
|
||||
switch (mr_attr) { |
||||
case STRONG_ORDER: |
||||
attr = SHAREABLE; |
||||
break; |
||||
case SHARED_WRITE_BUFFERED: |
||||
attr = BUFFERABLE; |
||||
break; |
||||
case O_I_WT_NO_WR_ALLOC: |
||||
attr = CACHEABLE; |
||||
break; |
||||
case O_I_WB_NO_WR_ALLOC: |
||||
attr = CACHEABLE | BUFFERABLE; |
||||
break; |
||||
case O_I_NON_CACHEABLE: |
||||
attr = 1 << TEX_SHIFT; |
||||
break; |
||||
case O_I_WB_RD_WR_ALLOC: |
||||
attr = (1 << TEX_SHIFT) | CACHEABLE | BUFFERABLE; |
||||
break; |
||||
case DEVICE_NON_SHARED: |
||||
attr = (2 << TEX_SHIFT) | BUFFERABLE; |
||||
break; |
||||
default: |
||||
attr = 0; /* strongly ordered */ |
||||
break; |
||||
}; |
||||
|
||||
return attr; |
||||
} |
||||
|
||||
#endif /* _ASM_ARMV7_MPU_H */ |
@ -1,66 +0,0 @@ |
||||
/* SPDX-License-Identifier: GPL-2.0+ */ |
||||
/*
|
||||
* Copyright (C) 2017, STMicroelectronics - All Rights Reserved |
||||
* Author(s): Vikas Manocha, <vikas.manocha@st.com> for STMicroelectronics. |
||||
*/ |
||||
|
||||
enum region_number { |
||||
REGION_0 = 0, |
||||
REGION_1, |
||||
REGION_2, |
||||
REGION_3, |
||||
REGION_4, |
||||
REGION_5, |
||||
REGION_6, |
||||
REGION_7, |
||||
}; |
||||
|
||||
enum ap { |
||||
NO_ACCESS = 0, |
||||
PRIV_RW_USR_NO, |
||||
PRIV_RW_USR_RO, |
||||
PRIV_RW_USR_RW, |
||||
UNPREDICTABLE, |
||||
PRIV_RO_USR_NO, |
||||
PRIV_RO_USR_RO, |
||||
}; |
||||
|
||||
enum mr_attr { |
||||
STRONG_ORDER = 0, |
||||
SHARED_WRITE_BUFFERED, |
||||
O_I_WT_NO_WR_ALLOC, |
||||
O_I_WB_NO_WR_ALLOC, |
||||
O_I_NON_CACHEABLE, |
||||
O_I_WB_RD_WR_ALLOC, |
||||
DEVICE_NON_SHARED, |
||||
}; |
||||
enum size { |
||||
REGION_8MB = 22, |
||||
REGION_16MB, |
||||
REGION_32MB, |
||||
REGION_64MB, |
||||
REGION_128MB, |
||||
REGION_256MB, |
||||
REGION_512MB, |
||||
REGION_1GB, |
||||
REGION_2GB, |
||||
REGION_4GB, |
||||
}; |
||||
|
||||
enum xn { |
||||
XN_DIS = 0, |
||||
XN_EN, |
||||
}; |
||||
|
||||
struct mpu_region_config { |
||||
uint32_t start_addr; |
||||
enum region_number region_no; |
||||
enum xn xn; |
||||
enum ap ap; |
||||
enum mr_attr mr_attr; |
||||
enum size reg_size; |
||||
}; |
||||
|
||||
void disable_mpu(void); |
||||
void enable_mpu(void); |
||||
void mpu_config(struct mpu_region_config *reg_config); |
Loading…
Reference in new issue