mirror of
https://github.com/holub/mame
synced 2025-04-24 01:11:11 +03:00
(MESS) New system marked as GAME_NOT_WORKING
------------------------------------ Sega Visual Memory Unit [Sandro Ronco]
This commit is contained in:
parent
7bf498af91
commit
023d855b01
1
.gitattributes
vendored
1
.gitattributes
vendored
@ -6054,6 +6054,7 @@ src/mess/drivers/supercon.c svneol=native#text/plain
|
||||
src/mess/drivers/supracan.c svneol=native#text/plain
|
||||
src/mess/drivers/svi318.c svneol=native#text/plain
|
||||
src/mess/drivers/svision.c svneol=native#text/plain
|
||||
src/mess/drivers/svmu.c svneol=native#text/plain
|
||||
src/mess/drivers/swtpc.c svneol=native#text/plain
|
||||
src/mess/drivers/sym1.c svneol=native#text/plain
|
||||
src/mess/drivers/sys2900.c svneol=native#text/plain
|
||||
|
361
src/mess/drivers/svmu.c
Normal file
361
src/mess/drivers/svmu.c
Normal file
@ -0,0 +1,361 @@
|
||||
/***************************************************************************
|
||||
|
||||
Sega Visual Memory Unit
|
||||
|
||||
driver by Sandro Ronco
|
||||
|
||||
TODO:
|
||||
- add more bios versions
|
||||
- layout for LCD symbols
|
||||
- serial
|
||||
|
||||
****************************************************************************/
|
||||
|
||||
#include "emu.h"
|
||||
#include "cpu/lc8670/lc8670.h"
|
||||
#include "imagedev/snapquik.h"
|
||||
#include "machine/intelfsh.h"
|
||||
#include "sound/speaker.h"
|
||||
#include "rendlay.h"
|
||||
|
||||
#define PIXEL_SIZE 7
|
||||
#define PIXEL_DISTANCE 1
|
||||
|
||||
class svmu_state : public driver_device
|
||||
{
|
||||
public:
|
||||
svmu_state(const machine_config &mconfig, device_type type, const char *tag)
|
||||
: driver_device(mconfig, type, tag),
|
||||
m_maincpu(*this, "maincpu"),
|
||||
m_flash(*this, "flash"),
|
||||
m_speaker(*this, SPEAKER_TAG)
|
||||
{ }
|
||||
|
||||
required_device<lc8670_cpu_device> m_maincpu;
|
||||
required_device<intelfsh8_device> m_flash;
|
||||
required_device<device_t> m_speaker;
|
||||
|
||||
virtual void palette_init();
|
||||
virtual void machine_reset();
|
||||
|
||||
DECLARE_WRITE8_MEMBER(page_w);
|
||||
DECLARE_READ8_MEMBER(flash_r);
|
||||
DECLARE_WRITE8_MEMBER(flash_w);
|
||||
DECLARE_READ8_MEMBER(prog_r);
|
||||
DECLARE_WRITE8_MEMBER(prog_w);
|
||||
DECLARE_READ8_MEMBER(p1_r);
|
||||
DECLARE_WRITE8_MEMBER(p1_w);
|
||||
DECLARE_READ8_MEMBER(p7_r);
|
||||
|
||||
private:
|
||||
UINT8 * m_bios;
|
||||
UINT8 m_page;
|
||||
};
|
||||
|
||||
|
||||
WRITE8_MEMBER(svmu_state::page_w)
|
||||
{
|
||||
m_page = data & 0x03;
|
||||
}
|
||||
|
||||
READ8_MEMBER(svmu_state::flash_r)
|
||||
{
|
||||
return m_flash->read(offset);
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(svmu_state::flash_w)
|
||||
{
|
||||
m_flash->write(offset, data);
|
||||
}
|
||||
|
||||
READ8_MEMBER(svmu_state::prog_r)
|
||||
{
|
||||
if (m_page == 1)
|
||||
return m_flash->read(offset);
|
||||
else if (m_page == 2)
|
||||
return m_flash->read(0x10000 + offset);
|
||||
else
|
||||
return m_bios[offset];
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(svmu_state::prog_w)
|
||||
{
|
||||
if (m_page == 1)
|
||||
m_flash->write(offset, data);
|
||||
else if (m_page == 2)
|
||||
m_flash->write(0x10000 + offset, data);
|
||||
}
|
||||
|
||||
/*
|
||||
Port 1
|
||||
|
||||
x--- ---- PWM output
|
||||
-x-- ---- BUZ
|
||||
--x- ---- SCK1
|
||||
---x ---- SB1
|
||||
---- x--- SO1
|
||||
---- -x-- SCK0
|
||||
---- --x- SB0
|
||||
---- ---x SO0
|
||||
|
||||
*/
|
||||
|
||||
READ8_MEMBER(svmu_state::p1_r)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
WRITE8_MEMBER(svmu_state::p1_w)
|
||||
{
|
||||
speaker_level_w(m_speaker, BIT(data, 7));
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
Port 7
|
||||
|
||||
---- x--- ID1
|
||||
---- -x-- ID0
|
||||
---- --x- battery low voltage
|
||||
---- ---x 5V detection
|
||||
*/
|
||||
|
||||
READ8_MEMBER(svmu_state::p7_r)
|
||||
{
|
||||
return (ioport("BATTERY")->read()<<1);
|
||||
}
|
||||
|
||||
|
||||
static ADDRESS_MAP_START(svmu_mem, AS_PROGRAM, 8, svmu_state)
|
||||
AM_RANGE( 0x0000, 0xffff ) AM_READWRITE(prog_r, prog_w)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
static ADDRESS_MAP_START(svmu_io_mem, AS_IO, 8, svmu_state)
|
||||
AM_RANGE( LC8670_PORT1, LC8670_PORT1 ) AM_READWRITE(p1_r, p1_w)
|
||||
AM_RANGE( LC8670_PORT3, LC8670_PORT3 ) AM_READ_PORT("P3")
|
||||
AM_RANGE( LC8670_PORT7, LC8670_PORT7 ) AM_READ(p7_r)
|
||||
ADDRESS_MAP_END
|
||||
|
||||
/* Input ports */
|
||||
static INPUT_PORTS_START( svmu )
|
||||
PORT_START( "P3" )
|
||||
PORT_BIT( 0x01, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("UP") PORT_CODE( KEYCODE_UP )
|
||||
PORT_BIT( 0x02, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("DOWN") PORT_CODE( KEYCODE_DOWN )
|
||||
PORT_BIT( 0x04, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("LEFT") PORT_CODE( KEYCODE_LEFT )
|
||||
PORT_BIT( 0x08, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("RIGHT") PORT_CODE( KEYCODE_RIGHT )
|
||||
PORT_BIT( 0x10, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("A") PORT_CODE( KEYCODE_A )
|
||||
PORT_BIT( 0x20, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("B") PORT_CODE( KEYCODE_B )
|
||||
PORT_BIT( 0x40, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("MODE") PORT_CODE( KEYCODE_M )
|
||||
PORT_BIT( 0x80, IP_ACTIVE_LOW, IPT_KEYPAD ) PORT_NAME("SLEEP") PORT_CODE( KEYCODE_S )
|
||||
PORT_START("BATTERY")
|
||||
PORT_CONFNAME( 0x01, 0x01, "Battery" )
|
||||
PORT_CONFSETTING( 0x01, "Good" )
|
||||
PORT_CONFSETTING( 0x00, "Poor" )
|
||||
INPUT_PORTS_END
|
||||
|
||||
void svmu_state::machine_reset()
|
||||
{
|
||||
m_bios = (UINT8*)(*memregion("bios"));
|
||||
m_page = 0;
|
||||
}
|
||||
|
||||
void svmu_state::palette_init()
|
||||
{
|
||||
palette_set_color(machine(), 0, MAKE_RGB(138, 146, 148));
|
||||
palette_set_color(machine(), 1, MAKE_RGB(92, 83, 88));
|
||||
}
|
||||
|
||||
static LC8670_LCD_UPDATE( svmu_lcd_update )
|
||||
{
|
||||
if (lcd_enabled)
|
||||
{
|
||||
for (int y=0; y<32; y++)
|
||||
for (int x=0; x<6; x++)
|
||||
{
|
||||
int gfx = vram[y*6 + x];
|
||||
|
||||
for (int b=0; b<8; b++)
|
||||
bitmap.plot_box((x*8 + b) * (PIXEL_SIZE + PIXEL_DISTANCE), y * (PIXEL_SIZE + PIXEL_DISTANCE), PIXEL_SIZE, PIXEL_SIZE, BIT(gfx,7-b));
|
||||
}
|
||||
|
||||
popmessage("%c %c %c %c\n", BIT(vram[0xc1],6) ? 'F' : ' ', // File icon
|
||||
BIT(vram[0xc2],4) ? 'G' : ' ', // Game icon
|
||||
BIT(vram[0xc3],2) ? 'C' : ' ', // Clock icon
|
||||
BIT(vram[0xc4],0) ? 'A' : ' '); // Flash icon
|
||||
}
|
||||
else
|
||||
{
|
||||
bitmap.fill(0, cliprect);
|
||||
}
|
||||
|
||||
output_set_value("file_icon" , lcd_enabled ? BIT(vram[0xc1],6) : 0);
|
||||
output_set_value("game_icon" , lcd_enabled ? BIT(vram[0xc2],4) : 0);
|
||||
output_set_value("clock_icon", lcd_enabled ? BIT(vram[0xc3],2) : 0);
|
||||
output_set_value("flash_icon", lcd_enabled ? BIT(vram[0xc4],0) : 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
inline void vmufat_write_byte(UINT8* flash, UINT8 block, offs_t offset, UINT8 data)
|
||||
{
|
||||
flash[(block * 512) + offset] = data;
|
||||
}
|
||||
|
||||
inline void vmufat_write_word(UINT8* flash, UINT8 block, offs_t offset, UINT16 data)
|
||||
{
|
||||
// 16-bit data are stored in little endian
|
||||
flash[(block * 512) + offset + 0] = data & 0xff;
|
||||
flash[(block * 512) + offset + 1] = (data>>8) & 0xff;
|
||||
}
|
||||
|
||||
static QUICKLOAD_LOAD( svmu )
|
||||
{
|
||||
running_machine &machine = image.device().machine();
|
||||
svmu_state *state = machine.driver_data<svmu_state>();
|
||||
UINT32 size = image.length();
|
||||
UINT8 *flash = (UINT8*)state->m_flash->space().get_read_ptr(0);
|
||||
|
||||
image.fread(flash, size);
|
||||
|
||||
// verify if image is already a valid VMUFAT file system
|
||||
bool valid_vmufat = true;
|
||||
if (size == 0x20000)
|
||||
{
|
||||
for (int i=0; i<0x10; i++)
|
||||
if (flash[255 * 512 + i] != 0x55)
|
||||
{
|
||||
valid_vmufat = false;
|
||||
break;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
valid_vmufat = false;
|
||||
}
|
||||
|
||||
if (!valid_vmufat)
|
||||
{
|
||||
// more info about the VMUFAT here: http://mc.pp.se/dc/vms/flashmem.html
|
||||
|
||||
//-------------------------------- Formatting --------------------------------
|
||||
memset(flash + 241*512, 0, 15*512); // clears the last 15 blocks that contain file system information
|
||||
|
||||
for (int i=0; i<0x10; i++)
|
||||
vmufat_write_byte(flash, 255, i, 0x55); // first 16 bytes should be 0x55 to indicate a properly formatted card
|
||||
|
||||
vmufat_write_byte(flash, 255, 0x10, 0x00); // custom VMS colour (1 = use custom colours, 0 = standard colour)
|
||||
vmufat_write_byte(flash, 255, 0x11, 0x00); // VMS colour blue component
|
||||
vmufat_write_byte(flash, 255, 0x12, 0x00); // VMS colour green component
|
||||
vmufat_write_byte(flash, 255, 0x13, 0x00); // VMS colour red component
|
||||
vmufat_write_byte(flash, 255, 0x14, 0x00); // VMS colour alpha component
|
||||
vmufat_write_byte(flash, 255, 0x30, 0x19); // Century (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x31, 0x99); // Year (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x32, 0x01); // Month (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x33, 0x01); // Day (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x34, 0x00); // Hour (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x35, 0x00); // Minute (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x36, 0x00); // Second (BCD)
|
||||
vmufat_write_byte(flash, 255, 0x37, 0x00); // Day of week (0 = Monday, 6 = Sunday)
|
||||
vmufat_write_word(flash, 255, 0x44, 0x00ff); // location of Root
|
||||
vmufat_write_word(flash, 255, 0x46, 0x00fe); // location of FAT (254)
|
||||
vmufat_write_word(flash, 255, 0x48, 0x0001); // size of FAT in blocks (1)
|
||||
vmufat_write_word(flash, 255, 0x4a, 0x00fd); // location of Directory (253)
|
||||
vmufat_write_word(flash, 255, 0x4c, 0x000d); // size of Directory in blocks (13)
|
||||
vmufat_write_word(flash, 255, 0x4e, 0x0000); // icon shape for this VMS (0-123)
|
||||
vmufat_write_word(flash, 255, 0x50, 0x00c8); // number of user blocks (200)
|
||||
|
||||
for (int i=0; i<256; i++)
|
||||
vmufat_write_word(flash, 254, i<<1, 0xfffc); // marks all blocks as unallocated
|
||||
|
||||
for (int i=253; i>241; --i)
|
||||
vmufat_write_word(flash, 254, i<<1, i - 1); // marsk all Directory blocks as allocate
|
||||
|
||||
vmufat_write_word(flash, 254, 0x1e2, 0xfffa); // marks last Directory block
|
||||
vmufat_write_word(flash, 254, 0x1fc, 0xfffa); // marks FAT block as allocated
|
||||
vmufat_write_word(flash, 254, 0x1fe, 0xfffa); // marks Root block as allocated
|
||||
|
||||
//-------------------------------- Create the vms file --------------------------------
|
||||
int vms_blocks = (size / 512) + (size & 0x1ff ? 1 : 0); // number of blocks required for store the vms file
|
||||
|
||||
for (int i=0; i<vms_blocks - 1; i++)
|
||||
vmufat_write_word(flash, 254, i<<1, i + 1); // marks blocks where the file is allocated
|
||||
|
||||
vmufat_write_word(flash, 254, (vms_blocks-1)<<1, 0xfffa); // last block for this file
|
||||
|
||||
vmufat_write_byte(flash, 253, 0x00, 0xcc); // file type (0x00 = no file, 0x33 = data, 0xcc = game)
|
||||
vmufat_write_byte(flash, 253, 0x01, 0x00); // copy protect (0x00 = no, 0xff = yes)
|
||||
vmufat_write_word(flash, 253, 0x02, 0x0000); // location of first file block
|
||||
|
||||
const char *vms_filename = image.basename_noext();
|
||||
for (int i=0; i<12; i++)
|
||||
{
|
||||
if (i < strlen(vms_filename))
|
||||
vmufat_write_byte(flash, 253, i + 4, vms_filename[i]); // 12 bytes filename
|
||||
else
|
||||
vmufat_write_byte(flash, 253, i + 4, 0x20); // space padded
|
||||
}
|
||||
|
||||
vmufat_write_byte(flash, 253, 0x10, 0x19); // Century (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x11, 0x99); // Year (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x12, 0x01); // Month (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x13, 0x01); // Day (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x14, 0x00); // Hour (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x15, 0x00); // Minute (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x16, 0x00); // Second (BCD)
|
||||
vmufat_write_byte(flash, 253, 0x17, 0x00); // Day of week (0 = Monday, 6 = Sunday)
|
||||
vmufat_write_word(flash, 253, 0x18, vms_blocks); // file size (in blocks)
|
||||
vmufat_write_word(flash, 253, 0x1a, 0x0001); // offset of header (in blocks) from file start
|
||||
}
|
||||
|
||||
return IMAGE_INIT_PASS;
|
||||
}
|
||||
|
||||
|
||||
static MACHINE_CONFIG_START( svmu, svmu_state )
|
||||
/* basic machine hardware */
|
||||
MCFG_CPU_ADD("maincpu", LC8670, XTAL_32_768kHz)
|
||||
MCFG_CPU_PROGRAM_MAP(svmu_mem)
|
||||
MCFG_CPU_IO_MAP(svmu_io_mem)
|
||||
|
||||
/* specific LC8670 configurations */
|
||||
MCFG_LC8670_SET_CLOCK_SOURCES(XTAL_32_768kHz, 600000, XTAL_6MHz) // tolerance range of the RC oscillator is 600kHz to 1200kHz
|
||||
MCFG_LC8670_BANKSWITCH_CB(WRITE8(svmu_state, page_w))
|
||||
MCFG_LC8670_LCD_UPDATE_CB(svmu_lcd_update)
|
||||
|
||||
/* video hardware */
|
||||
MCFG_SCREEN_ADD("screen", LCD)
|
||||
MCFG_SCREEN_REFRESH_RATE(60)
|
||||
MCFG_SCREEN_VBLANK_TIME(ATTOSECONDS_IN_USEC(2500)) // not accurate
|
||||
MCFG_SCREEN_SIZE(48 * (PIXEL_SIZE + PIXEL_DISTANCE), 32 * (PIXEL_SIZE + PIXEL_DISTANCE))
|
||||
MCFG_SCREEN_VISIBLE_AREA(0, 48*(PIXEL_SIZE + PIXEL_DISTANCE) - 1, 0, 32*(PIXEL_SIZE + PIXEL_DISTANCE) - 1)
|
||||
MCFG_SCREEN_UPDATE_DEVICE("maincpu", lc8670_cpu_device, screen_update)
|
||||
MCFG_DEFAULT_LAYOUT(layout_lcd)
|
||||
MCFG_PALETTE_LENGTH(2)
|
||||
|
||||
/* sound hardware */
|
||||
MCFG_SPEAKER_STANDARD_MONO("mono")
|
||||
MCFG_SOUND_ADD(SPEAKER_TAG, SPEAKER_SOUND, 0)
|
||||
MCFG_SOUND_ROUTE(ALL_OUTPUTS, "mono", 0.50)
|
||||
|
||||
/* devices */
|
||||
MCFG_ATMEL_29C010_ADD("flash")
|
||||
MCFG_QUICKLOAD_ADD("quickload", svmu, "vms,bin", 0)
|
||||
MACHINE_CONFIG_END
|
||||
|
||||
|
||||
/* ROM definition */
|
||||
ROM_START( svmu )
|
||||
ROM_REGION( 0x10000, "bios", 0 )
|
||||
// these ROMs come from the Sega Katana SDK and are scrambled, a simple way to restore it is to remove the first 4 bytes and xor the whole file for the xor key.
|
||||
ROM_SYSTEM_BIOS(0, "jp1004", "Japan v1.004")
|
||||
ROMX_LOAD( "fbios.bin", 0x0000, 0x10000, CRC(8e0f867a) SHA1(dc2fa2963138a1049a43f7f36439ad0a416ee8b4), ROM_BIOS(1)) // from Sega Katana SDK (original file: fbios.sbf, CRC: c7c77b3c, xor key: 0x37)
|
||||
ROM_SYSTEM_BIOS(1, "jp1004q", "Japan v1.004 (quick start)") // automatically boot the first game found in the flash
|
||||
ROMX_LOAD( "qbios.bin", 0x0000, 0x10000, CRC(395e25f2) SHA1(37dea034322b5b80b35b2de784298d32c71ba7a3), ROM_BIOS(2)) // from Sega Katana SDK (original file: qbios.sbf, CRC: eed5524c, xor key: 0x43)
|
||||
ROM_END
|
||||
|
||||
|
||||
/* Driver */
|
||||
|
||||
/* YEAR NAME PARENT COMPAT MACHINE INPUT INIT COMPANY FULLNAME FLAGS */
|
||||
COMP( 1998, svmu, 0, 0, svmu , svmu , driver_device, 0, "Sega", "Visual Memory Unit", GAME_NOT_WORKING | GAME_NO_SOUND)
|
@ -144,6 +144,8 @@ dceu // 1999 Sega Dreamcast (Europe)
|
||||
dcdev // 1998 Sega HKT-0120 Sega Dreamcast Development Box
|
||||
dcprt // 1998 Sega Katana Set 5 Prototype
|
||||
|
||||
svmu // 1998 Sega Visual Memory Unit
|
||||
|
||||
// Sony
|
||||
psj // 1994 Sony PlayStation (Japan)
|
||||
psu // 1995 Sony PlayStation (USA)
|
||||
|
@ -120,6 +120,7 @@ CPUS += HCD62121
|
||||
CPUS += PPS4
|
||||
CPUS += UPD7725
|
||||
CPUS += HD61700
|
||||
CPUS += LC8670
|
||||
|
||||
#-------------------------------------------------
|
||||
# specify available sound cores; some of these are
|
||||
@ -1596,6 +1597,7 @@ $(MESSOBJ)/sega.a: \
|
||||
$(MESS_MACHINE)/dccons.o \
|
||||
$(MESS_MACHINE)/sms.o \
|
||||
$(MESS_DRIVERS)/sms.o \
|
||||
$(MESS_DRIVERS)/svmu.o \
|
||||
|
||||
$(MESSOBJ)/sgi.a: \
|
||||
$(MESS_MACHINE)/sgi.o \
|
||||
|
Loading…
Reference in New Issue
Block a user