blob: ec4641dcdbe2bf20851139875fe17022cd735c42 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Eddy Petrișor9702ec02016-06-05 03:43:00 +03002/*
3 * (C) Copyright 2013-2016, Freescale Semiconductor, Inc.
Eddy Petrișor9702ec02016-06-05 03:43:00 +03004 */
5
6#include <common.h>
Simon Glassd96c2602019-12-28 10:44:58 -07007#include <clock_legacy.h>
Simon Glass9a3b4ce2019-12-28 10:45:01 -07008#include <cpu_func.h>
Eddy Petrișor9702ec02016-06-05 03:43:00 +03009#include <asm/io.h>
10#include <asm/arch/imx-regs.h>
11#include <asm/arch/clock.h>
12#include <asm/arch/mc_cgm_regs.h>
13#include <asm/arch/mc_me_regs.h>
14#include <asm/arch/mc_rgm_regs.h>
15#include <netdev.h>
16#include <div64.h>
17#include <errno.h>
18
19u32 get_cpu_rev(void)
20{
21 struct mscm_ir *mscmir = (struct mscm_ir *)MSCM_BASE_ADDR;
22 u32 cpu = readl(&mscmir->cpxtype);
23
24 return cpu;
25}
26
27DECLARE_GLOBAL_DATA_PTR;
28
29static uintptr_t get_pllfreq(u32 pll, u32 refclk_freq, u32 plldv,
30 u32 pllfd, u32 selected_output)
31{
32 u32 vco = 0, plldv_prediv = 0, plldv_mfd = 0, pllfd_mfn = 0;
33 u32 plldv_rfdphi_div = 0, fout = 0;
34 u32 dfs_portn = 0, dfs_mfn = 0, dfs_mfi = 0;
35
36 if (selected_output > DFS_MAXNUMBER) {
37 return -1;
38 }
39
40 plldv_prediv =
41 (plldv & PLLDIG_PLLDV_PREDIV_MASK) >> PLLDIG_PLLDV_PREDIV_OFFSET;
42 plldv_mfd = (plldv & PLLDIG_PLLDV_MFD_MASK);
43
44 pllfd_mfn = (pllfd & PLLDIG_PLLFD_MFN_MASK);
45
46 plldv_prediv = plldv_prediv == 0 ? 1 : plldv_prediv;
47
48 /* The formula for VCO is from TR manual, rev. D */
49 vco = refclk_freq / plldv_prediv * (plldv_mfd + pllfd_mfn / 20481);
50
51 if (selected_output != 0) {
52 /* Determine the RFDPHI for PHI1 */
53 plldv_rfdphi_div =
54 (plldv & PLLDIG_PLLDV_RFDPHI1_MASK) >>
55 PLLDIG_PLLDV_RFDPHI1_OFFSET;
56 plldv_rfdphi_div = plldv_rfdphi_div == 0 ? 1 : plldv_rfdphi_div;
57 if (pll == ARM_PLL || pll == ENET_PLL || pll == DDR_PLL) {
58 dfs_portn =
59 readl(DFS_DVPORTn(pll, selected_output - 1));
60 dfs_mfi =
61 (dfs_portn & DFS_DVPORTn_MFI_MASK) >>
62 DFS_DVPORTn_MFI_OFFSET;
63 dfs_mfn =
64 (dfs_portn & DFS_DVPORTn_MFI_MASK) >>
65 DFS_DVPORTn_MFI_OFFSET;
66 fout = vco / (dfs_mfi + (dfs_mfn / 256));
67 } else {
68 fout = vco / plldv_rfdphi_div;
69 }
70
71 } else {
72 /* Determine the RFDPHI for PHI0 */
73 plldv_rfdphi_div =
74 (plldv & PLLDIG_PLLDV_RFDPHI_MASK) >>
75 PLLDIG_PLLDV_RFDPHI_OFFSET;
76 fout = vco / plldv_rfdphi_div;
77 }
78
79 return fout;
80
81}
82
83/* Implemented for ARMPLL, PERIPH_PLL, ENET_PLL, DDR_PLL, VIDEO_LL */
84static uintptr_t decode_pll(enum pll_type pll, u32 refclk_freq,
85 u32 selected_output)
86{
87 u32 plldv, pllfd;
88
89 plldv = readl(PLLDIG_PLLDV(pll));
90 pllfd = readl(PLLDIG_PLLFD(pll));
91
92 return get_pllfreq(pll, refclk_freq, plldv, pllfd, selected_output);
93}
94
95static u32 get_mcu_main_clk(void)
96{
97 u32 coreclk_div;
98 u32 sysclk_sel;
99 u32 freq = 0;
100
101 sysclk_sel = readl(CGM_SC_SS(MC_CGM1_BASE_ADDR)) & MC_CGM_SC_SEL_MASK;
102 sysclk_sel >>= MC_CGM_SC_SEL_OFFSET;
103
104 coreclk_div =
105 readl(CGM_SC_DCn(MC_CGM1_BASE_ADDR, 0)) & MC_CGM_SC_DCn_PREDIV_MASK;
106 coreclk_div >>= MC_CGM_SC_DCn_PREDIV_OFFSET;
107 coreclk_div += 1;
108
109 switch (sysclk_sel) {
110 case MC_CGM_SC_SEL_FIRC:
111 freq = FIRC_CLK_FREQ;
112 break;
113 case MC_CGM_SC_SEL_XOSC:
114 freq = XOSC_CLK_FREQ;
115 break;
116 case MC_CGM_SC_SEL_ARMPLL:
117 /* ARMPLL has as source XOSC and CORE_CLK has as input PHI0 */
118 freq = decode_pll(ARM_PLL, XOSC_CLK_FREQ, 0);
119 break;
120 case MC_CGM_SC_SEL_CLKDISABLE:
121 printf("Sysclk is disabled\n");
122 break;
123 default:
124 printf("unsupported system clock select\n");
125 }
126
127 return freq / coreclk_div;
128}
129
130static u32 get_sys_clk(u32 number)
131{
132 u32 sysclk_div, sysclk_div_number;
133 u32 sysclk_sel;
134 u32 freq = 0;
135
136 switch (number) {
137 case 3:
138 sysclk_div_number = 0;
139 break;
140 case 6:
141 sysclk_div_number = 1;
142 break;
143 default:
144 printf("unsupported system clock \n");
145 return -1;
146 }
147 sysclk_sel = readl(CGM_SC_SS(MC_CGM0_BASE_ADDR)) & MC_CGM_SC_SEL_MASK;
148 sysclk_sel >>= MC_CGM_SC_SEL_OFFSET;
149
150 sysclk_div =
151 readl(CGM_SC_DCn(MC_CGM1_BASE_ADDR, sysclk_div_number)) &
152 MC_CGM_SC_DCn_PREDIV_MASK;
153 sysclk_div >>= MC_CGM_SC_DCn_PREDIV_OFFSET;
154 sysclk_div += 1;
155
156 switch (sysclk_sel) {
157 case MC_CGM_SC_SEL_FIRC:
158 freq = FIRC_CLK_FREQ;
159 break;
160 case MC_CGM_SC_SEL_XOSC:
161 freq = XOSC_CLK_FREQ;
162 break;
163 case MC_CGM_SC_SEL_ARMPLL:
164 /* ARMPLL has as source XOSC and SYSn_CLK has as input DFS1 */
165 freq = decode_pll(ARM_PLL, XOSC_CLK_FREQ, 1);
166 break;
167 case MC_CGM_SC_SEL_CLKDISABLE:
168 printf("Sysclk is disabled\n");
169 break;
170 default:
171 printf("unsupported system clock select\n");
172 }
173
174 return freq / sysclk_div;
175}
176
177static u32 get_peripherals_clk(void)
178{
179 u32 aux5clk_div;
180 u32 freq = 0;
181
182 aux5clk_div =
183 readl(CGM_ACn_DCm(MC_CGM0_BASE_ADDR, 5, 0)) &
184 MC_CGM_ACn_DCm_PREDIV_MASK;
185 aux5clk_div >>= MC_CGM_ACn_DCm_PREDIV_OFFSET;
186 aux5clk_div += 1;
187
188 freq = decode_pll(PERIPH_PLL, XOSC_CLK_FREQ, 0);
189
190 return freq / aux5clk_div;
191
192}
193
194static u32 get_uart_clk(void)
195{
196 u32 auxclk3_div, auxclk3_sel, freq = 0;
197
198 auxclk3_sel =
199 readl(CGM_ACn_SS(MC_CGM0_BASE_ADDR, 3)) & MC_CGM_ACn_SEL_MASK;
200 auxclk3_sel >>= MC_CGM_ACn_SEL_OFFSET;
201
202 auxclk3_div =
203 readl(CGM_ACn_DCm(MC_CGM0_BASE_ADDR, 3, 0)) &
204 MC_CGM_ACn_DCm_PREDIV_MASK;
205 auxclk3_div >>= MC_CGM_ACn_DCm_PREDIV_OFFSET;
206 auxclk3_div += 1;
207
208 switch (auxclk3_sel) {
209 case MC_CGM_ACn_SEL_FIRC:
210 freq = FIRC_CLK_FREQ;
211 break;
212 case MC_CGM_ACn_SEL_XOSC:
213 freq = XOSC_CLK_FREQ;
214 break;
215 case MC_CGM_ACn_SEL_PERPLLDIVX:
216 freq = get_peripherals_clk() / 3;
217 break;
218 case MC_CGM_ACn_SEL_SYSCLK:
219 freq = get_sys_clk(6);
220 break;
221 default:
222 printf("unsupported system clock select\n");
223 }
224
225 return freq / auxclk3_div;
226}
227
228static u32 get_fec_clk(void)
229{
230 u32 aux2clk_div;
231 u32 freq = 0;
232
233 aux2clk_div =
234 readl(CGM_ACn_DCm(MC_CGM0_BASE_ADDR, 2, 0)) &
235 MC_CGM_ACn_DCm_PREDIV_MASK;
236 aux2clk_div >>= MC_CGM_ACn_DCm_PREDIV_OFFSET;
237 aux2clk_div += 1;
238
239 freq = decode_pll(ENET_PLL, XOSC_CLK_FREQ, 0);
240
241 return freq / aux2clk_div;
242}
243
244static u32 get_usdhc_clk(void)
245{
246 u32 aux15clk_div;
247 u32 freq = 0;
248
249 aux15clk_div =
250 readl(CGM_ACn_DCm(MC_CGM0_BASE_ADDR, 15, 0)) &
251 MC_CGM_ACn_DCm_PREDIV_MASK;
252 aux15clk_div >>= MC_CGM_ACn_DCm_PREDIV_OFFSET;
253 aux15clk_div += 1;
254
255 freq = decode_pll(ENET_PLL, XOSC_CLK_FREQ, 4);
256
257 return freq / aux15clk_div;
258}
259
260static u32 get_i2c_clk(void)
261{
262 return get_peripherals_clk();
263}
264
265/* return clocks in Hz */
266unsigned int mxc_get_clock(enum mxc_clock clk)
267{
268 switch (clk) {
269 case MXC_ARM_CLK:
270 return get_mcu_main_clk();
271 case MXC_PERIPHERALS_CLK:
272 return get_peripherals_clk();
273 case MXC_UART_CLK:
274 return get_uart_clk();
275 case MXC_FEC_CLK:
276 return get_fec_clk();
277 case MXC_I2C_CLK:
278 return get_i2c_clk();
279 case MXC_USDHC_CLK:
280 return get_usdhc_clk();
281 default:
282 break;
283 }
284 printf("Error: Unsupported function to read the frequency! \
285 Please define it correctly!");
286 return -1;
287}
288
289/* Not yet implemented - int soc_clk_dump(); */
290
291#if defined(CONFIG_DISPLAY_CPUINFO)
292static char *get_reset_cause(void)
293{
294 u32 cause = readl(MC_RGM_BASE_ADDR + 0x300);
295
296 switch (cause) {
297 case F_SWT4:
298 return "WDOG";
299 case F_JTAG:
300 return "JTAG";
301 case F_FCCU_SOFT:
302 return "FCCU soft reaction";
303 case F_FCCU_HARD:
304 return "FCCU hard reaction";
305 case F_SOFT_FUNC:
306 return "Software Functional reset";
307 case F_ST_DONE:
308 return "Self Test done reset";
309 case F_EXT_RST:
310 return "External reset";
311 default:
312 return "unknown reset";
313 }
314
315}
316
317#define SRC_SCR_SW_RST (1<<12)
318
319void reset_cpu(ulong addr)
320{
321 printf("Feature not supported.\n");
322};
323
324int print_cpuinfo(void)
325{
326 printf("CPU: Freescale Treerunner S32V234 at %d MHz\n",
327 mxc_get_clock(MXC_ARM_CLK) / 1000000);
328 printf("Reset cause: %s\n", get_reset_cause());
329
330 return 0;
331}
332#endif
333
334int cpu_eth_init(bd_t * bis)
335{
336 int rc = -ENODEV;
337
338#if defined(CONFIG_FEC_MXC)
339 rc = fecmxc_initialize(bis);
340#endif
341
342 return rc;
343}
344
345int get_clocks(void)
346{
Yangbo Lue37ac712019-06-21 11:42:28 +0800347#ifdef CONFIG_FSL_ESDHC_IMX
Eddy Petrișor9702ec02016-06-05 03:43:00 +0300348 gd->arch.sdhc_clk = mxc_get_clock(MXC_USDHC_CLK);
349#endif
350 return 0;
351}