1 /*
2  * Copyright (c) 2016 Piotr Mienkowski
3  * Copyright (c) 2019-2023 Gerson Fernando Budke <nandojve@gmail.com>
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 /** @file
8  * @brief Atmel SAM V71 MCU initialization code
9  *
10  * This file provides routines to initialize and support board-level hardware
11  * for the Atmel SAM V71 MCU.
12  */
13 
14 #include <zephyr/kernel.h>
15 #include <zephyr/device.h>
16 #include <zephyr/init.h>
17 #include <zephyr/cache.h>
18 #include <zephyr/arch/cache.h>
19 #include <soc.h>
20 #include <cmsis_core.h>
21 #include <zephyr/logging/log.h>
22 
23 #define LOG_LEVEL CONFIG_SOC_LOG_LEVEL
24 LOG_MODULE_REGISTER(soc);
25 
26 /**
27  * @brief Setup various clocks on SoC at boot time.
28  *
29  * Setup Slow, Main, PLLA, Processor and Master clocks during the device boot.
30  * It is assumed that the relevant registers are at their reset value.
31  */
clock_init(void)32 static ALWAYS_INLINE void clock_init(void)
33 {
34 	/* Switch the main clock to the internal OSC with 12MHz */
35 	soc_pmc_switch_mainck_to_fastrc(SOC_PMC_FAST_RC_FREQ_12MHZ);
36 
37 	/* Switch MCK (Master Clock) to the main clock */
38 	soc_pmc_mck_set_source(SOC_PMC_MCK_SRC_MAIN_CLK);
39 
40 	EFC->EEFC_FMR = EEFC_FMR_FWS(0) | EEFC_FMR_CLOE;
41 
42 	soc_pmc_enable_clock_failure_detector();
43 
44 	if (IS_ENABLED(CONFIG_SOC_ATMEL_SAM_EXT_SLCK)) {
45 		soc_supc_slow_clock_select_crystal_osc();
46 	}
47 
48 
49 	if (IS_ENABLED(CONFIG_SOC_ATMEL_SAM_EXT_MAINCK)) {
50 		/*
51 		 * Setup main external crystal oscillator.
52 		 */
53 
54 		/* We select maximum setup time.
55 		 * While start up time could be shortened
56 		 * this optimization is not deemed
57 		 * critical now.
58 		 */
59 		soc_pmc_switch_mainck_to_xtal(false, 0xff);
60 	}
61 
62 	/*
63 	 * Set FWS (Flash Wait State) value before increasing Master Clock
64 	 * (MCK) frequency.
65 	 * TODO: set FWS based on the actual MCK frequency and VDDIO value
66 	 * rather than maximum supported 150 MHz at standard VDDIO=2.7V
67 	 */
68 	EFC->EEFC_FMR = EEFC_FMR_FWS(5) | EEFC_FMR_CLOE;
69 
70 	/*
71 	 * Setup PLLA
72 	 */
73 
74 	/*
75 	 * PLL clock = Main * (MULA + 1) / DIVA
76 	 *
77 	 * By default, MULA == 24, DIVA == 1.
78 	 * With main crystal running at 12 MHz,
79 	 * PLL = 12 * (24 + 1) / 1 = 300 MHz
80 	 *
81 	 * With Processor Clock prescaler at 1
82 	 * Processor Clock (HCLK)=300 MHz.
83 	 */
84 	soc_pmc_enable_pllack(CONFIG_SOC_ATMEL_SAM_PLLA_MULA, 0x3Fu,
85 			      CONFIG_SOC_ATMEL_SAM_PLLA_DIVA);
86 
87 
88 	soc_pmc_enable_upllck(0x3Fu);
89 
90 	/*
91 	 * Final setup of the Master Clock
92 	 */
93 
94 	/* Setting PLLA as MCK, first prescaler, then divider and source last */
95 	soc_pmc_mck_set_prescaler(1);
96 	soc_pmc_mck_set_divider(CONFIG_SOC_ATMEL_SAM_MDIV);
97 	soc_pmc_mck_set_source(SOC_PMC_MCK_SRC_PLLA_CLK);
98 
99 	/* Disable internal fast RC if we have an external crystal oscillator */
100 	if (IS_ENABLED(CONFIG_SOC_ATMEL_SAM_EXT_MAINCK)) {
101 		soc_pmc_osc_disable_fastrc();
102 	}
103 }
104 
soc_reset_hook(void)105 void soc_reset_hook(void)
106 {
107 	if (IS_ENABLED(CONFIG_SOC_ATMEL_SAM_WAIT_MODE)) {
108 		/*
109 		 * Instruct CPU to enter Wait mode instead of Sleep mode to
110 		 * keep Processor Clock (HCLK) and thus be able to debug
111 		 * CPU using JTAG.
112 		 */
113 		soc_pmc_enable_waitmode();
114 	}
115 
116 	/*
117 	 * DTCM is enabled by default at reset, therefore we have to disable
118 	 * it first to get the caches into a state where then the
119 	 * sys_cache*-functions can enable them, if requested by the
120 	 * configuration.
121 	 */
122 	SCB_InvalidateDCache();
123 	SCB_DisableDCache();
124 
125 	/*
126 	 * Enable the caches only if configured to do so.
127 	 */
128 	sys_cache_instr_enable();
129 	sys_cache_data_enable();
130 
131 	/* Setup system clocks */
132 	clock_init();
133 }
134 
135 extern void atmel_samv71_config(void);
136 /**
137  * @brief Perform basic hardware initialization at boot.
138  *
139  * This needs to be run at the very beginning.
140  */
soc_early_init_hook(void)141 void soc_early_init_hook(void)
142 {
143 	/* Check that the CHIP CIDR matches the HAL one */
144 	if (CHIPID->CHIPID_CIDR != CHIP_CIDR) {
145 		LOG_WRN("CIDR mismatch: chip = 0x%08x vs HAL = 0x%08x",
146 			(uint32_t)CHIPID->CHIPID_CIDR, (uint32_t)CHIP_CIDR);
147 	}
148 	atmel_samv71_config();
149 }
150