#pragma ident "%Z%%M% %I% %E% SMI"
#include <sys/fpu/fpu_simulator.h>
#include <sys/fpu/globals.h>
#include <sys/regset.h>
#include <sys/privregs.h>
#include <sys/vis_simulator.h>
static enum ftt_type
fbcc_sim(
fp_inst_type pinst,
struct regs *pregs,
kfpu_t *pfpu)
{
fsr_type fsr;
int fbpcc = 0;
union {
fp_inst_type fi;
int32_t i;
} fp;
enum fcc_type fcc;
enum icc_type {
fbn, fbne, fblg, fbul, fbl, fbug, fbg, fbu,
fba, fbe, fbue, fbge, fbuge, fble, fbule, fbo
} icc;
uint_t annul, takeit;
if (((pinst.op3 >> 3) & 0xf) == 5)
fbpcc = 1;
fsr.ll = pfpu->fpu_fsr;
if (fbpcc) {
uint_t nfcc = (pinst.op3 >> 1) & 0x3;
switch (nfcc) {
case fcc_0:
fcc = fsr.fcc0;
break;
case fcc_1:
fcc = fsr.fcc1;
break;
case fcc_2:
fcc = fsr.fcc2;
break;
case fcc_3:
fcc = fsr.fcc3;
break;
}
} else {
fcc = fsr.fcc0;
}
icc = (enum icc_type) (pinst.rd & 0xf);
annul = pinst.rd & 0x10;
switch (icc) {
case fbn:
takeit = 0;
break;
case fbl:
takeit = fcc == fcc_less;
break;
case fbg:
takeit = fcc == fcc_greater;
break;
case fbu:
takeit = fcc == fcc_unordered;
break;
case fbe:
takeit = fcc == fcc_equal;
break;
case fblg:
takeit = (fcc == fcc_less) || (fcc == fcc_greater);
break;
case fbul:
takeit = (fcc == fcc_unordered) || (fcc == fcc_less);
break;
case fbug:
takeit = (fcc == fcc_unordered) || (fcc == fcc_greater);
break;
case fbue:
takeit = (fcc == fcc_unordered) || (fcc == fcc_equal);
break;
case fbge:
takeit = (fcc == fcc_greater) || (fcc == fcc_equal);
break;
case fble:
takeit = (fcc == fcc_less) || (fcc == fcc_equal);
break;
case fbne:
takeit = fcc != fcc_equal;
break;
case fbuge:
takeit = fcc != fcc_less;
break;
case fbule:
takeit = fcc != fcc_greater;
break;
case fbo:
takeit = fcc != fcc_unordered;
break;
case fba:
takeit = 1;
break;
}
if (takeit) {
uintptr_t tpc;
fp.fi = pinst;
tpc = pregs->r_pc;
if (annul && (icc == fba)) {
if (fbpcc) {
pregs->r_pc = tpc +
(int)((fp.i << 13) >> 11);
} else {
pregs->r_pc = tpc +
(int)((fp.i << 10) >> 8);
}
pregs->r_npc = pregs->r_pc + 4;
} else {
pregs->r_pc = pregs->r_npc;
if (fbpcc) {
pregs->r_npc = tpc +
(int)((fp.i << 13) >> 11);
} else {
pregs->r_npc = tpc +
(int)((fp.i << 10) >> 8);
}
}
} else {
if (annul) {
pregs->r_pc = pregs->r_npc + 4;
pregs->r_npc += 8;
} else {
pregs->r_pc = pregs->r_npc;
pregs->r_npc += 4;
}
}
return (ftt_none);
}
enum ftt_type
_fp_iu_simulator(
fp_simd_type *pfpsd,
fp_inst_type pinst,
struct regs *pregs,
void *prw,
kfpu_t *pfpu)
{
switch (pinst.hibits) {
case 0:
return (fbcc_sim(pinst, pregs, pfpu));
case 2:
switch (pinst.op3) {
case 0x28:
if (pinst.rs1 == 0x13)
return (vis_rdgsr(pfpsd, pinst, pregs,
prw, pfpu));
else
return (ftt_unimplemented);
case 0x30:
if (pinst.rd == 0x13)
return (vis_wrgsr(pfpsd, pinst, pregs,
prw, pfpu));
else
return (ftt_unimplemented);
case 0x2C:
return (movcc(pfpsd, pinst, pregs, prw, pfpu));
default:
return (ftt_unimplemented);
}
case 3:
return (fldst(pfpsd, pinst, pregs, prw));
default:
return (ftt_unimplemented);
}
}