1 /*
2  * Copyright (c) 2017, NXP
3  *
4  * SPDX-License-Identifier: Apache-2.0
5  */
6 
7 #include <zephyr/init.h>
8 #include <soc.h>
9 #include <zephyr/dt-bindings/rdc/imx_rdc.h>
10 #include <zephyr/devicetree.h>
11 #include "wdog_imx.h"
12 
13 /* Initialize clock. */
SOC_ClockInit(void)14 void SOC_ClockInit(void)
15 {
16 	/* OSC/PLL is already initialized by Cortex-A7 (u-boot) */
17 
18 	/*
19 	 * Disable WDOG3
20 	 *	Note : The WDOG clock Root is shared by all the 4 WDOGs,
21 	 *	so Zephyr code should avoid closing it
22 	 */
23 	CCM_UpdateRoot(CCM, ccmRootWdog, ccmRootmuxWdogOsc24m, 0, 0);
24 	CCM_EnableRoot(CCM, ccmRootWdog);
25 	CCM_ControlGate(CCM, ccmCcgrGateWdog3, ccmClockNeededRun);
26 
27 	RDC_SetPdapAccess(RDC, rdcPdapWdog3,
28 			RDC_DOMAIN_PERM(M4_DOMAIN_ID, RDC_DOMAIN_PERM_RW),
29 			false, false);
30 
31 	WDOG_DisablePowerdown(WDOG3);
32 
33 	CCM_ControlGate(CCM, ccmCcgrGateWdog3, ccmClockNotNeeded);
34 
35 	/* We need system PLL Div2 to run M4 core */
36 	CCM_ControlGate(CCM, ccmPllGateSys, ccmClockNeededRun);
37 	CCM_ControlGate(CCM, ccmPllGateSysDiv2, ccmClockNeededRun);
38 
39 	/* Enable clock gate for IP bridge and IO mux */
40 	CCM_ControlGate(CCM, ccmCcgrGateIpmux1, ccmClockNeededRun);
41 	CCM_ControlGate(CCM, ccmCcgrGateIpmux2, ccmClockNeededRun);
42 	CCM_ControlGate(CCM, ccmCcgrGateIpmux3, ccmClockNeededRun);
43 	CCM_ControlGate(CCM, ccmCcgrGateIomux, ccmClockNeededRun);
44 	CCM_ControlGate(CCM, ccmCcgrGateIomuxLpsr, ccmClockNeededRun);
45 
46 	/* Enable clock gate for RDC */
47 	CCM_ControlGate(CCM, ccmCcgrGateRdc, ccmClockNeededRun);
48 }
49 
SOC_RdcInit(void)50 void SOC_RdcInit(void)
51 {
52 	/* Move M4 core to specific RDC domain */
53 	RDC_SetDomainID(RDC, rdcMdaM4, M4_DOMAIN_ID, false);
54 }
55 
56 #ifdef CONFIG_GPIO_IMX
nxp_mcimx7_gpio_config(void)57 static void nxp_mcimx7_gpio_config(void)
58 {
59 
60 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(gpio1))
61 	RDC_SetPdapAccess(RDC, rdcPdapGpio1, RDC_DT_VAL(gpio1), false, false);
62 	/* Enable gpio clock gate */
63 	CCM_ControlGate(CCM, ccmCcgrGateGpio1, ccmClockNeededRunWait);
64 #endif
65 
66 
67 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(gpio2))
68 	RDC_SetPdapAccess(RDC, rdcPdapGpio2, RDC_DT_VAL(gpio2), false, false);
69 	/* Enable gpio clock gate */
70 	CCM_ControlGate(CCM, ccmCcgrGateGpio2, ccmClockNeededRunWait);
71 #endif
72 
73 
74 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(gpio7))
75 	RDC_SetPdapAccess(RDC, rdcPdapGpio7, RDC_DT_VAL(gpio7), false, false);
76 	/* Enable gpio clock gate */
77 	CCM_ControlGate(CCM, ccmCcgrGateGpio7, ccmClockNeededRunWait);
78 #endif
79 
80 }
81 #endif /* CONFIG_GPIO_IMX */
82 
83 #ifdef CONFIG_UART_IMX
nxp_mcimx7_uart_config(void)84 static void nxp_mcimx7_uart_config(void)
85 {
86 
87 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(uart2))
88 	/* We need to grasp board uart exclusively */
89 	RDC_SetPdapAccess(RDC, rdcPdapUart2, RDC_DT_VAL(uart2), false, false);
90 	/* Select clock derived from OSC clock(24M) */
91 	CCM_UpdateRoot(CCM, ccmRootUart2, ccmRootmuxUartOsc24m, 0, 0);
92 	/* Enable uart clock */
93 	CCM_EnableRoot(CCM, ccmRootUart2);
94 	/*
95 	 * IC Limitation
96 	 * M4 stop will cause A7 UART lose functionality
97 	 * So we need UART clock all the time
98 	 */
99 	CCM_ControlGate(CCM, ccmCcgrGateUart2, ccmClockNeededAll);
100 #endif
101 
102 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(uart6))
103 	/* We need to grasp board uart exclusively */
104 	RDC_SetPdapAccess(RDC, rdcPdapUart6, RDC_DT_VAL(uart6), false, false);
105 	/* Select clock derived from OSC clock(24M) */
106 	CCM_UpdateRoot(CCM, ccmRootUart6, ccmRootmuxUartOsc24m, 0, 0);
107 	/* Enable uart clock */
108 	CCM_EnableRoot(CCM, ccmRootUart6);
109 	/*
110 	 * IC Limitation
111 	 * M4 stop will cause A7 UART lose functionality
112 	 * So we need UART clock all the time
113 	 */
114 	CCM_ControlGate(CCM, ccmCcgrGateUart6, ccmClockNeededAll);
115 #endif
116 }
117 #endif /* CONFIG_UART_IMX */
118 
119 
120 #ifdef CONFIG_I2C_IMX
nxp_mcimx7_i2c_config(void)121 static void nxp_mcimx7_i2c_config(void)
122 {
123 
124 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(i2c1))
125 	/* In this example, we need to grasp board I2C exclusively */
126 	RDC_SetPdapAccess(RDC, rdcPdapI2c1, RDC_DT_VAL(i2c1), false, false);
127 	/* Select I2C clock derived from OSC clock(24M) */
128 	CCM_UpdateRoot(CCM, ccmRootI2c1, ccmRootmuxI2cOsc24m, 0, 0);
129 	/* Enable I2C clock */
130 	CCM_EnableRoot(CCM, ccmRootI2c1);
131 	CCM_ControlGate(CCM, ccmCcgrGateI2c1, ccmClockNeededRunWait);
132 #endif
133 
134 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(i2c2))
135 	/* In this example, we need to grasp board I2C exclusively */
136 	RDC_SetPdapAccess(RDC, rdcPdapI2c2, RDC_DT_VAL(i2c2), false, false);
137 	/* Select I2C clock derived from OSC clock(24M) */
138 	CCM_UpdateRoot(CCM, ccmRootI2c2, ccmRootmuxI2cOsc24m, 0, 0);
139 	/* Enable I2C clock */
140 	CCM_EnableRoot(CCM, ccmRootI2c2);
141 	CCM_ControlGate(CCM, ccmCcgrGateI2c2, ccmClockNeededRunWait);
142 #endif
143 
144 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(i2c3))
145 	/* In this example, we need to grasp board I2C exclusively */
146 	RDC_SetPdapAccess(RDC, rdcPdapI2c3, RDC_DT_VAL(i2c3), false, false);
147 	/* Select I2C clock derived from OSC clock(24M) */
148 	CCM_UpdateRoot(CCM, ccmRootI2c3, ccmRootmuxI2cOsc24m, 0, 0);
149 	/* Enable I2C clock */
150 	CCM_EnableRoot(CCM, ccmRootI2c3);
151 	CCM_ControlGate(CCM, ccmCcgrGateI2c3, ccmClockNeededRunWait);
152 #endif
153 
154 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(i2c4))
155 	/* In this example, we need to grasp board I2C exclusively */
156 	RDC_SetPdapAccess(RDC, rdcPdapI2c4, RDC_DT_VAL(i2c4), false, false);
157 	/* Select I2C clock derived from OSC clock(24M) */
158 	CCM_UpdateRoot(CCM, ccmRootI2c4, ccmRootmuxI2cOsc24m, 0, 0);
159 	/* Enable I2C clock */
160 	CCM_EnableRoot(CCM, ccmRootI2c4);
161 	CCM_ControlGate(CCM, ccmCcgrGateI2c4, ccmClockNeededRunWait);
162 #endif
163 
164 }
165 #endif /* CONFIG_I2C_IMX */
166 
167 #ifdef CONFIG_PWM_IMX
nxp_mcimx7_pwm_config(void)168 static void nxp_mcimx7_pwm_config(void)
169 {
170 
171 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(pwm1))
172 	/* We need to grasp board pwm exclusively */
173 	RDC_SetPdapAccess(RDC, rdcPdapPwm1, RDC_DT_VAL(pwm1), false, false);
174 	/* Select clock derived from OSC clock(24M) */
175 	CCM_UpdateRoot(CCM, ccmRootPwm1, ccmRootmuxPwmOsc24m, 0, 0);
176 	/* Enable pwm clock */
177 	CCM_EnableRoot(CCM, ccmRootPwm1);
178 	CCM_ControlGate(CCM, ccmCcgrGatePwm1, ccmClockNeededAll);
179 #endif
180 
181 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(pwm2))
182 	/* We need to grasp board pwm exclusively */
183 	RDC_SetPdapAccess(RDC, rdcPdapPwm2, RDC_DT_VAL(pwm2), false, false);
184 	/* Select clock derived from OSC clock(24M) */
185 	CCM_UpdateRoot(CCM, ccmRootPwm2, ccmRootmuxPwmOsc24m, 0, 0);
186 	/* Enable pwm clock */
187 	CCM_EnableRoot(CCM, ccmRootPwm2);
188 	CCM_ControlGate(CCM, ccmCcgrGatePwm2, ccmClockNeededAll);
189 #endif
190 
191 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(pwm3))
192 	/* We need to grasp board pwm exclusively */
193 	RDC_SetPdapAccess(RDC, rdcPdapPwm3, RDC_DT_VAL(pwm3), false, false);
194 	/* Select clock derived from OSC clock(24M) */
195 	CCM_UpdateRoot(CCM, ccmRootPwm3, ccmRootmuxPwmOsc24m, 0, 0);
196 	/* Enable pwm clock */
197 	CCM_EnableRoot(CCM, ccmRootPwm3);
198 	CCM_ControlGate(CCM, ccmCcgrGatePwm3, ccmClockNeededAll);
199 #endif
200 
201 #if DT_NODE_HAS_STATUS_OKAY(DT_NODELABEL(pwm4))
202 	/* We need to grasp board pwm exclusively */
203 	RDC_SetPdapAccess(RDC, rdcPdapPwm4, RDC_DT_VAL(pwm4), false, false);
204 	/* Select clock derived from OSC clock(24M) */
205 	CCM_UpdateRoot(CCM, ccmRootPwm4, ccmRootmuxPwmOsc24m, 0, 0);
206 	/* Enable pwm clock */
207 	CCM_EnableRoot(CCM, ccmRootPwm4);
208 	CCM_ControlGate(CCM, ccmCcgrGatePwm4, ccmClockNeededAll);
209 #endif
210 
211 }
212 #endif /* CONFIG_PWM_IMX */
213 
214 #ifdef CONFIG_IPM_IMX
nxp_mcimx7_mu_config(void)215 static void nxp_mcimx7_mu_config(void)
216 {
217 	/* Set access to MU B for M4 core */
218 	RDC_SetPdapAccess(RDC, rdcPdapMuB, RDC_DT_VAL(mub), false, false);
219 
220 	/* Enable clock gate for MU*/
221 	CCM_ControlGate(CCM, ccmCcgrGateMu, ccmClockNeededRun);
222 }
223 #endif /* CONFIG_IPM_IMX */
224 
soc_early_init_hook(void)225 void soc_early_init_hook(void)
226 {
227 
228 	/* SoC specific RDC settings */
229 	SOC_RdcInit();
230 
231 	/* BoC specific clock settings */
232 	SOC_ClockInit();
233 
234 #ifdef CONFIG_GPIO_IMX
235 	nxp_mcimx7_gpio_config();
236 #endif /* CONFIG_GPIO_IMX */
237 
238 #ifdef CONFIG_UART_IMX
239 	nxp_mcimx7_uart_config();
240 #endif /* CONFIG_UART_IMX */
241 
242 #ifdef CONFIG_I2C_IMX
243 	nxp_mcimx7_i2c_config();
244 #endif /* CONFIG_I2C_IMX */
245 
246 #ifdef CONFIG_PWM_IMX
247 	nxp_mcimx7_pwm_config();
248 #endif /* CONFIG_PWM_IMX */
249 
250 #ifdef CONFIG_IPM_IMX
251 	nxp_mcimx7_mu_config();
252 #endif /* CONFIG_IPM_IMX */
253 }
254