1 /*
2 ** ###################################################################
3 ** Processors: LPC55S28JBD100
4 ** LPC55S28JBD64
5 ** LPC55S28JEV98
6 **
7 ** Compilers: GNU C Compiler
8 ** IAR ANSI C/C++ Compiler for ARM
9 ** Keil ARM C/C++ Compiler
10 ** MCUXpresso Compiler
11 **
12 ** Reference manual: LPC55S6x/LPC55S2x/LPC552x User manual(UM11126) Rev.1.3 16 May 2019
13 ** Version: rev. 1.1, 2019-05-16
14 ** Build: b220117
15 **
16 ** Abstract:
17 ** Provides a system configuration function and a global variable that
18 ** contains the system frequency. It configures the device and initializes
19 ** the oscillator (PLL) that is part of the microcontroller device.
20 **
21 ** Copyright 2016 Freescale Semiconductor, Inc.
22 ** Copyright 2016-2022 NXP
23 ** All rights reserved.
24 **
25 ** SPDX-License-Identifier: BSD-3-Clause
26 **
27 ** http: www.nxp.com
28 ** mail: support@nxp.com
29 **
30 ** Revisions:
31 ** - rev. 1.0 (2018-08-22)
32 ** Initial version based on v0.2UM
33 ** - rev. 1.1 (2019-05-16)
34 ** Initial A1 version based on v1.3UM
35 **
36 ** ###################################################################
37 */
38
39 /*!
40 * @file LPC55S28
41 * @version 1.1
42 * @date 2019-05-16
43 * @brief Device specific configuration file for LPC55S28 (implementation file)
44 *
45 * Provides a system configuration function and a global variable that contains
46 * the system frequency. It configures the device and initializes the oscillator
47 * (PLL) that is part of the microcontroller device.
48 */
49
50 #include <stdint.h>
51 #include "fsl_device_registers.h"
52
53 /* PLL0 SSCG control1 */
54 #define PLL_SSCG_MD_FRACT_P 0U
55 #define PLL_SSCG_MD_INT_P 25U
56 #define PLL_SSCG_MD_FRACT_M (0x1FFFFFFUL << PLL_SSCG_MD_FRACT_P)
57 #define PLL_SSCG_MD_INT_M ((uint64_t)0xFFUL << PLL_SSCG_MD_INT_P)
58
59 /* Get predivider (N) from PLL0 NDEC setting */
findPll0PreDiv(void)60 static uint32_t findPll0PreDiv(void)
61 {
62 uint32_t preDiv = 1UL;
63
64 /* Direct input is not used? */
65 if ((SYSCON->PLL0CTRL & SYSCON_PLL0CTRL_BYPASSPREDIV_MASK) == 0UL)
66 {
67 preDiv = SYSCON->PLL0NDEC & SYSCON_PLL0NDEC_NDIV_MASK;
68 if (preDiv == 0UL)
69 {
70 preDiv = 1UL;
71 }
72 }
73 return preDiv;
74 }
75
76 /* Get postdivider (P) from PLL0 PDEC setting */
findPll0PostDiv(void)77 static uint32_t findPll0PostDiv(void)
78 {
79 uint32_t postDiv = 1;
80
81 if ((SYSCON->PLL0CTRL & SYSCON_PLL0CTRL_BYPASSPOSTDIV_MASK) == 0UL)
82 {
83 if ((SYSCON->PLL0CTRL & SYSCON_PLL0CTRL_BYPASSPOSTDIV2_MASK) != 0UL)
84 {
85 postDiv = SYSCON->PLL0PDEC & SYSCON_PLL0PDEC_PDIV_MASK;
86 }
87 else
88 {
89 postDiv = 2UL * (SYSCON->PLL0PDEC & SYSCON_PLL0PDEC_PDIV_MASK);
90 }
91 if (postDiv == 0UL)
92 {
93 postDiv = 2UL;
94 }
95 }
96 return postDiv;
97 }
98
99 /* Get multiplier (M) from PLL0 SSCG and SEL_EXT settings */
findPll0MMult(void)100 static float findPll0MMult(void)
101 {
102 float mMult = 1.0F;
103 float mMult_fract;
104 uint32_t mMult_int;
105
106 if ((SYSCON->PLL0SSCG1 & SYSCON_PLL0SSCG1_SEL_EXT_MASK) != 0UL)
107 {
108 mMult = (float)(uint32_t)((SYSCON->PLL0SSCG1 & SYSCON_PLL0SSCG1_MDIV_EXT_MASK) >> SYSCON_PLL0SSCG1_MDIV_EXT_SHIFT);
109 }
110 else
111 {
112 mMult_int = ((SYSCON->PLL0SSCG1 & SYSCON_PLL0SSCG1_MD_MBS_MASK) << 7U);
113 mMult_int = mMult_int | ((SYSCON->PLL0SSCG0) >> PLL_SSCG_MD_INT_P);
114 mMult_fract = ((float)(uint32_t)((SYSCON->PLL0SSCG0) & PLL_SSCG_MD_FRACT_M) /
115 (float)(uint32_t)(1UL << PLL_SSCG_MD_INT_P));
116 mMult = (float)mMult_int + mMult_fract;
117 }
118 if (0ULL == ((uint64_t)mMult))
119 {
120 mMult = 1.0F;
121 }
122 return mMult;
123 }
124
125 /* Get predivider (N) from PLL1 NDEC setting */
findPll1PreDiv(void)126 static uint32_t findPll1PreDiv(void)
127 {
128 uint32_t preDiv = 1UL;
129
130 /* Direct input is not used? */
131 if ((SYSCON->PLL1CTRL & SYSCON_PLL1CTRL_BYPASSPREDIV_MASK) == 0UL)
132 {
133 preDiv = SYSCON->PLL1NDEC & SYSCON_PLL1NDEC_NDIV_MASK;
134 if (preDiv == 0UL)
135 {
136 preDiv = 1UL;
137 }
138 }
139 return preDiv;
140 }
141
142 /* Get postdivider (P) from PLL1 PDEC setting */
findPll1PostDiv(void)143 static uint32_t findPll1PostDiv(void)
144 {
145 uint32_t postDiv = 1UL;
146
147 if ((SYSCON->PLL1CTRL & SYSCON_PLL1CTRL_BYPASSPOSTDIV_MASK) == 0UL)
148 {
149 if ((SYSCON->PLL1CTRL & SYSCON_PLL1CTRL_BYPASSPOSTDIV2_MASK) != 0UL)
150 {
151 postDiv = SYSCON->PLL1PDEC & SYSCON_PLL1PDEC_PDIV_MASK;
152 }
153 else
154 {
155 postDiv = 2UL * (SYSCON->PLL1PDEC & SYSCON_PLL1PDEC_PDIV_MASK);
156 }
157 if (postDiv == 0UL)
158 {
159 postDiv = 2UL;
160 }
161 }
162 return postDiv;
163 }
164
165 /* Get multiplier (M) from PLL1 MDEC settings */
findPll1MMult(void)166 static uint32_t findPll1MMult(void)
167 {
168 uint32_t mMult = 1UL;
169
170 mMult = SYSCON->PLL1MDEC & SYSCON_PLL1MDEC_MDIV_MASK;
171
172 if (mMult == 0UL)
173 {
174 mMult = 1UL;
175 }
176 return mMult;
177 }
178
179 /* Get FRO 12M Clk */
180 /*! brief Return Frequency of FRO 12MHz
181 * return Frequency of FRO 12MHz
182 */
GetFro12MFreq(void)183 static uint32_t GetFro12MFreq(void)
184 {
185 return ((ANACTRL->FRO192M_CTRL & ANACTRL_FRO192M_CTRL_ENA_12MHZCLK_MASK) != 0UL) ? 12000000U : 0U;
186 }
187
188 /* Get FRO 1M Clk */
189 /*! brief Return Frequency of FRO 1MHz
190 * return Frequency of FRO 1MHz
191 */
GetFro1MFreq(void)192 static uint32_t GetFro1MFreq(void)
193 {
194 return ((SYSCON->CLOCK_CTRL & SYSCON_CLOCK_CTRL_FRO1MHZ_CLK_ENA_MASK) != 0UL) ? 1000000U : 0U;
195 }
196
197 /* Get EXT OSC Clk */
198 /*! brief Return Frequency of External Clock
199 * return Frequency of External Clock. If no external clock is used returns 0.
200 */
GetExtClkFreq(void)201 static uint32_t GetExtClkFreq(void)
202 {
203 return ((ANACTRL->XO32M_CTRL & ANACTRL_XO32M_CTRL_ENABLE_SYSTEM_CLK_OUT_MASK) != 0UL) ? CLK_CLK_IN : 0U;
204 }
205
206 /* Get HF FRO Clk */
207 /*! brief Return Frequency of High-Freq output of FRO
208 * return Frequency of High-Freq output of FRO
209 */
GetFroHfFreq(void)210 static uint32_t GetFroHfFreq(void)
211 {
212 return ((ANACTRL->FRO192M_CTRL & ANACTRL_FRO192M_CTRL_ENA_96MHZCLK_MASK) != 0UL) ? 96000000U : 0U;
213 }
214
215 /* Get RTC OSC Clk */
216 /*! brief Return Frequency of 32kHz osc
217 * return Frequency of 32kHz osc
218 */
GetOsc32KFreq(void)219 static uint32_t GetOsc32KFreq(void)
220 {
221 return ((0UL == (PMC->PDRUNCFG0 & PMC_PDRUNCFG0_PDEN_FRO32K_MASK)) && (0UL == (PMC->RTCOSC32K & PMC_RTCOSC32K_SEL_MASK))) ?
222 CLK_RTC_32K_CLK :
223 ((0UL == (PMC->PDRUNCFG0 & PMC_PDRUNCFG0_PDEN_XTAL32K_MASK)) && ((PMC->RTCOSC32K & PMC_RTCOSC32K_SEL_MASK) != 0UL)) ?
224 CLK_RTC_32K_CLK :
225 0U;
226 }
227
228
229
230 /* ----------------------------------------------------------------------------
231 -- Core clock
232 ---------------------------------------------------------------------------- */
233
234 uint32_t SystemCoreClock = DEFAULT_SYSTEM_CLOCK;
235
236 /* ----------------------------------------------------------------------------
237 -- SystemInit()
238 ---------------------------------------------------------------------------- */
239
SystemInit(void)240 __attribute__ ((weak)) void SystemInit (void) {
241 #if ((__FPU_PRESENT == 1) && (__FPU_USED == 1))
242 SCB->CPACR |= ((3UL << 10*2) | (3UL << 11*2)); /* set CP10, CP11 Full Access in Secure mode */
243 #if defined (__ARM_FEATURE_CMSE) && (__ARM_FEATURE_CMSE == 3U)
244 SCB_NS->CPACR |= ((3UL << 10*2) | (3UL << 11*2)); /* set CP10, CP11 Full Access in Non-secure mode */
245 #endif /* (__ARM_FEATURE_CMSE) && (__ARM_FEATURE_CMSE == 3U) */
246 #endif /* ((__FPU_PRESENT == 1) && (__FPU_USED == 1)) */
247
248 SCB->CPACR |= ((3UL << 0*2) | (3UL << 1*2)); /* set CP0, CP1 Full Access in Secure mode (enable PowerQuad) */
249 #if defined (__ARM_FEATURE_CMSE) && (__ARM_FEATURE_CMSE == 3U)
250 SCB_NS->CPACR |= ((3UL << 0*2) | (3UL << 1*2)); /* set CP0, CP1 Full Access in Normal mode (enable PowerQuad) */
251 #endif /* (__ARM_FEATURE_CMSE) && (__ARM_FEATURE_CMSE == 3U) */
252
253 SCB->NSACR |= ((3UL << 0) | (3UL << 10)); /* enable CP0, CP1, CP10, CP11 Non-secure Access */
254
255 #if defined(__MCUXPRESSO)
256 extern void(*const g_pfnVectors[]) (void);
257 SCB->VTOR = (uint32_t) &g_pfnVectors;
258 #else
259 extern void *__Vectors;
260 SCB->VTOR = (uint32_t) &__Vectors;
261 #endif
262 SYSCON->TRACECLKDIV = 0;
263 /* Optionally enable RAM banks that may be off by default at reset */
264 #if !defined(DONT_ENABLE_DISABLED_RAMBANKS)
265 SYSCON->AHBCLKCTRLSET[0] = SYSCON_AHBCLKCTRL0_SRAM_CTRL1_MASK | SYSCON_AHBCLKCTRL0_SRAM_CTRL2_MASK
266 | SYSCON_AHBCLKCTRL0_SRAM_CTRL3_MASK | SYSCON_AHBCLKCTRL0_SRAM_CTRL4_MASK;
267 #endif
268 SystemInitHook();
269 }
270
271 /* ----------------------------------------------------------------------------
272 -- SystemCoreClockUpdate()
273 ---------------------------------------------------------------------------- */
274
SystemCoreClockUpdate(void)275 void SystemCoreClockUpdate (void) {
276 uint32_t clkRate = 0;
277 uint32_t prediv, postdiv;
278 uint64_t workRate;
279 uint64_t workRate1;
280
281 switch (SYSCON->MAINCLKSELB & SYSCON_MAINCLKSELB_SEL_MASK)
282 {
283 case 0x00: /* MAINCLKSELA clock (main_clk_a)*/
284 switch (SYSCON->MAINCLKSELA & SYSCON_MAINCLKSELA_SEL_MASK)
285 {
286 case 0x00: /* FRO 12 MHz (fro_12m) */
287 clkRate = GetFro12MFreq();
288 break;
289 case 0x01: /* CLKIN (clk_in) */
290 clkRate = GetExtClkFreq();
291 break;
292 case 0x02: /* Fro 1MHz (fro_1m) */
293 clkRate = GetFro1MFreq();
294 break;
295 default: /* = 0x03 = FRO 96 MHz (fro_hf) */
296 clkRate = GetFroHfFreq();
297 break;
298 }
299 break;
300 case 0x01: /* PLL0 clock (pll0_clk)*/
301 switch (SYSCON->PLL0CLKSEL & SYSCON_PLL0CLKSEL_SEL_MASK)
302 {
303 case 0x00: /* FRO 12 MHz (fro_12m) */
304 clkRate = GetFro12MFreq();
305 break;
306 case 0x01: /* CLKIN (clk_in) */
307 clkRate = GetExtClkFreq();
308 break;
309 case 0x02: /* Fro 1MHz (fro_1m) */
310 clkRate = GetFro1MFreq();
311 break;
312 case 0x03: /* RTC oscillator 32 kHz output (32k_clk) */
313 clkRate = GetOsc32KFreq();
314 break;
315 default:
316 clkRate = 0UL;
317 break;
318 }
319 if (((SYSCON->PLL0CTRL & SYSCON_PLL0CTRL_BYPASSPLL_MASK) == 0UL) && ((SYSCON->PLL0CTRL & SYSCON_PLL0CTRL_CLKEN_MASK) != 0UL) && ((PMC->PDRUNCFG0 & PMC_PDRUNCFG0_PDEN_PLL0_MASK) == 0UL) && ((PMC->PDRUNCFG0 & PMC_PDRUNCFG0_PDEN_PLL0_SSCG_MASK) == 0UL))
320 {
321 prediv = findPll0PreDiv();
322 postdiv = findPll0PostDiv();
323 /* Adjust input clock */
324 clkRate = clkRate / prediv;
325 /* MDEC used for rate */
326 workRate = (uint64_t)clkRate * (uint64_t)findPll0MMult();
327 clkRate = (uint32_t)(workRate / ((uint64_t)postdiv));
328 }
329 break;
330 case 0x02: /* PLL1 clock (pll1_clk)*/
331 switch (SYSCON->PLL1CLKSEL & SYSCON_PLL1CLKSEL_SEL_MASK)
332 {
333 case 0x00: /* FRO 12 MHz (fro_12m) */
334 clkRate = GetFro12MFreq();
335 break;
336 case 0x01: /* CLKIN (clk_in) */
337 clkRate = GetExtClkFreq();
338 break;
339 case 0x02: /* Fro 1MHz (fro_1m) */
340 clkRate = GetFro1MFreq();
341 break;
342 case 0x03: /* RTC oscillator 32 kHz output (32k_clk) */
343 clkRate = GetOsc32KFreq();
344 break;
345 default:
346 clkRate = 0UL;
347 break;
348 }
349 if (((SYSCON->PLL1CTRL & SYSCON_PLL1CTRL_BYPASSPLL_MASK) == 0UL) && ((SYSCON->PLL1CTRL & SYSCON_PLL1CTRL_CLKEN_MASK) != 0UL) && ((PMC->PDRUNCFG0 & PMC_PDRUNCFG0_PDEN_PLL1_MASK) == 0UL))
350 {
351 /* PLL is not in bypass mode, get pre-divider, post-divider, and M divider */
352 prediv = findPll1PreDiv();
353 postdiv = findPll1PostDiv();
354 /* Adjust input clock */
355 clkRate = clkRate / prediv;
356
357 /* MDEC used for rate */
358 workRate1 = (uint64_t)clkRate * (uint64_t)findPll1MMult();
359 clkRate = (uint32_t)(workRate1 / ((uint64_t)postdiv));
360 }
361 break;
362 case 0x03: /* RTC oscillator 32 kHz output (32k_clk) */
363 clkRate = GetOsc32KFreq();
364 break;
365 default:
366 clkRate = 0UL;
367 break;
368 }
369 SystemCoreClock = clkRate / ((SYSCON->AHBCLKDIV & 0xFFUL) + 1UL);
370 }
371
372 /* ----------------------------------------------------------------------------
373 -- SystemInitHook()
374 ---------------------------------------------------------------------------- */
375
SystemInitHook(void)376 __attribute__ ((weak)) void SystemInitHook (void) {
377 /* Void implementation of the weak function. */
378 }
379