mirror of
https://github.com/holub/mame
synced 2025-04-26 18:23:08 +03:00
2544 lines
76 KiB
C++
2544 lines
76 KiB
C++
// license:BSD-3-Clause
|
|
// copyright-holders:Steve Ellenoff, Manuel Abadia, Couriersud
|
|
/*****************************************************************************
|
|
*
|
|
* i8051.c
|
|
* Portable MCS-51 Family Emulator
|
|
*
|
|
* Chips in the family:
|
|
* 8051 Product Line (8031,8051,8751)
|
|
* 8052 Product Line (8032,8052,8752)
|
|
* 8054 Product Line (8054)
|
|
* 8058 Product Line (8058)
|
|
*
|
|
* Copyright Steve Ellenoff, all rights reserved.
|
|
*
|
|
* This work is based on:
|
|
* #1) 'Intel(tm) MC51 Microcontroller Family Users Manual' and
|
|
* #2) 8051 simulator by Travis Marlatte
|
|
* #3) Portable UPI-41/8041/8741/8042/8742 emulator V0.1 by Juergen Buchmueller (MAME CORE)
|
|
*
|
|
*****************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
* DS5002FP emulator by Manuel Abadia
|
|
*
|
|
* October 2008, couriersud: Merged back in mcs51
|
|
*
|
|
* What has been added?
|
|
* - Extra SFRs
|
|
* - Bytewide Bus Support
|
|
* - Memory Partition and Memory Range
|
|
* - Bootstrap Configuration
|
|
* - Power Fail Interrupt
|
|
* - Timed Access
|
|
* - Stop Mode
|
|
* - Idle Mode
|
|
*
|
|
* What is not implemented?
|
|
* - Peripherals and Reprogrammable Peripheral Controller
|
|
* - CRC-16
|
|
* - Watchdog timer
|
|
*
|
|
* The main features of the DS5002FP are:
|
|
* - 100% code-compatible with 8051
|
|
* - Directly addresses 64kB program/64kB data memory
|
|
* - Nonvolatile memory control circuitry
|
|
* - 10-year data retention in the absence of power
|
|
* - In-system reprogramming via serial port
|
|
* - Dedicated memory bus, preserving four 8-bit ports for general purpose I/O
|
|
* - Power-fail reset
|
|
* - Early warning power-fail interrupt
|
|
* - Watchdog timer
|
|
* - Accesses up to 128kB on the bytewide bus
|
|
* - Decodes memory for 32kB x 8 or 128kB x 8 SRAMs
|
|
* - Four additional decoded peripheral-chip enables
|
|
* - CRC hardware for checking memory validity
|
|
* - Optionally emulates an 8042-style slave interface
|
|
* - Memory encryption using an 80-bit encryption key
|
|
* - Automatic random generation of encryption keys
|
|
* - Self-destruct input for tamper protection
|
|
* - Optional top-coating prevents microprobe
|
|
*
|
|
*****************************************************************************/
|
|
|
|
/******************************************************************************
|
|
* Notes:
|
|
*
|
|
* The term cycles is used here to really refer to clock oscilations, because 1 machine cycle
|
|
* actually takes 12 oscilations.
|
|
*
|
|
* Read/Write/Modify Instruction -
|
|
* Data is read from the Port Latch (not the Port Pin!), possibly modified, and
|
|
* written back to (the pin? and) the latch!
|
|
*
|
|
* The following all perform this on a port address..
|
|
* (anl, orl, xrl, jbc, cpl, inc, dec, djnz, mov px.y,c, clr px.y, setb px.y)
|
|
*
|
|
* Serial UART emulation is not really accurate, but faked enough to work as far as i can tell
|
|
*
|
|
* August 27,2003: Currently support for only 8031/8051/8751 chips (ie 128 RAM)
|
|
* October 14,2003: Added initial support for the 8752 (ie 256 RAM)
|
|
* October 22,2003: Full support for the 8752 (ie 256 RAM)
|
|
* July 28,2004: Fixed MOVX command and added External Ram Paging Support
|
|
* July 31,2004: Added Serial Mode 0 Support & Fixed Interrupt Flags for Serial Port
|
|
*
|
|
* October, 2008, Couriersud - Major rewrite
|
|
*
|
|
*****************************************************************************/
|
|
|
|
/* TODO: Varios
|
|
* - EA pin - defined by architecture, must implement:
|
|
* 1 means external access, bypassing internal ROM
|
|
* - T0 output clock ?
|
|
*
|
|
* - Implement 80C52 extended serial capabilities
|
|
* - Fix serial communication - This is a big hack (but working) right now.
|
|
* - Implement 83C751 in sslam.c
|
|
* - Fix cardline.c
|
|
* most likely due to different behaviour of I/O pins. The boards
|
|
* actually use 80CXX, i.e. CMOS versions.
|
|
* "Normal" 805X will return a 0 if reading from a output port which has
|
|
* a 0 written to it's latch. At least cardline expects a 1 here.
|
|
*
|
|
* Done: (Couriersud)
|
|
* - Merged DS5002FP
|
|
* - Disassembler now uses type specific memory names
|
|
* - Merged DS5002FP disasm
|
|
* - added 83C751 memory names to disassembler
|
|
* - Pointer-ified
|
|
* - Implemented cmos features
|
|
* - Implemented 80C52 interrupt handling
|
|
* - Fix segas18.c (segaic16.c) memory handling.
|
|
* - Fix sslam.c
|
|
* - Fix limenko.c videopkr.c : Issue with core allocation of ram (duplicate savestate)
|
|
* - Handle internal ram better (debugger visible)
|
|
* - Fixed port reading
|
|
* - Rewrote Macros for better readability
|
|
* - Fixed and rewrote Interrupt handling
|
|
* - Now returns INTERNAL_DIVIDER, adjusted cycle counts
|
|
* - Remove unnecessary and duplicated code
|
|
* - Remove unnecessary functions
|
|
* - Rewrite to have sfr-registers stored in int_ram.
|
|
* - Debugger may now watch sfr-registers as well.
|
|
* - implemented interrupt callbacks (HOLD_LINE now supported)
|
|
* - Runtime switch for processor type - remove ifdefs
|
|
* - internal memory maps for internal rom versions (internal ram now displayed in debugger)
|
|
* - more timer cleanups from manual
|
|
*/
|
|
|
|
#include "emu.h"
|
|
#include "debugger.h"
|
|
#include "mcs51.h"
|
|
#include "mcs51dasm.h"
|
|
|
|
#define VERBOSE 0
|
|
|
|
#define LOG(x) do { if (VERBOSE) logerror x; } while (0)
|
|
|
|
/***************************************************************************
|
|
CONSTANTS
|
|
***************************************************************************/
|
|
|
|
enum
|
|
{
|
|
FEATURE_NONE = 0x00,
|
|
FEATURE_I8052 = 0x01,
|
|
FEATURE_CMOS = 0x02,
|
|
FEATURE_I80C52 = 0x04,
|
|
FEATURE_DS5002FP = 0x08
|
|
};
|
|
|
|
/* Internal address in SFR of registers */
|
|
enum
|
|
{
|
|
ADDR_PSW = 0xd0,
|
|
ADDR_ACC = 0xe0,
|
|
ADDR_B = 0xf0,
|
|
|
|
ADDR_P0 = 0x80,
|
|
ADDR_SP = 0x81,
|
|
ADDR_DPL = 0x82,
|
|
ADDR_DPH = 0x83,
|
|
ADDR_PCON = 0x87,
|
|
ADDR_TCON = 0x88,
|
|
ADDR_TMOD = 0x89,
|
|
ADDR_TL0 = 0x8a,
|
|
ADDR_TL1 = 0x8b,
|
|
ADDR_TH0 = 0x8c,
|
|
ADDR_TH1 = 0x8d,
|
|
ADDR_P1 = 0x90,
|
|
ADDR_SCON = 0x98,
|
|
ADDR_SBUF = 0x99,
|
|
ADDR_P2 = 0xa0,
|
|
ADDR_IE = 0xa8,
|
|
ADDR_P3 = 0xb0,
|
|
ADDR_IP = 0xb8,
|
|
|
|
/* 8052 Only registers */
|
|
ADDR_T2CON = 0xc8,
|
|
ADDR_RCAP2L = 0xca,
|
|
ADDR_RCAP2H = 0xcb,
|
|
ADDR_TL2 = 0xcc,
|
|
ADDR_TH2 = 0xcd,
|
|
|
|
/* 80C52 Only registers */
|
|
ADDR_IPH = 0xb7,
|
|
ADDR_SADDR = 0xa9,
|
|
ADDR_SADEN = 0xb9,
|
|
|
|
/* Philips 80C52 */
|
|
ADDR_AUXR = 0x8e,
|
|
ADDR_AUXR1 = 0xa2,
|
|
|
|
/* DS5002FP */
|
|
ADDR_CRCR = 0xc1,
|
|
ADDR_CRCL = 0xc2,
|
|
ADDR_CRCH = 0xc3,
|
|
ADDR_MCON = 0xc6,
|
|
ADDR_TA = 0xc7,
|
|
ADDR_RNR = 0xcf,
|
|
ADDR_RPCTL = 0xd8,
|
|
ADDR_RPS = 0xda
|
|
|
|
};
|
|
|
|
/* PC vectors */
|
|
|
|
enum
|
|
{
|
|
V_RESET = 0x000, /* power on address */
|
|
V_IE0 = 0x003, /* External Interrupt 0 */
|
|
V_TF0 = 0x00b, /* Timer 0 Overflow */
|
|
V_IE1 = 0x013, /* External Interrupt 1 */
|
|
V_TF1 = 0x01b, /* Timer 1 Overflow */
|
|
V_RITI = 0x023, /* Serial Receive/Transmit */
|
|
|
|
/* 8052 Only Vectors */
|
|
V_TF2 = 0x02b, /* Timer 2 Overflow */
|
|
|
|
/* DS5002FP */
|
|
V_PFI = 0x02b /* Power Failure Interrupt */
|
|
};
|
|
|
|
|
|
DEFINE_DEVICE_TYPE(I8031, i8031_device, "i8031", "I8031")
|
|
DEFINE_DEVICE_TYPE(I8032, i8032_device, "i8032", "I8032")
|
|
DEFINE_DEVICE_TYPE(I8051, i8051_device, "i8051", "I8051")
|
|
DEFINE_DEVICE_TYPE(I8751, i8751_device, "i8751", "I8751")
|
|
DEFINE_DEVICE_TYPE(I8052, i8052_device, "i8052", "I8052")
|
|
DEFINE_DEVICE_TYPE(I8752, i8752_device, "i8752", "I8752")
|
|
DEFINE_DEVICE_TYPE(I80C31, i80c31_device, "i80c31", "I80C31")
|
|
DEFINE_DEVICE_TYPE(I80C51, i80c51_device, "i80c51", "I80C51")
|
|
DEFINE_DEVICE_TYPE(I87C51, i87c51_device, "i87c51", "I87C51")
|
|
DEFINE_DEVICE_TYPE(I80C32, i80c32_device, "i80c32", "I80C32")
|
|
DEFINE_DEVICE_TYPE(I80C52, i80c52_device, "i80c52", "I80C52")
|
|
DEFINE_DEVICE_TYPE(I87C52, i87c52_device, "i87c52", "I87C52")
|
|
DEFINE_DEVICE_TYPE(AT89C4051, at89c4051_device, "at89c4051", "AT89C4051")
|
|
DEFINE_DEVICE_TYPE(DS5002FP, ds5002fp_device, "ds5002fp", "DS5002FP")
|
|
|
|
|
|
/***************************************************************************
|
|
ADDRESS MAPS
|
|
***************************************************************************/
|
|
|
|
static ADDRESS_MAP_START(program_12bit, AS_PROGRAM, 8, mcs51_cpu_device)
|
|
AM_RANGE(0x00, 0x0fff) AM_ROM
|
|
ADDRESS_MAP_END
|
|
|
|
static ADDRESS_MAP_START(program_13bit, AS_PROGRAM, 8, mcs51_cpu_device)
|
|
AM_RANGE(0x00, 0x1fff) AM_ROM
|
|
ADDRESS_MAP_END
|
|
|
|
static ADDRESS_MAP_START(data_7bit, AS_DATA, 8, mcs51_cpu_device)
|
|
AM_RANGE(0x0000, 0x007f) AM_RAM AM_SHARE("scratchpad")
|
|
AM_RANGE(0x0100, 0x01ff) AM_RAM AM_SHARE("sfr_ram") /* SFR */
|
|
ADDRESS_MAP_END
|
|
|
|
static ADDRESS_MAP_START(data_8bit, AS_DATA, 8, mcs51_cpu_device)
|
|
AM_RANGE(0x0000, 0x00ff) AM_RAM AM_SHARE("scratchpad")
|
|
AM_RANGE(0x0100, 0x01ff) AM_RAM AM_SHARE("sfr_ram") /* SFR */
|
|
ADDRESS_MAP_END
|
|
|
|
|
|
|
|
mcs51_cpu_device::mcs51_cpu_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, uint32_t clock, int program_width, int data_width, uint8_t features)
|
|
: cpu_device(mconfig, type, tag, owner, clock)
|
|
, m_program_config("program", ENDIANNESS_LITTLE, 8, 16, 0,
|
|
(program_width == 12) ? ADDRESS_MAP_NAME(program_12bit) : (program_width == 13) ? ADDRESS_MAP_NAME(program_13bit) : nullptr)
|
|
, m_data_config("data", ENDIANNESS_LITTLE, 8, 9, 0,
|
|
(data_width == 7) ? ADDRESS_MAP_NAME(data_7bit) : (data_width == 8) ? ADDRESS_MAP_NAME(data_8bit) : nullptr )
|
|
, m_io_config("io", ENDIANNESS_LITTLE, 8, 18, 0)
|
|
, m_pc(0)
|
|
, m_features(features)
|
|
, m_ram_mask( (data_width == 8) ? 0xFF : 0x7F )
|
|
, m_num_interrupts(5)
|
|
, m_sfr_ram(*this, "sfr_ram")
|
|
, m_scratchpad(*this, "scratchpad")
|
|
, m_serial_tx_cb(*this)
|
|
, m_serial_rx_cb(*this)
|
|
, m_rtemp(0)
|
|
{
|
|
m_ds5002fp.mcon = 0;
|
|
m_ds5002fp.rpctl = 0;
|
|
m_ds5002fp.crc = 0;
|
|
|
|
/* default to standard cmos interfacing */
|
|
|
|
for (auto & elem : m_forced_inputs)
|
|
elem = 0;
|
|
}
|
|
|
|
|
|
i8031_device::i8031_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: mcs51_cpu_device(mconfig, I8031, tag, owner, clock, 0, 7)
|
|
{
|
|
}
|
|
|
|
i8051_device::i8051_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: mcs51_cpu_device(mconfig, I8051, tag, owner, clock, 12, 7)
|
|
{
|
|
}
|
|
|
|
i8751_device::i8751_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: mcs51_cpu_device(mconfig, I8751, tag, owner, clock, 12, 7)
|
|
{
|
|
}
|
|
|
|
i8052_device::i8052_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, uint32_t clock, int program_width, int data_width, uint8_t features)
|
|
: mcs51_cpu_device(mconfig, type, tag, owner, clock, program_width, data_width, features | FEATURE_I8052)
|
|
{
|
|
m_num_interrupts = 6;
|
|
}
|
|
|
|
i8052_device::i8052_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i8052_device(mconfig, I8052, tag, owner, clock, 13, 8)
|
|
{
|
|
}
|
|
|
|
i8032_device::i8032_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i8052_device(mconfig, I8032, tag, owner, clock, 0, 8)
|
|
{
|
|
}
|
|
|
|
i8752_device::i8752_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i8052_device(mconfig, I8752, tag, owner, clock, 13, 8)
|
|
{
|
|
}
|
|
|
|
i80c31_device::i80c31_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i8052_device(mconfig, I80C31, tag, owner, clock, 0, 7)
|
|
{
|
|
}
|
|
|
|
i80c51_device::i80c51_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, uint32_t clock, int program_width, int data_width, uint8_t features)
|
|
: mcs51_cpu_device(mconfig, type, tag, owner, clock, program_width, data_width, features | FEATURE_CMOS)
|
|
{
|
|
}
|
|
|
|
i80c51_device::i80c51_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i80c51_device(mconfig, I80C51, tag, owner, clock, 12, 7)
|
|
{
|
|
}
|
|
|
|
i87c51_device::i87c51_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i80c51_device(mconfig, I87C51, tag, owner, clock, 12, 7)
|
|
{
|
|
}
|
|
|
|
|
|
i80c52_device::i80c52_device(const machine_config &mconfig, device_type type, const char *tag, device_t *owner, uint32_t clock, int program_width, int data_width, uint8_t features)
|
|
: i8052_device(mconfig, type, tag, owner, clock, program_width, data_width, features | FEATURE_I80C52 | FEATURE_CMOS)
|
|
{
|
|
}
|
|
|
|
i80c52_device::i80c52_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i80c52_device(mconfig, I80C52, tag, owner, clock, 13, 8)
|
|
{
|
|
}
|
|
|
|
i80c32_device::i80c32_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i80c52_device(mconfig, I80C32, tag, owner, clock, 0, 8)
|
|
{
|
|
}
|
|
|
|
|
|
i87c52_device::i87c52_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i80c52_device(mconfig, I87C52, tag, owner, clock, 13, 8)
|
|
{
|
|
}
|
|
|
|
at89c4051_device::at89c4051_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: i80c51_device(mconfig, AT89C4051, tag, owner, clock, 12, 7)
|
|
{
|
|
}
|
|
|
|
/* program width field is set to 0 because technically the SRAM isn't internal */
|
|
ds5002fp_device::ds5002fp_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock)
|
|
: mcs51_cpu_device(mconfig, DS5002FP, tag, owner, clock, 0, 7, FEATURE_DS5002FP | FEATURE_CMOS)
|
|
, device_nvram_interface(mconfig, *this)
|
|
, m_region(*this, "internal")
|
|
{
|
|
}
|
|
|
|
device_memory_interface::space_config_vector mcs51_cpu_device::memory_space_config() const
|
|
{
|
|
return space_config_vector {
|
|
std::make_pair(AS_PROGRAM, &m_program_config),
|
|
std::make_pair(AS_DATA, &m_data_config),
|
|
std::make_pair(AS_IO, &m_io_config)
|
|
};
|
|
}
|
|
|
|
|
|
/***************************************************************************
|
|
MACROS
|
|
***************************************************************************/
|
|
|
|
/* Read Opcode/Opcode Arguments from Program Code */
|
|
#define ROP(pc) m_direct->read_byte(pc)
|
|
#define ROP_ARG(pc) m_direct->read_byte(pc)
|
|
|
|
/* Read a byte from External Code Memory (Usually Program Rom(s) Space) */
|
|
#define CODEMEM_R(a) (uint8_t)m_program->read_byte(a)
|
|
|
|
/* Read/Write a byte from/to External Data Memory (Usually RAM or other I/O) */
|
|
#define DATAMEM_R(a) (uint8_t)m_io->read_byte(a)
|
|
#define DATAMEM_W(a,v) m_io->write_byte(a, v)
|
|
|
|
/* Read/Write a byte from/to the Internal RAM */
|
|
|
|
#define IRAM_R(a) iram_read(a)
|
|
#define IRAM_W(a, d) iram_write(a, d)
|
|
|
|
/* Read/Write a byte from/to the Internal RAM indirectly */
|
|
/* (called from indirect addressing) */
|
|
uint8_t mcs51_cpu_device::iram_iread(offs_t a) { return (a <= m_ram_mask) ? m_data->read_byte(a) : 0xff; }
|
|
void mcs51_cpu_device::iram_iwrite(offs_t a, uint8_t d) { if (a <= m_ram_mask) m_data->write_byte(a, d); }
|
|
|
|
#define IRAM_IR(a) iram_iread(a)
|
|
#define IRAM_IW(a, d) iram_iwrite(a, d)
|
|
|
|
/* Form an Address to Read/Write to External RAM indirectly */
|
|
/* (called from indirect addressing) */
|
|
#define ERAM_ADDR(a,m) external_ram_iaddr(a,m)
|
|
|
|
/* Read/Write a bit from Bit Addressable Memory */
|
|
#define BIT_R(a) bit_address_r(a)
|
|
#define BIT_W(a,v) bit_address_w(a, v)
|
|
|
|
/* Input/Output a byte from given I/O port */
|
|
#define IN(port) ((uint8_t)m_io->read_byte(port))
|
|
#define OUT(port,value) m_io->write_byte(port,value)
|
|
|
|
|
|
/***************************************************************************
|
|
SHORTCUTS
|
|
***************************************************************************/
|
|
|
|
#define PPC m_ppc
|
|
#define PC m_pc
|
|
#define RWM m_rwm
|
|
|
|
/* SFR Registers - These are accessed directly for speed on read */
|
|
/* Read accessors */
|
|
|
|
#define SFR_A(a) m_sfr_ram[(a)]
|
|
#define SET_SFR_A(a,v) do { SFR_A(a) = (v); } while (0)
|
|
|
|
#define ACC SFR_A(ADDR_ACC)
|
|
#define PSW SFR_A(ADDR_PSW)
|
|
|
|
#define P0 ((const uint8_t) SFR_A(ADDR_P0))
|
|
#define P1 ((const uint8_t) SFR_A(ADDR_P1))
|
|
#define P2 ((const uint8_t) SFR_A(ADDR_P2))
|
|
#define P3 ((const uint8_t) SFR_A(ADDR_P3))
|
|
|
|
#define SP SFR_A(ADDR_SP)
|
|
#define DPL SFR_A(ADDR_DPL)
|
|
#define DPH SFR_A(ADDR_DPH)
|
|
#define PCON SFR_A(ADDR_PCON)
|
|
#define TCON SFR_A(ADDR_TCON)
|
|
#define TMOD SFR_A(ADDR_TMOD)
|
|
#define TL0 SFR_A(ADDR_TL0)
|
|
#define TL1 SFR_A(ADDR_TL1)
|
|
#define TH0 SFR_A(ADDR_TH0)
|
|
#define TH1 SFR_A(ADDR_TH1)
|
|
#define SCON SFR_A(ADDR_SCON)
|
|
#define IE SFR_A(ADDR_IE)
|
|
#define IP SFR_A(ADDR_IP)
|
|
#define B SFR_A(ADDR_B)
|
|
#define SBUF SFR_A(ADDR_SBUF)
|
|
|
|
#define R_REG(r) m_scratchpad[(r) | (PSW & 0x18)]
|
|
#define DPTR ((DPH<<8) | DPL)
|
|
|
|
/* 8052 Only registers */
|
|
#define T2CON SFR_A(ADDR_T2CON)
|
|
#define RCAP2L SFR_A(ADDR_RCAP2L)
|
|
#define RCAP2H SFR_A(ADDR_RCAP2H)
|
|
#define TL2 SFR_A(ADDR_TL2)
|
|
#define TH2 SFR_A(ADDR_TH2)
|
|
|
|
/* 80C52 Only registers */
|
|
#define IPH SFR_A(ADDR_IPH)
|
|
#define SADDR SFR_A(ADDR_SADDR)
|
|
#define SADEN SFR_A(ADDR_SADEN)
|
|
|
|
/* Philips 80C52 */
|
|
/* ============= */
|
|
/* Reduced EMI Mode
|
|
* The AO bit (AUXR.0) in the AUXR register when set disables the
|
|
* ALE output.
|
|
*/
|
|
#define AUXR SFR_A(ADDR_AUXR)
|
|
|
|
/* The dual DPTR structure (see Figure 12) is a way by which the
|
|
* 80C52/54/58 will specify the address of an external data memory
|
|
* location. There are two 16-bit DPTR registers that address the
|
|
* external memory, and a single bit called DPS = AUXR1/bit0 that
|
|
* allows the program code to switch between them.
|
|
*/
|
|
#define AUXR1 SFR_A(ADDR_AUXR1)
|
|
|
|
/* DS5002FP only registers */
|
|
#define CRCR SFR_A(ADDR_CRCR)
|
|
#define CRCL SFR_A(ADDR_CRCL)
|
|
#define CRCH SFR_A(ADDR_CRCH)
|
|
#define MCON SFR_A(ADDR_MCON)
|
|
#define TA SFR_A(ADDR_TA)
|
|
#define RNR SFR_A(ADDR_RNR)
|
|
#define RPCTL SFR_A(ADDR_RPCTL)
|
|
#define RPS SFR_A(ADDR_RPS)
|
|
|
|
|
|
/* WRITE accessors */
|
|
|
|
/* Shortcuts */
|
|
|
|
#define SET_PSW(v) do { SFR_A(ADDR_PSW) = (v); SET_PARITY(); } while (0)
|
|
#define SET_ACC(v) do { SFR_A(ADDR_ACC) = (v); SET_PARITY(); } while (0)
|
|
|
|
/* These trigger actions on modification and have to be written through SFR_W */
|
|
#define SET_P0(v) IRAM_W(ADDR_P0, v)
|
|
#define SET_P1(v) IRAM_W(ADDR_P1, v)
|
|
#define SET_P2(v) IRAM_W(ADDR_P2, v)
|
|
#define SET_P3(v) IRAM_W(ADDR_P3, v)
|
|
|
|
/* Within the cpu core, do not trigger a send */
|
|
#define SET_SBUF(v) SET_SFR_A(ADDR_SBUF, v)
|
|
|
|
/* No actions triggered on write */
|
|
#define SET_REG(r, v) do { m_scratchpad[(r) | (PSW & 0x18)] = (v); } while (0)
|
|
|
|
#define SET_DPTR(n) do { DPH = ((n) >> 8) & 0xff; DPL = (n) & 0xff; } while (0)
|
|
|
|
/* Macros for Setting Flags */
|
|
#define SET_X(R, v) do { R = (v);} while (0)
|
|
|
|
#define SET_CY(n) SET_PSW((PSW & 0x7f) | (n<<7)) //Carry Flag
|
|
#define SET_AC(n) SET_PSW((PSW & 0xbf) | (n<<6)) //Aux.Carry Flag
|
|
#define SET_FO(n) SET_PSW((PSW & 0xdf) | (n<<5)) //User Flag
|
|
#define SET_RS(n) SET_PSW((PSW & 0xe7) | (n<<3)) //R Bank Select
|
|
#define SET_OV(n) SET_PSW((PSW & 0xfb) | (n<<2)) //Overflow Flag
|
|
#define SET_P(n) SET_PSW((PSW & 0xfe) | (n<<0)) //Parity Flag
|
|
|
|
#define SET_BIT(R, n, v) do { R = (R & ~(1<<(n))) | ((v) << (n));} while (0)
|
|
#define GET_BIT(R, n) (((R)>>(n)) & 0x01)
|
|
|
|
#define SET_EA(n) SET_BIT(IE, 7, n) //Global Interrupt Enable/Disable
|
|
#define SET_ES(n) SET_BIT(IE, 4, v) //Serial Interrupt Enable/Disable
|
|
#define SET_ET1(n) SET_BIT(IE, 3, n) //Timer 1 Interrupt Enable/Disable
|
|
#define SET_EX1(n) SET_BIT(IE, 2, n) //External Int 1 Interrupt Enable/Disable
|
|
#define SET_ET0(n) SET_BIT(IE, 1, n) //Timer 0 Interrupt Enable/Disable
|
|
#define SET_EX0(n) SET_BIT(IE, 0, n) //External Int 0 Interrupt Enable/Disable
|
|
/* 8052 Only flags */
|
|
#define SET_ET2(n) SET_BIT(IE, 5, n) //Timer 2 Interrupt Enable/Disable
|
|
|
|
/* 8052 Only flags */
|
|
#define SET_PT2(n) SET_BIT(IP, 5, n); //Set Timer 2 Priority Level
|
|
|
|
#define SET_PS0(n) SET_BIT(IP, 4, n) //Set Serial Priority Level
|
|
#define SET_PT1(n) SET_BIT(IP, 3, n) //Set Timer 1 Priority Level
|
|
#define SET_PX1(n) SET_BIT(IP, 2, n) //Set External Int 1 Priority Level
|
|
#define SET_PT0(n) SET_BIT(IP, 1, n) //Set Timer 0 Priority Level
|
|
#define SET_PX0(n) SET_BIT(IP, 0, n) //Set External Int 0 Priority Level
|
|
|
|
#define SET_TF1(n) SET_BIT(TCON, 7, n) //Indicated Timer 1 Overflow Int Triggered
|
|
#define SET_TR1(n) SET_BIT(TCON, 6, n) //IndicateS Timer 1 is running
|
|
#define SET_TF0(n) SET_BIT(TCON, 5, n) //Indicated Timer 0 Overflow Int Triggered
|
|
#define SET_TR0(n) SET_BIT(TCON, 4, n) //IndicateS Timer 0 is running
|
|
#define SET_IE1(n) SET_BIT(TCON, 3, n) //Indicated External Int 1 Triggered
|
|
#define SET_IT1(n) SET_BIT(TCON, 2, n) //Indicates how External Int 1 is Triggered
|
|
#define SET_IE0(n) SET_BIT(TCON, 1, n) //Indicated External Int 0 Triggered
|
|
#define SET_IT0(n) SET_BIT(TCON, 0, n) //Indicates how External Int 0 is Triggered
|
|
|
|
#define SET_SM0(n) SET_BIT(SCON, 7, n) //Sets Serial Port Mode
|
|
#define SET_SM1(n) SET_BIT(SCON, 6, n) //Sets Serial Port Mode
|
|
#define SET_SM2(n) SET_BIT(SCON, 5, n) //Sets Serial Port Mode (Multiprocesser mode)
|
|
#define SET_REN(n) SET_BIT(SCON, 4, n) //Sets Serial Port Receive Enable
|
|
#define SET_TB8(n) SET_BIT(SCON, 3, n) //Transmit 8th Bit
|
|
#define SET_RB8(n) SET_BIT(SCON, 2, n) //Receive 8th Bit
|
|
#define SET_TI(n) SET_BIT(SCON, 1, n) //Indicates Transmit Interrupt Occurred
|
|
#define SET_RI(n) SET_BIT(SCON, 0, n) //Indicates Receive Interrupt Occurred
|
|
|
|
#define SET_GATE1(n) SET_BIT(TMOD, 7, n) //Timer 1 Gate Mode
|
|
#define SET_CT1(n) SET_BIT(TMOD, 6, n) //Timer 1 Counter Mode
|
|
#define SET_M1_1(n) SET_BIT(TMOD, 5, n) //Timer 1 Timer Mode Bit 1
|
|
#define SET_M1_0(n) SET_BIT(TMOD, 4, n) //Timer 1 Timer Mode Bit 0
|
|
#define SET_GATE0(n) SET_BIT(TMOD, 3, n) //Timer 0 Gate Mode
|
|
#define SET_CT0(n) SET_BIT(TMOD, 2, n) //Timer 0 Counter Mode
|
|
#define SET_M0_1(n) SET_BIT(TMOD, 1, n) //Timer 0 Timer Mode Bit 1
|
|
#define SET_M0_0(n) SET_BIT(TMOD, 0, n) //Timer 0 Timer Mode Bit 0
|
|
|
|
|
|
|
|
/* 8052 Only flags - T2CON Flags */
|
|
#define SET_TF2(n) SET_BIT(T2CON, 7, n) //Indicated Timer 2 Overflow Int Triggered
|
|
#define SET_EXF2(n) SET_BIT(T2CON, 6, n) //Indicates Timer 2 External Flag
|
|
#define SET_RCLK(n) SET_BIT(T2CON, 5, n) //Receive Clock
|
|
#define SET_TCLK(n) SET_BIT(T2CON, 4, n) //Transmit Clock
|
|
#define SET_EXEN2(n) SET_BIT(T2CON, 3, n) //Timer 2 External Interrupt Enable
|
|
#define SET_TR2(n) SET_BIT(T2CON, 2, n) //Indicates Timer 2 is running
|
|
#define SET_CT2(n) SET_BIT(T2CON, 1, n) //Sets Timer 2 Counter/Timer Mode
|
|
#define SET_CP(n) SET_BIT(T2CON, 0, n) //Sets Timer 2 Capture/Reload Mode
|
|
|
|
#define SET_GF1(n) SET_BIT(PCON, 3, n)
|
|
#define SET_GF0(n) SET_BIT(PCON, 2, n)
|
|
#define SET_PD(n) SET_BIT(PCON, 1, n)
|
|
#define SET_IDL(n) SET_BIT(PCON, 0, n)
|
|
|
|
/* Macros for accessing flags */
|
|
|
|
#define GET_CY GET_BIT(PSW, 7)
|
|
#define GET_AC GET_BIT(PSW, 6)
|
|
#define GET_FO GET_BIT(PSW, 5)
|
|
#define GET_RS GET_BIT(PSW, 3)
|
|
#define GET_OV GET_BIT(PSW, 2)
|
|
#define GET_P GET_BIT(PSW, 0)
|
|
|
|
#define GET_EA GET_BIT(IE, 7)
|
|
#define GET_ET2 GET_BIT(IE, 5)
|
|
#define GET_ES GET_BIT(IE, 4)
|
|
#define GET_ET1 GET_BIT(IE, 3)
|
|
#define GET_EX1 GET_BIT(IE, 2)
|
|
#define GET_ET0 GET_BIT(IE, 1)
|
|
#define GET_EX0 GET_BIT(IE, 0)
|
|
|
|
/* 8052 Only flags */
|
|
#define GET_PT2 GET_BIT(IP, 5)
|
|
|
|
#define GET_PS GET_BIT(IP, 4)
|
|
#define GET_PT1 GET_BIT(IP, 3)
|
|
#define GET_PX1 GET_BIT(IP, 2)
|
|
#define GET_PT0 GET_BIT(IP, 1)
|
|
#define GET_PX0 GET_BIT(IP, 0)
|
|
|
|
#define GET_TF1 GET_BIT(TCON, 7)
|
|
#define GET_TR1 GET_BIT(TCON, 6)
|
|
#define GET_TF0 GET_BIT(TCON, 5)
|
|
#define GET_TR0 GET_BIT(TCON, 4)
|
|
#define GET_IE1 GET_BIT(TCON, 3)
|
|
#define GET_IT1 GET_BIT(TCON, 2)
|
|
#define GET_IE0 GET_BIT(TCON, 1)
|
|
#define GET_IT0 GET_BIT(TCON, 0)
|
|
|
|
#define GET_SM0 GET_BIT(SCON, 7)
|
|
#define GET_SM1 GET_BIT(SCON, 6)
|
|
#define GET_SM2 GET_BIT(SCON, 5)
|
|
#define GET_REN GET_BIT(SCON, 4)
|
|
#define GET_TB8 GET_BIT(SCON, 3)
|
|
#define GET_RB8 GET_BIT(SCON, 2)
|
|
#define GET_TI GET_BIT(SCON, 1)
|
|
#define GET_RI GET_BIT(SCON, 0)
|
|
|
|
#define GET_GATE1 GET_BIT(TMOD, 7)
|
|
#define GET_CT1 GET_BIT(TMOD, 6)
|
|
#define GET_M1_1 GET_BIT(TMOD, 5)
|
|
#define GET_M1_0 GET_BIT(TMOD, 4)
|
|
#define GET_GATE0 GET_BIT(TMOD, 3)
|
|
#define GET_CT0 GET_BIT(TMOD, 2)
|
|
#define GET_M0_1 GET_BIT(TMOD, 1)
|
|
#define GET_M0_0 GET_BIT(TMOD, 0)
|
|
|
|
#define GET_SMOD GET_BIT(PCON, 7)
|
|
|
|
/* Only in 80C51BH & other cmos */
|
|
|
|
#define GET_GF1 GET_BIT(PCON, 3)
|
|
#define GET_GF0 GET_BIT(PCON, 2)
|
|
#define GET_PD GET_BIT(PCON, 1)
|
|
#define GET_IDL (GET_BIT(PCON, 0) & ~(GET_PD)) /* PD takes precedence! */
|
|
|
|
/* 8052 Only flags */
|
|
#define GET_TF2 GET_BIT(T2CON, 7)
|
|
#define GET_EXF2 GET_BIT(T2CON, 6)
|
|
#define GET_RCLK GET_BIT(T2CON, 5)
|
|
#define GET_TCLK GET_BIT(T2CON, 4)
|
|
#define GET_EXEN2 GET_BIT(T2CON, 3)
|
|
#define GET_TR2 GET_BIT(T2CON, 2)
|
|
#define GET_CT2 GET_BIT(T2CON, 1)
|
|
#define GET_CP GET_BIT(T2CON, 0)
|
|
|
|
/* DS5002FP Only flags */
|
|
|
|
/* PCON Flags - DS5002FP */
|
|
|
|
#define GET_POR GET_BIT(PCON, 6)
|
|
#define GET_PFW GET_BIT(PCON, 5)
|
|
#define GET_WTR GET_BIT(PCON, 4)
|
|
#define GET_EPFW GET_BIT(PCON, 3)
|
|
#define GET_EWT GET_BIT(PCON, 2)
|
|
|
|
#define SET_PFW(n) SET_BIT(PCON, 5, n)
|
|
|
|
/* MCON Flags - DS5002FP */
|
|
|
|
#define GET_PA ((MCON & 0xf0)>>4)
|
|
#define GET_RG1 GET_BIT(MCON, 3)
|
|
#define GET_PES GET_BIT(MCON, 2)
|
|
#define GET_PM GET_BIT(MCON, 1)
|
|
#define GET_SL GET_BIT(MCON, 0)
|
|
|
|
/* RPCTL Flags - DS5002FP */
|
|
#define GET_RNR GET_BIT(RPCTL, 7) /* Bit 6 ?? */
|
|
#define GET_EXBS GET_BIT(RPCTL, 5)
|
|
#define GET_AE GET_BIT(RPCTL, 4)
|
|
#define GET_IBI GET_BIT(RPCTL, 3)
|
|
#define GET_DMA GET_BIT(RPCTL, 2)
|
|
#define GET_RPCON GET_BIT(RPCTL, 1)
|
|
#define GET_RG0 GET_BIT(RPCTL, 0)
|
|
|
|
|
|
/*Add and Subtract Flag settings*/
|
|
#define DO_ADD_FLAGS(a,d,c) do_add_flags(a, d, c)
|
|
#define DO_SUB_FLAGS(a,d,c) do_sub_flags(a, d, c)
|
|
|
|
#define SET_PARITY() do {m_recalc_parity |= 1;} while (0)
|
|
#define PUSH_PC() push_pc()
|
|
#define POP_PC() pop_pc()
|
|
|
|
/* Clear Current IRQ */
|
|
#define CLEAR_CURRENT_IRQ() clear_current_irq()
|
|
|
|
|
|
/* Hold callback functions so they can be set by caller (before the cpu reset) */
|
|
|
|
/***************************************************************************
|
|
INLINE FUNCTIONS
|
|
***************************************************************************/
|
|
|
|
void mcs51_cpu_device::clear_current_irq()
|
|
{
|
|
if (m_cur_irq_prio >= 0)
|
|
m_irq_active &= ~(1 << m_cur_irq_prio);
|
|
if (m_irq_active & 4)
|
|
m_cur_irq_prio = 2;
|
|
else if (m_irq_active & 2)
|
|
m_cur_irq_prio = 1;
|
|
else if (m_irq_active & 1)
|
|
m_cur_irq_prio = 0;
|
|
else
|
|
m_cur_irq_prio = -1;
|
|
LOG(("New: %d %02x\n", m_cur_irq_prio, m_irq_active));
|
|
}
|
|
|
|
uint8_t mcs51_cpu_device::r_acc() { return SFR_A(ADDR_ACC); }
|
|
|
|
uint8_t mcs51_cpu_device::r_psw() { return SFR_A(ADDR_PSW); }
|
|
|
|
|
|
/* Generate an external ram address for read/writing using indirect addressing mode */
|
|
|
|
/*The lowest 8 bits of the address are passed in (from the R0/R1 register), however
|
|
the hardware can be configured to set the rest of the address lines to any available output port pins, which
|
|
means the only way we can implement this is to allow the driver to setup a callback to generate the
|
|
address as defined by the specific hardware setup. We'll assume the address won't be bigger than 32 bits
|
|
|
|
Couriersud, October 2008:
|
|
There is no way external hardware can distinguish between 8bit access and 16 bit access.
|
|
During 16bit access the high order byte of the address is output on port 2. We therefore
|
|
assume that most hardware will use port 2 for 8bit access as well.
|
|
|
|
On configurations where 8 bit access in conjunction with other ports is used,
|
|
it is up to the driver to use AM_MIRROR to mask out the high level address and
|
|
provide it's own mapping.
|
|
*/
|
|
|
|
/*
|
|
The DS5002FP has 2 16 bits data address buses (the byte-wide bus and the expanded bus). The exact memory position accessed depends on the
|
|
partition mode, the memory range and the expanded bus select. The partition mode and the expanded bus select can be changed at any time.
|
|
|
|
In order to simplify memory mapping to the data address bus, the following address map is assumed for partitioned mode:
|
|
|
|
0x00000-0x0ffff -> data memory on the expanded bus
|
|
0x10000-0x1ffff -> data memory on the byte-wide bus
|
|
|
|
For non-partitioned mode the following memory map is assumed:
|
|
|
|
0x0000-0xffff -> data memory (the bus used to access it does not matter)
|
|
*/
|
|
|
|
offs_t mcs51_cpu_device::external_ram_iaddr(offs_t offset, offs_t mem_mask)
|
|
{
|
|
/* Memory Range (RG1 and RG0 @ MCON and RPCTL registers) */
|
|
static const uint16_t ds5002fp_ranges[4] = { 0x1fff, 0x3fff, 0x7fff, 0xffff };
|
|
/* Memory Partition Table (RG1 & RG0 @ MCON & RPCTL registers) */
|
|
static const uint32_t ds5002fp_partitions[16] = {
|
|
0x0000, 0x1000, 0x2000, 0x3000, 0x4000, 0x5000, 0x6000, 0x7000,
|
|
0x8000, 0x9000, 0xa000, 0xb000, 0xc000, 0xd000, 0xe000, 0x10000 };
|
|
|
|
/* if partition mode is set, adjust offset based on the bus */
|
|
if (m_features & FEATURE_DS5002FP)
|
|
{
|
|
if (!GET_PM) {
|
|
if (!GET_EXBS) {
|
|
if ((offset >= ds5002fp_partitions[GET_PA]) && (offset <= ds5002fp_ranges[m_ds5002fp.range])) {
|
|
offset += 0x10000;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (mem_mask == 0x00ff)
|
|
return (offset & mem_mask) | (P2 << 8);
|
|
}
|
|
return offset;
|
|
}
|
|
|
|
/* Internal ram read/write */
|
|
|
|
uint8_t mcs51_cpu_device::iram_read(size_t offset)
|
|
{
|
|
return (((offset) < 0x80) ? m_data->read_byte(offset) : sfr_read(offset));
|
|
}
|
|
|
|
void mcs51_cpu_device::iram_write(size_t offset, uint8_t data)
|
|
{
|
|
if ((offset) < 0x80)
|
|
m_data->write_byte(offset, data);
|
|
else
|
|
sfr_write(offset, data);
|
|
}
|
|
|
|
/*Push the current PC to the stack*/
|
|
void mcs51_cpu_device::push_pc()
|
|
{
|
|
uint8_t tmpSP = SP+1; //Grab and Increment Stack Pointer
|
|
IRAM_IW(tmpSP, (PC & 0xff)); //Store low byte of PC to Internal Ram (Use IRAM_IW to store stack above 128 bytes)
|
|
tmpSP++; // ""
|
|
SP = tmpSP; // ""
|
|
IRAM_IW(tmpSP, ( (PC & 0xff00) >> 8)); //Store hi byte of PC to next address in Internal Ram (Use IRAM_IW to store stack above 128 bytes)
|
|
}
|
|
|
|
/*Pop the current PC off the stack and into the pc*/
|
|
void mcs51_cpu_device::pop_pc()
|
|
{
|
|
uint8_t tmpSP = SP; //Grab Stack Pointer
|
|
PC = (IRAM_IR(tmpSP--) & 0xff) << 8; //Store hi byte to PC (must use IRAM_IR to access stack pointing above 128 bytes)
|
|
PC = PC | IRAM_IR(tmpSP--); //Store lo byte to PC (must use IRAM_IR to access stack pointing above 128 bytes)
|
|
SP = tmpSP; //Decrement Stack Pointer
|
|
}
|
|
|
|
//Set the PSW Parity Flag
|
|
void mcs51_cpu_device::set_parity()
|
|
{
|
|
//This flag will be set when the accumulator contains an odd # of bits set..
|
|
uint8_t p = 0;
|
|
int i;
|
|
uint8_t a = ACC;
|
|
|
|
for (i=0; i<8; i++) { //Test for each of the 8 bits in the ACC!
|
|
p ^= (a & 1);
|
|
a = (a >> 1);
|
|
}
|
|
|
|
SET_P(p & 1);
|
|
}
|
|
|
|
uint8_t mcs51_cpu_device::bit_address_r(uint8_t offset)
|
|
{
|
|
uint8_t word;
|
|
uint8_t mask;
|
|
int bit_pos;
|
|
int distance; /* distance between bit addressable words */
|
|
/* 1 for normal bits, 8 for sfr bit addresses */
|
|
|
|
//User defined bit addresses 0x20-0x2f (values are 0x0-0x7f)
|
|
if (offset < 0x80) {
|
|
distance = 1;
|
|
word = ( (offset & 0x78) >> 3) * distance + 0x20;
|
|
bit_pos = offset & 0x7;
|
|
mask = (0x1 << bit_pos);
|
|
return((IRAM_R(word) & mask) >> bit_pos);
|
|
}
|
|
//SFR bit addressable registers
|
|
else {
|
|
distance = 8;
|
|
word = ( (offset & 0x78) >> 3) * distance + 0x80;
|
|
bit_pos = offset & 0x7;
|
|
mask = (0x1 << bit_pos);
|
|
return ((IRAM_R(word) & mask) >> bit_pos);
|
|
}
|
|
}
|
|
|
|
|
|
void mcs51_cpu_device::bit_address_w(uint8_t offset, uint8_t bit)
|
|
{
|
|
int word;
|
|
uint8_t mask;
|
|
int bit_pos;
|
|
uint8_t result;
|
|
int distance;
|
|
|
|
/* User defined bit addresses 0x20-0x2f (values are 0x0-0x7f) */
|
|
if (offset < 0x80) {
|
|
distance = 1;
|
|
word = ((offset & 0x78) >> 3) * distance + 0x20;
|
|
bit_pos = offset & 0x7;
|
|
bit = (bit & 0x1) << bit_pos;
|
|
mask = ~(1 << bit_pos) & 0xff;
|
|
result = IRAM_R(word) & mask;
|
|
result = result | bit;
|
|
IRAM_W(word, result);
|
|
}
|
|
/* SFR bit addressable registers */
|
|
else {
|
|
distance = 8;
|
|
word = ((offset & 0x78) >> 3) * distance + 0x80;
|
|
bit_pos = offset & 0x7;
|
|
bit = (bit & 0x1) << bit_pos;
|
|
mask = ~(1 << bit_pos) & 0xff;
|
|
result = IRAM_R(word) & mask;
|
|
result = result | bit;
|
|
IRAM_W(word, result);
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::do_add_flags(uint8_t a, uint8_t data, uint8_t c)
|
|
{
|
|
uint16_t result = a+data+c;
|
|
int16_t result1 = (int8_t)a+(int8_t)data+c;
|
|
|
|
SET_CY((result & 0x100) >> 8);
|
|
result = (a&0x0f)+(data&0x0f)+c;
|
|
SET_AC((result & 0x10) >> 4);
|
|
SET_OV(result1 < -128 || result1 > 127);
|
|
}
|
|
|
|
void mcs51_cpu_device::do_sub_flags(uint8_t a, uint8_t data, uint8_t c)
|
|
{
|
|
uint16_t result = a-(data+c);
|
|
int16_t result1 = (int8_t)a-(int8_t)(data+c);
|
|
|
|
SET_CY((result & 0x100) >> 8);
|
|
result = (a&0x0f)-((data&0x0f)+c);
|
|
SET_AC((result & 0x10) >> 4);
|
|
SET_OV((result1 < -128 || result1 > 127));
|
|
}
|
|
|
|
void mcs51_cpu_device::transmit_receive(int source)
|
|
{
|
|
int mode = (GET_SM0<<1) | GET_SM1;
|
|
|
|
if (source == 1) /* timer1 */
|
|
m_uart.smod_div = (m_uart.smod_div + 1) & (2-GET_SMOD);
|
|
|
|
switch(mode) {
|
|
//8 bit shifter ( + start,stop bit ) - baud set by clock freq / 12
|
|
case 0:
|
|
m_uart.rx_clk += (source == 0) ? 16 : 0; /* clock / 12 */
|
|
m_uart.tx_clk += (source == 0) ? 16 : 0; /* clock / 12 */
|
|
break;
|
|
//8 bit uart ( + start,stop bit ) - baud set by timer1 or timer2
|
|
case 1:
|
|
case 3:
|
|
if (source == 1)
|
|
{
|
|
m_uart.tx_clk += (GET_TCLK ? 0 : !m_uart.smod_div);
|
|
m_uart.rx_clk += (GET_RCLK ? 0 : !m_uart.smod_div);
|
|
}
|
|
if (source == 2)
|
|
{
|
|
m_uart.tx_clk += (GET_TCLK ? 1 : 0);
|
|
m_uart.rx_clk += (GET_RCLK ? 1 : 0);
|
|
}
|
|
break;
|
|
//9 bit uart
|
|
case 2:
|
|
m_uart.rx_clk += (source == 0) ? (GET_SMOD ? 6 : 3) : 0; /* clock / 12 * 3 / 8 (16) = clock / 32 (64)*/
|
|
m_uart.tx_clk += (source == 0) ? (GET_SMOD ? 6 : 3) : 0; /* clock / 12 */
|
|
break;
|
|
}
|
|
/* transmit ? */
|
|
if (m_uart.tx_clk >= 16)
|
|
{
|
|
m_uart.tx_clk &= 0x0f;
|
|
if(m_uart.bits_to_send)
|
|
{
|
|
m_uart.bits_to_send--;
|
|
if(m_uart.bits_to_send == 0) {
|
|
//Call the callback function
|
|
m_serial_tx_cb(*m_io, 0, m_uart.data_out, 0xff);
|
|
//Set Interrupt Flag
|
|
SET_TI(1);
|
|
}
|
|
}
|
|
|
|
}
|
|
/* receive */
|
|
if (m_uart.rx_clk >= 16)
|
|
{
|
|
m_uart.rx_clk &= 0x0f;
|
|
if (m_uart.delay_cycles>0)
|
|
{
|
|
m_uart.delay_cycles--;
|
|
if (m_uart.delay_cycles == 0)
|
|
{
|
|
int data = 0;
|
|
//Call our callball function to retrieve the data
|
|
data = m_serial_rx_cb(*m_io, 0, 0xff);
|
|
LOG(("RX Deliver %d\n", data));
|
|
SET_SBUF(data);
|
|
//Flag the IRQ
|
|
SET_RI(1);
|
|
SET_RB8(1); // HACK force 2nd stop bit
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void mcs51_cpu_device::update_timer_t0(int cycles)
|
|
{
|
|
int mode = (GET_M0_1<<1) | GET_M0_0;
|
|
uint32_t count;
|
|
|
|
if (GET_TR0)
|
|
{
|
|
uint32_t delta;
|
|
|
|
/* counter / external input */
|
|
delta = GET_CT0 ? m_t0_cnt : cycles;
|
|
/* taken, reset */
|
|
m_t0_cnt = 0;
|
|
/* TODO: Not sure about IE0. The manual specifies INT0=high,
|
|
* which in turn means CLEAR_LINE.
|
|
* IE0 may be edge triggered depending on IT0 */
|
|
if (GET_GATE0 && !GET_IE0)
|
|
delta = 0;
|
|
|
|
switch(mode) {
|
|
case 0: /* 13 Bit Timer Mode */
|
|
count = ((TH0<<5) | ( TL0 & 0x1f ) );
|
|
count += delta;
|
|
if ( count & 0xffffe000 ) /* Check for overflow */
|
|
SET_TF0(1);
|
|
TH0 = (count>>5) & 0xff;
|
|
TL0 = count & 0x1f ;
|
|
break;
|
|
case 1: /* 16 Bit Timer Mode */
|
|
count = ((TH0<<8) | TL0);
|
|
count += delta;
|
|
if ( count & 0xffff0000 ) /* Check for overflow */
|
|
SET_TF0(1);
|
|
TH0 = (count>>8) & 0xff;
|
|
TL0 = count & 0xff;
|
|
break;
|
|
case 2: /* 8 Bit Autoreload */
|
|
count = ((uint32_t) TL0) + delta;
|
|
if ( count & 0xffffff00 ) /* Check for overflow */
|
|
{
|
|
SET_TF0(1);
|
|
count += TH0; /* Reload timer */
|
|
}
|
|
/* Update new values of the counter */
|
|
TL0 = count & 0xff;
|
|
break;
|
|
case 3:
|
|
/* Split Timer 1 */
|
|
count = ((uint32_t) TL0) + delta;
|
|
if ( count & 0xffffff00 ) /* Check for overflow */
|
|
SET_TF0(1);
|
|
TL0 = count & 0xff; /* Update new values of the counter */
|
|
break;
|
|
}
|
|
}
|
|
if (GET_TR1)
|
|
{
|
|
switch(mode)
|
|
{
|
|
case 3:
|
|
/* Split Timer 2 */
|
|
count = ((uint32_t) TH0) + cycles; /* No gate control or counting !*/
|
|
if ( count & 0xffffff00 ) /* Check for overflow */
|
|
SET_TF1(1);
|
|
TH0 = count & 0xff; /* Update new values of the counter */
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* From the DS5002FP User Manual
|
|
When Timer 1 is selected for operation in Mode 3, it stops counting and holds its current value. This
|
|
action is the same as setting TR1 = 0. When Timer 0 is selected in Mode 3, Timer 1???s control bits are
|
|
stolen as described above. As a result, Timer 1???s functions are limited in this MODE. It is forced to
|
|
operate as a timer whose clock in-put is 12 tCLK and it cannot generate an interrupt on overflow. In
|
|
addition, it also cannot be used with the GATE function. However, it can be started and stopped by
|
|
switching it into or out of Mode 3 or it can be assigned as a baud rate generator for the serial port.
|
|
*/
|
|
|
|
/* Intel documentation:
|
|
* Timer 1 may still be used in modes 0, 1, and 2, while timer 0
|
|
* is in mode 3. With one important exception: No interrupts
|
|
* will be generated by timer 1 while timer 0 is using the TF1
|
|
* overflow flag
|
|
*/
|
|
|
|
void mcs51_cpu_device::update_timer_t1(int cycles)
|
|
{
|
|
uint8_t mode = (GET_M1_1<<1) | GET_M1_0;
|
|
uint8_t mode_0 = (GET_M0_1<<1) | GET_M0_0;
|
|
uint32_t count;
|
|
|
|
if (mode_0 != 3)
|
|
{
|
|
if (GET_TR1)
|
|
{
|
|
uint32_t delta;
|
|
uint32_t overflow = 0;
|
|
|
|
/* counter / external input */
|
|
delta = GET_CT1 ? m_t1_cnt : cycles;
|
|
/* taken, reset */
|
|
m_t1_cnt = 0;
|
|
/* TODO: Not sure about IE0. The manual specifies INT0=high,
|
|
* which in turn means CLEAR_LINE. Change to access last_state?
|
|
* IE0 may be edge triggered depending on IT0 */
|
|
if (GET_GATE1 && !GET_IE1)
|
|
delta = 0;
|
|
|
|
switch(mode) {
|
|
case 0: /* 13 Bit Timer Mode */
|
|
count = ((TH1<<5) | ( TL1 & 0x1f ) );
|
|
count += delta;
|
|
overflow = count & 0xffffe000; /* Check for overflow */
|
|
TH1 = (count>>5) & 0xff;
|
|
TL1 = count & 0x1f ;
|
|
break;
|
|
case 1: /* 16 Bit Timer Mode */
|
|
count = ((TH1<<8) | TL1);
|
|
count += delta;
|
|
overflow = count & 0xffff0000; /* Check for overflow */
|
|
TH1 = (count>>8) & 0xff;
|
|
TL1 = count & 0xff;
|
|
break;
|
|
case 2: /* 8 Bit Autoreload */
|
|
count = ((uint32_t) TL1) + delta;
|
|
overflow = count & 0xffffff00; /* Check for overflow */
|
|
if ( overflow )
|
|
{
|
|
count += TH1; /* Reload timer */
|
|
}
|
|
/* Update new values of the counter */
|
|
TL1 = count & 0xff;
|
|
break;
|
|
case 3:
|
|
/* do nothing */
|
|
break;
|
|
}
|
|
if (overflow)
|
|
{
|
|
SET_TF1(1);
|
|
transmit_receive(1);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32_t delta;
|
|
uint32_t overflow = 0;
|
|
|
|
delta = cycles;
|
|
/* taken, reset */
|
|
m_t1_cnt = 0;
|
|
switch(mode) {
|
|
case 0: /* 13 Bit Timer Mode */
|
|
count = ((TH1<<5) | ( TL1 & 0x1f ) );
|
|
count += delta;
|
|
overflow = count & 0xffffe000; /* Check for overflow */
|
|
TH1 = (count>>5) & 0xff;
|
|
TL1 = count & 0x1f ;
|
|
break;
|
|
case 1: /* 16 Bit Timer Mode */
|
|
count = ((TH1<<8) | TL1);
|
|
count += delta;
|
|
overflow = count & 0xffff0000; /* Check for overflow */
|
|
TH1 = (count>>8) & 0xff;
|
|
TL1 = count & 0xff;
|
|
break;
|
|
case 2: /* 8 Bit Autoreload */
|
|
count = ((uint32_t) TL1) + delta;
|
|
overflow = count & 0xffffff00; /* Check for overflow */
|
|
if ( overflow )
|
|
{
|
|
count += TH1; /* Reload timer */
|
|
}
|
|
/* Update new values of the counter */
|
|
TL1 = count & 0xff;
|
|
break;
|
|
case 3:
|
|
/* do nothing */
|
|
break;
|
|
}
|
|
if (overflow)
|
|
{
|
|
transmit_receive(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::update_timer_t2(int cycles)
|
|
{
|
|
/* Update Timer 2 */
|
|
if(GET_TR2) {
|
|
int mode = ((GET_TCLK | GET_RCLK) << 1) | GET_CP;
|
|
int delta = GET_CT2 ? m_t2_cnt : (mode & 2) ? cycles * (12/2) : cycles;
|
|
|
|
uint32_t count = ((TH2<<8) | TL2) + delta;
|
|
m_t2_cnt = 0;
|
|
|
|
switch (mode)
|
|
{
|
|
case 0: /* 16 Bit Auto Reload */
|
|
if ( count & 0xffff0000 )
|
|
{
|
|
SET_TF2(1);
|
|
count += ((RCAP2H<<8) | RCAP2L);
|
|
}
|
|
else if (GET_EXEN2 && m_t2ex_cnt>0)
|
|
{
|
|
count += ((RCAP2H<<8) | RCAP2L);
|
|
m_t2ex_cnt = 0;
|
|
}
|
|
TH2 = (count>>8) & 0xff;
|
|
TL2 = count & 0xff;
|
|
break;
|
|
case 1: /* 16 Bit Capture */
|
|
if ( count & 0xffff0000 )
|
|
SET_TF2(1);
|
|
TH2 = (count>>8) & 0xff;
|
|
TL2 = count & 0xff;
|
|
|
|
if (GET_EXEN2 && m_t2ex_cnt>0)
|
|
{
|
|
RCAP2H = TH2;
|
|
RCAP2L = TL2;
|
|
m_t2ex_cnt = 0;
|
|
}
|
|
break;
|
|
case 2:
|
|
case 3: /* Baud rate */
|
|
if ( count & 0xffff0000 )
|
|
{
|
|
count += ((RCAP2H<<8) | RCAP2L);
|
|
transmit_receive(2);
|
|
}
|
|
TH2 = (count>>8) & 0xff;
|
|
TL2 = count & 0xff;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::update_timers(int cycles)
|
|
{
|
|
while (cycles--)
|
|
{
|
|
update_timer_t0(1);
|
|
update_timer_t1(1);
|
|
|
|
if (m_features & FEATURE_I8052)
|
|
{
|
|
update_timer_t2(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
//Set up to transmit data out of serial port
|
|
//NOTE: Enable Serial Port Interrupt bit is NOT required to send/receive data!
|
|
|
|
void mcs51_cpu_device::serial_transmit(uint8_t data)
|
|
{
|
|
int mode = (GET_SM0<<1) | GET_SM1;
|
|
|
|
//Flag that we're sending data
|
|
m_uart.data_out = data;
|
|
LOG(("serial_transmit: %x %x\n", mode, data));
|
|
switch(mode) {
|
|
//8 bit shifter ( + start,stop bit ) - baud set by clock freq / 12
|
|
case 0:
|
|
m_uart.bits_to_send = 8+2;
|
|
break;
|
|
//8 bit uart ( + start,stop bit ) - baud set by timer1 or timer2
|
|
case 1:
|
|
m_uart.bits_to_send = 8+2;
|
|
break;
|
|
//9 bit uart
|
|
case 2:
|
|
case 3:
|
|
m_uart.bits_to_send = 8+3;
|
|
break;
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::serial_receive()
|
|
{
|
|
int mode = (GET_SM0<<1) | GET_SM1;
|
|
|
|
if (GET_REN) {
|
|
switch(mode) {
|
|
//8 bit shifter ( + start,stop bit ) - baud set by clock freq / 12
|
|
case 0:
|
|
m_uart.delay_cycles = 8+2;
|
|
break;
|
|
//8 bit uart ( + start,stop bit ) - baud set by timer1 or timer2
|
|
case 1:
|
|
m_uart.delay_cycles = 8+2;
|
|
break;
|
|
//9 bit uart
|
|
case 2:
|
|
case 3:
|
|
m_uart.delay_cycles = 8+3;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Check and update status of serial port */
|
|
void mcs51_cpu_device::update_serial(int cycles)
|
|
{
|
|
while (--cycles>=0)
|
|
transmit_receive(0);
|
|
}
|
|
|
|
/* Check and update status of serial port */
|
|
void mcs51_cpu_device::update_irq_prio(uint8_t ipl, uint8_t iph)
|
|
{
|
|
int i;
|
|
for (i=0; i<8; i++)
|
|
m_irq_prio[i] = ((ipl >> i) & 1) | (((iph >>i ) & 1) << 1);
|
|
}
|
|
|
|
|
|
/***************************************************************************
|
|
OPCODES
|
|
***************************************************************************/
|
|
|
|
#define OPHANDLER( _name ) void mcs51_cpu_device::_name (uint8_t r)
|
|
|
|
#include "mcs51ops.hxx"
|
|
|
|
|
|
void mcs51_cpu_device::execute_op(uint8_t op)
|
|
{
|
|
if (m_recalc_parity)
|
|
{
|
|
set_parity();
|
|
m_recalc_parity = 0;
|
|
}
|
|
|
|
switch( op )
|
|
{
|
|
case 0x00: nop(op); break; //NOP
|
|
case 0x01: ajmp(op); break; //AJMP code addr
|
|
case 0x02: ljmp(op); break; //LJMP code addr
|
|
case 0x03: rr_a(op); break; //RR A
|
|
case 0x04: inc_a(op); break; //INC A
|
|
case 0x05: RWM=1; inc_mem(op); RWM=0; break; //INC data addr
|
|
|
|
case 0x06:
|
|
case 0x07: inc_ir(op&1); break; //INC @R0/@R1
|
|
|
|
case 0x08:
|
|
case 0x09:
|
|
case 0x0a:
|
|
case 0x0b:
|
|
case 0x0c:
|
|
case 0x0d:
|
|
case 0x0e:
|
|
case 0x0f: inc_r(op&7); break; //INC R0 to R7
|
|
|
|
case 0x10: RWM=1; jbc(op); RWM=0; break; //JBC bit addr, code addr
|
|
case 0x11: acall(op); break; //ACALL code addr
|
|
case 0x12: lcall(op); break; //LCALL code addr
|
|
case 0x13: rrc_a(op); break; //RRC A
|
|
case 0x14: dec_a(op); break; //DEC A
|
|
case 0x15: RWM=1; dec_mem(op); RWM=0; break; //DEC data addr
|
|
|
|
case 0x16:
|
|
case 0x17: dec_ir(op&1); break; //DEC @R0/@R1
|
|
|
|
case 0x18:
|
|
case 0x19:
|
|
case 0x1a:
|
|
case 0x1b:
|
|
case 0x1c:
|
|
case 0x1d:
|
|
case 0x1e:
|
|
case 0x1f: dec_r(op&7); break; //DEC R0 to R7
|
|
|
|
case 0x20: jb(op); break; //JB bit addr, code addr
|
|
case 0x21: ajmp(op); break; //AJMP code addr
|
|
case 0x22: ret(op); break; //RET
|
|
case 0x23: rl_a(op); break; //RL A
|
|
case 0x24: add_a_byte(op); break; //ADD A, #data
|
|
case 0x25: add_a_mem(op); break; //ADD A, data addr
|
|
|
|
case 0x26:
|
|
case 0x27: add_a_ir(op&1); break; //ADD A, @R0/@R1
|
|
|
|
case 0x28:
|
|
case 0x29:
|
|
case 0x2a:
|
|
case 0x2b:
|
|
case 0x2c:
|
|
case 0x2d:
|
|
case 0x2e:
|
|
case 0x2f: add_a_r(op&7); break; //ADD A, R0 to R7
|
|
|
|
case 0x30: jnb(op); break; //JNB bit addr, code addr
|
|
case 0x31: acall(op); break; //ACALL code addr
|
|
case 0x32: reti(op); break; //RETI
|
|
case 0x33: rlc_a(op); break; //RLC A
|
|
case 0x34: addc_a_byte(op); break; //ADDC A, #data
|
|
case 0x35: addc_a_mem(op); break; //ADDC A, data addr
|
|
|
|
case 0x36:
|
|
case 0x37: addc_a_ir(op&1); break; //ADDC A, @R0/@R1
|
|
|
|
case 0x38:
|
|
case 0x39:
|
|
case 0x3a:
|
|
case 0x3b:
|
|
case 0x3c:
|
|
case 0x3d:
|
|
case 0x3e:
|
|
case 0x3f: addc_a_r(op&7); break; //ADDC A, R0 to R7
|
|
|
|
case 0x40: jc(op); break; //JC code addr
|
|
case 0x41: ajmp(op); break; //AJMP code addr
|
|
case 0x42: RWM=1; orl_mem_a(op); RWM=0; break; //ORL data addr, A
|
|
case 0x43: RWM=1; orl_mem_byte(op); RWM=0; break; //ORL data addr, #data
|
|
case 0x44: orl_a_byte(op); break;
|
|
case 0x45: orl_a_mem(op); break; //ORL A, data addr
|
|
|
|
case 0x46:
|
|
case 0x47: orl_a_ir(op&1); break; //ORL A, @RO/@R1
|
|
|
|
case 0x48:
|
|
case 0x49:
|
|
case 0x4a:
|
|
case 0x4b:
|
|
case 0x4c:
|
|
case 0x4d:
|
|
case 0x4e:
|
|
case 0x4f: orl_a_r(op&7); break; //ORL A, RO to R7
|
|
|
|
case 0x50: jnc(op); break; //JNC code addr
|
|
case 0x51: acall(op); break; //ACALL code addr
|
|
case 0x52: RWM=1; anl_mem_a(op); RWM=0; break; //ANL data addr, A
|
|
case 0x53: RWM=1; anl_mem_byte(op); RWM=0; break; //ANL data addr, #data
|
|
case 0x54: anl_a_byte(op); break; //ANL A, #data
|
|
case 0x55: anl_a_mem(op); break; //ANL A, data addr
|
|
|
|
case 0x56:
|
|
case 0x57: anl_a_ir(op&1); break; //ANL A, @RO/@R1
|
|
|
|
case 0x58:
|
|
case 0x59:
|
|
case 0x5a:
|
|
case 0x5b:
|
|
case 0x5c:
|
|
case 0x5d:
|
|
case 0x5e:
|
|
case 0x5f: anl_a_r(op&7); break; //ANL A, RO to R7
|
|
|
|
case 0x60: jz(op); break; //JZ code addr
|
|
case 0x61: ajmp(op); break; //AJMP code addr
|
|
case 0x62: RWM=1; xrl_mem_a(op); RWM=0; break; //XRL data addr, A
|
|
case 0x63: RWM=1; xrl_mem_byte(op); RWM=0; break; //XRL data addr, #data
|
|
case 0x64: xrl_a_byte(op); break; //XRL A, #data
|
|
case 0x65: xrl_a_mem(op); break; //XRL A, data addr
|
|
|
|
case 0x66:
|
|
case 0x67: xrl_a_ir(op&1); break; //XRL A, @R0/@R1
|
|
|
|
case 0x68:
|
|
case 0x69:
|
|
case 0x6a:
|
|
case 0x6b:
|
|
case 0x6c:
|
|
case 0x6d:
|
|
case 0x6e:
|
|
case 0x6f: xrl_a_r(op&7); break; //XRL A, R0 to R7
|
|
|
|
case 0x70: jnz(op); break; //JNZ code addr
|
|
case 0x71: acall(op); break; //ACALL code addr
|
|
case 0x72: orl_c_bitaddr(op); break; //ORL C, bit addr
|
|
case 0x73: jmp_iadptr(op); break; //JMP @A+DPTR
|
|
case 0x74: mov_a_byte(op); break; //MOV A, #data
|
|
case 0x75: mov_mem_byte(op); break; //MOV data addr, #data
|
|
|
|
case 0x76:
|
|
case 0x77: mov_ir_byte(op&1); break; //MOV @R0/@R1, #data
|
|
|
|
case 0x78:
|
|
case 0x79:
|
|
case 0x7a:
|
|
case 0x7b:
|
|
case 0x7c:
|
|
case 0x7d:
|
|
case 0x7e:
|
|
case 0x7f: mov_r_byte(op&7); break; //MOV R0 to R7, #data
|
|
|
|
case 0x80: sjmp(op); break; //SJMP code addr
|
|
case 0x81: ajmp(op); break; //AJMP code addr
|
|
case 0x82: anl_c_bitaddr(op); break; //ANL C, bit addr
|
|
case 0x83: movc_a_iapc(op); break; //MOVC A, @A + PC
|
|
case 0x84: div_ab(op); break; //DIV AB
|
|
case 0x85: mov_mem_mem(op); break; //MOV data addr, data addr
|
|
|
|
case 0x86:
|
|
case 0x87: mov_mem_ir(op&1); break; //MOV data addr, @R0/@R1
|
|
|
|
case 0x88:
|
|
case 0x89:
|
|
case 0x8a:
|
|
case 0x8b:
|
|
case 0x8c:
|
|
case 0x8d:
|
|
case 0x8e:
|
|
case 0x8f: mov_mem_r(op&7); break; //MOV data addr,R0 to R7
|
|
|
|
case 0x90: mov_dptr_byte(op); break; //MOV DPTR, #data
|
|
case 0x91: acall(op); break; //ACALL code addr
|
|
case 0x92: RWM = 1; mov_bitaddr_c(op); RWM = 0; break; //MOV bit addr, C
|
|
case 0x93: movc_a_iadptr(op); break; //MOVC A, @A + DPTR
|
|
case 0x94: subb_a_byte(op); break; //SUBB A, #data
|
|
case 0x95: subb_a_mem(op); break; //SUBB A, data addr
|
|
|
|
case 0x96:
|
|
case 0x97: subb_a_ir(op&1); break; //SUBB A, @R0/@R1
|
|
|
|
case 0x98:
|
|
case 0x99:
|
|
case 0x9a:
|
|
case 0x9b:
|
|
case 0x9c:
|
|
case 0x9d:
|
|
case 0x9e:
|
|
case 0x9f: subb_a_r(op&7); break; //SUBB A, R0 to R7
|
|
|
|
case 0xa0: orl_c_nbitaddr(op); break; //ORL C, /bit addr
|
|
case 0xa1: ajmp(op); break; //AJMP code addr
|
|
case 0xa2: mov_c_bitaddr(op); break; //MOV C, bit addr
|
|
case 0xa3: inc_dptr(op); break; //INC DPTR
|
|
case 0xa4: mul_ab(op); break; //MUL AB
|
|
case 0xa5: illegal(op); break; //reserved
|
|
|
|
case 0xa6:
|
|
case 0xa7: mov_ir_mem(op&1); break; //MOV @R0/@R1, data addr
|
|
|
|
case 0xa8:
|
|
case 0xa9:
|
|
case 0xaa:
|
|
case 0xab:
|
|
case 0xac:
|
|
case 0xad:
|
|
case 0xae:
|
|
case 0xaf: mov_r_mem(op&7); break; //MOV R0 to R7, data addr
|
|
|
|
case 0xb0: anl_c_nbitaddr(op); break; //ANL C,/bit addr
|
|
case 0xb1: acall(op); break; //ACALL code addr
|
|
case 0xb2: RWM=1; cpl_bitaddr(op); RWM=0; break; //CPL bit addr
|
|
case 0xb3: cpl_c(op); break; //CPL C
|
|
case 0xb4: cjne_a_byte(op); break; //CJNE A, #data, code addr
|
|
case 0xb5: cjne_a_mem(op); break; //CJNE A, data addr, code addr
|
|
|
|
case 0xb6:
|
|
case 0xb7: cjne_ir_byte(op&1); break; //CJNE @R0/@R1, #data, code addr
|
|
|
|
case 0xb8:
|
|
case 0xb9:
|
|
case 0xba:
|
|
case 0xbb:
|
|
case 0xbc:
|
|
case 0xbd:
|
|
case 0xbe:
|
|
case 0xbf: cjne_r_byte(op&7); break; //CJNE R0 to R7, #data, code addr
|
|
|
|
case 0xc0: push(op); break; //PUSH data addr
|
|
case 0xc1: ajmp(op); break; //AJMP code addr
|
|
case 0xc2: RWM=1; clr_bitaddr(op); RWM=0; break; //CLR bit addr
|
|
case 0xc3: clr_c(op); break; //CLR C
|
|
case 0xc4: swap_a(op); break; //SWAP A
|
|
case 0xc5: xch_a_mem(op); break; //XCH A, data addr
|
|
|
|
case 0xc6:
|
|
case 0xc7: xch_a_ir(op&1); break; //XCH A, @RO/@R1
|
|
|
|
case 0xc8:
|
|
case 0xc9:
|
|
case 0xca:
|
|
case 0xcb:
|
|
case 0xcc:
|
|
case 0xcd:
|
|
case 0xce:
|
|
case 0xcf: xch_a_r(op&7); break; //XCH A, RO to R7
|
|
|
|
case 0xd0: pop(op); break; //POP data addr
|
|
case 0xd1: acall(op); break; //ACALL code addr
|
|
case 0xd2: RWM=1; setb_bitaddr(op); RWM=0; break; //SETB bit addr
|
|
case 0xd3: setb_c(op); break; //SETB C
|
|
case 0xd4: da_a(op); break; //DA A
|
|
case 0xd5: RWM=1; djnz_mem(op); RWM=0; break; //DJNZ data addr, code addr
|
|
|
|
case 0xd6:
|
|
case 0xd7: xchd_a_ir(op&1); break; //XCHD A, @R0/@R1
|
|
|
|
case 0xd8:
|
|
case 0xd9:
|
|
case 0xda:
|
|
case 0xdb:
|
|
case 0xdc:
|
|
case 0xdd:
|
|
case 0xde:
|
|
case 0xdf: djnz_r(op&7); break; //DJNZ R0 to R7,code addr
|
|
|
|
case 0xe0: movx_a_idptr(op); break; //MOVX A,@DPTR
|
|
case 0xe1: ajmp(op); break; //AJMP code addr
|
|
|
|
case 0xe2:
|
|
case 0xe3: movx_a_ir(op&1); break; //MOVX A, @R0/@R1
|
|
|
|
case 0xe4: clr_a(op); break; //CLR A
|
|
case 0xe5: mov_a_mem(op); break; //MOV A, data addr
|
|
case 0xe6:
|
|
case 0xe7: mov_a_ir(op&1); break; //MOV A,@RO/@R1
|
|
|
|
case 0xe8:
|
|
case 0xe9:
|
|
case 0xea:
|
|
case 0xeb:
|
|
case 0xec:
|
|
case 0xed:
|
|
case 0xee:
|
|
case 0xef: mov_a_r(op&7); break; //MOV A,R0 to R7
|
|
|
|
case 0xf0: movx_idptr_a(op); break; //MOVX @DPTR,A
|
|
case 0xf1: acall(op); break; //ACALL code addr
|
|
|
|
case 0xf2:
|
|
case 0xf3: movx_ir_a(op&1); break; //MOVX @R0/@R1,A
|
|
|
|
case 0xf4: cpl_a(op); break; //CPL A
|
|
case 0xf5: mov_mem_a(op); break; //MOV data addr, A
|
|
|
|
case 0xf6:
|
|
case 0xf7: mov_ir_a(op&1); break; //MOV @R0/@R1, A
|
|
|
|
case 0xf8:
|
|
case 0xf9:
|
|
case 0xfa:
|
|
case 0xfb:
|
|
case 0xfc:
|
|
case 0xfd:
|
|
case 0xfe:
|
|
case 0xff: mov_r_a(op&7); break; //MOV R0 to R7, A
|
|
default:
|
|
illegal(op);
|
|
}
|
|
}
|
|
|
|
/***************************************************************************
|
|
OPCODE CYCLES
|
|
***************************************************************************/
|
|
|
|
/* # of oscilations each opcode requires*/
|
|
const uint8_t mcs51_cpu_device::mcs51_cycles[256] = {
|
|
1,2,2,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,2,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,2,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,2,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,1,2,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,1,2,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,1,2,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,2,2,1,2,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,2,2,4,2,2,2,2,2,2,2,2,2,2,2,
|
|
2,2,2,2,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,1,2,4,1,2,2,2,2,2,2,2,2,2,2,
|
|
2,2,1,1,2,2,2,2,2,2,2,2,2,2,2,2,
|
|
2,2,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,1,1,1,2,1,1,2,2,2,2,2,2,2,2,
|
|
2,2,2,2,1,1,1,1,1,1,1,1,1,1,1,1,
|
|
2,2,2,2,1,1,1,1,1,1,1,1,1,1,1,1
|
|
};
|
|
|
|
/***********************************************************************************
|
|
Check for pending Interrupts and process - returns # of cycles used for the int
|
|
|
|
Note about priority & interrupting interrupts..
|
|
1) A high priority interrupt cannot be interrupted by anything!
|
|
2) A low priority interrupt can ONLY be interrupted by a high priority interrupt
|
|
3) If more than 1 Interrupt Flag is set (ie, 2 simultaneous requests occur),
|
|
the following logic works as follows:
|
|
1) If two requests come in of different priority levels, the higher one is selected..
|
|
2) If the requests are of the same level, an internal order is used:
|
|
a) IEO
|
|
b) TFO
|
|
c) IE1
|
|
d) TF1
|
|
e) RI+TI
|
|
f) TF2+EXF2
|
|
**********************************************************************************/
|
|
void mcs51_cpu_device::check_irqs()
|
|
{
|
|
uint8_t ints = (GET_IE0 | (GET_TF0<<1) | (GET_IE1<<2) | (GET_TF1<<3)
|
|
| ((GET_RI|GET_TI)<<4));
|
|
uint8_t int_vec = 0;
|
|
uint8_t int_mask;
|
|
int priority_request = -1;
|
|
int i;
|
|
|
|
//If All Inerrupts Disabled or no pending abort..
|
|
int_mask = (GET_EA ? IE : 0x00);
|
|
|
|
if (m_features & FEATURE_I8052)
|
|
ints |= ((GET_TF2|GET_EXF2)<<5);
|
|
|
|
if (m_features & FEATURE_DS5002FP)
|
|
{
|
|
ints |= ((GET_PFW)<<5);
|
|
m_irq_prio[6] = 3; /* force highest priority */
|
|
/* mask out interrupts not enabled */
|
|
ints &= ((int_mask & 0x1f) | ((GET_EPFW)<<5));
|
|
}
|
|
else
|
|
{
|
|
/* mask out interrupts not enabled */
|
|
ints &= int_mask;
|
|
}
|
|
|
|
if (!ints) return;
|
|
|
|
/* CLear IDL - got enabled interrupt */
|
|
if (m_features & FEATURE_CMOS)
|
|
{
|
|
/* any interrupt terminates idle mode */
|
|
SET_IDL(0);
|
|
/* external interrupt wakes up */
|
|
if (ints & (GET_IE0 | GET_IE1))
|
|
/* but not the DS5002FP */
|
|
if (!(m_features & FEATURE_DS5002FP))
|
|
SET_PD(0);
|
|
}
|
|
|
|
for (i=0; i<m_num_interrupts; i++)
|
|
{
|
|
if (ints & (1<<i))
|
|
{
|
|
if (m_irq_prio[i] > priority_request)
|
|
{
|
|
priority_request = m_irq_prio[i];
|
|
int_vec = (i<<3) | 3;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Skip the interrupt request if currently processing interrupt
|
|
* and the new request does not have a higher priority
|
|
*/
|
|
|
|
LOG(("Request: %d\n", priority_request));
|
|
if (m_irq_active && (priority_request <= m_cur_irq_prio))
|
|
{
|
|
LOG(("higher or equal priority irq (%u) in progress already, skipping ...\n", m_cur_irq_prio));
|
|
return;
|
|
}
|
|
|
|
/* also break out of jb int0,<self> loops */
|
|
if (ROP(PC) == 0x20 && ROP_ARG(PC+1) == 0xb2 && ROP_ARG(PC+2) == 0xfd)
|
|
PC += 3;
|
|
|
|
//Save current pc to stack, set pc to new interrupt vector
|
|
push_pc();
|
|
PC = int_vec;
|
|
|
|
/* interrupts take 24 cycles */
|
|
m_inst_cycles += 2;
|
|
|
|
//Set current Irq & Priority being serviced
|
|
m_cur_irq_prio = priority_request;
|
|
m_irq_active |= (1 << priority_request);
|
|
|
|
LOG(("Take: %d %02x\n", m_cur_irq_prio, m_irq_active));
|
|
|
|
//Clear any interrupt flags that should be cleared since we're servicing the irq!
|
|
switch(int_vec) {
|
|
case V_IE0:
|
|
//External Int Flag only cleared when configured as Edge Triggered..
|
|
if(GET_IT0) /* for some reason having this, breaks alving dmd games */
|
|
SET_IE0(0);
|
|
|
|
/* indicate we took the external IRQ */
|
|
standard_irq_callback(0);
|
|
|
|
break;
|
|
case V_TF0:
|
|
//Timer 0 - Always clear Flag
|
|
SET_TF0(0);
|
|
break;
|
|
case V_IE1:
|
|
//External Int Flag only cleared when configured as Edge Triggered..
|
|
if(GET_IT1) /* for some reason having this, breaks alving dmd games */
|
|
SET_IE1(0);
|
|
/* indicate we took the external IRQ */
|
|
standard_irq_callback(1);
|
|
|
|
break;
|
|
case V_TF1:
|
|
//Timer 1 - Always clear Flag
|
|
SET_TF1(0);
|
|
break;
|
|
case V_RITI:
|
|
/* no flags are cleared, TI and RI remain set until reset by software */
|
|
break;
|
|
/* I8052 specific */
|
|
case V_TF2:
|
|
/* no flags are cleared according to manual */
|
|
break;
|
|
/* DS5002FP specific */
|
|
/* case V_PFI:
|
|
* no flags are cleared, PFW is reset by software
|
|
* This has the same vector as V_TF2.
|
|
*/
|
|
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::burn_cycles(int cycles)
|
|
{
|
|
/* Update Timer (if any timers are running) */
|
|
update_timers(cycles);
|
|
|
|
/* Update Serial (only for mode 0) */
|
|
update_serial(cycles);
|
|
|
|
/* check_irqs */
|
|
check_irqs();
|
|
}
|
|
|
|
void mcs51_cpu_device::execute_set_input(int irqline, int state)
|
|
{
|
|
/* From the manual:
|
|
*
|
|
* <cite>In operation all the interrupt flags are latched into the
|
|
* interrupt control system during State 5 of every machine cycle.
|
|
* The samples are polled during the following machine cycle.</cite>
|
|
*
|
|
* ==> Since we do not emulate sub-states, this assumes that the signal is present
|
|
* for at least one cycle (12 states)
|
|
*
|
|
*/
|
|
uint32_t new_state = (m_last_line_state & ~(1 << irqline)) | ((state != CLEAR_LINE) << irqline);
|
|
/* detect 0->1 transitions */
|
|
uint32_t tr_state = (~m_last_line_state) & new_state;
|
|
|
|
switch( irqline )
|
|
{
|
|
//External Interrupt 0
|
|
case MCS51_INT0_LINE:
|
|
//Line Asserted?
|
|
if (state != CLEAR_LINE) {
|
|
//Need cleared->active line transition? (Logical 1-0 Pulse on the line) - CLEAR->ASSERT Transition since INT0 active lo!
|
|
if (GET_IT0) {
|
|
if (GET_BIT(tr_state, MCS51_INT0_LINE))
|
|
SET_IE0(1);
|
|
}
|
|
else
|
|
SET_IE0(1); //Nope, just set it..
|
|
}
|
|
else
|
|
{
|
|
if (!GET_IT0) /* clear if level triggered */
|
|
SET_IE0(0);
|
|
}
|
|
|
|
break;
|
|
|
|
//External Interrupt 1
|
|
case MCS51_INT1_LINE:
|
|
|
|
//Line Asserted?
|
|
if (state != CLEAR_LINE) {
|
|
//Need cleared->active line transition? (Logical 1-0 Pulse on the line) - CLEAR->ASSERT Transition since INT1 active lo!
|
|
if(GET_IT1){
|
|
if (GET_BIT(tr_state, MCS51_INT1_LINE))
|
|
SET_IE1(1);
|
|
}
|
|
else
|
|
SET_IE1(1); //Nope, just set it..
|
|
}
|
|
else
|
|
{
|
|
if (!GET_IT1) /* clear if level triggered */
|
|
SET_IE1(0);
|
|
}
|
|
break;
|
|
|
|
case MCS51_T0_LINE:
|
|
if (GET_BIT(tr_state, MCS51_T0_LINE) && GET_TR0)
|
|
m_t0_cnt++;
|
|
break;
|
|
|
|
case MCS51_T1_LINE:
|
|
if (GET_BIT(tr_state, MCS51_T1_LINE) && GET_TR1)
|
|
m_t1_cnt++;
|
|
break;
|
|
|
|
case MCS51_T2_LINE:
|
|
if (m_features & FEATURE_I8052)
|
|
{
|
|
if (GET_BIT(tr_state, MCS51_T2_LINE) && GET_TR1)
|
|
m_t2_cnt++;
|
|
}
|
|
else
|
|
fatalerror("mcs51: Trying to set T2_LINE on a non I8052 type cpu.\n");
|
|
break;
|
|
|
|
case MCS51_T2EX_LINE:
|
|
if (m_features & FEATURE_I8052)
|
|
{
|
|
if (GET_BIT(tr_state, MCS51_T2EX_LINE))
|
|
{
|
|
SET_EXF2(1);
|
|
m_t2ex_cnt++;
|
|
}
|
|
}
|
|
else
|
|
fatalerror("mcs51: Trying to set T2EX_LINE on a non I8052 type cpu.\n");
|
|
break;
|
|
|
|
case MCS51_RX_LINE: /* Serial Port Receive */
|
|
/* Is the enable flags for this interrupt set? */
|
|
if (state != CLEAR_LINE)
|
|
{
|
|
serial_receive();
|
|
}
|
|
break;
|
|
|
|
/* Power Fail Interrupt */
|
|
case DS5002FP_PFI_LINE:
|
|
if (m_features & FEATURE_DS5002FP)
|
|
{
|
|
/* Need cleared->active line transition? (Logical 1-0 Pulse on the line) - CLEAR->ASSERT Transition since INT1 active lo! */
|
|
if (GET_BIT(tr_state, MCS51_INT1_LINE))
|
|
SET_PFW(1);
|
|
}
|
|
else
|
|
fatalerror("mcs51: Trying to set DS5002FP_PFI_LINE on a non DS5002FP type cpu.\n");
|
|
break;
|
|
}
|
|
m_last_line_state = new_state;
|
|
}
|
|
|
|
/* Execute cycles - returns number of cycles actually run */
|
|
void mcs51_cpu_device::execute_run()
|
|
{
|
|
uint8_t op;
|
|
|
|
/* external interrupts may have been set since we last checked */
|
|
m_inst_cycles = 0;
|
|
check_irqs();
|
|
|
|
/* if in powerdown, just return */
|
|
if ((m_features & FEATURE_CMOS) && GET_PD)
|
|
{
|
|
m_icount = 0;
|
|
return;
|
|
}
|
|
|
|
m_icount -= m_inst_cycles;
|
|
burn_cycles(m_inst_cycles);
|
|
|
|
if ((m_features & FEATURE_CMOS) && GET_IDL)
|
|
{
|
|
do
|
|
{
|
|
/* burn the cycles */
|
|
m_icount--;
|
|
burn_cycles(1);
|
|
} while( m_icount > 0 );
|
|
return;
|
|
}
|
|
|
|
do
|
|
{
|
|
/* Read next opcode */
|
|
PPC = PC;
|
|
debugger_instruction_hook(this, PC);
|
|
op = m_direct->read_byte(PC++);
|
|
|
|
/* process opcode and count cycles */
|
|
m_inst_cycles = mcs51_cycles[op];
|
|
execute_op(op);
|
|
|
|
/* burn the cycles */
|
|
m_icount -= m_inst_cycles;
|
|
|
|
/* if in powerdown, just return */
|
|
if ((m_features & FEATURE_CMOS) && GET_PD)
|
|
return;
|
|
|
|
burn_cycles(m_inst_cycles);
|
|
|
|
/* decrement the timed access window */
|
|
if (m_features & FEATURE_DS5002FP)
|
|
m_ds5002fp.ta_window = (m_ds5002fp.ta_window ? (m_ds5002fp.ta_window - 1) : 0x00);
|
|
|
|
/* If the chip entered in idle mode, end the loop */
|
|
if ((m_features & FEATURE_CMOS) && GET_IDL)
|
|
return;
|
|
|
|
} while( m_icount > 0 );
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* MCS51/8051 Section
|
|
****************************************************************************/
|
|
|
|
void mcs51_cpu_device::sfr_write(size_t offset, uint8_t data)
|
|
{
|
|
/* update register */
|
|
assert(offset >= 0x80 && offset <= 0xff);
|
|
|
|
switch (offset)
|
|
{
|
|
case ADDR_P0: OUT(MCS51_PORT_P0,data); break;
|
|
case ADDR_P1: OUT(MCS51_PORT_P1,data); break;
|
|
case ADDR_P2: OUT(MCS51_PORT_P2,data); break;
|
|
case ADDR_P3: OUT(MCS51_PORT_P3,data); break;
|
|
case ADDR_SBUF: serial_transmit(data); break;
|
|
case ADDR_PSW: SET_PARITY(); break;
|
|
case ADDR_ACC: SET_PARITY(); break;
|
|
case ADDR_IP: update_irq_prio(data, 0); break;
|
|
/* R_SBUF = data; //This register is used only for "Receiving data coming in!" */
|
|
|
|
case ADDR_B:
|
|
case ADDR_SP:
|
|
case ADDR_DPL:
|
|
case ADDR_DPH:
|
|
case ADDR_PCON:
|
|
case ADDR_TCON:
|
|
case ADDR_TMOD:
|
|
case ADDR_IE:
|
|
case ADDR_TL0:
|
|
case ADDR_TL1:
|
|
case ADDR_TH0:
|
|
case ADDR_TH1:
|
|
case ADDR_SCON:
|
|
break;
|
|
default:
|
|
LOG(("mcs51 '%s': attemping to write to an invalid/non-implemented SFR address: %x at 0x%04x, data=%x\n", tag(), (uint32_t)offset,PC,data));
|
|
/* no write in this case according to manual */
|
|
return;
|
|
}
|
|
m_data->write_byte((size_t)offset | 0x100, data);
|
|
}
|
|
|
|
uint8_t mcs51_cpu_device::sfr_read(size_t offset)
|
|
{
|
|
assert(offset >= 0x80 && offset <= 0xff);
|
|
|
|
switch (offset)
|
|
{
|
|
/* Read/Write/Modify operations read the port latch ! */
|
|
/* Move to memory map */
|
|
case ADDR_P0: return RWM ? P0 : (P0 | m_forced_inputs[0]) & IN(MCS51_PORT_P0);
|
|
case ADDR_P1: return RWM ? P1 : (P1 | m_forced_inputs[1]) & IN(MCS51_PORT_P1);
|
|
case ADDR_P2: return RWM ? P2 : (P2 | m_forced_inputs[2]) & IN(MCS51_PORT_P2);
|
|
case ADDR_P3: return RWM ? P3 : (P3 | m_forced_inputs[3]) & IN(MCS51_PORT_P3);
|
|
|
|
case ADDR_PSW:
|
|
case ADDR_ACC:
|
|
case ADDR_B:
|
|
case ADDR_SP:
|
|
case ADDR_DPL:
|
|
case ADDR_DPH:
|
|
case ADDR_PCON:
|
|
case ADDR_TCON:
|
|
case ADDR_TMOD:
|
|
case ADDR_TL0:
|
|
case ADDR_TL1:
|
|
case ADDR_TH0:
|
|
case ADDR_TH1:
|
|
case ADDR_SCON:
|
|
case ADDR_SBUF:
|
|
case ADDR_IE:
|
|
case ADDR_IP:
|
|
return m_data->read_byte((size_t) offset | 0x100);
|
|
/* Illegal or non-implemented sfr */
|
|
default:
|
|
LOG(("mcs51 '%s': attemping to read an invalid/non-implemented SFR address: %x at 0x%04x\n", tag(), (uint32_t)offset,PC));
|
|
/* according to the manual, the read may return random bits */
|
|
return 0xff;
|
|
}
|
|
}
|
|
|
|
|
|
void mcs51_cpu_device::device_start()
|
|
{
|
|
m_program = &space(AS_PROGRAM);
|
|
m_direct = m_program->direct<0>();
|
|
m_data = &space(AS_DATA);
|
|
m_io = &space(AS_IO);
|
|
|
|
m_serial_rx_cb.resolve_safe(0);
|
|
m_serial_tx_cb.resolve_safe();
|
|
|
|
/* Save states */
|
|
|
|
save_item(NAME(m_ppc));
|
|
save_item(NAME(m_pc));
|
|
save_item(NAME(m_rwm) );
|
|
save_item(NAME(m_cur_irq_prio) );
|
|
save_item(NAME(m_last_line_state) );
|
|
save_item(NAME(m_t0_cnt) );
|
|
save_item(NAME(m_t1_cnt) );
|
|
save_item(NAME(m_t2_cnt) );
|
|
save_item(NAME(m_t2ex_cnt) );
|
|
save_item(NAME(m_recalc_parity) );
|
|
save_item(NAME(m_irq_prio) );
|
|
save_item(NAME(m_irq_active) );
|
|
save_item(NAME(m_ds5002fp.previous_ta) );
|
|
save_item(NAME(m_ds5002fp.ta_window) );
|
|
save_item(NAME(m_ds5002fp.range) );
|
|
save_item(NAME(m_uart.data_out));
|
|
save_item(NAME(m_uart.bits_to_send));
|
|
save_item(NAME(m_uart.smod_div));
|
|
save_item(NAME(m_uart.rx_clk));
|
|
save_item(NAME(m_uart.tx_clk));
|
|
save_item(NAME(m_uart.delay_cycles));
|
|
|
|
state_add( MCS51_PC, "PC", m_pc).formatstr("%04X");
|
|
state_add( MCS51_SP, "SP", SP).formatstr("%02X");
|
|
state_add( MCS51_PSW, "PSW", PSW).formatstr("%02X");
|
|
state_add( MCS51_ACC, "A", ACC).formatstr("%02X");
|
|
state_add( MCS51_B, "B", B).formatstr("%02X");
|
|
state_add<uint16_t>( MCS51_DPTR, "DPTR", [this](){ return DPTR; }, [this](uint16_t dp){ SET_DPTR(dp); }).formatstr("%04X");
|
|
state_add( MCS51_DPH, "DPH", DPH).noshow();
|
|
state_add( MCS51_DPL, "DPL", DPL).noshow();
|
|
state_add( MCS51_IE, "IE", IE).formatstr("%02X");
|
|
state_add<uint8_t>( MCS51_P0, "P0", [this](){ return P0; }, [this](uint8_t p){ SET_P0(p); }).formatstr("%02X");
|
|
state_add<uint8_t>( MCS51_P1, "P1", [this](){ return P1; }, [this](uint8_t p){ SET_P1(p); }).formatstr("%02X");
|
|
state_add<uint8_t>( MCS51_P2, "P2", [this](){ return P2; }, [this](uint8_t p){ SET_P2(p); }).formatstr("%02X");
|
|
state_add<uint8_t>( MCS51_P3, "P3", [this](){ return P3; }, [this](uint8_t p){ SET_P3(p); }).formatstr("%02X");
|
|
state_add( MCS51_R0, "R0", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R1, "R1", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R2, "R2", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R3, "R3", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R4, "R4", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R5, "R5", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R6, "R6", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_R7, "R7", m_rtemp).callimport().callexport().formatstr("%02X");
|
|
state_add( MCS51_RB, "RB", m_rtemp).mask(0x03).callimport().callexport().formatstr("%02X");
|
|
|
|
state_add( STATE_GENPC, "GENPC", m_pc ).noshow();
|
|
state_add( STATE_GENPCBASE, "CURPC", m_pc ).noshow();
|
|
state_add( STATE_GENFLAGS, "GENFLAGS", m_rtemp).formatstr("%8s").noshow();
|
|
|
|
m_icountptr = &m_icount;
|
|
}
|
|
|
|
|
|
void mcs51_cpu_device::state_import(const device_state_entry &entry)
|
|
{
|
|
switch (entry.index())
|
|
{
|
|
case MCS51_R0:
|
|
case MCS51_R1:
|
|
case MCS51_R2:
|
|
case MCS51_R3:
|
|
case MCS51_R4:
|
|
case MCS51_R5:
|
|
case MCS51_R6:
|
|
case MCS51_R7:
|
|
SET_REG( entry.index() - MCS51_R0, m_rtemp );
|
|
break;
|
|
|
|
case MCS51_RB:
|
|
SET_RS( m_rtemp );
|
|
break;
|
|
|
|
default:
|
|
fatalerror("CPU_IMPORT_STATE(mcs48) called for unexpected value\n");
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::state_export(const device_state_entry &entry)
|
|
{
|
|
switch (entry.index())
|
|
{
|
|
case MCS51_R0:
|
|
case MCS51_R1:
|
|
case MCS51_R2:
|
|
case MCS51_R3:
|
|
case MCS51_R4:
|
|
case MCS51_R5:
|
|
case MCS51_R6:
|
|
case MCS51_R7:
|
|
m_rtemp = R_REG(entry.index() - MCS51_R0);
|
|
break;
|
|
|
|
case MCS51_RB:
|
|
m_rtemp = ((PSW & 0x18)>>3);
|
|
break;
|
|
|
|
default:
|
|
fatalerror("CPU_EXPORT_STATE(mcs51) called for unexpected value\n");
|
|
}
|
|
}
|
|
|
|
void mcs51_cpu_device::state_string_export(const device_state_entry &entry, std::string &str) const
|
|
{
|
|
switch (entry.index())
|
|
{
|
|
case STATE_GENFLAGS:
|
|
str = string_format("%c%c%c%c%c%c%c%c",
|
|
PSW & 0x80 ? 'C':'.',
|
|
PSW & 0x40 ? 'A':'.',
|
|
PSW & 0x20 ? 'F':'.',
|
|
PSW & 0x10 ? '0':'.',
|
|
PSW & 0x08 ? '1':'.',
|
|
PSW & 0x04 ? 'V':'.',
|
|
PSW & 0x02 ? '?':'.',
|
|
PSW & 0x01 ? 'P':'.');
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Reset registers to the initial values */
|
|
void mcs51_cpu_device::device_reset()
|
|
{
|
|
m_last_line_state = 0;
|
|
m_t0_cnt = 0;
|
|
m_t1_cnt = 0;
|
|
m_t2_cnt = 0;
|
|
m_t2ex_cnt = 0;
|
|
/* Flag as NO IRQ in Progress */
|
|
m_irq_active = 0;
|
|
m_cur_irq_prio = -1;
|
|
|
|
/* these are all defined reset states */
|
|
PC = 0;
|
|
SP = 0x7;
|
|
SET_PSW(0);
|
|
SET_ACC(0);
|
|
DPH = 0;
|
|
DPL = 0;
|
|
B = 0;
|
|
IP = 0;
|
|
update_irq_prio(IP, 0);
|
|
IE = 0;
|
|
SCON = 0;
|
|
TCON = 0;
|
|
TMOD = 0;
|
|
PCON = 0;
|
|
TH1 = 0;
|
|
TH0 = 0;
|
|
TL1 = 0;
|
|
TL0 = 0;
|
|
/* set the port configurations to all 1's */
|
|
SET_P3(0xff);
|
|
SET_P2(0xff);
|
|
SET_P1(0xff);
|
|
SET_P0(0xff);
|
|
|
|
/* 8052 Only registers */
|
|
if (m_features & FEATURE_I8052)
|
|
{
|
|
T2CON = 0;
|
|
RCAP2L = 0;
|
|
RCAP2H = 0;
|
|
TL2 = 0;
|
|
TH2 = 0;
|
|
}
|
|
|
|
/* 80C52 Only registers */
|
|
if (m_features & FEATURE_I80C52)
|
|
{
|
|
IPH = 0;
|
|
update_irq_prio(IP, IPH);
|
|
SADDR = 0;
|
|
SADEN = 0;
|
|
}
|
|
|
|
/* DS5002FP Only registers */
|
|
if (m_features & FEATURE_DS5002FP)
|
|
{
|
|
// set initial values (some of them are set using the bootstrap loader)
|
|
PCON = 0;
|
|
MCON = m_sfr_ram[ADDR_MCON-0x80];
|
|
RPCTL = m_sfr_ram[ADDR_RPCTL-0x80];
|
|
RPS = 0;
|
|
RNR = 0;
|
|
CRCR = m_sfr_ram[ADDR_CRCR-0x80];
|
|
CRCL = 0;
|
|
CRCH = 0;
|
|
TA = 0;
|
|
|
|
// set internal CPU state
|
|
m_ds5002fp.previous_ta = 0;
|
|
m_ds5002fp.ta_window = 0;
|
|
m_ds5002fp.range = (GET_RG1 << 1) | GET_RG0;
|
|
}
|
|
|
|
m_uart.data_out = 0;
|
|
m_uart.rx_clk = 0;
|
|
m_uart.tx_clk = 0;
|
|
m_uart.bits_to_send = 0;
|
|
m_uart.delay_cycles = 0;
|
|
m_uart.smod_div = 0;
|
|
|
|
m_recalc_parity = 0;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* 8052 Section
|
|
****************************************************************************/
|
|
|
|
void i8052_device::sfr_write(size_t offset, uint8_t data)
|
|
{
|
|
switch (offset)
|
|
{
|
|
/* 8052 family specific */
|
|
case ADDR_T2CON:
|
|
case ADDR_RCAP2L:
|
|
case ADDR_RCAP2H:
|
|
case ADDR_TL2:
|
|
case ADDR_TH2:
|
|
m_data->write_byte((size_t) offset | 0x100, data);
|
|
break;
|
|
|
|
default:
|
|
mcs51_cpu_device::sfr_write(offset, data);
|
|
}
|
|
}
|
|
|
|
uint8_t i8052_device::sfr_read(size_t offset)
|
|
{
|
|
switch (offset)
|
|
{
|
|
/* 8052 family specific */
|
|
case ADDR_T2CON:
|
|
case ADDR_RCAP2L:
|
|
case ADDR_RCAP2H:
|
|
case ADDR_TL2:
|
|
case ADDR_TH2:
|
|
return m_data->read_byte((size_t) offset | 0x100);
|
|
default:
|
|
return mcs51_cpu_device::sfr_read(offset);
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* 80C52 Section
|
|
****************************************************************************/
|
|
|
|
void i80c52_device::sfr_write(size_t offset, uint8_t data)
|
|
{
|
|
switch (offset)
|
|
{
|
|
/* 80c52 family specific */
|
|
case ADDR_IP:
|
|
update_irq_prio(data, IPH);
|
|
break;
|
|
case ADDR_IPH:
|
|
update_irq_prio(IP, data);
|
|
break;
|
|
case ADDR_SADDR:
|
|
case ADDR_SADEN:
|
|
break;
|
|
|
|
default:
|
|
i8052_device::sfr_write(offset, data);
|
|
return;
|
|
}
|
|
m_data->write_byte((size_t) offset | 0x100, data);
|
|
}
|
|
|
|
uint8_t i80c52_device::sfr_read(size_t offset)
|
|
{
|
|
switch (offset)
|
|
{
|
|
/* 80c52 family specific */
|
|
case ADDR_IPH:
|
|
case ADDR_SADDR:
|
|
case ADDR_SADEN:
|
|
return m_data->read_byte((size_t) offset | 0x100);
|
|
default:
|
|
return i8052_device::sfr_read(offset);
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* DS5002FP Section
|
|
****************************************************************************/
|
|
|
|
|
|
#define DS5_LOGW(a, d) LOG(("ds5002fp '%s': write to " # a " register at 0x%04x, data=%x\n", tag(), PC, d))
|
|
#define DS5_LOGR(a, d) LOG(("ds5002fp '%s': read from " # a " register at 0x%04x\n", tag(), PC))
|
|
|
|
uint8_t mcs51_cpu_device::ds5002fp_protected(size_t offset, uint8_t data, uint8_t ta_mask, uint8_t mask)
|
|
{
|
|
uint8_t is_timed_access;
|
|
|
|
is_timed_access = (m_ds5002fp.ta_window > 0) && (TA == 0x55);
|
|
if (is_timed_access)
|
|
{
|
|
ta_mask = 0xff;
|
|
}
|
|
data = (m_sfr_ram[offset] & (~ta_mask)) | (data & ta_mask);
|
|
return (m_sfr_ram[offset] & (~mask)) | (data & mask);
|
|
}
|
|
|
|
void ds5002fp_device::sfr_write(size_t offset, uint8_t data)
|
|
{
|
|
switch (offset)
|
|
{
|
|
case ADDR_TA:
|
|
m_ds5002fp.previous_ta = TA;
|
|
/* init the time window after having wrote 0xaa */
|
|
if ((data == 0xaa) && (m_ds5002fp.ta_window == 0))
|
|
{
|
|
m_ds5002fp.ta_window = 6; /* 4*12 + 2*12 */
|
|
LOG(("ds5002fp '%s': TA window initiated at 0x%04x\n", tag(), PC));
|
|
}
|
|
break;
|
|
case ADDR_MCON: data = ds5002fp_protected(ADDR_MCON, data, 0x0f, 0xf7); DS5_LOGW(MCON, data); break;
|
|
case ADDR_RPCTL: data = ds5002fp_protected(ADDR_RPCTL, data, 0xef, 0xfe); DS5_LOGW(RPCTL, data); break;
|
|
case ADDR_CRCR: data = ds5002fp_protected(ADDR_CRCR, data, 0xff, 0x0f); DS5_LOGW(CRCR, data); break;
|
|
case ADDR_PCON: data = ds5002fp_protected(ADDR_PCON, data, 0xb9, 0xff); break;
|
|
case ADDR_IP: data = ds5002fp_protected(ADDR_IP, data, 0x7f, 0xff); break;
|
|
case ADDR_CRCL: DS5_LOGW(CRCL, data); break;
|
|
case ADDR_CRCH: DS5_LOGW(CRCH, data); break;
|
|
case ADDR_RNR: DS5_LOGW(RNR, data); break;
|
|
case ADDR_RPS: DS5_LOGW(RPS, data); break;
|
|
default:
|
|
mcs51_cpu_device::sfr_write(offset, data);
|
|
return;
|
|
}
|
|
m_data->write_byte((size_t) offset | 0x100, data);
|
|
}
|
|
|
|
uint8_t ds5002fp_device::sfr_read(size_t offset)
|
|
{
|
|
switch (offset)
|
|
{
|
|
case ADDR_CRCR: DS5_LOGR(CRCR, data); break;
|
|
case ADDR_CRCL: DS5_LOGR(CRCL, data); break;
|
|
case ADDR_CRCH: DS5_LOGR(CRCH, data); break;
|
|
case ADDR_MCON: DS5_LOGR(MCON, data); break;
|
|
case ADDR_TA: DS5_LOGR(TA, data); break;
|
|
case ADDR_RNR: DS5_LOGR(RNR, data); break;
|
|
case ADDR_RPCTL: DS5_LOGR(RPCTL, data); return 0x80; break; /* touchgo stalls unless bit 7 is set, why? documentation is unclear */
|
|
case ADDR_RPS: DS5_LOGR(RPS, data); break;
|
|
case ADDR_PCON:
|
|
SET_PFW(0); /* reset PFW flag */
|
|
return mcs51_cpu_device::sfr_read(offset);
|
|
default:
|
|
return mcs51_cpu_device::sfr_read(offset);
|
|
}
|
|
return m_data->read_byte((size_t) offset | 0x100);
|
|
}
|
|
|
|
/*
|
|
Documentation states that having the battery connected "maintains the internal scratchpad RAM" and "certain SFRs"
|
|
(although it isn't clear exactly which SFRs except for those explicitly mentioned)
|
|
*/
|
|
|
|
void ds5002fp_device::nvram_default()
|
|
{
|
|
memset( m_scratchpad, 0, 0x80 );
|
|
memset( m_sfr_ram, 0, 0x80 );
|
|
|
|
int expected_bytes = 0x80 + 0x80;
|
|
|
|
if (!m_region.found())
|
|
{
|
|
logerror( "ds5002fp_device region not found\n" );
|
|
}
|
|
else if( m_region->bytes() != expected_bytes )
|
|
{
|
|
logerror( "ds5002fp_device region length 0x%x expected 0x%x\n", m_region->bytes(), expected_bytes );
|
|
}
|
|
else
|
|
{
|
|
uint8_t *region = m_region->base();
|
|
|
|
memcpy( m_scratchpad, region, 0x80 ); region += 0x80;
|
|
memcpy( m_sfr_ram, region, 0x80 ); region += 0x80;
|
|
/* does anything else need storing? any registers that aren't in sfr ram?
|
|
It isn't clear if the various initial MCON registers etc. are just stored in sfr ram
|
|
or if the DS5002FP stores them elsewhere and the bootstrap copies them */
|
|
}
|
|
}
|
|
|
|
void ds5002fp_device::nvram_read( emu_file &file )
|
|
{
|
|
file.read( m_scratchpad, 0x80 );
|
|
file.read( m_sfr_ram, 0x80 );
|
|
}
|
|
|
|
void ds5002fp_device::nvram_write( emu_file &file )
|
|
{
|
|
file.write( m_scratchpad, 0x80 );
|
|
file.write( m_sfr_ram, 0x80 );
|
|
}
|
|
|
|
util::disasm_interface *mcs51_cpu_device::create_disassembler()
|
|
{
|
|
return new i8051_disassembler;
|
|
}
|
|
|
|
util::disasm_interface *i8052_device::create_disassembler()
|
|
{
|
|
return new i8052_disassembler;
|
|
}
|
|
|
|
util::disasm_interface *i80c31_device::create_disassembler()
|
|
{
|
|
return new i80c51_disassembler;
|
|
}
|
|
|
|
util::disasm_interface *i80c51_device::create_disassembler()
|
|
{
|
|
return new i80c51_disassembler;
|
|
}
|
|
|
|
util::disasm_interface *i80c52_device::create_disassembler()
|
|
{
|
|
return new i80c52_disassembler;
|
|
}
|
|
|
|
util::disasm_interface *ds5002fp_device::create_disassembler()
|
|
{
|
|
return new ds5002fp_disassembler;
|
|
}
|