mame/src/devices/cpu/sm510/sm590core.cpp

206 lines
6.1 KiB
C++

// license:BSD-3-Clause
// copyright-holders:hap, Jonathan Gevaryahu
/*
Sharp SM590 MCU core implementation
TODO:
- finish SM590/SM595 emulation (NES/SNES CIC)
http://bitsavers.informatik.uni-stuttgart.de/pdf/sharp/_dataBooks/1990_Sharp_Microcomputers_Data_Book.pdf
pdf page 35/doc page 26 thru pdf page 44/doc page 35
*/
#include "emu.h"
#include "sm590.h"
#include "sm510d.h"
#include "debugger.h"
// MCU types
DEFINE_DEVICE_TYPE(SM590, sm590_device, "sm590", "SM590") // 512x8 ROM, 32x4 RAM
//DEFINE_DEVICE_TYPE(SM591, sm591_device, "sm591", "SM591") // 1kx8 ROM, 56x4 RAM
//DEFINE_DEVICE_TYPE(SM595, sm595_device, "sm595", "SM595") // 768x8 ROM, 32x4 RAM
// internal memory maps
ADDRESS_MAP_START(sm590_device::program_1x128x4)
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*/
ADDRESS_MAP_START(sm590_device::data_16x2x4)
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)
: sm590_device(mconfig, SM590, tag, owner, clock, 4 /* stack levels */, 9 /* prg width */, address_map_constructor(FUNC(sm590_device::program_1x128x4), this), 5 /* data width */, address_map_constructor(FUNC(sm590_device::data_16x2x4), this))
{
}
//sm591_device::sm591_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock)
// : sm510_base_device(mconfig, SM591, tag, owner, clock, 4 /* stack levels */, 10 /* prg width */, address_map_constructor(FUNC(sm591_device::program_2x128x4), this), 6 /* data width */, address_map_constructor(FUNC(sm591_device::data_16x3.5x4), this))
//{
//}
//sm595_device::sm595_device(const machine_config &mconfig, const char *tag, device_t *owner, u32 clock)
// : sm510_base_device(mconfig, SM595, tag, owner, clock, 4 /* stack levels */, 10 /* prg width */, address_map_constructor(FUNC(sm595_device::program_1x128x4_1x128x2), this), 5 /* data width */, address_map_constructor(FUNC(sm595_device::data_16x2x4), this))
//{
//}
sm590_device::sm590_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, u32 clock, int stack_levels, int prgwidth, address_map_constructor program, int datawidth, address_map_constructor data)
: sm510_base_device(mconfig, type, tag, owner, clock, stack_levels, prgwidth, program, datawidth, data)
{
}
util::disasm_interface *sm590_device::create_disassembler()
{
return new sm590_disassembler;
}
//-------------------------------------------------
// device_reset - device-specific reset
//-------------------------------------------------
void sm590_device::device_reset()
{
// ACL
m_skip = false;
m_halt = false;
m_sbm = false; // needed?
m_op = m_prev_op = 0;
reset_vector();
m_prev_pc = m_pc;
m_clk_div = 4; // 4 clock oscillations per cycle on SM59x, see datasheet page 30/pdf page 39
m_rports[0] = m_rports[1] = m_rports[2] = m_rports[3] = 0;
//m_write_r(0, 0, 0xff); // TODO: are the four ports zeroed on reset?
}
//-------------------------------------------------
// wake from suspend mode
//-------------------------------------------------
bool sm590_device::wake_me_up()
{
// in halt mode, wake up after R2.2 goes high
if (m_rports[2]&0x4)
{
m_halt = false;
do_branch(0, 1, 0); // field 0, page 1, step 0
standard_irq_callback(0);
return true;
}
else
return false;
}
//-------------------------------------------------
// 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
}