Add skeleton CPU device and disassembler for HPC architecture

This commit is contained in:
AJR 2019-03-03 16:56:13 -05:00
parent 542a331fd2
commit 156619b401
9 changed files with 993 additions and 13 deletions

View File

@ -2800,3 +2800,20 @@ if (CPUS["ALPHA"]~=null or _OPTIONS["with-tools"]) then
table.insert(disasm_files , MAME_DIR .. "src/devices/cpu/alpha/alphad.cpp")
table.insert(disasm_files , MAME_DIR .. "src/devices/cpu/alpha/alphad.h")
end
--------------------------------------------------
-- National Semiconductor HPC
--@src/devices/cpu/hpc/hpc.h,CPUS["HPC"] = true
--------------------------------------------------
if (CPUS["HPC"]~=null) then
files {
MAME_DIR .. "src/devices/cpu/hpc/hpc.cpp",
MAME_DIR .. "src/devices/cpu/hpc/hpc.h",
}
end
if (CPUS["HPC"]~=null or _OPTIONS["with-tools"]) then
table.insert(disasm_files , MAME_DIR .. "src/devices/cpu/hpc/hpcdasm.cpp")
table.insert(disasm_files , MAME_DIR .. "src/devices/cpu/hpc/hpcdasm.h")
end

View File

@ -133,6 +133,7 @@ CPUS["HMCS40"] = true
--CPUS["SM510"] = true
CPUS["ST62XX"] = true
CPUS["DSPP"] = true
CPUS["HPC"] = true
--------------------------------------------------
-- specify available sound cores

View File

@ -138,6 +138,7 @@ CPUS["CLIPPER"] = true
CPUS["CAPRICORN"] = true
CPUS["ALPHA"] = true
--CPUS["DSPP"] = true
--CPUS["HPC"] = true
--------------------------------------------------
-- specify available sound cores; some of these are

102
src/devices/cpu/hpc/hpc.cpp Normal file
View File

@ -0,0 +1,102 @@
// license:BSD-3-Clause
// copyright-holders:AJR
/***************************************************************************
National Semiconductor High-Performance microController (HPC)
Currently this device is just a stub with no actual execution core.
***************************************************************************/
#include "emu.h"
#include "hpc.h"
#include "hpcdasm.h"
// device type definitions
DEFINE_DEVICE_TYPE(HPC46104, hpc46104_device, "hpc46104", "HPC46104")
void hpc46104_device::internal_map(address_map &map)
{
map(0x0000, 0x00bf).ram();
map(0x00c0, 0x00c0).rw(FUNC(hpc46104_device::psw_r), FUNC(hpc46104_device::psw_w));
map(0x00c4, 0x00cf).ram().share("core_regs");
// TODO: many other internal registers
map(0x01c0, 0x02ff).ram();
}
std::unique_ptr<util::disasm_interface> hpc46104_device::create_disassembler()
{
return std::make_unique<hpc16164_disassembler>();
}
hpc_device::hpc_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, u32 clock, address_map_constructor map)
: cpu_device(mconfig, type, tag, owner, clock)
, m_program_config("program", ENDIANNESS_LITTLE, 16, 16, 0, map)
, m_program(nullptr)
, m_core_regs(*this, "core_regs")
, m_psw(0)
, m_icount(0)
{
}
hpc46104_device::hpc46104_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock)
: hpc_device(mconfig, HPC46104, tag, owner, clock, address_map_constructor(FUNC(hpc46104_device::internal_map), this))
{
}
device_memory_interface::space_config_vector hpc_device::memory_space_config() const
{
return space_config_vector {
std::make_pair(AS_PROGRAM, &m_program_config),
};
}
void hpc_device::device_start()
{
m_program = &space(AS_PROGRAM);
set_icountptr(m_icount);
state_add(HPC_PSW, "PSW", m_psw);
state_add(HPC_SP, "SP", m_core_regs[0]);
state_add(HPC_PC, "PC", m_core_regs[1]);
state_add(STATE_GENPC, "GENPC", m_core_regs[1]).callimport().noshow();
state_add(STATE_GENPCBASE, "CURPC", m_core_regs[1]).callimport().noshow();
state_add(HPC_A, "A", m_core_regs[2]);
state_add(HPC_K, "K", m_core_regs[3]);
state_add(HPC_B, "B", m_core_regs[4]);
state_add(HPC_X, "X", m_core_regs[5]);
save_item(NAME(m_psw));
}
void hpc_device::device_reset()
{
m_psw = 0x00;
std::fill_n(&m_core_regs[0], 6, 0x0000);
}
u8 hpc_device::psw_r()
{
return m_psw;
}
void hpc_device::psw_w(u8 data)
{
m_psw = data;
}
void hpc_device::execute_run()
{
m_icount = 0;
}
void hpc_device::execute_set_input(int inputnum, int state)
{
// TODO
}

73
src/devices/cpu/hpc/hpc.h Normal file
View File

@ -0,0 +1,73 @@
// license:BSD-3-Clause
// copyright-holders:AJR
#ifndef MAME_CPU_HPC_HPC_H
#define MAME_CPU_HPC_HPC_H
#pragma once
class hpc_device : public cpu_device
{
public:
enum {
HPC_PSW,
HPC_SP,
HPC_PC,
HPC_A,
HPC_K,
HPC_B,
HPC_X
};
protected:
// construction/destruction
hpc_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, u32 clock, address_map_constructor map);
// device-level overrides
virtual void device_start() override;
virtual void device_reset() override;
// device_execute_interface overrides
virtual void execute_run() override;
virtual void execute_set_input(int inputnum, int state) override;
// device_memory_interface overrides
virtual space_config_vector memory_space_config() const override;
// internal register access
u8 psw_r();
void psw_w(u8 data);
private:
// address space
address_space_config m_program_config;
address_space *m_program;
// internal state
required_shared_ptr<u16> m_core_regs;
u8 m_psw;
// execution sequencing
s32 m_icount;
};
class hpc46104_device : public hpc_device
{
public:
// construction/destruction
hpc46104_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock);
protected:
// device_disasm_interface overrides
virtual std::unique_ptr<util::disasm_interface> create_disassembler() override;
private:
void internal_map(address_map &map);
};
// device type declarations
DECLARE_DEVICE_TYPE(HPC46104, hpc46104_device)
#endif // MAME_CPU_HPC_HPC_H

View File

@ -0,0 +1,739 @@
// license:BSD-3-Clause
// copyright-holders:AJR
/***************************************************************************
National Semiconductor HPC disassembler
Note that though all 16-bit fields in instructions have the MSB first,
the HPC's memory organization is in fact little-endian. This is why r16
is not used to read them.
***************************************************************************/
#include "util/disasmintf.h"
#include "hpcdasm.h"
#include "util/strformat.h"
#include <ctype.h>
const char *const hpc16164_disassembler::s_regs[128] =
{
"psw", nullptr, "SP", "PC", "A", "K", "B", "X",
"enir", "irpd", "ircd", "sio", "porti", nullptr, "halten", "romdump",
"porta", "portb", nullptr, "upic", nullptr, nullptr, nullptr, nullptr,
"dira", "dirb", "bfun", nullptr, nullptr, nullptr, nullptr, nullptr,
"adcr1", "adcr2", "portd", "adcr3", nullptr, nullptr, nullptr, nullptr,
"ad0", "ad1", "ad2", "ad3", "ad4", "ad5", "ad6", "ad7",
"enu", "enui", "rbuf", "tbuf", "enur", nullptr, nullptr, nullptr,
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr,
"t4", "r4", "t5", "r5", "t6", "r6", "t7", "r7",
"pwmode", "portp", nullptr, nullptr, nullptr, nullptr, "eicon", "eicr",
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr,
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr,
"i4cr", "i3cr", "i2cr", "r2", "t2", "r3", "t3", "divby",
"tmmode", "t0con", "watchdog", nullptr, nullptr, nullptr, nullptr, nullptr,
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr,
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr, nullptr
};
hpc_disassembler::hpc_disassembler(const char *const regs[])
: util::disasm_interface()
, m_regs(regs)
{
}
u32 hpc_disassembler::opcode_alignment() const
{
return 1;
}
void hpc_disassembler::format_register(std::ostream &stream, u16 reg) const
{
if (reg >= 0x00c0 && reg < 0x01c0)
{
const char *name = m_regs[(reg - 0x00c0) >> 1];
if (name != nullptr)
{
stream << name;
if (BIT(reg, 0))
stream << "+1";
return;
}
}
util::stream_format(stream, "0%X", reg);
}
void hpc_disassembler::format_immediate_byte(std::ostream &stream, u8 data) const
{
stream << "#";
if (data >= 0x10)
stream << "0";
util::stream_format(stream, "%02X", data);
}
void hpc_disassembler::format_immediate_word(std::ostream &stream, u16 data) const
{
stream << "#";
if (data >= 0x1000)
stream << "0";
util::stream_format(stream, "%04X", data);
}
void hpc_disassembler::disassemble_op(std::ostream &stream, const char *op, u16 reg, u16 src, bool imm, bool indir, bool idx, bool w) const
{
util::stream_format(stream, "%-8s", op);
if (idx)
stream << "A";
else
format_register(stream, reg);
stream << ",";
if (imm)
{
if (w)
format_immediate_word(stream, src);
else
format_immediate_byte(stream, src);
}
else
{
if (idx)
util::stream_format(stream, "0%X", u16(reg));
if (indir)
stream << "[";
format_register(stream, src);
if (indir)
stream << "]";
if (w)
stream << ".w";
else
stream << ".b";
}
}
void hpc_disassembler::disassemble_unary_op(std::ostream &stream, const char *op, u16 offset, u16 src, bool indir, bool idx, bool w) const
{
util::stream_format(stream, "%-8s", op);
if (idx)
util::stream_format(stream, "0%X", offset);
if (indir)
stream << "[";
format_register(stream, src);
if (indir)
stream << "]";
if (w)
{
if (src < 0x00c4 || src >= 0x00d0 || indir)
stream << ".w";
}
else
stream << ".b";
}
void hpc_disassembler::disassemble_bit_op(std::ostream &stream, const char *op, u8 bit, u16 offset, u16 src, bool indir, bool idx) const
{
if (src >= 0x00c0 && src < 0x01c0 && BIT(src, 0) && !indir && m_regs[(src - 0x00c0) >> 1] != nullptr)
{
src &= 0xfffe;
bit += 8;
}
util::stream_format(stream, "%-8s%d,", op, bit);
if (idx)
util::stream_format(stream, "0%X", offset);
if (indir)
stream << "[";
format_register(stream, src);
if (indir)
stream << "]";
}
offs_t hpc_disassembler::disassemble(std::ostream &stream, offs_t pc, const hpc_disassembler::data_buffer &opcodes, const hpc_disassembler::data_buffer &params)
{
u8 opcode = opcodes.r8(pc);
u16 reg = REGISTER_A;
u16 src = REGISTER_B;
bool imm = false;
bool dmode = false;
bool indir = true;
bool idx = false;
bool jmp = false;
offs_t bytes = 1;
switch (opcode)
{
case 0x20: case 0x21: case 0x22: case 0x23:
case 0x24: case 0x25: case 0x26: case 0x27:
case 0x28: case 0x29: case 0x2a: case 0x2b:
case 0x2c: case 0x2d: case 0x2e: case 0x2f:
jmp = true;
src = 0xffd0 + (opcode & 0x0f) * 2;
bytes = 1;
break;
case 0x30: case 0x31: case 0x32: case 0x33:
jmp = true;
src = pc + 2 + ((opcode & 0x03) << 8 | opcodes.r8(pc + 1));
bytes = 2;
break;
case 0x34: case 0x35: case 0x36: case 0x37:
jmp = true;
src = pc - ((opcode & 0x03) << 8 | opcodes.r8(pc + 1));
bytes = 2;
break;
case 0x3f:
case 0x89: case 0x8a:
case 0xa9: case 0xaa: case 0xaf:
indir = false;
src = opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x40: case 0x41: case 0x42: case 0x43:
case 0x44: case 0x45: case 0x46: case 0x47:
case 0x48: case 0x49: case 0x4a: case 0x4b:
case 0x4c: case 0x4d: case 0x4e: case 0x4f:
case 0x50: case 0x51: case 0x52: case 0x53:
case 0x54: case 0x55: case 0x56: case 0x57:
case 0x58: case 0x59: case 0x5a: case 0x5b:
case 0x5c: case 0x5d: case 0x5e: case 0x5f:
jmp = true;
src = pc + 1 + (opcode & 0x1f);
break;
case 0x60: case 0x61: case 0x62: case 0x63:
case 0x64: case 0x65: case 0x66: case 0x67:
case 0x68: case 0x69: case 0x6a: case 0x6b:
case 0x6c: case 0x6d: case 0x6e: case 0x6f:
case 0x70: case 0x71: case 0x72: case 0x73:
case 0x74: case 0x75: case 0x76: case 0x77:
case 0x78: case 0x79: case 0x7a: case 0x7b:
case 0x7c: case 0x7d: case 0x7e: case 0x7f:
jmp = true;
src = pc - (opcode & 0x1f);
break;
case 0x80: case 0x82:
case 0xa0:
dmode = true;
indir = false;
if (BIT(opcode, 1))
imm = true;
src = opcodes.r8(pc + 1);
reg = opcodes.r8(pc + 2);
opcode = opcodes.r8(pc + 3);
bytes = 4;
break;
case 0x84: case 0x86:
case 0xa4:
dmode = true;
indir = false;
if (BIT(opcode, 1))
imm = true;
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
reg = opcodes.r8(pc + 3);
opcode = opcodes.r8(pc + 4);
bytes = 5;
break;
case 0x81: case 0x83:
case 0xa1:
dmode = true;
indir = false;
if (BIT(opcode, 1))
imm = true;
src = opcodes.r8(pc + 1);
reg = (opcodes.r8(pc + 2) << 8) | opcodes.r8(pc + 3);
opcode = opcodes.r8(pc + 4);
bytes = 5;
break;
case 0xa2:
idx = true;
reg = opcodes.r8(pc + 1);
src = opcodes.r8(pc + 2);
opcode = opcodes.r8(pc + 3);
bytes = 4;
break;
case 0x85: case 0x87:
case 0xa5:
dmode = true;
indir = false;
if (BIT(opcode, 1))
imm = true;
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
reg = (opcodes.r8(pc + 3) << 8) | opcodes.r8(pc + 4);
opcode = opcodes.r8(pc + 5);
bytes = 6;
break;
case 0xa6:
idx = true;
reg = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
src = opcodes.r8(pc + 3);
opcode = opcodes.r8(pc + 4);
bytes = 5;
break;
case 0x88: case 0x8b: case 0x8e:
case 0xa8: case 0xab: case 0xae:
indir = false;
src = opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x8c:
case 0xac:
indir = false;
src = opcodes.r8(pc + 1);
reg = opcodes.r8(pc + 2);
bytes = 3;
break;
case 0x8d:
imm = true;
src = opcodes.r8(pc + 1);
reg = opcodes.r8(pc + 2);
bytes = 3;
break;
case 0x8f:
src = REGISTER_X;
opcode = opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x90: case 0x91: case 0x92: case 0x93:
imm = true;
reg = 0x00c8 | ((opcode & 0x03) << 1);
src = opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x98: case 0x99: case 0x9a: case 0x9b:
case 0x9c: case 0x9d: case 0x9e: case 0x9f:
imm = true;
src = opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x94:
jmp = true;
src = pc + 2 + opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x95:
jmp = true;
src = pc - opcodes.r8(pc + 1);
bytes = 2;
break;
case 0x96:
indir = false;
src = opcodes.r8(pc + 1);
opcode = opcodes.r8(pc + 2);
bytes = 3;
break;
case 0x97:
indir = false;
imm = true;
src = opcodes.r8(pc + 1);
reg = opcodes.r8(pc + 2);
bytes = 3;
break;
case 0xa7:
imm = true;
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
reg = (opcodes.r8(pc + 3) << 8) | opcodes.r8(pc + 4);
bytes = 5;
break;
case 0xad:
src = opcodes.r8(pc + 1);
opcode = opcodes.r8(pc + 2);
bytes = 3;
break;
case 0xb0: case 0xb1: case 0xb2: case 0xb3:
imm = true;
reg = 0x00c8 | ((opcode & 0x03) << 1);
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
bytes = 3;
break;
case 0xb8: case 0xb9: case 0xba: case 0xbb:
case 0xbc: case 0xbd: case 0xbe: case 0xbf:
imm = true;
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
bytes = 3;
break;
case 0xb4: case 0xb5:
jmp = true;
src = pc + 3 + ((opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2));
bytes = 3;
break;
case 0xb6:
indir = false;
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
opcode = opcodes.r8(pc + 3);
bytes = 4;
break;
case 0xb7:
imm = true;
src = (opcodes.r8(pc + 1) << 8) | opcodes.r8(pc + 2);
reg = opcodes.r8(pc + 3);
bytes = 4;
break;
case 0xd4: case 0xd5: case 0xd6:
case 0xf4: case 0xf5: case 0xf6:
src = REGISTER_X;
break;
}
switch (opcode)
{
case 0x00:
util::stream_format(stream, "%-8sA", "clr");
break;
case 0x01:
util::stream_format(stream, "%-8sA", "comp");
break;
case 0x02:
stream << "sc";
break;
case 0x03:
stream << "rc";
break;
case 0x04:
util::stream_format(stream, "%-8sA", "inc");
break;
case 0x05:
util::stream_format(stream, "%-8sA", "dec");
break;
case 0x06:
stream << "ifnc";
break;
case 0x07:
stream << "ifc";
break;
case 0x08: case 0x09: case 0x0a: case 0x0b:
case 0x0c: case 0x0d: case 0x0e: case 0x0f:
disassemble_bit_op(stream, "sbit", opcode & 0x07, reg, src, indir, idx);
break;
case 0x10: case 0x11: case 0x12: case 0x13:
case 0x14: case 0x15: case 0x16: case 0x17:
disassemble_bit_op(stream, "ifbit", opcode & 0x07, reg, src, indir, idx);
break;
case 0x18: case 0x19: case 0x1a: case 0x1b:
case 0x1c: case 0x1d: case 0x1e: case 0x1f:
disassemble_bit_op(stream, "rbit", opcode & 0x07, reg, src, indir, idx);
break;
case 0x20: case 0x21: case 0x22: case 0x23:
case 0x24: case 0x25: case 0x26: case 0x27:
case 0x28: case 0x29: case 0x2a: case 0x2b:
case 0x2c: case 0x2d: case 0x2e: case 0x2f:
util::stream_format(stream, "%-8s", "jsrp");
if (jmp)
util::stream_format(stream, "[0%X]", src);
else
stream << "???";
bytes |= STEP_OVER;
break;
case 0x30: case 0x31: case 0x32: case 0x33:
case 0x34: case 0x35: case 0x36: case 0x37:
util::stream_format(stream, "%-8s", "jsr");
if (jmp)
util::stream_format(stream, "0%X", src);
else
stream << "???";
bytes |= STEP_OVER;
break;
case 0x38:
util::stream_format(stream, "%-8sX,", "rbit");
format_register(stream, reg);
break;
case 0x39:
util::stream_format(stream, "%-8sX,", "sbit");
format_register(stream, reg);
break;
case 0x3a:
util::stream_format(stream, "%-8sX,", "ifbit");
format_register(stream, reg);
break;
case 0x3b:
util::stream_format(stream, "%-8sA", "swap");
break;
case 0x3c:
stream << "ret";
bytes |= STEP_OUT;
break;
case 0x3d:
stream << "retsk";
bytes |= STEP_OUT;
break;
case 0x3e:
stream << "reti";
bytes |= STEP_OUT;
break;
case 0x3f:
util::stream_format(stream, "%-8s", "pop");
format_register(stream, src);
break;
case 0x40:
stream << "nop";
break;
case 0x41: case 0x42: case 0x43:
case 0x44: case 0x45: case 0x46: case 0x47:
case 0x48: case 0x49: case 0x4a: case 0x4b:
case 0x4c: case 0x4d: case 0x4e: case 0x4f:
case 0x50: case 0x51: case 0x52: case 0x53:
case 0x54: case 0x55: case 0x56: case 0x57:
case 0x58: case 0x59: case 0x5a: case 0x5b:
case 0x5c: case 0x5d: case 0x5e: case 0x5f:
case 0x60: case 0x61: case 0x62: case 0x63:
case 0x64: case 0x65: case 0x66: case 0x67:
case 0x68: case 0x69: case 0x6a: case 0x6b:
case 0x6c: case 0x6d: case 0x6e: case 0x6f:
case 0x70: case 0x71: case 0x72: case 0x73:
case 0x74: case 0x75: case 0x76: case 0x77:
case 0x78: case 0x79: case 0x7a: case 0x7b:
case 0x7c: case 0x7d: case 0x7e: case 0x7f:
util::stream_format(stream, "%-8s", "jp");
if (jmp)
util::stream_format(stream, "0%X", src);
else
stream << "???";
break;
case 0x88: case 0x8c:
case 0x90: case 0x91: case 0x92: case 0x93:
case 0x97:
case 0xa8: case 0xac:
case 0xb0: case 0xb1: case 0xb2: case 0xb3:
case 0xb7:
case 0xc4:
case 0xd4:
case 0xe4:
case 0xf4:
disassemble_op(stream, "ld", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x89:
case 0xa9:
disassemble_unary_op(stream, "inc", reg, src, indir, idx, BIT(opcode, 5));
break;
case 0x8a:
case 0xaa:
disassemble_unary_op(stream, "decsz", reg, src, indir, idx, BIT(opcode, 5));
bytes |= STEP_OVER | (1 << OVERINSTSHIFT);
break;
case 0x8b:
case 0xab:
case 0xc6:
case 0xd6:
case 0xe6:
case 0xf6:
disassemble_op(stream, dmode ? "ld" : "st", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x8d:
util::stream_format(stream, "%-8sBK,", "ld");
format_immediate_byte(stream, src);
stream << ",";
format_immediate_byte(stream, reg);
break;
case 0x8e:
case 0xae:
case 0xc5:
case 0xd5:
case 0xe5:
case 0xf5:
disassemble_op(stream, "x", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x94: case 0x95:
util::stream_format(stream, "%-8s", "jp");
if (jmp)
util::stream_format(stream, "0%X", src);
else
stream << "???";
break;
case 0x98: case 0xb8: case 0xd8: case 0xf8:
disassemble_op(stream, "add", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x99: case 0xb9: case 0xd9: case 0xf9:
disassemble_op(stream, "and", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x9a: case 0xba: case 0xda: case 0xfa:
disassemble_op(stream, "or", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x9b: case 0xbb: case 0xdb: case 0xfb:
disassemble_op(stream, "xor", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x9c: case 0xbc: case 0xdc: case 0xfc:
disassemble_op(stream, "ifeq", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x9d: case 0xbd: case 0xdd: case 0xfd:
disassemble_op(stream, "ifgt", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x9e: case 0xbe: case 0xde: case 0xfe:
disassemble_op(stream, "mult", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0x9f: case 0xbf: case 0xdf: case 0xff:
disassemble_op(stream, "div", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0xa7:
util::stream_format(stream, "%-8sBK,", "ld");
format_immediate_word(stream, src);
stream << ",";
format_immediate_word(stream, reg);
break;
case 0xaf:
util::stream_format(stream, "%-8s", "push");
format_register(stream, src);
break;
case 0xb4:
util::stream_format(stream, "%-8s", "jmpl");
if (jmp)
util::stream_format(stream, "0%X", src);
else
stream << "???";
break;
case 0xb5:
util::stream_format(stream, "%-8s", "jsrl");
if (jmp)
util::stream_format(stream, "0%X", src);
else
stream << "???";
bytes |= STEP_OVER;
break;
case 0xc0: case 0xc2:
case 0xe0: case 0xe2:
util::stream_format(stream, "%-8sA,[B%c].%c", "lds",
BIT(opcode, 1) ? '-' : '+',
BIT(opcode, 5) ? 'w' : 'b');
bytes |= STEP_OVER | (1 << OVERINSTSHIFT);
break;
case 0xd0: case 0xd2:
case 0xf0: case 0xf2:
util::stream_format(stream, "%-8sA,[X%c].%c", "ld",
BIT(opcode, 1) ? '-' : '+',
BIT(opcode, 5) ? 'w' : 'b');
break;
case 0xc1: case 0xc3:
case 0xe1: case 0xe3:
util::stream_format(stream, "%-8sA,[B%c].%c", "xs",
BIT(opcode, 1) ? '-' : '+',
BIT(opcode, 5) ? 'w' : 'b');
bytes |= STEP_OVER | (1 << OVERINSTSHIFT);
break;
case 0xd1: case 0xd3:
case 0xf1: case 0xf3:
util::stream_format(stream, "%-8sA,[X%c].%c", "x",
BIT(opcode, 1) ? '-' : '+',
BIT(opcode, 5) ? 'w' : 'b');
break;
case 0xc7: case 0xe7:
util::stream_format(stream, "%-8sA", BIT(opcode, 5) ? "shl" : "shr");
break;
case 0xd7: case 0xf7:
util::stream_format(stream, "%-8sA", BIT(opcode, 5) ? "rlc" : "rrc");
break;
case 0xc8: case 0xe8:
disassemble_op(stream, "adc", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0xc9: case 0xe9:
disassemble_op(stream, "dadc", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0xca: case 0xea:
disassemble_op(stream, "dsubc", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0xcb: case 0xeb:
disassemble_op(stream, "subc", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
case 0xcc: case 0xec:
stream << "jid";
if (BIT(opcode, 5))
stream << "w";
break;
case 0xcf: case 0xef:
disassemble_op(stream, "divd", reg, src, imm, indir, idx, BIT(opcode, 5));
break;
default:
stream << "???";
break;
}
return bytes | SUPPORTED;
}

View File

@ -0,0 +1,49 @@
// license:BSD-3-Clause
// copyright-holders:AJR
#ifndef MAME_CPU_HPC_HPCDASM_H
#define MAME_CPU_HPC_HPCDASM_H
#pragma once
class hpc_disassembler : public util::disasm_interface
{
protected:
enum : u16
{
REGISTER_A = 0x00c8,
REGISTER_K = 0x00ca,
REGISTER_B = 0x00cc,
REGISTER_X = 0x00ce
};
// construction/destruction
hpc_disassembler(const char *const regs[]);
// disassembler overrides
virtual u32 opcode_alignment() const override;
virtual offs_t disassemble(std::ostream &stream, offs_t pc, const data_buffer &opcodes, const data_buffer &params) override;
private:
// internal helpers
void format_register(std::ostream &stream, u16 reg) const;
void format_immediate_byte(std::ostream &stream, u8 data) const;
void format_immediate_word(std::ostream &stream, u16 data) const;
void disassemble_op(std::ostream &stream, const char *op, u16 reg, u16 src, bool imm, bool indir, bool idx, bool w) const;
void disassemble_unary_op(std::ostream &stream, const char *op, u16 offset, u16 src, bool indir, bool idx, bool w) const;
void disassemble_bit_op(std::ostream &stream, const char *op, u8 bit, u16 offset, u16 src, bool indir, bool idx) const;
// register names
const char *const *m_regs;
};
class hpc16164_disassembler : public hpc_disassembler
{
public:
hpc16164_disassembler() : hpc_disassembler(s_regs) { }
private:
static const char *const s_regs[128];
};
#endif // MAME_CPU_HPC_HPCDASM_H

View File

@ -13,6 +13,7 @@ Device is a 27c256 location U3
*/
#include "emu.h"
#include "cpu/hpc/hpc.h"
#include "speaker.h"
@ -21,7 +22,7 @@ class age_candy_state : public driver_device
public:
age_candy_state(const machine_config &mconfig, device_type type, const char *tag)
: driver_device(mconfig, type, tag)
// ,m_maincpu(*this, "maincpu")
, m_maincpu(*this, "maincpu")
{ }
void age_candy(machine_config &config);
@ -31,7 +32,7 @@ private:
virtual void machine_reset() override;
void age_candy_map(address_map &map);
// required_device<mcs51_cpu_device> m_maincpu;
required_device<hpc_device> m_maincpu;
};
static INPUT_PORTS_START( age_candy )
@ -48,24 +49,19 @@ void age_candy_state::machine_reset()
}
#ifdef UNUSED_DEFINITION
void age_candy_state::age_candy_map(address_map &map)
{
map(0xc000, 0xffff).rom().region("maincpu", 0x4000);
map(0x8000, 0xffff).rom().region("maincpu", 0);
}
#endif
MACHINE_CONFIG_START(age_candy_state::age_candy)
/* basic machine hardware */
// MCFG_DEVICE_ADD("maincpu", HPC46104, 8000000) // unknown clock; HPC emulation needed
// MCFG_DEVICE_PROGRAM_MAP(age_candy_map)
// MCFG_DEVICE_IO_MAP(age_candy_io)
void age_candy_state::age_candy(machine_config &config)
{
HPC46104(config, m_maincpu, 8000000); // unknown clock; HPC still not actually emulated
m_maincpu->set_addrmap(AS_PROGRAM, &age_candy_state::age_candy_map);
/* sound hardware */
SPEAKER(config, "mono").front_center();
MACHINE_CONFIG_END
}
ROM_START( age_cand )

View File

@ -55,6 +55,7 @@ using util::BIT;
#include "cpu/hcd62121/hcd62121d.h"
#include "cpu/hd61700/hd61700d.h"
#include "cpu/hmcs40/hmcs40d.h"
#include "cpu/hpc/hpcdasm.h"
#include "cpu/hphybrid/hphybrid_dasm.h"
#include "cpu/i386/i386dasm.h"
#include "cpu/i8008/8008dasm.h"
@ -360,6 +361,7 @@ static const dasm_table_entry dasm_table[] =
{ "hp_5061_3001", be, -1, []() -> util::disasm_interface * { return new hp_5061_3001_disassembler; } },
{ "hp_5061_3011", be, -1, []() -> util::disasm_interface * { return new hp_5061_3011_disassembler; } },
{ "hp_09825_67907", be, -1, []() -> util::disasm_interface * { return new hp_09825_67907_disassembler; } },
{ "hpc16164", le, 0, []() -> util::disasm_interface * { return new hpc16164_disassembler; } },
{ "hyperstone", be, 0, []() -> util::disasm_interface * { return new hyperstone_disassembler(&hyperstone_unidasm); } },
{ "i4004", le, 0, []() -> util::disasm_interface * { return new i4004_disassembler; } },
{ "i4040", le, 0, []() -> util::disasm_interface * { return new i4040_disassembler; } },