1 /*
2 * Copyright (c) 2020-2023 Gerson Fernando Budke <nandojve@gmail.com>
3 * SPDX-License-Identifier: Apache-2.0
4 */
5
6 /**
7 * @file
8 * @brief Atmel SAM4L MCU series initialization code
9 *
10 * This module provides routines to initialize and support board-level hardware
11 * for the Atmel SAM4L series processor.
12 */
13
14 #include <zephyr/device.h>
15 #include <zephyr/init.h>
16 #include <soc.h>
17 #include <zephyr/sys/util.h>
18
19 /** Watchdog control register first write keys */
20 #define WDT_FIRST_KEY 0x55ul
21 /** Watchdog control register second write keys */
22 #define WDT_SECOND_KEY 0xAAul
23
24 /**
25 * @brief Sets the WatchDog Timer Control register to the \a ctrl value thanks
26 * to the WatchDog Timer key.
27 *
28 * @param ctrl Value to set the WatchDog Timer Control register to.
29 */
wdt_set_ctrl(uint32_t ctrl)30 static ALWAYS_INLINE void wdt_set_ctrl(uint32_t ctrl)
31 {
32 volatile uint32_t dly;
33
34 /** Calculate delay for internal synchronization
35 * see 45.1.3 WDT errata
36 */
37 dly = DIV_ROUND_UP(48000000 * 2, 115000);
38 dly >>= 3; /* ~8 cycles for one while loop */
39 while (dly--) {
40 ;
41 }
42 WDT->CTRL = ctrl | WDT_CTRL_KEY(WDT_FIRST_KEY);
43 WDT->CTRL = ctrl | WDT_CTRL_KEY(WDT_SECOND_KEY);
44 }
45
46 #define XTAL_FREQ 12000000
47 #define NR_PLLS 1
48 #define PLL_MAX_STARTUP_CYCLES (SCIF_PLL_PLLCOUNT_Msk >> SCIF_PLL_PLLCOUNT_Pos)
49
50 /**
51 * Fcpu = 48MHz
52 * Fpll = (Fclk * PLL_mul) / PLL_div
53 */
54 #define PLL0_MUL (192000000 / XTAL_FREQ)
55 #define PLL0_DIV 4
56
pll_is_locked(uint32_t pll_id)57 static inline bool pll_is_locked(uint32_t pll_id)
58 {
59 return !!(SCIF->PCLKSR & (1U << (6 + pll_id)));
60 }
61
osc_is_ready(uint8_t id)62 static inline bool osc_is_ready(uint8_t id)
63 {
64 switch (id) {
65 case OSC_ID_OSC0:
66 return !!(SCIF->PCLKSR & SCIF_PCLKSR_OSC0RDY);
67 case OSC_ID_OSC32:
68 return !!(BSCIF->PCLKSR & BSCIF_PCLKSR_OSC32RDY);
69 case OSC_ID_RC32K:
70 return !!(BSCIF->RC32KCR & (BSCIF_RC32KCR_EN));
71 case OSC_ID_RC80M:
72 return !!(SCIF->RC80MCR & (SCIF_RC80MCR_EN));
73 case OSC_ID_RCFAST:
74 return !!(SCIF->RCFASTCFG & (SCIF_RCFASTCFG_EN));
75 case OSC_ID_RC1M:
76 return !!(BSCIF->RC1MCR & (BSCIF_RC1MCR_CLKOE));
77 case OSC_ID_RCSYS:
78 /* RCSYS is always ready */
79 return true;
80 default:
81 /* unhandled_case(id); */
82 return false;
83 }
84 }
85
86 /**
87 * The PLL options #PLL_OPT_VCO_RANGE_HIGH and #PLL_OPT_OUTPUT_DIV will
88 * be set automatically based on the calculated target frequency.
89 */
pll_config_init(uint32_t divide,uint32_t mul)90 static inline uint32_t pll_config_init(uint32_t divide, uint32_t mul)
91 {
92 #define SCIF0_PLL_VCO_RANGE1_MAX_FREQ 240000000
93 #define SCIF_PLL0_VCO_RANGE1_MIN_FREQ 160000000
94 #define SCIF_PLL0_VCO_RANGE0_MAX_FREQ 180000000
95 #define SCIF_PLL0_VCO_RANGE0_MIN_FREQ 80000000
96 /* VCO frequency range is 160-240 MHz (80-180 MHz if unset) */
97 #define PLL_OPT_VCO_RANGE_HIGH 0
98 /* Divide output frequency by two */
99 #define PLL_OPT_OUTPUT_DIV 1
100 /* The threshold above which to set the #PLL_OPT_VCO_RANGE_HIGH option */
101 #define PLL_VCO_LOW_THRESHOLD ((SCIF_PLL0_VCO_RANGE1_MIN_FREQ \
102 + SCIF_PLL0_VCO_RANGE0_MAX_FREQ) / 2)
103 #define PLL_MIN_HZ 40000000
104 #define PLL_MAX_HZ 240000000
105 #define MUL_MIN 2
106 #define MUL_MAX 16
107 #define DIV_MIN 0
108 #define DIV_MAX 15
109
110 uint32_t pll_value;
111 uint32_t vco_hz;
112
113 /* Calculate internal VCO frequency */
114 vco_hz = XTAL_FREQ * mul;
115 vco_hz /= divide;
116
117 pll_value = 0;
118
119 /* Bring the internal VCO frequency up to the minimum value */
120 if ((vco_hz < PLL_MIN_HZ * 2) && (mul <= 8)) {
121 mul *= 2;
122 vco_hz *= 2;
123 pll_value |= (1U << (SCIF_PLL_PLLOPT_Pos +
124 PLL_OPT_OUTPUT_DIV));
125 }
126
127 /* Set VCO frequency range according to calculated value */
128 if (vco_hz >= PLL_VCO_LOW_THRESHOLD) {
129 pll_value |= 1U << (SCIF_PLL_PLLOPT_Pos +
130 PLL_OPT_VCO_RANGE_HIGH);
131 }
132
133 pll_value |= ((mul - 1) << SCIF_PLL_PLLMUL_Pos) |
134 (divide << SCIF_PLL_PLLDIV_Pos) |
135 (PLL_MAX_STARTUP_CYCLES << SCIF_PLL_PLLCOUNT_Pos);
136
137 return pll_value;
138 }
139
flashcalw_set_wait_state(uint32_t wait_state)140 static inline void flashcalw_set_wait_state(uint32_t wait_state)
141 {
142 HFLASHC->FCR = (HFLASHC->FCR & ~FLASHCALW_FCR_FWS) |
143 (wait_state ?
144 FLASHCALW_FCR_FWS_1 :
145 FLASHCALW_FCR_FWS_0);
146 }
147
flashcalw_is_ready(void)148 static inline bool flashcalw_is_ready(void)
149 {
150 return ((HFLASHC->FSR & FLASHCALW_FSR_FRDY) != 0);
151 }
152
flashcalw_issue_command(uint32_t command,int page_number)153 static inline void flashcalw_issue_command(uint32_t command, int page_number)
154 {
155 uint32_t time;
156
157 flashcalw_is_ready();
158 time = HFLASHC->FCMD;
159
160 /* Clear the command bitfield. */
161 time &= ~FLASHCALW_FCMD_CMD_Msk;
162 if (page_number >= 0) {
163 time = (FLASHCALW_FCMD_KEY_KEY |
164 FLASHCALW_FCMD_PAGEN(page_number) | command);
165 } else {
166 time |= (FLASHCALW_FCMD_KEY_KEY | command);
167 }
168
169 HFLASHC->FCMD = time;
170 flashcalw_is_ready();
171 }
172
173 /**
174 * @brief Setup various clock on SoC at boot time.
175 *
176 * Setup the SoC clocks according to section 28.12 in datasheet.
177 *
178 * Setup Slow, Main, PLLA, Processor and Master clocks during the device boot.
179 * It is assumed that the relevant registers are at their reset value.
180 */
clock_init(void)181 static ALWAYS_INLINE void clock_init(void)
182 {
183 /* Disable PicoCache and Enable HRAMC1 as extended RAM */
184 soc_pmc_peripheral_enable(
185 PM_CLOCK_MASK(PM_CLK_GRP_HSB, SYSCLK_HRAMC1_DATA));
186 soc_pmc_peripheral_enable(
187 PM_CLOCK_MASK(PM_CLK_GRP_PBB, SYSCLK_HRAMC1_REGS));
188
189 HCACHE->CTRL = HCACHE_CTRL_CEN_NO;
190
191 while (HCACHE->SR & HCACHE_SR_CSTS_EN) {
192 ;
193 }
194
195 /* Enable PLL */
196 if (!pll_is_locked(0)) {
197 /* This assumes external 12MHz Crystal */
198 SCIF->UNLOCK = SCIF_UNLOCK_KEY(0xAAu) |
199 SCIF_UNLOCK_ADDR((uint32_t)&SCIF->OSCCTRL0 -
200 (uint32_t)SCIF);
201 SCIF->OSCCTRL0 = SCIF_OSCCTRL0_STARTUP(2) |
202 SCIF_OSCCTRL0_GAIN(3) |
203 SCIF_OSCCTRL0_MODE |
204 SCIF_OSCCTRL0_OSCEN;
205
206 while (!osc_is_ready(OSC_ID_OSC0)) {
207 ;
208 }
209 uint32_t pll_config = pll_config_init(PLL0_DIV,
210 PLL0_MUL);
211
212 SCIF->UNLOCK = SCIF_UNLOCK_KEY(0xAAu) |
213 SCIF_UNLOCK_ADDR((uint32_t)&SCIF->PLL[0] -
214 (uint32_t)SCIF);
215 SCIF->PLL[0] = pll_config | SCIF_PLL_PLLEN;
216
217 while (!pll_is_locked(0)) {
218 ;
219 }
220 }
221
222 /** Set a flash wait state depending on the new cpu frequency.
223 */
224 flashcalw_set_wait_state(1);
225 flashcalw_issue_command(FLASHCALW_FCMD_CMD_HSEN, -1);
226
227 /** Set Clock CPU/BUS dividers
228 */
229 PM->UNLOCK = PM_UNLOCK_KEY(0xAAu) |
230 PM_UNLOCK_ADDR((uint32_t)&PM->CPUSEL - (uint32_t)PM);
231 PM->CPUSEL = PM_CPUSEL_CPUSEL(0);
232
233 PM->UNLOCK = PM_UNLOCK_KEY(0xAAu) |
234 PM_UNLOCK_ADDR((uint32_t)&PM->PBASEL - (uint32_t)PM);
235 PM->PBASEL = PM_PBASEL_PBSEL(0);
236
237 PM->UNLOCK = PM_UNLOCK_KEY(0xAAu) |
238 PM_UNLOCK_ADDR((uint32_t)&PM->PBBSEL - (uint32_t)PM);
239 PM->PBBSEL = PM_PBBSEL_PBSEL(0);
240
241 PM->UNLOCK = PM_UNLOCK_KEY(0xAAu) |
242 PM_UNLOCK_ADDR((uint32_t)&PM->PBCSEL - (uint32_t)PM);
243 PM->PBCSEL = PM_PBCSEL_PBSEL(0);
244
245 PM->UNLOCK = PM_UNLOCK_KEY(0xAAu) |
246 PM_UNLOCK_ADDR((uint32_t)&PM->PBDSEL - (uint32_t)PM);
247 PM->PBDSEL = PM_PBDSEL_PBSEL(0);
248
249 /** Set PLL0 as source clock
250 */
251 PM->UNLOCK = PM_UNLOCK_KEY(0xAAu) |
252 PM_UNLOCK_ADDR((uint32_t)&PM->MCCTRL - (uint32_t)PM);
253 PM->MCCTRL = OSC_SRC_PLL0;
254 }
255
soc_reset_hook(void)256 void soc_reset_hook(void)
257 {
258 #if defined(CONFIG_WDT_DISABLE_AT_BOOT)
259 wdt_set_ctrl(WDT->CTRL & ~WDT_CTRL_EN);
260 while (WDT->CTRL & WDT_CTRL_EN) {
261 ;
262 }
263 #endif
264
265 /* Setup system clocks. */
266 clock_init();
267 }
268