mirror of
https://github.com/holub/mame
synced 2025-06-04 11:56:28 +03:00
179 lines
5.5 KiB
C++
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
|
|
}
|