1 //*****************************************************************************
2 //
3 //! @file am_hal_mpu.c
4 //!
5 //! @brief Hardware abstraction for the Memory Protection Unit.
6 //!
7 //! @addtogroup mpu_4p MPU - Memory Protection Unit
8 //! @ingroup apollo4p_hal
9 //! @{
10 //
11 //*****************************************************************************
12
13 //*****************************************************************************
14 //
15 // Copyright (c) 2023, Ambiq Micro, Inc.
16 // All rights reserved.
17 //
18 // Redistribution and use in source and binary forms, with or without
19 // modification, are permitted provided that the following conditions are met:
20 //
21 // 1. Redistributions of source code must retain the above copyright notice,
22 // this list of conditions and the following disclaimer.
23 //
24 // 2. Redistributions in binary form must reproduce the above copyright
25 // notice, this list of conditions and the following disclaimer in the
26 // documentation and/or other materials provided with the distribution.
27 //
28 // 3. Neither the name of the copyright holder nor the names of its
29 // contributors may be used to endorse or promote products derived from this
30 // software without specific prior written permission.
31 //
32 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
33 // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
34 // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
35 // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
36 // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
37 // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
38 // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
39 // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
40 // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
41 // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
42 // POSSIBILITY OF SUCH DAMAGE.
43 //
44 // This is part of revision release_sdk_4_4_0-3c5977e664 of the AmbiqSuite Development Package.
45 //
46 //*****************************************************************************
47
48 #include "am_hal_mpu.h"
49
50 //*****************************************************************************
51 //
52 // Returns the contents of the MPU_TYPE register
53 //
54 //*****************************************************************************
55 uint32_t
am_hal_mpu_type_get(uint32_t * pui32Type)56 am_hal_mpu_type_get(uint32_t *pui32Type)
57 {
58 *pui32Type = MPU->TYPE ;
59
60 return AM_HAL_STATUS_SUCCESS;
61 }
62
63 //*****************************************************************************
64 //
65 // Sets the global configuration of the MPU
66 //
67 //*****************************************************************************
68 uint32_t
am_hal_mpu_global_configure(bool bMPUEnable,bool bPrivilegedDefault,bool bFaultNMIProtect)69 am_hal_mpu_global_configure(bool bMPUEnable,
70 bool bPrivilegedDefault,
71 bool bFaultNMIProtect)
72 {
73 __DMB();
74
75 MPU->CTRL = _VAL2FLD(MPU_CTRL_ENABLE, (uint32_t)bMPUEnable ) |
76 _VAL2FLD(MPU_CTRL_HFNMIENA, (uint32_t)bFaultNMIProtect ) |
77 _VAL2FLD(MPU_CTRL_PRIVDEFENA, (uint32_t)bPrivilegedDefault ) ;
78
79
80 SCB->SHCSR |= SCB_SHCSR_MEMFAULTENA_Msk;
81
82 __DSB();
83 __ISB();
84
85 return AM_HAL_STATUS_SUCCESS;
86 }
87
88 //*****************************************************************************
89 //
90 // Configures an MPU region.
91 //
92 //*****************************************************************************
93 uint32_t
am_hal_mpu_region_configure(tMPURegion * psConfig,bool bEnableNow)94 am_hal_mpu_region_configure(tMPURegion *psConfig, bool bEnableNow)
95 {
96
97 #ifndef AM_HAL_DISABLE_API_VALIDATION
98
99 if (psConfig->ui8RegionNumber > _FLD2VAL( MPU_RBAR_REGION , 0xFFFFFFFF ) ) //(MPU_RBAR_REGION_Msk >>MPU_RBAR_REGION_Pos) )
100 {
101 return AM_HAL_STATUS_INVALID_ARG;
102 }
103 if (psConfig->eAccessPermission > _FLD2VAL( MPU_RASR_AP, 0xFFFFFFFF)) // (MPU_RASR_AP_Msk >> MPU_RASR_AP_Pos))
104 {
105 return AM_HAL_STATUS_INVALID_ARG;
106 }
107 if (psConfig->ui16SubRegionDisable > _FLD2VAL(MPU_RASR_SRD, 0xFFFFFFFF)) // (MPU_RASR_SRD_Msk >> MPU_RASR_SRD_Pos))
108 {
109 return AM_HAL_STATUS_INVALID_ARG;
110 }
111 if (psConfig->ui8Size > _FLD2VAL(MPU_RASR_SIZE, 0xFFFFFFFF)) // (MPU_RASR_SIZE_Msk >> MPU_RASR_SIZE_Pos))
112 {
113 return AM_HAL_STATUS_INVALID_ARG;
114 }
115
116 #endif // AM_HAL_DISABLE_API_VALIDATION
117
118 //
119 // Set the new base address for the specified region.
120 //
121 MPU->RBAR = (psConfig->ui32BaseAddress & MPU_RBAR_ADDR_Msk) |
122 _VAL2FLD(MPU_RBAR_REGION, psConfig->ui8RegionNumber ) |
123 _VAL2FLD(MPU_RBAR_VALID, 1 ) ;
124
125 //
126 // Set the attributes for this region based on the input structure.
127 //
128 MPU->RASR = _VAL2FLD(MPU_RASR_XN, (uint32_t) psConfig->bExecuteNever ) |
129 _VAL2FLD(MPU_RASR_AP, psConfig->eAccessPermission) |
130 _VAL2FLD(MPU_RASR_SRD, psConfig->ui16SubRegionDisable) |
131 _VAL2FLD(MPU_RASR_SIZE, psConfig->ui8Size ) |
132 _VAL2FLD(MPU_RASR_ENABLE, (uint32_t) bEnableNow ) |
133 MPU_DEFAULT_TEXSCB;
134
135 return AM_HAL_STATUS_SUCCESS;
136 }
137
138 //*****************************************************************************
139 //
140 // Enable an MPU region.
141 //
142 //*****************************************************************************
143 uint32_t
am_hal_mpu_region_enable(uint8_t ui8RegionNumber)144 am_hal_mpu_region_enable(uint8_t ui8RegionNumber)
145 {
146 #ifndef AM_HAL_DISABLE_API_VALIDATION
147 if (ui8RegionNumber > _FLD2VAL(MPU_RBAR_REGION, 0xFFFFFFFF))
148 {
149 return AM_HAL_STATUS_INVALID_ARG;
150 }
151 #endif
152 //
153 // Set the region number in the MPU_RNR register, and set the enable bit.
154 //
155
156 MPU->RNR = _VAL2FLD(MPU_RBAR_REGION, ui8RegionNumber);
157 MPU->RASR |= MPU_RASR_ENABLE_Msk;
158
159 return AM_HAL_STATUS_SUCCESS;
160 }
161
162 //*****************************************************************************
163 //
164 // Disable an MPU region.
165 //
166 //*****************************************************************************
167 uint32_t
am_hal_mpu_region_disable(uint8_t ui8RegionNumber)168 am_hal_mpu_region_disable(uint8_t ui8RegionNumber)
169 {
170 #ifndef AM_HAL_DISABLE_API_VALIDATION
171 if (ui8RegionNumber > _FLD2VAL(MPU_RBAR_REGION, 0xFFFFFFFF))
172 {
173 return AM_HAL_STATUS_INVALID_ARG;
174 }
175 #endif
176 //
177 // Set the region number in the MPU_RNR register, and clear the enable bit.
178 //
179 MPU->RNR = _VAL2FLD(MPU_RBAR_REGION, ui8RegionNumber);
180 MPU->RASR &= ~(MPU_RASR_ENABLE_Msk);
181
182 return AM_HAL_STATUS_SUCCESS;
183 }
184
185 //*****************************************************************************
186 //
187 // Get the MPU region number.
188 //
189 //*****************************************************************************
190 uint32_t
am_hal_mpu_get_region_number(uint32_t * ui32pMpuRNR)191 am_hal_mpu_get_region_number(uint32_t *ui32pMpuRNR)
192 {
193 *ui32pMpuRNR = _FLD2VAL( MPU_RBAR_REGION, MPU->RNR);
194
195 return AM_HAL_STATUS_SUCCESS;
196 }
197
198 //*****************************************************************************
199 //
200 // End Doxygen group.
201 //! @}
202 //
203 //*****************************************************************************
204
205