getting conditional branch now
This commit is contained in:
+2
-1
@@ -24,6 +24,7 @@ add_subdirectory(external/capstone)
|
|||||||
add_executable(weee main.cpp core/mem.cpp core/loaders/elf.cpp
|
add_executable(weee main.cpp core/mem.cpp core/loaders/elf.cpp
|
||||||
core/loaders/dol.cpp
|
core/loaders/dol.cpp
|
||||||
core/broadway.cpp
|
core/broadway.cpp
|
||||||
core/broadway/instructions.cpp)
|
core/broadway/instructions.cpp
|
||||||
|
core/broadway/mmio/vi.cpp)
|
||||||
target_link_libraries(weee PUBLIC capstone)
|
target_link_libraries(weee PUBLIC capstone)
|
||||||
target_include_directories(weee PUBLIC core)
|
target_include_directories(weee PUBLIC core)
|
||||||
+16
-1
@@ -22,7 +22,7 @@ void broadway::run(mem &mem) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ircolib::u32 broadway::fetch(mem &mem) {
|
ircolib::u32 broadway::fetch(mem &mem) {
|
||||||
ircolib::u32 val = mem.read(pc);
|
ircolib::u32 val = mem.read<ircolib::u32>(pc);
|
||||||
pc += 4;
|
pc += 4;
|
||||||
|
|
||||||
return val;
|
return val;
|
||||||
@@ -30,12 +30,27 @@ ircolib::u32 broadway::fetch(mem &mem) {
|
|||||||
|
|
||||||
void broadway::execute(ircolib::u32 instr, mem &mem) {
|
void broadway::execute(ircolib::u32 instr, mem &mem) {
|
||||||
switch (utils::primary(instr)) {
|
switch (utils::primary(instr)) {
|
||||||
|
case 14: // addi rd, ra, simm
|
||||||
|
addi(instr);
|
||||||
|
break;
|
||||||
case 15: // addis rd, ra, simm
|
case 15: // addis rd, ra, simm
|
||||||
addis(instr);
|
addis(instr);
|
||||||
break;
|
break;
|
||||||
|
case 18: // bx target
|
||||||
|
bx(instr);
|
||||||
|
break;
|
||||||
case 24: // ori ra, rs, uimm
|
case 24: // ori ra, rs, uimm
|
||||||
ori(instr);
|
ori(instr);
|
||||||
break;
|
break;
|
||||||
|
case 31: // mfspr rd, spr
|
||||||
|
move_spr(instr);
|
||||||
|
break;
|
||||||
|
case 37: // stwu rs, d(ra)
|
||||||
|
stwu(instr, mem);
|
||||||
|
break;
|
||||||
|
case 44: // stwu rs, d(ra)
|
||||||
|
sth(instr, mem);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
std::println("broadway::execute unimplemented instruction 0x{:08X} / 0b{:032b}", instr, instr);
|
std::println("broadway::execute unimplemented instruction 0x{:08X} / 0b{:032b}", instr, instr);
|
||||||
std::println("disassembly:");
|
std::println("disassembly:");
|
||||||
|
|||||||
+6
-1
@@ -17,7 +17,7 @@ struct broadway {
|
|||||||
void execute(ircolib::u32, mem &);
|
void execute(ircolib::u32, mem &);
|
||||||
|
|
||||||
bool disasm_available = true;
|
bool disasm_available = true;
|
||||||
ircolib::u32 pc = 0;
|
ircolib::u32 pc = 0, lr = 0, ctr = 0;
|
||||||
std::array<ircolib::u32, 32> gpr{};
|
std::array<ircolib::u32, 32> gpr{};
|
||||||
// ircolib::u32 const_gpr_lookup{};
|
// ircolib::u32 const_gpr_lookup{};
|
||||||
csh capstone;
|
csh capstone;
|
||||||
@@ -25,6 +25,11 @@ struct broadway {
|
|||||||
|
|
||||||
// instructions
|
// instructions
|
||||||
void addis(ircolib::u32);
|
void addis(ircolib::u32);
|
||||||
|
void addi(ircolib::u32);
|
||||||
void ori(ircolib::u32);
|
void ori(ircolib::u32);
|
||||||
|
void bx(ircolib::u32);
|
||||||
|
void move_spr(ircolib::u32);
|
||||||
|
void stwu(ircolib::u32, mem &);
|
||||||
|
void sth(ircolib::u32, mem &);
|
||||||
};
|
};
|
||||||
} // namespace weee::core
|
} // namespace weee::core
|
||||||
|
|||||||
@@ -1,6 +1,17 @@
|
|||||||
#include <broadway/utils.hpp>
|
#include <broadway/utils.hpp>
|
||||||
|
#include "ircolib/log.hpp"
|
||||||
|
#include <mem.hpp>
|
||||||
|
|
||||||
namespace weee::core {
|
namespace weee::core {
|
||||||
|
void broadway::addi(ircolib::u32 instr) {
|
||||||
|
if (utils::RA(instr) == 0) { // lis
|
||||||
|
gpr[utils::RD(instr)] = ircolib::s32(utils::SIMM(instr));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
gpr[utils::RD(instr)] = gpr[utils::RA(instr)] + (ircolib::s32(utils::SIMM(instr)));
|
||||||
|
}
|
||||||
|
|
||||||
void broadway::addis(ircolib::u32 instr) {
|
void broadway::addis(ircolib::u32 instr) {
|
||||||
if (utils::RA(instr) == 0) { // lis
|
if (utils::RA(instr) == 0) { // lis
|
||||||
gpr[utils::RD(instr)] = ircolib::s32(utils::SIMM(instr)) << 16;
|
gpr[utils::RD(instr)] = ircolib::s32(utils::SIMM(instr)) << 16;
|
||||||
@@ -11,4 +22,76 @@ void broadway::addis(ircolib::u32 instr) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void broadway::ori(ircolib::u32 instr) { gpr[utils::RA(instr)] = gpr[utils::RS(instr)] | utils::UIMM(instr); }
|
void broadway::ori(ircolib::u32 instr) { gpr[utils::RA(instr)] = gpr[utils::RS(instr)] | utils::UIMM(instr); }
|
||||||
|
|
||||||
|
void broadway::bx(ircolib::u32 instr) {
|
||||||
|
const bool link = instr & 1;
|
||||||
|
const bool absolute = instr & 2;
|
||||||
|
|
||||||
|
ircolib::s32 LI = (ircolib::s32(instr << 6) >> 6) & 0x03FFFFFC;
|
||||||
|
if (!absolute)
|
||||||
|
LI += pc - 4;
|
||||||
|
|
||||||
|
if (link)
|
||||||
|
lr = pc;
|
||||||
|
|
||||||
|
pc = LI;
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
ircolib::panic("broadway::move_spr with unknown sub op {}", direction);
|
||||||
|
}
|
||||||
|
|
||||||
|
void broadway::stwu(ircolib::u32 instr, mem &mem) {
|
||||||
|
if (utils::RA(instr) == 0)
|
||||||
|
ircolib::panic("broadway::stwu with ra == 0");
|
||||||
|
|
||||||
|
const ircolib::s32 d = utils::SIMM(instr);
|
||||||
|
const ircolib::u32 EA = gpr[utils::RA(instr)] + d;
|
||||||
|
mem.write(EA, gpr[utils::RS(instr)]);
|
||||||
|
gpr[utils::RA(instr)] = EA;
|
||||||
|
}
|
||||||
|
|
||||||
|
void broadway::sth(ircolib::u32 instr, mem &mem) {
|
||||||
|
ircolib::u32 b = gpr[utils::RA(instr)];
|
||||||
|
if (utils::RA(instr) == 0)
|
||||||
|
b = 0;
|
||||||
|
|
||||||
|
const ircolib::s32 d = utils::SIMM(instr);
|
||||||
|
const ircolib::u32 EA = b + d;
|
||||||
|
|
||||||
|
mem.write(EA, ircolib::u16(gpr[utils::RS(instr)]));
|
||||||
|
}
|
||||||
} // namespace weee::core
|
} // namespace weee::core
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
#include <broadway/mmio/vi.hpp>
|
||||||
|
#include "ircolib/log.hpp"
|
||||||
|
|
||||||
|
namespace weee::core {
|
||||||
|
void video_interface::write(ircolib::u32 addr, ircolib::u16 value) {
|
||||||
|
addr -= 0x0c002000;
|
||||||
|
switch (addr) {
|
||||||
|
case 2:
|
||||||
|
dcr.raw = value;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ircolib::panic("video_interface::write to unimplemented addr 0x{:04X} with value 0x{:04X}", addr, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} // namespace weee::core
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
#pragma once
|
||||||
|
#include <ircolib/types.hpp>
|
||||||
|
|
||||||
|
namespace weee::core {
|
||||||
|
union DCR {
|
||||||
|
struct {
|
||||||
|
unsigned e : 1;
|
||||||
|
unsigned r : 1;
|
||||||
|
unsigned i : 1;
|
||||||
|
unsigned d : 1;
|
||||||
|
unsigned le0 : 2;
|
||||||
|
unsigned le1 : 2;
|
||||||
|
unsigned fmt : 2;
|
||||||
|
unsigned : 6;
|
||||||
|
};
|
||||||
|
|
||||||
|
ircolib::u16 raw;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct video_interface {
|
||||||
|
void write(ircolib::u32, ircolib::u16);
|
||||||
|
|
||||||
|
private:
|
||||||
|
DCR dcr{};
|
||||||
|
};
|
||||||
|
} // namespace weee::core
|
||||||
@@ -35,6 +35,7 @@ void mem::set(const ircolib::u8 val, const ircolib::u32 size, const ircolib::u32
|
|||||||
memset(&mem1[offset], val, size);
|
memset(&mem1[offset], val, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <>
|
||||||
ircolib::u32 mem::read(ircolib::u32 addr) {
|
ircolib::u32 mem::read(ircolib::u32 addr) {
|
||||||
addr &= 0x0FFFFFFF;
|
addr &= 0x0FFFFFFF;
|
||||||
if (addr > 0x017FFFFF)
|
if (addr > 0x017FFFFF)
|
||||||
@@ -43,6 +44,15 @@ ircolib::u32 mem::read(ircolib::u32 addr) {
|
|||||||
return ircolib::read_access<ircolib::u32>(mem1, addr);
|
return ircolib::read_access<ircolib::u32>(mem1, addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <>
|
||||||
|
ircolib::u16 mem::read(ircolib::u32 addr) {
|
||||||
|
addr &= 0x0FFFFFFF;
|
||||||
|
if (addr > 0x017FFFFF)
|
||||||
|
ircolib::panic("mem::read unimplemented outside mem1 (0x{:08X})", addr);
|
||||||
|
|
||||||
|
return ircolib::read_access<ircolib::u16>(mem1, addr);
|
||||||
|
}
|
||||||
|
|
||||||
void mem::write(ircolib::u32 addr, ircolib::u32 value) {
|
void mem::write(ircolib::u32 addr, ircolib::u32 value) {
|
||||||
addr &= 0x0FFFFFFF;
|
addr &= 0x0FFFFFFF;
|
||||||
if (addr > 0x017FFFFF)
|
if (addr > 0x017FFFFF)
|
||||||
@@ -50,4 +60,17 @@ void mem::write(ircolib::u32 addr, ircolib::u32 value) {
|
|||||||
|
|
||||||
ircolib::write_access<ircolib::u32>(mem1, addr, value);
|
ircolib::write_access<ircolib::u32>(mem1, addr, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void mem::write(ircolib::u32 addr, ircolib::u16 value) {
|
||||||
|
addr &= 0x0FFFFFFF;
|
||||||
|
if (addr > 0x017FFFFF && !ircolib::is_inside_range(addr, 0x0C002000, 0x0C0020FF))
|
||||||
|
ircolib::panic("mem::write unimplemented outside mem1 and vi (0x{:08X} = 0x{:08X})", addr, value);
|
||||||
|
|
||||||
|
if (addr <= 0x017FFFFF) {
|
||||||
|
ircolib::write_access<ircolib::u16>(mem1, addr, value);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
vi.write(addr, value);
|
||||||
|
}
|
||||||
} // namespace weee::core
|
} // namespace weee::core
|
||||||
|
|||||||
+5
-1
@@ -1,18 +1,22 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <ircolib/types.hpp>
|
#include <ircolib/types.hpp>
|
||||||
|
#include "broadway/mmio/vi.hpp"
|
||||||
|
|
||||||
namespace weee::core {
|
namespace weee::core {
|
||||||
struct mem {
|
struct mem {
|
||||||
mem();
|
mem();
|
||||||
|
|
||||||
ircolib::u32 read(ircolib::u32);
|
template <typename T>
|
||||||
|
T read(ircolib::u32);
|
||||||
void write(ircolib::u32, ircolib::u32);
|
void write(ircolib::u32, ircolib::u32);
|
||||||
|
void write(ircolib::u32, ircolib::u16);
|
||||||
void copy(std::vector<ircolib::u8> &src, const ircolib::u32 offset);
|
void copy(std::vector<ircolib::u8> &src, const ircolib::u32 offset);
|
||||||
void copy(ircolib::u8 *src, const ircolib::u32 size, const ircolib::u32 offset);
|
void copy(ircolib::u8 *src, const ircolib::u32 size, const ircolib::u32 offset);
|
||||||
void set(const ircolib::u8 val, const ircolib::u32 size, const ircolib::u32 offset);
|
void set(const ircolib::u8 val, const ircolib::u32 size, const ircolib::u32 offset);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::vector<ircolib::u8> mem1;
|
std::vector<ircolib::u8> mem1;
|
||||||
|
video_interface vi;
|
||||||
};
|
};
|
||||||
} // namespace weee::core
|
} // namespace weee::core
|
||||||
|
|||||||
Reference in New Issue
Block a user