mame/src/mame/drivers/taitowlf.cpp

442 lines
14 KiB
C++

// license:BSD-3-Clause
// copyright-holders:Ville Linde
/* Taito Wolf System
Driver by Ville Linde
Three board system consisting of a P5TX-LA PC motherboard, a Taito main board and a rom board.
Hardware configuration:
P5TX-LA Motherboard:
-CPU: Intel SL27J Pentium MMX @ 200 MHz
-Onboard sound: Crystal CS4237B ISA Audio
-Onboard VGA: ATI Rage II 3D Graph (removed from motherboard)
Chipsets (430TX PCIset):
-82439TX Northbridge
-82371EB PIIX4 PCI-ISA Southbridge
Taito W Main Board:
-AMD M4-128N/64 CPLD stamped 'E58-01'
-AMD MACH231 CPLD stamped 'E58-02'
-AMD MACH211 CPLD stamped 'E58-03'
-Panasonic MN1020019 (MN10200 based) Sound CPU
-Zoom ZFX-2 DSP (TMS57002 DSP)
-Zoom ZSG-2 Sound PCM chip
-Taito TC0510NIO I/O chip
-1x RAM NEC 42S4260
-1x RAM GM71C4400
-12x RAM Alliance AS4C256K16E0-35 (256k x 16)
-Mitsubishi M66220 256 x 8-bit CMOS memory
-Fujitsu MB87078 6-bit, 4-channel Electronic Volume Controller
-Atmel 93C66 EEPROM (4kb probably for high scores, test mode settings etc)
-ICS5342-3 GENDAC 16-Bit Integrated Clock-LUT-DAC
-3DFX 500-0003-03 F805281.1 FBI
-3DFX 500-0004-02 F804701.1 TMU
-Rom: E58-04 (bootscreen)
-XTALs 50MHz (near 3DFX) and 14.31818MHz (near RAMDAC)
Taito W Rom Board:
-AMD M4-128N/64 CPLD stamped 'E58-05'
-Program, Sound roms
TODO:
- program ROM is read via parallel port (for offset write, encrypted) and game port!?
- Emulation of the entire Taito Wolf main board which plugs into the PC motherboard's only PCI slot.
- PCI comms between both boards have yet to be understood.
*/
#include "emu.h"
#define TAITOWLF_ENABLE_VGA 0
#include "cpu/i386/i386.h"
#include "machine/lpci.h"
#include "machine/pcshare.h"
#include "machine/pckeybrd.h"
#if TAITOWLF_ENABLE_VGA
#include "video/pc_vga.h"
#endif
#include "emupal.h"
#include "screen.h"
namespace {
class taitowlf_state : public pcat_base_state
{
public:
taitowlf_state(const machine_config &mconfig, device_type type, const char *tag) :
pcat_base_state(mconfig, type, tag),
m_bootscreen_rom(*this, "bootscreen"),
m_bank1(*this, "bank1"),
m_palette(*this, "palette")
{ }
void taitowlf(machine_config &config);
void init_taitowlf();
protected:
virtual void machine_start() override;
virtual void machine_reset() override;
private:
required_region_ptr<uint8_t> m_bootscreen_rom;
required_memory_bank m_bank1;
optional_device<palette_device> m_palette;
void pnp_config_w(offs_t offset, uint32_t data, uint32_t mem_mask = ~0);
void pnp_data_w(offs_t offset, uint32_t data, uint32_t mem_mask = ~0);
void bios_ram_w(offs_t offset, uint32_t data, uint32_t mem_mask = ~0);
#if !TAITOWLF_ENABLE_VGA
void taitowlf_palette(palette_device &palette) const;
#endif
uint32_t screen_update_taitowlf(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect);
void intel82439tx_init();
void taitowlf_io(address_map &map);
void taitowlf_map(address_map &map);
uint8_t mtxc_config_r(int function, int reg);
void mtxc_config_w(int function, int reg, uint8_t data);
uint32_t intel82439tx_pci_r(int function, int reg, uint32_t mem_mask);
void intel82439tx_pci_w(int function, int reg, uint32_t data, uint32_t mem_mask);
uint8_t piix4_config_r(int function, int reg);
void piix4_config_w(int function, int reg, uint8_t data);
uint32_t intel82371ab_pci_r(int function, int reg, uint32_t mem_mask);
void intel82371ab_pci_w(int function, int reg, uint32_t data, uint32_t mem_mask);
std::unique_ptr<uint32_t[]> m_bios_ram;
uint8_t m_mtxc_config_reg[256];
uint8_t m_piix4_config_reg[4][256];
};
#if !TAITOWLF_ENABLE_VGA
uint32_t taitowlf_state::screen_update_taitowlf(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect)
{
bitmap.fill(m_palette->black_pen(), cliprect);
int count = 0;
for(int y=0;y<256;y++)
{
for(int x=0;x<512;x++)
{
uint32_t color = m_bootscreen_rom[count] & 0xff;
if(cliprect.contains(x+0, y))
bitmap.pix(y, x+0) = m_palette->pen(color);
count++;
}
}
return 0;
}
#endif
// Intel 82439TX System Controller (MTXC)
uint8_t taitowlf_state::mtxc_config_r(int function, int reg)
{
// osd_printf_debug("MTXC: read %d, %02X\n", function, reg);
return m_mtxc_config_reg[reg];
}
void taitowlf_state::mtxc_config_w(int function, int reg, uint8_t data)
{
// osd_printf_debug("%s:MTXC: write %d, %02X, %02X\n", machine().describe_context(), function, reg, data);
switch(reg)
{
case 0x59: // PAM0
{
if (data & 0x10) // enable RAM access to region 0xf0000 - 0xfffff
{
m_bank1->set_entry(1);
}
else // disable RAM access (reads go to BIOS ROM)
{
m_bank1->set_entry(0);
}
break;
}
}
m_mtxc_config_reg[reg] = data;
}
void taitowlf_state::intel82439tx_init()
{
m_mtxc_config_reg[0x60] = 0x02;
m_mtxc_config_reg[0x61] = 0x02;
m_mtxc_config_reg[0x62] = 0x02;
m_mtxc_config_reg[0x63] = 0x02;
m_mtxc_config_reg[0x64] = 0x02;
m_mtxc_config_reg[0x65] = 0x02;
}
uint32_t taitowlf_state::intel82439tx_pci_r(int function, int reg, uint32_t mem_mask)
{
uint32_t r = 0;
if (ACCESSING_BITS_24_31)
r |= mtxc_config_r(function, reg + 3) << 24;
if (ACCESSING_BITS_16_23)
r |= mtxc_config_r(function, reg + 2) << 16;
if (ACCESSING_BITS_8_15)
r |= mtxc_config_r(function, reg + 1) << 8;
if (ACCESSING_BITS_0_7)
r |= mtxc_config_r(function, reg + 0) << 0;
return r;
}
void taitowlf_state::intel82439tx_pci_w(int function, int reg, uint32_t data, uint32_t mem_mask)
{
if (ACCESSING_BITS_24_31)
mtxc_config_w(function, reg + 3, (data >> 24) & 0xff);
if (ACCESSING_BITS_16_23)
mtxc_config_w(function, reg + 2, (data >> 16) & 0xff);
if (ACCESSING_BITS_8_15)
mtxc_config_w(function, reg + 1, (data >> 8) & 0xff);
if (ACCESSING_BITS_0_7)
mtxc_config_w(function, reg + 0, (data >> 0) & 0xff);
}
// Intel 82371AB PCI-to-ISA / IDE bridge (PIIX4)
uint8_t taitowlf_state::piix4_config_r(int function, int reg)
{
// osd_printf_debug("PIIX4: read %d, %02X\n", function, reg);
return m_piix4_config_reg[function][reg];
}
void taitowlf_state::piix4_config_w(int function, int reg, uint8_t data)
{
// osd_printf_debug("%s:PIIX4: write %d, %02X, %02X\n", machine().describe_context(), function, reg, data);
m_piix4_config_reg[function][reg] = data;
}
uint32_t taitowlf_state::intel82371ab_pci_r(int function, int reg, uint32_t mem_mask)
{
uint32_t r = 0;
if (ACCESSING_BITS_24_31)
r |= piix4_config_r(function, reg + 3) << 24;
if (ACCESSING_BITS_16_23)
r |= piix4_config_r(function, reg + 2) << 16;
if (ACCESSING_BITS_8_15)
r |= piix4_config_r(function, reg + 1) << 8;
if (ACCESSING_BITS_0_7)
r |= piix4_config_r(function, reg + 0) << 0;
return r;
}
void taitowlf_state::intel82371ab_pci_w(int function, int reg, uint32_t data, uint32_t mem_mask)
{
if (ACCESSING_BITS_24_31)
piix4_config_w(function, reg + 3, (data >> 24) & 0xff);
if (ACCESSING_BITS_16_23)
piix4_config_w(function, reg + 2, (data >> 16) & 0xff);
if (ACCESSING_BITS_8_15)
piix4_config_w(function, reg + 1, (data >> 8) & 0xff);
if (ACCESSING_BITS_0_7)
piix4_config_w(function, reg + 0, (data >> 0) & 0xff);
}
// ISA Plug-n-Play
void taitowlf_state::pnp_config_w(offs_t offset, uint32_t data, uint32_t mem_mask)
{
if (ACCESSING_BITS_8_15)
{
// osd_printf_debug("PNP Config: %02X\n", (data >> 8) & 0xff);
}
}
void taitowlf_state::pnp_data_w(offs_t offset, uint32_t data, uint32_t mem_mask)
{
if (ACCESSING_BITS_8_15)
{
// osd_printf_debug("PNP Data: %02X\n", (data >> 8) & 0xff);
}
}
void taitowlf_state::bios_ram_w(offs_t offset, uint32_t data, uint32_t mem_mask)
{
if (m_mtxc_config_reg[0x59] & 0x20) // write to RAM if this region is write-enabled
{
COMBINE_DATA(m_bios_ram.get() + offset);
}
}
void taitowlf_state::taitowlf_map(address_map &map)
{
map(0x00000000, 0x0009ffff).ram();
#if TAITOWLF_ENABLE_VGA
map(0x000a0000, 0x000bffff).rw("vga", FUNC(vga_device::mem_r), FUNC(vga_device::mem_w));
#else
map(0x000a0000, 0x000bffff).ram();
#endif
#if TAITOWLF_ENABLE_VGA
map(0x000c0000, 0x000c7fff).ram().region("video_bios", 0);
#else
map(0x000c0000, 0x000c7fff).noprw();
#endif
map(0x000e0000, 0x000effff).ram();
map(0x000f0000, 0x000fffff).bankr("bank1");
map(0x000f0000, 0x000fffff).w(FUNC(taitowlf_state::bios_ram_w));
map(0x00100000, 0x01ffffff).ram();
// map(0xf8000000, 0xf83fffff).rom().region("user3", 0);
map(0xfffc0000, 0xffffffff).rom().region("bios", 0); /* System BIOS */
}
void taitowlf_state::taitowlf_io(address_map &map)
{
pcat32_io_common(map);
map(0x00e8, 0x00eb).noprw();
map(0x0300, 0x03af).noprw();
map(0x0278, 0x027b).w(FUNC(taitowlf_state::pnp_config_w));
#if TAITOWLF_ENABLE_VGA
map(0x03b0, 0x03bf).rw("vga", FUNC(vga_device::port_03b0_r), FUNC(vga_device::port_03b0_w));
map(0x03c0, 0x03cf).rw("vga", FUNC(vga_device::port_03c0_r), FUNC(vga_device::port_03c0_w));
map(0x03d0, 0x03df).rw("vga", FUNC(vga_device::port_03d0_r), FUNC(vga_device::port_03d0_w));
#else
map(0x03b0, 0x03df).noprw();
#endif
map(0x0a78, 0x0a7b).w(FUNC(taitowlf_state::pnp_data_w));
map(0x0cf8, 0x0cff).rw("pcibus", FUNC(pci_bus_legacy_device::read), FUNC(pci_bus_legacy_device::write));
}
/*****************************************************************************/
void taitowlf_state::machine_start()
{
for (int i = 0; i < 4; i++)
std::fill(std::begin(m_piix4_config_reg[i]), std::end(m_piix4_config_reg[i]), 0);
}
void taitowlf_state::machine_reset()
{
// disable RAM access (reads go to BIOS ROM)
m_bank1->set_entry(0);
}
#if !TAITOWLF_ENABLE_VGA
/* debug purpose*/
void taitowlf_state::taitowlf_palette(palette_device &palette) const
{
palette.set_pen_color(0x70, rgb_t(0xff, 0xff, 0xff));
palette.set_pen_color(0x71, rgb_t(0xff, 0xff, 0xff));
palette.set_pen_color(0x01, rgb_t(0x55, 0x00, 0x00));
palette.set_pen_color(0x10, rgb_t(0xaa, 0x00, 0x00));
palette.set_pen_color(0x00, rgb_t(0x00, 0x00, 0x00));
}
#endif
void taitowlf_state::taitowlf(machine_config &config)
{
/* basic machine hardware */
PENTIUM(config, m_maincpu, 200000000);
m_maincpu->set_addrmap(AS_PROGRAM, &taitowlf_state::taitowlf_map);
m_maincpu->set_addrmap(AS_IO, &taitowlf_state::taitowlf_io);
m_maincpu->set_irq_acknowledge_callback("pic8259_1", FUNC(pic8259_device::inta_cb));
pci_bus_legacy_device &pcibus(PCI_BUS_LEGACY(config, "pcibus", 0, 0));
pcibus.set_device(0, FUNC(taitowlf_state::intel82439tx_pci_r), FUNC(taitowlf_state::intel82439tx_pci_w));
pcibus.set_device(7, FUNC(taitowlf_state::intel82371ab_pci_r), FUNC(taitowlf_state::intel82371ab_pci_w));
pcat_common(config);
/* video hardware */
#if TAITOWLF_ENABLE_VGA
pcvideo_vga(config);
#else
screen_device &screen(SCREEN(config, "screen", SCREEN_TYPE_RASTER));
screen.set_refresh_hz(60);
screen.set_vblank_time(ATTOSECONDS_IN_USEC(0));
screen.set_size(512, 256);
screen.set_visarea(0, 512-1, 0, 256-1);
screen.set_screen_update(FUNC(taitowlf_state::screen_update_taitowlf));
PALETTE(config, m_palette, FUNC(taitowlf_state::taitowlf_palette), 256);
#endif
}
void taitowlf_state::init_taitowlf()
{
m_bios_ram = std::make_unique<uint32_t[]>(0x10000/4);
m_bank1->configure_entry(1, m_bios_ram.get());
m_bank1->configure_entry(0, memregion("bios")->base() + 0x30000);
intel82439tx_init();
}
/*****************************************************************************/
ROM_START(pf2012)
ROM_REGION32_LE(0x40000, "bios", 0)
ROM_LOAD("p5tx-la.bin", 0x00000, 0x40000, CRC(072e6d51) SHA1(70414349b37e478fc28ecbaba47ad1033ae583b7))
#if TAITOWLF_ENABLE_VGA
ROM_REGION( 0x8000, "video_bios", 0 ) // debug
ROM_LOAD16_BYTE( "trident_tgui9680_bios.bin", 0x0000, 0x4000, BAD_DUMP CRC(1eebde64) SHA1(67896a854d43a575037613b3506aea6dae5d6a19) )
ROM_CONTINUE( 0x0001, 0x4000 )
#endif
ROM_REGION(0x400000, "user3", 0) // Program ROM
ROM_LOAD("u1.bin", 0x000000, 0x200000, CRC(8f4c09cb) SHA1(0969a92fec819868881683c580f9e01cbedf4ad2))
ROM_LOAD("u2.bin", 0x200000, 0x200000, CRC(59881781) SHA1(85ff074ab2a922eac37cf96f0bf153a2dac55aa4))
ROM_REGION(0x4000000, "user4", 0) // Data ROM
ROM_LOAD("e59-01.u20", 0x0000000, 0x800000, CRC(701d3a9a) SHA1(34c9f34f4da34bb8eed85a4efd1d9eea47a21d77) )
ROM_LOAD("e59-02.u23", 0x0800000, 0x800000, CRC(626df682) SHA1(35bb4f91201734ce7ccdc640a75030aaca3d1151) )
ROM_LOAD("e59-03.u26", 0x1000000, 0x800000, CRC(74e4efde) SHA1(630235c2e4a11f615b5f3b8c93e1e645da09eefe) )
ROM_LOAD("e59-04.u21", 0x1800000, 0x800000, CRC(c900e8df) SHA1(93c06b8f5082e33f0dcc41f1be6a79283de16c40) )
ROM_LOAD("e59-05.u24", 0x2000000, 0x800000, CRC(85b0954c) SHA1(1b533d5888d56d1510c79f790e4fa708f77e836f) )
ROM_LOAD("e59-06.u27", 0x2800000, 0x800000, CRC(0573a113) SHA1(ee76a71dfd31289a9a5428653a36d01d914fc5d9) )
ROM_LOAD("e59-07.u22", 0x3000000, 0x800000, CRC(1f0ddcdc) SHA1(72ffe08f5effab093bdfe9863f8a11f80e914272) )
ROM_LOAD("e59-08.u25", 0x3800000, 0x800000, CRC(8db38ffd) SHA1(4b71ea86fb774ba6a8ac45abf4191af64af007e7) )
ROM_REGION(0x1400000, "samples", 0) // ZOOM sample data
ROM_LOAD("e59-09.u29", 0x0000000, 0x800000, CRC(d0da5c50) SHA1(56fb3c38f35244720d32a44fed28e6b58c7851f7) )
ROM_LOAD("e59-10.u32", 0x0800000, 0x800000, CRC(4c0e0a5c) SHA1(6454befa3a1dd532eb2a760129dcd7e611508730) )
ROM_LOAD("e59-11.u33", 0x1000000, 0x400000, CRC(c90a896d) SHA1(2b62992f20e4ca9634e7953fe2c553906de44f04) )
ROM_REGION(0x180000, "cpu1", 0) // MN10200 program
ROM_LOAD("e59-12.u13", 0x000000, 0x80000, CRC(9a473a7e) SHA1(b0ec7b0ae2b33a32da98899aa79d44e8e318ceb7) )
ROM_LOAD("e59-13.u15", 0x080000, 0x80000, CRC(77719880) SHA1(8382dd2dfb0dae60a3831ed6d3ff08539e2d94eb) )
ROM_LOAD("e59-14.u14", 0x100000, 0x40000, CRC(d440887c) SHA1(d965871860d757bc9111e9adb2303a633c662d6b) )
ROM_LOAD("e59-15.u16", 0x140000, 0x40000, CRC(eae8e523) SHA1(8a054d3ded7248a7906c4f0bec755ddce53e2023) )
ROM_REGION(0x20000, "bootscreen", 0) // bootscreen
ROM_LOAD("e58-04.u71", 0x000000, 0x20000, CRC(500e6113) SHA1(93226706517c02e336f96bdf9443785158e7becf) )
ROM_END
} // Anonymous namespace
/*****************************************************************************/
GAME(1997, pf2012, 0, taitowlf, 0, taitowlf_state, init_taitowlf, ROT0, "Taito", "Psychic Force 2012", MACHINE_NOT_WORKING | MACHINE_NO_SOUND)