refactor: make disassembler optional
Signed-off-by: Amneesh Singh <natto@weirdnatto.in>
This commit is contained in:
@@ -1 +1,2 @@
|
|||||||
option('tests', type : 'boolean', value : true, description: 'enable tests')
|
option('tests', type : 'boolean', value : true, description: 'enable tests')
|
||||||
|
option('disassembler', type: 'boolean', value: false, description: 'enable disassembler')
|
||||||
|
235
src/cpu/arm/disassembler.cc
Normal file
235
src/cpu/arm/disassembler.cc
Normal file
@@ -0,0 +1,235 @@
|
|||||||
|
#include "instruction.hh"
|
||||||
|
#include "util/bits.hh"
|
||||||
|
|
||||||
|
namespace matar {
|
||||||
|
namespace arm {
|
||||||
|
std::string
|
||||||
|
Instruction::disassemble() {
|
||||||
|
auto condition = stringify(this->condition);
|
||||||
|
|
||||||
|
return std::visit(
|
||||||
|
overloaded{
|
||||||
|
[condition](BranchAndExchange& data) {
|
||||||
|
return fmt::format("BX{} R{:d}", condition, data.rn);
|
||||||
|
},
|
||||||
|
[condition](Branch& data) {
|
||||||
|
return fmt::format(
|
||||||
|
"B{}{} 0x{:06X}", (data.link ? "L" : ""), condition, data.offset);
|
||||||
|
},
|
||||||
|
[condition](Multiply& data) {
|
||||||
|
if (data.acc) {
|
||||||
|
return fmt::format("MLA{}{} R{:d},R{:d},R{:d},R{:d}",
|
||||||
|
condition,
|
||||||
|
(data.set ? "S" : ""),
|
||||||
|
data.rd,
|
||||||
|
data.rm,
|
||||||
|
data.rs,
|
||||||
|
data.rn);
|
||||||
|
} else {
|
||||||
|
return fmt::format("MUL{}{} R{:d},R{:d},R{:d}",
|
||||||
|
condition,
|
||||||
|
(data.set ? "S" : ""),
|
||||||
|
data.rd,
|
||||||
|
data.rm,
|
||||||
|
data.rs);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[condition](MultiplyLong& data) {
|
||||||
|
return fmt::format("{}{}{}{} R{:d},R{:d},R{:d},R{:d}",
|
||||||
|
(data.uns ? 'U' : 'S'),
|
||||||
|
(data.acc ? "MLAL" : "MULL"),
|
||||||
|
condition,
|
||||||
|
(data.set ? "S" : ""),
|
||||||
|
data.rdlo,
|
||||||
|
data.rdhi,
|
||||||
|
data.rm,
|
||||||
|
data.rs);
|
||||||
|
},
|
||||||
|
[](Undefined) { return std::string("UND"); },
|
||||||
|
[condition](SingleDataSwap& data) {
|
||||||
|
return fmt::format("SWP{}{} R{:d},R{:d},[R{:d}]",
|
||||||
|
condition,
|
||||||
|
(data.byte ? "B" : ""),
|
||||||
|
data.rd,
|
||||||
|
data.rm,
|
||||||
|
data.rn);
|
||||||
|
},
|
||||||
|
[condition](SingleDataTransfer& data) {
|
||||||
|
std::string expression;
|
||||||
|
std::string address;
|
||||||
|
|
||||||
|
if (const uint16_t* offset = std::get_if<uint16_t>(&data.offset)) {
|
||||||
|
if (*offset == 0) {
|
||||||
|
expression = "";
|
||||||
|
} else {
|
||||||
|
expression =
|
||||||
|
fmt::format(",{}#{:d}", (data.up ? '+' : '-'), *offset);
|
||||||
|
}
|
||||||
|
} else if (const Shift* shift = std::get_if<Shift>(&data.offset)) {
|
||||||
|
// Shifts are always immediate in single data transfer
|
||||||
|
expression = fmt::format(",{}R{:d},{} #{:d}",
|
||||||
|
(data.up ? '+' : '-'),
|
||||||
|
shift->rm,
|
||||||
|
stringify(shift->data.type),
|
||||||
|
shift->data.operand);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt::format(
|
||||||
|
"{}{}{}{} R{:d},[R{:d}{}]{}",
|
||||||
|
(data.load ? "LDR" : "STR"),
|
||||||
|
condition,
|
||||||
|
(data.byte ? "B" : ""),
|
||||||
|
(!data.pre && data.write ? "T" : ""),
|
||||||
|
data.rd,
|
||||||
|
data.rn,
|
||||||
|
(data.pre ? expression : ""),
|
||||||
|
(data.pre ? (data.write ? "!" : "") : expression));
|
||||||
|
},
|
||||||
|
[condition](HalfwordTransfer& data) {
|
||||||
|
std::string expression;
|
||||||
|
|
||||||
|
if (data.imm) {
|
||||||
|
if (data.offset == 0) {
|
||||||
|
expression = "";
|
||||||
|
} else {
|
||||||
|
expression = fmt::format(
|
||||||
|
",{}#{:d}", (data.up ? '+' : '-'), data.offset);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
expression =
|
||||||
|
fmt::format(",{}R{:d}", (data.up ? '+' : '-'), data.offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt::format(
|
||||||
|
"{}{}{}{} R{:d},[R{:d}{}]{}",
|
||||||
|
(data.load ? "LDR" : "STR"),
|
||||||
|
condition,
|
||||||
|
(data.sign ? "S" : ""),
|
||||||
|
(data.half ? 'H' : 'B'),
|
||||||
|
data.rd,
|
||||||
|
data.rn,
|
||||||
|
(data.pre ? expression : ""),
|
||||||
|
(data.pre ? (data.write ? "!" : "") : expression));
|
||||||
|
},
|
||||||
|
[condition](BlockDataTransfer& data) {
|
||||||
|
std::string regs;
|
||||||
|
|
||||||
|
for (uint8_t i = 0; i < 16; i++) {
|
||||||
|
if (get_bit(data.regs, i))
|
||||||
|
fmt::format_to(std::back_inserter(regs), "R{:d},", i);
|
||||||
|
};
|
||||||
|
|
||||||
|
regs.pop_back();
|
||||||
|
|
||||||
|
return fmt::format("{}{}{}{} R{:d}{},{{{}}}{}",
|
||||||
|
(data.load ? "LDM" : "STM"),
|
||||||
|
condition,
|
||||||
|
(data.up ? 'I' : 'D'),
|
||||||
|
(data.pre ? 'B' : 'A'),
|
||||||
|
data.rn,
|
||||||
|
(data.write ? "!" : ""),
|
||||||
|
regs,
|
||||||
|
(data.s ? "^" : ""));
|
||||||
|
},
|
||||||
|
[condition](PsrTransfer& data) {
|
||||||
|
if (data.type == PsrTransfer::Type::Mrs) {
|
||||||
|
return fmt::format("MRS{} R{:d},{}",
|
||||||
|
condition,
|
||||||
|
data.operand,
|
||||||
|
(data.spsr ? "SPSR_all" : "CPSR_all"));
|
||||||
|
} else {
|
||||||
|
return fmt::format(
|
||||||
|
"MSR{} {}_{},{}{}",
|
||||||
|
condition,
|
||||||
|
(data.spsr ? "SPSR" : "CPSR"),
|
||||||
|
(data.type == PsrTransfer::Type::Msr_flg ? "flg" : "all"),
|
||||||
|
(data.imm ? '#' : 'R'),
|
||||||
|
data.operand);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[condition](DataProcessing& data) {
|
||||||
|
using OpCode = DataProcessing::OpCode;
|
||||||
|
|
||||||
|
std::string op_2;
|
||||||
|
|
||||||
|
if (const uint32_t* operand =
|
||||||
|
std::get_if<uint32_t>(&data.operand)) {
|
||||||
|
op_2 = fmt::format("#{:d}", *operand);
|
||||||
|
} else if (const Shift* shift = std::get_if<Shift>(&data.operand)) {
|
||||||
|
op_2 = fmt::format("R{:d},{} {}{:d}",
|
||||||
|
shift->rm,
|
||||||
|
stringify(shift->data.type),
|
||||||
|
(shift->data.immediate ? '#' : 'R'),
|
||||||
|
shift->data.operand);
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (data.opcode) {
|
||||||
|
case OpCode::MOV:
|
||||||
|
case OpCode::MVN:
|
||||||
|
return fmt::format("{}{}{} R{:d},{}",
|
||||||
|
stringify(data.opcode),
|
||||||
|
condition,
|
||||||
|
(data.set ? "S" : ""),
|
||||||
|
data.rd,
|
||||||
|
op_2);
|
||||||
|
case OpCode::TST:
|
||||||
|
case OpCode::TEQ:
|
||||||
|
case OpCode::CMP:
|
||||||
|
case OpCode::CMN:
|
||||||
|
return fmt::format("{}{} R{:d},{}",
|
||||||
|
stringify(data.opcode),
|
||||||
|
condition,
|
||||||
|
data.rn,
|
||||||
|
op_2);
|
||||||
|
default:
|
||||||
|
return fmt::format("{}{}{} R{:d},R{:d},{}",
|
||||||
|
stringify(data.opcode),
|
||||||
|
condition,
|
||||||
|
(data.set ? "S" : ""),
|
||||||
|
data.rd,
|
||||||
|
data.rn,
|
||||||
|
op_2);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[condition](SoftwareInterrupt) {
|
||||||
|
return fmt::format("SWI{}", condition);
|
||||||
|
},
|
||||||
|
[condition](CoprocessorDataTransfer& data) {
|
||||||
|
std::string expression = fmt::format(",#{:d}", data.offset);
|
||||||
|
return fmt::format(
|
||||||
|
"{}{}{} p{:d},c{:d},[R{:d}{}]{}",
|
||||||
|
(data.load ? "LDC" : "STC"),
|
||||||
|
condition,
|
||||||
|
(data.len ? "L" : ""),
|
||||||
|
data.cpn,
|
||||||
|
data.crd,
|
||||||
|
data.rn,
|
||||||
|
(data.pre ? expression : ""),
|
||||||
|
(data.pre ? (data.write ? "!" : "") : expression));
|
||||||
|
},
|
||||||
|
[condition](CoprocessorDataOperation& data) {
|
||||||
|
return fmt::format("CDP{} p{},{},c{},c{},c{},{}",
|
||||||
|
condition,
|
||||||
|
data.cpn,
|
||||||
|
data.cp_opc,
|
||||||
|
data.crd,
|
||||||
|
data.crn,
|
||||||
|
data.crm,
|
||||||
|
data.cp);
|
||||||
|
},
|
||||||
|
[condition](CoprocessorRegisterTransfer& data) {
|
||||||
|
return fmt::format("{}{} p{},{},R{},c{},c{},{}",
|
||||||
|
(data.load ? "MRC" : "MCR"),
|
||||||
|
condition,
|
||||||
|
data.cpn,
|
||||||
|
data.cp_opc,
|
||||||
|
data.rd,
|
||||||
|
data.crn,
|
||||||
|
data.crm,
|
||||||
|
data.cp);
|
||||||
|
},
|
||||||
|
[](auto) { return std::string("unknown instruction"); } },
|
||||||
|
data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -274,230 +274,5 @@ Instruction::Instruction(uint32_t insn)
|
|||||||
data = Undefined{};
|
data = Undefined{};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string
|
|
||||||
Instruction::disassemble() {
|
|
||||||
// goddamn this is gore
|
|
||||||
// TODO: make this less ugly
|
|
||||||
return std::visit(
|
|
||||||
overloaded{
|
|
||||||
[this](BranchAndExchange& data) {
|
|
||||||
return fmt::format("BX{} R{:d}", condition, data.rn);
|
|
||||||
},
|
|
||||||
[this](Branch& data) {
|
|
||||||
return fmt::format(
|
|
||||||
"B{}{} 0x{:06X}", (data.link ? "L" : ""), condition, data.offset);
|
|
||||||
},
|
|
||||||
[this](Multiply& data) {
|
|
||||||
if (data.acc) {
|
|
||||||
return fmt::format("MLA{}{} R{:d},R{:d},R{:d},R{:d}",
|
|
||||||
condition,
|
|
||||||
(data.set ? "S" : ""),
|
|
||||||
data.rd,
|
|
||||||
data.rm,
|
|
||||||
data.rs,
|
|
||||||
data.rn);
|
|
||||||
} else {
|
|
||||||
return fmt::format("MUL{}{} R{:d},R{:d},R{:d}",
|
|
||||||
condition,
|
|
||||||
(data.set ? "S" : ""),
|
|
||||||
data.rd,
|
|
||||||
data.rm,
|
|
||||||
data.rs);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[this](MultiplyLong& data) {
|
|
||||||
return fmt::format("{}{}{}{} R{:d},R{:d},R{:d},R{:d}",
|
|
||||||
(data.uns ? 'U' : 'S'),
|
|
||||||
(data.acc ? "MLAL" : "MULL"),
|
|
||||||
condition,
|
|
||||||
(data.set ? "S" : ""),
|
|
||||||
data.rdlo,
|
|
||||||
data.rdhi,
|
|
||||||
data.rm,
|
|
||||||
data.rs);
|
|
||||||
},
|
|
||||||
[](Undefined) { return std::string("UND"); },
|
|
||||||
[this](SingleDataSwap& data) {
|
|
||||||
return fmt::format("SWP{}{} R{:d},R{:d},[R{:d}]",
|
|
||||||
condition,
|
|
||||||
(data.byte ? "B" : ""),
|
|
||||||
data.rd,
|
|
||||||
data.rm,
|
|
||||||
data.rn);
|
|
||||||
},
|
|
||||||
[this](SingleDataTransfer& data) {
|
|
||||||
std::string expression;
|
|
||||||
std::string address;
|
|
||||||
|
|
||||||
if (const uint16_t* offset = std::get_if<uint16_t>(&data.offset)) {
|
|
||||||
if (*offset == 0) {
|
|
||||||
expression = "";
|
|
||||||
} else {
|
|
||||||
expression =
|
|
||||||
fmt::format(",{}#{:d}", (data.up ? '+' : '-'), *offset);
|
|
||||||
}
|
|
||||||
} else if (const Shift* shift = std::get_if<Shift>(&data.offset)) {
|
|
||||||
// Shifts are always immediate in single data transfer
|
|
||||||
expression = fmt::format(",{}R{:d},{} #{:d}",
|
|
||||||
(data.up ? '+' : '-'),
|
|
||||||
shift->rm,
|
|
||||||
shift->data.type,
|
|
||||||
shift->data.operand);
|
|
||||||
}
|
|
||||||
|
|
||||||
return fmt::format(
|
|
||||||
"{}{}{}{} R{:d},[R{:d}{}]{}",
|
|
||||||
(data.load ? "LDR" : "STR"),
|
|
||||||
condition,
|
|
||||||
(data.byte ? "B" : ""),
|
|
||||||
(!data.pre && data.write ? "T" : ""),
|
|
||||||
data.rd,
|
|
||||||
data.rn,
|
|
||||||
(data.pre ? expression : ""),
|
|
||||||
(data.pre ? (data.write ? "!" : "") : expression));
|
|
||||||
},
|
|
||||||
[this](HalfwordTransfer& data) {
|
|
||||||
std::string expression;
|
|
||||||
|
|
||||||
if (data.imm) {
|
|
||||||
if (data.offset == 0) {
|
|
||||||
expression = "";
|
|
||||||
} else {
|
|
||||||
expression = fmt::format(
|
|
||||||
",{}#{:d}", (data.up ? '+' : '-'), data.offset);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
expression =
|
|
||||||
fmt::format(",{}R{:d}", (data.up ? '+' : '-'), data.offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
return fmt::format(
|
|
||||||
"{}{}{}{} R{:d},[R{:d}{}]{}",
|
|
||||||
(data.load ? "LDR" : "STR"),
|
|
||||||
condition,
|
|
||||||
(data.sign ? "S" : ""),
|
|
||||||
(data.half ? 'H' : 'B'),
|
|
||||||
data.rd,
|
|
||||||
data.rn,
|
|
||||||
(data.pre ? expression : ""),
|
|
||||||
(data.pre ? (data.write ? "!" : "") : expression));
|
|
||||||
},
|
|
||||||
[this](BlockDataTransfer& data) {
|
|
||||||
std::string regs;
|
|
||||||
|
|
||||||
for (uint8_t i = 0; i < 16; i++) {
|
|
||||||
if (get_bit(data.regs, i))
|
|
||||||
fmt::format_to(std::back_inserter(regs), "R{:d},", i);
|
|
||||||
};
|
|
||||||
|
|
||||||
regs.pop_back();
|
|
||||||
|
|
||||||
return fmt::format("{}{}{}{} R{:d}{},{{{}}}{}",
|
|
||||||
(data.load ? "LDM" : "STM"),
|
|
||||||
condition,
|
|
||||||
(data.up ? 'I' : 'D'),
|
|
||||||
(data.pre ? 'B' : 'A'),
|
|
||||||
data.rn,
|
|
||||||
(data.write ? "!" : ""),
|
|
||||||
regs,
|
|
||||||
(data.s ? "^" : ""));
|
|
||||||
},
|
|
||||||
[this](PsrTransfer& data) {
|
|
||||||
if (data.type == PsrTransfer::Type::Mrs) {
|
|
||||||
return fmt::format("MRS{} R{:d},{}",
|
|
||||||
condition,
|
|
||||||
data.operand,
|
|
||||||
(data.spsr ? "SPSR_all" : "CPSR_all"));
|
|
||||||
} else {
|
|
||||||
return fmt::format(
|
|
||||||
"MSR{} {}_{},{}{}",
|
|
||||||
condition,
|
|
||||||
(data.spsr ? "SPSR" : "CPSR"),
|
|
||||||
(data.type == PsrTransfer::Type::Msr_flg ? "flg" : "all"),
|
|
||||||
(data.imm ? '#' : 'R'),
|
|
||||||
data.operand);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[this](DataProcessing& data) {
|
|
||||||
using OpCode = DataProcessing::OpCode;
|
|
||||||
|
|
||||||
std::string op_2;
|
|
||||||
|
|
||||||
if (const uint32_t* operand =
|
|
||||||
std::get_if<uint32_t>(&data.operand)) {
|
|
||||||
op_2 = fmt::format("#{:d}", *operand);
|
|
||||||
} else if (const Shift* shift = std::get_if<Shift>(&data.operand)) {
|
|
||||||
op_2 = fmt::format("R{:d},{} {}{:d}",
|
|
||||||
shift->rm,
|
|
||||||
shift->data.type,
|
|
||||||
(shift->data.immediate ? '#' : 'R'),
|
|
||||||
shift->data.operand);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (data.opcode) {
|
|
||||||
case OpCode::MOV:
|
|
||||||
case OpCode::MVN:
|
|
||||||
return fmt::format("{}{}{} R{:d},{}",
|
|
||||||
data.opcode,
|
|
||||||
condition,
|
|
||||||
(data.set ? "S" : ""),
|
|
||||||
data.rd,
|
|
||||||
op_2);
|
|
||||||
case OpCode::TST:
|
|
||||||
case OpCode::TEQ:
|
|
||||||
case OpCode::CMP:
|
|
||||||
case OpCode::CMN:
|
|
||||||
return fmt::format(
|
|
||||||
"{}{} R{:d},{}", data.opcode, condition, data.rn, op_2);
|
|
||||||
default:
|
|
||||||
return fmt::format("{}{}{} R{:d},R{:d},{}",
|
|
||||||
data.opcode,
|
|
||||||
condition,
|
|
||||||
(data.set ? "S" : ""),
|
|
||||||
data.rd,
|
|
||||||
data.rn,
|
|
||||||
op_2);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[this](SoftwareInterrupt) { return fmt::format("SWI{}", condition); },
|
|
||||||
[this](CoprocessorDataTransfer& data) {
|
|
||||||
std::string expression = fmt::format(",#{:d}", data.offset);
|
|
||||||
return fmt::format(
|
|
||||||
"{}{}{} p{:d},c{:d},[R{:d}{}]{}",
|
|
||||||
(data.load ? "LDC" : "STC"),
|
|
||||||
condition,
|
|
||||||
(data.len ? "L" : ""),
|
|
||||||
data.cpn,
|
|
||||||
data.crd,
|
|
||||||
data.rn,
|
|
||||||
(data.pre ? expression : ""),
|
|
||||||
(data.pre ? (data.write ? "!" : "") : expression));
|
|
||||||
},
|
|
||||||
[this](CoprocessorDataOperation& data) {
|
|
||||||
return fmt::format("CDP{} p{},{},c{},c{},c{},{}",
|
|
||||||
condition,
|
|
||||||
data.cpn,
|
|
||||||
data.cp_opc,
|
|
||||||
data.crd,
|
|
||||||
data.crn,
|
|
||||||
data.crm,
|
|
||||||
data.cp);
|
|
||||||
},
|
|
||||||
[this](CoprocessorRegisterTransfer& data) {
|
|
||||||
return fmt::format("{}{} p{},{},R{},c{},c{},{}",
|
|
||||||
(data.load ? "MRC" : "MCR"),
|
|
||||||
condition,
|
|
||||||
data.cpn,
|
|
||||||
data.cp_opc,
|
|
||||||
data.rd,
|
|
||||||
data.crn,
|
|
||||||
data.crm,
|
|
||||||
data.cp);
|
|
||||||
},
|
|
||||||
[](auto) { return std::string("unknown instruction"); } },
|
|
||||||
data);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -218,7 +218,9 @@ struct Instruction {
|
|||||||
: condition(condition)
|
: condition(condition)
|
||||||
, data(data){};
|
, data(data){};
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
std::string disassemble();
|
std::string disassemble();
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -2,3 +2,7 @@ lib_sources += files(
|
|||||||
'instruction.cc',
|
'instruction.cc',
|
||||||
'exec.cc'
|
'exec.cc'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if get_option('disassembler')
|
||||||
|
lib_sources += files('disassembler.cc')
|
||||||
|
endif
|
@@ -121,11 +121,13 @@ CpuImpl::step() {
|
|||||||
if (cpsr.state() == State::Arm) {
|
if (cpsr.state() == State::Arm) {
|
||||||
uint32_t x = bus->read_word(cur_pc);
|
uint32_t x = bus->read_word(cur_pc);
|
||||||
arm::Instruction instruction(x);
|
arm::Instruction instruction(x);
|
||||||
|
|
||||||
|
exec(instruction);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
glogger.info("{:#034b}", x);
|
glogger.info("{:#034b}", x);
|
||||||
|
|
||||||
arm(instruction);
|
|
||||||
|
|
||||||
glogger.info("0x{:08X} : {}", cur_pc, instruction.disassemble());
|
glogger.info("0x{:08X} : {}", cur_pc, instruction.disassemble());
|
||||||
|
#endif
|
||||||
|
|
||||||
if (is_flushed) {
|
if (is_flushed) {
|
||||||
// if flushed, do not increment the PC, instead set it to two
|
// if flushed, do not increment the PC, instead set it to two
|
||||||
|
@@ -11,7 +11,11 @@ lib_cpp_args = [ ]
|
|||||||
fmt = dependency('fmt', version : '>=10.1.0', static: true)
|
fmt = dependency('fmt', version : '>=10.1.0', static: true)
|
||||||
if not fmt.found()
|
if not fmt.found()
|
||||||
fmt = dependency('fmt', version : '>=10.1.0', static: false)
|
fmt = dependency('fmt', version : '>=10.1.0', static: false)
|
||||||
lib_cpp_args += 'DFMT_HEADER_ONLY'
|
lib_cpp_args += '-DFMT_HEADER_ONLY'
|
||||||
|
endif
|
||||||
|
|
||||||
|
if get_option('disassembler')
|
||||||
|
lib_cpp_args += '-DDISASSEMBLER'
|
||||||
endif
|
endif
|
||||||
|
|
||||||
lib = library(
|
lib = library(
|
||||||
|
@@ -16,7 +16,9 @@ TEST_CASE("Branch and Exchange", TAG) {
|
|||||||
|
|
||||||
CHECK(bx->rn == 10);
|
CHECK(bx->rn == 10);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "BXGT R10");
|
CHECK(instruction.disassemble() == "BXGT R10");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Branch", TAG) {
|
TEST_CASE("Branch", TAG) {
|
||||||
@@ -33,10 +35,12 @@ TEST_CASE("Branch", TAG) {
|
|||||||
CHECK(b->offset == 0xFE15FF14);
|
CHECK(b->offset == 0xFE15FF14);
|
||||||
CHECK(b->link == true);
|
CHECK(b->link == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "BL 0xFE15FF14");
|
CHECK(instruction.disassemble() == "BL 0xFE15FF14");
|
||||||
|
|
||||||
b->link = false;
|
b->link = false;
|
||||||
CHECK(instruction.disassemble() == "B 0xFE15FF14");
|
CHECK(instruction.disassemble() == "B 0xFE15FF14");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Multiply", TAG) {
|
TEST_CASE("Multiply", TAG) {
|
||||||
@@ -54,11 +58,13 @@ TEST_CASE("Multiply", TAG) {
|
|||||||
CHECK(mul->acc == true);
|
CHECK(mul->acc == true);
|
||||||
CHECK(mul->set == true);
|
CHECK(mul->set == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "MLAEQS R10,R0,R15,R14");
|
CHECK(instruction.disassemble() == "MLAEQS R10,R0,R15,R14");
|
||||||
|
|
||||||
mul->acc = false;
|
mul->acc = false;
|
||||||
mul->set = false;
|
mul->set = false;
|
||||||
CHECK(instruction.disassemble() == "MULEQ R10,R0,R15");
|
CHECK(instruction.disassemble() == "MULEQ R10,R0,R15");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Multiply Long", TAG) {
|
TEST_CASE("Multiply Long", TAG) {
|
||||||
@@ -77,6 +83,7 @@ TEST_CASE("Multiply Long", TAG) {
|
|||||||
CHECK(mull->set == true);
|
CHECK(mull->set == true);
|
||||||
CHECK(mull->uns == true);
|
CHECK(mull->uns == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "UMULLNES R7,R14,R2,R6");
|
CHECK(instruction.disassemble() == "UMULLNES R7,R14,R2,R6");
|
||||||
|
|
||||||
mull->acc = true;
|
mull->acc = true;
|
||||||
@@ -85,6 +92,7 @@ TEST_CASE("Multiply Long", TAG) {
|
|||||||
mull->uns = false;
|
mull->uns = false;
|
||||||
mull->set = false;
|
mull->set = false;
|
||||||
CHECK(instruction.disassemble() == "SMLALNE R7,R14,R2,R6");
|
CHECK(instruction.disassemble() == "SMLALNE R7,R14,R2,R6");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Undefined", TAG) {
|
TEST_CASE("Undefined", TAG) {
|
||||||
@@ -94,7 +102,10 @@ TEST_CASE("Undefined", TAG) {
|
|||||||
Instruction instruction(raw);
|
Instruction instruction(raw);
|
||||||
|
|
||||||
CHECK(instruction.condition == Condition::AL);
|
CHECK(instruction.condition == Condition::AL);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "UND");
|
CHECK(instruction.disassemble() == "UND");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Single Data Swap", TAG) {
|
TEST_CASE("Single Data Swap", TAG) {
|
||||||
@@ -110,10 +121,12 @@ TEST_CASE("Single Data Swap", TAG) {
|
|||||||
CHECK(swp->rn == 9);
|
CHECK(swp->rn == 9);
|
||||||
CHECK(swp->byte == false);
|
CHECK(swp->byte == false);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "SWPGE R5,R6,[R9]");
|
CHECK(instruction.disassemble() == "SWPGE R5,R6,[R9]");
|
||||||
|
|
||||||
swp->byte = true;
|
swp->byte = true;
|
||||||
CHECK(instruction.disassemble() == "SWPGEB R5,R6,[R9]");
|
CHECK(instruction.disassemble() == "SWPGEB R5,R6,[R9]");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Single Data Transfer", TAG) {
|
TEST_CASE("Single Data Transfer", TAG) {
|
||||||
@@ -138,6 +151,7 @@ TEST_CASE("Single Data Transfer", TAG) {
|
|||||||
CHECK(ldr->up == true);
|
CHECK(ldr->up == true);
|
||||||
CHECK(ldr->pre == true);
|
CHECK(ldr->pre == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
ldr->load = true;
|
ldr->load = true;
|
||||||
ldr->byte = true;
|
ldr->byte = true;
|
||||||
ldr->write = false;
|
ldr->write = false;
|
||||||
@@ -153,6 +167,7 @@ TEST_CASE("Single Data Transfer", TAG) {
|
|||||||
|
|
||||||
ldr->pre = true;
|
ldr->pre = true;
|
||||||
CHECK(instruction.disassemble() == "LDRB R10,[R2,-#9023]");
|
CHECK(instruction.disassemble() == "LDRB R10,[R2,-#9023]");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Halfword Transfer", TAG) {
|
TEST_CASE("Halfword Transfer", TAG) {
|
||||||
@@ -176,6 +191,7 @@ TEST_CASE("Halfword Transfer", TAG) {
|
|||||||
CHECK(ldr->up == true);
|
CHECK(ldr->up == true);
|
||||||
CHECK(ldr->pre == true);
|
CHECK(ldr->pre == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "STRCCH R2,[R15,+R6]!");
|
CHECK(instruction.disassemble() == "STRCCH R2,[R15,+R6]!");
|
||||||
|
|
||||||
ldr->pre = false;
|
ldr->pre = false;
|
||||||
@@ -193,6 +209,7 @@ TEST_CASE("Halfword Transfer", TAG) {
|
|||||||
ldr->imm = 1;
|
ldr->imm = 1;
|
||||||
ldr->offset = 90;
|
ldr->offset = 90;
|
||||||
CHECK(instruction.disassemble() == "STRCCSB R2,[R15],-#90");
|
CHECK(instruction.disassemble() == "STRCCSB R2,[R15],-#90");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Block Data Transfer", TAG) {
|
TEST_CASE("Block Data Transfer", TAG) {
|
||||||
@@ -223,6 +240,7 @@ TEST_CASE("Block Data Transfer", TAG) {
|
|||||||
CHECK(ldm->up == false);
|
CHECK(ldm->up == false);
|
||||||
CHECK(ldm->pre == true);
|
CHECK(ldm->pre == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "LDMLSDB R7,{R0,R2,R3,R5,R6,R8,R14}^");
|
CHECK(instruction.disassemble() == "LDMLSDB R7,{R0,R2,R3,R5,R6,R8,R14}^");
|
||||||
|
|
||||||
ldm->write = true;
|
ldm->write = true;
|
||||||
@@ -238,6 +256,7 @@ TEST_CASE("Block Data Transfer", TAG) {
|
|||||||
ldm->pre = false;
|
ldm->pre = false;
|
||||||
|
|
||||||
CHECK(instruction.disassemble() == "STMLSIA R7!,{R0,R2,R5,R14}");
|
CHECK(instruction.disassemble() == "STMLSIA R7!,{R0,R2,R5,R14}");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("PSR Transfer", TAG) {
|
TEST_CASE("PSR Transfer", TAG) {
|
||||||
@@ -256,7 +275,9 @@ TEST_CASE("PSR Transfer", TAG) {
|
|||||||
CHECK(mrs->operand == 10);
|
CHECK(mrs->operand == 10);
|
||||||
CHECK(mrs->spsr == true);
|
CHECK(mrs->spsr == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "MRSMI R10,SPSR_all");
|
CHECK(instruction.disassemble() == "MRSMI R10,SPSR_all");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("MSR") {
|
SECTION("MSR") {
|
||||||
@@ -272,7 +293,9 @@ TEST_CASE("PSR Transfer", TAG) {
|
|||||||
CHECK(msr->operand == 8);
|
CHECK(msr->operand == 8);
|
||||||
CHECK(msr->spsr == false);
|
CHECK(msr->spsr == false);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "MSR CPSR_all,R8");
|
CHECK(instruction.disassemble() == "MSR CPSR_all,R8");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("MSR_flg with register operand") {
|
SECTION("MSR_flg with register operand") {
|
||||||
@@ -287,7 +310,9 @@ TEST_CASE("PSR Transfer", TAG) {
|
|||||||
CHECK(msr->operand == 8);
|
CHECK(msr->operand == 8);
|
||||||
CHECK(msr->spsr == false);
|
CHECK(msr->spsr == false);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "MSRVS CPSR_flg,R8");
|
CHECK(instruction.disassemble() == "MSRVS CPSR_flg,R8");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("MSR_flg with immediate operand") {
|
SECTION("MSR_flg with immediate operand") {
|
||||||
@@ -304,7 +329,9 @@ TEST_CASE("PSR Transfer", TAG) {
|
|||||||
CHECK(msr->operand == 27262976);
|
CHECK(msr->operand == 27262976);
|
||||||
CHECK(msr->spsr == true);
|
CHECK(msr->spsr == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "MSR SPSR_flg,#27262976");
|
CHECK(instruction.disassemble() == "MSR SPSR_flg,#27262976");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -331,6 +358,7 @@ TEST_CASE("Data Processing", TAG) {
|
|||||||
CHECK(alu->set == true);
|
CHECK(alu->set == true);
|
||||||
CHECK(alu->opcode == OpCode::AND);
|
CHECK(alu->opcode == OpCode::AND);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "ANDS R7,R14,R1,ROR #22");
|
CHECK(instruction.disassemble() == "ANDS R7,R14,R1,ROR #22");
|
||||||
|
|
||||||
shift->data.immediate = false;
|
shift->data.immediate = false;
|
||||||
@@ -392,6 +420,7 @@ TEST_CASE("Data Processing", TAG) {
|
|||||||
alu->opcode = OpCode::MVN;
|
alu->opcode = OpCode::MVN;
|
||||||
CHECK(instruction.disassemble() == "MVN R7,#3300012");
|
CHECK(instruction.disassemble() == "MVN R7,#3300012");
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Coprocessor Data Transfer", TAG) {
|
TEST_CASE("Coprocessor Data Transfer", TAG) {
|
||||||
@@ -412,6 +441,7 @@ TEST_CASE("Coprocessor Data Transfer", TAG) {
|
|||||||
CHECK(ldc->up == true);
|
CHECK(ldc->up == true);
|
||||||
CHECK(ldc->pre == true);
|
CHECK(ldc->pre == true);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "STCGE p1,c15,[R5,#70]!");
|
CHECK(instruction.disassemble() == "STCGE p1,c15,[R5,#70]!");
|
||||||
|
|
||||||
ldc->load = true;
|
ldc->load = true;
|
||||||
@@ -420,6 +450,7 @@ TEST_CASE("Coprocessor Data Transfer", TAG) {
|
|||||||
ldc->len = true;
|
ldc->len = true;
|
||||||
|
|
||||||
CHECK(instruction.disassemble() == "LDCGEL p1,c15,[R5],#70");
|
CHECK(instruction.disassemble() == "LDCGEL p1,c15,[R5],#70");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Coprocessor Operand Operation", TAG) {
|
TEST_CASE("Coprocessor Operand Operation", TAG) {
|
||||||
@@ -437,7 +468,9 @@ TEST_CASE("Coprocessor Operand Operation", TAG) {
|
|||||||
CHECK(cdp->crn == 5);
|
CHECK(cdp->crn == 5);
|
||||||
CHECK(cdp->cp_opc == 10);
|
CHECK(cdp->cp_opc == 10);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "CDP p1,10,c15,c5,c6,2");
|
CHECK(instruction.disassemble() == "CDP p1,10,c15,c5,c6,2");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Coprocessor Register Transfer", TAG) {
|
TEST_CASE("Coprocessor Register Transfer", TAG) {
|
||||||
@@ -457,7 +490,9 @@ TEST_CASE("Coprocessor Register Transfer", TAG) {
|
|||||||
CHECK(mrc->load == false);
|
CHECK(mrc->load == false);
|
||||||
CHECK(mrc->cp_opc == 5);
|
CHECK(mrc->cp_opc == 5);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "MCR p1,5,R15,c5,c6,2");
|
CHECK(instruction.disassemble() == "MCR p1,5,R15,c5,c6,2");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("Software Interrupt", TAG) {
|
TEST_CASE("Software Interrupt", TAG) {
|
||||||
@@ -465,5 +500,8 @@ TEST_CASE("Software Interrupt", TAG) {
|
|||||||
Instruction instruction(raw);
|
Instruction instruction(raw);
|
||||||
|
|
||||||
CHECK(instruction.condition == Condition::EQ);
|
CHECK(instruction.condition == Condition::EQ);
|
||||||
|
|
||||||
|
#ifdef DISASSEMBLER
|
||||||
CHECK(instruction.disassemble() == "SWIEQ");
|
CHECK(instruction.disassemble() == "SWIEQ");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
@@ -13,6 +13,12 @@ tests_sources = files(
|
|||||||
subdir('cpu')
|
subdir('cpu')
|
||||||
subdir('util')
|
subdir('util')
|
||||||
|
|
||||||
|
tests_cpp_args = []
|
||||||
|
|
||||||
|
if get_option('disassembler')
|
||||||
|
tests_cpp_args += '-DDISASSEMBLER'
|
||||||
|
endif
|
||||||
|
|
||||||
catch2 = dependency('catch2', version: '>=3.4.0', static: true)
|
catch2 = dependency('catch2', version: '>=3.4.0', static: true)
|
||||||
catch2_tests = executable(
|
catch2_tests = executable(
|
||||||
'matar_tests',
|
'matar_tests',
|
||||||
@@ -21,6 +27,7 @@ catch2_tests = executable(
|
|||||||
link_with: tests_deps,
|
link_with: tests_deps,
|
||||||
include_directories: [inc, src],
|
include_directories: [inc, src],
|
||||||
build_by_default: false,
|
build_by_default: false,
|
||||||
|
cpp_args: tests_cpp_args
|
||||||
)
|
)
|
||||||
|
|
||||||
test('catch2 tests', catch2_tests)
|
test('catch2 tests', catch2_tests)
|
||||||
|
Reference in New Issue
Block a user