parallel-rsp/main.cpp

392 lines
9.4 KiB
C++
Raw Normal View History

2020-01-29 13:58:56 +00:00
#include "rsp.hpp"
2020-01-29 20:39:26 +00:00
#include "rsp_jit.hpp"
2020-01-29 13:58:56 +00:00
#include <stdio.h>
2020-01-29 20:39:26 +00:00
#include <vector>
2020-01-29 13:58:56 +00:00
using namespace std;
static inline uint32_t flip_endian(uint32_t v)
{
2020-01-29 20:39:26 +00:00
return (v >> 24) | (v << 24) | ((v >> 8) & 0x0000ff00) | ((v << 8) & 0x00ff0000);
2020-01-29 13:58:56 +00:00
}
static vector<uint32_t> read_binary(const char *path, bool flip)
{
2020-01-29 20:39:26 +00:00
FILE *f = fopen(path, "rb");
if (!f)
return {};
fseek(f, 0, SEEK_END);
long len = ftell(f);
rewind(f);
vector<uint32_t> v(len / 4);
fread(v.data(), sizeof(uint32_t), v.size(), f);
fclose(f);
if (flip)
for (auto &value : v)
value = flip_endian(value);
return v;
2020-01-29 13:58:56 +00:00
}
2020-01-30 13:50:20 +00:00
#if 0
2020-01-29 13:58:56 +00:00
static bool read_tag_validate(FILE *file, const char *tag)
{
2020-01-29 20:39:26 +00:00
char tmp[9] = {};
if (fread(tmp, 1, 8, file) != 8)
throw runtime_error("Failed to read tag.");
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
if (strcmp(tmp, "EOF ") == 0)
return false;
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
if (strcmp(tmp, tag))
throw runtime_error("Unexpected tag.");
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
return true;
2020-01-29 13:58:56 +00:00
}
static bool read_block(FILE *file, const char *tag, void *buffer, size_t size)
{
2020-01-29 20:39:26 +00:00
if (!read_tag_validate(file, tag))
return false;
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
uint32_t block_size;
if (fread(&block_size, sizeof(block_size), 1, file) != 1)
throw runtime_error("EOF");
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
if (size != block_size)
throw runtime_error("Unexpected size");
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
if (fread(buffer, size, 1, file) != 1)
throw runtime_error("EOF");
2020-01-29 13:58:56 +00:00
2020-01-29 20:39:26 +00:00
return true;
2020-01-29 13:58:56 +00:00
}
static bool read_poke(FILE *file, RSP::CPU &cpu)
{
2020-01-29 20:39:26 +00:00
char tmp[9] = {};
if (fread(tmp, 1, 8, file) != 8)
throw runtime_error("Failed to read tag.");
if (strcmp(tmp, "ENDDMA ") == 0)
return false;
if (strcmp(tmp, "POKE "))
throw runtime_error("Unexpected tag.");
uint32_t offset;
uint32_t len;
if (fread(&offset, sizeof(offset), 1, file) != 1)
throw runtime_error("Wrong EOF");
if (fread(&len, sizeof(len), 1, file) != 1)
throw runtime_error("Wrong EOF");
if (offset >= 0x1000)
{
if (fread(reinterpret_cast<uint8_t *>(cpu.get_state().imem) + offset - 0x1000, len, 1, file) != 1)
throw runtime_error("Wrong EOF");
}
else
{
if (fread(reinterpret_cast<uint8_t *>(cpu.get_state().dmem) + offset, len, 1, file) != 1)
throw runtime_error("Wrong EOF");
}
return true;
2020-01-29 13:58:56 +00:00
}
static void validate_trace(RSP::CPU &cpu, const char *path)
{
2020-01-29 20:39:26 +00:00
auto &state = cpu.get_state();
uint32_t dmem[1024];
uint32_t imem[1024];
cpu.set_dmem(dmem);
cpu.set_imem(imem);
FILE *file = fopen(path, "rb");
if (!file)
throw runtime_error("Failed to load trace.");
try
{
read_tag_validate(file, "RSPDUMP1");
unsigned index = 0;
while (read_tag_validate(file, "BEGIN "))
{
read_block(file, "DMEM ", state.dmem, 0x1000);
read_block(file, "IMEM ", state.imem, 0x1000);
read_block(file, "SR32 ", state.sr, sizeof(state.sr));
read_block(file, "VR32 ", state.cp2.regs, sizeof(state.cp2.regs));
read_block(file, "VLO ", state.cp2.acc.e + RSP::RSP_ACC_LO, sizeof(uint16_t) * 8);
read_block(file, "VMD ", state.cp2.acc.e + RSP::RSP_ACC_MD, sizeof(uint16_t) * 8);
read_block(file, "VHI ", state.cp2.acc.e + RSP::RSP_ACC_HI, sizeof(uint16_t) * 8);
read_block(file, "PC ", &state.pc, sizeof(state.pc));
int16_t VCO, VCC, VCE;
read_block(file, "VCO ", &VCO, sizeof(VCO));
read_block(file, "VCC ", &VCC, sizeof(VCC));
read_block(file, "VCE ", &VCE, sizeof(VCE));
rsp_set_flags(state.cp2.flags[RSP::RSP_VCO].e, VCO);
rsp_set_flags(state.cp2.flags[RSP::RSP_VCC].e, VCC);
rsp_set_flags(state.cp2.flags[RSP::RSP_VCE].e, VCE);
RSP::ReturnMode mode = RSP::MODE_CONTINUE;
do
{
*state.cp0.cr[RSP::CP0_REGISTER_SP_STATUS] = 0;
cpu.invalidate_imem();
// Run till break.
mode = cpu.run();
if (mode == RSP::MODE_DMA_READ)
{
if (!read_tag_validate(file, "BEGINDMA"))
throw runtime_error("Expected BEGINDMA.");
while (read_poke(file, cpu))
;
}
} while (mode != RSP::MODE_BREAK);
uint32_t dmem[0x1000 >> 2];
uint32_t imem[0x1000 >> 2];
uint32_t sr[32];
uint16_t vr[32 * 8];
uint16_t vlo[8];
uint16_t vmd[8];
uint16_t vhi[8];
read_block(file, "DMEM END", dmem, sizeof(dmem));
read_block(file, "IMEM END", imem, sizeof(imem));
read_block(file, "SR32 END", sr, sizeof(sr));
read_block(file, "VR32 END", vr, sizeof(vr));
read_block(file, "VLO END", vlo, sizeof(vlo));
read_block(file, "VMD END", vmd, sizeof(vmd));
read_block(file, "VHI END", vhi, sizeof(vhi));
read_block(file, "VCO END", &VCO, sizeof(VCO));
read_block(file, "VCC END", &VCC, sizeof(VCC));
read_block(file, "VCE END", &VCE, sizeof(VCE));
unsigned errors = 0;
fprintf(stderr, "==== Trace #%u ====\n", index);
// Validate DMEM
for (unsigned i = 0; i < (0x1000 >> 2); i++)
{
if (state.dmem[i] != dmem[i])
{
fprintf(stderr, "DMEM32[0x%03x] fault. Expected 0x%08x, got 0x%08x!\n", i, dmem[i], state.dmem[i]);
errors++;
}
}
// Validate IMEM (in case of DMA)
for (unsigned i = 0; i < (0x1000 >> 2); i++)
{
if (state.imem[i] != imem[i])
{
fprintf(stderr, "IMEM32[0x%03x] fault. Expected 0x%08x, got 0x%08x!\n", i, dmem[i], state.dmem[i]);
errors++;
}
}
// Validate SR
for (unsigned i = 0; i < 32; i++)
{
if (sr[i] != state.sr[i])
{
fprintf(stderr, "SR[%02u] fault. Expected 0x%08x, got 0x%08x!\n", i, sr[i], state.sr[i]);
errors++;
}
}
// Validate VR
for (unsigned i = 0; i < 16 * 8; i++)
{
if (vr[i] != state.cp2.regs[i >> 3].e[i & 7])
{
fprintf(stderr, "VR[%02u][%u] fault. Expected 0x%04x, got 0x%04x!\n", i >> 3, i & 7, vr[i],
state.cp2.regs[i >> 3].e[i & 7]);
errors++;
}
}
// Validate VLO
for (unsigned i = 0; i < 8; i++)
{
if (vlo[i] != state.cp2.acc.e[RSP::RSP_ACC_LO + i])
{
fprintf(stderr, "VLO[%u] fault. Expected 0x%04x, got 0x%04x!\n", i, vlo[i],
state.cp2.acc.e[RSP::RSP_ACC_LO + i]);
errors++;
}
}
// Validate VMD
for (unsigned i = 0; i < 8; i++)
{
if (vmd[i] != state.cp2.acc.e[RSP::RSP_ACC_MD + i])
{
fprintf(stderr, "VMD[%u] fault. Expected 0x%04x, got 0x%04x!\n", i, vmd[i],
state.cp2.acc.e[RSP::RSP_ACC_MD + i]);
errors++;
}
}
// Validate VHI
for (unsigned i = 0; i < 8; i++)
{
if (vhi[i] != state.cp2.acc.e[RSP::RSP_ACC_HI + i])
{
fprintf(stderr, "VHI[%u] fault. Expected 0x%04x, got 0x%04x!\n", i, vhi[i],
state.cp2.acc.e[RSP::RSP_ACC_HI + i]);
errors++;
}
}
// Validate flags
if (VCO != rsp_get_flags(state.cp2.flags[RSP::RSP_VCO].e))
{
fprintf(stderr, "VCO fault. Expected 0x%04x, got 0x%04x!\n", VCO,
rsp_get_flags(state.cp2.flags[RSP::RSP_VCO].e));
errors++;
}
if (VCC != rsp_get_flags(state.cp2.flags[RSP::RSP_VCC].e))
{
fprintf(stderr, "VCC fault. Expected 0x%04x, got 0x%04x!\n", VCC,
rsp_get_flags(state.cp2.flags[RSP::RSP_VCC].e));
errors++;
}
if (VCE != rsp_get_flags(state.cp2.flags[RSP::RSP_VCE].e))
{
fprintf(stderr, "VCE fault. Expected 0x%04x, got 0x%04x!\n", VCE,
rsp_get_flags(state.cp2.flags[RSP::RSP_VCE].e));
errors++;
}
read_tag_validate(file, "END ");
if (errors == 0)
fprintf(stderr, "SUCCESS! :D\n");
else
fprintf(stderr, "%u ERRORS! :{\n", errors);
fprintf(stderr, "======================\n\n");
index++;
}
}
catch (const std::exception &e)
{
fprintf(stderr, "Exception: %s\n", e.what());
}
fclose(file);
2020-01-29 13:58:56 +00:00
}
2020-01-30 13:50:20 +00:00
#endif
2020-01-29 13:58:56 +00:00
int main(int argc, char *argv[])
{
2020-01-30 13:50:20 +00:00
RSP::JIT::CPU cpu;
RSP::CPU reference_cpu;
2020-01-29 20:39:26 +00:00
auto &state = cpu.get_state();
auto &reference_state = reference_cpu.get_state();
2020-01-29 20:39:26 +00:00
uint32_t cr[16] = {};
for (unsigned i = 0; i < 16; i++)
state.cp0.cr[i] = &cr[i];
uint32_t reference_cr[16] = {};
for (unsigned i = 0; i < 16; i++)
reference_state.cp0.cr[i] = &reference_cr[i];
2020-01-29 20:39:26 +00:00
if (argc == 3)
{
auto dmem = read_binary(argv[1], true);
auto imem = read_binary(argv[2], true);
if (imem.empty())
return 1;
dmem.resize(0x1000);
imem.resize(0x1000);
auto reference_dmem = dmem;
auto reference_imem = imem;
2020-01-29 20:39:26 +00:00
cpu.set_dmem(dmem.data());
cpu.set_imem(imem.data());
reference_cpu.set_dmem(reference_dmem.data());
reference_cpu.set_imem(reference_imem.data());
printf("=== Running Lightning CPU ===\n");
fflush(stdout);
fflush(stderr);
cpu.invalidate_imem();
cpu.run();
fflush(stdout);
fflush(stderr);
printf("=== Running reference CPU ===\n");
reference_cpu.invalidate_imem();
reference_cpu.run();
fflush(stdout);
fflush(stderr);
bool mismatch = false;
for (unsigned i = 0; i < 32; i++)
{
if (state.sr[i] != reference_state.sr[i])
{
fprintf(stderr, "SR[%u] mismatch (got 0x%x, reference 0x%x)!\n", i, state.sr[i], reference_state.sr[i]);
mismatch = true;
}
}
if (state.pc != reference_state.pc)
{
fprintf(stderr, "PC mismatch (got 0x%x, reference 0x%x)!\n", state.pc, reference_state.pc);
mismatch = true;
}
2020-01-29 20:39:26 +00:00
for (unsigned i = 0; i < 16; i++)
2020-01-29 20:39:26 +00:00
{
if (cr[i] != reference_cr[i])
{
fprintf(stderr, "COP0 CR [%u] mismatch (got %u, reference %u)!\n", i, cr[i], reference_cr[i]);
mismatch = true;
}
2020-01-29 20:39:26 +00:00
}
for (unsigned i = 0; i < 0x1000; i++)
{
if (dmem[i] != reference_dmem[i])
{
fprintf(stderr, "DMEM[0x%03x] mismatch (got 0x%02x, reference 0x%02x)!\n", i, dmem[i], reference_dmem[i]);
mismatch = true;
}
}
2020-01-31 21:53:36 +00:00
if (memcmp(&state.cp2, &reference_state.cp2, sizeof(state.cp2)) != 0)
{
fprintf(stderr, "CP2 register state mismatch.\n");
mismatch = true;
}
if (mismatch)
return EXIT_FAILURE;
2020-01-29 20:39:26 +00:00
}
2020-01-30 13:50:20 +00:00
#if 0
2020-01-29 20:39:26 +00:00
else if (argc == 2)
validate_trace(cpu, argv[1]);
2020-01-30 13:50:20 +00:00
#endif
2020-01-29 20:39:26 +00:00
else
return 1;
2020-01-29 13:58:56 +00:00
}