mirror of
https://github.com/holub/mame
synced 2025-07-01 16:19:38 +03:00
pdp8: Rewrite disassembler; add alternate disassembly for HD-6120
* unidasm: Add option to use octal instead of hexadecimal for output
This commit is contained in:
parent
a068ff3644
commit
2fa2ef8b0e
File diff suppressed because it is too large
Load Diff
@ -1,10 +1,5 @@
|
|||||||
// license:BSD-3-Clause
|
// license:BSD-3-Clause
|
||||||
// copyright-holders:Ryan Holtz
|
// copyright-holders:AJR
|
||||||
/*
|
|
||||||
First-gen DEC PDP-8 disassembler
|
|
||||||
|
|
||||||
Written by Ryan Holtz
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef MAME_CPU_PDP8_PDP8DASM_H
|
#ifndef MAME_CPU_PDP8_PDP8DASM_H
|
||||||
#define MAME_CPU_PDP8_PDP8DASM_H
|
#define MAME_CPU_PDP8_PDP8DASM_H
|
||||||
@ -14,11 +9,36 @@
|
|||||||
class pdp8_disassembler : public util::disasm_interface
|
class pdp8_disassembler : public util::disasm_interface
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
pdp8_disassembler() = default;
|
pdp8_disassembler(bool has_r3l = false);
|
||||||
virtual ~pdp8_disassembler() = default;
|
|
||||||
|
|
||||||
|
protected:
|
||||||
|
// util::disasm_interface overrides
|
||||||
virtual u32 opcode_alignment() const override;
|
virtual u32 opcode_alignment() const override;
|
||||||
virtual offs_t disassemble(std::ostream &stream, offs_t pc, const data_buffer &opcodes, const data_buffer ¶ms) override;
|
virtual offs_t disassemble(std::ostream &stream, offs_t pc, const data_buffer &opcodes, const data_buffer ¶ms) override;
|
||||||
|
|
||||||
|
void dasm_memory_reference(std::ostream &stream, u16 inst, offs_t pc);
|
||||||
|
virtual offs_t dasm_iot(std::ostream &stream, u16 dev, offs_t pc);
|
||||||
|
void dasm_opr_group1(std::ostream &stream, u16 inst);
|
||||||
|
void dasm_opr_group2(std::ostream &stream, u16 inst);
|
||||||
|
virtual offs_t dasm_opr_group3(std::ostream &stream, u16 inst, offs_t pc, const data_buffer &opcodes);
|
||||||
|
|
||||||
|
private:
|
||||||
|
const bool m_has_r3l;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
class hd6120_disassembler : public pdp8_disassembler
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
hd6120_disassembler();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
// util::disasm_interface overrides
|
||||||
|
virtual u32 interface_flags() const override;
|
||||||
|
virtual u32 page_address_bits() const override;
|
||||||
|
virtual u32 page2_address_bits() const override;
|
||||||
|
|
||||||
|
virtual offs_t dasm_iot(std::ostream &stream, u16 dev, offs_t pc) override;
|
||||||
|
virtual offs_t dasm_opr_group3(std::ostream &stream, u16 inst, offs_t pc, const data_buffer &opcodes) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // MAME_CPU_PDP8_PDP8DASM_H
|
||||||
|
@ -357,6 +357,7 @@ struct options
|
|||||||
const dasm_table_entry *dasm;
|
const dasm_table_entry *dasm;
|
||||||
uint32_t skip;
|
uint32_t skip;
|
||||||
uint32_t count;
|
uint32_t count;
|
||||||
|
bool octal;
|
||||||
};
|
};
|
||||||
|
|
||||||
static const dasm_table_entry dasm_table[] =
|
static const dasm_table_entry dasm_table[] =
|
||||||
@ -426,6 +427,7 @@ static const dasm_table_entry dasm_table[] =
|
|||||||
{ "h16", be, 0, []() -> util::disasm_interface * { return new h16_disassembler; } },
|
{ "h16", be, 0, []() -> util::disasm_interface * { return new h16_disassembler; } },
|
||||||
{ "hc11", be, 0, []() -> util::disasm_interface * { return new hc11_disassembler; } },
|
{ "hc11", be, 0, []() -> util::disasm_interface * { return new hc11_disassembler; } },
|
||||||
{ "hcd62121", le, 0, []() -> util::disasm_interface * { return new hcd62121_disassembler; } },
|
{ "hcd62121", le, 0, []() -> util::disasm_interface * { return new hcd62121_disassembler; } },
|
||||||
|
{ "hd6120", be, -1, []() -> util::disasm_interface * { return new hd6120_disassembler; } },
|
||||||
{ "hd61700", le, -1, []() -> util::disasm_interface * { return new hd61700_disassembler; } },
|
{ "hd61700", le, -1, []() -> util::disasm_interface * { return new hd61700_disassembler; } },
|
||||||
{ "hd6301", be, 0, []() -> util::disasm_interface * { return new m680x_disassembler(6301); } },
|
{ "hd6301", be, 0, []() -> util::disasm_interface * { return new m680x_disassembler(6301); } },
|
||||||
{ "hd6309", be, 0, []() -> util::disasm_interface * { return new hd6309_disassembler; } },
|
{ "hd6309", be, 0, []() -> util::disasm_interface * { return new hd6309_disassembler; } },
|
||||||
@ -1110,6 +1112,8 @@ static int parse_options(int argc, char *argv[], options *opts)
|
|||||||
opts->upper = true;
|
opts->upper = true;
|
||||||
else if(tolower((uint8_t)curarg[1]) == 'x')
|
else if(tolower((uint8_t)curarg[1]) == 'x')
|
||||||
opts->xchbytes = true;
|
opts->xchbytes = true;
|
||||||
|
else if(tolower((uint8_t)curarg[1]) == 'o')
|
||||||
|
opts->octal = true;
|
||||||
else
|
else
|
||||||
goto usage;
|
goto usage;
|
||||||
|
|
||||||
@ -1183,7 +1187,7 @@ static int parse_options(int argc, char *argv[], options *opts)
|
|||||||
usage:
|
usage:
|
||||||
printf("Usage: %s <filename> -arch <architecture> [-basepc <pc>] \n", argv[0]);
|
printf("Usage: %s <filename> -arch <architecture> [-basepc <pc>] \n", argv[0]);
|
||||||
printf(" [-mode <n>] [-norawbytes] [-xchbytes] [-flipped] [-upper] [-lower]\n");
|
printf(" [-mode <n>] [-norawbytes] [-xchbytes] [-flipped] [-upper] [-lower]\n");
|
||||||
printf(" [-skip <n>] [-count <n>]\n");
|
printf(" [-skip <n>] [-count <n>] [-octal]\n");
|
||||||
printf("\n");
|
printf("\n");
|
||||||
printf("Supported architectures:");
|
printf("Supported architectures:");
|
||||||
const int colwidth = 1 + std::strlen(std::max_element(std::begin(dasm_table), std::end(dasm_table), [](const dasm_table_entry &a, const dasm_table_entry &b) { return std::strlen(a.name) < std::strlen(b.name); })->name);
|
const int colwidth = 1 + std::strlen(std::max_element(std::begin(dasm_table), std::end(dasm_table), [](const dasm_table_entry &a, const dasm_table_entry &b) { return std::strlen(a.name) < std::strlen(b.name); })->name);
|
||||||
@ -1350,7 +1354,7 @@ int main(int argc, char *argv[])
|
|||||||
// pc to string conversion
|
// pc to string conversion
|
||||||
std::function<std::string (offs_t pc)> pc_to_string;
|
std::function<std::string (offs_t pc)> pc_to_string;
|
||||||
int aw = 32 - count_leading_zeros(pc_mask);
|
int aw = 32 - count_leading_zeros(pc_mask);
|
||||||
bool is_octal = false; // Parameter? Per-cpu config?
|
bool is_octal = opts.octal; // Parameter? Per-cpu config?
|
||||||
if((flags & util::disasm_interface::PAGED2LEVEL) == util::disasm_interface::PAGED2LEVEL) {
|
if((flags & util::disasm_interface::PAGED2LEVEL) == util::disasm_interface::PAGED2LEVEL) {
|
||||||
int bits1 = disasm->page_address_bits();
|
int bits1 = disasm->page_address_bits();
|
||||||
int bits2 = disasm->page2_address_bits();
|
int bits2 = disasm->page2_address_bits();
|
||||||
@ -1471,6 +1475,65 @@ int main(int argc, char *argv[])
|
|||||||
max_len >>= granularity_shift;
|
max_len >>= granularity_shift;
|
||||||
std::function<std::string (offs_t pc, offs_t size)> dump_raw_bytes;
|
std::function<std::string (offs_t pc, offs_t size)> dump_raw_bytes;
|
||||||
|
|
||||||
|
if(is_octal) {
|
||||||
|
switch(granularity) {
|
||||||
|
case 1:
|
||||||
|
dump_raw_bytes = [step = disasm->opcode_alignment(), next_pc, base_buffer](offs_t pc, offs_t size) -> std::string {
|
||||||
|
std::string result = "";
|
||||||
|
for(offs_t i=0; i != size; i++) {
|
||||||
|
if(i)
|
||||||
|
result += ' ';
|
||||||
|
result += util::string_format("%03o", base_buffer.r8(pc));
|
||||||
|
pc = next_pc(pc, step);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
max_len = (max_len * 4) - 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
dump_raw_bytes = [step = disasm->opcode_alignment(), next_pc, base_buffer](offs_t pc, offs_t size) -> std::string {
|
||||||
|
std::string result = "";
|
||||||
|
for(offs_t i=0; i != size; i++) {
|
||||||
|
if(i)
|
||||||
|
result += ' ';
|
||||||
|
result += util::string_format("%06o", base_buffer.r16(pc));
|
||||||
|
pc = next_pc(pc, step);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
max_len = (max_len * 7) - 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 4:
|
||||||
|
dump_raw_bytes = [step = disasm->opcode_alignment(), next_pc, base_buffer](offs_t pc, offs_t size) -> std::string {
|
||||||
|
std::string result = "";
|
||||||
|
for(offs_t i=0; i != size; i++) {
|
||||||
|
if(i)
|
||||||
|
result += ' ';
|
||||||
|
result += util::string_format("%011o", base_buffer.r32(pc));
|
||||||
|
pc = next_pc(pc, step);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
max_len = (max_len * 12) - 1;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 8:
|
||||||
|
dump_raw_bytes = [step = disasm->opcode_alignment(), next_pc, base_buffer](offs_t pc, offs_t size) -> std::string {
|
||||||
|
std::string result = "";
|
||||||
|
for(offs_t i=0; i != size; i++) {
|
||||||
|
if(i)
|
||||||
|
result += ' ';
|
||||||
|
result += util::string_format("%022o", base_buffer.r64(pc));
|
||||||
|
pc = next_pc(pc, step);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
max_len = (max_len * 23) - 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
switch(granularity) {
|
switch(granularity) {
|
||||||
case 1:
|
case 1:
|
||||||
dump_raw_bytes = [step = disasm->opcode_alignment(), next_pc, base_buffer](offs_t pc, offs_t size) -> std::string {
|
dump_raw_bytes = [step = disasm->opcode_alignment(), next_pc, base_buffer](offs_t pc, offs_t size) -> std::string {
|
||||||
@ -1528,6 +1591,7 @@ int main(int argc, char *argv[])
|
|||||||
max_len = (max_len * 17) - 1;
|
max_len = (max_len * 17) - 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if(opts.flipped) {
|
if(opts.flipped) {
|
||||||
if(opts.norawbytes)
|
if(opts.norawbytes)
|
||||||
|
Loading…
Reference in New Issue
Block a user