target-unicore32: Clean up ENV_GET_CPU() usage
[qemu/cris-port.git] / target-sparc / cpu.c
blob5806e59af374f3d10242494a00cced9a0b8bfbd4
1 /*
2 * Sparc CPU init helpers
4 * Copyright (c) 2003-2005 Fabrice Bellard
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
20 #include "cpu.h"
22 //#define DEBUG_FEATURES
24 static int cpu_sparc_find_by_name(sparc_def_t *cpu_def, const char *cpu_model);
26 /* CPUClass::reset() */
27 static void sparc_cpu_reset(CPUState *s)
29 SPARCCPU *cpu = SPARC_CPU(s);
30 SPARCCPUClass *scc = SPARC_CPU_GET_CLASS(cpu);
31 CPUSPARCState *env = &cpu->env;
33 scc->parent_reset(s);
35 memset(env, 0, offsetof(CPUSPARCState, breakpoints));
36 tlb_flush(env, 1);
37 env->cwp = 0;
38 #ifndef TARGET_SPARC64
39 env->wim = 1;
40 #endif
41 env->regwptr = env->regbase + (env->cwp * 16);
42 CC_OP = CC_OP_FLAGS;
43 #if defined(CONFIG_USER_ONLY)
44 #ifdef TARGET_SPARC64
45 env->cleanwin = env->nwindows - 2;
46 env->cansave = env->nwindows - 2;
47 env->pstate = PS_RMO | PS_PEF | PS_IE;
48 env->asi = 0x82; /* Primary no-fault */
49 #endif
50 #else
51 #if !defined(TARGET_SPARC64)
52 env->psret = 0;
53 env->psrs = 1;
54 env->psrps = 1;
55 #endif
56 #ifdef TARGET_SPARC64
57 env->pstate = PS_PRIV|PS_RED|PS_PEF|PS_AG;
58 env->hpstate = cpu_has_hypervisor(env) ? HS_PRIV : 0;
59 env->tl = env->maxtl;
60 cpu_tsptr(env)->tt = TT_POWER_ON_RESET;
61 env->lsu = 0;
62 #else
63 env->mmuregs[0] &= ~(MMU_E | MMU_NF);
64 env->mmuregs[0] |= env->def->mmu_bm;
65 #endif
66 env->pc = 0;
67 env->npc = env->pc + 4;
68 #endif
69 env->cache_control = 0;
72 static int cpu_sparc_register(CPUSPARCState *env, const char *cpu_model)
74 sparc_def_t def1, *def = &def1;
76 if (cpu_sparc_find_by_name(def, cpu_model) < 0) {
77 return -1;
80 env->def = g_new0(sparc_def_t, 1);
81 memcpy(env->def, def, sizeof(*def));
82 #if defined(CONFIG_USER_ONLY)
83 if ((env->def->features & CPU_FEATURE_FLOAT)) {
84 env->def->features |= CPU_FEATURE_FLOAT128;
86 #endif
87 env->version = def->iu_version;
88 env->fsr = def->fpu_version;
89 env->nwindows = def->nwindows;
90 #if !defined(TARGET_SPARC64)
91 env->mmuregs[0] |= def->mmu_version;
92 cpu_sparc_set_id(env, 0);
93 env->mxccregs[7] |= def->mxcc_version;
94 #else
95 env->mmu_version = def->mmu_version;
96 env->maxtl = def->maxtl;
97 env->version |= def->maxtl << 8;
98 env->version |= def->nwindows - 1;
99 #endif
100 return 0;
103 SPARCCPU *cpu_sparc_init(const char *cpu_model)
105 SPARCCPU *cpu;
106 CPUSPARCState *env;
108 cpu = SPARC_CPU(object_new(TYPE_SPARC_CPU));
109 env = &cpu->env;
111 if (cpu_sparc_register(env, cpu_model) < 0) {
112 object_unref(OBJECT(cpu));
113 return NULL;
116 object_property_set_bool(OBJECT(cpu), true, "realized", NULL);
118 return cpu;
121 void cpu_sparc_set_id(CPUSPARCState *env, unsigned int cpu)
123 #if !defined(TARGET_SPARC64)
124 env->mxccregs[7] = ((cpu + 8) & 0xf) << 24;
125 #endif
128 static const sparc_def_t sparc_defs[] = {
129 #ifdef TARGET_SPARC64
131 .name = "Fujitsu Sparc64",
132 .iu_version = ((0x04ULL << 48) | (0x02ULL << 32) | (0ULL << 24)),
133 .fpu_version = 0x00000000,
134 .mmu_version = mmu_us_12,
135 .nwindows = 4,
136 .maxtl = 4,
137 .features = CPU_DEFAULT_FEATURES,
140 .name = "Fujitsu Sparc64 III",
141 .iu_version = ((0x04ULL << 48) | (0x03ULL << 32) | (0ULL << 24)),
142 .fpu_version = 0x00000000,
143 .mmu_version = mmu_us_12,
144 .nwindows = 5,
145 .maxtl = 4,
146 .features = CPU_DEFAULT_FEATURES,
149 .name = "Fujitsu Sparc64 IV",
150 .iu_version = ((0x04ULL << 48) | (0x04ULL << 32) | (0ULL << 24)),
151 .fpu_version = 0x00000000,
152 .mmu_version = mmu_us_12,
153 .nwindows = 8,
154 .maxtl = 5,
155 .features = CPU_DEFAULT_FEATURES,
158 .name = "Fujitsu Sparc64 V",
159 .iu_version = ((0x04ULL << 48) | (0x05ULL << 32) | (0x51ULL << 24)),
160 .fpu_version = 0x00000000,
161 .mmu_version = mmu_us_12,
162 .nwindows = 8,
163 .maxtl = 5,
164 .features = CPU_DEFAULT_FEATURES,
167 .name = "TI UltraSparc I",
168 .iu_version = ((0x17ULL << 48) | (0x10ULL << 32) | (0x40ULL << 24)),
169 .fpu_version = 0x00000000,
170 .mmu_version = mmu_us_12,
171 .nwindows = 8,
172 .maxtl = 5,
173 .features = CPU_DEFAULT_FEATURES,
176 .name = "TI UltraSparc II",
177 .iu_version = ((0x17ULL << 48) | (0x11ULL << 32) | (0x20ULL << 24)),
178 .fpu_version = 0x00000000,
179 .mmu_version = mmu_us_12,
180 .nwindows = 8,
181 .maxtl = 5,
182 .features = CPU_DEFAULT_FEATURES,
185 .name = "TI UltraSparc IIi",
186 .iu_version = ((0x17ULL << 48) | (0x12ULL << 32) | (0x91ULL << 24)),
187 .fpu_version = 0x00000000,
188 .mmu_version = mmu_us_12,
189 .nwindows = 8,
190 .maxtl = 5,
191 .features = CPU_DEFAULT_FEATURES,
194 .name = "TI UltraSparc IIe",
195 .iu_version = ((0x17ULL << 48) | (0x13ULL << 32) | (0x14ULL << 24)),
196 .fpu_version = 0x00000000,
197 .mmu_version = mmu_us_12,
198 .nwindows = 8,
199 .maxtl = 5,
200 .features = CPU_DEFAULT_FEATURES,
203 .name = "Sun UltraSparc III",
204 .iu_version = ((0x3eULL << 48) | (0x14ULL << 32) | (0x34ULL << 24)),
205 .fpu_version = 0x00000000,
206 .mmu_version = mmu_us_12,
207 .nwindows = 8,
208 .maxtl = 5,
209 .features = CPU_DEFAULT_FEATURES,
212 .name = "Sun UltraSparc III Cu",
213 .iu_version = ((0x3eULL << 48) | (0x15ULL << 32) | (0x41ULL << 24)),
214 .fpu_version = 0x00000000,
215 .mmu_version = mmu_us_3,
216 .nwindows = 8,
217 .maxtl = 5,
218 .features = CPU_DEFAULT_FEATURES,
221 .name = "Sun UltraSparc IIIi",
222 .iu_version = ((0x3eULL << 48) | (0x16ULL << 32) | (0x34ULL << 24)),
223 .fpu_version = 0x00000000,
224 .mmu_version = mmu_us_12,
225 .nwindows = 8,
226 .maxtl = 5,
227 .features = CPU_DEFAULT_FEATURES,
230 .name = "Sun UltraSparc IV",
231 .iu_version = ((0x3eULL << 48) | (0x18ULL << 32) | (0x31ULL << 24)),
232 .fpu_version = 0x00000000,
233 .mmu_version = mmu_us_4,
234 .nwindows = 8,
235 .maxtl = 5,
236 .features = CPU_DEFAULT_FEATURES,
239 .name = "Sun UltraSparc IV+",
240 .iu_version = ((0x3eULL << 48) | (0x19ULL << 32) | (0x22ULL << 24)),
241 .fpu_version = 0x00000000,
242 .mmu_version = mmu_us_12,
243 .nwindows = 8,
244 .maxtl = 5,
245 .features = CPU_DEFAULT_FEATURES | CPU_FEATURE_CMT,
248 .name = "Sun UltraSparc IIIi+",
249 .iu_version = ((0x3eULL << 48) | (0x22ULL << 32) | (0ULL << 24)),
250 .fpu_version = 0x00000000,
251 .mmu_version = mmu_us_3,
252 .nwindows = 8,
253 .maxtl = 5,
254 .features = CPU_DEFAULT_FEATURES,
257 .name = "Sun UltraSparc T1",
258 /* defined in sparc_ifu_fdp.v and ctu.h */
259 .iu_version = ((0x3eULL << 48) | (0x23ULL << 32) | (0x02ULL << 24)),
260 .fpu_version = 0x00000000,
261 .mmu_version = mmu_sun4v,
262 .nwindows = 8,
263 .maxtl = 6,
264 .features = CPU_DEFAULT_FEATURES | CPU_FEATURE_HYPV | CPU_FEATURE_CMT
265 | CPU_FEATURE_GL,
268 .name = "Sun UltraSparc T2",
269 /* defined in tlu_asi_ctl.v and n2_revid_cust.v */
270 .iu_version = ((0x3eULL << 48) | (0x24ULL << 32) | (0x02ULL << 24)),
271 .fpu_version = 0x00000000,
272 .mmu_version = mmu_sun4v,
273 .nwindows = 8,
274 .maxtl = 6,
275 .features = CPU_DEFAULT_FEATURES | CPU_FEATURE_HYPV | CPU_FEATURE_CMT
276 | CPU_FEATURE_GL,
279 .name = "NEC UltraSparc I",
280 .iu_version = ((0x22ULL << 48) | (0x10ULL << 32) | (0x40ULL << 24)),
281 .fpu_version = 0x00000000,
282 .mmu_version = mmu_us_12,
283 .nwindows = 8,
284 .maxtl = 5,
285 .features = CPU_DEFAULT_FEATURES,
287 #else
289 .name = "Fujitsu MB86904",
290 .iu_version = 0x04 << 24, /* Impl 0, ver 4 */
291 .fpu_version = 4 << 17, /* FPU version 4 (Meiko) */
292 .mmu_version = 0x04 << 24, /* Impl 0, ver 4 */
293 .mmu_bm = 0x00004000,
294 .mmu_ctpr_mask = 0x00ffffc0,
295 .mmu_cxr_mask = 0x000000ff,
296 .mmu_sfsr_mask = 0x00016fff,
297 .mmu_trcr_mask = 0x00ffffff,
298 .nwindows = 8,
299 .features = CPU_DEFAULT_FEATURES,
302 .name = "Fujitsu MB86907",
303 .iu_version = 0x05 << 24, /* Impl 0, ver 5 */
304 .fpu_version = 4 << 17, /* FPU version 4 (Meiko) */
305 .mmu_version = 0x05 << 24, /* Impl 0, ver 5 */
306 .mmu_bm = 0x00004000,
307 .mmu_ctpr_mask = 0xffffffc0,
308 .mmu_cxr_mask = 0x000000ff,
309 .mmu_sfsr_mask = 0x00016fff,
310 .mmu_trcr_mask = 0xffffffff,
311 .nwindows = 8,
312 .features = CPU_DEFAULT_FEATURES,
315 .name = "TI MicroSparc I",
316 .iu_version = 0x41000000,
317 .fpu_version = 4 << 17,
318 .mmu_version = 0x41000000,
319 .mmu_bm = 0x00004000,
320 .mmu_ctpr_mask = 0x007ffff0,
321 .mmu_cxr_mask = 0x0000003f,
322 .mmu_sfsr_mask = 0x00016fff,
323 .mmu_trcr_mask = 0x0000003f,
324 .nwindows = 7,
325 .features = CPU_FEATURE_FLOAT | CPU_FEATURE_SWAP | CPU_FEATURE_MUL |
326 CPU_FEATURE_DIV | CPU_FEATURE_FLUSH | CPU_FEATURE_FSQRT |
327 CPU_FEATURE_FMUL,
330 .name = "TI MicroSparc II",
331 .iu_version = 0x42000000,
332 .fpu_version = 4 << 17,
333 .mmu_version = 0x02000000,
334 .mmu_bm = 0x00004000,
335 .mmu_ctpr_mask = 0x00ffffc0,
336 .mmu_cxr_mask = 0x000000ff,
337 .mmu_sfsr_mask = 0x00016fff,
338 .mmu_trcr_mask = 0x00ffffff,
339 .nwindows = 8,
340 .features = CPU_DEFAULT_FEATURES,
343 .name = "TI MicroSparc IIep",
344 .iu_version = 0x42000000,
345 .fpu_version = 4 << 17,
346 .mmu_version = 0x04000000,
347 .mmu_bm = 0x00004000,
348 .mmu_ctpr_mask = 0x00ffffc0,
349 .mmu_cxr_mask = 0x000000ff,
350 .mmu_sfsr_mask = 0x00016bff,
351 .mmu_trcr_mask = 0x00ffffff,
352 .nwindows = 8,
353 .features = CPU_DEFAULT_FEATURES,
356 .name = "TI SuperSparc 40", /* STP1020NPGA */
357 .iu_version = 0x41000000, /* SuperSPARC 2.x */
358 .fpu_version = 0 << 17,
359 .mmu_version = 0x00000800, /* SuperSPARC 2.x, no MXCC */
360 .mmu_bm = 0x00002000,
361 .mmu_ctpr_mask = 0xffffffc0,
362 .mmu_cxr_mask = 0x0000ffff,
363 .mmu_sfsr_mask = 0xffffffff,
364 .mmu_trcr_mask = 0xffffffff,
365 .nwindows = 8,
366 .features = CPU_DEFAULT_FEATURES,
369 .name = "TI SuperSparc 50", /* STP1020PGA */
370 .iu_version = 0x40000000, /* SuperSPARC 3.x */
371 .fpu_version = 0 << 17,
372 .mmu_version = 0x01000800, /* SuperSPARC 3.x, no MXCC */
373 .mmu_bm = 0x00002000,
374 .mmu_ctpr_mask = 0xffffffc0,
375 .mmu_cxr_mask = 0x0000ffff,
376 .mmu_sfsr_mask = 0xffffffff,
377 .mmu_trcr_mask = 0xffffffff,
378 .nwindows = 8,
379 .features = CPU_DEFAULT_FEATURES,
382 .name = "TI SuperSparc 51",
383 .iu_version = 0x40000000, /* SuperSPARC 3.x */
384 .fpu_version = 0 << 17,
385 .mmu_version = 0x01000000, /* SuperSPARC 3.x, MXCC */
386 .mmu_bm = 0x00002000,
387 .mmu_ctpr_mask = 0xffffffc0,
388 .mmu_cxr_mask = 0x0000ffff,
389 .mmu_sfsr_mask = 0xffffffff,
390 .mmu_trcr_mask = 0xffffffff,
391 .mxcc_version = 0x00000104,
392 .nwindows = 8,
393 .features = CPU_DEFAULT_FEATURES,
396 .name = "TI SuperSparc 60", /* STP1020APGA */
397 .iu_version = 0x40000000, /* SuperSPARC 3.x */
398 .fpu_version = 0 << 17,
399 .mmu_version = 0x01000800, /* SuperSPARC 3.x, no MXCC */
400 .mmu_bm = 0x00002000,
401 .mmu_ctpr_mask = 0xffffffc0,
402 .mmu_cxr_mask = 0x0000ffff,
403 .mmu_sfsr_mask = 0xffffffff,
404 .mmu_trcr_mask = 0xffffffff,
405 .nwindows = 8,
406 .features = CPU_DEFAULT_FEATURES,
409 .name = "TI SuperSparc 61",
410 .iu_version = 0x44000000, /* SuperSPARC 3.x */
411 .fpu_version = 0 << 17,
412 .mmu_version = 0x01000000, /* SuperSPARC 3.x, MXCC */
413 .mmu_bm = 0x00002000,
414 .mmu_ctpr_mask = 0xffffffc0,
415 .mmu_cxr_mask = 0x0000ffff,
416 .mmu_sfsr_mask = 0xffffffff,
417 .mmu_trcr_mask = 0xffffffff,
418 .mxcc_version = 0x00000104,
419 .nwindows = 8,
420 .features = CPU_DEFAULT_FEATURES,
423 .name = "TI SuperSparc II",
424 .iu_version = 0x40000000, /* SuperSPARC II 1.x */
425 .fpu_version = 0 << 17,
426 .mmu_version = 0x08000000, /* SuperSPARC II 1.x, MXCC */
427 .mmu_bm = 0x00002000,
428 .mmu_ctpr_mask = 0xffffffc0,
429 .mmu_cxr_mask = 0x0000ffff,
430 .mmu_sfsr_mask = 0xffffffff,
431 .mmu_trcr_mask = 0xffffffff,
432 .mxcc_version = 0x00000104,
433 .nwindows = 8,
434 .features = CPU_DEFAULT_FEATURES,
437 .name = "LEON2",
438 .iu_version = 0xf2000000,
439 .fpu_version = 4 << 17, /* FPU version 4 (Meiko) */
440 .mmu_version = 0xf2000000,
441 .mmu_bm = 0x00004000,
442 .mmu_ctpr_mask = 0x007ffff0,
443 .mmu_cxr_mask = 0x0000003f,
444 .mmu_sfsr_mask = 0xffffffff,
445 .mmu_trcr_mask = 0xffffffff,
446 .nwindows = 8,
447 .features = CPU_DEFAULT_FEATURES | CPU_FEATURE_TA0_SHUTDOWN,
450 .name = "LEON3",
451 .iu_version = 0xf3000000,
452 .fpu_version = 4 << 17, /* FPU version 4 (Meiko) */
453 .mmu_version = 0xf3000000,
454 .mmu_bm = 0x00000000,
455 .mmu_ctpr_mask = 0xfffffffc,
456 .mmu_cxr_mask = 0x000000ff,
457 .mmu_sfsr_mask = 0xffffffff,
458 .mmu_trcr_mask = 0xffffffff,
459 .nwindows = 8,
460 .features = CPU_DEFAULT_FEATURES | CPU_FEATURE_TA0_SHUTDOWN |
461 CPU_FEATURE_ASR17 | CPU_FEATURE_CACHE_CTRL | CPU_FEATURE_POWERDOWN |
462 CPU_FEATURE_CASA,
464 #endif
467 static const char * const feature_name[] = {
468 "float",
469 "float128",
470 "swap",
471 "mul",
472 "div",
473 "flush",
474 "fsqrt",
475 "fmul",
476 "vis1",
477 "vis2",
478 "fsmuld",
479 "hypv",
480 "cmt",
481 "gl",
484 static void print_features(FILE *f, fprintf_function cpu_fprintf,
485 uint32_t features, const char *prefix)
487 unsigned int i;
489 for (i = 0; i < ARRAY_SIZE(feature_name); i++) {
490 if (feature_name[i] && (features & (1 << i))) {
491 if (prefix) {
492 (*cpu_fprintf)(f, "%s", prefix);
494 (*cpu_fprintf)(f, "%s ", feature_name[i]);
499 static void add_flagname_to_bitmaps(const char *flagname, uint32_t *features)
501 unsigned int i;
503 for (i = 0; i < ARRAY_SIZE(feature_name); i++) {
504 if (feature_name[i] && !strcmp(flagname, feature_name[i])) {
505 *features |= 1 << i;
506 return;
509 fprintf(stderr, "CPU feature %s not found\n", flagname);
512 static int cpu_sparc_find_by_name(sparc_def_t *cpu_def, const char *cpu_model)
514 unsigned int i;
515 const sparc_def_t *def = NULL;
516 char *s = g_strdup(cpu_model);
517 char *featurestr, *name = strtok(s, ",");
518 uint32_t plus_features = 0;
519 uint32_t minus_features = 0;
520 uint64_t iu_version;
521 uint32_t fpu_version, mmu_version, nwindows;
523 for (i = 0; i < ARRAY_SIZE(sparc_defs); i++) {
524 if (strcasecmp(name, sparc_defs[i].name) == 0) {
525 def = &sparc_defs[i];
528 if (!def) {
529 goto error;
531 memcpy(cpu_def, def, sizeof(*def));
533 featurestr = strtok(NULL, ",");
534 while (featurestr) {
535 char *val;
537 if (featurestr[0] == '+') {
538 add_flagname_to_bitmaps(featurestr + 1, &plus_features);
539 } else if (featurestr[0] == '-') {
540 add_flagname_to_bitmaps(featurestr + 1, &minus_features);
541 } else if ((val = strchr(featurestr, '='))) {
542 *val = 0; val++;
543 if (!strcmp(featurestr, "iu_version")) {
544 char *err;
546 iu_version = strtoll(val, &err, 0);
547 if (!*val || *err) {
548 fprintf(stderr, "bad numerical value %s\n", val);
549 goto error;
551 cpu_def->iu_version = iu_version;
552 #ifdef DEBUG_FEATURES
553 fprintf(stderr, "iu_version %" PRIx64 "\n", iu_version);
554 #endif
555 } else if (!strcmp(featurestr, "fpu_version")) {
556 char *err;
558 fpu_version = strtol(val, &err, 0);
559 if (!*val || *err) {
560 fprintf(stderr, "bad numerical value %s\n", val);
561 goto error;
563 cpu_def->fpu_version = fpu_version;
564 #ifdef DEBUG_FEATURES
565 fprintf(stderr, "fpu_version %x\n", fpu_version);
566 #endif
567 } else if (!strcmp(featurestr, "mmu_version")) {
568 char *err;
570 mmu_version = strtol(val, &err, 0);
571 if (!*val || *err) {
572 fprintf(stderr, "bad numerical value %s\n", val);
573 goto error;
575 cpu_def->mmu_version = mmu_version;
576 #ifdef DEBUG_FEATURES
577 fprintf(stderr, "mmu_version %x\n", mmu_version);
578 #endif
579 } else if (!strcmp(featurestr, "nwindows")) {
580 char *err;
582 nwindows = strtol(val, &err, 0);
583 if (!*val || *err || nwindows > MAX_NWINDOWS ||
584 nwindows < MIN_NWINDOWS) {
585 fprintf(stderr, "bad numerical value %s\n", val);
586 goto error;
588 cpu_def->nwindows = nwindows;
589 #ifdef DEBUG_FEATURES
590 fprintf(stderr, "nwindows %d\n", nwindows);
591 #endif
592 } else {
593 fprintf(stderr, "unrecognized feature %s\n", featurestr);
594 goto error;
596 } else {
597 fprintf(stderr, "feature string `%s' not in format "
598 "(+feature|-feature|feature=xyz)\n", featurestr);
599 goto error;
601 featurestr = strtok(NULL, ",");
603 cpu_def->features |= plus_features;
604 cpu_def->features &= ~minus_features;
605 #ifdef DEBUG_FEATURES
606 print_features(stderr, fprintf, cpu_def->features, NULL);
607 #endif
608 g_free(s);
609 return 0;
611 error:
612 g_free(s);
613 return -1;
616 void sparc_cpu_list(FILE *f, fprintf_function cpu_fprintf)
618 unsigned int i;
620 for (i = 0; i < ARRAY_SIZE(sparc_defs); i++) {
621 (*cpu_fprintf)(f, "Sparc %16s IU " TARGET_FMT_lx
622 " FPU %08x MMU %08x NWINS %d ",
623 sparc_defs[i].name,
624 sparc_defs[i].iu_version,
625 sparc_defs[i].fpu_version,
626 sparc_defs[i].mmu_version,
627 sparc_defs[i].nwindows);
628 print_features(f, cpu_fprintf, CPU_DEFAULT_FEATURES &
629 ~sparc_defs[i].features, "-");
630 print_features(f, cpu_fprintf, ~CPU_DEFAULT_FEATURES &
631 sparc_defs[i].features, "+");
632 (*cpu_fprintf)(f, "\n");
634 (*cpu_fprintf)(f, "Default CPU feature flags (use '-' to remove): ");
635 print_features(f, cpu_fprintf, CPU_DEFAULT_FEATURES, NULL);
636 (*cpu_fprintf)(f, "\n");
637 (*cpu_fprintf)(f, "Available CPU feature flags (use '+' to add): ");
638 print_features(f, cpu_fprintf, ~CPU_DEFAULT_FEATURES, NULL);
639 (*cpu_fprintf)(f, "\n");
640 (*cpu_fprintf)(f, "Numerical features (use '=' to set): iu_version "
641 "fpu_version mmu_version nwindows\n");
644 static void cpu_print_cc(FILE *f, fprintf_function cpu_fprintf,
645 uint32_t cc)
647 cpu_fprintf(f, "%c%c%c%c", cc & PSR_NEG ? 'N' : '-',
648 cc & PSR_ZERO ? 'Z' : '-', cc & PSR_OVF ? 'V' : '-',
649 cc & PSR_CARRY ? 'C' : '-');
652 #ifdef TARGET_SPARC64
653 #define REGS_PER_LINE 4
654 #else
655 #define REGS_PER_LINE 8
656 #endif
658 void sparc_cpu_dump_state(CPUState *cs, FILE *f, fprintf_function cpu_fprintf,
659 int flags)
661 SPARCCPU *cpu = SPARC_CPU(cs);
662 CPUSPARCState *env = &cpu->env;
663 int i, x;
665 cpu_fprintf(f, "pc: " TARGET_FMT_lx " npc: " TARGET_FMT_lx "\n", env->pc,
666 env->npc);
668 for (i = 0; i < 8; i++) {
669 if (i % REGS_PER_LINE == 0) {
670 cpu_fprintf(f, "%%g%d-%d:", i, i + REGS_PER_LINE - 1);
672 cpu_fprintf(f, " " TARGET_FMT_lx, env->gregs[i]);
673 if (i % REGS_PER_LINE == REGS_PER_LINE - 1) {
674 cpu_fprintf(f, "\n");
677 for (x = 0; x < 3; x++) {
678 for (i = 0; i < 8; i++) {
679 if (i % REGS_PER_LINE == 0) {
680 cpu_fprintf(f, "%%%c%d-%d: ",
681 x == 0 ? 'o' : (x == 1 ? 'l' : 'i'),
682 i, i + REGS_PER_LINE - 1);
684 cpu_fprintf(f, TARGET_FMT_lx " ", env->regwptr[i + x * 8]);
685 if (i % REGS_PER_LINE == REGS_PER_LINE - 1) {
686 cpu_fprintf(f, "\n");
691 for (i = 0; i < TARGET_DPREGS; i++) {
692 if ((i & 3) == 0) {
693 cpu_fprintf(f, "%%f%02d: ", i * 2);
695 cpu_fprintf(f, " %016" PRIx64, env->fpr[i].ll);
696 if ((i & 3) == 3) {
697 cpu_fprintf(f, "\n");
700 #ifdef TARGET_SPARC64
701 cpu_fprintf(f, "pstate: %08x ccr: %02x (icc: ", env->pstate,
702 (unsigned)cpu_get_ccr(env));
703 cpu_print_cc(f, cpu_fprintf, cpu_get_ccr(env) << PSR_CARRY_SHIFT);
704 cpu_fprintf(f, " xcc: ");
705 cpu_print_cc(f, cpu_fprintf, cpu_get_ccr(env) << (PSR_CARRY_SHIFT - 4));
706 cpu_fprintf(f, ") asi: %02x tl: %d pil: %x\n", env->asi, env->tl,
707 env->psrpil);
708 cpu_fprintf(f, "cansave: %d canrestore: %d otherwin: %d wstate: %d "
709 "cleanwin: %d cwp: %d\n",
710 env->cansave, env->canrestore, env->otherwin, env->wstate,
711 env->cleanwin, env->nwindows - 1 - env->cwp);
712 cpu_fprintf(f, "fsr: " TARGET_FMT_lx " y: " TARGET_FMT_lx " fprs: "
713 TARGET_FMT_lx "\n", env->fsr, env->y, env->fprs);
714 #else
715 cpu_fprintf(f, "psr: %08x (icc: ", cpu_get_psr(env));
716 cpu_print_cc(f, cpu_fprintf, cpu_get_psr(env));
717 cpu_fprintf(f, " SPE: %c%c%c) wim: %08x\n", env->psrs ? 'S' : '-',
718 env->psrps ? 'P' : '-', env->psret ? 'E' : '-',
719 env->wim);
720 cpu_fprintf(f, "fsr: " TARGET_FMT_lx " y: " TARGET_FMT_lx "\n",
721 env->fsr, env->y);
722 #endif
723 cpu_fprintf(f, "\n");
726 static void sparc_cpu_set_pc(CPUState *cs, vaddr value)
728 SPARCCPU *cpu = SPARC_CPU(cs);
730 cpu->env.pc = value;
731 cpu->env.npc = value + 4;
734 static void sparc_cpu_synchronize_from_tb(CPUState *cs, TranslationBlock *tb)
736 SPARCCPU *cpu = SPARC_CPU(cs);
738 cpu->env.pc = tb->pc;
739 cpu->env.npc = tb->cs_base;
742 static void sparc_cpu_realizefn(DeviceState *dev, Error **errp)
744 SPARCCPUClass *scc = SPARC_CPU_GET_CLASS(dev);
746 qemu_init_vcpu(CPU(dev));
748 scc->parent_realize(dev, errp);
751 static void sparc_cpu_initfn(Object *obj)
753 CPUState *cs = CPU(obj);
754 SPARCCPU *cpu = SPARC_CPU(obj);
755 CPUSPARCState *env = &cpu->env;
757 cs->env_ptr = env;
758 cpu_exec_init(env);
760 if (tcg_enabled()) {
761 gen_intermediate_code_init(env);
765 static void sparc_cpu_uninitfn(Object *obj)
767 SPARCCPU *cpu = SPARC_CPU(obj);
768 CPUSPARCState *env = &cpu->env;
770 g_free(env->def);
773 static void sparc_cpu_class_init(ObjectClass *oc, void *data)
775 SPARCCPUClass *scc = SPARC_CPU_CLASS(oc);
776 CPUClass *cc = CPU_CLASS(oc);
777 DeviceClass *dc = DEVICE_CLASS(oc);
779 scc->parent_realize = dc->realize;
780 dc->realize = sparc_cpu_realizefn;
782 scc->parent_reset = cc->reset;
783 cc->reset = sparc_cpu_reset;
785 cc->do_interrupt = sparc_cpu_do_interrupt;
786 cc->dump_state = sparc_cpu_dump_state;
787 #if !defined(TARGET_SPARC64) && !defined(CONFIG_USER_ONLY)
788 cc->memory_rw_debug = sparc_cpu_memory_rw_debug;
789 #endif
790 cc->set_pc = sparc_cpu_set_pc;
791 cc->synchronize_from_tb = sparc_cpu_synchronize_from_tb;
792 cc->gdb_read_register = sparc_cpu_gdb_read_register;
793 cc->gdb_write_register = sparc_cpu_gdb_write_register;
794 #ifndef CONFIG_USER_ONLY
795 cc->do_unassigned_access = sparc_cpu_unassigned_access;
796 cc->get_phys_page_debug = sparc_cpu_get_phys_page_debug;
797 #endif
799 #if defined(TARGET_SPARC64) && !defined(TARGET_ABI32)
800 cc->gdb_num_core_regs = 86;
801 #else
802 cc->gdb_num_core_regs = 72;
803 #endif
806 static const TypeInfo sparc_cpu_type_info = {
807 .name = TYPE_SPARC_CPU,
808 .parent = TYPE_CPU,
809 .instance_size = sizeof(SPARCCPU),
810 .instance_init = sparc_cpu_initfn,
811 .instance_finalize = sparc_cpu_uninitfn,
812 .abstract = false,
813 .class_size = sizeof(SPARCCPUClass),
814 .class_init = sparc_cpu_class_init,
817 static void sparc_cpu_register_types(void)
819 type_register_static(&sparc_cpu_type_info);
822 type_init(sparc_cpu_register_types)