mirror of
https://github.com/holub/mame
synced 2025-04-16 13:34:55 +03:00
- split off the handhelds that seem to be based around gpl162xx technology into their own file from generalplus_gpl_unknown.cpp (#7312)
* - split off the handhelds that seem to be based around gpl162xx technology into their own file from generalplus_gpl_unknown.cpp to generalplus_gpl162xx_lcdtype.cpp - identify unkunsp based on new information * typo
This commit is contained in:
parent
51a4222299
commit
ddb13a5861
@ -4032,6 +4032,7 @@ files {
|
||||
MAME_DIR .. "src/mame/machine/generalplus_gpl16250.cpp",
|
||||
MAME_DIR .. "src/mame/machine/generalplus_gpl16250.h",
|
||||
MAME_DIR .. "src/mame/drivers/generalplus_gpl32612.cpp",
|
||||
MAME_DIR .. "src/mame/drivers/generalplus_gpl162xx_lcdtype.cpp",
|
||||
MAME_DIR .. "src/mame/drivers/generalplus_gpl_unknown.cpp",
|
||||
MAME_DIR .. "src/mame/drivers/xavix.cpp",
|
||||
MAME_DIR .. "src/mame/includes/xavix.h",
|
||||
|
886
src/mame/drivers/generalplus_gpl162xx_lcdtype.cpp
Normal file
886
src/mame/drivers/generalplus_gpl162xx_lcdtype.cpp
Normal file
@ -0,0 +1,886 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:David Haywood
|
||||
|
||||
/* These contain a similar game selection to the devices in unk6502_st2xxx.cpp but on updated hardware
|
||||
|
||||
The hardware appears to be an abuse of the GPL16250 SoC. The palette and sprite banks are used, but as work-ram
|
||||
rather than for their intended purpose, and the rest of the GPL16250 video hardware is either entirely bypassed
|
||||
or doesn't exist. All video is software rendered and output directly to the LCD Controller.
|
||||
|
||||
If this is confirmed via a decap then this should be merged with the GPL16250 implementation.
|
||||
|
||||
The coding of these is similar to the unk6502_st2xxx.cpp too, with all game specific function calls being loaded
|
||||
on the fly from the SPI to a tiny portion of work RAM, with graphics likewise being loaded and decompressed for
|
||||
every draw call.
|
||||
|
||||
pcp8718 / pcp8728 / bkid218
|
||||
to access test mode hold up + left on startup.
|
||||
(bkid218 cursor in test menu can't be moved in emulation, works on unit, why?)
|
||||
*/
|
||||
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
#include "cpu/unsp/unsp.h"
|
||||
#include "machine/bl_handhelds_menucontrol.h"
|
||||
|
||||
#include "emupal.h"
|
||||
#include "screen.h"
|
||||
#include "speaker.h"
|
||||
|
||||
#define LOG_GPL162XX_LCDTYPE (1U << 1)
|
||||
#define LOG_GPL162XX_LCDTYPE_SELECT_SIM (1U << 2)
|
||||
#define LOG_GPL162XX_LCDTYPE_IO_PORT (1U << 3)
|
||||
|
||||
#define LOG_ALL (LOG_GPL162XX_LCDTYPE | LOG_GPL162XX_LCDTYPE_SELECT_SIM | LOG_GPL162XX_LCDTYPE_IO_PORT)
|
||||
#define VERBOSE (0)
|
||||
|
||||
#include "logmacro.h"
|
||||
|
||||
|
||||
class gpl162xx_lcdtype_state : public driver_device
|
||||
{
|
||||
public:
|
||||
gpl162xx_lcdtype_state(const machine_config &mconfig, device_type type, const char *tag) :
|
||||
driver_device(mconfig, type, tag),
|
||||
m_maincpu(*this, "maincpu"),
|
||||
m_mainrom(*this, "maincpu"),
|
||||
m_mainram(*this, "mainram"),
|
||||
m_palette(*this, "palette"),
|
||||
m_screen(*this, "screen"),
|
||||
m_spirom(*this, "spi"),
|
||||
m_io_in0(*this, "IN0"),
|
||||
m_io_in1(*this, "IN1"),
|
||||
m_menucontrol(*this, "menucontrol")
|
||||
{ }
|
||||
|
||||
void gpl162xx_lcdtype(machine_config &config);
|
||||
|
||||
private:
|
||||
virtual void machine_start() override;
|
||||
virtual void machine_reset() override;
|
||||
|
||||
uint32_t screen_update(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect);
|
||||
|
||||
required_device<unsp_20_device> m_maincpu;
|
||||
required_region_ptr<uint16_t> m_mainrom;
|
||||
required_shared_ptr<uint16_t> m_mainram;
|
||||
|
||||
required_device<palette_device> m_palette;
|
||||
required_device<screen_device> m_screen;
|
||||
|
||||
void map(address_map &map);
|
||||
|
||||
required_region_ptr<uint8_t> m_spirom;
|
||||
|
||||
uint16_t unk_7abf_r();
|
||||
uint16_t io_7860_r();
|
||||
uint16_t unk_780f_r();
|
||||
|
||||
void io_7860_w(uint16_t data);
|
||||
uint16_t m_7860;
|
||||
|
||||
void unk_7862_w(uint16_t data);
|
||||
void unk_7863_w(uint16_t data);
|
||||
|
||||
void unk_7868_w(uint16_t data);
|
||||
uint16_t unk_7868_r();
|
||||
uint16_t m_7868;
|
||||
|
||||
void bankswitch_707e_w(uint16_t data);
|
||||
uint16_t bankswitch_707e_r();
|
||||
uint16_t m_707e_bank;
|
||||
|
||||
void bankswitch_703a_w(uint16_t data);
|
||||
uint16_t bankswitch_703a_r();
|
||||
uint16_t m_703a_bank;
|
||||
|
||||
void bankedram_7300_w(offs_t offset, uint16_t data);
|
||||
uint16_t bankedram_7300_r(offs_t offset);
|
||||
uint16_t m_bankedram_7300[0x400];
|
||||
|
||||
void bankedram_7400_w(offs_t offset, uint16_t data);
|
||||
uint16_t bankedram_7400_r(offs_t offset);
|
||||
uint16_t m_bankedram_7400[0x800];
|
||||
|
||||
void system_dma_params_channel0_w(offs_t offset, uint16_t data);
|
||||
uint16_t system_dma_params_channel0_r(offs_t offset);
|
||||
|
||||
uint16_t m_dmaregs[8];
|
||||
|
||||
void lcd_w(uint16_t data);
|
||||
void lcd_command_w(uint16_t data);
|
||||
|
||||
uint16_t spi_misc_control_r();
|
||||
uint16_t spi_rx_fifo_r();
|
||||
void spi_tx_fifo_w(uint16_t data);
|
||||
|
||||
void spi_control_w(uint16_t data);
|
||||
|
||||
void spi_process_tx_data(uint8_t data);
|
||||
uint8_t spi_process_rx();
|
||||
uint8_t spi_rx();
|
||||
uint8_t spi_rx_fast();
|
||||
|
||||
uint8_t m_rx_fifo[5]; // actually 8 bytes? or 8 half-bytes?
|
||||
|
||||
uint32_t m_spiaddress;
|
||||
|
||||
uint16_t unk_78a1_r();
|
||||
uint16_t m_78a1;
|
||||
uint16_t unk_78d8_r();
|
||||
void unk_78d8_w(uint16_t data);
|
||||
|
||||
enum spistate : const int
|
||||
{
|
||||
SPI_STATE_READY = 0,
|
||||
SPI_STATE_WAITING_HIGH_ADDR = 1,
|
||||
SPI_STATE_WAITING_MID_ADDR = 2,
|
||||
SPI_STATE_WAITING_LOW_ADDR = 3,
|
||||
// probably not
|
||||
SPI_STATE_WAITING_DUMMY1_ADDR = 4,
|
||||
SPI_STATE_WAITING_DUMMY2_ADDR = 5,
|
||||
SPI_STATE_READING = 6,
|
||||
|
||||
SPI_STATE_WAITING_HIGH_ADDR_FAST = 8,
|
||||
SPI_STATE_WAITING_MID_ADDR_FAST = 9,
|
||||
SPI_STATE_WAITING_LOW_ADDR_FAST = 10,
|
||||
SPI_STATE_WAITING_LOW_ADDR_FAST_DUMMY = 11,
|
||||
SPI_STATE_READING_FAST = 12
|
||||
|
||||
};
|
||||
|
||||
spistate m_spistate;
|
||||
|
||||
enum lcdstate : const int
|
||||
{
|
||||
LCD_STATE_READY = 0,
|
||||
LCD_STATE_WAITING_FOR_COMMAND = 1,
|
||||
LCD_STATE_PROCESSING_COMMAND = 2
|
||||
};
|
||||
|
||||
lcdstate m_lcdstate;
|
||||
int m_lastlcdcommand;
|
||||
|
||||
uint8_t m_displaybuffer[0x40000];
|
||||
int m_lcdaddr;
|
||||
|
||||
uint16_t io_7870_r();
|
||||
required_ioport m_io_in0;
|
||||
required_ioport m_io_in1;
|
||||
required_device<bl_handhelds_menucontrol_device> m_menucontrol;
|
||||
|
||||
DECLARE_WRITE_LINE_MEMBER(screen_vblank);
|
||||
};
|
||||
|
||||
uint32_t gpl162xx_lcdtype_state::screen_update(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect)
|
||||
{
|
||||
int count = 0;
|
||||
for (int y = 0; y < 256; y++)
|
||||
{
|
||||
uint32_t *const dst = &bitmap.pix(y);
|
||||
|
||||
for (int x = 0; x < 320; x++)
|
||||
{
|
||||
// 8-bit values get pumped through a 256 word table in internal ROM and converted to words
|
||||
uint16_t dat = m_displaybuffer[(count * 2) + 1] | (m_displaybuffer[(count * 2) + 0] << 8);
|
||||
|
||||
int b = ((dat >> 0) & 0x1f) << 3;
|
||||
int g = ((dat >> 5) & 0x3f) << 2;
|
||||
int r = ((dat >> 11) & 0x1f) << 3;
|
||||
|
||||
dst[x] = (r << 16) | (g << 8) | (b << 0);
|
||||
count++;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static INPUT_PORTS_START( gpl162xx_lcdtype )
|
||||
PORT_START("IN0")
|
||||
PORT_BIT( 0x0001, IP_ACTIVE_HIGH, IPT_UNUSED ) // causes lag if state is inverted, investigate
|
||||
PORT_DIPNAME( 0x0002, 0x0002, "P0:0002" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0002, "0002" )
|
||||
PORT_DIPNAME( 0x0004, 0x0000, "Show Vs in Test Mode" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0004, "0004" )
|
||||
PORT_BIT( 0x0008, IP_ACTIVE_HIGH, IPT_CUSTOM ) PORT_READ_LINE_DEVICE_MEMBER("menucontrol", bl_handhelds_menucontrol_device, status_r)
|
||||
PORT_BIT( 0x0010, IP_ACTIVE_HIGH, IPT_CUSTOM ) PORT_READ_LINE_DEVICE_MEMBER("menucontrol", bl_handhelds_menucontrol_device, data_r)
|
||||
PORT_DIPNAME( 0x0020, 0x0020, "P0:0020" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0020, "0020" )
|
||||
PORT_DIPNAME( 0x0040, 0x0040, "P0:0040" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0040, "0040" )
|
||||
PORT_DIPNAME( 0x0080, 0x0080, "P0:0080" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0080, "0080" )
|
||||
PORT_DIPNAME( 0x0100, 0x0100, "P0:0100" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0100, "0100" )
|
||||
PORT_DIPNAME( 0x0200, 0x0000, "Battery Level" )
|
||||
PORT_DIPSETTING( 0x0000, "Normal" )
|
||||
PORT_DIPSETTING( 0x0200, "Low" )
|
||||
PORT_DIPNAME( 0x0400, 0x0400, "P0:0400" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0400, "0400" )
|
||||
PORT_DIPNAME( 0x0800, 0x0800, "P0:0800" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0800, "0800" )
|
||||
PORT_DIPNAME( 0x1000, 0x1000, "P0:1000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x1000, "1000" )
|
||||
PORT_DIPNAME( 0x2000, 0x2000, "P0:2000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x2000, "2000" )
|
||||
PORT_BIT( 0x4000, IP_ACTIVE_LOW, IPT_START1 )
|
||||
PORT_BIT( 0x8000, IP_ACTIVE_LOW, IPT_JOYSTICK_UP )
|
||||
|
||||
PORT_START("IN1")
|
||||
PORT_DIPNAME( 0x0001, 0x0001, "P1:0001" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0001, "0001" )
|
||||
PORT_DIPNAME( 0x0002, 0x0002, "P1:0002" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0002, "0002" )
|
||||
PORT_DIPNAME( 0x0004, 0x0004, "P1:0004" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0004, "0004" )
|
||||
PORT_DIPNAME( 0x0008, 0x0008, "P1:0008" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0008, "0008" )
|
||||
PORT_DIPNAME( 0x0010, 0x0010, "P1:0010" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0010, "0010" )
|
||||
PORT_BIT( 0x0020, IP_ACTIVE_LOW, IPT_BUTTON3 ) PORT_NAME("SOUND")
|
||||
PORT_BIT( 0x0040, IP_ACTIVE_LOW, IPT_JOYSTICK_DOWN )
|
||||
PORT_BIT( 0x0080, IP_ACTIVE_LOW, IPT_JOYSTICK_LEFT )
|
||||
PORT_BIT( 0x0100, IP_ACTIVE_LOW, IPT_JOYSTICK_RIGHT )
|
||||
PORT_BIT( 0x0200, IP_ACTIVE_LOW, IPT_BUTTON1 ) PORT_NAME("A")
|
||||
PORT_BIT( 0x0400, IP_ACTIVE_LOW, IPT_BUTTON2 ) PORT_NAME("B")
|
||||
PORT_BIT( 0x0800, IP_ACTIVE_LOW, IPT_BUTTON4 ) PORT_NAME("ON/OFF")
|
||||
PORT_DIPNAME( 0x1000, 0x1000, "P1:1000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x1000, "1000" )
|
||||
PORT_DIPNAME( 0x2000, 0x2000, "P1:2000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x2000, "2000" )
|
||||
PORT_DIPNAME( 0x4000, 0x4000, "P1:4000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x4000, "4000" )
|
||||
PORT_DIPNAME( 0x8000, 0x8000, "P1:8000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x8000, "8000" )
|
||||
INPUT_PORTS_END
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_7abf_r()
|
||||
{
|
||||
return 0x0001;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::io_7860_r()
|
||||
{
|
||||
uint16_t ret = m_io_in0->read();
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE_IO_PORT, "%s: io_7860_r %02x\n", machine().describe_context(), ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::io_7860_w(uint16_t data)
|
||||
{
|
||||
m_menucontrol->data_w((data & 0x10)>>4);
|
||||
m_menucontrol->clock_w((data & 0x20)>>5);
|
||||
|
||||
m_7860 = data;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::unk_7862_w(uint16_t data)
|
||||
{
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::unk_7863_w(uint16_t data)
|
||||
{
|
||||
// probably port direction (or 7862 is?)
|
||||
if (data == 0x3cf7)
|
||||
{
|
||||
m_menucontrol->reset_w(1);
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_780f_r()
|
||||
{
|
||||
return 0x0002;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::spi_misc_control_r()
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_misc_control_r\n", machine().describe_context());
|
||||
return 0x0000;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::spi_rx_fifo_r()
|
||||
{
|
||||
if (m_spistate == SPI_STATE_READING_FAST)
|
||||
return spi_rx_fast();
|
||||
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_rx_fifo_r\n", machine().describe_context());
|
||||
return spi_rx();
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::spi_process_tx_data(uint8_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"transmitting %02x\n", data);
|
||||
|
||||
switch (m_spistate)
|
||||
{
|
||||
case SPI_STATE_READY:
|
||||
{
|
||||
if (data == 0x03)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to read mode (need address) %02x\n", data);
|
||||
m_spistate = SPI_STATE_WAITING_HIGH_ADDR;
|
||||
}
|
||||
else if (data == 0x0b)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to fast read mode (need address) %02x\n", data);
|
||||
m_spistate = SPI_STATE_WAITING_HIGH_ADDR_FAST;
|
||||
//machine().debug_break();
|
||||
}
|
||||
else
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"invalid state request %02x\n", data);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_HIGH_ADDR:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xff00ffff) | data << 16;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to high address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_MID_ADDR;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_MID_ADDR:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffff00ff) | data << 8;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to mid address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_LOW_ADDR;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_LOW_ADDR:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffffff00) | data;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to low address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_READING;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_READING:
|
||||
{
|
||||
// writes when in read mode clock in data?
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"write while in read mode (clock data?)\n", data, m_spiaddress);
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_DUMMY1_ADDR:
|
||||
{
|
||||
m_spistate = SPI_STATE_WAITING_DUMMY2_ADDR;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_DUMMY2_ADDR:
|
||||
{
|
||||
// m_spistate = SPI_STATE_READY;
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
case SPI_STATE_WAITING_HIGH_ADDR_FAST:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xff00ffff) | data << 16;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to high address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_MID_ADDR_FAST;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_MID_ADDR_FAST:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffff00ff) | data << 8;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to mid address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_LOW_ADDR_FAST;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_LOW_ADDR_FAST:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffffff00) | data;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to low address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_LOW_ADDR_FAST_DUMMY;
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_LOW_ADDR_FAST_DUMMY:
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"dummy write %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_READING_FAST;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_READING_FAST:
|
||||
{
|
||||
// writes when in read mode clock in data?
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"write while in read mode (clock data?)\n", data, m_spiaddress);
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
uint8_t gpl162xx_lcdtype_state::spi_process_rx()
|
||||
{
|
||||
|
||||
switch (m_spistate)
|
||||
{
|
||||
case SPI_STATE_READING:
|
||||
case SPI_STATE_READING_FAST:
|
||||
{
|
||||
uint8_t dat = m_spirom[m_spiaddress & 0x7fffff];
|
||||
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"reading SPI %02x from SPI Address %08x (adjusted word offset %08x)\n", dat, m_spiaddress, (m_spiaddress/2)+0x20000);
|
||||
m_spiaddress++;
|
||||
return dat;
|
||||
}
|
||||
|
||||
default:
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"reading FIFO in unknown state\n");
|
||||
return 0x00;
|
||||
}
|
||||
}
|
||||
|
||||
return 0x00;
|
||||
}
|
||||
|
||||
|
||||
uint8_t gpl162xx_lcdtype_state::spi_rx()
|
||||
{
|
||||
uint8_t ret = m_rx_fifo[0];
|
||||
|
||||
m_rx_fifo[0] = m_rx_fifo[1];
|
||||
m_rx_fifo[1] = m_rx_fifo[2];
|
||||
m_rx_fifo[2] = m_rx_fifo[3];
|
||||
m_rx_fifo[3] = spi_process_rx();
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
uint8_t gpl162xx_lcdtype_state::spi_rx_fast()
|
||||
{
|
||||
uint8_t ret = m_rx_fifo[0];
|
||||
|
||||
m_rx_fifo[0] = m_rx_fifo[1];
|
||||
m_rx_fifo[1] = m_rx_fifo[2];
|
||||
m_rx_fifo[2] = m_rx_fifo[3];
|
||||
m_rx_fifo[3] = m_rx_fifo[4];
|
||||
m_rx_fifo[4] = spi_process_rx();
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::spi_tx_fifo_w(uint16_t data)
|
||||
{
|
||||
data &= 0x00ff;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_tx_fifo_w %04x\n", machine().describe_context(), data);
|
||||
|
||||
spi_process_tx_data(data);
|
||||
}
|
||||
|
||||
// this is probably 'port b' but when SPI is enabled some points of this can become SPI control pins
|
||||
// it's accessed after each large data transfer, probably to reset the SPI into 'ready for command' state?
|
||||
void gpl162xx_lcdtype_state::unk_7868_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE_IO_PORT, "%s: unk_7868_w %04x (Port B + SPI reset?)\n", machine().describe_context(), data);
|
||||
|
||||
if ((m_7868 & 0x0100) != (data & 0x0100))
|
||||
{
|
||||
if (!(data & 0x0100))
|
||||
{
|
||||
for (int i = 0; i < 4; i++)
|
||||
m_rx_fifo[i] = 0xff;
|
||||
|
||||
m_spistate = SPI_STATE_READY;
|
||||
}
|
||||
}
|
||||
|
||||
m_7868 = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_7868_r()
|
||||
{
|
||||
return m_7868;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::bankswitch_707e_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: bankswitch_707e_w %04x\n", machine().describe_context(), data);
|
||||
m_707e_bank = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankswitch_707e_r()
|
||||
{
|
||||
return m_707e_bank;
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::bankswitch_703a_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: bankswitch_703a_w %04x\n", machine().describe_context(), data);
|
||||
m_703a_bank = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankswitch_703a_r()
|
||||
{
|
||||
return m_703a_bank;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::bankedram_7300_w(offs_t offset, uint16_t data)
|
||||
{
|
||||
offset |= (m_703a_bank & 0x000c) << 6;
|
||||
m_bankedram_7300[offset] = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankedram_7300_r(offs_t offset)
|
||||
{
|
||||
offset |= (m_703a_bank & 0x000c) << 6;
|
||||
return m_bankedram_7300[offset];
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::bankedram_7400_w(offs_t offset, uint16_t data)
|
||||
{
|
||||
if (m_707e_bank & 1)
|
||||
{
|
||||
m_bankedram_7400[offset + 0x400] = data;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_bankedram_7400[offset] = data;
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankedram_7400_r(offs_t offset)
|
||||
{
|
||||
if (m_707e_bank & 1)
|
||||
{
|
||||
return m_bankedram_7400[offset + 0x400];
|
||||
}
|
||||
else
|
||||
{
|
||||
return m_bankedram_7400[offset];
|
||||
}
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::system_dma_params_channel0_w(offs_t offset, uint16_t data)
|
||||
{
|
||||
m_dmaregs[offset] = data;
|
||||
|
||||
switch (offset)
|
||||
{
|
||||
case 0:
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Mode)\n", machine().describe_context(), offset, data);
|
||||
|
||||
uint16_t mode = m_dmaregs[0];
|
||||
uint32_t source = m_dmaregs[1] | (m_dmaregs[4] << 16);
|
||||
uint32_t dest = m_dmaregs[2] | (m_dmaregs[5] << 16) ;
|
||||
uint32_t length = m_dmaregs[3] | (m_dmaregs[6] << 16);
|
||||
|
||||
if ((mode != 0x0200) && (mode != 0x4009) && (mode != 0x6009))
|
||||
fatalerror("unknown dma mode write %04x\n", data);
|
||||
|
||||
if ((mode == 0x4009) || (mode == 0x6009))
|
||||
{
|
||||
address_space& mem = m_maincpu->space(AS_PROGRAM);
|
||||
|
||||
for (int i = 0; i < length; i++)
|
||||
{
|
||||
uint16_t dat = mem.read_word(source);
|
||||
|
||||
if (mode & 0x2000)
|
||||
{
|
||||
// Racing Car and Elevator Action need this logic
|
||||
mem.write_word(dest, dat & 0xff);
|
||||
dest++;
|
||||
mem.write_word(dest, dat >> 8);
|
||||
dest++;
|
||||
}
|
||||
else
|
||||
{
|
||||
mem.write_word(dest, dat);
|
||||
dest++;
|
||||
}
|
||||
|
||||
source++;
|
||||
}
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
case 1:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Source Low)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 2:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Dest Low)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 3:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Length Low)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 4:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Source High)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 5:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Dest High)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 6:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Length High)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 7:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA unknown)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::system_dma_params_channel0_r(offs_t offset)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_r %01x\n", machine().describe_context(), offset);
|
||||
return m_dmaregs[offset];
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::io_7870_r()
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE_IO_PORT, "%s: io_7870_r (IO port)\n", machine().describe_context() );
|
||||
return m_io_in1->read();
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::spi_control_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_control_w %04x\n", machine().describe_context(), data);
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_78a1_r()
|
||||
{
|
||||
// checked in interrupt, code skipped entirely if this isn't set
|
||||
return m_78a1;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_78d8_r()
|
||||
{
|
||||
return 0xffff;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::unk_78d8_w(uint16_t data)
|
||||
{
|
||||
// written in IRQ, possible ack
|
||||
if (data & 0x8000)
|
||||
m_78a1 &= ~0x8000;
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::map(address_map &map)
|
||||
{
|
||||
// there are calls to 01xxx and 02xxx regions
|
||||
// (RAM populated by internal ROM?, TODO: check to make sure code copied there isn't from SPI ROM like the GPL16250 bootstrap
|
||||
// does from NAND, it doesn't seem to have a header in the same format at least)
|
||||
map(0x000000, 0x006fff).ram().share("mainram");
|
||||
|
||||
// registers at 7xxx are similar to GPL16250, but not identical? (different video system? or just GPL16250 with the video part unused?)
|
||||
|
||||
map(0x00703a, 0x00703a).rw(FUNC(gpl162xx_lcdtype_state::bankswitch_703a_r), FUNC(gpl162xx_lcdtype_state::bankswitch_703a_w));
|
||||
map(0x00707e, 0x00707e).rw(FUNC(gpl162xx_lcdtype_state::bankswitch_707e_r), FUNC(gpl162xx_lcdtype_state::bankswitch_707e_w));
|
||||
|
||||
map(0x007100, 0x0071ff).ram(); // rowscroll on gpl16250
|
||||
map(0x007300, 0x0073ff).rw(FUNC(gpl162xx_lcdtype_state::bankedram_7300_r), FUNC(gpl162xx_lcdtype_state::bankedram_7300_w)); // palette on gpl16250
|
||||
map(0x007400, 0x0077ff).rw(FUNC(gpl162xx_lcdtype_state::bankedram_7400_r), FUNC(gpl162xx_lcdtype_state::bankedram_7400_w)); // spriteram on gpl16250
|
||||
|
||||
map(0x00780f, 0x00780f).r(FUNC(gpl162xx_lcdtype_state::unk_780f_r));
|
||||
|
||||
|
||||
map(0x007860, 0x007860).rw(FUNC(gpl162xx_lcdtype_state::io_7860_r),FUNC(gpl162xx_lcdtype_state::io_7860_w)); // Port A?
|
||||
map(0x007862, 0x007862).w(FUNC(gpl162xx_lcdtype_state::unk_7862_w));
|
||||
map(0x007863, 0x007863).w(FUNC(gpl162xx_lcdtype_state::unk_7863_w));
|
||||
|
||||
map(0x007868, 0x007868).rw(FUNC(gpl162xx_lcdtype_state::unk_7868_r), FUNC(gpl162xx_lcdtype_state::unk_7868_w)); // Port B?
|
||||
|
||||
map(0x007870, 0x007870).r(FUNC(gpl162xx_lcdtype_state::io_7870_r)); // Port C?
|
||||
|
||||
map(0x0078a1, 0x0078a1).r(FUNC(gpl162xx_lcdtype_state::unk_78a1_r));
|
||||
|
||||
map(0x0078d8, 0x0078d8).rw(FUNC(gpl162xx_lcdtype_state::unk_78d8_r), FUNC(gpl162xx_lcdtype_state::unk_78d8_w));
|
||||
|
||||
|
||||
map(0x007940, 0x007940).w(FUNC(gpl162xx_lcdtype_state::spi_control_w));
|
||||
// 7941 SPI Transmit Status
|
||||
map(0x007942, 0x007942).w(FUNC(gpl162xx_lcdtype_state::spi_tx_fifo_w));
|
||||
// 7943 SPI Receive Status
|
||||
map(0x007944, 0x007944).r(FUNC(gpl162xx_lcdtype_state::spi_rx_fifo_r));
|
||||
map(0x007945, 0x007945).r(FUNC(gpl162xx_lcdtype_state::spi_misc_control_r));
|
||||
|
||||
map(0x007a80, 0x007a87).rw(FUNC(gpl162xx_lcdtype_state::system_dma_params_channel0_r), FUNC(gpl162xx_lcdtype_state::system_dma_params_channel0_w));
|
||||
|
||||
map(0x007abf, 0x007abf).r(FUNC(gpl162xx_lcdtype_state::unk_7abf_r));
|
||||
|
||||
// there are calls to 0x0f000 (internal ROM?)
|
||||
map(0x00f000, 0x00ffff).rom().region("maincpu", 0x00000);
|
||||
|
||||
// external LCD controller
|
||||
map(0x200000, 0x200000).w(FUNC(gpl162xx_lcdtype_state::lcd_command_w));
|
||||
map(0x20fc00, 0x20fc00).w(FUNC(gpl162xx_lcdtype_state::lcd_w));
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::lcd_command_w(uint16_t data)
|
||||
{
|
||||
data &= 0xff;
|
||||
|
||||
switch (m_lcdstate)
|
||||
{
|
||||
case LCD_STATE_READY:
|
||||
case LCD_STATE_PROCESSING_COMMAND:
|
||||
{
|
||||
if (data == 0x0000)
|
||||
{
|
||||
m_lcdstate = LCD_STATE_WAITING_FOR_COMMAND;
|
||||
m_lastlcdcommand = 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case LCD_STATE_WAITING_FOR_COMMAND:
|
||||
{
|
||||
m_lastlcdcommand = data;
|
||||
m_lcdstate = LCD_STATE_PROCESSING_COMMAND;
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::lcd_w(uint16_t data)
|
||||
{
|
||||
data &= 0xff; // definitely looks like 8-bit port as 16-bit values are shifted and rewritten
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: lcd_w %02x\n", machine().describe_context(), data);
|
||||
|
||||
if ((m_lcdstate == LCD_STATE_PROCESSING_COMMAND) && (m_lastlcdcommand == 0x22))
|
||||
{
|
||||
m_displaybuffer[m_lcdaddr] = data;
|
||||
m_lcdaddr++;
|
||||
|
||||
if (m_lcdaddr >= ((320 * 240) * 2))
|
||||
m_lcdaddr = 0;
|
||||
|
||||
//m_lcdaddr &= 0x3ffff;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::machine_start()
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::machine_reset()
|
||||
{
|
||||
m_spistate = SPI_STATE_READY;
|
||||
m_spiaddress = 0;
|
||||
|
||||
for (int i = 0; i < 320*240*2; i++)
|
||||
m_displaybuffer[i] = 0x00;
|
||||
|
||||
m_lcdaddr = 0;
|
||||
|
||||
m_lcdstate = LCD_STATE_READY;
|
||||
m_lastlcdcommand = 0;
|
||||
|
||||
m_78a1 = 0;
|
||||
|
||||
// first menu index is stored here
|
||||
// m_spirom[0x16000] = gamenum & 0xff;
|
||||
// m_spirom[0x16001] = (gamenum>>8) & 0xff;
|
||||
}
|
||||
|
||||
WRITE_LINE_MEMBER(gpl162xx_lcdtype_state::screen_vblank)
|
||||
{
|
||||
if (state)
|
||||
{
|
||||
// probably a timer
|
||||
m_maincpu->set_input_line(UNSP_IRQ4_LINE, ASSERT_LINE);
|
||||
m_78a1 |= 0x8000;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_maincpu->set_input_line(UNSP_IRQ4_LINE, CLEAR_LINE);
|
||||
}
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::gpl162xx_lcdtype(machine_config &config)
|
||||
{
|
||||
|
||||
UNSP_20(config, m_maincpu, 96000000); // unknown CPU, unsp20 based, 96Mhz is listed as the maximum for most unSP2.0 chips, and appears correct here
|
||||
m_maincpu->set_addrmap(AS_PROGRAM, &gpl162xx_lcdtype_state::map);
|
||||
|
||||
SCREEN(config, m_screen, SCREEN_TYPE_RASTER);
|
||||
m_screen->set_refresh_hz(60);
|
||||
m_screen->set_vblank_time(ATTOSECONDS_IN_USEC(10));
|
||||
m_screen->set_size(64*8, 32*8);
|
||||
m_screen->set_visarea(0*8, 320-1, 0*8, 240-1);
|
||||
m_screen->set_screen_update(FUNC(gpl162xx_lcdtype_state::screen_update));
|
||||
m_screen->screen_vblank().set(FUNC(gpl162xx_lcdtype_state::screen_vblank));
|
||||
|
||||
PALETTE(config, m_palette).set_format(palette_device::xBGR_555, 0x8000);
|
||||
|
||||
BL_HANDHELDS_MENUCONTROL(config, m_menucontrol, 0);
|
||||
m_menucontrol->set_is_unsp_type_hack();
|
||||
|
||||
}
|
||||
|
||||
// pcp8718 and pcp8728 both contain user data (player name?) and will need to be factory defaulted once they work
|
||||
// the ROM code is slightly different between them
|
||||
|
||||
ROM_START( pcp8718 )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "internal.rom", 0x000000, 0x2000, CRC(ea119561) SHA1(a2680577e20fe1155efc40a5781cf1ec80ccec3a) )
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF )
|
||||
//ROM_LOAD16_WORD_SWAP( "8718_en25f32.bin", 0x000000, 0x400000, CRC(cc138db4) SHA1(379af3d94ae840f52c06416d6cf32e25923af5ae) ) // bad dump, some blocks are corrupt
|
||||
ROM_LOAD( "eyecare_25q32av1g_ef4016.bin", 0x000000, 0x400000, CRC(58415e10) SHA1(b1adcc03f2ad8d741544204671677740e904ce1a) )
|
||||
ROM_END
|
||||
|
||||
ROM_START( pcp8728 )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "internal.rom", 0x000000, 0x2000, CRC(ea119561) SHA1(a2680577e20fe1155efc40a5781cf1ec80ccec3a) )
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "pcp 8728 788 in 1.bin", 0x000000, 0x400000, CRC(60115f21) SHA1(e15c39f11e442a76fae3823b6d510178f6166926) )
|
||||
ROM_END
|
||||
|
||||
ROM_START( bkid218 )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "internal.rom", 0x000000, 0x2000, CRC(ea119561) SHA1(a2680577e20fe1155efc40a5781cf1ec80ccec3a) )
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "218n1_25q64csig_c84017.bin", 0x000000, 0x800000, CRC(94f35dbd) SHA1(a1bd6defd2465ae14753cd83be5c31f99e9158ec) )
|
||||
ROM_END
|
||||
|
||||
|
||||
CONS( 200?, pcp8718, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "PCP", "PCP 8718 - HD 360 Degrees Rocker Palm Eyecare Console - 788 in 1", MACHINE_NOT_WORKING | MACHINE_NO_SOUND )
|
||||
CONS( 200?, pcp8728, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "PCP", "PCP 8728 - 788 in 1", MACHINE_NOT_WORKING | MACHINE_NO_SOUND ) // what name was this sold under?
|
||||
CONS( 200?, bkid218, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "BornKid", "Handheld Game Console BC-19 - 218 in 1", MACHINE_NOT_WORKING | MACHINE_NO_SOUND )
|
@ -1,60 +1,31 @@
|
||||
// license:BSD-3-Clause
|
||||
// copyright-holders:David Haywood
|
||||
|
||||
/* These contain a similar game selection to the devices in unk6502_st2xxx.cpp but on updated hardware
|
||||
|
||||
The hardware appears to be an abuse of the GPL16250 SoC. The palette and sprite banks are used, but as work-ram
|
||||
rather than for their intended purpose, and the rest of the GPL16250 video hardware is either entirely bypassed
|
||||
or doesn't exist. All video is software rendered and output directly to the LCD Controller.
|
||||
|
||||
If this is confirmed via a decap then this should be merged with the GPL16250 implementation.
|
||||
|
||||
The coding of these is similar to the unk6502_st2xxx.cpp too, with all game specific function calls being loaded
|
||||
on the fly from the SPI to a tiny portion of work RAM, with graphics likewise being loaded and decompressed for
|
||||
every draw call.
|
||||
|
||||
pcp8718 / pcp8728 / bkid218
|
||||
to access test mode hold up + left on startup.
|
||||
(bkid218 cursor in test menu can't be moved in emulation, works on unit, why?)
|
||||
*/
|
||||
|
||||
|
||||
#include "emu.h"
|
||||
|
||||
#include "cpu/unsp/unsp.h"
|
||||
#include "machine/bl_handhelds_menucontrol.h"
|
||||
|
||||
#include "emupal.h"
|
||||
#include "screen.h"
|
||||
#include "speaker.h"
|
||||
|
||||
#define LOG_GPL162XX_LCDTYPE (1U << 1)
|
||||
#define LOG_GPL162XX_LCDTYPE_SELECT_SIM (1U << 2)
|
||||
#define LOG_GPL162XX_LCDTYPE_IO_PORT (1U << 3)
|
||||
|
||||
#define LOG_ALL (LOG_GPL162XX_LCDTYPE | LOG_GPL162XX_LCDTYPE_SELECT_SIM | LOG_GPL162XX_LCDTYPE_IO_PORT)
|
||||
#define VERBOSE (0)
|
||||
|
||||
#include "logmacro.h"
|
||||
|
||||
|
||||
class gpl162xx_lcdtype_state : public driver_device
|
||||
class generalplus_gpl_unknown_state : public driver_device
|
||||
{
|
||||
public:
|
||||
gpl162xx_lcdtype_state(const machine_config &mconfig, device_type type, const char *tag) :
|
||||
generalplus_gpl_unknown_state(const machine_config &mconfig, device_type type, const char *tag) :
|
||||
driver_device(mconfig, type, tag),
|
||||
m_maincpu(*this, "maincpu"),
|
||||
m_mainrom(*this, "maincpu"),
|
||||
m_mainram(*this, "mainram"),
|
||||
//m_mainrom(*this, "maincpu"),
|
||||
//m_mainram(*this, "mainram"),
|
||||
m_palette(*this, "palette"),
|
||||
m_screen(*this, "screen"),
|
||||
m_spirom(*this, "spi"),
|
||||
m_io_in0(*this, "IN0"),
|
||||
m_io_in1(*this, "IN1"),
|
||||
m_menucontrol(*this, "menucontrol")
|
||||
m_spirom(*this, "spi")
|
||||
{ }
|
||||
|
||||
void gpl162xx_lcdtype(machine_config &config);
|
||||
|
||||
void generalplus_gpl_unknown(machine_config &config);
|
||||
|
||||
private:
|
||||
virtual void machine_start() override;
|
||||
@ -63,835 +34,76 @@ private:
|
||||
uint32_t screen_update(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect);
|
||||
|
||||
required_device<unsp_20_device> m_maincpu;
|
||||
required_region_ptr<uint16_t> m_mainrom;
|
||||
required_shared_ptr<uint16_t> m_mainram;
|
||||
//required_region_ptr<uint16_t> m_mainrom;
|
||||
//required_shared_ptr<uint16_t> m_mainram;
|
||||
|
||||
required_device<palette_device> m_palette;
|
||||
required_device<screen_device> m_screen;
|
||||
|
||||
void map(address_map &map);
|
||||
|
||||
required_region_ptr<uint8_t> m_spirom;
|
||||
|
||||
uint16_t unk_7abf_r();
|
||||
uint16_t io_7860_r();
|
||||
uint16_t unk_780f_r();
|
||||
|
||||
void io_7860_w(uint16_t data);
|
||||
uint16_t m_7860;
|
||||
|
||||
void unk_7862_w(uint16_t data);
|
||||
void unk_7863_w(uint16_t data);
|
||||
|
||||
void unk_7868_w(uint16_t data);
|
||||
uint16_t unk_7868_r();
|
||||
uint16_t m_7868;
|
||||
|
||||
void bankswitch_707e_w(uint16_t data);
|
||||
uint16_t bankswitch_707e_r();
|
||||
uint16_t m_707e_bank;
|
||||
|
||||
void bankswitch_703a_w(uint16_t data);
|
||||
uint16_t bankswitch_703a_r();
|
||||
uint16_t m_703a_bank;
|
||||
|
||||
void bankedram_7300_w(offs_t offset, uint16_t data);
|
||||
uint16_t bankedram_7300_r(offs_t offset);
|
||||
uint16_t m_bankedram_7300[0x400];
|
||||
|
||||
void bankedram_7400_w(offs_t offset, uint16_t data);
|
||||
uint16_t bankedram_7400_r(offs_t offset);
|
||||
uint16_t m_bankedram_7400[0x800];
|
||||
|
||||
void system_dma_params_channel0_w(offs_t offset, uint16_t data);
|
||||
uint16_t system_dma_params_channel0_r(offs_t offset);
|
||||
|
||||
uint16_t m_dmaregs[8];
|
||||
|
||||
void lcd_w(uint16_t data);
|
||||
void lcd_command_w(uint16_t data);
|
||||
|
||||
uint16_t spi_misc_control_r();
|
||||
uint16_t spi_rx_fifo_r();
|
||||
void spi_tx_fifo_w(uint16_t data);
|
||||
|
||||
void spi_control_w(uint16_t data);
|
||||
|
||||
void spi_process_tx_data(uint8_t data);
|
||||
uint8_t spi_process_rx();
|
||||
uint8_t spi_rx();
|
||||
uint8_t spi_rx_fast();
|
||||
|
||||
uint8_t m_rx_fifo[5]; // actually 8 bytes? or 8 half-bytes?
|
||||
|
||||
uint32_t m_spiaddress;
|
||||
|
||||
uint16_t unk_78a1_r();
|
||||
uint16_t m_78a1;
|
||||
uint16_t unk_78d8_r();
|
||||
void unk_78d8_w(uint16_t data);
|
||||
|
||||
enum spistate : const int
|
||||
{
|
||||
SPI_STATE_READY = 0,
|
||||
SPI_STATE_WAITING_HIGH_ADDR = 1,
|
||||
SPI_STATE_WAITING_MID_ADDR = 2,
|
||||
SPI_STATE_WAITING_LOW_ADDR = 3,
|
||||
// probably not
|
||||
SPI_STATE_WAITING_DUMMY1_ADDR = 4,
|
||||
SPI_STATE_WAITING_DUMMY2_ADDR = 5,
|
||||
SPI_STATE_READING = 6,
|
||||
|
||||
SPI_STATE_WAITING_HIGH_ADDR_FAST = 8,
|
||||
SPI_STATE_WAITING_MID_ADDR_FAST = 9,
|
||||
SPI_STATE_WAITING_LOW_ADDR_FAST = 10,
|
||||
SPI_STATE_WAITING_LOW_ADDR_FAST_DUMMY = 11,
|
||||
SPI_STATE_READING_FAST = 12
|
||||
|
||||
};
|
||||
|
||||
spistate m_spistate;
|
||||
|
||||
enum lcdstate : const int
|
||||
{
|
||||
LCD_STATE_READY = 0,
|
||||
LCD_STATE_WAITING_FOR_COMMAND = 1,
|
||||
LCD_STATE_PROCESSING_COMMAND = 2
|
||||
};
|
||||
|
||||
lcdstate m_lcdstate;
|
||||
int m_lastlcdcommand;
|
||||
|
||||
uint8_t m_displaybuffer[0x40000];
|
||||
int m_lcdaddr;
|
||||
|
||||
uint16_t io_7870_r();
|
||||
required_ioport m_io_in0;
|
||||
required_ioport m_io_in1;
|
||||
required_device<bl_handhelds_menucontrol_device> m_menucontrol;
|
||||
|
||||
DECLARE_WRITE_LINE_MEMBER(screen_vblank);
|
||||
void map(address_map &map);
|
||||
};
|
||||
|
||||
uint32_t gpl162xx_lcdtype_state::screen_update(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect)
|
||||
uint32_t generalplus_gpl_unknown_state::screen_update(screen_device &screen, bitmap_rgb32 &bitmap, const rectangle &cliprect)
|
||||
{
|
||||
int count = 0;
|
||||
for (int y = 0; y < 256; y++)
|
||||
{
|
||||
uint32_t *const dst = &bitmap.pix(y);
|
||||
|
||||
for (int x = 0; x < 320; x++)
|
||||
{
|
||||
// 8-bit values get pumped through a 256 word table in internal ROM and converted to words
|
||||
uint16_t dat = m_displaybuffer[(count * 2) + 1] | (m_displaybuffer[(count * 2) + 0] << 8);
|
||||
|
||||
int b = ((dat >> 0) & 0x1f) << 3;
|
||||
int g = ((dat >> 5) & 0x3f) << 2;
|
||||
int r = ((dat >> 11) & 0x1f) << 3;
|
||||
|
||||
dst[x] = (r << 16) | (g << 8) | (b << 0);
|
||||
count++;
|
||||
}
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static INPUT_PORTS_START( gpl162xx_lcdtype )
|
||||
PORT_START("IN0")
|
||||
PORT_BIT( 0x0001, IP_ACTIVE_HIGH, IPT_UNUSED ) // causes lag if state is inverted, investigate
|
||||
PORT_DIPNAME( 0x0002, 0x0002, "P0:0002" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0002, "0002" )
|
||||
PORT_DIPNAME( 0x0004, 0x0000, "Show Vs in Test Mode" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0004, "0004" )
|
||||
PORT_BIT( 0x0008, IP_ACTIVE_HIGH, IPT_CUSTOM ) PORT_READ_LINE_DEVICE_MEMBER("menucontrol", bl_handhelds_menucontrol_device, status_r)
|
||||
PORT_BIT( 0x0010, IP_ACTIVE_HIGH, IPT_CUSTOM ) PORT_READ_LINE_DEVICE_MEMBER("menucontrol", bl_handhelds_menucontrol_device, data_r)
|
||||
PORT_DIPNAME( 0x0020, 0x0020, "P0:0020" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0020, "0020" )
|
||||
PORT_DIPNAME( 0x0040, 0x0040, "P0:0040" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0040, "0040" )
|
||||
PORT_DIPNAME( 0x0080, 0x0080, "P0:0080" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0080, "0080" )
|
||||
PORT_DIPNAME( 0x0100, 0x0100, "P0:0100" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0100, "0100" )
|
||||
PORT_DIPNAME( 0x0200, 0x0000, "Battery Level" )
|
||||
PORT_DIPSETTING( 0x0000, "Normal" )
|
||||
PORT_DIPSETTING( 0x0200, "Low" )
|
||||
PORT_DIPNAME( 0x0400, 0x0400, "P0:0400" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0400, "0400" )
|
||||
PORT_DIPNAME( 0x0800, 0x0800, "P0:0800" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0800, "0800" )
|
||||
PORT_DIPNAME( 0x1000, 0x1000, "P0:1000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x1000, "1000" )
|
||||
PORT_DIPNAME( 0x2000, 0x2000, "P0:2000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x2000, "2000" )
|
||||
PORT_BIT( 0x4000, IP_ACTIVE_LOW, IPT_START1 )
|
||||
PORT_BIT( 0x8000, IP_ACTIVE_LOW, IPT_JOYSTICK_UP )
|
||||
|
||||
PORT_START("IN1")
|
||||
PORT_DIPNAME( 0x0001, 0x0001, "P1:0001" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0001, "0001" )
|
||||
PORT_DIPNAME( 0x0002, 0x0002, "P1:0002" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0002, "0002" )
|
||||
PORT_DIPNAME( 0x0004, 0x0004, "P1:0004" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0004, "0004" )
|
||||
PORT_DIPNAME( 0x0008, 0x0008, "P1:0008" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0008, "0008" )
|
||||
PORT_DIPNAME( 0x0010, 0x0010, "P1:0010" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x0010, "0010" )
|
||||
PORT_BIT( 0x0020, IP_ACTIVE_LOW, IPT_BUTTON3 ) PORT_NAME("SOUND")
|
||||
PORT_BIT( 0x0040, IP_ACTIVE_LOW, IPT_JOYSTICK_DOWN )
|
||||
PORT_BIT( 0x0080, IP_ACTIVE_LOW, IPT_JOYSTICK_LEFT )
|
||||
PORT_BIT( 0x0100, IP_ACTIVE_LOW, IPT_JOYSTICK_RIGHT )
|
||||
PORT_BIT( 0x0200, IP_ACTIVE_LOW, IPT_BUTTON1 ) PORT_NAME("A")
|
||||
PORT_BIT( 0x0400, IP_ACTIVE_LOW, IPT_BUTTON2 ) PORT_NAME("B")
|
||||
PORT_BIT( 0x0800, IP_ACTIVE_LOW, IPT_BUTTON4 ) PORT_NAME("ON/OFF")
|
||||
PORT_DIPNAME( 0x1000, 0x1000, "P1:1000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x1000, "1000" )
|
||||
PORT_DIPNAME( 0x2000, 0x2000, "P1:2000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x2000, "2000" )
|
||||
PORT_DIPNAME( 0x4000, 0x4000, "P1:4000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x4000, "4000" )
|
||||
PORT_DIPNAME( 0x8000, 0x8000, "P1:8000" )
|
||||
PORT_DIPSETTING( 0x0000, "0000" )
|
||||
PORT_DIPSETTING( 0x8000, "8000" )
|
||||
static INPUT_PORTS_START( generalplus_gpl_unknown )
|
||||
INPUT_PORTS_END
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_7abf_r()
|
||||
|
||||
void generalplus_gpl_unknown_state::map(address_map &map)
|
||||
{
|
||||
return 0x0001;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::io_7860_r()
|
||||
{
|
||||
uint16_t ret = m_io_in0->read();
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE_IO_PORT, "%s: io_7860_r %02x\n", machine().describe_context(), ret);
|
||||
return ret;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::io_7860_w(uint16_t data)
|
||||
{
|
||||
m_menucontrol->data_w((data & 0x10)>>4);
|
||||
m_menucontrol->clock_w((data & 0x20)>>5);
|
||||
|
||||
m_7860 = data;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::unk_7862_w(uint16_t data)
|
||||
{
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::unk_7863_w(uint16_t data)
|
||||
{
|
||||
// probably port direction (or 7862 is?)
|
||||
if (data == 0x3cf7)
|
||||
{
|
||||
m_menucontrol->reset_w(1);
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_780f_r()
|
||||
{
|
||||
return 0x0002;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::spi_misc_control_r()
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_misc_control_r\n", machine().describe_context());
|
||||
return 0x0000;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::spi_rx_fifo_r()
|
||||
{
|
||||
if (m_spistate == SPI_STATE_READING_FAST)
|
||||
return spi_rx_fast();
|
||||
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_rx_fifo_r\n", machine().describe_context());
|
||||
return spi_rx();
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::spi_process_tx_data(uint8_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"transmitting %02x\n", data);
|
||||
|
||||
switch (m_spistate)
|
||||
{
|
||||
case SPI_STATE_READY:
|
||||
{
|
||||
if (data == 0x03)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to read mode (need address) %02x\n", data);
|
||||
m_spistate = SPI_STATE_WAITING_HIGH_ADDR;
|
||||
}
|
||||
else if (data == 0x0b)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to fast read mode (need address) %02x\n", data);
|
||||
m_spistate = SPI_STATE_WAITING_HIGH_ADDR_FAST;
|
||||
//machine().debug_break();
|
||||
}
|
||||
else
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"invalid state request %02x\n", data);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_HIGH_ADDR:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xff00ffff) | data << 16;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to high address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_MID_ADDR;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_MID_ADDR:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffff00ff) | data << 8;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to mid address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_LOW_ADDR;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_LOW_ADDR:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffffff00) | data;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to low address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_READING;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_READING:
|
||||
{
|
||||
// writes when in read mode clock in data?
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"write while in read mode (clock data?)\n", data, m_spiaddress);
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_DUMMY1_ADDR:
|
||||
{
|
||||
m_spistate = SPI_STATE_WAITING_DUMMY2_ADDR;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_DUMMY2_ADDR:
|
||||
{
|
||||
// m_spistate = SPI_STATE_READY;
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
case SPI_STATE_WAITING_HIGH_ADDR_FAST:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xff00ffff) | data << 16;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to high address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_MID_ADDR_FAST;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_MID_ADDR_FAST:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffff00ff) | data << 8;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to mid address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_LOW_ADDR_FAST;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_LOW_ADDR_FAST:
|
||||
{
|
||||
m_spiaddress = (m_spiaddress & 0xffffff00) | data;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"set to low address %02x address is now %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_WAITING_LOW_ADDR_FAST_DUMMY;
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_WAITING_LOW_ADDR_FAST_DUMMY:
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"dummy write %08x\n", data, m_spiaddress);
|
||||
m_spistate = SPI_STATE_READING_FAST;
|
||||
break;
|
||||
}
|
||||
|
||||
case SPI_STATE_READING_FAST:
|
||||
{
|
||||
// writes when in read mode clock in data?
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"write while in read mode (clock data?)\n", data, m_spiaddress);
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
uint8_t gpl162xx_lcdtype_state::spi_process_rx()
|
||||
{
|
||||
|
||||
switch (m_spistate)
|
||||
{
|
||||
case SPI_STATE_READING:
|
||||
case SPI_STATE_READING_FAST:
|
||||
{
|
||||
uint8_t dat = m_spirom[m_spiaddress & 0x7fffff];
|
||||
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"reading SPI %02x from SPI Address %08x (adjusted word offset %08x)\n", dat, m_spiaddress, (m_spiaddress/2)+0x20000);
|
||||
m_spiaddress++;
|
||||
return dat;
|
||||
}
|
||||
|
||||
default:
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"reading FIFO in unknown state\n");
|
||||
return 0x00;
|
||||
}
|
||||
}
|
||||
|
||||
return 0x00;
|
||||
}
|
||||
|
||||
|
||||
uint8_t gpl162xx_lcdtype_state::spi_rx()
|
||||
{
|
||||
uint8_t ret = m_rx_fifo[0];
|
||||
|
||||
m_rx_fifo[0] = m_rx_fifo[1];
|
||||
m_rx_fifo[1] = m_rx_fifo[2];
|
||||
m_rx_fifo[2] = m_rx_fifo[3];
|
||||
m_rx_fifo[3] = spi_process_rx();
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
uint8_t gpl162xx_lcdtype_state::spi_rx_fast()
|
||||
{
|
||||
uint8_t ret = m_rx_fifo[0];
|
||||
|
||||
m_rx_fifo[0] = m_rx_fifo[1];
|
||||
m_rx_fifo[1] = m_rx_fifo[2];
|
||||
m_rx_fifo[2] = m_rx_fifo[3];
|
||||
m_rx_fifo[3] = m_rx_fifo[4];
|
||||
m_rx_fifo[4] = spi_process_rx();
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::spi_tx_fifo_w(uint16_t data)
|
||||
{
|
||||
data &= 0x00ff;
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_tx_fifo_w %04x\n", machine().describe_context(), data);
|
||||
|
||||
spi_process_tx_data(data);
|
||||
}
|
||||
|
||||
// this is probably 'port b' but when SPI is enabled some points of this can become SPI control pins
|
||||
// it's accessed after each large data transfer, probably to reset the SPI into 'ready for command' state?
|
||||
void gpl162xx_lcdtype_state::unk_7868_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE_IO_PORT, "%s: unk_7868_w %04x (Port B + SPI reset?)\n", machine().describe_context(), data);
|
||||
|
||||
if ((m_7868 & 0x0100) != (data & 0x0100))
|
||||
{
|
||||
if (!(data & 0x0100))
|
||||
{
|
||||
for (int i = 0; i < 4; i++)
|
||||
m_rx_fifo[i] = 0xff;
|
||||
|
||||
m_spistate = SPI_STATE_READY;
|
||||
}
|
||||
}
|
||||
|
||||
m_7868 = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_7868_r()
|
||||
{
|
||||
return m_7868;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::bankswitch_707e_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: bankswitch_707e_w %04x\n", machine().describe_context(), data);
|
||||
m_707e_bank = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankswitch_707e_r()
|
||||
{
|
||||
return m_707e_bank;
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::bankswitch_703a_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: bankswitch_703a_w %04x\n", machine().describe_context(), data);
|
||||
m_703a_bank = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankswitch_703a_r()
|
||||
{
|
||||
return m_703a_bank;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::bankedram_7300_w(offs_t offset, uint16_t data)
|
||||
{
|
||||
offset |= (m_703a_bank & 0x000c) << 6;
|
||||
m_bankedram_7300[offset] = data;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankedram_7300_r(offs_t offset)
|
||||
{
|
||||
offset |= (m_703a_bank & 0x000c) << 6;
|
||||
return m_bankedram_7300[offset];
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::bankedram_7400_w(offs_t offset, uint16_t data)
|
||||
{
|
||||
if (m_707e_bank & 1)
|
||||
{
|
||||
m_bankedram_7400[offset + 0x400] = data;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_bankedram_7400[offset] = data;
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::bankedram_7400_r(offs_t offset)
|
||||
{
|
||||
if (m_707e_bank & 1)
|
||||
{
|
||||
return m_bankedram_7400[offset + 0x400];
|
||||
}
|
||||
else
|
||||
{
|
||||
return m_bankedram_7400[offset];
|
||||
}
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::system_dma_params_channel0_w(offs_t offset, uint16_t data)
|
||||
{
|
||||
m_dmaregs[offset] = data;
|
||||
|
||||
switch (offset)
|
||||
{
|
||||
case 0:
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Mode)\n", machine().describe_context(), offset, data);
|
||||
|
||||
uint16_t mode = m_dmaregs[0];
|
||||
uint32_t source = m_dmaregs[1] | (m_dmaregs[4] << 16);
|
||||
uint32_t dest = m_dmaregs[2] | (m_dmaregs[5] << 16) ;
|
||||
uint32_t length = m_dmaregs[3] | (m_dmaregs[6] << 16);
|
||||
|
||||
if ((mode != 0x0200) && (mode != 0x4009) && (mode != 0x6009))
|
||||
fatalerror("unknown dma mode write %04x\n", data);
|
||||
|
||||
if ((mode == 0x4009) || (mode == 0x6009))
|
||||
{
|
||||
address_space& mem = m_maincpu->space(AS_PROGRAM);
|
||||
|
||||
for (int i = 0; i < length; i++)
|
||||
{
|
||||
uint16_t dat = mem.read_word(source);
|
||||
|
||||
if (mode & 0x2000)
|
||||
{
|
||||
// Racing Car and Elevator Action need this logic
|
||||
mem.write_word(dest, dat & 0xff);
|
||||
dest++;
|
||||
mem.write_word(dest, dat >> 8);
|
||||
dest++;
|
||||
}
|
||||
else
|
||||
{
|
||||
mem.write_word(dest, dat);
|
||||
dest++;
|
||||
}
|
||||
|
||||
source++;
|
||||
}
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
case 1:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Source Low)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 2:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Dest Low)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 3:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Length Low)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 4:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Source High)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 5:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Dest High)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 6:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA Length High)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
|
||||
case 7:
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_w %01x %04x (DMA unknown)\n", machine().describe_context(), offset, data);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::system_dma_params_channel0_r(offs_t offset)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: system_dma_params_channel0_r %01x\n", machine().describe_context(), offset);
|
||||
return m_dmaregs[offset];
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::io_7870_r()
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE_IO_PORT, "%s: io_7870_r (IO port)\n", machine().describe_context() );
|
||||
return m_io_in1->read();
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::spi_control_w(uint16_t data)
|
||||
{
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: spi_control_w %04x\n", machine().describe_context(), data);
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_78a1_r()
|
||||
{
|
||||
// checked in interrupt, code skipped entirely if this isn't set
|
||||
return m_78a1;
|
||||
}
|
||||
|
||||
uint16_t gpl162xx_lcdtype_state::unk_78d8_r()
|
||||
{
|
||||
return 0xffff;
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::unk_78d8_w(uint16_t data)
|
||||
{
|
||||
// written in IRQ, possible ack
|
||||
if (data & 0x8000)
|
||||
m_78a1 &= ~0x8000;
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::map(address_map &map)
|
||||
{
|
||||
// there are calls to 01xxx and 02xxx regions
|
||||
// (RAM populated by internal ROM?, TODO: check to make sure code copied there isn't from SPI ROM like the GPL16250 bootstrap
|
||||
// does from NAND, it doesn't seem to have a header in the same format at least)
|
||||
map(0x000000, 0x006fff).ram().share("mainram");
|
||||
|
||||
// registers at 7xxx are similar to GPL16250, but not identical? (different video system? or just GPL16250 with the video part unused?)
|
||||
|
||||
map(0x00703a, 0x00703a).rw(FUNC(gpl162xx_lcdtype_state::bankswitch_703a_r), FUNC(gpl162xx_lcdtype_state::bankswitch_703a_w));
|
||||
map(0x00707e, 0x00707e).rw(FUNC(gpl162xx_lcdtype_state::bankswitch_707e_r), FUNC(gpl162xx_lcdtype_state::bankswitch_707e_w));
|
||||
|
||||
map(0x007100, 0x0071ff).ram(); // rowscroll on gpl16250
|
||||
map(0x007300, 0x0073ff).rw(FUNC(gpl162xx_lcdtype_state::bankedram_7300_r), FUNC(gpl162xx_lcdtype_state::bankedram_7300_w)); // palette on gpl16250
|
||||
map(0x007400, 0x0077ff).rw(FUNC(gpl162xx_lcdtype_state::bankedram_7400_r), FUNC(gpl162xx_lcdtype_state::bankedram_7400_w)); // spriteram on gpl16250
|
||||
|
||||
map(0x00780f, 0x00780f).r(FUNC(gpl162xx_lcdtype_state::unk_780f_r));
|
||||
|
||||
|
||||
map(0x007860, 0x007860).rw(FUNC(gpl162xx_lcdtype_state::io_7860_r),FUNC(gpl162xx_lcdtype_state::io_7860_w)); // Port A?
|
||||
map(0x007862, 0x007862).w(FUNC(gpl162xx_lcdtype_state::unk_7862_w));
|
||||
map(0x007863, 0x007863).w(FUNC(gpl162xx_lcdtype_state::unk_7863_w));
|
||||
|
||||
map(0x007868, 0x007868).rw(FUNC(gpl162xx_lcdtype_state::unk_7868_r), FUNC(gpl162xx_lcdtype_state::unk_7868_w)); // Port B?
|
||||
|
||||
map(0x007870, 0x007870).r(FUNC(gpl162xx_lcdtype_state::io_7870_r)); // Port C?
|
||||
|
||||
map(0x0078a1, 0x0078a1).r(FUNC(gpl162xx_lcdtype_state::unk_78a1_r));
|
||||
|
||||
map(0x0078d8, 0x0078d8).rw(FUNC(gpl162xx_lcdtype_state::unk_78d8_r), FUNC(gpl162xx_lcdtype_state::unk_78d8_w));
|
||||
|
||||
|
||||
map(0x007940, 0x007940).w(FUNC(gpl162xx_lcdtype_state::spi_control_w));
|
||||
// 7941 SPI Transmit Status
|
||||
map(0x007942, 0x007942).w(FUNC(gpl162xx_lcdtype_state::spi_tx_fifo_w));
|
||||
// 7943 SPI Receive Status
|
||||
map(0x007944, 0x007944).r(FUNC(gpl162xx_lcdtype_state::spi_rx_fifo_r));
|
||||
map(0x007945, 0x007945).r(FUNC(gpl162xx_lcdtype_state::spi_misc_control_r));
|
||||
|
||||
map(0x007a80, 0x007a87).rw(FUNC(gpl162xx_lcdtype_state::system_dma_params_channel0_r), FUNC(gpl162xx_lcdtype_state::system_dma_params_channel0_w));
|
||||
|
||||
map(0x007abf, 0x007abf).r(FUNC(gpl162xx_lcdtype_state::unk_7abf_r));
|
||||
map(0x000000, 0x00ffff).ram(); // unknown, RAM/ROM? there are jumps to this region, doesn't seem the usual unSP 8000-ffff ROM mapping
|
||||
|
||||
// there are calls to 0x0f000 (internal ROM?)
|
||||
map(0x00f000, 0x00ffff).rom().region("maincpu", 0x00000);
|
||||
|
||||
// external LCD controller
|
||||
map(0x200000, 0x200000).w(FUNC(gpl162xx_lcdtype_state::lcd_command_w));
|
||||
map(0x20fc00, 0x20fc00).w(FUNC(gpl162xx_lcdtype_state::lcd_w));
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::lcd_command_w(uint16_t data)
|
||||
{
|
||||
data &= 0xff;
|
||||
|
||||
switch (m_lcdstate)
|
||||
{
|
||||
case LCD_STATE_READY:
|
||||
case LCD_STATE_PROCESSING_COMMAND:
|
||||
{
|
||||
if (data == 0x0000)
|
||||
{
|
||||
m_lcdstate = LCD_STATE_WAITING_FOR_COMMAND;
|
||||
m_lastlcdcommand = 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case LCD_STATE_WAITING_FOR_COMMAND:
|
||||
{
|
||||
m_lastlcdcommand = data;
|
||||
m_lcdstate = LCD_STATE_PROCESSING_COMMAND;
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::lcd_w(uint16_t data)
|
||||
{
|
||||
data &= 0xff; // definitely looks like 8-bit port as 16-bit values are shifted and rewritten
|
||||
LOGMASKED(LOG_GPL162XX_LCDTYPE,"%s: lcd_w %02x\n", machine().describe_context(), data);
|
||||
|
||||
if ((m_lcdstate == LCD_STATE_PROCESSING_COMMAND) && (m_lastlcdcommand == 0x22))
|
||||
{
|
||||
m_displaybuffer[m_lcdaddr] = data;
|
||||
m_lcdaddr++;
|
||||
|
||||
if (m_lcdaddr >= ((320 * 240) * 2))
|
||||
m_lcdaddr = 0;
|
||||
|
||||
//m_lcdaddr &= 0x3ffff;
|
||||
}
|
||||
map(0x200000, 0x3fffff).ram(); // almost certainly RAM where the SPI ROM gets copied
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::machine_start()
|
||||
void generalplus_gpl_unknown_state::machine_start()
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
void gpl162xx_lcdtype_state::machine_reset()
|
||||
void generalplus_gpl_unknown_state::machine_reset()
|
||||
{
|
||||
m_spistate = SPI_STATE_READY;
|
||||
m_spiaddress = 0;
|
||||
|
||||
for (int i = 0; i < 320*240*2; i++)
|
||||
m_displaybuffer[i] = 0x00;
|
||||
|
||||
m_lcdaddr = 0;
|
||||
|
||||
m_lcdstate = LCD_STATE_READY;
|
||||
m_lastlcdcommand = 0;
|
||||
|
||||
m_78a1 = 0;
|
||||
|
||||
// first menu index is stored here
|
||||
// m_spirom[0x16000] = gamenum & 0xff;
|
||||
// m_spirom[0x16001] = (gamenum>>8) & 0xff;
|
||||
}
|
||||
|
||||
WRITE_LINE_MEMBER(gpl162xx_lcdtype_state::screen_vblank)
|
||||
{
|
||||
if (state)
|
||||
address_space& mem = m_maincpu->space(AS_PROGRAM);
|
||||
for (int j = 0; j < 0x200000/2; j++)
|
||||
{
|
||||
// probably a timer
|
||||
m_maincpu->set_input_line(UNSP_IRQ4_LINE, ASSERT_LINE);
|
||||
m_78a1 |= 0x8000;
|
||||
}
|
||||
else
|
||||
{
|
||||
m_maincpu->set_input_line(UNSP_IRQ4_LINE, CLEAR_LINE);
|
||||
uint16_t data = (m_spirom[(j*2)+1] << 8) | (m_spirom[(j*2)+0]);
|
||||
mem.write_word(0x200000+j, data);
|
||||
}
|
||||
|
||||
m_maincpu->set_state_int(UNSP_PC, 0x1000); // this is not the boot code, everything in the SPI ROM seems like relatively small functions (that never touch internal registers) only the very last one in SPI ROM has no callers from in the SPI ROM code
|
||||
m_maincpu->set_state_int(UNSP_SR, 0x0020);
|
||||
}
|
||||
|
||||
void gpl162xx_lcdtype_state::gpl162xx_lcdtype(machine_config &config)
|
||||
void generalplus_gpl_unknown_state::generalplus_gpl_unknown(machine_config &config)
|
||||
{
|
||||
|
||||
UNSP_20(config, m_maincpu, 96000000); // unknown CPU, unsp20 based, 96Mhz is listed as the maximum for most unSP2.0 chips, and appears correct here
|
||||
m_maincpu->set_addrmap(AS_PROGRAM, &gpl162xx_lcdtype_state::map);
|
||||
UNSP_20(config, m_maincpu, 96000000); // unknown CPU, no unSP2.0 opcodes?
|
||||
m_maincpu->set_addrmap(AS_PROGRAM, &generalplus_gpl_unknown_state::map);
|
||||
|
||||
SCREEN(config, m_screen, SCREEN_TYPE_RASTER);
|
||||
m_screen->set_refresh_hz(60);
|
||||
m_screen->set_vblank_time(ATTOSECONDS_IN_USEC(10));
|
||||
m_screen->set_size(64*8, 32*8);
|
||||
m_screen->set_visarea(0*8, 320-1, 0*8, 240-1);
|
||||
m_screen->set_screen_update(FUNC(gpl162xx_lcdtype_state::screen_update));
|
||||
m_screen->screen_vblank().set(FUNC(gpl162xx_lcdtype_state::screen_vblank));
|
||||
m_screen->set_screen_update(FUNC(generalplus_gpl_unknown_state::screen_update));
|
||||
//m_screen->screen_vblank().set(FUNC(generalplus_gpl_unknown_state::screen_vblank));
|
||||
|
||||
PALETTE(config, m_palette).set_format(palette_device::xBGR_555, 0x8000);
|
||||
|
||||
BL_HANDHELDS_MENUCONTROL(config, m_menucontrol, 0);
|
||||
m_menucontrol->set_is_unsp_type_hack();
|
||||
|
||||
}
|
||||
|
||||
// pcp8718 and pcp8728 both contain user data (player name?) and will need to be factory defaulted once they work
|
||||
// the ROM code is slightly different between them
|
||||
|
||||
ROM_START( pcp8718 )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "internal.rom", 0x000000, 0x2000, CRC(ea119561) SHA1(a2680577e20fe1155efc40a5781cf1ec80ccec3a) )
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF )
|
||||
//ROM_LOAD16_WORD_SWAP( "8718_en25f32.bin", 0x000000, 0x400000, CRC(cc138db4) SHA1(379af3d94ae840f52c06416d6cf32e25923af5ae) ) // bad dump, some blocks are corrupt
|
||||
ROM_LOAD( "eyecare_25q32av1g_ef4016.bin", 0x000000, 0x400000, CRC(58415e10) SHA1(b1adcc03f2ad8d741544204671677740e904ce1a) )
|
||||
ROM_END
|
||||
|
||||
ROM_START( pcp8728 )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "internal.rom", 0x000000, 0x2000, CRC(ea119561) SHA1(a2680577e20fe1155efc40a5781cf1ec80ccec3a) )
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "pcp 8728 788 in 1.bin", 0x000000, 0x400000, CRC(60115f21) SHA1(e15c39f11e442a76fae3823b6d510178f6166926) )
|
||||
ROM_END
|
||||
|
||||
ROM_START( bkid218 )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "internal.rom", 0x000000, 0x2000, CRC(ea119561) SHA1(a2680577e20fe1155efc40a5781cf1ec80ccec3a) )
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF )
|
||||
ROM_LOAD( "218n1_25q64csig_c84017.bin", 0x000000, 0x800000, CRC(94f35dbd) SHA1(a1bd6defd2465ae14753cd83be5c31f99e9158ec) )
|
||||
ROM_END
|
||||
|
||||
ROM_START( unkunsp )
|
||||
ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
ROM_LOAD16_WORD_SWAP( "internal.rom", 0x000000, 0x2000, NO_DUMP ) // exact size unknown
|
||||
//ROM_REGION( 0x2000, "maincpu", ROMREGION_ERASEFF )
|
||||
//ROM_LOAD16_WORD_SWAP( "internal.rom", 0x000000, 0x2000, NO_DUMP ) // exact size unknown
|
||||
|
||||
ROM_REGION( 0x800000, "spi", ROMREGION_ERASEFF ) // this simply maps at 200000, and contains jumps to lower areas that suggest even the ROM/internal RAM mapping is very different
|
||||
ROM_LOAD( "fm25q16a.bin", 0x000000, 0x200000, CRC(aeb472ac) SHA1(500c24b725f6d3308ef8cbdf4259f5be556c7c92) )
|
||||
ROM_END
|
||||
|
||||
|
||||
CONS( 200?, pcp8718, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "PCP", "PCP 8718 - HD 360 Degrees Rocker Palm Eyecare Console - 788 in 1", MACHINE_NOT_WORKING | MACHINE_NO_SOUND )
|
||||
CONS( 200?, pcp8728, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "PCP", "PCP 8728 - 788 in 1", MACHINE_NOT_WORKING | MACHINE_NO_SOUND ) // what name was this sold under?
|
||||
CONS( 200?, bkid218, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "BornKid", "Handheld Game Console BC-19 - 218 in 1", MACHINE_NOT_WORKING | MACHINE_NO_SOUND )
|
||||
|
||||
// different hardware, first 0x2000 bytes in ROM are blank, maps fully in RAM at 200000
|
||||
CONS( 200?, unkunsp, 0, 0, gpl162xx_lcdtype, gpl162xx_lcdtype, gpl162xx_lcdtype_state, empty_init, "<unknown>", "unknown unSP-based handheld", MACHINE_IS_SKELETON )
|
||||
CONS( 2017, unkunsp, 0, 0, generalplus_gpl_unknown, generalplus_gpl_unknown, generalplus_gpl_unknown_state, empty_init, "Super Impulse", "Pac-Man (Micro Arcade)", MACHINE_IS_SKELETON )
|
||||
|
@ -14807,9 +14807,11 @@ suprpokrb // (c) 1986 Grayhound Electronics
|
||||
rp200
|
||||
|
||||
@source:generalplus_gpl_unknown.cpp
|
||||
unkunsp
|
||||
|
||||
@source:generalplus_gpl162xx_lcdtype.cpp
|
||||
pcp8718
|
||||
pcp8728
|
||||
unkunsp
|
||||
bkid218
|
||||
|
||||
@source:generalplus_gpl16250_mobigo.cpp
|
||||
|
@ -333,6 +333,7 @@ gb.cpp
|
||||
gba.cpp
|
||||
gem_rp.cpp
|
||||
generalplus_gpl_unknown.cpp
|
||||
generalplus_gpl162xx_lcdtype.cpp
|
||||
generalplus_gpl16250_mobigo.cpp
|
||||
generalplus_gpl16250_nand.cpp
|
||||
generalplus_gpl16250_rom.cpp
|
||||
|
Loading…
Reference in New Issue
Block a user