tmpz84c015: Add WDT emulation

This commit is contained in:
AJR 2020-12-07 14:57:31 -05:00
parent 1c13556555
commit cc954c2ade
2 changed files with 94 additions and 7 deletions

View File

@ -8,7 +8,7 @@
TODO:
- SIO configuration, or should that be up to the driver?
- CGC (clock generator/controller)
- WDT (watchdog timer)
- Halt modes
***************************************************************************/
@ -17,22 +17,26 @@
DEFINE_DEVICE_TYPE(TMPZ84C015, tmpz84c015_device, "tmpz84c015", "Toshiba TMPZ84C015")
void tmpz84c015_device::tmpz84c015_internal_io_map(address_map &map)
void tmpz84c015_device::internal_io_map(address_map &map)
{
map(0x10, 0x13).mirror(0xff00).rw(m_ctc, FUNC(z80ctc_device::read), FUNC(z80ctc_device::write));
map(0x18, 0x1b).mirror(0xff00).rw(m_sio, FUNC(z80sio_device::ba_cd_r), FUNC(z80sio_device::ba_cd_w));
map(0x1c, 0x1f).mirror(0xff00).rw(m_pio, FUNC(z80pio_device::read_alt), FUNC(z80pio_device::write_alt));
map(0xf0, 0xf0).mirror(0xff00).rw(FUNC(tmpz84c015_device::wdtmr_r), FUNC(tmpz84c015_device::wdtmr_w));
map(0xf1, 0xf1).mirror(0xff00).w(FUNC(tmpz84c015_device::wdtcr_w));
map(0xf4, 0xf4).mirror(0xff00).w(FUNC(tmpz84c015_device::irq_priority_w));
}
tmpz84c015_device::tmpz84c015_device(const machine_config &mconfig, const char *tag, device_t *owner, uint32_t clock) :
z80_device(mconfig, TMPZ84C015, tag, owner, clock),
m_io_space_config( "io", ENDIANNESS_LITTLE, 8, 16, 0, address_map_constructor(FUNC(tmpz84c015_device::tmpz84c015_internal_io_map), this)),
m_io_space_config( "io", ENDIANNESS_LITTLE, 8, 16, 0, address_map_constructor(FUNC(tmpz84c015_device::internal_io_map), this)),
m_ctc(*this, "tmpz84c015_ctc"),
m_sio(*this, "tmpz84c015_sio"),
m_pio(*this, "tmpz84c015_pio"),
m_irq_priority(-1), // !
m_wdtmr(0xfb),
m_watchdog_timer(nullptr),
m_out_txda_cb(*this),
m_out_dtra_cb(*this),
@ -59,7 +63,9 @@ tmpz84c015_device::tmpz84c015_device(const machine_config &mconfig, const char *
m_in_pb_cb(*this),
m_out_pb_cb(*this),
m_out_brdy_cb(*this)
m_out_brdy_cb(*this),
m_wdtout_cb(*this)
{
}
@ -107,8 +113,14 @@ void tmpz84c015_device::device_start()
m_out_pb_cb.resolve_safe();
m_out_brdy_cb.resolve_safe();
m_wdtout_cb.resolve();
// setup watchdog timer
m_watchdog_timer = machine().scheduler().timer_alloc(timer_expired_delegate(FUNC(tmpz84c015_device::watchdog_timeout), this));
// register for save states
save_item(NAME(m_irq_priority));
save_item(NAME(m_wdtmr));
}
@ -119,6 +131,9 @@ void tmpz84c015_device::device_start()
void tmpz84c015_device::device_reset()
{
irq_priority_w(0);
m_wdtmr = 0xfb;
watchdog_clear();
z80_device::device_reset();
}
@ -176,6 +191,65 @@ void tmpz84c015_device::irq_priority_w(uint8_t data)
}
}
uint8_t tmpz84c015_device::wdtmr_r()
{
return m_wdtmr;
}
void tmpz84c015_device::wdtmr_w(uint8_t data)
{
if ((data & 0x07) != 0x03)
logerror("%s: Writing %d%d%d to WDTMR reserved bits\n", machine().describe_context(), BIT(data, 2), BIT(data, 1), BIT(data, 0));
// check for watchdog timer enable
uint8_t old_data = std::exchange(m_wdtmr, data);
if (!BIT(old_data, 7) && BIT(data, 7))
watchdog_clear();
}
void tmpz84c015_device::wdtcr_w(uint8_t data)
{
// write specific values only
switch (data)
{
case 0x4e:
watchdog_clear();
break;
case 0xb1:
// WDTER must be cleared first
if (!BIT(m_wdtmr, 7))
{
logerror("%s: Watchdog timer disabled\n", machine().describe_context());
m_watchdog_timer->adjust(attotime::never);
}
break;
default:
logerror("%s: Writing %02X to WDTCR\n", machine().describe_context(), data);
break;
}
}
void tmpz84c015_device::watchdog_clear()
{
if (!m_wdtout_cb.isnull())
m_wdtout_cb(CLEAR_LINE);
if (BIT(m_wdtmr, 7))
m_watchdog_timer->adjust(cycles_to_attotime(0x10000 << (BIT(m_wdtmr, 5, 2) * 2)));
}
TIMER_CALLBACK_MEMBER(tmpz84c015_device::watchdog_timeout)
{
if (!m_wdtout_cb.isnull())
m_wdtout_cb(ASSERT_LINE);
else
logerror("Watchdog timeout\n");
}
void tmpz84c015_device::device_add_mconfig(machine_config &config)
{
/* basic machine hardware */

View File

@ -47,6 +47,8 @@ public:
auto out_rxdrqb_callback() { return m_out_rxdrqb_cb.bind(); }
auto out_txdrqb_callback() { return m_out_txdrqb_cb.bind(); }
auto wdtout_cb() { return m_wdtout_cb.bind(); }
// CTC callbacks
template<unsigned N> auto zc_callback() { return m_zc_cb[N].bind(); }
@ -108,9 +110,6 @@ public:
/////////////////////////////////////////////////////////
void irq_priority_w(uint8_t data);
void tmpz84c015_internal_io_map(address_map &map);
protected:
// device-level overrides
virtual void device_add_mconfig(machine_config &config) override;
@ -130,6 +129,8 @@ private:
// internal state
uint8_t m_irq_priority;
uint8_t m_wdtmr;
emu_timer *m_watchdog_timer;
// callbacks
devcb_write_line m_out_txda_cb;
@ -159,6 +160,18 @@ private:
devcb_write8 m_out_pb_cb;
devcb_write_line m_out_brdy_cb;
devcb_write_line m_wdtout_cb;
uint8_t wdtmr_r();
void wdtmr_w(uint8_t data);
void wdtcr_w(uint8_t data);
void watchdog_clear();
TIMER_CALLBACK_MEMBER(watchdog_timeout);
void irq_priority_w(uint8_t data);
void internal_io_map(address_map &map);
DECLARE_WRITE_LINE_MEMBER( out_txda_cb_trampoline_w ) { m_out_txda_cb(state); }
DECLARE_WRITE_LINE_MEMBER( out_dtra_cb_trampoline_w ) { m_out_dtra_cb(state); }
DECLARE_WRITE_LINE_MEMBER( out_rtsa_cb_trampoline_w ) { m_out_rtsa_cb(state); }