binutils-gdb/sim/h8500/compile.c

2520 lines
53 KiB
C

/* Simulator for the Hitachi H8/500 architecture.
Written by Steve Chamberlain of Cygnus Support.
sac@cygnus.com
This file is part of H8/500 sim
THIS SOFTWARE IS NOT COPYRIGHTED
Cygnus offers the following for use in the public domain. Cygnus
makes no warranty with regard to the software or it's performance
and the user accepts the software "AS IS" with all faults.
CYGNUS DISCLAIMS ANY WARRANTIES, EXPRESS OR IMPLIED, WITH REGARD TO
THIS SOFTWARE INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
*/
#include "config.h"
#include <signal.h>
#ifdef HAVE_STDLIB_H
#include <stdlib.h>
#endif
#ifdef HAVE_TIME_H
#include <time.h>
#endif
#include <sys/param.h>
#include <setjmp.h>
#include "ansidecl.h"
#include "bfd.h"
#include "gdb/callback.h"
#include "gdb/remote-sim.h"
#define O_RECOMPILE 85
#define DEFINE_TABLE
#define DISASSEMBLER_TABLE
/* FIXME: Needs to live in header file.
This header should also include the things in remote-sim.h.
One could move this to remote-sim.h but this function isn't needed
by gdb. */
void sim_set_simcache_size PARAMS ((int));
int debug;
host_callback *sim_callback;
static SIM_OPEN_KIND sim_kind;
static char *myname;
/* This code can be compiled with any old C compiler, in which case
four or five switch statements will be executed for each
instruction simulated. It can be compiled with GCC, then the
simulated instructions thread through the code fragments, and
everything goes much faster.
These definitions make the code work either way
*/
#ifdef __GNUC__
#define DISPATCH(X) goto *(X); do
#define LABEL(X) X##_L
#define LABELN(X,N) X##_L##N
#define LABEL_REF(X) &&X##_L
#define LABEL_REFN(X,N) &&X##_L##N
#define ENDDISPATCH while (0);
#define fastref void *
#define DEFAULT ;
#define INLINE __inline__
#else
#define DEFAULT default :
#define DISPATCH(X) switch (X)
#define LABEL(X) case X
#define LABELN(X,N) case X
#define LABEL_REF(X) X
#define LABEL_REFN(X,N) X
#define ENDDISPATCH
#define fastref int
#define INLINE
#define STORE_REG_B 1
#define STORE_REG_W 2
#define STORE_INC_B 3
#define STORE_INC_W 4
#define STORE_DEC_B 5
#define STORE_DEC_W 6
#define STORE_DISP_B 7
#define STORE_DISP_W 8
#define STORE_CRB 9
#define STORE_CRW 10
#define STORE_REG_L 11
#define STORE_NOP 12
#define FETCH_NOP 9
#define FETCH_REG_B 10
#define FETCH_REG_W 11
#define FETCH_INC_B 12
#define FETCH_INC_W 13
#define FETCH_DEC_B 14
#define FETCH_DEC_W 15
#define FETCH_DISP_B 16
#define FETCH_DISP_W 17
#define FETCH_IMM 18
#define FETCH_CRB 19
#define FETCH_CRW 20
#define FETCH_LVAL 21
#define FETCH_LVAL24 22
#define FETCH_REG_L 23
#define FLAG_m 20
#define FLAG_M 21
#define FLAG_A 22
#define FLAG_NONE 23
#define FLAG_NOSTORE 24
#define FLAG_CLEAR 25
#define FLAG_a 26
#define FLAG_BRANCH 27
#define FLAG_special 28
#define FLAG_shiftword 29
#define FLAG_shiftbyte 30
#define FLAG_multword 31
#define FLAG_multbyte 32
#endif
#define h8500_table h8500_compile_table
#include "../opcodes/h8500-opc.h"
#include "inst.h"
#define LOW_BYTE(x) ((x) & 0xff)
#define HIGH_BYTE(x) (((x)>>8) & 0xff)
#define NORMAL_CP ((cpu.regs[R_CP].c - cpu.memory)>>16)
#define NORMAL_DP ((cpu.regs[R_DP].c - cpu.memory)>>16)
#define NORMAL_EP ((cpu.regs[R_EP].c - cpu.memory)>>16)
#define NORMAL_TP ((cpu.regs[R_TP].c - cpu.memory)>>16)
#define SET_NORMREG(x,y) ((cpu.regs[x].l = (y)))
#define GET_NORMREG(x) (cpu.regs[x].l)
#define SET_SEGREG(x,y) { cpu.regs[x].c = ((y) & 0xff0000) + cpu.memory;}
#define GET_SEGREG(x) ( (cpu.regs[x].c - cpu.memory ) >> 16)
#define SET_NORMAL_CPPC(x) { pc = (x) & 0xffff; SET_SEGREG(R_CP, (x));}
#define NORMAL_SR ((N<<3)|(Z<<2)|(V<<1)|(C))
#define P(X,Y) ((X<<8) | Y)
#define BUILDSR() cpu.regs[R_SR].s[LOW] = (N << 3) | (Z << 2) | (V<<1) | C;
#define GETSR() \
C = (cpu.regs[R_SR].s[LOW] >> 0) & 1;\
V = (cpu.regs[R_SR].s[LOW] >> 1) & 1;\
Z = (cpu.regs[R_SR].s[LOW] >> 2) & 1;\
N = (cpu.regs[R_SR].s[LOW] >> 3) & 1;
#ifdef __CHAR_IS_SIGNED__
#define SEXTCHAR(x) ((char)(x))
#endif
#ifndef SEXTCHAR
#define SEXTCHAR(x) ((x & 0x80) ? (x | ~0xff):x)
#endif
#define SEXTSHORT(x) ((short)(x))
/* Which segment registers go with which pointer registers */
static unsigned char **segmap[R_LAST];
static unsigned char *(regptr[R_LAST][3]);
static unsigned char *(segregptr[R_LAST][3]);
static cpu_state_type cpu;
static int segforreg[] = {R_DP, R_DP, R_DP, R_DP,
R_EP, R_EP, R_TP, R_TP,
R_DP, R_DP, R_DP, R_DP,
R_EP, R_EP, R_TP, R_TP};
int LOW;
int HIGH;
/* routines for getting and storing args */
#define elval(struct, lit) \
(((*(struct.reg.wptr) + lit) & 0xffff) + (*(struct.r2.segreg)))
#define displval(s) elval((s),(s).literal)
#define ireglval(struct) elval(struct, 0)
#define wordat(x) (((x)[0] << 8) | (x)[1])
#define longat(x) ((wordat((x))<<16)|(wordat((x)+2)))
#define byteat(x) ((x)[0])
#define setwordat(x,y) {x[0] =( y)>>8; x[1] = y;}
#define setbyteat(x,y) {x[0] = y;}
/*#define setalignedwordat(x,y) {((short *)x)[0] =y;}*/
/*
statics
*/
ea_type rd;
ea_type rs;
ea_type imm;
ea_type cr;
ea_type ea;
ea_type nop;
ea_type lval;
ea_type lval24;
ea_type eavector[2];
int disp;
#define JBYTE 0
#define JWORD 1
#define JLONG 2
typedef union
{
struct
{
fastref srcabyte;
fastref srcaword;
fastref srcalong;
fastref srcbbyte;
fastref srcbword;
fastref srcblong;
fastref dstbyte;
fastref dstword;
fastref dstlong;
} s;
struct
{
fastref byte;
fastref word;
fastref lon;
} a[3];
fastref j[9];
} size_ptr;
union
{
struct ea_struct
{
size_ptr ea_nop;
size_ptr ea_reg;
size_ptr ea_inc;
size_ptr ea_dec;
size_ptr ea_disp;
size_ptr ea_imm;
size_ptr ea_cr;
size_ptr ea_lval;
size_ptr ea_lval24;
} s;
#define N_EATYPES (sizeof(struct ea_struct) / sizeof(size_ptr))
size_ptr a[N_EATYPES];
} eas;
/* This function takes an ea structure filled in for the 1st source
operand and modifies it to be for either the 1st, 2nd or dst operand */
static void
howto_workout (encoded, semiencoded, n)
ea_type *encoded;
ea_type *semiencoded;
int n;
{
int i;
*encoded = *semiencoded;
for (i = 0; i < N_EATYPES; i++)
{
if (encoded->type == eas.a[i].s.srcabyte)
{
encoded->type = eas.a[i].a[n].byte;
return;
}
else if (encoded->type == eas.a[i].s.srcaword)
{
encoded->type = eas.a[i].a[n].word;
return;
}
else if (encoded->type == eas.a[i].s.srcalong)
{
encoded->type = eas.a[i].a[n].lon;
return;
}
}
abort ();
}
fastref flag_shiftword;
fastref flag_shiftbyte;
fastref flag_multword;
fastref flag_multbyte;
fastref flag_mp;
fastref flag_special;
fastref flag_Mp;
fastref flag_ap;
fastref flag_Ap;
fastref flag_nonep;
fastref flag_nostorep;
fastref flag_clearp;
fastref flag_branch;
fastref exec_dispatch[100];
static int
get_now ()
{
return time (0);
}
static int
now_persec ()
{
return 1;
}
static void
gotcr (ptr, n)
ea_type *ptr;
int n;
{
int size;
n &= 0x7;
if (n == 0)
{
abort ();
}
else
{
ptr->type = eas.s.ea_cr.j[JBYTE];
ptr->reg.bptr = segregptr[n][JLONG];
}
}
static void
gotreg (ptr, n, size)
ea_type *ptr;
int n;
int size;
{
n &= 0x7;
ptr->type = eas.s.ea_reg.j[size];
ptr->reg.bptr = regptr[n][size];
}
static void
gotinc (ptr, n, inc, size)
ea_type *ptr;
int n;
int size;
{
n &= 0x7;
if (inc > 0)
{
ptr->type = eas.s.ea_inc.j[size];
}
else
{
ptr->type = eas.s.ea_dec.j[size];
}
ptr->reg.bptr = regptr[n][JWORD];
ptr->r2.segreg = segmap[n];
}
static void
gotabs (ptr, disp, reg, size)
ea_type *ptr;
int disp;
int reg;
int size;
{
ptr->type = eas.s.ea_disp.j[size];
ptr->reg.bptr = regptr[reg][JWORD];
ptr->r2.segreg = segmap[reg];
ptr->literal = disp;
}
static void
gotind (ptr, disp, reg, size)
ea_type *ptr;
int disp;
int reg;
int size;
{
gotabs (ptr, disp, reg & 0x7, size);
}
static void
gotimm (ptr, val)
ea_type *ptr;
int val;
{
ptr->type = eas.s.ea_imm.j[0];
ptr->literal = val;
}
static void
indoff (ptr)
ea_type *ptr;
{
int i;
for (i = 0; i < 6; i++)
{
if (ptr->type == eas.s.ea_disp.j[i])
{
ptr->type = eas.s.ea_lval.j[i];
return;
}
}
}
thinkabout_shifts (d, bytesized)
decoded_inst *d;
int bytesized;
{
if (bytesized)
{
/* Got a byte shift, fake up second arg */
d->srcb.type = eas.s.ea_imm.s.srcbword;
d->srcb.literal = 8;
}
else
{
/* got a word shift, fake up second arg */
d->srcb.type = eas.s.ea_imm.s.srcbword;
d->srcb.literal = 16;
}
}
/* Calculate the number of cycles required to run this
instruction
*/
static void
compcycles (dst, opcode)
decoded_inst *dst;
h8500_opcode_info *opcode;
{
int cycles = 0;
/* Guess for the time being - 1 cycle for the first two bytes in the
opcode - to fecth the operand, and 3 cycles for all the rest of
the bytes, since they mean that there is probably an operand to
fetch */
switch (opcode->length)
{
case 1:
case 2:
cycles += opcode->length;
break;
default:
cycles += opcode->length * 3;
break;
}
dst->cycles = cycles;
}
static void
translate (ptr, from, to)
ea_type *ptr;
fastref from;
fastref to;
{
if (ptr->reg.wptr == &cpu.regs[7].s[LOW]
&& ptr->type == from)
{
ptr->type = to;
}
}
static
void
fix_incdecs (dst)
decoded_inst *dst;
{
if (dst->dst.type == eas.s.ea_inc.s.dstbyte
&& (dst->srca.type == eas.s.ea_inc.s.srcabyte
|| dst->srcb.type == eas.s.ea_inc.s.srcbbyte))
{
dst->dst.type = eas.s.ea_disp.s.dstbyte;
}
if (dst->dst.type == eas.s.ea_inc.s.dstword
&& (dst->srca.type == eas.s.ea_inc.s.srcaword
|| dst->srcb.type == eas.s.ea_inc.s.srcbword))
{
dst->dst.type = eas.s.ea_disp.s.dstword;
}
if (dst->dst.type == eas.s.ea_dec.s.dstbyte
|| dst->dst.type == eas.s.ea_dec.s.dstword)
{
if (dst->srca.type == eas.s.ea_dec.s.srcabyte)
{
dst->srca.type = eas.s.ea_disp.s.srcabyte;
}
else if (dst->srca.type == eas.s.ea_dec.s.srcaword)
{
dst->srca.type = eas.s.ea_disp.s.srcaword;
}
else if (dst->srcb.type == eas.s.ea_dec.s.srcbbyte)
{
dst->srcb.type = eas.s.ea_disp.s.srcbbyte;
}
else if (dst->srcb.type == eas.s.ea_dec.s.srcbword)
{
dst->srcb.type = eas.s.ea_disp.s.srcbword;
}
}
/* Turn a byte ops from the sp into word ops */
translate (&dst->dst, eas.s.ea_dec.s.dstbyte, eas.s.ea_dec.s.dstword);
translate (&dst->dst, eas.s.ea_inc.s.dstbyte, eas.s.ea_inc.s.dstword);
translate (&dst->srca, eas.s.ea_dec.s.srcabyte, eas.s.ea_dec.s.srcaword);
translate (&dst->srca, eas.s.ea_inc.s.srcabyte, eas.s.ea_inc.s.srcaword);
translate (&dst->srcb, eas.s.ea_dec.s.srcbbyte, eas.s.ea_dec.s.srcbword);
translate (&dst->srcb, eas.s.ea_inc.s.srcbbyte, eas.s.ea_inc.s.srcbword);
}
static void
find (pc, buffer, dst)
int pc;
unsigned char *buffer;
decoded_inst *dst;
{
h8500_opcode_info *opcode;
int i;
int idx;
int hadimm = 0;
dst->srca.reg.rptr = 0;
/* Run down the table to find the one which matches */
for (opcode = h8500_table; opcode->name; opcode++)
{
int byte;
int rn;
int rd;
int rs;
int disp;
int abs;
int imm;
int pcrel;
int qim;
int i;
int cr;
dst->opcode = exec_dispatch[opcode->flavor & 0x7f];
for (byte = 0; byte < opcode->length; byte++)
{
if ((buffer[byte] & opcode->bytes[byte].mask)
!= (opcode->bytes[byte].contents))
{
goto next;
}
else
{
/* extract any info parts */
switch (opcode->bytes[byte].insert)
{
case 0:
case FP:
break;
default:
abort ();
break;
case RN:
rn = buffer[byte] & 0x7;
break;
case RS:
rs = buffer[byte] & 0x7;
break;
case CRB:
cr = buffer[byte] & 0x7;
if (cr == 0)
goto next;
break;
case CRW:
cr = buffer[byte] & 0x7;
if (cr != 0)
goto next;
break;
case DISP16:
disp = (buffer[byte] << 8) | (buffer[byte + 1]);
break;
case FPIND_D8:
case DISP8:
disp = ((char) (buffer[byte]));
break;
case RD:
case RDIND:
rd = buffer[byte] & 0x7;
break;
case ABS24:
abs =
(buffer[byte] << 16)
| (buffer[byte + 1] << 8)
| (buffer[byte + 2]);
break;
case ABS16:
abs = (buffer[byte] << 8) | (buffer[byte + 1]);
break;
case ABS8:
abs = (buffer[byte]);
break;
case IMM16:
imm = (buffer[byte] << 8) | (buffer[byte + 1]);
break;
case IMM4:
imm = (buffer[byte]) & 0xf;
break;
case IMM8:
case RLIST:
imm = SEXTCHAR (buffer[byte]);
break;
case PCREL16:
pcrel = SEXTSHORT ((buffer[byte] << 8) | (buffer[byte + 1]));
break;
case PCREL8:
pcrel = SEXTCHAR ((buffer[byte]));
break;
case QIM:
switch (buffer[byte] & 0x7)
{
case 0:
imm = 1;
break;
case 1:
imm = 2;
break;
case 4:
imm = -1;
break;
case 5:
imm = -2;
break;
}
break;
}
}
}
if (opcode->flavor & O_BYTE)
{
idx = 0;
switch (opcode->flags)
{
case 'h':
dst->flags = flag_shiftbyte;
break;
case 'p':
dst->flags = flag_multbyte;
break;
case 'B':
dst->flags = flag_branch;
break;
case 'm':
dst->flags = flag_mp;
break;
case 'a':
dst->flags = flag_ap;
break;
case '-':
dst->flags = flag_nonep;
break;
case 0:
dst->flags = flag_nostorep;
break;
case 'c':
dst->flags = flag_clearp;
break;
case 's':
/* special */
dst->flags = flag_special;
}
}
else
{
idx = 1;
switch (opcode->flags)
{
case 'h':
dst->flags = flag_shiftword;
break;
case 'p':
dst->flags = flag_multword;
break;
case 'B':
dst->flags = flag_branch;
break;
case 'm':
dst->flags = flag_Mp;
break;
case 'a':
dst->flags = flag_Ap;
break;
case '-':
dst->flags = flag_nonep;
break;
case 0:
dst->flags = flag_nostorep;
break;
case 'c':
dst->flags = flag_clearp;
break;
case 's':
/* special */
dst->flags = flag_special;
break;
}
}
for (i = 0; i < opcode->nargs; i++)
{
ea_type *p = eavector + i;
switch (opcode->arg_type[i])
{
default:
abort ();
case FP:
gotreg (p, 6, idx);
break;
case RNIND:
disp = 0;
case RNIND_D16:
case RNIND_D8:
gotind (p, disp, rn, idx);
break;
break;
case RDIND:
disp = 0;
case RDIND_D16:
case RDIND_D8:
gotind (p, disp, rd, idx);
break;
case FPIND_D8:
gotind (p, disp, 6, idx);
break;
case CRB:
case CRW:
gotcr (p, cr);
break;
case RN:
gotreg (p, rn, idx);
break;
case RD:
gotreg (p, rd, idx);
break;
case RS:
gotreg (p, rs, idx);
break;
case RNDEC:
gotinc (p, rn, -1, idx);
break;
case RNINC:
gotinc (p, rn, 1, idx);
break;
case SPINC:
gotinc (p, 7, 1, idx);
break;
case SPDEC:
gotinc (p, 7, -1, idx);
break;
case ABS24:
case ABS16:
gotabs (p, abs, R_HARD_0, idx);
break;
case ABS8:
gotabs (p, abs, R_HARD8_0, idx);
break;
case IMM16:
case RLIST:
case QIM:
case IMM4:
case IMM8:
gotimm (p, imm);
break;
case PCREL16:
case PCREL8:
gotimm (p,
((pcrel + pc + opcode->length) & 0xffff) | (pc & 0xff0000),
R_HARD_0, JLONG);
}
}
/* Finished and done - turn from two operand stuff into three */
dst->srca.type = eas.s.ea_nop.s.srcabyte;
dst->srcb.type = eas.s.ea_nop.s.srcbbyte;
dst->dst.type = eas.s.ea_nop.s.dstbyte;
if (opcode->nargs)
{
switch (opcode->nargs)
{
case 1:
howto_workout (&dst->srca, &eavector[0], 0);
if (opcode->dst != '!')
howto_workout (&dst->dst, &eavector[0], 2);
break;
case 2:
if (opcode->src2 == '!')
{
howto_workout (&dst->srca, &eavector[0], 0);
howto_workout (&dst->dst, &eavector[1], 2);
}
else
{
howto_workout (&dst->srca, &eavector[0], 0);
howto_workout (&dst->srcb, &eavector[1], 1);
if (opcode->dst != '!')
{
howto_workout (&dst->dst, &eavector[1], 2);
}
}
break;
}
/* Some extra stuff with pre inc and post dec,
make sure that if the same ea is there twice, only one of the
ops is auto inc/dec */
fix_incdecs (dst);
/* Some special cases */
if (dst->opcode == exec_dispatch[O_PJSR]
|| dst->opcode == exec_dispatch[O_PJMP])
{
/* Both the @abs:24 and @rn turn into a disp word,
chose the right a mode since @abs:24 is 4 bytes
long */
if (opcode->length == 4)
{
dst->srca.type = eas.s.ea_lval24.s.srcabyte;
}
else
{
dst->srca.type = eas.s.ea_reg.s.srcalong;
}
dst->srca.r2.rptr = &cpu.regs[R_HARD_0];
/* For [P]JSR, keep return address precomputed */
dst->srcb.literal = pc + opcode->length;
dst->srcb.type = eas.s.ea_imm.s.srcbword;
}
else if (dst->opcode == exec_dispatch[O_MULXU])
{
/* This is a multiply -fix the destination op */
if (dst->dst.type == eas.s.ea_reg.s.dstword)
{
dst->dst.type = eas.s.ea_reg.s.dstlong;
}
else
{
dst->dst.type = eas.s.ea_reg.s.dstword;
}
dst->dst.reg.bptr = regptr[rd][JWORD];
}
else if (dst->opcode == exec_dispatch[O_DIVXU])
{
/* This is a wider than normal, fix the source operand */
dst->srcb.type
= (dst->srcb.type == eas.s.ea_reg.s.srcbword)
? eas.s.ea_reg.s.srcblong
: eas.s.ea_reg.s.srcbword;
dst->dst.type
= (dst->dst.type == eas.s.ea_reg.s.dstword)
? eas.s.ea_reg.s.dstlong
: eas.s.ea_reg.s.dstword;
}
else if (dst->opcode == exec_dispatch[O_LDM])
{
/* Turn of the stack ref */
dst->srca.type = eas.s.ea_nop.s.srcabyte;
}
else if (dst->opcode == exec_dispatch[O_STM])
{
/* Turn of the stack ref */
dst->srcb.type = eas.s.ea_nop.s.srcbbyte;
}
/* extends read one size and write another */
else if (dst->opcode == exec_dispatch[O_EXTS]
|| dst->opcode == exec_dispatch[O_EXTU])
{
dst->dst.type = eas.s.ea_reg.s.dstword;
dst->dst.reg.bptr = regptr[rd][JWORD];
dst->flags = flag_Ap;
}
if (opcode->flags == 'h')
thinkabout_shifts (dst, opcode->flavor & O_BYTE);
/* For a branch, turn off one level of indirection */
if (opcode->src1 == 'B')
{
indoff (&dst->srca, 0);
}
}
dst->next_pc = pc + opcode->length;
compcycles (dst, opcode);
return;
next:;
}
/* Couldn't understand anything */
dst->opcode = exec_dispatch[O_TRAPA];
dst->next_pc = pc + 1;
}
compile (pc)
{
int idx;
/* find the next cache entry to use */
idx = cpu.cache_top + 1;
cpu.compiles++;
if (idx >= cpu.csize)
{
idx = 1;
}
cpu.cache_top = idx;
/* Throw away its old meaning */
cpu.cache_idx[cpu.cache[idx].oldpc] = 0;
/* set to new address */
cpu.cache[idx].oldpc = pc;
/* fill in instruction info */
find (pc, cpu.memory + pc, cpu.cache + idx);
/* point to new cache entry */
cpu.cache_idx[pc] = idx;
}
baddefault (x)
{
printf ("bad default %d\n", x);
}
static int fetch_l (arg)
ea_type *arg;
{
int l, r;
int h = *(arg->reg.wptr);
r = (union rtype *) (arg->reg.wptr) - &cpu.regs[0];
r++;
l = cpu.regs[r].s[LOW];
return (h << 16) | l;
}
#define FETCH(dst, arg, n) \
{ \
int r; unsigned char*lval; \
DISPATCH((arg).type) \
{ LABELN(FETCH_NOP,n): \
dst= 0; \
break; \
DEFAULT baddefault((arg).type); break; \
LABELN(FETCH_LVAL,n): \
dst = (*(((arg).reg.wptr)) + (arg.literal)) ; \
break; \
LABELN(FETCH_LVAL24,n): \
dst = (*(((arg).reg.wptr)) + *(((arg).r2.wptr)) + (arg.literal)) &0xffffff; \
break; \
LABELN(FETCH_CRB,n): \
dst = (*((arg).reg.segptr) - cpu.memory)>>16; \
break; \
LABELN(FETCH_CRW,n): \
dst = BUILDSR();\
break; \
LABELN(FETCH_REG_B,n): \
dst = *((arg).reg.bptr); \
break; \
LABELN(FETCH_REG_W,n): \
dst = *((arg).reg.wptr); \
break; \
LABELN(FETCH_REG_L,n): \
dst = fetch_l(&(arg));\
break; \
LABELN(FETCH_INC_B,n): \
lval = elval ((arg), 0); \
dst = byteat (lval); \
(*((arg).reg.wptr))++; \
break; \
LABELN(FETCH_INC_W,n): \
lval = elval ((arg), 0); \
dst = wordat (lval); \
(*(((arg).reg.wptr))) += 2; \
break; \
LABELN(FETCH_DEC_B, n): \
(*(arg).reg.wptr)--; \
lval = elval ((arg), 0); \
r = byteat (lval); \
dst = r; \
break; \
LABELN(FETCH_DEC_W, n): \
(*((arg).reg.wptr)) -= 2; \
lval = elval ((arg), 0); \
r = wordat (lval); \
dst = r; \
break; \
LABELN(FETCH_DISP_B,n): \
lval = displval ((arg)); \
dst = byteat (lval); \
break; \
LABELN(FETCH_DISP_W,n): \
lval = displval ((arg)); \
dst = wordat (lval); \
break; \
LABELN(FETCH_IMM, n): \
dst = (arg).literal; \
break; \
} \
ENDDISPATCH; \
}
static union
{
short int i;
struct
{
char low;
char high;
}
u;
}
littleendian;
static
void
init_pointers ()
{
static int init;
if (!init)
{
int i;
init = 1;
littleendian.i = 1;
for (i = 0; i < (int) R_LAST; i++)
{
if (littleendian.u.high)
{
/* big endian host */
LOW = 1;
HIGH = 0;
regptr[i][0] = ((unsigned char *) (cpu.regs + i)) + 3;
regptr[i][1] = ((unsigned char *) (cpu.regs + i)) + 2;
}
else
{
LOW = 0;
HIGH = 1;
regptr[i][0] = (unsigned char *) &(cpu.regs[i]);
regptr[i][1] = (unsigned char *) (&(cpu.regs[i]));
}
regptr[i][2] = (unsigned char *) &(cpu.regs[i]);
}
memcpy (segregptr + 0, regptr + R_SR, sizeof (segregptr[0]));
memcpy (segregptr + 1, regptr + R_TP, sizeof (segregptr[1]));
memcpy (segregptr + 3, regptr + R_BR, sizeof (segregptr[3]));
memcpy (segregptr + 4, regptr + R_EP, sizeof (segregptr[4]));
memcpy (segregptr + 5, regptr + R_DP, sizeof (segregptr[5]));
memcpy (segregptr + 6, regptr + R_CP, sizeof (segregptr[6]));
memcpy (segregptr + 7, regptr + R_TP, sizeof (segregptr[7]));
/* Pointers to into the cpu state for the seg registers */
segmap[R0] = &cpu.regs[R_DP].c;
segmap[R1] = &cpu.regs[R_DP].c;
segmap[R2] = &cpu.regs[R_DP].c;
segmap[R3] = &cpu.regs[R_DP].c;
segmap[R4] = &cpu.regs[R_EP].c;
segmap[R5] = &cpu.regs[R_EP].c;
segmap[R6] = &cpu.regs[R_TP].c;
segmap[R7] = &cpu.regs[R_TP].c;
segmap[R_HARD_0] = &cpu.regs[R_DP].c;
segmap[R_HARD8_0] = &cpu.regs[R_BP].c;
cpu.memory = (unsigned char *) calloc (sizeof (char), H8500_MSIZE);
cpu.cache_idx = (unsigned short *) calloc (sizeof (short), H8500_MSIZE);
/* initialize the seg registers */
cpu.regs[R_DP].c = cpu.memory;
cpu.regs[R_TP].c = cpu.memory;
cpu.regs[R_CP].c = cpu.memory;
cpu.regs[R_BP].c = cpu.memory;
cpu.regs[R_EP].c = cpu.memory;
cpu.regs[R7].s[LOW] = 0xfffe;
cpu.regs[R6].s[LOW] = 0xfffe;
if (!cpu.cache)
sim_set_simcache_size (CSIZE);
}
}
#define PUSHWORD(x) \
{ \
int sp = cpu.regs[R7].s[LOW]; \
unsigned char *p; \
\
sp -= 2; \
p = (sp & 0xffff) + (cpu.regs[R_TP].c); \
cpu.regs[R7].s[LOW] = sp; \
setwordat (p, x); \
} \
#define POPWORD(d) \
{ \
int spx= cpu.regs[R7].s[LOW]; \
unsigned char *p; \
\
p = (spx& 0xffff) + (cpu.regs[R_TP].c); \
spx+= 2; \
cpu.regs[R7].s[LOW] = spx; \
d = wordat (p); \
} \
/* simulate a monitor trap */
trap ()
{
switch (cpu.regs[R3].s[LOW] & 0xff)
{
case 33:
/* exit */
cpu.exception = SIGQUIT;
break;
case 34:
/* abort */
cpu.exception = SIGABRT;
break;
case 6:
/* print char in r0 */
printf ("%c", cpu.regs[R0].s[LOW]);
break;
}
}
void
control_c (sig, code, scp, addr)
int sig;
int code;
char *scp;
char *addr;
{
cpu.exception = SIGINT;
}
static jmp_buf jbuf;
static void
segv ()
{
cpu.exception = SIGSEGV;
longjmp (jbuf, 1);
}
int
sim_stop (sd)
SIM_DESC sd;
{
cpu.exception = SIGINT;
return 1;
}
void
sim_resume (sd, step, siggnal)
SIM_DESC sd;
{
static int init1;
int res;
int tmp;
int arga;
int argb;
int bit;
int pc;
int C, Z, V, N;
int cycles = 0;
int insts = 0;
int tick_start = get_now ();
void (*prev) ();
void (*prev_seg) ();
if (!init1)
{
int i;
init1 = 1;
init_pointers ();
for (i = 0; i < N_EATYPES; i++)
{
eas.a[i].s.srcabyte = LABEL_REFN (FETCH_NOP, 0);
eas.a[i].s.srcaword = LABEL_REFN (FETCH_NOP, 0);
eas.a[i].s.srcalong = LABEL_REFN (FETCH_NOP, 0);
eas.a[i].s.srcbbyte = LABEL_REFN (FETCH_NOP, 1);
eas.a[i].s.srcbword = LABEL_REFN (FETCH_NOP, 1);
eas.a[i].s.srcblong = LABEL_REFN (FETCH_NOP, 1);
eas.a[i].s.dstbyte = LABEL_REF (STORE_NOP);
eas.a[i].s.dstword = LABEL_REF (STORE_NOP);
eas.a[i].s.dstlong = LABEL_REF (STORE_NOP);
}
eas.s.ea_lval.s.srcabyte = LABEL_REFN (FETCH_LVAL, 0);
eas.s.ea_lval.s.srcaword = LABEL_REFN (FETCH_LVAL, 0);
eas.s.ea_lval24.s.srcabyte = LABEL_REFN (FETCH_LVAL24, 0);
eas.s.ea_lval24.s.srcaword = LABEL_REFN (FETCH_LVAL24, 0);
eas.s.ea_nop.s.srcabyte = LABEL_REFN (FETCH_NOP, 0);
eas.s.ea_nop.s.srcaword = LABEL_REFN (FETCH_NOP, 0);
eas.s.ea_nop.s.srcbbyte = LABEL_REFN (FETCH_NOP, 1);
eas.s.ea_nop.s.srcbword = LABEL_REFN (FETCH_NOP, 1);
eas.s.ea_nop.s.dstbyte = LABEL_REF (STORE_NOP);
eas.s.ea_nop.s.dstword = LABEL_REF (STORE_NOP);
eas.s.ea_cr.s.srcabyte = LABEL_REFN (FETCH_CRB, 0);
eas.s.ea_cr.s.srcaword = LABEL_REFN (FETCH_CRW, 0);
eas.s.ea_cr.s.srcbbyte = LABEL_REFN (FETCH_CRB, 1);
eas.s.ea_cr.s.srcbword = LABEL_REFN (FETCH_CRW, 1);
eas.s.ea_cr.s.dstbyte = LABEL_REF (STORE_CRB);
eas.s.ea_cr.s.dstword = LABEL_REF (STORE_CRW);
eas.s.ea_reg.s.srcabyte = LABEL_REFN (FETCH_REG_B, 0);
eas.s.ea_reg.s.srcaword = LABEL_REFN (FETCH_REG_W, 0);
eas.s.ea_reg.s.srcalong = LABEL_REFN (FETCH_REG_L, 0);
eas.s.ea_reg.s.srcbbyte = LABEL_REFN (FETCH_REG_B, 1);
eas.s.ea_reg.s.srcbword = LABEL_REFN (FETCH_REG_W, 1);
eas.s.ea_reg.s.srcblong = LABEL_REFN (FETCH_REG_L, 1);
eas.s.ea_reg.s.dstbyte = LABEL_REF (STORE_REG_B);
eas.s.ea_reg.s.dstword = LABEL_REF (STORE_REG_W);
eas.s.ea_reg.s.dstlong = LABEL_REF (STORE_REG_L);
eas.s.ea_inc.s.srcabyte = LABEL_REFN (FETCH_INC_B, 0);
eas.s.ea_inc.s.srcaword = LABEL_REFN (FETCH_INC_W, 0);
eas.s.ea_inc.s.srcbbyte = LABEL_REFN (FETCH_INC_B, 1);
eas.s.ea_inc.s.srcbword = LABEL_REFN (FETCH_INC_W, 1);
eas.s.ea_inc.s.dstbyte = LABEL_REF (STORE_INC_B);
eas.s.ea_inc.s.dstword = LABEL_REF (STORE_INC_W);
eas.s.ea_dec.s.srcabyte = LABEL_REFN (FETCH_DEC_B, 0);
eas.s.ea_dec.s.srcaword = LABEL_REFN (FETCH_DEC_W, 0);
eas.s.ea_dec.s.srcbbyte = LABEL_REFN (FETCH_DEC_B, 1);
eas.s.ea_dec.s.srcbword = LABEL_REFN (FETCH_DEC_W, 1);
eas.s.ea_dec.s.dstbyte = LABEL_REF (STORE_DEC_B);
eas.s.ea_dec.s.dstword = LABEL_REF (STORE_DEC_W);
eas.s.ea_disp.s.srcabyte = LABEL_REFN (FETCH_DISP_B, 0);
eas.s.ea_disp.s.srcaword = LABEL_REFN (FETCH_DISP_W, 0);
eas.s.ea_disp.s.srcbbyte = LABEL_REFN (FETCH_DISP_B, 1);
eas.s.ea_disp.s.srcbword = LABEL_REFN (FETCH_DISP_W, 1);
eas.s.ea_disp.s.dstbyte = LABEL_REF (STORE_DISP_B);
eas.s.ea_disp.s.dstword = LABEL_REF (STORE_DISP_W);
eas.s.ea_imm.s.srcabyte = LABEL_REFN (FETCH_IMM, 0);
eas.s.ea_imm.s.srcaword = LABEL_REFN (FETCH_IMM, 0);
eas.s.ea_imm.s.srcbbyte = LABEL_REFN (FETCH_IMM, 1);
eas.s.ea_imm.s.srcbword = LABEL_REFN (FETCH_IMM, 1);
flag_special = LABEL_REF (FLAG_special);
flag_mp = LABEL_REF (FLAG_m);
flag_Mp = LABEL_REF (FLAG_M);
flag_ap = LABEL_REF (FLAG_a);
flag_Ap = LABEL_REF (FLAG_A);
flag_nonep = LABEL_REF (FLAG_NONE);
flag_nostorep = LABEL_REF (FLAG_NOSTORE);
flag_clearp = LABEL_REF (FLAG_CLEAR);
flag_shiftbyte = LABEL_REF (FLAG_shiftbyte);
flag_shiftword = LABEL_REF (FLAG_shiftword);
flag_multbyte = LABEL_REF (FLAG_multbyte);
flag_multword = LABEL_REF (FLAG_multword);
exec_dispatch[O_ADDS] = LABEL_REF (O_ADDS);
exec_dispatch[O_ADDX] = LABEL_REF (O_ADDX);
exec_dispatch[O_ADD] = LABEL_REF (O_ADD);
exec_dispatch[O_ANDC] = LABEL_REF (O_ANDC);
exec_dispatch[O_AND] = LABEL_REF (O_AND);
exec_dispatch[O_BCC] = LABEL_REF (O_BCC);
exec_dispatch[O_BCLR] = LABEL_REF (O_BCLR);
exec_dispatch[O_BCS] = LABEL_REF (O_BCS);
exec_dispatch[O_BEQ] = LABEL_REF (O_BEQ);
exec_dispatch[O_BF] = LABEL_REF (O_BF);
exec_dispatch[O_BGE] = LABEL_REF (O_BGE);
exec_dispatch[O_BGT] = LABEL_REF (O_BGT);
exec_dispatch[O_BHI] = LABEL_REF (O_BHI);
exec_dispatch[O_BHS] = LABEL_REF (O_BHS);
exec_dispatch[O_BLE] = LABEL_REF (O_BLE);
exec_dispatch[O_BLO] = LABEL_REF (O_BLO);
exec_dispatch[O_BLS] = LABEL_REF (O_BLS);
exec_dispatch[O_BLT] = LABEL_REF (O_BLT);
exec_dispatch[O_BMI] = LABEL_REF (O_BMI);
exec_dispatch[O_BNE] = LABEL_REF (O_BNE);
exec_dispatch[O_BNOT] = LABEL_REF (O_BNOT);
exec_dispatch[O_BPL] = LABEL_REF (O_BPL);
exec_dispatch[O_BPT] = LABEL_REF (O_BPT);
exec_dispatch[O_BRA] = LABEL_REF (O_BRA);
exec_dispatch[O_BRN] = LABEL_REF (O_BRN);
exec_dispatch[O_BSET] = LABEL_REF (O_BSET);
exec_dispatch[O_BSR] = LABEL_REF (O_BSR);
exec_dispatch[O_BTST] = LABEL_REF (O_BTST);
exec_dispatch[O_BT] = LABEL_REF (O_BT);
exec_dispatch[O_BVC] = LABEL_REF (O_BVC);
exec_dispatch[O_BVS] = LABEL_REF (O_BVS);
exec_dispatch[O_CLR] = LABEL_REF (O_CLR);
exec_dispatch[O_CMP] = LABEL_REF (O_CMP);
exec_dispatch[O_DADD] = LABEL_REF (O_DADD);
exec_dispatch[O_DIVXU] = LABEL_REF (O_DIVXU);
exec_dispatch[O_DSUB] = LABEL_REF (O_DSUB);
exec_dispatch[O_EXTS] = LABEL_REF (O_EXTS);
exec_dispatch[O_EXTU] = LABEL_REF (O_EXTU);
exec_dispatch[O_JMP] = LABEL_REF (O_JMP);
exec_dispatch[O_JSR] = LABEL_REF (O_JSR);
exec_dispatch[O_LDC] = LABEL_REF (O_LDC);
exec_dispatch[O_LDM] = LABEL_REF (O_LDM);
exec_dispatch[O_LINK] = LABEL_REF (O_LINK);
exec_dispatch[O_MOVFPE] = LABEL_REF (O_MOVFPE);
exec_dispatch[O_MOVTPE] = LABEL_REF (O_MOVTPE);
exec_dispatch[O_MOV] = LABEL_REF (O_MOV);
exec_dispatch[O_MULXU] = LABEL_REF (O_MULXU);
exec_dispatch[O_NEG] = LABEL_REF (O_NEG);
exec_dispatch[O_NOP] = LABEL_REF (O_NOP);
exec_dispatch[O_NOT] = LABEL_REF (O_NOT);
exec_dispatch[O_ORC] = LABEL_REF (O_ORC);
exec_dispatch[O_OR] = LABEL_REF (O_OR);
exec_dispatch[O_PJMP] = LABEL_REF (O_PJMP);
exec_dispatch[O_PJSR] = LABEL_REF (O_PJSR);
exec_dispatch[O_PRTD] = LABEL_REF (O_PRTD);
exec_dispatch[O_PRTS] = LABEL_REF (O_PRTS);
exec_dispatch[O_RECOMPILE] = LABEL_REF (O_RECOMPILE);
exec_dispatch[O_ROTL] = LABEL_REF (O_ROTL);
exec_dispatch[O_ROTR] = LABEL_REF (O_ROTR);
exec_dispatch[O_ROTXL] = LABEL_REF (O_ROTXL);
exec_dispatch[O_ROTXR] = LABEL_REF (O_ROTXR);
exec_dispatch[O_RTD] = LABEL_REF (O_RTD);
exec_dispatch[O_RTS] = LABEL_REF (O_RTS);
exec_dispatch[O_SCB_EQ] = LABEL_REF (O_SCB_EQ);
exec_dispatch[O_SCB_F] = LABEL_REF (O_SCB_F);
exec_dispatch[O_SCB_NE] = LABEL_REF (O_SCB_NE);
exec_dispatch[O_SHAL] = LABEL_REF (O_SHAL);
exec_dispatch[O_SHAR] = LABEL_REF (O_SHAR);
exec_dispatch[O_SHLL] = LABEL_REF (O_SHLL);
exec_dispatch[O_SHLR] = LABEL_REF (O_SHLR);
exec_dispatch[O_SLEEP] = LABEL_REF (O_SLEEP);
exec_dispatch[O_STC] = LABEL_REF (O_STC);
exec_dispatch[O_STM] = LABEL_REF (O_STM);
exec_dispatch[O_SUBS] = LABEL_REF (O_SUBS);
exec_dispatch[O_SUBX] = LABEL_REF (O_SUBX);
exec_dispatch[O_SUB] = LABEL_REF (O_SUB);
exec_dispatch[O_SWAP] = LABEL_REF (O_SWAP);
exec_dispatch[O_TAS] = LABEL_REF (O_TAS);
exec_dispatch[O_TRAPA] = LABEL_REF (O_TRAPA);
exec_dispatch[O_TRAP_VS] = LABEL_REF (O_TRAP_VS);
exec_dispatch[O_TST] = LABEL_REF (O_TST);
exec_dispatch[O_UNLK] = LABEL_REF (O_UNLK);
exec_dispatch[O_XCH] = LABEL_REF (O_XCH);
exec_dispatch[O_XORC] = LABEL_REF (O_XORC);
exec_dispatch[O_XOR] = LABEL_REF (O_XOR);
nop.type = eas.s.ea_nop.s.srcabyte;
cpu.cache[0].opcode = exec_dispatch[O_RECOMPILE];
cpu.cache[0].srca.type = eas.s.ea_nop.s.srcabyte;
cpu.cache[0].srcb.type = eas.s.ea_nop.s.srcbbyte;
}
prev = signal (SIGINT, control_c);
prev_seg = signal (SIGSEGV, segv);
if (step)
{
cpu.exception = SIGTRAP;
}
else
{
cpu.exception = 0;
}
pc = cpu.regs[R_PC].s[LOW] + (NORMAL_CP << 16);
GETSR ();
if (setjmp (jbuf) == 0) {
do
{
int cidx;
decoded_inst *code;
top:
cidx = cpu.cache_idx[pc];
code = cpu.cache + cidx;
FETCH (arga, code->srca, 0);
FETCH (argb, code->srcb, 1);
#ifdef DEBUG
if (debug)
{
printf ("%x %d %s\n", pc, code->opcode,
code->op ? code->op->name : "**");
}
#endif
cycles += code->cycles;
insts++;
DISPATCH (code->opcode)
{
LABEL (O_RECOMPILE):
/* This opcode is a fake for when we get to an instruction which
hasn't been compiled */
compile (pc);
goto top;
break;
LABEL (O_NEG):
arga = -arga;
argb = 0;
res = arga + argb;
break;
LABEL (O_SUBX):
arga += C;
LABEL (O_SUB):
LABEL (O_SUBS):
arga = -arga;
LABEL (O_ADD):
LABEL (O_ADDS):
res = arga + argb;
break;
LABEL (O_ADDX):
res = arga + argb + C;
break;
LABEL (O_AND):
LABEL (O_ANDC):
res = arga & argb;
break;
break;
LABEL (O_BCLR):
arga &= 0xf;
bit = (argb & (1 << arga));
res = argb & ~(1 << arga);
goto bitop;
LABEL (O_BRA):
LABEL (O_BT):
if (1)
goto condtrue;
LABEL (O_BRN):
LABEL (O_BF):
if (0)
goto condtrue;
break;
LABEL (O_BHI):
if ((C || Z) == 0)
goto condtrue;
break;
LABEL (O_BLS):
if ((C || Z))
goto condtrue;
break;
LABEL (O_BCS):
LABEL (O_BLO):
if ((C == 1))
goto condtrue;
break;
LABEL (O_BCC):
LABEL (O_BHS):
if ((C == 0))
goto condtrue;
break;
LABEL (O_BEQ):
if (Z)
goto condtrue;
break;
LABEL (O_BGT):
if (((Z || (N ^ V)) == 0))
goto condtrue;
break;
LABEL (O_BLE):
if (((Z || (N ^ V)) == 1))
goto condtrue;
break;
LABEL (O_BGE):
if ((N ^ V) == 0)
goto condtrue;
break;
LABEL (O_BLT):
if ((N ^ V))
goto condtrue;
break;
LABEL (O_BMI):
if ((N))
goto condtrue;
break;
LABEL (O_BNE):
if ((Z == 0))
goto condtrue;
break;
LABEL (O_BPL):
if (N == 0)
goto condtrue;
break;
break;
LABEL (O_BVC):
if ((V == 0))
goto condtrue;
break;
LABEL (O_BVS):
if ((V == 1))
goto condtrue;
break;
LABEL (O_BNOT):
bit = argb & (1<<(arga & 0xf));
res = argb ^ (1<<(arga & 0xf));
goto bitop;
break;
LABEL (O_BSET):
arga = 1 << (arga & 0xf);
bit = argb & arga;
res = argb | arga;
goto bitop;
break;
LABEL (O_PJMP):
pc = arga;
goto next;
LABEL (O_UNLK):
{
int t;
SET_NORMREG (R7, GET_NORMREG (R6));
POPWORD (t);
SET_NORMREG (R6, t);
pc = code->next_pc;
goto next;
}
LABEL (O_RTS):
{
int cp = pc & 0xff0000;
POPWORD (pc);
pc |= cp;
goto next;
}
break;
LABEL (O_PRTS):
{
int cp;
int off;
POPWORD (cp);
POPWORD (off);
cp <<= 16;
SET_SEGREG (R_CP, cp);
pc = cp + off;
}
goto next;
LABEL (O_PJSR):
PUSHWORD (argb & 0xffff);
PUSHWORD (argb >> 16);
pc = (arga & 0xffffff);
goto next;
LABEL (O_BSR):
LABEL (O_JSR):
PUSHWORD (code->next_pc);
pc = arga | (pc & 0xff0000);
goto next;
LABEL (O_BTST):
Z = (((argb >> (arga & 0xf)) & 1) == 0);
pc = code->next_pc;
goto next;
LABEL (O_CLR):
res = 0;
break;
LABEL (O_CMP):
arga = -arga;
res = arga + argb;
break;
LABEL (O_DADD):
res = arga + argb + C;
if (res > 99)
{
res -= 100;
C = 1;
}
else
{
C = 0;
}
Z = Z && (res == 0);
break;
LABEL (O_DSUB):
res = argb - arga - C;
if (res < 0)
{
res += 100;
C = 1;
}
else
{
C = 0;
}
Z = Z && (res == 0);
break;
LABEL (O_EXTS):
res = SEXTCHAR (arga);
break;
LABEL (O_EXTU):
res = (unsigned char) arga;
break;
LABEL (O_JMP):
pc = arga | (pc & 0xff0000);
goto next;
break;
LABEL (O_LDM):
for (tmp = 0; tmp < 7; tmp++)
{
if (argb & (1 << tmp))
{
POPWORD (cpu.regs[tmp].s[LOW]);
}
}
if (argb & 0x80)
POPWORD (tmp); /* dummy ready for sp */
goto nextpc;
break;
LABEL (O_LINK):
PUSHWORD (cpu.regs[R6].s[LOW]);
cpu.regs[R6].s[LOW] = cpu.regs[R7].s[LOW];
cpu.regs[R7].s[LOW] += argb;
goto nextpc;
LABEL (O_STC):
LABEL (O_LDC):
LABEL (O_MOVFPE):
LABEL (O_MOVTPE):
LABEL (O_MOV):
LABEL (O_TST):
res = arga;
break;
LABEL (O_TRAPA):
if (arga == 15)
{
trap ();
}
else
{
PUSHWORD (pc & 0xffff);
if (cpu.maximum)
{
PUSHWORD (NORMAL_CP);
}
PUSHWORD (NORMAL_SR);
if (cpu.maximum)
{
arga = arga * 4 + 0x40;
SET_NORMAL_CPPC (longat (cpu.memory + arga));
}
else
{
arga = arga * 2 + 0x20;
SET_NORMAL_CPPC (wordat (cpu.memory + arga));
}
}
break;
LABEL (O_OR):
LABEL (O_ORC):
res = arga | argb;
break;
LABEL (O_XOR):
LABEL (O_XORC):
res = arga ^ argb;
break;
LABEL (O_SCB_F):
{
scb_f:
res = arga - 1;
code->srca.reg.wptr[0] = res;
if (res != -1)
{
pc = argb;
goto next;
}
}
break;
LABEL (O_SCB_EQ):
if (Z == 1)
break;
else
goto scb_f;
LABEL (O_SCB_NE):
if (Z == 0)
break;
else
goto scb_f;
LABEL (O_NOP):
/* If only they were all as simple as this */
break;
LABEL (O_ROTL):
res = arga << 1;
C = (res >> argb) & 1;
res |= C;
break;
LABEL (O_ROTR):
C = arga & 1;
res = arga >> 1;
res |= (C << (argb - 1));
break;
LABEL (O_ROTXL):
res = arga << 1;
res |= C;
C = (res >> argb) & 1;
break;
LABEL (O_ROTXR):
res = arga >> 1;
res |= (C << (argb - 1));
C = arga & 1;
break;
LABEL (O_SHAL):
res = arga << 1;
if (argb == 16)
{
C = (res >> (16)) & 1;
Z = ((res & 0xffff) == 0);
N = ((res & 0x8000) != 0);
}
else
{
C = (res >> (8)) & 1;
Z = ((res & 0xff) == 0);
N = ((res & 0x80) != 0);
}
V = C ^ N;
goto none;
LABEL (O_SHAR):
C = arga & 1;
if (argb == 16)
{
res = ((short) arga) >> 1;
}
else
{
res = (SEXTCHAR (arga)) >> 1;
}
break;
LABEL (O_SHLL):
res = arga << 1;
C = (res >> argb) & 1;
break;
LABEL (O_SHLR):
C = arga & 1;
res = arga >> 1;
break;
LABEL (O_DIVXU):
if (arga == 0)
{
N = V = C = 0;
Z = 1;
cpu.exception = SIGILL;
}
else
{
int d = argb / arga;
int m = argb % arga;
if (code->dst.type == eas.s.ea_reg.s.dstlong)
{
res = (m << 16) | (d & 0xffff);
}
else
{
res = (m << 8) | (d & 0xff);
}
}
break;
LABEL (O_MULXU):
res = arga * argb;
break;
LABEL (O_NOT):
res = ~arga;
break;
LABEL (O_SWAP):
res = ((arga >> 8) & 0xff) | ((arga << 8) & 0xff00);
break;
LABEL (O_STM):
for (tmp = 7; tmp >= 0; tmp--)
{
if (arga & (1 << tmp))
{
PUSHWORD (cpu.regs[tmp].s[LOW]);
}
}
goto nextpc;
LABEL (O_TAS):
C = 0;
V = 0;
Z = arga == 0;
N = arga < 0;
res = arga | 0x80;
goto none;
LABEL (O_PRTD):
LABEL (O_XCH):
LABEL (O_RTD):
cpu.exception = SIGILL;
goto next;
LABEL (O_TRAP_VS):
LABEL (O_SLEEP):
LABEL (O_BPT):
cpu.exception = SIGTRAP;
goto next;
break;
}
ENDDISPATCH;
DISPATCH (code->flags)
{
bitop:
Z = (res & bit) == 0;
pc = code->next_pc;
break;
LABEL (FLAG_multword):
Z = (res & 0xffff) == 0;
N = (res & 0x8000) != 0;
V = 0;
C = 0;
pc = code->next_pc;
break;
LABEL (FLAG_multbyte):
/* 8*8 -> 16 */
Z = (res & 0xff) == 0;
N = (res & 0x80) != 0;
V = 0;
C = 0;
pc = code->next_pc;
break;
LABEL (FLAG_shiftword):
N = (res & 0x8000) != 0;
Z = (res & 0xffff) == 0;
V = 0;
pc = code->next_pc;
break;
LABEL (FLAG_shiftbyte):
N = (res & 0x80) != 0;
Z = (res & 0xff) == 0;
V = 0;
pc = code->next_pc;
break;
LABEL (FLAG_special):
pc = code->next_pc;
break;
LABEL (FLAG_m):
/* Move byte flags */
/* after a logical instruction */
N = (res & 0x80) != 0;
Z = (res & 0xff) == 0;
V = (((~arga & ~argb & res) | (arga & argb & ~res)) & 0x80) != 0;
pc = code->next_pc;
break;
LABEL (FLAG_M):
/* Move word flags */
/* after a logical instruction */
N = (res & 0x8000) != 0;
Z = (res & 0xffff) == 0;
V = (((~arga & ~argb & res) | (arga & argb & ~res)) & 0x8000) != 0;
pc = code->next_pc;
break;
LABEL (FLAG_a):
/* after byte sized arith */
C = (res & 0x100) != 0;
N = (res & 0x80) != 0;
Z = (res & 0xff) == 0;
V = (((~arga & ~argb & res) | (arga & argb & ~res)) & 0x80) != 0;
pc = code->next_pc;
break;
LABEL (FLAG_A):
/* after word sized arith */
C = (res & 0x10000) != 0;
N = (res & 0x8000) != 0;
Z = (res & 0xffff) == 0;
V = (((~arga & ~argb & res) | (arga & argb & ~res)) & 0x8000) != 0;
pc = code->next_pc;
break;
LABEL (FLAG_NONE):
none:;
/* no flags but store */
pc = code->next_pc;
break;
LABEL (FLAG_NOSTORE):
/* no flags and no store */
pc = code->next_pc;
break;
LABEL (FLAG_CLEAR):
/* clear flags */
N = 0;
Z = 1;
V = 0;
C = 0;
pc = code->next_pc;
break;
condtrue:
pc = arga;
goto next;
}
ENDDISPATCH;
DISPATCH (code->dst.type)
{
unsigned char *lval;
LABEL (STORE_CRB):
(*(code->dst.reg.segptr)) = cpu.memory + (res << 16);
break;
LABEL (STORE_NOP):
break;
LABEL (STORE_REG_B):
(*(code->dst.reg.bptr)) = res;
break;
LABEL (STORE_REG_W):
(*(code->dst.reg.wptr)) = res;
break;
LABEL (STORE_REG_L):
{
int l, r;
r = (union rtype *) (code->dst.reg.wptr) - &cpu.regs[0];
r++;
*(code->dst.reg.wptr) = res >> 16;
cpu.regs[r].s[LOW] = res & 0xffff;
}
break;
LABEL (STORE_DISP_W):
lval = displval (code->dst);
setwordat (lval, res);
break;
LABEL (STORE_DISP_B):
lval = displval (code->dst);
setbyteat (lval, res);
break;
LABEL (STORE_INC_B):
lval = elval (code->dst, 0);
setbyteat (lval, res);
(*(code->dst.reg.wptr))++;
break;
LABEL (STORE_INC_W):
lval = elval (code->dst, 0);
setwordat (lval, res);
(*(code->dst.reg.wptr)) += 2;
break;
LABEL (STORE_DEC_B):
(*(code->dst.reg.wptr))--;
lval = elval (code->dst, 0);
setbyteat (lval, res);
break;
LABEL (STORE_CRW):
/* Make an up to date sr from the flag state */
cpu.regs[R_SR].s[LOW] = res;
GETSR ();
break;
LABEL (STORE_DEC_W):
(*(code->dst.reg.wptr)) -= 2;
lval = elval (code->dst, 0);
setwordat (lval, res);
break;
nextpc:
pc = code->next_pc;
}
ENDDISPATCH;
next:;
}
while (!cpu.exception);
}
cpu.ticks += get_now () - tick_start;
cpu.cycles += cycles;
cpu.insts += insts;
cpu.regs[R_PC].s[LOW] = pc;
BUILDSR ();
signal (SIGINT, prev);
signal (SIGSEGV, prev_seg);
}
int
sim_write (sd, addr, buffer, size)
SIM_DESC sd;
SIM_ADDR addr;
unsigned char *buffer;
int size;
{
int i;
init_pointers ();
if (addr < 0 || addr + size > H8500_MSIZE)
return 0;
for (i = 0; i < size; i++)
{
cpu.memory[addr + i] = buffer[i];
cpu.cache_idx[addr + i] = 0;
}
return size;
}
int
sim_read (sd, addr, buffer, size)
SIM_DESC sd;
SIM_ADDR addr;
unsigned char *buffer;
int size;
{
init_pointers ();
if (addr < 0 || addr + size > H8500_MSIZE)
return 0;
memcpy (buffer, cpu.memory + addr, size);
return size;
}
/* Ripped off from tm-h8500.h */
#define R0_REGNUM 0
#define R1_REGNUM 1
#define R2_REGNUM 2
#define R3_REGNUM 3
#define R4_REGNUM 4
#define R5_REGNUM 5
#define R6_REGNUM 6
#define R7_REGNUM 7
/* As above, but with correct seg register glued on */
#define PR0_REGNUM 8
#define PR1_REGNUM 9
#define PR2_REGNUM 10
#define PR3_REGNUM 11
#define PR4_REGNUM 12
#define PR5_REGNUM 13
#define PR6_REGNUM 14
#define PR7_REGNUM 15
#define SP_REGNUM PR7_REGNUM /* Contains address of top of stack */
#define FP_REGNUM PR6_REGNUM /* Contains address of executing stack frame */
#define SEG_C_REGNUM 16 /* Segment registers */
#define SEG_D_REGNUM 17
#define SEG_E_REGNUM 18
#define SEG_T_REGNUM 19
#define CCR_REGNUM 20 /* Contains processor status */
#define PC_REGNUM 21 /* Contains program counter */
#define CYCLE_REGNUM 22
#define INST_REGNUM 23
#define TICK_REGNUM 24
int
sim_store_register (sd, rn, value, length)
SIM_DESC sd;
int rn;
unsigned char *value;
int length;
{
int seg = 0;
int reg = -1;
init_pointers ();
switch (rn)
{
case PC_REGNUM:
SET_SEGREG (R_CP, (value[1]<<16));
cpu.regs[R_PC].s[LOW] = (value[2] << 8) | value[3];
break;
case SEG_C_REGNUM:
case SEG_D_REGNUM:
case SEG_E_REGNUM:
case SEG_T_REGNUM:
seg = rn - SEG_C_REGNUM + R_CP;
reg = -1;
break;
default:
abort ();
case R0_REGNUM:
case R1_REGNUM:
case R2_REGNUM:
case R3_REGNUM:
case R4_REGNUM:
case R5_REGNUM:
case R6_REGNUM:
case R7_REGNUM:
seg = 0;
reg = rn - R0_REGNUM;
break;
case CCR_REGNUM:
seg = 0;
reg = R_SR;
break;
case CYCLE_REGNUM:
cpu.cycles = (value[0] << 24) | (value[1] << 16) | (value[2] << 8) | value[3];
return;
case INST_REGNUM:
cpu.insts = (value[0] << 24) | (value[1] << 16) | (value[2] << 8) | value[3];
return;
case TICK_REGNUM:
cpu.ticks = (value[0] << 24) | (value[1] << 16) | (value[2] << 8) | value[3];
return;
case PR0_REGNUM:
case PR1_REGNUM:
case PR2_REGNUM:
case PR3_REGNUM:
case PR4_REGNUM:
case PR5_REGNUM:
case PR6_REGNUM:
case PR7_REGNUM:
SET_SEGREG (segforreg[rn], value[1]);
reg = rn - PR0_REGNUM;
cpu.regs[reg].s[LOW] = (value[2] << 8) | value[3];
return;
}
if (seg)
SET_SEGREG (seg, value[0] << 16);
if (reg > 0)
{
cpu.regs[reg].s[LOW] = (value[0] << 8) | value[1];
}
return -1;
}
int
sim_fetch_register (sd, rn, buf, length)
SIM_DESC sd;
int rn;
unsigned char *buf;
int length;
{
init_pointers ();
switch (rn)
{
default:
abort ();
case SEG_C_REGNUM:
case SEG_D_REGNUM:
case SEG_E_REGNUM:
case SEG_T_REGNUM:
buf[0] = GET_SEGREG(rn - SEG_C_REGNUM + R_CP);
break;
case CCR_REGNUM:
buf[0] = cpu.regs[R_SR].s[HIGH];
buf[1] = cpu.regs[R_SR].s[LOW];
break;
case PC_REGNUM:
buf[0] = 0;
buf[1] = GET_SEGREG(R_CP);
buf[2] = HIGH_BYTE (cpu.regs[R_PC].s[LOW]);
buf[3] = LOW_BYTE (cpu.regs[R_PC].s[LOW]);
break;
case PR0_REGNUM:
case PR1_REGNUM:
case PR2_REGNUM:
case PR3_REGNUM:
case PR4_REGNUM:
case PR5_REGNUM:
case PR6_REGNUM:
case PR7_REGNUM:
rn -= PR0_REGNUM;
buf[0] = 0;
buf[1] = GET_SEGREG(segforreg[rn]);
buf[2] = HIGH_BYTE (cpu.regs[rn].s[LOW]);
buf[3] = LOW_BYTE (cpu.regs[rn].s[LOW]);
break;
case R0_REGNUM:
case R1_REGNUM:
case R2_REGNUM:
case R3_REGNUM:
case R4_REGNUM:
case R5_REGNUM:
case R6_REGNUM:
case R7_REGNUM:
buf[0] = HIGH_BYTE (cpu.regs[rn].s[LOW]);
buf[1] = LOW_BYTE (cpu.regs[rn].s[LOW]);
break;
case CYCLE_REGNUM:
buf[0] = cpu.cycles >> 24;
buf[1] = cpu.cycles >> 16;
buf[2] = cpu.cycles >> 8;
buf[3] = cpu.cycles >> 0;
break;
case TICK_REGNUM:
buf[0] = cpu.ticks >> 24;
buf[1] = cpu.ticks >> 16;
buf[2] = cpu.ticks >> 8;
buf[3] = cpu.ticks >> 0;
break;
case INST_REGNUM:
buf[0] = cpu.insts >> 24;
buf[1] = cpu.insts >> 16;
buf[2] = cpu.insts >> 8;
buf[3] = cpu.insts >> 0;
break;
}
return -1;
}
int
sim_trace (sd)
SIM_DESC sd;
{
int i;
for (i = 0; i < 12; i += 2)
{
unsigned char *p = cpu.regs[R_TP].c + ((cpu.regs[R6].s[LOW] + i) & 0xffff);
unsigned short *j = (unsigned short *) p;
printf ("%04x ", *j);
}
printf ("\n");
printf ("%02x %02x %02x %02x:%04x %04x %04x %04x %04x %04x %04x %04x %04x\n",
NORMAL_DP,
NORMAL_EP,
NORMAL_TP,
NORMAL_CP,
cpu.regs[R_PC].s[LOW],
cpu.regs[0].s[LOW],
cpu.regs[1].s[LOW],
cpu.regs[2].s[LOW],
cpu.regs[3].s[LOW],
cpu.regs[4].s[LOW],
cpu.regs[5].s[LOW],
cpu.regs[6].s[LOW],
cpu.regs[7].s[LOW]);
sim_resume (sd, 1, 0);
return 0;
}
void
sim_stop_reason (sd, reason, sigrc)
SIM_DESC sd;
enum sim_stop *reason;
int *sigrc;
{
*reason = sim_stopped;
*sigrc = cpu.exception;
}
void
sim_set_simcache_size (n)
{
if (cpu.cache)
free (cpu.cache);
if (n < 2)
n = 2;
cpu.cache = (decoded_inst *) malloc (sizeof (decoded_inst) * n);
cpu.csize = n;
}
void
sim_size (n)
int n;
{
/* Fixed size. */
}
void
sim_info (sd, verbose)
SIM_DESC sd;
int verbose;
{
double timetaken = (double) cpu.ticks / (double) now_persec ();
double virttime = cpu.cycles / 10.0e6;
(*sim_callback->printf_filtered) (sim_callback,
"\n\ninstructions executed %10d\n",
cpu.insts);
(*sim_callback->printf_filtered) (sim_callback,
"cycles (v approximate) %10d\n",
cpu.cycles);
(*sim_callback->printf_filtered) (sim_callback,
"real time taken %10.4f\n",
timetaken);
(*sim_callback->printf_filtered) (sim_callback,
"virtual time taked %10.4f\n",
virttime);
if (timetaken)
{
(*sim_callback->printf_filtered) (sim_callback,
"simulation ratio %10.4f\n",
virttime / timetaken);
}
(*sim_callback->printf_filtered) (sim_callback,
"compiles %10d\n",
cpu.compiles);
(*sim_callback->printf_filtered) (sim_callback,
"cache size %10d\n",
cpu.csize);
}
SIM_DESC
sim_open (kind, cb, abfd, argv)
SIM_OPEN_KIND kind;
host_callback *cb;
struct bfd *abfd;
char **argv;
{
sim_kind = kind;
myname = argv[0];
sim_callback = cb;
/* fudge our descriptor */
return (SIM_DESC) 1;
}
void
sim_close (sd, quitting)
SIM_DESC sd;
int quitting;
{
/* nothing to do */
}
SIM_RC
sim_load (sd, prog, abfd, from_tty)
SIM_DESC sd;
char *prog;
bfd *abfd;
int from_tty;
{
extern bfd *sim_load_file (); /* ??? Don't know where this should live. */
bfd *prog_bfd;
prog_bfd = sim_load_file (sd, myname, sim_callback, prog, abfd,
sim_kind == SIM_OPEN_DEBUG,
0, sim_write);
if (prog_bfd == NULL)
return SIM_RC_FAIL;
if (abfd == NULL)
bfd_close (prog_bfd);
return SIM_RC_OK;
}
SIM_RC
sim_create_inferior (sd, abfd, argv, env)
SIM_DESC sd;
struct bfd *abfd;
char **argv;
char **env;
{
int pc;
bfd_vma start_address;
if (abfd != NULL)
start_address = bfd_get_start_address (abfd);
else
start_address = 0;
/* ??? We assume this is a 4 byte quantity. */
pc = start_address;
sim_store_register (sd, PC_REGNUM, (unsigned char *) &pc, 4);
return SIM_RC_OK;
}
void
sim_do_command (sd, cmd)
SIM_DESC sd;
char *cmd;
{
(*sim_callback->printf_filtered) (sim_callback,
"This simulator does not accept any commands.\n");
}
void
sim_set_callbacks (ptr)
struct host_callback_struct *ptr;
{
sim_callback = ptr;
}