target: e2k: Reorg control flow.
This commit is contained in:
parent
ed1b58c984
commit
e83d027c6f
|
@ -6,6 +6,7 @@ e2k_ss.add(files(
|
||||||
'translate.c',
|
'translate.c',
|
||||||
'translate/state.c',
|
'translate/state.c',
|
||||||
'translate/win.c',
|
'translate/win.c',
|
||||||
|
'translate/control.c',
|
||||||
'translate/alc.c',
|
'translate/alc.c',
|
||||||
))
|
))
|
||||||
|
|
||||||
|
|
|
@ -202,450 +202,6 @@ void e2k_gen_exception(DisasContext *dc, int which)
|
||||||
dc->base.is_jmp = DISAS_NORETURN;
|
dc->base.is_jmp = DISAS_NORETURN;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void reset_is_jmp(void)
|
|
||||||
{
|
|
||||||
tcg_gen_movi_i32(e2k_cs.is_jmp, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void set_is_jmp(void)
|
|
||||||
{
|
|
||||||
tcg_gen_movi_i32(e2k_cs.is_jmp, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void gen_rcur_move(void)
|
|
||||||
{
|
|
||||||
TCGv_i32 tmp = tcg_temp_new_i32();
|
|
||||||
tcg_gen_addi_i32(tmp, e2k_cs.rcur, 2);
|
|
||||||
e2k_gen_wrap_i32(e2k_cs.rcur, tmp, e2k_cs.rsz);
|
|
||||||
tcg_temp_free_i32(tmp);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void gen_cs0(DisasContext *dc, CPUE2KState *env)
|
|
||||||
{
|
|
||||||
typedef enum {
|
|
||||||
NOTHING,
|
|
||||||
IBRANCH,
|
|
||||||
PREF,
|
|
||||||
PUTTSD,
|
|
||||||
DONE,
|
|
||||||
HRET,
|
|
||||||
GLAUNCH,
|
|
||||||
DISP,
|
|
||||||
SDISP,
|
|
||||||
GETTSD,
|
|
||||||
LDISP,
|
|
||||||
RETURN
|
|
||||||
} cs0_type;
|
|
||||||
|
|
||||||
static cs0_type cs0_ops[4][4] = {
|
|
||||||
{IBRANCH, PREF, PUTTSD, DONE},
|
|
||||||
{DISP, NOTHING, SDISP, GETTSD},
|
|
||||||
{DISP, LDISP, SDISP, GETTSD},
|
|
||||||
{DISP, NOTHING, SDISP, RETURN}
|
|
||||||
};
|
|
||||||
const UnpackedBundle *bundle = &dc->bundle;
|
|
||||||
uint32_t cs0 = bundle->cs0;
|
|
||||||
|
|
||||||
unsigned int ctpr = (cs0 & 0xc0000000) >> 30;
|
|
||||||
unsigned int ctp_opc = (cs0 & 0x30000000) >> 28;
|
|
||||||
unsigned int param_type = (cs0 & 0x00000007);
|
|
||||||
cs0_type type = cs0_ops[ctpr][ctp_opc];
|
|
||||||
|
|
||||||
if (type == RETURN && param_type == 1) {
|
|
||||||
type = GETTSD;
|
|
||||||
} else if (type == DONE) {
|
|
||||||
if (param_type == 3) {
|
|
||||||
type = HRET;
|
|
||||||
} else if (param_type == 4) {
|
|
||||||
type = GLAUNCH;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type == IBRANCH || type == DONE || type == HRET || type == GLAUNCH) {
|
|
||||||
/* IBRANCH, DONE, HRET and GLAUNCH are special because they require SS
|
|
||||||
to be properly encoded. */
|
|
||||||
if (! bundle->ss_present
|
|
||||||
/* SS.ctop should be equal to zero for IBRANCH, DONE, HRET and
|
|
||||||
GLAUNCH (see C.17.1.1, note that they don't mention the latter two
|
|
||||||
instructions there which is probably an omission ). */
|
|
||||||
|| (bundle->ss & 0x00000c00))
|
|
||||||
{
|
|
||||||
// TODO: invalid
|
|
||||||
abort();
|
|
||||||
}
|
|
||||||
/* Don't output either of the aforementioned instructions under "never"
|
|
||||||
condition. Don't disassemble CS0 being a part of HCALL. Unlike ldis
|
|
||||||
HCALL is currently disassembled on behalf of CS1. */
|
|
||||||
else if ((bundle->ss & 0x1ff)
|
|
||||||
&& !(bundle->cs1_present
|
|
||||||
/* CS1.opc == CALL */
|
|
||||||
&& (bundle->cs1 & 0xf0000000) >> 28 == 5
|
|
||||||
/* CS1.param.ctopc == HCALL */
|
|
||||||
&& (bundle->cs1 & 0x380) >> 7 == 2))
|
|
||||||
{
|
|
||||||
unsigned int cond = bundle->ss & 0x1ff;
|
|
||||||
if (type == IBRANCH) {
|
|
||||||
/* C0F2 has `disp' field. In `C0F1' it's called `param'. Is this
|
|
||||||
the only difference between these two formats? Funnily enough,
|
|
||||||
DONE is also C0F2 and thus has `disp', though it obviously
|
|
||||||
makes no sense for it. */
|
|
||||||
unsigned int disp = (cs0 & 0x0fffffff);
|
|
||||||
/// Calculate a signed displacement in bytes.
|
|
||||||
int sdisp = ((int) (disp << 4)) >> 1;
|
|
||||||
target_ulong tgt = dc->pc + sdisp;
|
|
||||||
TCGv dest = e2k_get_temp(dc);
|
|
||||||
tcg_gen_movi_tl(dest, tgt);
|
|
||||||
// TODO: temporary, need to move in ctcond evaluate
|
|
||||||
set_is_jmp();
|
|
||||||
dc->jmp.dest = dest;
|
|
||||||
dc->jmp.cond = cond;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
/* Note that according to Table B.4.1 it's possible to obtain
|
|
||||||
` gettsd %ctpr{1,2} with an invalid value for CS0.param.type. */
|
|
||||||
if (type == GETTSD && param_type != 1) {
|
|
||||||
// invalid
|
|
||||||
abort();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type == DISP
|
|
||||||
|| type == SDISP
|
|
||||||
|| type == LDISP
|
|
||||||
/* Note that RETURN is said to be COPF1. I can't understand what its
|
|
||||||
`CS0.param' is needed for: all of the bits except the three
|
|
||||||
lowermost ones are undefined, while the latter also known as "type"
|
|
||||||
field should be filled in with zeroes. */
|
|
||||||
|| type == RETURN
|
|
||||||
/* GETTSD has as meaningless `CS0.param' as RETURN. The only
|
|
||||||
difference is that its `CS0.param.type' should be equal to `1'. I
|
|
||||||
wonder if I should check for that and output something like
|
|
||||||
"invalid gettsd" if this turns out not to be the case . . . */
|
|
||||||
|| type == GETTSD)
|
|
||||||
{
|
|
||||||
// ctpr
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type == SDISP) {
|
|
||||||
// my_printf (", 0x%x", cs0 & 0x1f);
|
|
||||||
} else if (type == DISP
|
|
||||||
|| type == LDISP
|
|
||||||
|| type == PUTTSD)
|
|
||||||
{
|
|
||||||
// unsigned int disp = (cs0 & 0x0fffffff);
|
|
||||||
// int sgnd_disp = ((int) (disp << 4)) >> 1;
|
|
||||||
/* PUTTSD obviously doesn't take %ctpr{j} parameter. TODO: beware of
|
|
||||||
an optional predicate which may control its execution which is
|
|
||||||
encoded via `SS.ctcond.psrc' and `SS.ts_opc == PUTTSDC{P,N}' in
|
|
||||||
case of `SS.type == 1' (see C.21.4). I wonder if `ct %ctpr<j>'
|
|
||||||
encoded in `SS.ctop' under the same `SS.ctcond' takes an effect in
|
|
||||||
such a case. */
|
|
||||||
// my_printf ("%s0x%llx", type == PUTTSD ? "" : ", ",
|
|
||||||
/* FIXME: this way I ensure that it'll work correctly
|
|
||||||
both on 32 and 64-bit hosts. */
|
|
||||||
// (unsigned long long) (instr_addr + sgnd_disp));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (type == PREF) {
|
|
||||||
// unsigned int pdisp = (bundle->cs0 & 0x0ffffff0) >> 4;
|
|
||||||
// unsigned int ipd = (bundle->cs0 & 0x00000008) >> 3;
|
|
||||||
// unsigned int prefr = bundle->cs0 & 0x00000007;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void gen_cs1(DisasContext *dc, CPUE2KState *env)
|
|
||||||
{
|
|
||||||
enum {
|
|
||||||
SETR0,
|
|
||||||
SETR1,
|
|
||||||
SETEI,
|
|
||||||
WAIT,
|
|
||||||
SETBR,
|
|
||||||
CALL,
|
|
||||||
MAS_OPC,
|
|
||||||
FLUSHR,
|
|
||||||
BG
|
|
||||||
};
|
|
||||||
|
|
||||||
const UnpackedBundle *bundle = &dc->bundle;
|
|
||||||
unsigned int cs1 = bundle->cs1;
|
|
||||||
unsigned int opc = (cs1 & 0xf0000000) >> 28;
|
|
||||||
|
|
||||||
if (opc == SETR0 || opc == SETR1 || opc == SETBR) {
|
|
||||||
unsigned int setbp = (cs1 & 0x08000000) >> 27;
|
|
||||||
unsigned int setbn = (cs1 & 0x04000000) >> 26;
|
|
||||||
|
|
||||||
/* Try to follow the same order of these instructions as in LDIS.
|
|
||||||
Presumably `vfrpsz' should come first, while `setbp' should be placed
|
|
||||||
between `setwd' and `setbn', but this is to be verified. I don't have
|
|
||||||
a binary with these commands by hand right now. */
|
|
||||||
|
|
||||||
if (opc == SETR1) {
|
|
||||||
if (! bundle->lts_present[0]) {
|
|
||||||
// my_printf ("<bogus vfrpsz>");
|
|
||||||
} else {
|
|
||||||
/* Find out if VFRPSZ is always encoded together with SETWD. This
|
|
||||||
seems to be the case even if no SETWD has been explicitly
|
|
||||||
specified. */
|
|
||||||
// unsigned int rpsz = (bundle->lts[0] & 0x0001f000) >> 12;
|
|
||||||
// my_printf ("vfrpsz rpsz = 0x%x", rpsz);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// FIXME: Should windowing registers be precomputed or not?
|
|
||||||
|
|
||||||
if (opc == SETR0 || opc == SETR1) {
|
|
||||||
if (! bundle->lts_present[0]) {
|
|
||||||
// TODO: <bogus setwd>
|
|
||||||
abort();
|
|
||||||
} else {
|
|
||||||
unsigned int lts0 = bundle->lts[0];
|
|
||||||
unsigned int wsz = (lts0 & 0x00000fe0) >> 5;
|
|
||||||
unsigned int nfx = (lts0 & 0x00000010) >> 4;
|
|
||||||
|
|
||||||
tcg_gen_movi_i32(e2k_cs.wsz, wsz * 2);
|
|
||||||
tcg_gen_movi_i32(e2k_cs.nfx, nfx);
|
|
||||||
|
|
||||||
if (env->version >= 3) {
|
|
||||||
// DBL parameter of SETWD was added only starting from
|
|
||||||
// elbrus-v3.
|
|
||||||
unsigned int dbl = (lts0 & 0x00000008) >> 3;
|
|
||||||
tcg_gen_movi_i32(e2k_cs.dbl, dbl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (setbn) {
|
|
||||||
unsigned int rcur = (cs1 & 0x0003f000) >> 12;
|
|
||||||
unsigned int rsz = (cs1 & 0x00000fc0) >> 6;
|
|
||||||
unsigned int rbs = cs1 & 0x0000003f;
|
|
||||||
|
|
||||||
tcg_gen_movi_i32(e2k_cs.rcur, rcur * 2);
|
|
||||||
tcg_gen_movi_i32(e2k_cs.rsz, rsz * 2 + 2);
|
|
||||||
tcg_gen_movi_i32(e2k_cs.rbs, rbs * 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (setbp) {
|
|
||||||
unsigned int psz = (cs1 & 0x007c0000) >> 18;
|
|
||||||
|
|
||||||
tcg_gen_movi_i32(e2k_cs.psz, psz);
|
|
||||||
}
|
|
||||||
} else if (opc == SETEI) {
|
|
||||||
/* Verify that CS1.param.sft = CS1.param[27] is equal to zero as required
|
|
||||||
in C.14.3. */
|
|
||||||
unsigned int sft = (cs1 & 0x08000000) >> 27;
|
|
||||||
// unsigned int eir = (cs1 & 0x000000ff);
|
|
||||||
|
|
||||||
if (sft) {
|
|
||||||
// my_printf ("%s", mcpu >= 2 ? "setsft" : "unimp");
|
|
||||||
} else {
|
|
||||||
// my_printf ("setei 0x%x", eir);
|
|
||||||
}
|
|
||||||
} else if (opc == WAIT) {
|
|
||||||
// unsigned int ma_c = (cs1 & 0x00000020) >> 5;
|
|
||||||
// unsigned int fl_c = (cs1 & 0x00000010) >> 4;
|
|
||||||
unsigned int ld_c = (cs1 & 0x00000008) >> 3;
|
|
||||||
unsigned int st_c = (cs1 & 0x00000004) >> 2;
|
|
||||||
// unsigned int all_e = (cs1 & 0x00000002) >> 1;
|
|
||||||
// unsigned int all_c = cs1 & 0x00000001;
|
|
||||||
|
|
||||||
if (env->version >= 5) {
|
|
||||||
/* `sa{l,s}' fields are `elbrus-v5'-specific. Each of them makes sense
|
|
||||||
only in the presence of `{ld,st}_c == 1' respectively. */
|
|
||||||
if (ld_c) {
|
|
||||||
// unsigned int sal = (cs1 & 0x00000100) >> 8;
|
|
||||||
// my_printf ("sal = %d, ", sal);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (st_c) {
|
|
||||||
// unsigned int sas = (cs1 & 0x00000080) >> 7;
|
|
||||||
// my_printf ("sas = %d, ", sas);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (env->version >= 2) {
|
|
||||||
/* `trap' field was introduced starting from `elbrus-v2'. */
|
|
||||||
// unsigned int trap = (cs1 & 0x00000040) >> 6;
|
|
||||||
// my_printf ("trap = %d, ", trap);
|
|
||||||
}
|
|
||||||
|
|
||||||
// my_printf ("ma_c = %d, fl_c = %d, ld_c = %d, st_c = %d, all_e = %d, "
|
|
||||||
// "all_c = %d", ma_c, fl_c, ld_c, st_c, all_e, all_c);
|
|
||||||
} else if (opc == CALL) {
|
|
||||||
unsigned int ctop = (bundle->ss & 0x00000c00) >> 10;
|
|
||||||
/* In C.17.4 it's said that other bits in CS1.param except for the
|
|
||||||
seven lowermost ones are ignored. */
|
|
||||||
// unsigned int wbs = cs1 & 0x7f;
|
|
||||||
|
|
||||||
if (ctop) {
|
|
||||||
// my_printf ("call %%ctpr%d, wbs = 0x%x", ctop, wbs);
|
|
||||||
// print_ctcond (info, instr->ss & 0x1ff);
|
|
||||||
} else {
|
|
||||||
unsigned int cs1_ctopc = (cs1 & 0x380) >> 7;
|
|
||||||
/* CS1.param.ctpopc == HCALL. CS0 is required to encode HCALL. */
|
|
||||||
if (cs1_ctopc == 2 && bundle->cs0_present) {
|
|
||||||
unsigned int cs0 = bundle->cs0;
|
|
||||||
unsigned int cs0_opc = (cs0 & 0xf0000000) >> 28;
|
|
||||||
/* CS0.opc == HCALL, which means
|
|
||||||
CS0.opc.ctpr == CS0.opc.ctp_opc == 0 */
|
|
||||||
if (cs0_opc == 0) {
|
|
||||||
// unsigned int hdisp = (cs0 & 0x1e) >> 1;
|
|
||||||
// my_printf ("hcall 0x%x, wbs = 0x%x", hdisp, wbs);
|
|
||||||
// print_ctcond (info, instr->ss & 0x1ff);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// my_printf ("<bogus call>");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else if (opc == MAS_OPC) {
|
|
||||||
/* Note that LDIS doesn't print it out as a standalone instruction. */
|
|
||||||
// unsigned int mas = cs1 & 0x0fffffff;
|
|
||||||
|
|
||||||
// my_printf ("mas 0x%x", mas);
|
|
||||||
} else if (opc == FLUSHR) {
|
|
||||||
/* . . . these stupid engineers off! FLUSHR seems to be responsible for
|
|
||||||
encoding both `flushr' and `flushc'. Moreover, according to their
|
|
||||||
logic it should be possible to encode them simultaneously. */
|
|
||||||
|
|
||||||
/* Check for `CS1.param.flr'. */
|
|
||||||
if (cs1 & 0x00000001) {
|
|
||||||
// my_printf ("flushr");
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Check for `CS1.param.flc'. */
|
|
||||||
if (cs1 & 0x00000002) {
|
|
||||||
// my_printf ("flushc");
|
|
||||||
}
|
|
||||||
} else if (opc == BG) {
|
|
||||||
/* Hopefully, `vfbg' is the only instruction encoded by BG. I'm currently
|
|
||||||
unable to find other ones in `iset-v5.single' at least . . . */
|
|
||||||
// unsigned int chkm4 = (cs1 & 0x00010000) >> 16;
|
|
||||||
// unsigned int dmask = (cs1 & 0x0000ff00) >> 8;
|
|
||||||
// unsigned int umsk = cs1 & 0x000000ff;
|
|
||||||
|
|
||||||
/* Print its fields in the order proposed in C.14.10. */
|
|
||||||
// my_printf ("vfbg umask = 0x%x, dmask = 0x%x, chkm4 = 0x%x",
|
|
||||||
// umsk, dmask, chkm4);
|
|
||||||
} else {
|
|
||||||
// my_printf ("unimp");
|
|
||||||
abort();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void gen_jmp(DisasContext *dc, target_ulong next_pc)
|
|
||||||
{
|
|
||||||
unsigned int ctcond = dc->jmp.cond;
|
|
||||||
unsigned int cond_type = (ctcond & 0x1e0) >> 5;
|
|
||||||
unsigned int psrc = (ctcond & 0x01f);
|
|
||||||
// bool not_preg = cond_type == 3 || cond_type == 7 || cond_type == 0xe;
|
|
||||||
|
|
||||||
if (cond_type == 1) {
|
|
||||||
dc->base.is_jmp = DISAS_NORETURN;
|
|
||||||
tcg_gen_mov_tl(e2k_cs.pc, dc->jmp.dest);
|
|
||||||
tcg_gen_exit_tb(NULL, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: temporary only preg cond
|
|
||||||
if (cond_type == 2) {
|
|
||||||
TCGv_i64 preg = e2k_get_preg(dc, psrc);
|
|
||||||
TCGv_i64 cond = tcg_const_i64(1);
|
|
||||||
TCGv next = tcg_const_tl(next_pc);
|
|
||||||
dc->base.is_jmp = DISAS_NORETURN;
|
|
||||||
tcg_gen_movcond_i64(TCG_COND_EQ, e2k_cs.pc,
|
|
||||||
preg, cond,
|
|
||||||
dc->jmp.dest, next
|
|
||||||
);
|
|
||||||
tcg_gen_exit_tb(NULL, 0);
|
|
||||||
tcg_temp_free_i64(cond);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* These types of conditions involve a (probably negated) predicate
|
|
||||||
register. */
|
|
||||||
if (cond_type == 2
|
|
||||||
|| cond_type == 3
|
|
||||||
|| cond_type == 6
|
|
||||||
|| cond_type == 7
|
|
||||||
|| cond_type == 0xe
|
|
||||||
|| cond_type == 0xf)
|
|
||||||
{
|
|
||||||
// preg
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cond_type == 4
|
|
||||||
|| cond_type == 6
|
|
||||||
|| cond_type == 0xe)
|
|
||||||
{
|
|
||||||
if (cond_type == 6 || cond_type == 0xe) {
|
|
||||||
// or
|
|
||||||
}
|
|
||||||
|
|
||||||
// %LOOP_END
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cond_type == 5
|
|
||||||
|| cond_type == 7
|
|
||||||
|| cond_type == 0xf)
|
|
||||||
{
|
|
||||||
if(cond_type == 7
|
|
||||||
|| cond_type == 0xf)
|
|
||||||
{
|
|
||||||
// AND
|
|
||||||
}
|
|
||||||
// %NOT_LOOP_END
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cond_type == 8) {
|
|
||||||
// %MLOCK
|
|
||||||
/* It's not clearly said in C.17.1.2 of iset-vX.single if the uppermost
|
|
||||||
fourth bit in `psrc' has any meaning at all. */
|
|
||||||
if (psrc & 0xf) {
|
|
||||||
// static const int conv[] = {0, 1, 3, 4};
|
|
||||||
int i;
|
|
||||||
|
|
||||||
// %dt_al
|
|
||||||
for (i = 0; i < 4; i++) {
|
|
||||||
if (psrc & (1 << i)) {
|
|
||||||
// i
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* `lock_cond || pl_cond' control transfer conditions. */
|
|
||||||
if (cond_type == 9) {
|
|
||||||
unsigned int type = (psrc & 0x18) >> 3;
|
|
||||||
if (type == 0) {
|
|
||||||
// static const int cmp_num_to_alc[] = {0, 1, 3, 4};
|
|
||||||
// unsigned int cmp_num = (psrc & 0x6) >> 1;
|
|
||||||
// unsigned int neg = psrc & 0x1;
|
|
||||||
|
|
||||||
// my_printf ("%%MLOCK || %s%%cmp%d", neg ? "~" : "",
|
|
||||||
// cmp_num_to_alc[cmp_num]);
|
|
||||||
} else if (type == 1) {
|
|
||||||
// unsigned int cmp_jk = (psrc & 0x4) >> 2;
|
|
||||||
// unsigned int negj = (psrc & 0x2) >> 1;
|
|
||||||
// unsigned int negk = psrc & 0x1;
|
|
||||||
|
|
||||||
// my_printf ("%%MLOCK || %s%%cmp%d || %s%%cmp%d",
|
|
||||||
// negj ? "~" : "", cmp_jk == 0 ? 0 : 3,
|
|
||||||
// negk ? "~" : "", cmp_jk == 0 ? 1 : 4);
|
|
||||||
} else if (type == 2) {
|
|
||||||
// unsigned int clp_num = (psrc & 0x6) >> 1;
|
|
||||||
// unsigned int neg = psrc & 0x1;
|
|
||||||
|
|
||||||
// "%%MLOCK || %s%%clp%d", neg ? "~" : "", clp_num
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cond_type >= 0xa && cond_type <= 0xd) {
|
|
||||||
// reserved condition type
|
|
||||||
qemu_log_mask(LOG_UNIMP, "Undefined control transfer type %#x\n", cond_type);
|
|
||||||
abort();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static target_ulong disas_e2k_insn(DisasContext *dc, CPUState *cs)
|
static target_ulong disas_e2k_insn(DisasContext *dc, CPUState *cs)
|
||||||
{
|
{
|
||||||
E2KCPU *cpu = E2K_CPU(cs);
|
E2KCPU *cpu = E2K_CPU(cs);
|
||||||
|
@ -655,32 +211,50 @@ static target_ulong disas_e2k_insn(DisasContext *dc, CPUState *cs)
|
||||||
/* TODO: exception, check bundle_len */
|
/* TODO: exception, check bundle_len */
|
||||||
target_ulong pc_next = dc->pc + bundle_len;
|
target_ulong pc_next = dc->pc + bundle_len;
|
||||||
|
|
||||||
reset_is_jmp();
|
dc->jmp.cond = tcg_const_i64(0);
|
||||||
|
|
||||||
if (bundle->cs0_present) {
|
|
||||||
gen_cs0(dc, env);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bundle->cs1_present) {
|
|
||||||
gen_cs1(dc, env);
|
|
||||||
}
|
|
||||||
|
|
||||||
e2k_alc_gen(dc);
|
e2k_alc_gen(dc);
|
||||||
|
e2k_control_gen(dc);
|
||||||
|
|
||||||
e2k_alc_commit(dc);
|
e2k_alc_commit(dc);
|
||||||
e2k_win_commit(dc);
|
e2k_win_commit(dc);
|
||||||
|
|
||||||
// Control transfer
|
/* Control transfer */
|
||||||
if (env->interrupt_index != 0) {
|
switch(dc->base.is_jmp) {
|
||||||
|
case DISAS_NEXT:
|
||||||
|
case DISAS_TOO_MANY:
|
||||||
|
dc->base.pc_next = pc_next;
|
||||||
|
break;
|
||||||
|
case DISAS_NORETURN:
|
||||||
|
/* exception */
|
||||||
tcg_gen_movi_tl(e2k_cs.pc, dc->pc);
|
tcg_gen_movi_tl(e2k_cs.pc, dc->pc);
|
||||||
tcg_gen_exit_tb(NULL, 0);
|
tcg_gen_exit_tb(NULL, 0);
|
||||||
dc->base.is_jmp = DISAS_NORETURN;
|
break;
|
||||||
} else if (dc->jmp.cond != 0) {
|
case STATIC_JUMP:
|
||||||
// TODO: move condition compute before commit
|
tcg_gen_mov_i64(e2k_cs.pc, dc->jmp.dest);
|
||||||
gen_jmp(dc, pc_next);
|
tcg_gen_exit_tb(NULL, 0);
|
||||||
|
break;
|
||||||
|
case DYNAMIC_JUMP: {
|
||||||
|
TCGv_i64 one = tcg_const_i64(1);
|
||||||
|
TCGv_i64 next = tcg_const_i64(pc_next);
|
||||||
|
tcg_gen_movcond_i64(TCG_COND_EQ, e2k_cs.pc,
|
||||||
|
dc->jmp.cond, one,
|
||||||
|
dc->jmp.dest, next
|
||||||
|
);
|
||||||
|
tcg_temp_free_i64(next);
|
||||||
|
tcg_temp_free_i64(one);
|
||||||
|
tcg_gen_exit_tb(NULL, 0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
/* TODO: unreachable */
|
||||||
|
abort();
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Free temporary values.
|
tcg_temp_free_i64(dc->jmp.cond);
|
||||||
|
|
||||||
|
/* Free temporary values */
|
||||||
while(dc->t32_len) {
|
while(dc->t32_len) {
|
||||||
tcg_temp_free_i32(dc->t32[--dc->t32_len]);
|
tcg_temp_free_i32(dc->t32[--dc->t32_len]);
|
||||||
}
|
}
|
||||||
|
@ -699,8 +273,11 @@ static target_ulong disas_e2k_insn(DisasContext *dc, CPUState *cs)
|
||||||
static void e2k_tr_init_disas_context(DisasContextBase *db, CPUState *cs)
|
static void e2k_tr_init_disas_context(DisasContextBase *db, CPUState *cs)
|
||||||
{
|
{
|
||||||
DisasContext *dc = container_of(db, DisasContext, base);
|
DisasContext *dc = container_of(db, DisasContext, base);
|
||||||
|
E2KCPU *cpu = E2K_CPU(cs);
|
||||||
|
CPUE2KState *env = &cpu->env;
|
||||||
|
|
||||||
dc->pc = dc->base.pc_first;
|
dc->pc = dc->base.pc_first;
|
||||||
|
dc->version = env->version;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void e2k_tr_insn_start(DisasContextBase *db, CPUState *cs)
|
static void e2k_tr_insn_start(DisasContextBase *db, CPUState *cs)
|
||||||
|
@ -721,17 +298,11 @@ static void e2k_tr_translate_insn(DisasContextBase *db, CPUState *cs)
|
||||||
{
|
{
|
||||||
DisasContext *dc = container_of(db, DisasContext, base);
|
DisasContext *dc = container_of(db, DisasContext, base);
|
||||||
dc->pc = dc->base.pc_next;
|
dc->pc = dc->base.pc_next;
|
||||||
dc->base.pc_next = disas_e2k_insn(dc, cs);
|
disas_e2k_insn(dc, cs);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void e2k_tr_tb_start(DisasContextBase *db, CPUState *cs)
|
static void e2k_tr_tb_start(DisasContextBase *db, CPUState *cs)
|
||||||
{
|
{
|
||||||
DisasContext *dc = container_of(db, DisasContext, base);
|
|
||||||
// E2KCPU *cpu = E2K_CPU(cs);
|
|
||||||
// CPUE2KState *env = &cpu->env;
|
|
||||||
|
|
||||||
dc->jmp.cond = COND_NEVER;
|
|
||||||
dc->jmp.dest = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void e2k_tr_tb_stop(DisasContextBase *db, CPUState *cs)
|
static void e2k_tr_tb_stop(DisasContextBase *db, CPUState *cs)
|
||||||
|
@ -810,7 +381,6 @@ void e2k_tcg_initialize(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
e2k_cs.win_ptr = tcg_global_mem_new_ptr(cpu_env, offsetof(CPUE2KState, win_ptr), "win_ptr");
|
e2k_cs.win_ptr = tcg_global_mem_new_ptr(cpu_env, offsetof(CPUE2KState, win_ptr), "win_ptr");
|
||||||
e2k_cs.is_jmp = tcg_global_mem_new_i32(cpu_env, offsetof(CPUE2KState, is_jmp), "is_jmp");
|
|
||||||
|
|
||||||
for (i = 0; i < WREGS_SIZE; i++) {
|
for (i = 0; i < WREGS_SIZE; i++) {
|
||||||
snprintf(buf, ARRAY_SIZE(buf), "%%r%d", i);
|
snprintf(buf, ARRAY_SIZE(buf), "%%r%d", i);
|
||||||
|
|
|
@ -3,7 +3,8 @@
|
||||||
|
|
||||||
#include "exec/translator.h"
|
#include "exec/translator.h"
|
||||||
|
|
||||||
#define COND_NEVER 0
|
#define STATIC_JUMP DISAS_TARGET_0
|
||||||
|
#define DYNAMIC_JUMP DISAS_TARGET_1
|
||||||
|
|
||||||
#define GET_BIT(v, index) (((v) >> (index)) & 1)
|
#define GET_BIT(v, index) (((v) >> (index)) & 1)
|
||||||
#define GET_FIELD(v, start, end) \
|
#define GET_FIELD(v, start, end) \
|
||||||
|
@ -99,6 +100,8 @@ typedef struct DisasContext {
|
||||||
UnpackedBundle bundle;
|
UnpackedBundle bundle;
|
||||||
target_ulong pc;
|
target_ulong pc;
|
||||||
|
|
||||||
|
int version;
|
||||||
|
|
||||||
// Temporary values.
|
// Temporary values.
|
||||||
TCGv_i32 t32[16];
|
TCGv_i32 t32[16];
|
||||||
TCGv_i64 t64[16];
|
TCGv_i64 t64[16];
|
||||||
|
@ -110,13 +113,11 @@ typedef struct DisasContext {
|
||||||
|
|
||||||
Result alc[6];
|
Result alc[6];
|
||||||
struct {
|
struct {
|
||||||
// raw condition code from SS[8:0]
|
|
||||||
unsigned int cond;
|
|
||||||
TCGv dest;
|
TCGv dest;
|
||||||
|
TCGv_i64 cond;
|
||||||
} jmp;
|
} jmp;
|
||||||
} DisasContext;
|
} DisasContext;
|
||||||
|
|
||||||
|
|
||||||
static inline TCGv_i32 e2k_get_temp_i32(DisasContext *dc)
|
static inline TCGv_i32 e2k_get_temp_i32(DisasContext *dc)
|
||||||
{
|
{
|
||||||
assert(dc->t32_len < ARRAY_SIZE(dc->t32));
|
assert(dc->t32_len < ARRAY_SIZE(dc->t32));
|
||||||
|
@ -157,6 +158,8 @@ void e2k_gen_store_greg(int reg, TCGv_i64 val);
|
||||||
|
|
||||||
void e2k_gen_exception(DisasContext *dc, int which);
|
void e2k_gen_exception(DisasContext *dc, int which);
|
||||||
|
|
||||||
|
void e2k_control_gen(DisasContext *dc);
|
||||||
|
|
||||||
void e2k_alc_gen(DisasContext *dc);
|
void e2k_alc_gen(DisasContext *dc);
|
||||||
void e2k_alc_commit(DisasContext *dc);
|
void e2k_alc_commit(DisasContext *dc);
|
||||||
|
|
||||||
|
|
|
@ -133,8 +133,8 @@ static void gen_channel(DisasContext *dc, int chan)
|
||||||
{
|
{
|
||||||
const UnpackedBundle *bundle = &dc->bundle;
|
const UnpackedBundle *bundle = &dc->bundle;
|
||||||
unsigned int als = bundle->als[chan];
|
unsigned int als = bundle->als[chan];
|
||||||
int opc = (als >> 24) & 0x7f;
|
int opc = GET_FIELD(als, 24, 30);
|
||||||
int sm = als >> 31;
|
int sm = GET_BIT(als, 31);
|
||||||
unsigned int dst = als & 0xff;
|
unsigned int dst = als & 0xff;
|
||||||
bool is_cmp = false;
|
bool is_cmp = false;
|
||||||
Result res = { 0 };
|
Result res = { 0 };
|
||||||
|
@ -246,25 +246,23 @@ static void gen_channel(DisasContext *dc, int chan)
|
||||||
// TODO: getfd
|
// TODO: getfd
|
||||||
abort();
|
abort();
|
||||||
break;
|
break;
|
||||||
case 0x21: // cmp{op}sd
|
case 0x21: { // cmp{op}sd
|
||||||
do {
|
is_cmp = true;
|
||||||
is_cmp = true;
|
unsigned int cmp_op = GET_FIELD(als, 5, 7);
|
||||||
unsigned int cmp_op = (als & 0xe0) >> 5;
|
// TODO: move to separate function
|
||||||
// unsigned int index = als & 0x1f;
|
switch(cmp_op) {
|
||||||
// TODO: move to separate function
|
case 1: // unsigned less
|
||||||
switch(cmp_op) {
|
tcg_gen_setcond_i64(TCG_COND_LEU, tmp_dst, cpu_src1, cpu_src2);
|
||||||
case 1: // unsigned less
|
break;
|
||||||
tcg_gen_setcond_i64(TCG_COND_LEU, tmp_dst, cpu_src1, cpu_src2);
|
case 2: // equal
|
||||||
break;
|
tcg_gen_setcond_i64(TCG_COND_EQ, tmp_dst, cpu_src1, cpu_src2);
|
||||||
case 2: // equal
|
break;
|
||||||
tcg_gen_setcond_i64(TCG_COND_EQ, tmp_dst, cpu_src1, cpu_src2);
|
default:
|
||||||
break;
|
abort();
|
||||||
default:
|
break;
|
||||||
abort();
|
}
|
||||||
break;
|
|
||||||
}
|
|
||||||
} while(0);
|
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
case 0x40: // TODO: udivs used as temporary UD
|
case 0x40: // TODO: udivs used as temporary UD
|
||||||
e2k_gen_exception(dc, 1);
|
e2k_gen_exception(dc, 1);
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -0,0 +1,428 @@
|
||||||
|
#include "qemu/osdep.h"
|
||||||
|
#include "qemu.h"
|
||||||
|
#include "tcg/tcg-op.h"
|
||||||
|
#include "exec/log.h"
|
||||||
|
#include "exec/translator.h"
|
||||||
|
#include "translate.h"
|
||||||
|
|
||||||
|
static void gen_cs0(DisasContext *dc)
|
||||||
|
{
|
||||||
|
typedef enum {
|
||||||
|
NOTHING,
|
||||||
|
IBRANCH,
|
||||||
|
PREF,
|
||||||
|
PUTTSD,
|
||||||
|
DONE,
|
||||||
|
HRET,
|
||||||
|
GLAUNCH,
|
||||||
|
DISP,
|
||||||
|
SDISP,
|
||||||
|
GETTSD,
|
||||||
|
LDISP,
|
||||||
|
RETURN
|
||||||
|
} cs0_type;
|
||||||
|
|
||||||
|
static cs0_type cs0_ops[4][4] = {
|
||||||
|
{IBRANCH, PREF, PUTTSD, DONE},
|
||||||
|
{DISP, NOTHING, SDISP, GETTSD},
|
||||||
|
{DISP, LDISP, SDISP, GETTSD},
|
||||||
|
{DISP, NOTHING, SDISP, RETURN}
|
||||||
|
};
|
||||||
|
const UnpackedBundle *bundle = &dc->bundle;
|
||||||
|
uint32_t cs0 = bundle->cs0;
|
||||||
|
|
||||||
|
unsigned int ctpr = (cs0 & 0xc0000000) >> 30;
|
||||||
|
unsigned int ctp_opc = (cs0 & 0x30000000) >> 28;
|
||||||
|
unsigned int param_type = (cs0 & 0x00000007);
|
||||||
|
cs0_type type = cs0_ops[ctpr][ctp_opc];
|
||||||
|
|
||||||
|
if (type == RETURN && param_type == 1) {
|
||||||
|
type = GETTSD;
|
||||||
|
} else if (type == DONE) {
|
||||||
|
if (param_type == 3) {
|
||||||
|
type = HRET;
|
||||||
|
} else if (param_type == 4) {
|
||||||
|
type = GLAUNCH;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == IBRANCH || type == DONE || type == HRET || type == GLAUNCH) {
|
||||||
|
/* IBRANCH, DONE, HRET and GLAUNCH are special because they require SS
|
||||||
|
to be properly encoded. */
|
||||||
|
if (! bundle->ss_present
|
||||||
|
/* SS.ctop should be equal to zero for IBRANCH, DONE, HRET and
|
||||||
|
GLAUNCH (see C.17.1.1, note that they don't mention the latter two
|
||||||
|
instructions there which is probably an omission ). */
|
||||||
|
|| (bundle->ss & 0x00000c00))
|
||||||
|
{
|
||||||
|
// TODO: invalid
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
/* Don't output either of the aforementioned instructions under "never"
|
||||||
|
condition. Don't disassemble CS0 being a part of HCALL. Unlike ldis
|
||||||
|
HCALL is currently disassembled on behalf of CS1. */
|
||||||
|
else if ((bundle->ss & 0x1ff)
|
||||||
|
&& !(bundle->cs1_present
|
||||||
|
/* CS1.opc == CALL */
|
||||||
|
&& (bundle->cs1 & 0xf0000000) >> 28 == 5
|
||||||
|
/* CS1.param.ctopc == HCALL */
|
||||||
|
&& (bundle->cs1 & 0x380) >> 7 == 2))
|
||||||
|
{
|
||||||
|
if (type == IBRANCH) {
|
||||||
|
/* C0F2 has `disp' field. In `C0F1' it's called `param'. Is this
|
||||||
|
the only difference between these two formats? Funnily enough,
|
||||||
|
DONE is also C0F2 and thus has `disp', though it obviously
|
||||||
|
makes no sense for it. */
|
||||||
|
unsigned int disp = (cs0 & 0x0fffffff);
|
||||||
|
/* Calculate a signed displacement in bytes. */
|
||||||
|
int sdisp = ((int) (disp << 4)) >> 1;
|
||||||
|
target_ulong tgt = dc->pc + sdisp;
|
||||||
|
TCGv dest = e2k_get_temp(dc);
|
||||||
|
tcg_gen_movi_tl(dest, tgt);
|
||||||
|
dc->jmp.dest = dest;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* Note that according to Table B.4.1 it's possible to obtain
|
||||||
|
` gettsd %ctpr{1,2} with an invalid value for CS0.param.type. */
|
||||||
|
if (type == GETTSD && param_type != 1) {
|
||||||
|
// invalid
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == DISP
|
||||||
|
|| type == SDISP
|
||||||
|
|| type == LDISP
|
||||||
|
/* Note that RETURN is said to be COPF1. I can't understand what its
|
||||||
|
`CS0.param' is needed for: all of the bits except the three
|
||||||
|
lowermost ones are undefined, while the latter also known as "type"
|
||||||
|
field should be filled in with zeroes. */
|
||||||
|
|| type == RETURN
|
||||||
|
/* GETTSD has as meaningless `CS0.param' as RETURN. The only
|
||||||
|
difference is that its `CS0.param.type' should be equal to `1'. I
|
||||||
|
wonder if I should check for that and output something like
|
||||||
|
"invalid gettsd" if this turns out not to be the case . . . */
|
||||||
|
|| type == GETTSD)
|
||||||
|
{
|
||||||
|
// ctpr
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == SDISP) {
|
||||||
|
// my_printf (", 0x%x", cs0 & 0x1f);
|
||||||
|
} else if (type == DISP
|
||||||
|
|| type == LDISP
|
||||||
|
|| type == PUTTSD)
|
||||||
|
{
|
||||||
|
// unsigned int disp = (cs0 & 0x0fffffff);
|
||||||
|
// int sgnd_disp = ((int) (disp << 4)) >> 1;
|
||||||
|
/* PUTTSD obviously doesn't take %ctpr{j} parameter. TODO: beware of
|
||||||
|
an optional predicate which may control its execution which is
|
||||||
|
encoded via `SS.ctcond.psrc' and `SS.ts_opc == PUTTSDC{P,N}' in
|
||||||
|
case of `SS.type == 1' (see C.21.4). I wonder if `ct %ctpr<j>'
|
||||||
|
encoded in `SS.ctop' under the same `SS.ctcond' takes an effect in
|
||||||
|
such a case. */
|
||||||
|
// my_printf ("%s0x%llx", type == PUTTSD ? "" : ", ",
|
||||||
|
/* FIXME: this way I ensure that it'll work correctly
|
||||||
|
both on 32 and 64-bit hosts. */
|
||||||
|
// (unsigned long long) (instr_addr + sgnd_disp));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type == PREF) {
|
||||||
|
// unsigned int pdisp = (bundle->cs0 & 0x0ffffff0) >> 4;
|
||||||
|
// unsigned int ipd = (bundle->cs0 & 0x00000008) >> 3;
|
||||||
|
// unsigned int prefr = bundle->cs0 & 0x00000007;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gen_cs1(DisasContext *dc)
|
||||||
|
{
|
||||||
|
enum {
|
||||||
|
SETR0,
|
||||||
|
SETR1,
|
||||||
|
SETEI,
|
||||||
|
WAIT,
|
||||||
|
SETBR,
|
||||||
|
CALL,
|
||||||
|
MAS_OPC,
|
||||||
|
FLUSHR,
|
||||||
|
BG
|
||||||
|
};
|
||||||
|
|
||||||
|
const UnpackedBundle *bundle = &dc->bundle;
|
||||||
|
unsigned int cs1 = bundle->cs1;
|
||||||
|
unsigned int opc = (cs1 & 0xf0000000) >> 28;
|
||||||
|
|
||||||
|
if (opc == SETR0 || opc == SETR1 || opc == SETBR) {
|
||||||
|
unsigned int setbp = (cs1 & 0x08000000) >> 27;
|
||||||
|
unsigned int setbn = (cs1 & 0x04000000) >> 26;
|
||||||
|
|
||||||
|
/* Try to follow the same order of these instructions as in LDIS.
|
||||||
|
Presumably `vfrpsz' should come first, while `setbp' should be placed
|
||||||
|
between `setwd' and `setbn', but this is to be verified. I don't have
|
||||||
|
a binary with these commands by hand right now. */
|
||||||
|
|
||||||
|
if (opc == SETR1) {
|
||||||
|
if (! bundle->lts_present[0]) {
|
||||||
|
// my_printf ("<bogus vfrpsz>");
|
||||||
|
} else {
|
||||||
|
/* Find out if VFRPSZ is always encoded together with SETWD. This
|
||||||
|
seems to be the case even if no SETWD has been explicitly
|
||||||
|
specified. */
|
||||||
|
// unsigned int rpsz = (bundle->lts[0] & 0x0001f000) >> 12;
|
||||||
|
// my_printf ("vfrpsz rpsz = 0x%x", rpsz);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME: Should windowing registers be precomputed or not?
|
||||||
|
|
||||||
|
if (opc == SETR0 || opc == SETR1) {
|
||||||
|
if (! bundle->lts_present[0]) {
|
||||||
|
// TODO: <bogus setwd>
|
||||||
|
abort();
|
||||||
|
} else {
|
||||||
|
uint32_t lts0 = bundle->lts[0];
|
||||||
|
tcg_gen_movi_i32(e2k_cs.wsz, GET_FIELD(lts0, 5, 11) * 2);
|
||||||
|
tcg_gen_movi_i32(e2k_cs.nfx, GET_BIT(lts0, 4));
|
||||||
|
|
||||||
|
if (dc->version >= 3) {
|
||||||
|
tcg_gen_movi_i32(e2k_cs.dbl, GET_BIT(lts0, 3));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (setbn) {
|
||||||
|
unsigned int rcur = (cs1 & 0x0003f000) >> 12;
|
||||||
|
unsigned int rsz = (cs1 & 0x00000fc0) >> 6;
|
||||||
|
unsigned int rbs = cs1 & 0x0000003f;
|
||||||
|
|
||||||
|
tcg_gen_movi_i32(e2k_cs.rcur, rcur * 2);
|
||||||
|
tcg_gen_movi_i32(e2k_cs.rsz, rsz * 2 + 2);
|
||||||
|
tcg_gen_movi_i32(e2k_cs.rbs, rbs * 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (setbp) {
|
||||||
|
unsigned int psz = (cs1 & 0x007c0000) >> 18;
|
||||||
|
|
||||||
|
tcg_gen_movi_i32(e2k_cs.psz, psz);
|
||||||
|
}
|
||||||
|
} else if (opc == SETEI) {
|
||||||
|
/* Verify that CS1.param.sft = CS1.param[27] is equal to zero as required
|
||||||
|
in C.14.3. */
|
||||||
|
unsigned int sft = (cs1 & 0x08000000) >> 27;
|
||||||
|
// unsigned int eir = (cs1 & 0x000000ff);
|
||||||
|
|
||||||
|
if (sft) {
|
||||||
|
// my_printf ("%s", mcpu >= 2 ? "setsft" : "unimp");
|
||||||
|
} else {
|
||||||
|
// my_printf ("setei 0x%x", eir);
|
||||||
|
}
|
||||||
|
} else if (opc == WAIT) {
|
||||||
|
// unsigned int ma_c = (cs1 & 0x00000020) >> 5;
|
||||||
|
// unsigned int fl_c = (cs1 & 0x00000010) >> 4;
|
||||||
|
unsigned int ld_c = (cs1 & 0x00000008) >> 3;
|
||||||
|
unsigned int st_c = (cs1 & 0x00000004) >> 2;
|
||||||
|
// unsigned int all_e = (cs1 & 0x00000002) >> 1;
|
||||||
|
// unsigned int all_c = cs1 & 0x00000001;
|
||||||
|
|
||||||
|
if (dc->version >= 5) {
|
||||||
|
/* `sa{l,s}' fields are `elbrus-v5'-specific. Each of them makes sense
|
||||||
|
only in the presence of `{ld,st}_c == 1' respectively. */
|
||||||
|
if (ld_c) {
|
||||||
|
// unsigned int sal = (cs1 & 0x00000100) >> 8;
|
||||||
|
// my_printf ("sal = %d, ", sal);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (st_c) {
|
||||||
|
// unsigned int sas = (cs1 & 0x00000080) >> 7;
|
||||||
|
// my_printf ("sas = %d, ", sas);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dc->version >= 2) {
|
||||||
|
/* `trap' field was introduced starting from `elbrus-v2'. */
|
||||||
|
// unsigned int trap = (cs1 & 0x00000040) >> 6;
|
||||||
|
// my_printf ("trap = %d, ", trap);
|
||||||
|
}
|
||||||
|
|
||||||
|
// my_printf ("ma_c = %d, fl_c = %d, ld_c = %d, st_c = %d, all_e = %d, "
|
||||||
|
// "all_c = %d", ma_c, fl_c, ld_c, st_c, all_e, all_c);
|
||||||
|
} else if (opc == CALL) {
|
||||||
|
unsigned int ctop = (bundle->ss & 0x00000c00) >> 10;
|
||||||
|
/* In C.17.4 it's said that other bits in CS1.param except for the
|
||||||
|
seven lowermost ones are ignored. */
|
||||||
|
// unsigned int wbs = cs1 & 0x7f;
|
||||||
|
|
||||||
|
if (ctop) {
|
||||||
|
// my_printf ("call %%ctpr%d, wbs = 0x%x", ctop, wbs);
|
||||||
|
// print_ctcond (info, instr->ss & 0x1ff);
|
||||||
|
} else {
|
||||||
|
unsigned int cs1_ctopc = (cs1 & 0x380) >> 7;
|
||||||
|
/* CS1.param.ctpopc == HCALL. CS0 is required to encode HCALL. */
|
||||||
|
if (cs1_ctopc == 2 && bundle->cs0_present) {
|
||||||
|
unsigned int cs0 = bundle->cs0;
|
||||||
|
unsigned int cs0_opc = (cs0 & 0xf0000000) >> 28;
|
||||||
|
/* CS0.opc == HCALL, which means
|
||||||
|
CS0.opc.ctpr == CS0.opc.ctp_opc == 0 */
|
||||||
|
if (cs0_opc == 0) {
|
||||||
|
// unsigned int hdisp = (cs0 & 0x1e) >> 1;
|
||||||
|
// my_printf ("hcall 0x%x, wbs = 0x%x", hdisp, wbs);
|
||||||
|
// print_ctcond (info, instr->ss & 0x1ff);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// my_printf ("<bogus call>");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (opc == MAS_OPC) {
|
||||||
|
/* Note that LDIS doesn't print it out as a standalone instruction. */
|
||||||
|
// unsigned int mas = cs1 & 0x0fffffff;
|
||||||
|
|
||||||
|
// my_printf ("mas 0x%x", mas);
|
||||||
|
} else if (opc == FLUSHR) {
|
||||||
|
/* . . . these stupid engineers off! FLUSHR seems to be responsible for
|
||||||
|
encoding both `flushr' and `flushc'. Moreover, according to their
|
||||||
|
logic it should be possible to encode them simultaneously. */
|
||||||
|
|
||||||
|
/* Check for `CS1.param.flr'. */
|
||||||
|
if (cs1 & 0x00000001) {
|
||||||
|
// my_printf ("flushr");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check for `CS1.param.flc'. */
|
||||||
|
if (cs1 & 0x00000002) {
|
||||||
|
// my_printf ("flushc");
|
||||||
|
}
|
||||||
|
} else if (opc == BG) {
|
||||||
|
/* Hopefully, `vfbg' is the only instruction encoded by BG. I'm currently
|
||||||
|
unable to find other ones in `iset-v5.single' at least . . . */
|
||||||
|
// unsigned int chkm4 = (cs1 & 0x00010000) >> 16;
|
||||||
|
// unsigned int dmask = (cs1 & 0x0000ff00) >> 8;
|
||||||
|
// unsigned int umsk = cs1 & 0x000000ff;
|
||||||
|
|
||||||
|
/* Print its fields in the order proposed in C.14.10. */
|
||||||
|
// my_printf ("vfbg umask = 0x%x, dmask = 0x%x, chkm4 = 0x%x",
|
||||||
|
// umsk, dmask, chkm4);
|
||||||
|
} else {
|
||||||
|
// my_printf ("unimp");
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void gen_jmp(DisasContext *dc)
|
||||||
|
{
|
||||||
|
unsigned int ctcond = GET_FIELD(dc->bundle.ss, 0, 8);
|
||||||
|
unsigned int cond_type = (ctcond & 0x1e0) >> 5;
|
||||||
|
unsigned int psrc = (ctcond & 0x01f);
|
||||||
|
// bool not_preg = cond_type == 3 || cond_type == 7 || cond_type == 0xe;
|
||||||
|
|
||||||
|
if (cond_type == 1) {
|
||||||
|
dc->base.is_jmp = STATIC_JUMP;
|
||||||
|
tcg_gen_movi_i64(dc->jmp.cond, 1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cond_type == 2) {
|
||||||
|
dc->base.is_jmp = DYNAMIC_JUMP;
|
||||||
|
tcg_gen_mov_i64(dc->jmp.cond, e2k_get_preg(dc, psrc));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* These types of conditions involve a (probably negated) predicate
|
||||||
|
register. */
|
||||||
|
if (cond_type == 2
|
||||||
|
|| cond_type == 3
|
||||||
|
|| cond_type == 6
|
||||||
|
|| cond_type == 7
|
||||||
|
|| cond_type == 0xe
|
||||||
|
|| cond_type == 0xf)
|
||||||
|
{
|
||||||
|
// preg
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cond_type == 4
|
||||||
|
|| cond_type == 6
|
||||||
|
|| cond_type == 0xe)
|
||||||
|
{
|
||||||
|
if (cond_type == 6 || cond_type == 0xe) {
|
||||||
|
// or
|
||||||
|
}
|
||||||
|
|
||||||
|
// %LOOP_END
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cond_type == 5
|
||||||
|
|| cond_type == 7
|
||||||
|
|| cond_type == 0xf)
|
||||||
|
{
|
||||||
|
if(cond_type == 7
|
||||||
|
|| cond_type == 0xf)
|
||||||
|
{
|
||||||
|
// AND
|
||||||
|
}
|
||||||
|
// %NOT_LOOP_END
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cond_type == 8) {
|
||||||
|
// %MLOCK
|
||||||
|
/* It's not clearly said in C.17.1.2 of iset-vX.single if the uppermost
|
||||||
|
fourth bit in `psrc' has any meaning at all. */
|
||||||
|
if (psrc & 0xf) {
|
||||||
|
// static const int conv[] = {0, 1, 3, 4};
|
||||||
|
int i;
|
||||||
|
|
||||||
|
// %dt_al
|
||||||
|
for (i = 0; i < 4; i++) {
|
||||||
|
if (psrc & (1 << i)) {
|
||||||
|
// i
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* `lock_cond || pl_cond' control transfer conditions. */
|
||||||
|
if (cond_type == 9) {
|
||||||
|
unsigned int type = (psrc & 0x18) >> 3;
|
||||||
|
if (type == 0) {
|
||||||
|
// static const int cmp_num_to_alc[] = {0, 1, 3, 4};
|
||||||
|
// unsigned int cmp_num = (psrc & 0x6) >> 1;
|
||||||
|
// unsigned int neg = psrc & 0x1;
|
||||||
|
|
||||||
|
// my_printf ("%%MLOCK || %s%%cmp%d", neg ? "~" : "",
|
||||||
|
// cmp_num_to_alc[cmp_num]);
|
||||||
|
} else if (type == 1) {
|
||||||
|
// unsigned int cmp_jk = (psrc & 0x4) >> 2;
|
||||||
|
// unsigned int negj = (psrc & 0x2) >> 1;
|
||||||
|
// unsigned int negk = psrc & 0x1;
|
||||||
|
|
||||||
|
// my_printf ("%%MLOCK || %s%%cmp%d || %s%%cmp%d",
|
||||||
|
// negj ? "~" : "", cmp_jk == 0 ? 0 : 3,
|
||||||
|
// negk ? "~" : "", cmp_jk == 0 ? 1 : 4);
|
||||||
|
} else if (type == 2) {
|
||||||
|
// unsigned int clp_num = (psrc & 0x6) >> 1;
|
||||||
|
// unsigned int neg = psrc & 0x1;
|
||||||
|
|
||||||
|
// "%%MLOCK || %s%%clp%d", neg ? "~" : "", clp_num
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cond_type >= 0xa && cond_type <= 0xd) {
|
||||||
|
// reserved condition type
|
||||||
|
qemu_log_mask(LOG_UNIMP, "Undefined control transfer type %#x\n", cond_type);
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void e2k_control_gen(DisasContext *dc)
|
||||||
|
{
|
||||||
|
if (dc->bundle.cs0_present) {
|
||||||
|
gen_cs0(dc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dc->bundle.cs1_present) {
|
||||||
|
gen_cs1(dc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dc->bundle.ss_present) {
|
||||||
|
gen_jmp(dc);
|
||||||
|
}
|
||||||
|
}
|
|
@ -5,19 +5,19 @@
|
||||||
#include "exec/translator.h"
|
#include "exec/translator.h"
|
||||||
#include "translate.h"
|
#include "translate.h"
|
||||||
|
|
||||||
static inline void gen_abn_inc(TCGCond cond)
|
static inline void gen_abn_inc(DisasContext *dc, TCGCond cond)
|
||||||
{
|
{
|
||||||
TCGv_i32 t0 = tcg_temp_new_i32();
|
TCGv_i32 t0 = tcg_temp_new_i32();
|
||||||
TCGv_i32 t1 = tcg_temp_new_i32();
|
TCGv_i32 t1 = tcg_temp_new_i32();
|
||||||
TCGv_i32 zero = tcg_const_i32(0);
|
TCGv_i32 one = tcg_const_i32(1);
|
||||||
|
|
||||||
tcg_gen_addi_i32(t0, e2k_cs.rcur, 2);
|
tcg_gen_addi_i32(t0, e2k_cs.rcur, 2);
|
||||||
e2k_gen_wrap_i32(t1, t0, e2k_cs.rsz);
|
e2k_gen_wrap_i32(t1, t0, e2k_cs.rsz);
|
||||||
tcg_gen_movcond_i32(cond, e2k_cs.rcur,
|
tcg_gen_movcond_i32(cond, e2k_cs.rcur,
|
||||||
e2k_cs.is_jmp, zero,
|
dc->jmp.cond, one,
|
||||||
t1, e2k_cs.rcur);
|
t1, e2k_cs.rcur);
|
||||||
|
|
||||||
tcg_temp_free_i32(zero);
|
tcg_temp_free_i32(one);
|
||||||
tcg_temp_free_i32(t1);
|
tcg_temp_free_i32(t1);
|
||||||
tcg_temp_free_i32(t0);
|
tcg_temp_free_i32(t0);
|
||||||
}
|
}
|
||||||
|
@ -32,9 +32,9 @@ void e2k_win_commit(DisasContext *dc)
|
||||||
// unsigned int alc = (ss & 0x00030000) >> 16;
|
// unsigned int alc = (ss & 0x00030000) >> 16;
|
||||||
|
|
||||||
if (GET_BIT(ss, 21)) {
|
if (GET_BIT(ss, 21)) {
|
||||||
gen_abn_inc(TCG_COND_NE);
|
gen_abn_inc(dc, TCG_COND_EQ);
|
||||||
}
|
}
|
||||||
if (GET_BIT(ss, 22)) {
|
if (GET_BIT(ss, 22)) {
|
||||||
gen_abn_inc(TCG_COND_EQ);
|
gen_abn_inc(dc, TCG_COND_NE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue