1 /* $NetBSD: cpu.c,v 1.55 2004/02/13 11:36:10 wiz Exp $ */
4 * Copyright (c) 1995 Mark Brinicombe.
5 * Copyright (c) 1995 Brini.
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3. All advertising materials mentioning features or use of this software
17 * must display the following acknowledgement:
18 * This product includes software developed by Brini.
19 * 4. The name of the company nor the name of the author may be used to
20 * endorse or promote products derived from this software without specific
21 * prior written permission.
23 * THIS SOFTWARE IS PROVIDED BY BRINI ``AS IS'' AND ANY EXPRESS OR IMPLIED
24 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
25 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
26 * IN NO EVENT SHALL BRINI OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
27 * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
28 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
29 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
35 * RiscBSD kernel project
39 * Probing and configuration for the master CPU
44 #include <sys/cdefs.h>
45 __FBSDID("$FreeBSD$");
46 #include <sys/param.h>
47 #include <sys/systm.h>
49 #include <sys/kernel.h>
50 #include <sys/sysctl.h>
51 #include <machine/cpu.h>
52 #include <machine/endian.h>
54 #include <machine/md_var.h>
56 char machine[] = "arm";
58 SYSCTL_STRING(_hw, HW_MACHINE, machine, CTLFLAG_RD,
59 machine, 0, "Machine class");
61 static const char * const generic_steppings[16] = {
62 "rev 0", "rev 1", "rev 2", "rev 3",
63 "rev 4", "rev 5", "rev 6", "rev 7",
64 "rev 8", "rev 9", "rev 10", "rev 11",
65 "rev 12", "rev 13", "rev 14", "rev 15",
68 static const char * const xscale_steppings[16] = {
69 "step A-0", "step A-1", "step B-0", "step C-0",
70 "step D-0", "rev 5", "rev 6", "rev 7",
71 "rev 8", "rev 9", "rev 10", "rev 11",
72 "rev 12", "rev 13", "rev 14", "rev 15",
75 static const char * const i80219_steppings[16] = {
76 "step A-0", "rev 1", "rev 2", "rev 3",
77 "rev 4", "rev 5", "rev 6", "rev 7",
78 "rev 8", "rev 9", "rev 10", "rev 11",
79 "rev 12", "rev 13", "rev 14", "rev 15",
82 static const char * const i80321_steppings[16] = {
83 "step A-0", "step B-0", "rev 2", "rev 3",
84 "rev 4", "rev 5", "rev 6", "rev 7",
85 "rev 8", "rev 9", "rev 10", "rev 11",
86 "rev 12", "rev 13", "rev 14", "rev 15",
89 static const char * const i81342_steppings[16] = {
90 "step A-0", "rev 1", "rev 2", "rev 3",
91 "rev 4", "rev 5", "rev 6", "rev 7",
92 "rev 8", "rev 9", "rev 10", "rev 11",
93 "rev 12", "rev 13", "rev 14", "rev 15",
96 /* Steppings for PXA2[15]0 */
97 static const char * const pxa2x0_steppings[16] = {
98 "step A-0", "step A-1", "step B-0", "step B-1",
99 "step B-2", "step C-0", "rev 6", "rev 7",
100 "rev 8", "rev 9", "rev 10", "rev 11",
101 "rev 12", "rev 13", "rev 14", "rev 15",
104 /* Steppings for PXA255/26x.
105 * rev 5: PXA26x B0, rev 6: PXA255 A0
107 static const char * const pxa255_steppings[16] = {
108 "rev 0", "rev 1", "rev 2", "step A-0",
109 "rev 4", "step B-0", "step A-0", "rev 7",
110 "rev 8", "rev 9", "rev 10", "rev 11",
111 "rev 12", "rev 13", "rev 14", "rev 15",
114 /* Stepping for PXA27x */
115 static const char * const pxa27x_steppings[16] = {
116 "step A-0", "step A-1", "step B-0", "step B-1",
117 "step C-0", "rev 5", "rev 6", "rev 7",
118 "rev 8", "rev 9", "rev 10", "rev 11",
119 "rev 12", "rev 13", "rev 14", "rev 15",
124 enum cpu_class cpu_class;
125 const char *cpu_name;
126 const char * const *cpu_steppings;
129 const struct cpuidtab cpuids[] = {
130 { CPU_ID_ARM920T, CPU_CLASS_ARM9TDMI, "ARM920T",
132 { CPU_ID_ARM920T_ALT, CPU_CLASS_ARM9TDMI, "ARM920T",
134 { CPU_ID_ARM922T, CPU_CLASS_ARM9TDMI, "ARM922T",
136 { CPU_ID_ARM926EJS, CPU_CLASS_ARM9EJS, "ARM926EJ-S",
138 { CPU_ID_ARM940T, CPU_CLASS_ARM9TDMI, "ARM940T",
140 { CPU_ID_ARM946ES, CPU_CLASS_ARM9ES, "ARM946E-S",
142 { CPU_ID_ARM966ES, CPU_CLASS_ARM9ES, "ARM966E-S",
144 { CPU_ID_ARM966ESR1, CPU_CLASS_ARM9ES, "ARM966E-S",
146 { CPU_ID_FA526, CPU_CLASS_ARM9TDMI, "FA526",
148 { CPU_ID_FA626TE, CPU_CLASS_ARM9ES, "FA626TE",
151 { CPU_ID_TI925T, CPU_CLASS_ARM9TDMI, "TI ARM925T",
154 { CPU_ID_ARM1020E, CPU_CLASS_ARM10E, "ARM1020E",
156 { CPU_ID_ARM1022ES, CPU_CLASS_ARM10E, "ARM1022E-S",
158 { CPU_ID_ARM1026EJS, CPU_CLASS_ARM10EJ, "ARM1026EJ-S",
161 { CPU_ID_80200, CPU_CLASS_XSCALE, "i80200",
164 { CPU_ID_80321_400, CPU_CLASS_XSCALE, "i80321 400MHz",
166 { CPU_ID_80321_600, CPU_CLASS_XSCALE, "i80321 600MHz",
168 { CPU_ID_80321_400_B0, CPU_CLASS_XSCALE, "i80321 400MHz",
170 { CPU_ID_80321_600_B0, CPU_CLASS_XSCALE, "i80321 600MHz",
173 { CPU_ID_81342, CPU_CLASS_XSCALE, "i81342",
176 { CPU_ID_80219_400, CPU_CLASS_XSCALE, "i80219 400MHz",
178 { CPU_ID_80219_600, CPU_CLASS_XSCALE, "i80219 600MHz",
181 { CPU_ID_PXA27X, CPU_CLASS_XSCALE, "PXA27x",
183 { CPU_ID_PXA250A, CPU_CLASS_XSCALE, "PXA250",
185 { CPU_ID_PXA210A, CPU_CLASS_XSCALE, "PXA210",
187 { CPU_ID_PXA250B, CPU_CLASS_XSCALE, "PXA250",
189 { CPU_ID_PXA210B, CPU_CLASS_XSCALE, "PXA210",
191 { CPU_ID_PXA250C, CPU_CLASS_XSCALE, "PXA255",
193 { CPU_ID_PXA210C, CPU_CLASS_XSCALE, "PXA210",
196 { CPU_ID_MV88FR131, CPU_CLASS_MARVELL, "Feroceon 88FR131",
199 { CPU_ID_MV88FR571_VD, CPU_CLASS_MARVELL, "Feroceon 88FR571-VD",
202 { 0, CPU_CLASS_NONE, NULL, NULL }
205 struct cpu_classtab {
206 const char *class_name;
207 const char *class_option;
210 const struct cpu_classtab cpu_classes[] = {
211 { "unknown", NULL }, /* CPU_CLASS_NONE */
212 { "ARM9TDMI", "CPU_ARM9TDMI" }, /* CPU_CLASS_ARM9TDMI */
213 { "ARM9E-S", "CPU_ARM9E" }, /* CPU_CLASS_ARM9ES */
214 { "ARM9EJ-S", "CPU_ARM9E" }, /* CPU_CLASS_ARM9EJS */
215 { "ARM10E", "CPU_ARM10" }, /* CPU_CLASS_ARM10E */
216 { "ARM10EJ", "CPU_ARM10" }, /* CPU_CLASS_ARM10EJ */
217 { "XScale", "CPU_XSCALE_..." }, /* CPU_CLASS_XSCALE */
218 { "Marvell", "CPU_MARVELL" }, /* CPU_CLASS_MARVELL */
222 * Report the type of the specified arm processor. This uses the generic and
223 * arm specific information in the cpu structure to identify the processor.
224 * The remaining fields in the cpu structure are filled in appropriately.
227 static const char * const wtnames[] = {
233 "write-back-locking", /* XXX XScale-specific? */
234 "write-back-locking-A",
235 "write-back-locking-B",
242 "write-back-locking-C",
247 print_enadis(int enadis, char *s)
250 printf(" %s %sabled", s, (enadis == 0) ? "dis" : "en");
253 enum cpu_class cpu_class = CPU_CLASS_NONE;
256 identify_arm_cpu(void)
261 ctrl = cp15_sctlr_get();
262 cpuid = cp15_midr_get();
265 printf("Processor failed probe - no CPU ID\n");
269 for (i = 0; cpuids[i].cpuid != 0; i++)
270 if (cpuids[i].cpuid == (cpuid & CPU_ID_CPU_MASK)) {
271 cpu_class = cpuids[i].cpu_class;
272 printf("CPU: %s %s (%s core)\n",
274 cpuids[i].cpu_steppings[cpuid &
275 CPU_ID_REVISION_MASK],
276 cpu_classes[cpu_class].class_name);
279 if (cpuids[i].cpuid == 0)
280 printf("unknown CPU (ID = 0x%x)\n", cpuid);
284 if (ctrl & CPU_CONTROL_BEND_ENABLE)
285 printf(" Big-endian");
287 printf(" Little-endian");
290 case CPU_CLASS_ARM9TDMI:
291 case CPU_CLASS_ARM9ES:
292 case CPU_CLASS_ARM9EJS:
293 case CPU_CLASS_ARM10E:
294 case CPU_CLASS_ARM10EJ:
295 case CPU_CLASS_XSCALE:
296 case CPU_CLASS_MARVELL:
297 print_enadis(ctrl & CPU_CONTROL_DC_ENABLE, "DC");
298 print_enadis(ctrl & CPU_CONTROL_IC_ENABLE, "IC");
299 #if defined(SOC_MV_KIRKWOOD) || defined(SOC_MV_DISCOVERY)
300 i = sheeva_control_ext(0, 0);
301 print_enadis(i & MV_WA_ENABLE, "WA");
302 print_enadis(i & MV_DC_STREAM_ENABLE, "DC streaming");
304 print_enadis((i & MV_BTB_DISABLE) == 0, "BTB");
305 print_enadis(i & MV_L2_ENABLE, "L2");
306 print_enadis((i & MV_L2_PREFETCH_DISABLE) == 0,
315 print_enadis(ctrl & CPU_CONTROL_WBUF_ENABLE, "WB");
316 if (ctrl & CPU_CONTROL_LABT_ENABLE)
321 print_enadis(ctrl & CPU_CONTROL_BPRD_ENABLE, "branch prediction");
324 /* Print cache info. */
325 if (arm_picache_line_size == 0 && arm_pdcache_line_size == 0)
328 if (arm_pcache_unified) {
329 printf(" %dKB/%dB %d-way %s unified cache\n",
330 arm_pdcache_size / 1024,
331 arm_pdcache_line_size, arm_pdcache_ways,
332 wtnames[arm_pcache_type]);
334 printf(" %dKB/%dB %d-way instruction cache\n",
335 arm_picache_size / 1024,
336 arm_picache_line_size, arm_picache_ways);
337 printf(" %dKB/%dB %d-way %s data cache\n",
338 arm_pdcache_size / 1024,
339 arm_pdcache_line_size, arm_pdcache_ways,
340 wtnames[arm_pcache_type]);