mirror of
https://github.com/holub/mame
synced 2025-05-10 00:01:52 +03:00
(MESS) Half-way modernization for TI-990, removing usage of tokens.
Splitting driver to a video and simple terminal variant. (nw)
This commit is contained in:
parent
b57f9d5974
commit
fd38292291
@ -28,6 +28,7 @@
|
||||
Note that only interrupt levels 3-7 are supported by the board (8-15 are not wired).
|
||||
|
||||
TODO:
|
||||
* Split into two machines for either ASR or VDT
|
||||
* finish ASR emulation
|
||||
* programmer panel
|
||||
* emulate other devices: card reader, printer
|
||||
@ -36,17 +37,13 @@ TODO:
|
||||
Rewritten by Michael Zapf 2013
|
||||
*/
|
||||
|
||||
/*if 1, use 911 VDT; if 0, use 733 ASR */
|
||||
#define VIDEO_911 0
|
||||
|
||||
#include "emu.h"
|
||||
#include "cpu/tms9900/tms9900.h"
|
||||
#if VIDEO_911
|
||||
|
||||
#include "video/911_vdt.h"
|
||||
#include "sound/beep.h"
|
||||
#else
|
||||
#include "video/733_asr.h"
|
||||
#endif
|
||||
|
||||
#include "imagedev/flopdrv.h"
|
||||
#include "machine/ti99/990_dk.h"
|
||||
|
||||
@ -55,23 +52,26 @@ class ti990_4_state : public driver_device
|
||||
{
|
||||
public:
|
||||
ti990_4_state(const machine_config &mconfig, device_type type, const char *tag)
|
||||
: driver_device(mconfig, type, tag) ,
|
||||
m_maincpu(*this, "maincpu") { }
|
||||
: driver_device(mconfig, type, tag),
|
||||
m_maincpu(*this, "maincpu"),
|
||||
m_fd800(*this, "fd800") { }
|
||||
|
||||
device_t *m_terminal;
|
||||
DECLARE_READ8_MEMBER( panel_read );
|
||||
DECLARE_WRITE8_MEMBER( panel_write );
|
||||
DECLARE_WRITE8_MEMBER( external_operation );
|
||||
DECLARE_READ8_MEMBER( interrupt_level );
|
||||
DECLARE_WRITE_LINE_MEMBER( fd_interrupt );
|
||||
|
||||
DECLARE_DRIVER_INIT(ti990_4);
|
||||
DECLARE_DRIVER_INIT(ti990_4v);
|
||||
|
||||
DECLARE_MACHINE_RESET(ti990_4);
|
||||
|
||||
virtual void video_start();
|
||||
UINT32 screen_update_ti990_4(screen_device &screen, bitmap_ind16 &bitmap, const rectangle &cliprect);
|
||||
INTERRUPT_GEN_MEMBER(ti990_4_line_interrupt);
|
||||
void idle_callback(int state);
|
||||
required_device<tms9900_device> m_maincpu;
|
||||
|
||||
private:
|
||||
void hold_load();
|
||||
@ -87,6 +87,12 @@ private:
|
||||
void set_int6(int state);
|
||||
void set_int7(int state);
|
||||
bool m_ckon_state;
|
||||
|
||||
bool m_video;
|
||||
|
||||
// Connected devices
|
||||
required_device<tms9900_device> m_maincpu;
|
||||
required_device<fd800_legacy_device> m_fd800;
|
||||
};
|
||||
|
||||
enum
|
||||
@ -96,16 +102,18 @@ enum
|
||||
|
||||
void ti990_4_state::hold_load()
|
||||
{
|
||||
m_maincpu->set_input_line(INPUT_LINE_NMI, ASSERT_LINE);
|
||||
m_maincpu->set_input_line(INT_9900_LOAD, ASSERT_LINE);
|
||||
logerror("ti990_4: Triggering LOAD interrupt\n");
|
||||
m_nmi_timer->adjust(attotime::from_msec(100));
|
||||
}
|
||||
|
||||
/*
|
||||
NMI timer callback
|
||||
LOAD interrupt trigger callback
|
||||
*/
|
||||
void ti990_4_state::device_timer(emu_timer &timer, device_timer_id id, int param, void *ptr)
|
||||
{
|
||||
m_maincpu->set_input_line(INPUT_LINE_NMI, CLEAR_LINE);
|
||||
m_maincpu->set_input_line(INT_9900_LOAD, CLEAR_LINE);
|
||||
logerror("ti990_4: Released LOAD interrupt\n");
|
||||
}
|
||||
|
||||
READ8_MEMBER( ti990_4_state::panel_read )
|
||||
@ -131,7 +139,8 @@ void ti990_4_state::set_int_line(int line, int state)
|
||||
{
|
||||
m_int_level = 0;
|
||||
while ((m_intlines & (1 << m_int_level))==0) m_int_level++;
|
||||
m_maincpu->set_input_line(INT_9900_INTREQ, ASSERT_LINE); /* interrupt it, baby */
|
||||
logerror("ti990_4: Setting int level to %x\n", m_int_level);
|
||||
m_maincpu->set_input_line(INT_9900_INTREQ, ASSERT_LINE);
|
||||
}
|
||||
else
|
||||
m_maincpu->set_input_line(INT_9900_INTREQ, CLEAR_LINE);
|
||||
@ -162,63 +171,65 @@ void ti990_4_state::line_interrupt()
|
||||
if (m_ckon_state) set_int_line(5, ASSERT_LINE);
|
||||
}
|
||||
|
||||
/*
|
||||
Callback from the floppy controller.
|
||||
*/
|
||||
WRITE_LINE_MEMBER(ti990_4_state::fd_interrupt)
|
||||
{
|
||||
set_int7(state);
|
||||
}
|
||||
|
||||
INTERRUPT_GEN_MEMBER(ti990_4_state::ti990_4_line_interrupt)
|
||||
{
|
||||
#if VIDEO_911
|
||||
vdt911_keyboard(m_terminal);
|
||||
#else
|
||||
asr733_keyboard(m_terminal);
|
||||
#endif
|
||||
if (m_video)
|
||||
vdt911_keyboard(m_terminal);
|
||||
else
|
||||
asr733_keyboard(m_terminal);
|
||||
|
||||
line_interrupt();
|
||||
}
|
||||
|
||||
#if VIDEO_911
|
||||
|
||||
/*
|
||||
TI990/4 video emulation.
|
||||
|
||||
We emulate a single VDT911 CRT terminal.
|
||||
*/
|
||||
|
||||
void ti990_vdt_int(running_machine &machine, int state)
|
||||
{
|
||||
// set_int3
|
||||
}
|
||||
|
||||
static const vdt911_init_params_t vdt911_intf =
|
||||
{
|
||||
char_1920,
|
||||
vdt911_model_US,
|
||||
ti990_set_int3
|
||||
ti990_vdt_int
|
||||
};
|
||||
|
||||
void ti990_4_state::video_start()
|
||||
{
|
||||
m_terminal = machine().device("vdt911");
|
||||
if (m_video)
|
||||
m_terminal = machine().device("vdt911");
|
||||
else
|
||||
m_terminal = machine().device("asr733");
|
||||
}
|
||||
|
||||
UINT32 ti990_4_state::screen_update_ti990_4(screen_device &screen, bitmap_ind16 &bitmap, const rectangle &cliprect)
|
||||
{
|
||||
vdt911_refresh(m_terminal, bitmap, cliprect, 0, 0);
|
||||
if (m_video)
|
||||
vdt911_refresh(m_terminal, bitmap, cliprect, 0, 0);
|
||||
else
|
||||
asr733_refresh(m_terminal, bitmap, 0, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#else
|
||||
|
||||
static const asr733_init_params_t asr733_intf =
|
||||
{
|
||||
// set_int6
|
||||
};
|
||||
|
||||
void ti990_4_state::video_start()
|
||||
{
|
||||
m_terminal = machine().device("asr733");
|
||||
}
|
||||
|
||||
UINT32 ti990_4_state::screen_update_ti990_4(screen_device &screen, bitmap_ind16 &bitmap, const rectangle &cliprect)
|
||||
{
|
||||
asr733_refresh(m_terminal, bitmap, 0, 0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
WRITE8_MEMBER( ti990_4_state::external_operation )
|
||||
{
|
||||
static const char* extop[8] = { "inv1", "inv2", "IDLE", "RSET", "inv3", "CKON", "CKOF", "LREX" };
|
||||
@ -287,21 +298,28 @@ ADDRESS_MAP_END
|
||||
*/
|
||||
|
||||
static ADDRESS_MAP_START(cru_map, AS_IO, 8, ti990_4_state )
|
||||
#if VIDEO_911
|
||||
AM_RANGE(0x10, 0x11) AM_DEVREAD_LEGACY("vdt911", vdt911_cru_r)
|
||||
AM_RANGE(0x80, 0x8f) AM_DEVWRITE_LEGACY("vdt911", vdt911_cru_w)
|
||||
#else
|
||||
AM_RANGE(0x00, 0x01) AM_DEVREAD_LEGACY("asr733", asr733_cru_r)
|
||||
AM_RANGE(0x00, 0x0f) AM_DEVWRITE_LEGACY("asr733", asr733_cru_w)
|
||||
#endif
|
||||
|
||||
AM_RANGE(0x08, 0x0b) AM_READ_LEGACY(fd800_cru_r)
|
||||
AM_RANGE(0x40, 0x5f) AM_WRITE_LEGACY(fd800_cru_w)
|
||||
AM_RANGE(0x08, 0x0b) AM_DEVREAD( "fd800", fd800_legacy_device, cru_r )
|
||||
AM_RANGE(0x40, 0x5f) AM_DEVWRITE( "fd800", fd800_legacy_device, cru_w )
|
||||
|
||||
AM_RANGE(0x1fe, 0x1ff) AM_READ( panel_read )
|
||||
AM_RANGE(0xff0, 0xfff) AM_WRITE( panel_write )
|
||||
ADDRESS_MAP_END
|
||||
|
||||
static ADDRESS_MAP_START(cru_map_v, AS_IO, 8, ti990_4_state )
|
||||
AM_RANGE(0x10, 0x11) AM_DEVREAD_LEGACY("vdt911", vdt911_cru_r)
|
||||
AM_RANGE(0x80, 0x8f) AM_DEVWRITE_LEGACY("vdt911", vdt911_cru_w)
|
||||
|
||||
AM_RANGE(0x08, 0x0b) AM_DEVREAD( "fd800", fd800_legacy_device, cru_r )
|
||||
AM_RANGE(0x40, 0x5f) AM_DEVWRITE( "fd800", fd800_legacy_device, cru_w )
|
||||
|
||||
AM_RANGE(0x1fe, 0x1ff) AM_READ( panel_read )
|
||||
AM_RANGE(0xff0, 0xfff) AM_WRITE( panel_write )
|
||||
ADDRESS_MAP_END
|
||||
|
||||
|
||||
static const floppy_interface ti990_4_floppy_interface =
|
||||
{
|
||||
DEVCB_NULL,
|
||||
@ -309,7 +327,7 @@ static const floppy_interface ti990_4_floppy_interface =
|
||||
DEVCB_NULL,
|
||||
DEVCB_NULL,
|
||||
DEVCB_NULL,
|
||||
FLOPPY_STANDARD_5_25_DSHD,
|
||||
FLOPPY_STANDARD_8_DSSD,
|
||||
LEGACY_FLOPPY_OPTIONS_NAME(fd800),
|
||||
NULL,
|
||||
NULL
|
||||
@ -331,49 +349,80 @@ MACHINE_RESET_MEMBER(ti990_4_state,ti990_4)
|
||||
hold_load();
|
||||
reset_int_lines();
|
||||
m_ckon_state = false;
|
||||
// fd800_machine_init(machine(), ti990_4_state::set_int7);
|
||||
m_maincpu->set_ready(ASSERT_LINE);
|
||||
}
|
||||
|
||||
DRIVER_INIT_MEMBER(ti990_4_state, ti990_4)
|
||||
{
|
||||
m_video = false;
|
||||
m_nmi_timer = timer_alloc(NMI_TIMER_ID);
|
||||
asr733_init(machine());
|
||||
}
|
||||
|
||||
DRIVER_INIT_MEMBER(ti990_4_state, ti990_4v)
|
||||
{
|
||||
m_video = true;
|
||||
m_nmi_timer = timer_alloc(NMI_TIMER_ID);
|
||||
vdt911_init(machine());
|
||||
}
|
||||
|
||||
static INPUT_PORTS_START(ti990_4)
|
||||
ASR733_KEY_PORTS
|
||||
INPUT_PORTS_END
|
||||
|
||||
static INPUT_PORTS_START(ti990_4v)
|
||||
VDT911_KEY_PORTS
|
||||
INPUT_PORTS_END
|
||||
|
||||
static MACHINE_CONFIG_START( ti990_4, ti990_4_state )
|
||||
/* basic machine hardware */
|
||||
/* TMS9900 CPU @ 3.0(???) MHz */
|
||||
MCFG_TMS99xx_ADD("maincpu", TMS9900, 3000000, memmap, cru_map, cpuconf)
|
||||
MCFG_CPU_PERIODIC_INT_DRIVER(ti990_4_state, ti990_4_line_interrupt, 120/*or TIME_IN_HZ(100) in Europe*/)
|
||||
|
||||
MCFG_CPU_PERIODIC_INT_DRIVER(ti990_4_state, ti990_4_line_interrupt, 120/*or TIME_IN_HZ(100) in Europe*/)
|
||||
MCFG_MACHINE_RESET_OVERRIDE(ti990_4_state, ti990_4 )
|
||||
|
||||
/* video hardware - we emulate a single 911 vdt display */
|
||||
MCFG_SCREEN_ADD("screen", RASTER)
|
||||
MCFG_SCREEN_REFRESH_RATE(60)
|
||||
MCFG_SCREEN_VBLANK_TIME(ATTOSECONDS_IN_USEC(2500)) /* not accurate */
|
||||
MCFG_SCREEN_UPDATE_DRIVER(ti990_4_state, screen_update_ti990_4)
|
||||
#if VIDEO_911
|
||||
MCFG_SCREEN_SIZE(560, 280)
|
||||
MCFG_SCREEN_VISIBLE_AREA(0, 560-1, 0, /*250*/280-1)
|
||||
MCFG_SCREEN_PALETTE("vdt911:palette")
|
||||
#else
|
||||
|
||||
MCFG_SCREEN_SIZE(640, 480)
|
||||
MCFG_SCREEN_VISIBLE_AREA(0, 640-1, 0, 480-1)
|
||||
MCFG_SCREEN_PALETTE("asr733:palette")
|
||||
#endif
|
||||
|
||||
#if VIDEO_911
|
||||
MCFG_VDT911_VIDEO_ADD("vdt911", vdt911_intf)
|
||||
#else
|
||||
MCFG_ASR733_VIDEO_ADD("asr733", asr733_intf)
|
||||
#endif
|
||||
|
||||
#if VIDEO_911
|
||||
/* 911 VDT has a beep tone generator */
|
||||
// Floppy controller
|
||||
MCFG_FD800_ADD("fd800", WRITELINE(ti990_4_state, fd_interrupt))
|
||||
MCFG_LEGACY_FLOPPY_4_DRIVES_ADD(ti990_4_floppy_interface)
|
||||
MACHINE_CONFIG_END
|
||||
|
||||
static MACHINE_CONFIG_START( ti990_4v, ti990_4_state )
|
||||
/* basic machine hardware */
|
||||
/* TMS9900 CPU @ 3.0(???) MHz */
|
||||
MCFG_TMS99xx_ADD("maincpu", TMS9900, 3000000, memmap, cru_map_v, cpuconf)
|
||||
|
||||
MCFG_CPU_PERIODIC_INT_DRIVER(ti990_4_state, ti990_4_line_interrupt, 120/*or TIME_IN_HZ(100) in Europe*/)
|
||||
MCFG_MACHINE_RESET_OVERRIDE(ti990_4_state, ti990_4 )
|
||||
|
||||
MCFG_SCREEN_ADD("screen", RASTER)
|
||||
MCFG_SCREEN_REFRESH_RATE(60)
|
||||
MCFG_SCREEN_VBLANK_TIME(ATTOSECONDS_IN_USEC(2500)) /* not accurate */
|
||||
MCFG_SCREEN_UPDATE_DRIVER(ti990_4_state, screen_update_ti990_4)
|
||||
|
||||
MCFG_SCREEN_SIZE(560, 280)
|
||||
MCFG_SCREEN_VISIBLE_AREA(0, 560-1, 0, /*250*/280-1)
|
||||
MCFG_SCREEN_PALETTE("vdt911:palette")
|
||||
MCFG_VDT911_VIDEO_ADD("vdt911", vdt911_intf)
|
||||
MCFG_SPEAKER_STANDARD_MONO("mono")
|
||||
MCFG_SOUND_ADD("beeper", BEEP, 0)
|
||||
MCFG_SOUND_ROUTE(ALL_OUTPUTS, "mono", 0.50)
|
||||
#endif
|
||||
|
||||
// Floppy controller
|
||||
MCFG_FD800_ADD("fd800", WRITELINE(ti990_4_state, fd_interrupt))
|
||||
MCFG_LEGACY_FLOPPY_4_DRIVES_ADD(ti990_4_floppy_interface)
|
||||
|
||||
MACHINE_CONFIG_END
|
||||
|
||||
|
||||
/*
|
||||
ROM loading
|
||||
*/
|
||||
@ -398,41 +447,22 @@ ROM_START(ti990_4)
|
||||
ROMX_LOAD("94519116.u7", 0xFE01, 0x100, CRC(fa387bf3), ROM_NIBBLE | ROM_SHIFT_NIBBLE_LO | ROM_SKIP(1))
|
||||
|
||||
#else
|
||||
/* ROM set 945121-4(?): "Floppy disc loader with self test" (cf 945401-9701
|
||||
pp. 1-19) */
|
||||
|
||||
/* ROM set 945121-4(?): "Floppy disc loader with self test" (cf 945401-9701 pp. 1-19) */
|
||||
ROM_LOAD16_WORD("ti9904.rom", 0xFC00, 0x400, CRC(691e7d19) SHA1(58d9bed80490fdf71c743bfd3077c70840b7df8c))
|
||||
|
||||
#endif
|
||||
|
||||
#if VIDEO_911
|
||||
/* VDT911 character definitions */
|
||||
ROM_REGION(vdt911_chr_region_len, vdt911_chr_region, ROMREGION_ERASEFF)
|
||||
#else
|
||||
ROM_REGION(asr733_chr_region_len, asr733_chr_region, ROMREGION_ERASEFF)
|
||||
#endif
|
||||
|
||||
ROM_END
|
||||
|
||||
DRIVER_INIT_MEMBER(ti990_4_state,ti990_4)
|
||||
{
|
||||
m_nmi_timer = timer_alloc(NMI_TIMER_ID);
|
||||
|
||||
#if VIDEO_911
|
||||
vdt911_init(machine());
|
||||
#else
|
||||
asr733_init(machine());
|
||||
#endif
|
||||
}
|
||||
|
||||
static INPUT_PORTS_START(ti990_4)
|
||||
#if VIDEO_911
|
||||
VDT911_KEY_PORTS
|
||||
#else
|
||||
ASR733_KEY_PORTS
|
||||
#endif
|
||||
INPUT_PORTS_END
|
||||
ROM_START(ti990_4v)
|
||||
/*CPU memory space*/
|
||||
ROM_REGION16_BE(0x10000, "maincpu",0)
|
||||
ROM_LOAD16_WORD("ti9904.rom", 0xFC00, 0x400, CRC(691e7d19) SHA1(58d9bed80490fdf71c743bfd3077c70840b7df8c))
|
||||
|
||||
/* VDT911 character definitions */
|
||||
ROM_REGION(vdt911_chr_region_len, vdt911_chr_region, ROMREGION_ERASEFF)
|
||||
ROM_END
|
||||
|
||||
/* YEAR NAME PARENT COMPAT MACHINE INPUT INIT COMPANY FULLNAME */
|
||||
COMP( 1976, ti990_4, 0, 0, ti990_4, ti990_4, ti990_4_state, ti990_4, "Texas Instruments", "TI Model 990/4 Microcomputer System" , GAME_NOT_WORKING | GAME_NO_SOUND )
|
||||
COMP( 1976, ti990_4, 0, 0, ti990_4, ti990_4, ti990_4_state, ti990_4, "Texas Instruments", "TI 990/4 Minicomputer System" , GAME_NOT_WORKING | GAME_NO_SOUND )
|
||||
COMP( 1976, ti990_4v, ti990_4, 0, ti990_4v, ti990_4v, ti990_4_state, ti990_4v, "Texas Instruments", "TI 990/4 Minicomputer System with Video Display Terminal" , GAME_NOT_WORKING | GAME_NO_SOUND )
|
||||
|
@ -6,6 +6,11 @@
|
||||
This floppy disk controller supports IBM-format 8" SSSD and DSSD floppies.
|
||||
|
||||
Raphael Nabet 2003
|
||||
|
||||
Rewritten as class
|
||||
Michael Zapf 2014
|
||||
|
||||
TODO: Make it work
|
||||
*/
|
||||
|
||||
#include "emu.h"
|
||||
@ -14,40 +19,6 @@
|
||||
#include "imagedev/flopdrv.h"
|
||||
#include "990_dk.h"
|
||||
|
||||
#define MAX_FLOPPIES 4
|
||||
|
||||
enum buf_mode_t {
|
||||
bm_off, bm_read, bm_write
|
||||
};
|
||||
static struct
|
||||
{
|
||||
running_machine *machine;
|
||||
UINT16 recv_buf;
|
||||
UINT16 stat_reg;
|
||||
UINT16 xmit_buf;
|
||||
UINT16 cmd_reg;
|
||||
|
||||
int interrupt_f_f;
|
||||
void (*interrupt_callback)(running_machine &, int state);
|
||||
|
||||
UINT8 buf[128];
|
||||
int buf_pos;
|
||||
buf_mode_t buf_mode;
|
||||
int unit;
|
||||
int head;
|
||||
int sector;
|
||||
/*int non_seq_mode;*/
|
||||
int ddam;
|
||||
|
||||
struct
|
||||
{
|
||||
device_image_interface *img;
|
||||
int phys_cylinder;
|
||||
int log_cylinder[2];
|
||||
int seclen;
|
||||
} drv[MAX_FLOPPIES];
|
||||
} fd800;
|
||||
|
||||
/* status bits */
|
||||
enum
|
||||
{
|
||||
@ -69,64 +40,57 @@ enum
|
||||
status_unit_shift = 13
|
||||
};
|
||||
|
||||
LEGACY_FLOPPY_OPTIONS_START(fd800)
|
||||
#if 1
|
||||
/* SSSD 8" */
|
||||
LEGACY_FLOPPY_OPTION(fd800, "dsk", "TI990 8\" SSSD disk image", basicdsk_identify_default, basicdsk_construct_default, NULL,
|
||||
HEADS([1])
|
||||
TRACKS([77])
|
||||
SECTORS([26])
|
||||
SECTOR_LENGTH([128])
|
||||
FIRST_SECTOR_ID([1]))
|
||||
#elif 0
|
||||
/* DSSD 8" */
|
||||
LEGACY_FLOPPY_OPTION(fd800, "dsk", "TI990 8\" DSSD disk image", basicdsk_identify_default, basicdsk_construct_default, NULL,
|
||||
HEADS([2])
|
||||
TRACKS([77])
|
||||
SECTORS([26])
|
||||
SECTOR_LENGTH([128])
|
||||
FIRST_SECTOR_ID([1]))
|
||||
#endif
|
||||
LEGACY_FLOPPY_OPTIONS_END
|
||||
const device_type FD800 = &device_creator<fd800_legacy_device>;
|
||||
|
||||
static void fd800_field_interrupt(void)
|
||||
fd800_legacy_device::fd800_legacy_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock)
|
||||
: device_t(mconfig, FD800, "TI FD800 Diablo floppy disk controller", tag, owner, clock, "fd800", __FILE__),
|
||||
m_int_line(*this)
|
||||
{
|
||||
if (fd800.interrupt_callback)
|
||||
(*fd800.interrupt_callback)(*fd800.machine, (fd800.stat_reg & status_interrupt) && ! fd800.interrupt_f_f);
|
||||
}
|
||||
|
||||
static void fd800_unload_proc(device_image_interface &image)
|
||||
void fd800_legacy_device::set_interrupt_line()
|
||||
{
|
||||
int unit = floppy_get_drive(&image.device());
|
||||
|
||||
fd800.drv[unit].log_cylinder[0] = fd800.drv[unit].log_cylinder[1] = -1;
|
||||
if ((m_stat_reg & status_interrupt) && ! m_interrupt_f_f)
|
||||
m_int_line(ASSERT_LINE);
|
||||
else
|
||||
m_int_line(CLEAR_LINE);
|
||||
}
|
||||
|
||||
void fd800_machine_init(running_machine &machine, void (*interrupt_callback)(running_machine &machine, int state))
|
||||
|
||||
/* void fd800_legacy_device::unload_proc(device_image_interface &image)
|
||||
{
|
||||
int i;
|
||||
int unit = floppy_get_drive(&image.device());
|
||||
|
||||
fd800.machine = &machine;
|
||||
fd800.interrupt_callback = interrupt_callback;
|
||||
|
||||
fd800.stat_reg = 0;
|
||||
fd800.interrupt_f_f = 1;
|
||||
|
||||
fd800.buf_pos = 0;
|
||||
fd800.buf_mode = bm_off;
|
||||
|
||||
for (i=0; i<MAX_FLOPPIES; i++)
|
||||
{
|
||||
fd800.drv[i].img = dynamic_cast<device_image_interface *>(floppy_get_device(machine, i));
|
||||
fd800.drv[i].phys_cylinder = -1;
|
||||
fd800.drv[i].log_cylinder[0] = fd800.drv[i].log_cylinder[1] = -1;
|
||||
fd800.drv[i].seclen = 64;
|
||||
floppy_install_unload_proc(&fd800.drv[i].img->device(), fd800_unload_proc);
|
||||
}
|
||||
|
||||
fd800_field_interrupt();
|
||||
m_drv[unit].log_cylinder[0] = m_drv[unit].log_cylinder[1] = -1;
|
||||
}
|
||||
|
||||
|
||||
void fd800_machine_init(void (*interrupt_callback)(running_machine &machine, int state))
|
||||
{
|
||||
int i;
|
||||
|
||||
m_machine = &machine;
|
||||
m_interrupt_callback = interrupt_callback;
|
||||
|
||||
m_stat_reg = 0;
|
||||
m_interrupt_f_f = 1;
|
||||
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_off;
|
||||
|
||||
for (i=0; i<MAX_FLOPPIES; i++)
|
||||
{
|
||||
m_drv[i].img = dynamic_cast<device_image_interface *>(floppy_get_device(machine, i));
|
||||
m_drv[i].phys_cylinder = -1;
|
||||
m_drv[i].log_cylinder[0] = m_drv[i].log_cylinder[1] = -1;
|
||||
m_drv[i].seclen = 64;
|
||||
floppy_install_unload_proc(&m_drv[i].img->device(), unload_proc);
|
||||
}
|
||||
|
||||
set_interrupt_line();
|
||||
}
|
||||
*/
|
||||
|
||||
/*
|
||||
Read the first id field that can be found on the floppy disk.
|
||||
|
||||
@ -137,7 +101,7 @@ void fd800_machine_init(running_machine &machine, void (*interrupt_callback)(run
|
||||
|
||||
Return TRUE if an ID was found
|
||||
*/
|
||||
static int fd800_read_id(int unit, int head, int *cylinder_id, int *sector_id)
|
||||
int fd800_legacy_device::read_id(int unit, int head, int *cylinder_id, int *sector_id)
|
||||
{
|
||||
/*UINT8 revolution_count;*/
|
||||
chrn_id id;
|
||||
@ -146,7 +110,7 @@ static int fd800_read_id(int unit, int head, int *cylinder_id, int *sector_id)
|
||||
|
||||
/*while (revolution_count < 2)*/
|
||||
/*{*/
|
||||
if (floppy_drive_get_next_id(&fd800.drv[unit].img->device(), head, &id))
|
||||
if (floppy_drive_get_next_id(&m_drv[unit].img->device(), head, &id))
|
||||
{
|
||||
if (cylinder_id)
|
||||
*cylinder_id = id.C;
|
||||
@ -169,7 +133,7 @@ static int fd800_read_id(int unit, int head, int *cylinder_id, int *sector_id)
|
||||
|
||||
Return TRUE if the given sector ID was found
|
||||
*/
|
||||
static int fd800_find_sector(int unit, int head, int sector, int *data_id)
|
||||
int fd800_legacy_device::find_sector(int unit, int head, int sector, int *data_id)
|
||||
{
|
||||
UINT8 revolution_count;
|
||||
chrn_id id;
|
||||
@ -178,7 +142,7 @@ static int fd800_find_sector(int unit, int head, int sector, int *data_id)
|
||||
|
||||
while (revolution_count < 2)
|
||||
{
|
||||
if (floppy_drive_get_next_id(&fd800.drv[unit].img->device(), head, &id))
|
||||
if (floppy_drive_get_next_id(&m_drv[unit].img->device(), head, &id))
|
||||
{
|
||||
/* compare id */
|
||||
if ((id.R == sector) && (id.N == 0))
|
||||
@ -203,58 +167,58 @@ static int fd800_find_sector(int unit, int head, int sector, int *data_id)
|
||||
|
||||
Return FALSE if the seek was successful
|
||||
*/
|
||||
static int fd800_do_seek(int unit, int cylinder, int head)
|
||||
int fd800_legacy_device::do_seek(int unit, int cylinder, int head)
|
||||
{
|
||||
int retries;
|
||||
|
||||
if (cylinder > 76)
|
||||
{
|
||||
fd800.stat_reg |= status_invalid_cmd;
|
||||
m_stat_reg |= status_invalid_cmd;
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
if (!fd800.drv[unit].img->exists())
|
||||
if (m_drv[unit].img == NULL || !m_drv[unit].img->exists())
|
||||
{
|
||||
fd800.stat_reg |= status_drv_not_ready; /* right??? */
|
||||
m_stat_reg |= status_drv_not_ready; /* right??? */
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
if (fd800.drv[unit].log_cylinder[head] == -1)
|
||||
if (m_drv[unit].log_cylinder[head] == -1)
|
||||
{ /* current track ID is unknown: read it */
|
||||
if (!fd800_read_id(unit, head, &fd800.drv[unit].log_cylinder[head], NULL))
|
||||
if (!read_id(unit, head, &m_drv[unit].log_cylinder[head], NULL))
|
||||
{
|
||||
fd800.stat_reg |= status_ID_not_found;
|
||||
m_stat_reg |= status_ID_not_found;
|
||||
return TRUE;
|
||||
}
|
||||
}
|
||||
/* exit if we are already at the requested track */
|
||||
if (fd800.drv[unit].log_cylinder[head] == cylinder)
|
||||
if (m_drv[unit].log_cylinder[head] == cylinder)
|
||||
{
|
||||
/*fd800.stat_reg |= status_OP_complete;*/
|
||||
/*m_stat_reg |= status_OP_complete;*/
|
||||
return FALSE;
|
||||
}
|
||||
for (retries=0; retries<10; retries++)
|
||||
{ /* seek to requested track */
|
||||
floppy_drive_seek(&fd800.drv[unit].img->device(), cylinder-fd800.drv[unit].log_cylinder[head]);
|
||||
floppy_drive_seek(&m_drv[unit].img->device(), cylinder-m_drv[unit].log_cylinder[head]);
|
||||
/* update physical track position */
|
||||
if (fd800.drv[unit].phys_cylinder != -1)
|
||||
fd800.drv[unit].phys_cylinder += cylinder-fd800.drv[unit].log_cylinder[head];
|
||||
if (m_drv[unit].phys_cylinder != -1)
|
||||
m_drv[unit].phys_cylinder += cylinder-m_drv[unit].log_cylinder[head];
|
||||
/* read new track ID */
|
||||
if (!fd800_read_id(unit, head, &fd800.drv[unit].log_cylinder[head], NULL))
|
||||
if (!read_id(unit, head, &m_drv[unit].log_cylinder[head], NULL))
|
||||
{
|
||||
fd800.drv[unit].log_cylinder[head] = -1;
|
||||
fd800.stat_reg |= status_ID_not_found;
|
||||
m_drv[unit].log_cylinder[head] = -1;
|
||||
m_stat_reg |= status_ID_not_found;
|
||||
return TRUE;
|
||||
}
|
||||
/* exit if we have reached the requested track */
|
||||
if (fd800.drv[unit].log_cylinder[head] == cylinder)
|
||||
if (m_drv[unit].log_cylinder[head] == cylinder)
|
||||
{
|
||||
/*fd800.stat_reg |= status_OP_complete;*/
|
||||
/*m_stat_reg |= status_OP_complete;*/
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
/* track not found */
|
||||
fd800.stat_reg |= status_seek_err;
|
||||
m_stat_reg |= status_seek_err;
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
@ -265,32 +229,32 @@ static int fd800_do_seek(int unit, int cylinder, int head)
|
||||
|
||||
Return FALSE if the restore was successful
|
||||
*/
|
||||
static int fd800_do_restore(int unit)
|
||||
int fd800_legacy_device::do_restore(int unit)
|
||||
{
|
||||
int seek_count = 0;
|
||||
int seek_complete;
|
||||
|
||||
if (!fd800.drv[unit].img->exists())
|
||||
if (!m_drv[unit].img->exists())
|
||||
{
|
||||
fd800.stat_reg |= status_drv_not_ready; /* right??? */
|
||||
m_stat_reg |= status_drv_not_ready; /* right??? */
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
/* limit iterations to 76 to prevent an endless loop if the disc is locked */
|
||||
while (!(seek_complete = !floppy_tk00_r(&fd800.drv[unit].img->device())) && (seek_count < 76))
|
||||
while (!(seek_complete = !floppy_tk00_r(&m_drv[unit].img->device())) && (seek_count < 76))
|
||||
{
|
||||
floppy_drive_seek(&fd800.drv[unit].img->device(), -1);
|
||||
floppy_drive_seek(&m_drv[unit].img->device(), -1);
|
||||
seek_count++;
|
||||
}
|
||||
if (! seek_complete)
|
||||
{
|
||||
fd800.drv[unit].phys_cylinder = -1;
|
||||
fd800.stat_reg |= status_seek_err;
|
||||
m_drv[unit].phys_cylinder = -1;
|
||||
m_stat_reg |= status_seek_err;
|
||||
}
|
||||
else
|
||||
{
|
||||
fd800.drv[unit].phys_cylinder = 0;
|
||||
/*fd800.stat_reg |= status_OP_complete;*/
|
||||
m_drv[unit].phys_cylinder = 0;
|
||||
/*m_stat_reg |= status_OP_complete;*/
|
||||
}
|
||||
|
||||
return ! seek_complete;
|
||||
@ -299,60 +263,60 @@ static int fd800_do_restore(int unit)
|
||||
/*
|
||||
Perform a read operation for one sector
|
||||
*/
|
||||
static void fd800_do_read(void)
|
||||
void fd800_legacy_device::do_read(void)
|
||||
{
|
||||
int data_id;
|
||||
|
||||
if ((fd800.sector == 0) || (fd800.sector > 26))
|
||||
if ((m_sector == 0) || (m_sector > 26))
|
||||
{
|
||||
fd800.stat_reg |= status_invalid_cmd;
|
||||
m_stat_reg |= status_invalid_cmd;
|
||||
return;
|
||||
}
|
||||
|
||||
if (!fd800_find_sector(fd800.unit, fd800.head, fd800.sector, &data_id))
|
||||
if (!find_sector(m_unit, m_head, m_sector, &data_id))
|
||||
{
|
||||
fd800.stat_reg |= status_ID_not_found;
|
||||
m_stat_reg |= status_ID_not_found;
|
||||
return;
|
||||
}
|
||||
|
||||
floppy_drive_read_sector_data(&fd800.drv[fd800.unit].img->device(), fd800.head, data_id, fd800.buf, 128);
|
||||
fd800.buf_pos = 0;
|
||||
fd800.buf_mode = bm_read;
|
||||
fd800.recv_buf = (fd800.buf[fd800.buf_pos<<1] << 8) | fd800.buf[(fd800.buf_pos<<1)+1];
|
||||
floppy_drive_read_sector_data(&m_drv[m_unit].img->device(), m_head, data_id, m_buf, 128);
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_read;
|
||||
m_recv_buf = (m_buf[m_buf_pos<<1] << 8) | m_buf[(m_buf_pos<<1)+1];
|
||||
|
||||
fd800.stat_reg |= status_XFER_ready;
|
||||
fd800.stat_reg |= status_OP_complete; /* right??? */
|
||||
m_stat_reg |= status_XFER_ready;
|
||||
m_stat_reg |= status_OP_complete; /* right??? */
|
||||
}
|
||||
|
||||
/*
|
||||
Perform a write operation for one sector
|
||||
*/
|
||||
static void fd800_do_write(void)
|
||||
void fd800_legacy_device::do_write(void)
|
||||
{
|
||||
int data_id;
|
||||
|
||||
if (fd800.drv[fd800.unit].seclen < 64)
|
||||
if (m_drv[m_unit].seclen < 64)
|
||||
/* fill with 0s */
|
||||
memset(fd800.buf+(fd800.drv[fd800.unit].seclen<<1), 0, (64-fd800.drv[fd800.unit].seclen)<<1);
|
||||
memset(m_buf+(m_drv[m_unit].seclen<<1), 0, (64-m_drv[m_unit].seclen)<<1);
|
||||
|
||||
if (!fd800_find_sector(fd800.unit, fd800.head, fd800.sector, &data_id))
|
||||
if (!find_sector(m_unit, m_head, m_sector, &data_id))
|
||||
{
|
||||
fd800.stat_reg |= status_ID_not_found;
|
||||
m_stat_reg |= status_ID_not_found;
|
||||
return;
|
||||
}
|
||||
|
||||
floppy_drive_write_sector_data(&fd800.drv[fd800.unit].img->device(), fd800.head, data_id, fd800.buf, 128, fd800.ddam);
|
||||
fd800.buf_pos = 0;
|
||||
fd800.buf_mode = bm_write;
|
||||
floppy_drive_write_sector_data(&m_drv[m_unit].img->device(), m_head, data_id, m_buf, 128, m_ddam);
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_write;
|
||||
|
||||
fd800.stat_reg |= status_XFER_ready;
|
||||
fd800.stat_reg |= status_OP_complete; /* right??? */
|
||||
m_stat_reg |= status_XFER_ready;
|
||||
m_stat_reg |= status_OP_complete; /* right??? */
|
||||
}
|
||||
|
||||
/*
|
||||
Execute a fdc command
|
||||
*/
|
||||
static void fd800_do_cmd(void)
|
||||
void fd800_legacy_device::do_cmd(void)
|
||||
{
|
||||
int unit;
|
||||
int cylinder;
|
||||
@ -361,43 +325,43 @@ static void fd800_do_cmd(void)
|
||||
int sector;
|
||||
|
||||
|
||||
if (fd800.buf_mode != bm_off)
|
||||
if (m_buf_mode != bm_off)
|
||||
{ /* All commands in the midst of read or write are interpreted as Stop */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
fd800.buf_pos = 0;
|
||||
fd800.buf_mode = bm_off;
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_off;
|
||||
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
switch (fd800.cmd_reg >> 12)
|
||||
switch (m_cmd_reg >> 12)
|
||||
{
|
||||
case 0: /* select
|
||||
bits 16-25: 0s
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if (!fd800.drv[unit].img->exists())
|
||||
fd800.stat_reg |= status_drv_not_ready; /* right??? */
|
||||
else if (fd800.drv[unit].img->is_readonly())
|
||||
fd800.stat_reg |= status_write_prot;
|
||||
if (!m_drv[unit].img->exists())
|
||||
m_stat_reg |= status_drv_not_ready; /* right??? */
|
||||
else if (m_drv[unit].img->is_readonly())
|
||||
m_stat_reg |= status_write_prot;
|
||||
else
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 1: /* seek
|
||||
@ -405,57 +369,57 @@ static void fd800_do_cmd(void)
|
||||
bits 23-24: 0s
|
||||
bits 25: head number (1=upper)
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
head = (fd800.cmd_reg >> 9) & 1;
|
||||
cylinder = fd800.cmd_reg & 0x7f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
head = (m_cmd_reg >> 9) & 1;
|
||||
cylinder = m_cmd_reg & 0x7f;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if (!fd800_do_seek(unit, cylinder, head))
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
if (!do_seek(unit, cylinder, head))
|
||||
m_stat_reg |= status_OP_complete;
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 2: /* restore
|
||||
bits 16-25: 0s
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if (!fd800_do_restore(unit))
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
if (!do_restore(unit))
|
||||
m_stat_reg |= status_OP_complete;
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 3: /* sector length
|
||||
bits 16-22: sector word count (0-64)
|
||||
bits 23-25: 0s
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
seclen = fd800.cmd_reg & 0x7f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
seclen = m_cmd_reg & 0x7f;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if ((seclen > 64) || (seclen == 0))
|
||||
{
|
||||
fd800.stat_reg |= status_invalid_cmd;
|
||||
m_stat_reg |= status_invalid_cmd;
|
||||
}
|
||||
else
|
||||
{
|
||||
fd800.drv[unit].seclen = seclen;
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
m_drv[unit].seclen = seclen;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
}
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 4: /* read
|
||||
@ -464,47 +428,47 @@ static void fd800_do_cmd(void)
|
||||
bit 24: no sequential sectoring (1=active)
|
||||
bit 25: head number (1=upper)
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
head = (fd800.cmd_reg >> 9) & 1;
|
||||
/*non_seq_mode = (fd800.cmd_reg >> 8) & 1;*/
|
||||
sector = fd800.cmd_reg & 0x1f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
head = (m_cmd_reg >> 9) & 1;
|
||||
/*non_seq_mode = (m_cmd_reg >> 8) & 1;*/
|
||||
sector = m_cmd_reg & 0x1f;
|
||||
|
||||
fd800.unit = unit;
|
||||
fd800.head = head;
|
||||
fd800.sector = sector;
|
||||
/*fd800.non_seq_mode = non_seq_mode;*/
|
||||
m_unit = unit;
|
||||
m_head = head;
|
||||
m_sector = sector;
|
||||
/*m_non_seq_mode = non_seq_mode;*/
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
fd800_do_read();
|
||||
do_read();
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 5: /* read ID
|
||||
bits 16-24: 0s
|
||||
bit 25: head number (1=upper)
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
head = (fd800.cmd_reg >> 9) & 1;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
head = (m_cmd_reg >> 9) & 1;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if (!fd800_read_id(unit, head, &cylinder, §or))
|
||||
if (!read_id(unit, head, &cylinder, §or))
|
||||
{
|
||||
fd800.stat_reg |= status_ID_not_found;
|
||||
m_stat_reg |= status_ID_not_found;
|
||||
}
|
||||
else
|
||||
{
|
||||
fd800.recv_buf = (cylinder << 8) | sector;
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
m_recv_buf = (cylinder << 8) | sector;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
}
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 6: /* read unformatted
|
||||
@ -520,32 +484,32 @@ static void fd800_do_cmd(void)
|
||||
bits 21-24: 0s
|
||||
bit 25: head number (1=upper)
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
head = (fd800.cmd_reg >> 9) & 1;
|
||||
sector = fd800.cmd_reg & 0x1f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
head = (m_cmd_reg >> 9) & 1;
|
||||
sector = m_cmd_reg & 0x1f;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if ((fd800.sector == 0) || (fd800.sector > 26))
|
||||
if ((m_sector == 0) || (m_sector > 26))
|
||||
{
|
||||
fd800.stat_reg |= status_invalid_cmd;
|
||||
m_stat_reg |= status_invalid_cmd;
|
||||
}
|
||||
else
|
||||
{
|
||||
fd800.unit = unit;
|
||||
fd800.head = head;
|
||||
fd800.sector = sector;
|
||||
fd800.ddam = 0;
|
||||
m_unit = unit;
|
||||
m_head = head;
|
||||
m_sector = sector;
|
||||
m_ddam = 0;
|
||||
|
||||
fd800.buf_pos = 0;
|
||||
fd800.buf_mode = bm_write;
|
||||
fd800.stat_reg |= status_XFER_ready;
|
||||
fd800.stat_reg |= status_OP_complete; /* right??? */
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_write;
|
||||
m_stat_reg |= status_XFER_ready;
|
||||
m_stat_reg |= status_OP_complete; /* right??? */
|
||||
}
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 8: /* write delete
|
||||
@ -553,32 +517,32 @@ static void fd800_do_cmd(void)
|
||||
bits 21-24: 0s
|
||||
bit 25: head number (1=upper)
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
head = (fd800.cmd_reg >> 9) & 1;
|
||||
sector = fd800.cmd_reg & 0x1f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
head = (m_cmd_reg >> 9) & 1;
|
||||
sector = m_cmd_reg & 0x1f;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
if ((fd800.sector == 0) || (fd800.sector > 26))
|
||||
if ((m_sector == 0) || (m_sector > 26))
|
||||
{
|
||||
fd800.stat_reg |= status_invalid_cmd;
|
||||
m_stat_reg |= status_invalid_cmd;
|
||||
}
|
||||
else
|
||||
{
|
||||
fd800.unit = unit;
|
||||
fd800.head = head;
|
||||
fd800.sector = sector;
|
||||
fd800.ddam = 1;
|
||||
m_unit = unit;
|
||||
m_head = head;
|
||||
m_sector = sector;
|
||||
m_ddam = 1;
|
||||
|
||||
fd800.buf_pos = 0;
|
||||
fd800.buf_mode = bm_write;
|
||||
fd800.stat_reg |= status_XFER_ready;
|
||||
fd800.stat_reg |= status_OP_complete; /* right??? */
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_write;
|
||||
m_stat_reg |= status_XFER_ready;
|
||||
m_stat_reg |= status_OP_complete; /* right??? */
|
||||
}
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 9: /* format track
|
||||
@ -592,49 +556,49 @@ static void fd800_do_cmd(void)
|
||||
case 10: /* load int mask
|
||||
bit 16: bad mask for interrupt (0 = unmask or enable interrupt)
|
||||
bits 17-27: 0s */
|
||||
fd800.interrupt_f_f = fd800.cmd_reg & 1;
|
||||
fd800_field_interrupt();
|
||||
m_interrupt_f_f = m_cmd_reg & 1;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 11: /* stop
|
||||
bits 16-25: 0s
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
|
||||
/* reset status */
|
||||
fd800.stat_reg = unit << status_unit_shift;
|
||||
m_stat_reg = unit << status_unit_shift;
|
||||
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 12: /* step head
|
||||
bits 16-22: track number (0-76)
|
||||
bits 23-25: 0s
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
cylinder = fd800.cmd_reg & 0x7f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
cylinder = m_cmd_reg & 0x7f;
|
||||
|
||||
if (cylinder > 76)
|
||||
{
|
||||
fd800.stat_reg |= status_invalid_cmd;
|
||||
m_stat_reg |= status_invalid_cmd;
|
||||
}
|
||||
else if ((fd800.drv[unit].phys_cylinder != -1) || (!fd800_do_restore(unit)))
|
||||
else if ((m_drv[unit].phys_cylinder != -1) || (!do_restore(unit)))
|
||||
{
|
||||
floppy_drive_seek(&fd800.drv[unit].img->device(), cylinder-fd800.drv[unit].phys_cylinder);
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
floppy_drive_seek(&m_drv[unit].img->device(), cylinder-m_drv[unit].phys_cylinder);
|
||||
m_stat_reg |= status_OP_complete;
|
||||
}
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 13: /* maintenance commands
|
||||
bits 16-23: according to extended command code
|
||||
bits 24-27: extended command code (0-7) */
|
||||
switch ((fd800.cmd_reg >> 8) & 15)
|
||||
switch ((m_cmd_reg >> 8) & 15)
|
||||
{
|
||||
case 0: /* reset
|
||||
bits 16-23: 0s */
|
||||
@ -686,29 +650,29 @@ static void fd800_do_cmd(void)
|
||||
bit 24: no sequential sectoring (1=active)
|
||||
bit 25: head number (1=upper)
|
||||
bits 26-27: unit number (0-3) */
|
||||
unit = (fd800.cmd_reg >> 10) & 3;
|
||||
head = (fd800.cmd_reg >> 9) & 1;
|
||||
/*non_seq_mode = (fd800.cmd_reg >> 8) & 1;*/
|
||||
cylinder = fd800.cmd_reg & 0x7f;
|
||||
unit = (m_cmd_reg >> 10) & 3;
|
||||
head = (m_cmd_reg >> 9) & 1;
|
||||
/*non_seq_mode = (m_cmd_reg >> 8) & 1;*/
|
||||
cylinder = m_cmd_reg & 0x7f;
|
||||
|
||||
if (!fd800_do_seek(unit, cylinder, head))
|
||||
if (!do_seek(unit, cylinder, head))
|
||||
{
|
||||
fd800.unit = unit;
|
||||
fd800.head = head;
|
||||
fd800.sector = 1;
|
||||
/*fd800.non_seq_mode = non_seq_mode;*/
|
||||
m_unit = unit;
|
||||
m_head = head;
|
||||
m_sector = 1;
|
||||
/*m_non_seq_mode = non_seq_mode;*/
|
||||
|
||||
fd800_do_read();
|
||||
do_read();
|
||||
}
|
||||
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
|
||||
case 15: /* Clear Status port
|
||||
bits 16-27: 0s */
|
||||
fd800.stat_reg = 0;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg = 0;
|
||||
set_interrupt_line();
|
||||
break;
|
||||
}
|
||||
}
|
||||
@ -735,7 +699,7 @@ static void fd800_do_cmd(void)
|
||||
30: unit MSB
|
||||
31: Interrupt (CBUSY???) (1 -> controller is ready)
|
||||
*/
|
||||
READ8_HANDLER(fd800_cru_r)
|
||||
READ8_MEMBER( fd800_legacy_device::cru_r )
|
||||
{
|
||||
int reply = 0;
|
||||
|
||||
@ -744,13 +708,13 @@ static void fd800_do_cmd(void)
|
||||
case 0:
|
||||
case 1:
|
||||
/* receive buffer */
|
||||
reply = fd800.recv_buf >> (offset*8);
|
||||
reply = m_recv_buf >> (offset*8);
|
||||
break;
|
||||
|
||||
case 2:
|
||||
case 3:
|
||||
/* status register */
|
||||
reply = fd800.stat_reg >> ((offset-2)*8);
|
||||
reply = m_stat_reg >> ((offset-2)*8);
|
||||
break;
|
||||
}
|
||||
|
||||
@ -769,7 +733,7 @@ static void fd800_do_cmd(void)
|
||||
27: FD unit number MSB/extended command code
|
||||
28-31: command code
|
||||
*/
|
||||
WRITE8_HANDLER(fd800_cru_w)
|
||||
WRITE8_MEMBER( fd800_legacy_device::cru_w )
|
||||
{
|
||||
switch (offset)
|
||||
{
|
||||
@ -791,61 +755,61 @@ WRITE8_HANDLER(fd800_cru_w)
|
||||
case 15:
|
||||
/* transmit buffer */
|
||||
if (data)
|
||||
fd800.xmit_buf |= 1 << offset;
|
||||
m_xmit_buf |= 1 << offset;
|
||||
else
|
||||
fd800.xmit_buf &= ~(1 << offset);
|
||||
m_xmit_buf &= ~(1 << offset);
|
||||
if (offset == 15)
|
||||
{
|
||||
switch (fd800.buf_mode)
|
||||
switch (m_buf_mode)
|
||||
{
|
||||
case bm_off:
|
||||
break;
|
||||
case bm_read:
|
||||
fd800.buf_pos++;
|
||||
if (fd800.buf_pos == fd800.drv[fd800.unit].seclen)
|
||||
m_buf_pos++;
|
||||
if (m_buf_pos == m_drv[m_unit].seclen)
|
||||
{ /* end of sector */
|
||||
if (fd800.sector == 26)
|
||||
if (m_sector == 26)
|
||||
{ /* end of track -> end command (right???) */
|
||||
fd800.stat_reg &= ~status_XFER_ready;
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800.buf_mode = bm_off;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg &= ~status_XFER_ready;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
m_stat_reg |= status_interrupt;
|
||||
m_buf_mode = bm_off;
|
||||
set_interrupt_line();
|
||||
}
|
||||
else
|
||||
{ /* read next sector */
|
||||
fd800.sector++;
|
||||
fd800.stat_reg &= ~status_XFER_ready | status_OP_complete | status_interrupt;
|
||||
fd800_do_read();
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_sector++;
|
||||
m_stat_reg &= ~status_XFER_ready | status_OP_complete | status_interrupt;
|
||||
do_read();
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
}
|
||||
}
|
||||
else
|
||||
fd800.recv_buf = (fd800.buf[fd800.buf_pos<<1] << 8) | fd800.buf[(fd800.buf_pos<<1)+1];
|
||||
m_recv_buf = (m_buf[m_buf_pos<<1] << 8) | m_buf[(m_buf_pos<<1)+1];
|
||||
break;
|
||||
|
||||
case bm_write:
|
||||
fd800.buf[fd800.buf_pos<<1] = fd800.xmit_buf >> 8;
|
||||
fd800.buf[(fd800.buf_pos<<1)+1] = fd800.xmit_buf & 0xff;
|
||||
fd800.buf_pos++;
|
||||
if (fd800.buf_pos == fd800.drv[fd800.unit].seclen)
|
||||
m_buf[m_buf_pos<<1] = m_xmit_buf >> 8;
|
||||
m_buf[(m_buf_pos<<1)+1] = m_xmit_buf & 0xff;
|
||||
m_buf_pos++;
|
||||
if (m_buf_pos == m_drv[m_unit].seclen)
|
||||
{ /* end of sector */
|
||||
fd800_do_write();
|
||||
if (fd800.sector == 26)
|
||||
do_write();
|
||||
if (m_sector == 26)
|
||||
{
|
||||
/* end of track -> end command (right???) */
|
||||
fd800.stat_reg &= ~status_XFER_ready;
|
||||
fd800.stat_reg |= status_OP_complete;
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800.buf_mode = bm_off;
|
||||
fd800_field_interrupt();
|
||||
m_stat_reg &= ~status_XFER_ready;
|
||||
m_stat_reg |= status_OP_complete;
|
||||
m_stat_reg |= status_interrupt;
|
||||
m_buf_mode = bm_off;
|
||||
set_interrupt_line();
|
||||
}
|
||||
else
|
||||
{ /* increment to next sector */
|
||||
fd800.sector++;
|
||||
fd800.stat_reg |= status_interrupt;
|
||||
fd800_field_interrupt();
|
||||
m_sector++;
|
||||
m_stat_reg |= status_interrupt;
|
||||
set_interrupt_line();
|
||||
}
|
||||
}
|
||||
break;
|
||||
@ -871,11 +835,55 @@ WRITE8_HANDLER(fd800_cru_w)
|
||||
case 31:
|
||||
/* command register */
|
||||
if (data)
|
||||
fd800.cmd_reg |= 1 << (offset-16);
|
||||
m_cmd_reg |= 1 << (offset-16);
|
||||
else
|
||||
fd800.cmd_reg &= ~(1 << (offset-16));
|
||||
m_cmd_reg &= ~(1 << (offset-16));
|
||||
if (offset == 31)
|
||||
fd800_do_cmd();
|
||||
do_cmd();
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
LEGACY_FLOPPY_OPTIONS_START(fd800)
|
||||
#if 1
|
||||
/* SSSD 8" */
|
||||
LEGACY_FLOPPY_OPTION(fd800, "dsk", "TI990 8\" SSSD disk image", basicdsk_identify_default, basicdsk_construct_default, NULL,
|
||||
HEADS([1])
|
||||
TRACKS([77])
|
||||
SECTORS([26])
|
||||
SECTOR_LENGTH([128])
|
||||
FIRST_SECTOR_ID([1]))
|
||||
#elif 0
|
||||
/* DSSD 8" */
|
||||
LEGACY_FLOPPY_OPTION(fd800, "dsk", "TI990 8\" DSSD disk image", basicdsk_identify_default, basicdsk_construct_default, NULL,
|
||||
HEADS([2])
|
||||
TRACKS([77])
|
||||
SECTORS([26])
|
||||
SECTOR_LENGTH([128])
|
||||
FIRST_SECTOR_ID([1]))
|
||||
#endif
|
||||
LEGACY_FLOPPY_OPTIONS_END
|
||||
|
||||
void fd800_legacy_device::device_start(void)
|
||||
{
|
||||
logerror("fd800: start\n");
|
||||
m_int_line.resolve();
|
||||
|
||||
for (int i=0; i<MAX_FLOPPIES; i++)
|
||||
{
|
||||
m_drv[i].img = dynamic_cast<device_image_interface *>(floppy_get_device(machine(), i));
|
||||
m_drv[i].phys_cylinder = -1;
|
||||
m_drv[i].log_cylinder[0] = m_drv[i].log_cylinder[1] = -1;
|
||||
m_drv[i].seclen = 64;
|
||||
}
|
||||
}
|
||||
|
||||
void fd800_legacy_device::device_reset(void)
|
||||
{
|
||||
logerror("fd800: reset\n");
|
||||
m_stat_reg = 0;
|
||||
m_interrupt_f_f = 1;
|
||||
|
||||
m_buf_pos = 0;
|
||||
m_buf_mode = bm_off;
|
||||
}
|
||||
|
@ -1,9 +1,73 @@
|
||||
/*
|
||||
990_dk.h: include file for 990_dk.c
|
||||
*/
|
||||
|
||||
#ifndef __990_DK__
|
||||
#define __990_DK__
|
||||
|
||||
extern const device_type FD800;
|
||||
|
||||
#define MAX_FLOPPIES 4
|
||||
|
||||
class fd800_legacy_device : public device_t
|
||||
{
|
||||
public:
|
||||
fd800_legacy_device(const machine_config &mconfig, const char *tag, device_t *owner, UINT32 clock);
|
||||
|
||||
DECLARE_READ8_MEMBER( cru_r );
|
||||
DECLARE_WRITE8_MEMBER( cru_w );
|
||||
template<class _Object> static devcb2_base &static_set_int_callback(device_t &device, _Object object)
|
||||
{
|
||||
return downcast<fd800_legacy_device &>(device).m_int_line.set_callback(object);
|
||||
}
|
||||
|
||||
private:
|
||||
void device_start(void);
|
||||
void device_reset(void);
|
||||
void set_interrupt_line();
|
||||
|
||||
int read_id(int unit, int head, int *cylinder_id, int *sector_id);
|
||||
int find_sector(int unit, int head, int sector, int *data_id);
|
||||
int do_seek(int unit, int cylinder, int head);
|
||||
int do_restore(int unit);
|
||||
void do_read(void);
|
||||
void do_write(void);
|
||||
void do_cmd(void);
|
||||
|
||||
UINT16 m_recv_buf;
|
||||
UINT16 m_stat_reg;
|
||||
UINT16 m_xmit_buf;
|
||||
UINT16 m_cmd_reg;
|
||||
|
||||
int m_interrupt_f_f;
|
||||
devcb2_write_line m_int_line;
|
||||
|
||||
enum buf_mode_t {
|
||||
bm_off, bm_read, bm_write
|
||||
};
|
||||
|
||||
UINT8 m_buf[128];
|
||||
int m_buf_pos;
|
||||
buf_mode_t m_buf_mode;
|
||||
int m_unit;
|
||||
int m_head;
|
||||
int m_sector;
|
||||
/*int m_non_seq_mode;*/
|
||||
int m_ddam;
|
||||
|
||||
struct
|
||||
{
|
||||
device_image_interface *img;
|
||||
int phys_cylinder;
|
||||
int log_cylinder[2];
|
||||
int seclen;
|
||||
} m_drv[MAX_FLOPPIES];
|
||||
};
|
||||
|
||||
LEGACY_FLOPPY_OPTIONS_EXTERN(fd800);
|
||||
|
||||
void fd800_machine_init(running_machine &machine, void (*interrupt_callback)(running_machine &machine, int state));
|
||||
#define MCFG_FD800_ADD(_tag, _intcallb) \
|
||||
MCFG_DEVICE_ADD(_tag, FD800, 0) \
|
||||
devcb = &fd800_legacy_device::static_set_int_callback( *device, DEVCB2_##_intcallb );
|
||||
|
||||
extern DECLARE_READ8_HANDLER(fd800_cru_r);
|
||||
extern DECLARE_WRITE8_HANDLER(fd800_cru_w);
|
||||
#endif
|
||||
|
@ -353,7 +353,7 @@ mu100 // 1997 MU-100
|
||||
mt32
|
||||
cm32l
|
||||
d110
|
||||
sc55 // 1991 Sound Canvas SC-55
|
||||
sc55 // 1991 Sound Canvas SC-55
|
||||
|
||||
//***************COMPUTERS**************************************************
|
||||
|
||||
@ -995,6 +995,7 @@ ip244415 // IP24: Indigo 2, R4400, 150MHz
|
||||
// Texas Instruments
|
||||
ti990_10 // 1975 TI 990/10
|
||||
ti990_4 // 1976 TI 990/4
|
||||
ti990_4v // 1976 TI 990/4 with video display terminal
|
||||
990189 // 1978 TM 990/189
|
||||
990189v // 1980 TM 990/189 with Color Video Board
|
||||
|
||||
@ -1939,7 +1940,7 @@ batmantv // The Batman, 2004
|
||||
harriet // 1990
|
||||
|
||||
// Fanuc
|
||||
fanucs15 // 1990
|
||||
fanucs15 // 1990
|
||||
|
||||
//********** Misc **********************************************************
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user