xfb loop copy finished

This commit is contained in:
2026-05-13 17:46:37 +02:00
parent 9c3a6789f6
commit d057a31269
6 changed files with 257 additions and 38 deletions
+119 -32
View File
@@ -3,6 +3,8 @@
#include <mem.hpp>
namespace weee::core {
void broadway::add(ircolib::u32 instr) { gpr[utils::RD(instr)] = gpr[utils::RA(instr)] + gpr[utils::RB(instr)]; }
void broadway::addi(ircolib::u32 instr) {
if (utils::RA(instr) == 0) { // lis
gpr[utils::RD(instr)] = ircolib::s32(utils::SIMM(instr));
@@ -77,41 +79,38 @@ void broadway::bcx(ircolib::u32 instr) {
pc = bd;
}
void broadway::move_spr(ircolib::u32 instr) {
const size_t spr_field = (instr >> 12) & 0x3FF;
const size_t direction = (instr >> 1) & 0x3FF;
if (direction == 339) {
switch (spr_field) {
case 0x80:
gpr[utils::RD(instr)] = lr;
break;
case 0x90:
gpr[utils::RD(instr)] = ctr;
break;
default:
ircolib::panic("broadway::mfspr with unimplemented spr field of value {}", spr_field);
}
return;
void broadway::mfspr(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;
default:
ircolib::panic("broadway::mfspr with unimplemented spr field of value 0x{:04X}", spr_field);
}
}
if (direction == 467) {
switch (spr_field) {
case 0x80:
lr = gpr[utils::RS(instr)];
break;
case 0x90:
ctr = gpr[utils::RS(instr)];
break;
default:
ircolib::panic("broadway::mtspr with unimplemented spr field of value {}", spr_field);
}
return;
void broadway::mtspr(ircolib::u32 instr) {
const size_t spr_field = (instr >> 11) & 0x3FF;
switch (spr_field) {
case 0b0000100000:
xer.raw = gpr[utils::RS(instr)];
break;
case 0b0100000000:
lr = gpr[utils::RS(instr)];
break;
case 0b0100100000:
ctr = gpr[utils::RS(instr)];
break;
default:
ircolib::panic("broadway::mtspr with unimplemented spr field of value 0x{:04X}", spr_field);
}
ircolib::panic("broadway::move_spr with unknown sub op {}", direction);
}
void broadway::stwu(ircolib::u32 instr, mem &mem) {
@@ -151,4 +150,92 @@ void broadway::lwz(ircolib::u32 instr, mem &mem) {
ircolib::u32 ea = ircolib::s32(b) + utils::SIMM(instr);
gpr[utils::RD(instr)] = mem.read32(ea);
}
void broadway::bclrx(ircolib::u32 instr) {
const bool link = instr & 1;
ircolib::s32 bd = lr & 0xFFFFFFFC;
if (!ircolib::is_bit_set<ircolib::u32, 25>(instr)) {
if (!ircolib::is_bit_set<ircolib::u32, 24>(instr))
ircolib::panic("broadway::bclrx unimplemented variants with bit 24 == 0");
if (!ircolib::is_bit_set<ircolib::u32, 23>(instr))
ircolib::panic("broadway::bclrx unimplemented variants with bit 23 == 0");
const ircolib::u8 bi = (instr >> 16) & 0x1f;
if (ircolib::is_bit_set(cr, 32 - bi)) {
if (link)
lr = pc;
pc = bd;
}
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;
}
void broadway::cmpi(ircolib::u32 instr) {
const ircolib::s32 a = gpr[utils::RA(instr)];
const ircolib::s32 simm = utils::SIMM(instr);
ircolib::u8 c;
if (a < simm)
c = 0b001;
else if (a > simm)
c = 0b010;
else
c = 0b100;
const ircolib::u32 result = ((xer.so << 3) | c) << (28 - 4 * utils::crfD(instr));
cr &= ~(0xffff << (28 - 4 * utils::crfD(instr)));
cr |= result;
}
void broadway::rlwinm(ircolib::u32 instr) {
const ircolib::u8 sh = (instr >> 11) & 0x1f;
ircolib::u32 r = std::rotl(gpr[utils::RS(instr)], sh);
const ircolib::u8 mb = (instr >> 6) & 0x1f;
const ircolib::u8 me = (instr >> 1) & 0x1f;
r &= ~((0xFFFF'FFFF >> mb) << me);
gpr[utils::RA(instr)] = r;
}
void broadway::lwzu(ircolib::u32 instr, mem &mem) {
if (utils::RA(instr) == 0 || utils::RA(instr) == utils::RD(instr))
ircolib::panic("broadway::lwzu with ra == 0");
const ircolib::u32 EA = ircolib::s32(gpr[utils::RA(instr)]) + utils::SIMM(instr);
gpr[utils::RD(instr)] = mem.read32(EA);
gpr[utils::RA(instr)] = EA;
}
} // namespace weee::core
+25
View File
@@ -18,6 +18,14 @@ void video_interface::write16(ircolib::u32 addr, ircolib::u16 value) {
case 0x02:
dcr.raw = value;
break;
case 0x4a:
hsr.raw = value;
break;
case 0x6c:
viclk = value & 1;
break;
case 0x70:
break;
default:
ircolib::panic("video_interface::write16 to unimplemented addr 0x{:04X} with value 0x{:04X}", addr, value);
}
@@ -46,6 +54,23 @@ void video_interface::write32(ircolib::u32 addr, ircolib::u32 value) {
case 0x18:
bboi.raw = value;
break;
case 0x1c:
tfbl.raw = value;
break;
case 0x24:
bfbl.raw = value;
break;
case 0x4c:
case 0x50:
case 0x54:
fct0[addr - 0x4c].raw = value;
break;
case 0x58:
case 0x5c:
case 0x60:
case 0x64:
fct1[addr - 0x58].raw = value;
break;
default:
ircolib::panic("video_interface::write32 to unimplemented addr 0x{:04X} with value 0x{:08X}", addr, value);
}
+56
View File
@@ -79,5 +79,61 @@ struct video_interface {
ircolib::u32 raw;
} bbei, bboi;
union TFBL {
struct {
unsigned : 9;
unsigned fbb : 15;
unsigned xof : 4;
unsigned : 4;
};
ircolib::u32 raw;
} tfbl;
union BFBL {
struct {
unsigned : 9;
unsigned fbb : 15;
unsigned y : 8;
};
ircolib::u32 raw;
} bfbl;
union FCT012 {
struct {
unsigned tap0 : 10;
unsigned tap1 : 10;
unsigned tap2 : 10;
unsigned : 2;
};
ircolib::u32 raw;
} fct0[3];
union FCT3456 {
struct {
unsigned tap0 : 8;
unsigned tap1 : 8;
unsigned tap2 : 8;
unsigned tap3 : 8;
};
ircolib::u32 raw;
} fct1[4];
union HSR {
struct {
unsigned stp : 9;
unsigned : 3;
unsigned hs_en : 1;
unsigned : 3;
};
ircolib::u16 raw;
} hsr;
bool viclk;
};
} // namespace weee::core
+3
View File
@@ -3,9 +3,12 @@
namespace weee::core::utils {
static inline ircolib::u8 primary(ircolib::u32 instr) { return (instr >> 26) & 0x3f; }
static inline ircolib::u16 secondary(ircolib::u32 instr) { return (instr >> 1) & 0x1ff; }
static inline ircolib::u8 RD(ircolib::u32 instr) { return (instr >> 21) & 0x1f; }
static inline ircolib::u8 RB(ircolib::u32 instr) { return (instr >> 11) & 0x1f; }
static inline ircolib::u8 RS(ircolib::u32 instr) { return RD(instr); }
static inline ircolib::u8 RA(ircolib::u32 instr) { return (instr >> 16) & 0x1f; }
static inline ircolib::u16 UIMM(ircolib::u32 instr) { return instr & 0xffff; }
static inline ircolib::s16 SIMM(ircolib::u32 instr) { return UIMM(instr); }
static inline ircolib::u8 crfD(ircolib::u32 instr) { return (instr >> 23) & 7; }
} // namespace weee::core::utils