]>
Commit | Line | Data |
---|---|---|
83d290c5 | 1 | // SPDX-License-Identifier: GPL-2.0+ |
f046ccd1 | 2 | /* |
03051c3d | 3 | * Copyright (C) 2004-2007 Freescale Semiconductor, Inc. |
f046ccd1 EL |
4 | */ |
5 | ||
6 | /* | |
7 | * CPU specific code for the MPC83xx family. | |
8 | * | |
9 | * Derived from the MPC8260 and MPC85xx. | |
10 | */ | |
11 | ||
12 | #include <common.h> | |
30c7c434 | 13 | #include <cpu_func.h> |
36bf446b | 14 | #include <irq_func.h> |
049f8d6f | 15 | #include <time.h> |
2189d5f1 | 16 | #include <vsprintf.h> |
f046ccd1 EL |
17 | #include <watchdog.h> |
18 | #include <command.h> | |
19 | #include <mpc83xx.h> | |
20 | #include <asm/processor.h> | |
b08c8c48 | 21 | #include <linux/libfdt.h> |
75b9d4ae | 22 | #include <tsec.h> |
0e8454e9 | 23 | #include <netdev.h> |
e1ac387f | 24 | #include <fsl_esdhc.h> |
9403fc41 | 25 | #if defined(CONFIG_BOOTCOUNT_LIMIT) && !defined(CONFIG_ARCH_MPC831X) |
38d67a4e | 26 | #include <linux/immap_qe.h> |
f70fd13e HS |
27 | #include <asm/io.h> |
28 | #endif | |
f046ccd1 | 29 | |
d87080b7 WD |
30 | DECLARE_GLOBAL_DATA_PTR; |
31 | ||
19fbdca4 | 32 | #ifndef CONFIG_CPU_MPC83XX |
f046ccd1 EL |
33 | int checkcpu(void) |
34 | { | |
5f820439 | 35 | volatile immap_t *immr; |
f046ccd1 EL |
36 | ulong clock = gd->cpu_clk; |
37 | u32 pvr = get_pvr(); | |
5f820439 | 38 | u32 spridr; |
f046ccd1 | 39 | char buf[32]; |
d891ab95 | 40 | int ret; |
e5c4ade4 KP |
41 | int i; |
42 | ||
e5c4ade4 KP |
43 | const struct cpu_type { |
44 | char name[15]; | |
45 | u32 partid; | |
46 | } cpu_type_list [] = { | |
7c619ddc | 47 | CPU_TYPE_ENTRY(8308), |
a88731a6 | 48 | CPU_TYPE_ENTRY(8309), |
e5c4ade4 KP |
49 | CPU_TYPE_ENTRY(8311), |
50 | CPU_TYPE_ENTRY(8313), | |
51 | CPU_TYPE_ENTRY(8314), | |
52 | CPU_TYPE_ENTRY(8315), | |
53 | CPU_TYPE_ENTRY(8321), | |
54 | CPU_TYPE_ENTRY(8323), | |
55 | CPU_TYPE_ENTRY(8343), | |
56 | CPU_TYPE_ENTRY(8347_TBGA_), | |
57 | CPU_TYPE_ENTRY(8347_PBGA_), | |
58 | CPU_TYPE_ENTRY(8349), | |
59 | CPU_TYPE_ENTRY(8358_TBGA_), | |
60 | CPU_TYPE_ENTRY(8358_PBGA_), | |
61 | CPU_TYPE_ENTRY(8360), | |
62 | CPU_TYPE_ENTRY(8377), | |
63 | CPU_TYPE_ENTRY(8378), | |
64 | CPU_TYPE_ENTRY(8379), | |
65 | }; | |
f046ccd1 | 66 | |
6d0f6bcf | 67 | immr = (immap_t *)CONFIG_SYS_IMMR; |
5f820439 | 68 | |
d891ab95 SG |
69 | ret = prt_83xx_rsr(); |
70 | if (ret) | |
71 | return ret; | |
72 | ||
54b2d434 | 73 | puts("CPU: "); |
95e7ef89 SW |
74 | |
75 | switch (pvr & 0xffff0000) { | |
76 | case PVR_E300C1: | |
77 | printf("e300c1, "); | |
78 | break; | |
79 | ||
80 | case PVR_E300C2: | |
81 | printf("e300c2, "); | |
82 | break; | |
83 | ||
84 | case PVR_E300C3: | |
85 | printf("e300c3, "); | |
86 | break; | |
87 | ||
03051c3d DL |
88 | case PVR_E300C4: |
89 | printf("e300c4, "); | |
90 | break; | |
91 | ||
95e7ef89 SW |
92 | default: |
93 | printf("Unknown core, "); | |
f046ccd1 EL |
94 | } |
95 | ||
5f820439 | 96 | spridr = immr->sysconf.spridr; |
6902df56 | 97 | |
e5c4ade4 KP |
98 | for (i = 0; i < ARRAY_SIZE(cpu_type_list); i++) |
99 | if (cpu_type_list[i].partid == PARTID_NO_E(spridr)) { | |
100 | puts("MPC"); | |
101 | puts(cpu_type_list[i].name); | |
102 | if (IS_E_PROCESSOR(spridr)) | |
103 | puts("E"); | |
dfe812c7 KP |
104 | if ((SPR_FAMILY(spridr) == SPR_834X_FAMILY || |
105 | SPR_FAMILY(spridr) == SPR_836X_FAMILY) && | |
106 | REVID_MAJOR(spridr) >= 2) | |
e5c4ade4 KP |
107 | puts("A"); |
108 | printf(", Rev: %d.%d", REVID_MAJOR(spridr), | |
109 | REVID_MINOR(spridr)); | |
110 | break; | |
111 | } | |
112 | ||
113 | if (i == ARRAY_SIZE(cpu_type_list)) | |
114 | printf("(SPRIDR %08x unknown), ", spridr); | |
115 | ||
116 | printf(" at %s MHz, ", strmhz(buf, clock)); | |
117 | ||
c6731fe2 | 118 | printf("CSB: %s MHz\n", strmhz(buf, gd->arch.csb_clk)); |
54b2d434 | 119 | |
f046ccd1 EL |
120 | return 0; |
121 | } | |
19fbdca4 | 122 | #endif |
f046ccd1 | 123 | |
76fdad1f | 124 | #ifndef CONFIG_SYSRESET |
f046ccd1 | 125 | int |
54841ab5 | 126 | do_reset (cmd_tbl_t * cmdtp, int flag, int argc, char * const argv[]) |
f046ccd1 | 127 | { |
07a2505f WD |
128 | ulong msr; |
129 | #ifndef MPC83xx_RESET | |
130 | ulong addr; | |
131 | #endif | |
f046ccd1 | 132 | |
6d0f6bcf | 133 | volatile immap_t *immap = (immap_t *) CONFIG_SYS_IMMR; |
f046ccd1 | 134 | |
4c006dd1 MZ |
135 | puts("Resetting the board.\n"); |
136 | ||
f046ccd1 | 137 | #ifdef MPC83xx_RESET |
4c006dd1 | 138 | |
f046ccd1 | 139 | /* Interrupts and MMU off */ |
5c229985 MS |
140 | msr = mfmsr(); |
141 | msr &= ~(MSR_EE | MSR_IR | MSR_DR); | |
142 | mtmsr(msr); | |
f046ccd1 EL |
143 | |
144 | /* enable Reset Control Reg */ | |
145 | immap->reset.rpr = 0x52535445; | |
5c229985 MS |
146 | sync(); |
147 | isync(); | |
f046ccd1 EL |
148 | |
149 | /* confirm Reset Control Reg is enabled */ | |
5c229985 MS |
150 | while(!((immap->reset.rcer) & RCER_CRE)) |
151 | ; | |
f046ccd1 | 152 | |
f046ccd1 EL |
153 | udelay(200); |
154 | ||
155 | /* perform reset, only one bit */ | |
07a2505f WD |
156 | immap->reset.rcr = RCR_SWHR; |
157 | ||
158 | #else /* ! MPC83xx_RESET */ | |
f046ccd1 | 159 | |
07a2505f WD |
160 | immap->reset.rmr = RMR_CSRE; /* Checkstop Reset enable */ |
161 | ||
162 | /* Interrupts and MMU off */ | |
5c229985 | 163 | msr = mfmsr(); |
f046ccd1 | 164 | msr &= ~(MSR_ME | MSR_EE | MSR_IR | MSR_DR); |
5c229985 | 165 | mtmsr(msr); |
f046ccd1 EL |
166 | |
167 | /* | |
168 | * Trying to execute the next instruction at a non-existing address | |
169 | * should cause a machine check, resulting in reset | |
170 | */ | |
6d0f6bcf | 171 | addr = CONFIG_SYS_RESET_ADDRESS; |
f046ccd1 | 172 | |
f046ccd1 | 173 | ((void (*)(void)) addr) (); |
07a2505f WD |
174 | #endif /* MPC83xx_RESET */ |
175 | ||
f046ccd1 EL |
176 | return 1; |
177 | } | |
76fdad1f | 178 | #endif |
f046ccd1 EL |
179 | |
180 | /* | |
181 | * Get timebase clock frequency (like cpu_clk in Hz) | |
182 | */ | |
2c21749d | 183 | #ifndef CONFIG_TIMER |
f046ccd1 EL |
184 | unsigned long get_tbclk(void) |
185 | { | |
63a7578e | 186 | return (gd->bus_clk + 3L) / 4L; |
f046ccd1 | 187 | } |
2c21749d | 188 | #endif |
f046ccd1 EL |
189 | |
190 | #if defined(CONFIG_WATCHDOG) | |
191 | void watchdog_reset (void) | |
192 | { | |
2ad6b513 TT |
193 | int re_enable = disable_interrupts(); |
194 | ||
195 | /* Reset the 83xx watchdog */ | |
6d0f6bcf | 196 | volatile immap_t *immr = (immap_t *) CONFIG_SYS_IMMR; |
2ad6b513 TT |
197 | immr->wdt.swsrr = 0x556c; |
198 | immr->wdt.swsrr = 0xaa39; | |
199 | ||
200 | if (re_enable) | |
9d3915b2 | 201 | enable_interrupts(); |
f046ccd1 | 202 | } |
2ad6b513 | 203 | #endif |
62ec6418 | 204 | |
8835836a | 205 | #ifndef CONFIG_DM_ETH |
75b9d4ae AF |
206 | /* |
207 | * Initializes on-chip ethernet controllers. | |
208 | * to override, implement board_eth_init() | |
dd35479a | 209 | */ |
dd35479a BW |
210 | int cpu_eth_init(bd_t *bis) |
211 | { | |
8e55258f HW |
212 | #if defined(CONFIG_UEC_ETH) |
213 | uec_standard_init(bis); | |
0e8454e9 | 214 | #endif |
8e55258f | 215 | |
75b9d4ae AF |
216 | #if defined(CONFIG_TSEC_ENET) |
217 | tsec_standard_init(bis); | |
dd35479a BW |
218 | #endif |
219 | return 0; | |
220 | } | |
8835836a | 221 | #endif /* !CONFIG_DM_ETH */ |
e1ac387f AF |
222 | |
223 | /* | |
224 | * Initializes on-chip MMC controllers. | |
225 | * to override, implement board_mmc_init() | |
226 | */ | |
227 | int cpu_mmc_init(bd_t *bis) | |
228 | { | |
229 | #ifdef CONFIG_FSL_ESDHC | |
230 | return fsl_esdhc_mmc_init(bis); | |
231 | #else | |
232 | return 0; | |
233 | #endif | |
234 | } | |
1e718f43 MS |
235 | |
236 | void ppcDWstore(unsigned int *addr, unsigned int *value) | |
237 | { | |
238 | asm("lfd 1, 0(%1)\n\t" | |
239 | "stfd 1, 0(%0)" | |
240 | : | |
241 | : "r" (addr), "r" (value) | |
242 | : "memory"); | |
243 | } | |
244 | ||
245 | void ppcDWload(unsigned int *addr, unsigned int *ret) | |
246 | { | |
247 | asm("lfd 1, 0(%0)\n\t" | |
248 | "stfd 1, 0(%1)" | |
249 | : | |
250 | : "r" (addr), "r" (ret) | |
251 | : "memory"); | |
252 | } |