trying to simplify branches
This commit is contained in:
+108
-52
@@ -45,88 +45,140 @@ void broadway::bx(ircolib::u32 instr) {
|
||||
void broadway::bcx(ircolib::u32 instr) {
|
||||
const bool link = instr & 1;
|
||||
const bool absolute = instr & 2;
|
||||
const ircolib::u8 bi = (instr >> 16) & 0x1f;
|
||||
const ircolib::u8 bo = (instr >> 11) & 0x1f;
|
||||
if (ircolib::is_bit_set(bo, 2))
|
||||
ctr--;
|
||||
|
||||
const bool ctr_ok = ircolib::is_bit_set(bo, 2) || ((ctr != 0) ^ (ircolib::is_bit_set(bo, 3)));
|
||||
const bool cond_ok = ircolib::is_bit_set(bo, 0) || (ircolib::is_bit_set(cr, 32 - bi) == ircolib::is_bit_set(bo, 1));
|
||||
|
||||
ircolib::s32 bd = (ircolib::s32(instr << 16) >> 16) & 0xFFFFFFFC;
|
||||
if (!absolute)
|
||||
bd += pc - 4;
|
||||
if (ctr_ok && cond_ok) {
|
||||
if (!absolute)
|
||||
bd += pc - 4;
|
||||
|
||||
if (!ircolib::is_bit_set<ircolib::u32, 25>(instr))
|
||||
ircolib::panic("broadway::bcx unimplemented variants with bit 25 == 0");
|
||||
|
||||
if (ircolib::is_bit_set<ircolib::u32, 23>(instr)) { // always
|
||||
if (link)
|
||||
lr = pc;
|
||||
|
||||
pc = bd;
|
||||
return;
|
||||
}
|
||||
|
||||
if (ircolib::is_bit_set<ircolib::u32, 22>(instr)) { // --ctr == 0
|
||||
if (--ctr != 0)
|
||||
/*
|
||||
if (!ircolib::is_bit_set<ircolib::u32, 25>(instr)) {
|
||||
|
||||
if (!ircolib::is_bit_set<ircolib::u32, 24>(instr)) {
|
||||
}
|
||||
|
||||
if (!ircolib::is_bit_set<ircolib::u32, 23>(instr))
|
||||
ircolib::panic("broadway::bcx unimplemented variants with bit 23 == 0 (pc 0x{:08X})", pc - 4);
|
||||
|
||||
|
||||
if () {
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
if (ircolib::is_bit_set<ircolib::u32, 23>(instr)) { // always
|
||||
if (link)
|
||||
lr = pc;
|
||||
|
||||
pc = bd;
|
||||
return;
|
||||
}
|
||||
|
||||
if (ircolib::is_bit_set<ircolib::u32, 22>(instr)) { // --ctr == 0
|
||||
if (--ctr != 0)
|
||||
return;
|
||||
|
||||
if (link)
|
||||
lr = pc;
|
||||
|
||||
pc = bd;
|
||||
return;
|
||||
}
|
||||
|
||||
// --ctr != 0
|
||||
if (--ctr == 0)
|
||||
return;
|
||||
|
||||
if (link)
|
||||
lr = pc;
|
||||
|
||||
pc = bd;
|
||||
return;
|
||||
}
|
||||
|
||||
// --ctr != 0
|
||||
if (--ctr == 0)
|
||||
return;
|
||||
|
||||
if (link)
|
||||
lr = pc;
|
||||
|
||||
pc = bd;
|
||||
*/
|
||||
}
|
||||
|
||||
void broadway::mfspr(ircolib::u32 instr) {
|
||||
void broadway::mftspr(bool dir, ircolib::u32 instr) {
|
||||
const size_t spr_field = (instr >> 11) & 0x3FF;
|
||||
switch (spr_field) {
|
||||
case 0b0000100000:
|
||||
gpr[utils::RD(instr)] = xer.raw;
|
||||
break;
|
||||
case 0b0100000000:
|
||||
gpr[utils::RD(instr)] = lr;
|
||||
break;
|
||||
case 0b0100100000:
|
||||
gpr[utils::RD(instr)] = ctr;
|
||||
break;
|
||||
case 0b1101000000:
|
||||
gpr[utils::RD(instr)] = srr0;
|
||||
break;
|
||||
case 0b1101100000:
|
||||
// srr1
|
||||
break;
|
||||
default:
|
||||
ircolib::panic("broadway::mfspr with unimplemented spr field of value 0x{:04X} (pc 0x{:08X})", spr_field,
|
||||
pc - 4);
|
||||
}
|
||||
}
|
||||
|
||||
void broadway::mtspr(ircolib::u32 instr) {
|
||||
const size_t spr_field = (instr >> 11) & 0x3FF;
|
||||
auto move_to_or_from_spr = [&](ircolib::u32 &spr) {
|
||||
if (dir) { // mtspr
|
||||
spr = gpr[utils::RS(instr)];
|
||||
return;
|
||||
}
|
||||
// mfspr
|
||||
gpr[utils::RD(instr)] = spr;
|
||||
};
|
||||
|
||||
switch (spr_field) {
|
||||
case 0b0000100000:
|
||||
xer.raw = gpr[utils::RS(instr)];
|
||||
move_to_or_from_spr(xer.raw);
|
||||
break;
|
||||
case 0b0100000000:
|
||||
lr = gpr[utils::RS(instr)];
|
||||
move_to_or_from_spr(lr);
|
||||
break;
|
||||
case 0b0100100000:
|
||||
ctr = gpr[utils::RS(instr)];
|
||||
move_to_or_from_spr(ctr);
|
||||
break;
|
||||
case 0b1101000000:
|
||||
srr0 = gpr[utils::RS(instr)];
|
||||
move_to_or_from_spr(srr0);
|
||||
break;
|
||||
case 0b1101100000:
|
||||
// srr1
|
||||
case 0b1101100000: // srr1
|
||||
break;
|
||||
case 0b1000011111: // hid0
|
||||
break;
|
||||
case 0b1100011100: // hid2
|
||||
break;
|
||||
case 0b1100111111: // l2cr
|
||||
break;
|
||||
case 0b1000110000:
|
||||
move_to_or_from_spr(ibat[BAT_LOWER_OFFSET + 0]);
|
||||
break;
|
||||
case 0b1000010000:
|
||||
move_to_or_from_spr(ibat[BAT_UPPER_OFFSET + 0]);
|
||||
break;
|
||||
case 0b1001010000:
|
||||
move_to_or_from_spr(ibat[BAT_UPPER_OFFSET + 1]);
|
||||
break;
|
||||
case 0b1010010000:
|
||||
move_to_or_from_spr(ibat[BAT_UPPER_OFFSET + 2]);
|
||||
break;
|
||||
case 0b1011010000:
|
||||
move_to_or_from_spr(ibat[BAT_UPPER_OFFSET + 3]);
|
||||
break;
|
||||
case 0b1100110000:
|
||||
move_to_or_from_spr(dbat[BAT_LOWER_OFFSET + 0]);
|
||||
break;
|
||||
case 0b1101110000:
|
||||
move_to_or_from_spr(dbat[BAT_LOWER_OFFSET + 1]);
|
||||
break;
|
||||
case 0b1100010000:
|
||||
move_to_or_from_spr(dbat[BAT_UPPER_OFFSET + 0]);
|
||||
break;
|
||||
case 0b1101010000:
|
||||
move_to_or_from_spr(dbat[BAT_UPPER_OFFSET + 1]);
|
||||
break;
|
||||
case 0b1110010000:
|
||||
move_to_or_from_spr(dbat[BAT_UPPER_OFFSET + 2]);
|
||||
break;
|
||||
case 0b1111010000:
|
||||
move_to_or_from_spr(dbat[BAT_UPPER_OFFSET + 3]);
|
||||
break;
|
||||
default:
|
||||
ircolib::panic("broadway::mtspr with unimplemented spr field of value 0x{:04X} (pc 0x{:08X})", spr_field,
|
||||
pc - 4);
|
||||
ircolib::warn("broadway::m{}spr with unimplemented spr field of value 0b{:010b} (0x{:04X}) (pc 0x{:08X})",
|
||||
dir ? 't' : 'f', spr_field, spr_field, pc - 4);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -257,4 +309,8 @@ void broadway::lwzu(ircolib::u32 instr, mem &mem) {
|
||||
}
|
||||
|
||||
void broadway::rfi(ircolib::u32 instr) { pc = srr0; }
|
||||
|
||||
void broadway::andi(ircolib::u32 instr) {
|
||||
gpr[utils::RA(instr)] = gpr[utils::RS(instr)] & (ircolib::u32(utils::UIMM(instr)) << 16);
|
||||
}
|
||||
} // namespace weee::core
|
||||
|
||||
Reference in New Issue
Block a user