mame/src/devices/cpu/sm510/sm590core.cpp
2017-05-10 01:30:03 -04:00

179 lines
5.5 KiB
C++

// license:BSD-3-Clause
// copyright-holders:hap, Jonathan Gevaryahu
/*
Sharp SM590 MCU core implementation
*/
#include "emu.h"
#include "sm590.h"
#include "debugger.h"
// MCU types
const device_type SM590 = device_creator<sm590_device>;
//const device_type SM591 = device_creator<sm591_device>;
//const device_type SM595 = device_creator<sm595_device>;
// internal memory maps
static ADDRESS_MAP_START(program_1x128x4, AS_PROGRAM, 8, sm510_base_device)
AM_RANGE(0x000, 0x1ff) AM_ROM
ADDRESS_MAP_END
/*static ADDRESS_MAP_START(program_2x128x4, AS_PROGRAM, 8, sm510_base_device)
AM_RANGE(0x000, 0x3ff) AM_ROM
ADDRESS_MAP_END
static ADDRESS_MAP_START(program_1x128x4_1x128x2, AS_PROGRAM, 8, sm510_base_device)
AM_RANGE(0x000, 0x2ff) AM_ROM
ADDRESS_MAP_END*/
static ADDRESS_MAP_START(data_16x2x4, AS_DATA, 8, sm510_base_device)
AM_RANGE(0x00, 0x0f) AM_RAM
AM_RANGE(0x10, 0x1f) AM_RAM
ADDRESS_MAP_END
/*
static ADDRESS_MAP_START(data_16x3.5x4, AS_DATA, 8, sm510_base_device)
AM_RANGE(0x00, 0x0f) AM_RAM
AM_RANGE(0x10, 0x1f) AM_RAM
AM_RANGE(0x20, 0x2f) AM_RAM
AM_RANGE(0x30, 0x37) AM_RAM
ADDRESS_MAP_END
*/
// device definitions
sm590_device::sm590_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock)
: sm510_base_device(mconfig, SM590, "SM590", tag, owner, clock, 4 /* stack levels */, 10 /* prg width */, ADDRESS_MAP_NAME(program_1x128x4), 5 /* data width */, ADDRESS_MAP_NAME(data_16x2x4), "sm590", __FILE__)
{ }
//sm590_device::sm591_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock)
// : sm510_base_device(mconfig, SM591, "SM591", tag, owner, clock, 4 /* stack levels */, 10 /* prg width */, ADDRESS_MAP_NAME(program_2x128x4), 6 /* data width */, ADDRESS_MAP_NAME(data_16x3.5x4), "sm591", __FILE__)
//{ }
//sm590_device::sm595_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock)
// : sm510_base_device(mconfig, SM595, "SM595", tag, owner, clock, 4 /* stack levels */, 10 /* prg width */, ADDRESS_MAP_NAME(program_1x128x4_1x128x2), 5 /* data width */, ADDRESS_MAP_NAME(data_16x2x4), "sm595", __FILE__)
//{ }
sm590_device::sm590_device(const machine_config &mconfig, device_type type, const char *name, const char *tag, device_t *owner, u32 clock, int stack_levels, int prgwidth, address_map_constructor program, int datawidth, address_map_constructor data, const char *shortname, const char *source)
: sm510_base_device(mconfig, type, name, tag, owner, clock, stack_levels, prgwidth, program, datawidth, data, shortname, source)
{ }
// disasm
offs_t sm590_device::disasm_disassemble(std::ostream &stream, offs_t pc, const u8 *oprom, const u8 *opram, u32 options)
{
extern CPU_DISASSEMBLE(sm590);
return CPU_DISASSEMBLE_NAME(sm590)(this, stream, pc, oprom, opram, options);
}
//-------------------------------------------------
// device_reset - device-specific reset
//-------------------------------------------------
void sm590_device::device_reset()
{
// ACL
m_skip = false;
m_halt = false;
m_sbm = false;
m_op = m_prev_op = 0;
do_branch(0, 0, 0);
m_prev_pc = m_pc;
m_rports[0] = m_rports[1] = m_rports[2] = m_rports[3] = 0;
//m_write_r(0, 0, 0xff);
}
//-------------------------------------------------
// execute
//-------------------------------------------------
void sm590_device::increment_pc()
{
// PL(program counter low 7 bits) is a simple LFSR: newbit = (bit0==bit1)
// PU,PM(high bits) specify page, PL specifies steps within page
int feed = ((m_pc >> 1 ^ m_pc) & 1) ? 0 : 0x40;
m_pc = feed | (m_pc >> 1 & 0x3f) | (m_pc & ~0x7f);
}
void sm590_device::get_opcode_param()
{
// TL, TLS(TML) opcodes are 2 bytes
if ((m_op & 0xf8) == 0x78)
{
m_icount--;
m_param = m_program->read_byte(m_pc);
increment_pc();
}
}
void sm590_device::execute_one()
{
switch (m_op & 0xf0) // opcodes with 4 bit params
{
case 0x00: op_adx(); break;
case 0x10: op_tax(); break;
case 0x20: op_lblx(); break;
case 0x30: op_lax(); break;
case 0x80: case 0x90: case 0xa0: case 0xb0:
case 0xc0: case 0xd0: case 0xe0: case 0xf0:
op_t(); break; // aka tr
default: // opcodes with 2 bit params
switch (m_op & 0xfc)
{
case 0x60: op_tmi(); break; // aka tm
case 0x64: op_tba(); break;
case 0x68: op_rm(); break;
case 0x6c: op_sm(); break;
case 0x74: op_lbmx(); break;
case 0x78: op_tl(); break;
case 0x7c: op_tml(); break; // aka tls
default: // everything else
switch (m_op)
{
case 0x40: op_lda(); break;
case 0x41: op_exc(); break;
case 0x42: op_exci(); break;
case 0x43: op_excd(); break;
case 0x44: op_coma(); break;
case 0x45: op_tam(); break;
case 0x46: op_atr(); break;
case 0x47: op_mtr(); break;
case 0x48: op_rc(); break;
case 0x49: op_sc(); break;
case 0x4a: op_str(); break;
case 0x4b: op_cend(); break; // aka cctrl
case 0x4c: op_rtn0(); break; // aka rtn
case 0x4d: op_rtn1(); break; // aka rtns
// 4e, 4f illegal
case 0x50: op_inbm(); break;
case 0x51: op_debm(); break;
case 0x52: op_incb(); break; // aka inbl
case 0x53: op_decb(); break; // aka debl
case 0x54: op_tc(); break;
case 0x55: op_rta(); break;
case 0x56: op_blta(); break;
case 0x57: op_exbla(); break; // aka xbla
// 58, 59, 5a, 5b illegal
case 0x5c: op_atx(); break;
case 0x5d: op_exax(); break;
// 5e is illegal???
// 5f is illegal
case 0x70: op_add(); break;
case 0x71: op_ads(); break;
case 0x72: op_adc(); break;
case 0x73: op_add11(); break; // aka adcs
default: op_illegal(); break;
}
break; // 0xff
}
break; // 0xfc
} // big switch
}