2008-05-04 21:11:44 +08:00
|
|
|
#include "hw/hw.h"
|
|
|
|
#include "hw/boards.h"
|
2012-12-18 01:20:00 +08:00
|
|
|
#include "qemu/timer.h"
|
2008-05-04 21:11:44 +08:00
|
|
|
|
2011-06-20 04:38:22 +08:00
|
|
|
#include "cpu.h"
|
2008-05-04 21:11:44 +08:00
|
|
|
|
|
|
|
void cpu_save(QEMUFile *f, void *opaque)
|
|
|
|
{
|
2012-03-14 08:38:22 +08:00
|
|
|
CPUSPARCState *env = opaque;
|
2008-05-04 21:11:44 +08:00
|
|
|
int i;
|
|
|
|
uint32_t tmp;
|
|
|
|
|
2008-08-01 23:13:58 +08:00
|
|
|
// if env->cwp == env->nwindows - 1, this will set the ins of the last
|
|
|
|
// window as the outs of the first window
|
|
|
|
cpu_set_cwp(env, env->cwp);
|
|
|
|
|
2008-05-04 21:11:44 +08:00
|
|
|
for(i = 0; i < 8; i++)
|
|
|
|
qemu_put_betls(f, &env->gregs[i]);
|
2008-06-07 16:07:37 +08:00
|
|
|
qemu_put_be32s(f, &env->nwindows);
|
|
|
|
for(i = 0; i < env->nwindows * 16; i++)
|
2008-05-04 21:11:44 +08:00
|
|
|
qemu_put_betls(f, &env->regbase[i]);
|
|
|
|
|
|
|
|
/* FPU */
|
2011-10-18 01:42:49 +08:00
|
|
|
for (i = 0; i < TARGET_DPREGS; i++) {
|
|
|
|
qemu_put_be32(f, env->fpr[i].l.upper);
|
|
|
|
qemu_put_be32(f, env->fpr[i].l.lower);
|
2008-05-04 21:11:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
qemu_put_betls(f, &env->pc);
|
|
|
|
qemu_put_betls(f, &env->npc);
|
|
|
|
qemu_put_betls(f, &env->y);
|
2010-05-10 04:19:04 +08:00
|
|
|
tmp = cpu_get_psr(env);
|
2008-05-04 21:11:44 +08:00
|
|
|
qemu_put_be32(f, tmp);
|
|
|
|
qemu_put_betls(f, &env->fsr);
|
|
|
|
qemu_put_betls(f, &env->tbr);
|
2008-08-01 23:13:58 +08:00
|
|
|
tmp = env->interrupt_index;
|
|
|
|
qemu_put_be32(f, tmp);
|
|
|
|
qemu_put_be32s(f, &env->pil_in);
|
2008-05-04 21:11:44 +08:00
|
|
|
#ifndef TARGET_SPARC64
|
|
|
|
qemu_put_be32s(f, &env->wim);
|
|
|
|
/* MMU */
|
2008-07-24 19:28:51 +08:00
|
|
|
for (i = 0; i < 32; i++)
|
2008-05-04 21:11:44 +08:00
|
|
|
qemu_put_be32s(f, &env->mmuregs[i]);
|
2011-06-19 04:27:05 +08:00
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
qemu_put_be64s(f, &env->mxccdata[i]);
|
|
|
|
}
|
|
|
|
for (i = 0; i < 8; i++) {
|
|
|
|
qemu_put_be64s(f, &env->mxccregs[i]);
|
|
|
|
}
|
|
|
|
qemu_put_be32s(f, &env->mmubpctrv);
|
|
|
|
qemu_put_be32s(f, &env->mmubpctrc);
|
|
|
|
qemu_put_be32s(f, &env->mmubpctrs);
|
|
|
|
qemu_put_be64s(f, &env->mmubpaction);
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
qemu_put_be64s(f, &env->mmubpregs[i]);
|
|
|
|
}
|
2008-07-24 19:28:51 +08:00
|
|
|
#else
|
|
|
|
qemu_put_be64s(f, &env->lsu);
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
qemu_put_be64s(f, &env->immuregs[i]);
|
|
|
|
qemu_put_be64s(f, &env->dmmuregs[i]);
|
|
|
|
}
|
|
|
|
for (i = 0; i < 64; i++) {
|
2009-07-27 05:49:04 +08:00
|
|
|
qemu_put_be64s(f, &env->itlb[i].tag);
|
|
|
|
qemu_put_be64s(f, &env->itlb[i].tte);
|
|
|
|
qemu_put_be64s(f, &env->dtlb[i].tag);
|
|
|
|
qemu_put_be64s(f, &env->dtlb[i].tte);
|
2008-07-24 19:28:51 +08:00
|
|
|
}
|
|
|
|
qemu_put_be32s(f, &env->mmu_version);
|
2008-07-25 15:42:14 +08:00
|
|
|
for (i = 0; i < MAXTL_MAX; i++) {
|
2008-07-24 19:28:51 +08:00
|
|
|
qemu_put_be64s(f, &env->ts[i].tpc);
|
|
|
|
qemu_put_be64s(f, &env->ts[i].tnpc);
|
|
|
|
qemu_put_be64s(f, &env->ts[i].tstate);
|
|
|
|
qemu_put_be32s(f, &env->ts[i].tt);
|
|
|
|
}
|
|
|
|
qemu_put_be32s(f, &env->xcc);
|
|
|
|
qemu_put_be32s(f, &env->asi);
|
|
|
|
qemu_put_be32s(f, &env->pstate);
|
|
|
|
qemu_put_be32s(f, &env->tl);
|
|
|
|
qemu_put_be32s(f, &env->cansave);
|
|
|
|
qemu_put_be32s(f, &env->canrestore);
|
|
|
|
qemu_put_be32s(f, &env->otherwin);
|
|
|
|
qemu_put_be32s(f, &env->wstate);
|
|
|
|
qemu_put_be32s(f, &env->cleanwin);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_put_be64s(f, &env->agregs[i]);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_put_be64s(f, &env->bgregs[i]);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_put_be64s(f, &env->igregs[i]);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_put_be64s(f, &env->mgregs[i]);
|
|
|
|
qemu_put_be64s(f, &env->fprs);
|
|
|
|
qemu_put_be64s(f, &env->tick_cmpr);
|
|
|
|
qemu_put_be64s(f, &env->stick_cmpr);
|
2010-01-28 05:00:53 +08:00
|
|
|
cpu_put_timer(f, env->tick);
|
|
|
|
cpu_put_timer(f, env->stick);
|
2008-07-24 19:28:51 +08:00
|
|
|
qemu_put_be64s(f, &env->gsr);
|
|
|
|
qemu_put_be32s(f, &env->gl);
|
|
|
|
qemu_put_be64s(f, &env->hpstate);
|
2008-07-25 15:42:14 +08:00
|
|
|
for (i = 0; i < MAXTL_MAX; i++)
|
2008-07-24 19:28:51 +08:00
|
|
|
qemu_put_be64s(f, &env->htstate[i]);
|
|
|
|
qemu_put_be64s(f, &env->hintp);
|
|
|
|
qemu_put_be64s(f, &env->htba);
|
|
|
|
qemu_put_be64s(f, &env->hver);
|
|
|
|
qemu_put_be64s(f, &env->hstick_cmpr);
|
|
|
|
qemu_put_be64s(f, &env->ssr);
|
2010-01-28 05:00:53 +08:00
|
|
|
cpu_put_timer(f, env->hstick);
|
2008-05-04 21:11:44 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
int cpu_load(QEMUFile *f, void *opaque, int version_id)
|
|
|
|
{
|
2012-03-14 08:38:22 +08:00
|
|
|
CPUSPARCState *env = opaque;
|
2008-05-04 21:11:44 +08:00
|
|
|
int i;
|
|
|
|
uint32_t tmp;
|
|
|
|
|
2010-01-28 05:00:53 +08:00
|
|
|
if (version_id < 6)
|
2008-06-07 16:07:37 +08:00
|
|
|
return -EINVAL;
|
2008-05-04 21:11:44 +08:00
|
|
|
for(i = 0; i < 8; i++)
|
|
|
|
qemu_get_betls(f, &env->gregs[i]);
|
2008-06-07 16:07:37 +08:00
|
|
|
qemu_get_be32s(f, &env->nwindows);
|
|
|
|
for(i = 0; i < env->nwindows * 16; i++)
|
2008-05-04 21:11:44 +08:00
|
|
|
qemu_get_betls(f, &env->regbase[i]);
|
|
|
|
|
|
|
|
/* FPU */
|
2011-10-18 01:42:49 +08:00
|
|
|
for (i = 0; i < TARGET_DPREGS; i++) {
|
|
|
|
env->fpr[i].l.upper = qemu_get_be32(f);
|
|
|
|
env->fpr[i].l.lower = qemu_get_be32(f);
|
2008-05-04 21:11:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
qemu_get_betls(f, &env->pc);
|
|
|
|
qemu_get_betls(f, &env->npc);
|
|
|
|
qemu_get_betls(f, &env->y);
|
|
|
|
tmp = qemu_get_be32(f);
|
|
|
|
env->cwp = 0; /* needed to ensure that the wrapping registers are
|
|
|
|
correctly updated */
|
2010-05-10 04:19:04 +08:00
|
|
|
cpu_put_psr(env, tmp);
|
2008-05-04 21:11:44 +08:00
|
|
|
qemu_get_betls(f, &env->fsr);
|
|
|
|
qemu_get_betls(f, &env->tbr);
|
2008-08-01 23:13:58 +08:00
|
|
|
tmp = qemu_get_be32(f);
|
|
|
|
env->interrupt_index = tmp;
|
|
|
|
qemu_get_be32s(f, &env->pil_in);
|
2008-05-04 21:11:44 +08:00
|
|
|
#ifndef TARGET_SPARC64
|
|
|
|
qemu_get_be32s(f, &env->wim);
|
|
|
|
/* MMU */
|
2008-07-24 19:28:51 +08:00
|
|
|
for (i = 0; i < 32; i++)
|
2008-05-04 21:11:44 +08:00
|
|
|
qemu_get_be32s(f, &env->mmuregs[i]);
|
2011-06-19 04:27:05 +08:00
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
qemu_get_be64s(f, &env->mxccdata[i]);
|
|
|
|
}
|
|
|
|
for (i = 0; i < 8; i++) {
|
|
|
|
qemu_get_be64s(f, &env->mxccregs[i]);
|
|
|
|
}
|
|
|
|
qemu_get_be32s(f, &env->mmubpctrv);
|
|
|
|
qemu_get_be32s(f, &env->mmubpctrc);
|
|
|
|
qemu_get_be32s(f, &env->mmubpctrs);
|
|
|
|
qemu_get_be64s(f, &env->mmubpaction);
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
qemu_get_be64s(f, &env->mmubpregs[i]);
|
|
|
|
}
|
2008-07-24 19:28:51 +08:00
|
|
|
#else
|
|
|
|
qemu_get_be64s(f, &env->lsu);
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
qemu_get_be64s(f, &env->immuregs[i]);
|
|
|
|
qemu_get_be64s(f, &env->dmmuregs[i]);
|
|
|
|
}
|
|
|
|
for (i = 0; i < 64; i++) {
|
2009-07-27 05:49:04 +08:00
|
|
|
qemu_get_be64s(f, &env->itlb[i].tag);
|
|
|
|
qemu_get_be64s(f, &env->itlb[i].tte);
|
|
|
|
qemu_get_be64s(f, &env->dtlb[i].tag);
|
|
|
|
qemu_get_be64s(f, &env->dtlb[i].tte);
|
2008-07-24 19:28:51 +08:00
|
|
|
}
|
|
|
|
qemu_get_be32s(f, &env->mmu_version);
|
2008-07-25 15:42:14 +08:00
|
|
|
for (i = 0; i < MAXTL_MAX; i++) {
|
2008-07-24 19:28:51 +08:00
|
|
|
qemu_get_be64s(f, &env->ts[i].tpc);
|
|
|
|
qemu_get_be64s(f, &env->ts[i].tnpc);
|
|
|
|
qemu_get_be64s(f, &env->ts[i].tstate);
|
|
|
|
qemu_get_be32s(f, &env->ts[i].tt);
|
|
|
|
}
|
|
|
|
qemu_get_be32s(f, &env->xcc);
|
|
|
|
qemu_get_be32s(f, &env->asi);
|
|
|
|
qemu_get_be32s(f, &env->pstate);
|
|
|
|
qemu_get_be32s(f, &env->tl);
|
|
|
|
qemu_get_be32s(f, &env->cansave);
|
|
|
|
qemu_get_be32s(f, &env->canrestore);
|
|
|
|
qemu_get_be32s(f, &env->otherwin);
|
|
|
|
qemu_get_be32s(f, &env->wstate);
|
|
|
|
qemu_get_be32s(f, &env->cleanwin);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_get_be64s(f, &env->agregs[i]);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_get_be64s(f, &env->bgregs[i]);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_get_be64s(f, &env->igregs[i]);
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
qemu_get_be64s(f, &env->mgregs[i]);
|
|
|
|
qemu_get_be64s(f, &env->fprs);
|
|
|
|
qemu_get_be64s(f, &env->tick_cmpr);
|
|
|
|
qemu_get_be64s(f, &env->stick_cmpr);
|
2010-01-28 05:00:53 +08:00
|
|
|
cpu_get_timer(f, env->tick);
|
|
|
|
cpu_get_timer(f, env->stick);
|
2008-07-24 19:28:51 +08:00
|
|
|
qemu_get_be64s(f, &env->gsr);
|
|
|
|
qemu_get_be32s(f, &env->gl);
|
|
|
|
qemu_get_be64s(f, &env->hpstate);
|
2008-07-25 15:42:14 +08:00
|
|
|
for (i = 0; i < MAXTL_MAX; i++)
|
2008-07-24 19:28:51 +08:00
|
|
|
qemu_get_be64s(f, &env->htstate[i]);
|
|
|
|
qemu_get_be64s(f, &env->hintp);
|
|
|
|
qemu_get_be64s(f, &env->htba);
|
|
|
|
qemu_get_be64s(f, &env->hver);
|
|
|
|
qemu_get_be64s(f, &env->hstick_cmpr);
|
|
|
|
qemu_get_be64s(f, &env->ssr);
|
2010-01-28 05:00:53 +08:00
|
|
|
cpu_get_timer(f, env->hstick);
|
2008-05-04 21:11:44 +08:00
|
|
|
#endif
|
|
|
|
tlb_flush(env, 1);
|
|
|
|
return 0;
|
|
|
|
}
|