mirror of
https://github.com/ekeeke/Genesis-Plus-GX.git
synced 2025-01-28 10:55:28 +01:00
[Core/VDP]
--------------- * improved VBLANK flag accuracy, as observed on real hardware. * improved DMA operations accuracy, writes are now performed on a scanline basis: fixes Gaiares (flickering title screen). * improved DMA Fill timing accuracy. * fixed DMA with bad code values: fixes Williams Arcade Classics (corrupted gfx after soft reset). * fixed horizontal resolution changes during HBLANK: fixes Bugs Bunny in Double Trouble (2nd stage). * fixed Vertical Counter in interlace mode 1, as observed on real hardware. * fixed horizontal border width, as observed on real hardware. * various code improvments & optimizations. [Core/Extra] --------------- * improved savestate format: added DMA, SVP, cartridge mapping & internal registers state informations * improved unlicensed ROM mappers emulation * added Chinese Fighters III mapper support * added Top Fighter mapper support * fixed Barver Battle Saga mapper support * fixed cartridge hardware soft-reset (Game Genie, SVP, ...) * fixed Game Genie registers byte reads
This commit is contained in:
parent
edf479a257
commit
a4cfc2a77a
@ -151,9 +151,11 @@ void areplay_shutdown(void)
|
|||||||
action_replay.enabled = 0;
|
action_replay.enabled = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void areplay_reset(int hard_reset)
|
void areplay_reset(int hard)
|
||||||
{
|
{
|
||||||
if (action_replay.enabled)
|
if (action_replay.enabled)
|
||||||
|
{
|
||||||
|
if (hard || (action_replay.status == AR_SWITCH_TRAINER))
|
||||||
{
|
{
|
||||||
/* reset internal registers */
|
/* reset internal registers */
|
||||||
memset(action_replay.regs, 0, sizeof(action_replay.regs));
|
memset(action_replay.regs, 0, sizeof(action_replay.regs));
|
||||||
@ -164,12 +166,13 @@ void areplay_reset(int hard_reset)
|
|||||||
/* by default, internal ROM is mapped at $000000-$00FFFF */
|
/* by default, internal ROM is mapped at $000000-$00FFFF */
|
||||||
m68k_memory_map[0].base = action_replay.rom;
|
m68k_memory_map[0].base = action_replay.rom;
|
||||||
|
|
||||||
/* internal RAM is cleared on power ON */
|
/* reset internal RAM on power-on */
|
||||||
if (hard_reset)
|
if (hard)
|
||||||
{
|
{
|
||||||
memset(action_replay.ram,0xff,0x10000);
|
memset(action_replay.ram,0xff,0x10000);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int areplay_get_status(void)
|
int areplay_get_status(void)
|
||||||
|
@ -28,7 +28,7 @@
|
|||||||
|
|
||||||
extern void areplay_init(void);
|
extern void areplay_init(void);
|
||||||
extern void areplay_shutdown(void);
|
extern void areplay_shutdown(void);
|
||||||
extern void areplay_reset(int hard_reset);
|
extern void areplay_reset(int hard);
|
||||||
extern void areplay_set_status(int status);
|
extern void areplay_set_status(int status);
|
||||||
extern int areplay_get_status(void);
|
extern int areplay_get_status(void);
|
||||||
|
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
* Genesis Plus
|
* Genesis Plus
|
||||||
* Cartridge Hardware support
|
* Cartridge Hardware support
|
||||||
*
|
*
|
||||||
* Copyright (C) 2007, 2008, 2009 Eke-Eke (GCN/Wii port)
|
* Copyright (C) 2007, 2008, 2009, 2010 Eke-Eke (GCN/Wii port)
|
||||||
*
|
*
|
||||||
* Most cartridge protections documented by Haze
|
* Most cartridge protections documented by Haze
|
||||||
* (http://haze.mameworld.info/)
|
* (http://haze.mameworld.info/)
|
||||||
@ -27,7 +27,7 @@
|
|||||||
|
|
||||||
#include "shared.h"
|
#include "shared.h"
|
||||||
|
|
||||||
#define CART_CNT 28
|
#define CART_CNT 30
|
||||||
|
|
||||||
extern int emulate_address_error;
|
extern int emulate_address_error;
|
||||||
|
|
||||||
@ -45,15 +45,19 @@ typedef struct
|
|||||||
} T_CART_ENTRY;
|
} T_CART_ENTRY;
|
||||||
|
|
||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
static void sega_mapper_w(uint32 address, uint32 data);
|
static void mapper_sega_w(uint32 data);
|
||||||
static void special_mapper_w(uint32 address, uint32 data);
|
static void mapper_ssf2_w(uint32 address, uint32 data);
|
||||||
static void realtec_mapper_w(uint32 address, uint32 data);
|
static void mapper_realtec_w(uint32 address, uint32 data);
|
||||||
static void seganet_mapper_w(uint32 address, uint32 data);
|
static void mapper_seganet_w(uint32 address, uint32 data);
|
||||||
static uint32 radica_mapper_r(uint32 address);
|
static void mapper_32k_w(uint32 data);
|
||||||
|
static void mapper_64k_w(uint32 data);
|
||||||
|
static void mapper_64k_multi_w(uint32 address);
|
||||||
|
static uint32 mapper_radica_r(uint32 address);
|
||||||
static void default_time_w(uint32 address, uint32 data);
|
static void default_time_w(uint32 address, uint32 data);
|
||||||
static void default_regs_w(uint32 address, uint32 data);
|
static void default_regs_w(uint32 address, uint32 data);
|
||||||
static uint32 default_regs_r(uint32 address);
|
static uint32 default_regs_r(uint32 address);
|
||||||
static void special_regs_w(uint32 address, uint32 data);
|
static void custom_regs_w(uint32 address, uint32 data);
|
||||||
|
static void custom_alt_regs_w(uint32 address, uint32 data);
|
||||||
|
|
||||||
/* Games that need extra hardware emulation:
|
/* Games that need extra hardware emulation:
|
||||||
- copy protection device
|
- copy protection device
|
||||||
@ -62,31 +66,37 @@ static void special_regs_w(uint32 address, uint32 data);
|
|||||||
static const T_CART_ENTRY rom_database[CART_CNT] =
|
static const T_CART_ENTRY rom_database[CART_CNT] =
|
||||||
{
|
{
|
||||||
/* Funny World & Balloon Boy */
|
/* Funny World & Balloon Boy */
|
||||||
{0x0000,0x06ab,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},1,1,NULL,NULL,NULL,realtec_mapper_w}},
|
{0x0000,0x06ab,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},1,0,NULL,NULL,NULL,mapper_realtec_w}},
|
||||||
/* Whac-a-Critter */
|
/* Whac-a-Critter */
|
||||||
{0xffff,0xf863,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},1,1,NULL,NULL,NULL,realtec_mapper_w}},
|
{0xffff,0xf863,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},1,0,NULL,NULL,NULL,mapper_realtec_w}},
|
||||||
/* Earth Defense */
|
/* Earth Defense */
|
||||||
{0xffff,0x44fb,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},1,1,NULL,NULL,NULL,realtec_mapper_w}},
|
{0xffff,0x44fb,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},1,0,NULL,NULL,NULL,mapper_realtec_w}},
|
||||||
/* RADICA (Volume 1) (not byteswapped) */
|
/* RADICA (Volume 1) (not byteswapped) */
|
||||||
{0x0000,0x2326,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,1,radica_mapper_r,NULL,NULL,NULL}},
|
{0x0000,0x2326,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,0,mapper_radica_r,NULL,NULL,NULL}},
|
||||||
/* RADICA (Volume 2) */
|
/* RADICA (Volume 2) */
|
||||||
{0x4f10,0x0836,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,1,radica_mapper_r,NULL,NULL,NULL}},
|
{0x4f10,0x0836,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,0,mapper_radica_r,NULL,NULL,NULL}},
|
||||||
/* RADICA (Volume 1) */
|
/* RADICA (Volume 1) */
|
||||||
{0xf424,0x9f82,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,1,radica_mapper_r,NULL,NULL,NULL}},
|
{0xf424,0x9f82,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,0,mapper_radica_r,NULL,NULL,NULL}},
|
||||||
|
/* Chinese Fighters III */
|
||||||
|
{0x9490,0x8180,0x40,0x6f,{{0x00,0x00,0x00,0x00},{0xf0000c,0xf0000c,0xf0000c,0xf0000c},{0x400000,0x400004,0x400008,0x40000c},0,1,NULL,NULL,default_regs_r,custom_alt_regs_w}},
|
||||||
|
/* Top Fighter */
|
||||||
|
{0x4eb9,0x5d8b,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,custom_regs_w}},
|
||||||
/* Mulan */
|
/* Mulan */
|
||||||
{0x0404,0x1b40,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,special_regs_w}},
|
{0x0404,0x1b40,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,custom_regs_w}},
|
||||||
/* Pocket Monsters II */
|
/* Pocket Monsters II */
|
||||||
{0x47f9,0x17e5,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,special_regs_w}},
|
{0x47f9,0x17e5,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,custom_regs_w}},
|
||||||
/* Lion King 3 */
|
/* Lion King 3 */
|
||||||
{0x0000,0x507c,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,special_regs_w}},
|
{0x0000,0x507c,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,custom_regs_w}},
|
||||||
/* Super King Kong 99 */
|
/* Super King Kong 99 */
|
||||||
{0x0000,0x7d6e,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,special_regs_w}},
|
{0x0000,0x7d6e,0x60,0x7f,{{0x00,0x00,0x00,0x00},{0xf00007,0xf00007,0xf00007,0xffffff},{0x600001,0x600003,0x600005,0x000000},0,1,NULL,NULL,default_regs_r,custom_regs_w}},
|
||||||
/* Pokemon Stadium */
|
/* Pokemon Stadium */
|
||||||
{0x0000,0x843c,0x70,0x7f,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,1,NULL,NULL,NULL,special_regs_w}},
|
{0x0000,0x843c,0x70,0x7f,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,1,NULL,NULL,NULL,custom_regs_w}},
|
||||||
/* Lion King 2 */
|
/* Lion King 2 */
|
||||||
{0xffff,0x1d9b,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xfffffd,0xfffffd,0xffffff,0xffffff},{0x400000,0x400004,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,default_regs_w}},
|
{0xffff,0x1d9b,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xfffffd,0xfffffd,0xffffff,0xffffff},{0x400000,0x400004,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,default_regs_w}},
|
||||||
/* Squirell King */
|
/* Squirell King */
|
||||||
{0x0000,0x8ec8,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xfffffd,0xfffffd,0xffffff,0xffffff},{0x400000,0x400004,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,default_regs_w}},
|
{0x0000,0x8ec8,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xfffffd,0xfffffd,0xffffff,0xffffff},{0x400000,0x400004,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,default_regs_w}},
|
||||||
|
/* Lian Huan Pao - Barver Battle Saga */
|
||||||
|
{0x30b9,0x1c2a,0x40,0x40,{{0x00,0x00,0x00,0x00},{0xfffff4,0xffffff,0xffffff,0xffffff},{0x400004,0x000000,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,default_regs_w}},
|
||||||
/* Supper Bubble Bobble */
|
/* Supper Bubble Bobble */
|
||||||
{0x0000,0x16cd,0x40,0x40,{{0x55,0x0f,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x400000,0x400002,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,NULL}},
|
{0x0000,0x16cd,0x40,0x40,{{0x55,0x0f,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x400000,0x400002,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,NULL}},
|
||||||
/* Mahjong Lover */
|
/* Mahjong Lover */
|
||||||
@ -101,13 +111,9 @@ static const T_CART_ENTRY rom_database[CART_CNT] =
|
|||||||
{0x0000,0x0c5b,0x40,0x40,{{0x00,0x98,0xc9,0xF0},{0xffffff,0xffffff,0xffffff,0xffffff},{0x400000,0x400002,0x400004,0x400006},0,0,NULL,NULL,default_regs_r,NULL}},
|
{0x0000,0x0c5b,0x40,0x40,{{0x00,0x98,0xc9,0xF0},{0xffffff,0xffffff,0xffffff,0xffffff},{0x400000,0x400002,0x400004,0x400006},0,0,NULL,NULL,default_regs_r,NULL}},
|
||||||
/* King of Fighter 98 */
|
/* King of Fighter 98 */
|
||||||
{0x0000,0xd0a0,0x48,0x4f,{{0xaa,0xa0,0xf0,0xa0},{0xfc0000,0xffffff,0xffffff,0xffffff},{0x480000,0x4c82c0,0x4cdda0,0x4f8820},0,0,NULL,NULL,default_regs_r,NULL}},
|
{0x0000,0xd0a0,0x48,0x4f,{{0xaa,0xa0,0xf0,0xa0},{0xfc0000,0xffffff,0xffffff,0xffffff},{0x480000,0x4c82c0,0x4cdda0,0x4f8820},0,0,NULL,NULL,default_regs_r,NULL}},
|
||||||
/* Lian Huan Pao - Barver Battle Saga */
|
|
||||||
{0x30b9,0x1c2a,0x40,0x40,{{0x00,0x00,0x00,0x00},{0x000000,0x000000,0x000000,0x000000},{0x000000,0x000000,0x000000,0x000000},0,0,NULL,NULL,default_regs_r,NULL}},
|
|
||||||
/* Rockman X3 */
|
|
||||||
{0x0000,0x9d0e,0x40,0x40,{{0x0c,0x88,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0x400004,0x000000,0x000000},0,0,default_regs_r,NULL,default_regs_r,NULL}},
|
|
||||||
/* Super Mario 2 1998 */
|
/* Super Mario 2 1998 */
|
||||||
{0xffff,0x0474,0x00,0x00,{{0x0a,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0x000000,0x000000,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
{0xffff,0x0474,0x00,0x00,{{0x0a,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0x000000,0x000000,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
||||||
/* Super Mario 2 1998 */
|
/* Super Mario World */
|
||||||
{0x2020,0xb4eb,0x00,0x00,{{0x1c,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0x000000,0x000000,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
{0x2020,0xb4eb,0x00,0x00,{{0x1c,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0x000000,0x000000,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
||||||
/* A Bug's Life */
|
/* A Bug's Life */
|
||||||
{0x7f7f,0x2aad,0x00,0x00,{{0x28,0x1f,0x01,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0xa13002,0xa1303e,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
{0x7f7f,0x2aad,0x00,0x00,{{0x28,0x1f,0x01,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0xa13002,0xa1303e,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
||||||
@ -115,8 +121,10 @@ static const T_CART_ENTRY rom_database[CART_CNT] =
|
|||||||
{0x0000,0x021e,0x00,0x00,{{0x00,0x01,0x1f,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0xa13002,0xa1303e,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
{0x0000,0x021e,0x00,0x00,{{0x00,0x01,0x1f,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0xa13002,0xa1303e,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
||||||
/* Pocket Monster */
|
/* Pocket Monster */
|
||||||
{0xd6fc,0x1eb1,0x00,0x00,{{0x00,0x01,0x1f,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0xa13002,0xa1303e,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
{0xd6fc,0x1eb1,0x00,0x00,{{0x00,0x01,0x1f,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0xa13002,0xa1303e,0x000000},0,0,default_regs_r,NULL,NULL,NULL}},
|
||||||
|
/* Rockman X3 */
|
||||||
|
{0x0000,0x9d0e,0x40,0x40,{{0x0c,0x88,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0xa13000,0x400004,0x000000,0x000000},0,0,default_regs_r,NULL,default_regs_r,NULL}},
|
||||||
/* Game no Kanzume Otokuyou */
|
/* Game no Kanzume Otokuyou */
|
||||||
{0x0000,0xf9d1,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,0,NULL,seganet_mapper_w,NULL,NULL}}
|
{0x0000,0xf9d1,0x00,0x00,{{0x00,0x00,0x00,0x00},{0xffffff,0xffffff,0xffffff,0xffffff},{0x000000,0x000000,0x000000,0x000000},0,0,NULL,mapper_seganet_w,NULL,NULL}}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -311,12 +319,16 @@ void cart_hw_init()
|
|||||||
switch (config.lock_on)
|
switch (config.lock_on)
|
||||||
{
|
{
|
||||||
case TYPE_GG:
|
case TYPE_GG:
|
||||||
|
{
|
||||||
ggenie_init();
|
ggenie_init();
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case TYPE_AR:
|
case TYPE_AR:
|
||||||
|
{
|
||||||
areplay_init();
|
areplay_init();
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case TYPE_SK:
|
case TYPE_SK:
|
||||||
{
|
{
|
||||||
@ -359,15 +371,19 @@ void cart_hw_init()
|
|||||||
|
|
||||||
/*$000000-$1fffff is mapped to S&K ROM */
|
/*$000000-$1fffff is mapped to S&K ROM */
|
||||||
for (i=0x00; i<0x20; i++)
|
for (i=0x00; i<0x20; i++)
|
||||||
|
{
|
||||||
m68k_memory_map[i].base = (cart.rom + 0x600000) + (i<<16);
|
m68k_memory_map[i].base = (cart.rom + 0x600000) + (i<<16);
|
||||||
|
}
|
||||||
|
|
||||||
cart.lock_on = 1;
|
cart.lock_on = 1;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
{
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**********************************************
|
/**********************************************
|
||||||
Cartridge Extra Hardware
|
Cartridge Extra Hardware
|
||||||
@ -408,6 +424,22 @@ void cart_hw_init()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Realtec mapper */
|
||||||
|
if (cart.hw.realtec)
|
||||||
|
{
|
||||||
|
/* 8k BOOT ROM */
|
||||||
|
for (i=0; i<8; i++)
|
||||||
|
{
|
||||||
|
memcpy(cart.rom + 0x900000 + i*0x2000, cart.rom + 0x7e000, 0x2000);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* BOOT ROM is mapped to $000000-$3FFFFF */
|
||||||
|
for (i=0x00; i<0x40; i++)
|
||||||
|
{
|
||||||
|
m68k_memory_map[i].base = cart.rom + 0x900000;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#if M68K_EMULATE_ADDRESS_ERROR
|
#if M68K_EMULATE_ADDRESS_ERROR
|
||||||
/* default behavior */
|
/* default behavior */
|
||||||
emulate_address_error = config.addr_error;
|
emulate_address_error = config.addr_error;
|
||||||
@ -434,50 +466,42 @@ void cart_hw_init()
|
|||||||
{
|
{
|
||||||
/* assume SSF2 mapper */
|
/* assume SSF2 mapper */
|
||||||
cart.hw.bankshift = 1;
|
cart.hw.bankshift = 1;
|
||||||
cart.hw.time_w = sega_mapper_w;
|
cart.hw.time_w = mapper_ssf2_w;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* default write handler for !TIME signal */
|
/* default write handler for !TIME range ($A130xx)*/
|
||||||
if (!cart.hw.time_w)
|
if (!cart.hw.time_w)
|
||||||
|
{
|
||||||
cart.hw.time_w = default_time_w;
|
cart.hw.time_w = default_time_w;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* hardware that need to be reseted on power on */
|
/* hardware that need to be reseted on power on */
|
||||||
void cart_hw_reset()
|
void cart_hw_reset(int hard)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* reset bankshifting */
|
/* reset cartridge mapping */
|
||||||
if (cart.hw.bankshift)
|
if (cart.hw.bankshift)
|
||||||
{
|
{
|
||||||
for (i=0x00; i<0x40; i++)
|
for (i=0x00; i<0x40; i++)
|
||||||
|
{
|
||||||
m68k_memory_map[i].base = cart.rom + ((i<<16) & cart.mask);
|
m68k_memory_map[i].base = cart.rom + ((i<<16) & cart.mask);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Realtec mapper */
|
|
||||||
if (cart.hw.realtec & 1)
|
|
||||||
{
|
|
||||||
/* enable BOOTROM */
|
|
||||||
for (i=0; i<8; i++)
|
|
||||||
memcpy(cart.rom + 0x900000 + i*0x2000, cart.rom + 0x7e000, 0x2000);
|
|
||||||
for (i=0x00; i<0x40; i++)
|
|
||||||
m68k_memory_map[i].base = cart.rom + 0x900000;
|
|
||||||
cart.hw.realtec |= 2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* SVP chip */
|
/* SVP chip */
|
||||||
if (svp)
|
if (svp) svp_reset();
|
||||||
svp_reset();
|
|
||||||
|
|
||||||
/* Lock-ON */
|
/* Lock-ON */
|
||||||
switch (config.lock_on)
|
switch (config.lock_on)
|
||||||
{
|
{
|
||||||
case TYPE_GG:
|
case TYPE_GG:
|
||||||
ggenie_reset();
|
ggenie_reset(hard);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_AR:
|
case TYPE_AR:
|
||||||
areplay_reset(1);
|
areplay_reset(hard);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TYPE_SK:
|
case TYPE_SK:
|
||||||
@ -497,30 +521,107 @@ void cart_hw_reset()
|
|||||||
cart.base = m68k_memory_map[0].base;
|
cart.base = m68k_memory_map[0].base;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int cart_hw_context_save(uint8 *state)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int bufferptr = 0;
|
||||||
|
uint8 *base;
|
||||||
|
|
||||||
|
/* cartridge mapping */
|
||||||
|
for (i=0; i<0x40; i++)
|
||||||
|
{
|
||||||
|
/* get base address */
|
||||||
|
base = m68k_memory_map[i].base;
|
||||||
|
|
||||||
|
if (base == sram.sram)
|
||||||
|
{
|
||||||
|
/* SRAM */
|
||||||
|
state[bufferptr++] = 0xff;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* ROM */
|
||||||
|
state[bufferptr++] = ((base - cart.rom) >> 16) & 0xff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* hardware registers */
|
||||||
|
save_param(cart.hw.regs, sizeof(cart.hw.regs));
|
||||||
|
|
||||||
|
/* SVP */
|
||||||
|
if (svp)
|
||||||
|
{
|
||||||
|
save_param(svp->iram_rom, 0x800);
|
||||||
|
save_param(svp->dram,sizeof(svp->dram));
|
||||||
|
save_param(&svp->ssp1601,sizeof(ssp1601_t));
|
||||||
|
}
|
||||||
|
|
||||||
|
return bufferptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
int cart_hw_context_load(uint8 *state, char *version)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int bufferptr = 0;
|
||||||
|
uint8 offset;
|
||||||
|
|
||||||
|
/* extended state */
|
||||||
|
if (version[15] > 0x30)
|
||||||
|
{
|
||||||
|
/* cartridge mapping */
|
||||||
|
for (i=0; i<0x40; i++)
|
||||||
|
{
|
||||||
|
/* get offset */
|
||||||
|
offset = state[bufferptr++];
|
||||||
|
|
||||||
|
if (offset == 0xff)
|
||||||
|
{
|
||||||
|
/* SRAM */
|
||||||
|
m68k_memory_map[i].base = sram.sram;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* ROM */
|
||||||
|
m68k_memory_map[i].base = cart.rom + (offset << 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* hardware registers */
|
||||||
|
load_param(cart.hw.regs, sizeof(cart.hw.regs));
|
||||||
|
|
||||||
|
/* SVP */
|
||||||
|
if (svp)
|
||||||
|
{
|
||||||
|
load_param(svp->iram_rom, 0x800);
|
||||||
|
load_param(svp->dram,sizeof(svp->dram));
|
||||||
|
load_param(&svp->ssp1601,sizeof(ssp1601_t));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return bufferptr;
|
||||||
|
}
|
||||||
|
|
||||||
/************************************************************
|
/************************************************************
|
||||||
MAPPER handlers
|
MAPPER handlers
|
||||||
*************************************************************/
|
*************************************************************/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
"official" ROM/RAM switch
|
ROM/SRAM Bankswitch (Phantasy Star IV, Story of Thor/Beyond Oasis, Sonic 3 & Knuckles)
|
||||||
*/
|
*/
|
||||||
static void sega_mapper_w(uint32 address, uint32 data)
|
static void mapper_sega_w(uint32 data)
|
||||||
{
|
{
|
||||||
uint32 i,slot = (address >> 1) & 7;
|
int i;
|
||||||
uint8 *src;
|
|
||||||
|
|
||||||
switch (slot)
|
|
||||||
{
|
|
||||||
case 0:
|
|
||||||
/* ROM/SRAM switch (Phantasy Star IV, Story of Thor/Beyond Oasis, Sonic 3 & Knuckles) */
|
|
||||||
if (data & 1)
|
if (data & 1)
|
||||||
{
|
{
|
||||||
/* $200000-$3fffff is mapped to SRAM (only if SRAM exists) */
|
/* Only if SRAM is detected */
|
||||||
if (sram.on)
|
if (sram.on)
|
||||||
{
|
{
|
||||||
|
/* $200000-$3fffff is mapped to SRAM */
|
||||||
for (i=0x20; i<0x40; i++)
|
for (i=0x20; i<0x40; i++)
|
||||||
|
{
|
||||||
m68k_memory_map[i].base = sram.sram;
|
m68k_memory_map[i].base = sram.sram;
|
||||||
|
}
|
||||||
|
|
||||||
if (data & 2)
|
if (data & 2)
|
||||||
{
|
{
|
||||||
@ -544,71 +645,56 @@ static void sega_mapper_w(uint32 address, uint32 data)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cart.lock_on)
|
/* S&K lock-on chip */
|
||||||
|
if (cart.lock_on && (config.lock_on == TYPE_SK))
|
||||||
{
|
{
|
||||||
/* enable UPMEM chip at $300000-$3fffff */
|
/* $300000-$3fffff is mapped to S2K upmem chip */
|
||||||
for (i=0x30; i<0x40; i++)
|
for (i=0x30; i<0x40; i++)
|
||||||
|
{
|
||||||
m68k_memory_map[i].base = (cart.rom + 0x800000) + ((i & 3)<<16);
|
m68k_memory_map[i].base = (cart.rom + 0x800000) + ((i & 3)<<16);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* ROM enabled */
|
/* $200000-$3fffff is mapped to ROM */
|
||||||
for (i=0x20; i<0x40; i++)
|
for (i=0x20; i<0x40; i++)
|
||||||
m68k_memory_map[i].base = cart.rom + ((i<<16) & cart.mask);
|
|
||||||
|
|
||||||
if (cart.lock_on)
|
|
||||||
{
|
{
|
||||||
/* enable cartridge ROM at $300000-$3fffff */
|
|
||||||
for (i=0x30; i<0x40; i++)
|
|
||||||
m68k_memory_map[i].base = cart.rom + ((i<<16) & cart.mask);
|
m68k_memory_map[i].base = cart.rom + ((i<<16) & cart.mask);
|
||||||
|
m68k_memory_map[i].write8 = m68k_unused_8_w;
|
||||||
|
m68k_memory_map[i].write16 = m68k_unused_16_w;
|
||||||
|
zbank_memory_map[i].write = zbank_unused_w;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
}
|
||||||
|
|
||||||
default:
|
/*
|
||||||
/* ROM Bankswitch (Super Street Fighter 2)
|
Super Street Fighter 2 ROM Bankswitch
|
||||||
documented by Bart Trzynadlowski (http://www.trzy.org/files/ssf2.txt)
|
documented by Bart Trzynadlowski (http://www.trzy.org/files/ssf2.txt)
|
||||||
*/
|
*/
|
||||||
slot = slot << 3; /* 8 x 512k banks */
|
static void mapper_ssf2_w(uint32 address, uint32 data)
|
||||||
src = cart.rom + (data << 19);
|
{
|
||||||
|
/* 8 x 512k banks */
|
||||||
|
uint32 dst = (address << 2) & 0x38;
|
||||||
|
|
||||||
|
/* bank 0 remains unchanged */
|
||||||
|
if (dst)
|
||||||
|
{
|
||||||
|
uint32 i;
|
||||||
|
uint8 *src = cart.rom + (data << 19);
|
||||||
|
|
||||||
for (i=0; i<8; i++)
|
for (i=0; i<8; i++)
|
||||||
m68k_memory_map[slot++].base = src + (i<<16);
|
{
|
||||||
break;
|
m68k_memory_map[dst++].base = src + (i<<16);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
custom ROM Bankswitch used by pirate "Multi-in-1" carts
|
|
||||||
*/
|
|
||||||
static void multi_mapper_w(uint32 address, uint32 data)
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
cart.hw.bankshift = 1;
|
|
||||||
|
|
||||||
/* 64 x 64k banks */
|
|
||||||
for (i=0; i<64; i++)
|
|
||||||
m68k_memory_map[i].base = &cart.rom[((address++) & 0x3f) << 16];
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
Special ROM Bankswitch used for copy protection
|
|
||||||
Used by unlicensed cartridges (Lion King III, Super King Kong 99)
|
|
||||||
*/
|
|
||||||
static void special_mapper_w(uint32 address, uint32 data)
|
|
||||||
{
|
|
||||||
/* 1 x 32k bank */
|
|
||||||
memcpy(cart.rom + 0x900000, cart.rom + ((data & 0x7f) << 15), 0x8000);
|
|
||||||
memcpy(cart.rom + 0x908000, cart.rom + 0x8000, 0x8000);
|
|
||||||
m68k_memory_map[0].base = cart.rom + 0x900000;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Realtec ROM Bankswitch (Earth Defend, Balloon Boy & Funny World, Whac-A-Critter)
|
Realtec ROM Bankswitch (Earth Defend, Balloon Boy & Funny World, Whac-A-Critter)
|
||||||
(Note: register usage is inverted in TascoDlx documentation)
|
(Note: register usage is inverted in TascoDlx documentation)
|
||||||
*/
|
*/
|
||||||
static void realtec_mapper_w(uint32 address, uint32 data)
|
static void mapper_realtec_w(uint32 address, uint32 data)
|
||||||
{
|
{
|
||||||
switch (address)
|
switch (address)
|
||||||
{
|
{
|
||||||
@ -648,7 +734,7 @@ static void realtec_mapper_w(uint32 address, uint32 data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Game no Kanzume Otokuyou ROM Mapper */
|
/* Game no Kanzume Otokuyou ROM Mapper */
|
||||||
static void seganet_mapper_w(uint32 address, uint32 data)
|
static void mapper_seganet_w(uint32 address, uint32 data)
|
||||||
{
|
{
|
||||||
if ((address & 0xff) == 0xf1)
|
if ((address & 0xff) == 0xf1)
|
||||||
{
|
{
|
||||||
@ -677,18 +763,91 @@ static void seganet_mapper_w(uint32 address, uint32 data)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
RADICA ROM Bankswitch (use !TIME)
|
Custom ROM Bankswitch used in Top Fighter, Mulan, Pocket Monsters II, Lion King 3, Super King Kong 99, Pokemon Stadium
|
||||||
*/
|
*/
|
||||||
static uint32 radica_mapper_r(uint32 address)
|
static void mapper_32k_w(uint32 data)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* 64 x 32k banks */
|
||||||
|
if (data)
|
||||||
|
{
|
||||||
|
/* unverified (Top Fighter writes $2A instead $2E) */
|
||||||
|
if (data >> 2) data |= 4;
|
||||||
|
|
||||||
|
/* bank is mapped at $000000-$0FFFFF */
|
||||||
|
for (i=0; i<16; i++)
|
||||||
|
{
|
||||||
|
memcpy(cart.rom + 0x900000 + (i<<16), cart.rom + ((data & 0x3f) << 15), 0x8000);
|
||||||
|
memcpy(cart.rom + 0x908000 + (i<<16), cart.rom + ((data & 0x3f) << 15), 0x8000);
|
||||||
|
m68k_memory_map[i].base = cart.rom + 0x900000 + (i<<16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* reset default $000000-$0FFFFF mapping */
|
||||||
|
for (i=0; i<16; i++)
|
||||||
|
{
|
||||||
|
m68k_memory_map[i].base = &cart.rom[i << 16];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Custom ROM Bankswitch used in Chinese Fighter III
|
||||||
|
*/
|
||||||
|
static void mapper_64k_w(uint32 data)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* 16 x 64k banks */
|
||||||
|
if (data)
|
||||||
|
{
|
||||||
|
/* bank is mapped at $000000-$0FFFFF */
|
||||||
|
for (i=0; i<16; i++)
|
||||||
|
{
|
||||||
|
m68k_memory_map[i].base = &cart.rom[(data & 0xf) << 16];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* reset default $000000-$0FFFFF mapping */
|
||||||
|
for (i=0; i<16; i++)
|
||||||
|
{
|
||||||
|
m68k_memory_map[i].base = &cart.rom[(i & 0xf) << 16];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Custom ROM Bankswitch used in pirate "Multi-in-1" cartridges, A Bug's Life, King of Fighter 99, Pocket Monster, Rockman X3
|
||||||
|
*/
|
||||||
|
static void mapper_64k_multi_w(uint32 address)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* 64 x 64k banks */
|
||||||
|
for (i=0; i<64; i++)
|
||||||
|
{
|
||||||
|
m68k_memory_map[i].base = &cart.rom[((address++) & 0x3f) << 16];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Custom ROM Bankswitch used in RADICA cartridges
|
||||||
|
*/
|
||||||
|
static uint32 mapper_radica_r(uint32 address)
|
||||||
{
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
address = (address >> 1);
|
address = (address >> 1);
|
||||||
|
|
||||||
/* 64 x 64k banks */
|
/* 64 x 64k banks */
|
||||||
for (i = 0; i < 64; i++)
|
for (i = 0; i < 64; i++)
|
||||||
|
{
|
||||||
m68k_memory_map[i].base = &cart.rom[((address++)& 0x3f)<< 16];
|
m68k_memory_map[i].base = &cart.rom[((address++)& 0x3f)<< 16];
|
||||||
|
}
|
||||||
|
|
||||||
return 0xff;
|
return 0xffff;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -696,14 +855,17 @@ static uint32 radica_mapper_r(uint32 address)
|
|||||||
default !TIME signal handler
|
default !TIME signal handler
|
||||||
*************************************************************/
|
*************************************************************/
|
||||||
|
|
||||||
/* default ROM bankswitch */
|
|
||||||
static void default_time_w(uint32 address, uint32 data)
|
static void default_time_w(uint32 address, uint32 data)
|
||||||
{
|
{
|
||||||
if ((address & 0xf1) == 0xf1)
|
if (address < 0xa13040)
|
||||||
sega_mapper_w(address, data);
|
{
|
||||||
|
/* unlicensed cartridges mapper (default) */
|
||||||
|
mapper_64k_multi_w(address);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
else if (address < 0xa13040)
|
/* official cartridges mapper (default) */
|
||||||
multi_mapper_w(address, data);
|
mapper_sega_w(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -717,10 +879,10 @@ static uint32 default_regs_r(uint32 address)
|
|||||||
for (i=0; i<4; i++)
|
for (i=0; i<4; i++)
|
||||||
{
|
{
|
||||||
if ((address & cart.hw.mask[i]) == cart.hw.addr[i])
|
if ((address & cart.hw.mask[i]) == cart.hw.addr[i])
|
||||||
|
{
|
||||||
return cart.hw.regs[i];
|
return cart.hw.regs[i];
|
||||||
}
|
}
|
||||||
|
}
|
||||||
/* unused */
|
|
||||||
return m68k_read_bus_8(address);
|
return m68k_read_bus_8(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -738,22 +900,22 @@ static void default_regs_w(uint32 address, uint32 data)
|
|||||||
m68k_unused_8_w(address, data);
|
m68k_unused_8_w(address, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* special register behaviour (Lion King III, Super Donkey Kong 99, Mulan, Pocket Monsters II) */
|
/* custom register hardware (Top Fighter, Lion King III, Super Donkey Kong 99, Mulan, Pocket Monsters II, Pokemon Stadium) */
|
||||||
static void special_regs_w(uint32 address, uint32 data)
|
static void custom_regs_w(uint32 address, uint32 data)
|
||||||
{
|
{
|
||||||
/* ROM bankswitch */
|
/* ROM bankswitch */
|
||||||
if ((address >> 16) > 0x6f)
|
if ((address >> 16) > 0x6f)
|
||||||
{
|
{
|
||||||
special_mapper_w(address, data);
|
mapper_32k_w(data);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* write regs */
|
/* write register */
|
||||||
default_regs_w(address, data);
|
default_regs_w(address, data);
|
||||||
|
|
||||||
/* bitswapping (documented by Haze) */
|
/* bitswapping */
|
||||||
uint32 temp = cart.hw.regs[0];
|
uint32 temp = cart.hw.regs[0];
|
||||||
switch (cart.hw.regs[1])
|
switch (cart.hw.regs[1] & 3)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
cart.hw.regs[2] = (temp << 1);
|
cart.hw.regs[2] = (temp << 1);
|
||||||
@ -775,3 +937,17 @@ static void special_regs_w(uint32 address, uint32 data)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* alternate custom register hardware (Chinese Fighters III) */
|
||||||
|
static void custom_alt_regs_w(uint32 address, uint32 data)
|
||||||
|
{
|
||||||
|
/* ROM bankswitch */
|
||||||
|
if ((address >> 16) > 0x5f)
|
||||||
|
{
|
||||||
|
mapper_64k_w(data);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* write regs */
|
||||||
|
default_regs_w(address, data);
|
||||||
|
}
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
* Genesis Plus
|
* Genesis Plus
|
||||||
* Cartridge Hardware support
|
* Cartridge Hardware support
|
||||||
*
|
*
|
||||||
* Copyright (C) 2007, 2008, 2009 Eke-Eke (GCN/Wii port)
|
* Copyright (C) 2007, 2008, 2009, 2010 Eke-Eke (GCN/Wii port)
|
||||||
*
|
*
|
||||||
* Lots of protection mechanism have been discovered by Haze
|
* Lots of protection mechanism have been discovered by Haze
|
||||||
* (http://haze.mameworld.info/)
|
* (http://haze.mameworld.info/)
|
||||||
@ -38,8 +38,8 @@ typedef struct
|
|||||||
uint8 regs[4]; /* internal registers (R/W) */
|
uint8 regs[4]; /* internal registers (R/W) */
|
||||||
uint32 mask[4]; /* registers address mask */
|
uint32 mask[4]; /* registers address mask */
|
||||||
uint32 addr[4]; /* registers address */
|
uint32 addr[4]; /* registers address */
|
||||||
uint16 realtec; /* bit 0: realtec mapper detected, bit 1: bootrom enabled */
|
uint16 realtec; /* realtec mapper */
|
||||||
uint16 bankshift; /* cartridge with bankshift mecanism */
|
uint16 bankshift; /* cartridge with bankshift mecanism reseted on soft-reset */
|
||||||
unsigned int (*time_r)(unsigned int address); /* !TIME signal ($a130xx) read handler */
|
unsigned int (*time_r)(unsigned int address); /* !TIME signal ($a130xx) read handler */
|
||||||
void (*time_w)(unsigned int address, unsigned int data); /* !TIME signal ($a130xx) write handler */
|
void (*time_w)(unsigned int address, unsigned int data); /* !TIME signal ($a130xx) write handler */
|
||||||
unsigned int (*regs_r)(unsigned int address); /* cart hardware registers read handler */
|
unsigned int (*regs_r)(unsigned int address); /* cart hardware registers read handler */
|
||||||
@ -63,7 +63,9 @@ extern T_CART cart;
|
|||||||
|
|
||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
extern void cart_hw_init();
|
extern void cart_hw_init();
|
||||||
extern void cart_hw_reset();
|
extern void cart_hw_reset(int hard);
|
||||||
|
extern int cart_hw_context_save(uint8 *state);
|
||||||
|
extern int cart_hw_context_load(uint8 *state, char *version);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -34,10 +34,11 @@ static struct
|
|||||||
uint32 addr[6];
|
uint32 addr[6];
|
||||||
} ggenie;
|
} ggenie;
|
||||||
|
|
||||||
static void ggenie_write_byte(uint32 address, uint32 data);
|
static unsigned int ggenie_read_byte(unsigned int address);
|
||||||
static void ggenie_write_word(uint32 address, uint32 data);
|
static unsigned int ggenie_read_word(unsigned int address);
|
||||||
static void ggenie_write_regs(uint8 offset, uint32 data, uint8 type);
|
static void ggenie_write_byte(unsigned int address, unsigned int data);
|
||||||
static uint32 ggenie_read_regs(uint32 address);
|
static void ggenie_write_word(unsigned int address, unsigned int data);
|
||||||
|
static void ggenie_write_regs(unsigned int offset, unsigned int data);
|
||||||
|
|
||||||
void ggenie_init(void)
|
void ggenie_init(void)
|
||||||
{
|
{
|
||||||
@ -87,11 +88,12 @@ void ggenie_shutdown(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ggenie_reset(void)
|
void ggenie_reset(int hard)
|
||||||
{
|
{
|
||||||
if (!ggenie.enabled)
|
if (ggenie.enabled)
|
||||||
return;
|
{
|
||||||
|
if (hard)
|
||||||
|
{
|
||||||
/* clear codes */
|
/* clear codes */
|
||||||
ggenie_switch(0);
|
ggenie_switch(0);
|
||||||
|
|
||||||
@ -100,6 +102,7 @@ void ggenie_reset(void)
|
|||||||
memset(ggenie.old,0,sizeof(ggenie.old));
|
memset(ggenie.old,0,sizeof(ggenie.old));
|
||||||
memset(ggenie.data,0,sizeof(ggenie.data));
|
memset(ggenie.data,0,sizeof(ggenie.data));
|
||||||
memset(ggenie.addr,0,sizeof(ggenie.addr));
|
memset(ggenie.addr,0,sizeof(ggenie.addr));
|
||||||
|
}
|
||||||
|
|
||||||
/* Game Genie ROM is mapped at $000000-$007fff */
|
/* Game Genie ROM is mapped at $000000-$007fff */
|
||||||
m68k_memory_map[0].base = ggenie.rom;
|
m68k_memory_map[0].base = ggenie.rom;
|
||||||
@ -110,6 +113,7 @@ void ggenie_reset(void)
|
|||||||
|
|
||||||
/* Disable registers reads */
|
/* Disable registers reads */
|
||||||
m68k_memory_map[0].read16 = NULL;
|
m68k_memory_map[0].read16 = NULL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ggenie_switch(int enable)
|
void ggenie_switch(int enable)
|
||||||
@ -156,71 +160,75 @@ void ggenie_switch(int enable)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Byte write handler */
|
static unsigned int ggenie_read_byte(unsigned int address)
|
||||||
/* Note: 2nd revision of the Game Genie software use byte writes to set register values on exit */
|
{
|
||||||
static void ggenie_write_byte(uint32 address, uint32 data)
|
unsigned int data = ggenie.regs[(address >> 1) & 0x1f];
|
||||||
|
return ((address & 1) ? (data & 0xff) : ((data >> 8) & 0xff));
|
||||||
|
}
|
||||||
|
|
||||||
|
static unsigned int ggenie_read_word(unsigned int address)
|
||||||
|
{
|
||||||
|
return ggenie.regs[(address >> 1) & 0x1f];
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ggenie_write_byte(unsigned int address, unsigned int data)
|
||||||
{
|
{
|
||||||
/* Register offset */
|
/* Register offset */
|
||||||
uint8 offset = (address >> 1) & 0x1f;
|
uint8 offset = (address >> 1) & 0x1f;
|
||||||
|
|
||||||
/* Write internal register (lower or upper BYTE) */
|
/* /LWR and /UWR are used to decode writes */
|
||||||
ggenie_write_regs(offset,data,address & 1);
|
if (address & 1)
|
||||||
|
{
|
||||||
|
data = (ggenie.regs[offset] & 0xff00) | (data & 0xff);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
data = (ggenie.regs[offset] & 0x00ff) | ((data & 0xff) << 8);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Update internal register */
|
||||||
|
ggenie_write_regs(offset,data);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Word write handler */
|
static void ggenie_write_word(unsigned int address, unsigned int data)
|
||||||
static void ggenie_write_word(uint32 address, uint32 data)
|
|
||||||
{
|
{
|
||||||
/* Register offset */
|
/* Register offset */
|
||||||
uint8 offset = (address >> 1) & 0x1f;
|
uint8 offset = (address >> 1) & 0x1f;
|
||||||
|
|
||||||
/* Write internal register (full WORD) */
|
/* Write internal register (full WORD) */
|
||||||
ggenie_write_regs(offset,data,2);
|
ggenie_write_regs(offset,data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ggenie_write_regs(uint8 offset, uint32 data, uint8 type)
|
static void ggenie_write_regs(unsigned int offset, unsigned int data)
|
||||||
{
|
{
|
||||||
/* access type */
|
|
||||||
switch (type)
|
|
||||||
{
|
|
||||||
case 0: /* upper byte write */
|
|
||||||
data = (ggenie.regs[offset] & 0x00ff) | ((data & 0xff) << 8);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 1: /* lower byte write */
|
|
||||||
data = (ggenie.regs[offset] & 0xff00) | (data & 0xff);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* update internal register */
|
/* update internal register */
|
||||||
ggenie.regs[offset] = data;
|
ggenie.regs[offset] = data;
|
||||||
|
|
||||||
/* Mode Register */
|
/* Mode Register */
|
||||||
if (offset == 0)
|
if (offset == 0)
|
||||||
{
|
{
|
||||||
/* by default, registers are write only */
|
/* MODE bit */
|
||||||
m68k_memory_map[0].read16 = NULL;
|
|
||||||
|
|
||||||
/* MODE bits */
|
|
||||||
if (data & 0x400)
|
if (data & 0x400)
|
||||||
{
|
{
|
||||||
/* $0000-$7ffff reads mapped to Cartridge ROM */
|
/* $0000-$7ffff reads mapped to Cartridge ROM */
|
||||||
m68k_memory_map[0].base = cart.rom;
|
m68k_memory_map[0].base = cart.rom;
|
||||||
|
m68k_memory_map[0].read8 = NULL;
|
||||||
m68k_memory_map[0].read16 = NULL;
|
m68k_memory_map[0].read16 = NULL;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* $0000-$7ffff reads mapped to Game Genie ROM */
|
/* $0000-$7ffff reads mapped to Game Genie ROM */
|
||||||
m68k_memory_map[0].base = ggenie.rom;
|
m68k_memory_map[0].base = ggenie.rom;
|
||||||
|
m68k_memory_map[0].read8 = NULL;
|
||||||
m68k_memory_map[0].read16 = NULL;
|
m68k_memory_map[0].read16 = NULL;
|
||||||
|
|
||||||
|
/* READ_ENABLE bit */
|
||||||
if (data & 0x200)
|
if (data & 0x200)
|
||||||
{
|
{
|
||||||
/* $0000-$7ffff reads mapped to Game Genie Registers */
|
/* $0000-$7ffff reads mapped to Game Genie Registers */
|
||||||
/* code doing this should execute in RAM so we don't need to modify base address */
|
/* code doing this should execute in RAM so we don't need to modify base address */
|
||||||
m68k_memory_map[0].read16 = ggenie_read_regs;
|
m68k_memory_map[0].read8 = ggenie_read_byte;
|
||||||
|
m68k_memory_map[0].read16 = ggenie_read_word;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -253,6 +261,11 @@ static void ggenie_write_regs(uint8 offset, uint32 data, uint8 type)
|
|||||||
/* on real HW, address decoding would be done on each reads */
|
/* on real HW, address decoding would be done on each reads */
|
||||||
ggenie_switch(1);
|
ggenie_switch(1);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m68k_memory_map[0].write8 = ggenie_write_byte;
|
||||||
|
m68k_memory_map[0].write16 = ggenie_write_word;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* RESET register */
|
/* RESET register */
|
||||||
@ -261,9 +274,3 @@ static void ggenie_write_regs(uint8 offset, uint32 data, uint8 type)
|
|||||||
ggenie.regs[1] |= 1;
|
ggenie.regs[1] |= 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint32 ggenie_read_regs(uint32 address)
|
|
||||||
{
|
|
||||||
return ggenie.regs[(address >> 1) & 0x1f];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
@ -28,7 +28,7 @@
|
|||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
extern void ggenie_init(void);
|
extern void ggenie_init(void);
|
||||||
extern void ggenie_shutdown(void);
|
extern void ggenie_shutdown(void);
|
||||||
extern void ggenie_reset(void);
|
extern void ggenie_reset(int hard);
|
||||||
extern void ggenie_switch(int enable);
|
extern void ggenie_switch(int enable);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -99,7 +99,9 @@ static void lightgun_update(int num)
|
|||||||
{
|
{
|
||||||
/* External Interrupt ? */
|
/* External Interrupt ? */
|
||||||
if (reg[11] & 0x08)
|
if (reg[11] & 0x08)
|
||||||
irq_status = (irq_status & ~0x40) | 0x12;
|
{
|
||||||
|
irq_status = (irq_status & 4) | 0x12;
|
||||||
|
}
|
||||||
|
|
||||||
/* HV Counter Latch:
|
/* HV Counter Latch:
|
||||||
1) some games does not enable HVC latch but instead use bigger X offset
|
1) some games does not enable HVC latch but instead use bigger X offset
|
||||||
@ -107,11 +109,15 @@ static void lightgun_update(int num)
|
|||||||
2) for games using H40 mode, the gun routine scales up the Hcounter value
|
2) for games using H40 mode, the gun routine scales up the Hcounter value
|
||||||
--> H-Counter range is approx. 290 dot clocks
|
--> H-Counter range is approx. 290 dot clocks
|
||||||
*/
|
*/
|
||||||
hvc_latch = 0x10000 | (vctab[v_counter] << 8);
|
hvc_latch = 0x10000 | (v_counter << 8);
|
||||||
if (reg[12] & 1)
|
if (reg[12] & 1)
|
||||||
|
{
|
||||||
hvc_latch |= hc_320[((input.analog[num][0] * 290) / (2 * 320) + x_offset) % 210];
|
hvc_latch |= hc_320[((input.analog[num][0] * 290) / (2 * 320) + x_offset) % 210];
|
||||||
|
}
|
||||||
else
|
else
|
||||||
hvc_latch |= hc_256[(input.analog[num][0] / 2 + x_offset)%171];
|
{
|
||||||
|
hvc_latch |= hc_256[(input.analog[num][0] / 2 + x_offset) % 171];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -142,37 +142,44 @@ void gen_hardreset(void)
|
|||||||
mcycles_68k = mcycles_z80 = (uint32)((MCYCLES_PER_LINE * lines_per_frame) * ((double)rand() / (double)RAND_MAX));
|
mcycles_68k = mcycles_z80 = (uint32)((MCYCLES_PER_LINE * lines_per_frame) * ((double)rand() / (double)RAND_MAX));
|
||||||
|
|
||||||
/* Z80 bus is released & Z80 is stopped */
|
/* Z80 bus is released & Z80 is stopped */
|
||||||
|
zstate = 0;
|
||||||
m68k_memory_map[0xa0].read8 = m68k_read_bus_8;
|
m68k_memory_map[0xa0].read8 = m68k_read_bus_8;
|
||||||
m68k_memory_map[0xa0].read16 = m68k_read_bus_16;
|
m68k_memory_map[0xa0].read16 = m68k_read_bus_16;
|
||||||
m68k_memory_map[0xa0].write8 = m68k_unused_8_w;
|
m68k_memory_map[0xa0].write8 = m68k_unused_8_w;
|
||||||
m68k_memory_map[0xa0].write16 = m68k_unused_16_w;
|
m68k_memory_map[0xa0].write16 = m68k_unused_16_w;
|
||||||
zstate = 0;
|
|
||||||
|
|
||||||
/* Assume default bank is $000000-$007FFF */
|
/* Assume default bank is $000000-$007FFF */
|
||||||
zbank = 0;
|
zbank = 0;
|
||||||
|
|
||||||
/* Reset 68k, Z80 & YM2612 */
|
/* Reset 68k & Z80 */
|
||||||
m68k_pulse_reset();
|
m68k_pulse_reset();
|
||||||
z80_reset();
|
z80_reset();
|
||||||
YM2612ResetChip();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void gen_softreset(int state)
|
void gen_softreset(int state)
|
||||||
{
|
{
|
||||||
if (state)
|
if (state)
|
||||||
{
|
{
|
||||||
/* Halt 68k, Z80 & YM2612 */
|
/* Halt 68k */
|
||||||
m68k_pulse_halt();
|
m68k_pulse_halt();
|
||||||
|
|
||||||
|
/* Z80 bus is released & Z80 is reseted */
|
||||||
zstate = 0;
|
zstate = 0;
|
||||||
YM2612ResetChip();
|
m68k_memory_map[0xa0].read8 = m68k_read_bus_8;
|
||||||
|
m68k_memory_map[0xa0].read16 = m68k_read_bus_16;
|
||||||
|
m68k_memory_map[0xa0].write8 = m68k_unused_8_w;
|
||||||
|
m68k_memory_map[0xa0].write16 = m68k_unused_16_w;
|
||||||
|
|
||||||
|
/* Assume default bank is $000000-$007FFF */
|
||||||
|
zbank = 0;
|
||||||
|
|
||||||
|
/* Reset YM2612 */
|
||||||
|
fm_reset(0);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* Reset PRO Action Replay (if switch is in TRAINER position) */
|
/* Reset Cartridge Hardware */
|
||||||
if (areplay_get_status() == AR_SWITCH_TRAINER)
|
cart_hw_reset(0);
|
||||||
{
|
|
||||||
areplay_reset(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 68k & Z80 could restart anywhere in VDP frame (Bonkers, Eternal Champions, X-Men 2) */
|
/* 68k & Z80 could restart anywhere in VDP frame (Bonkers, Eternal Champions, X-Men 2) */
|
||||||
mcycles_68k = mcycles_z80 = (uint32)((MCYCLES_PER_LINE * lines_per_frame) * ((double)rand() / (double)RAND_MAX));
|
mcycles_68k = mcycles_z80 = (uint32)((MCYCLES_PER_LINE * lines_per_frame) * ((double)rand() / (double)RAND_MAX));
|
||||||
@ -180,7 +187,7 @@ void gen_softreset(int state)
|
|||||||
/* Reset 68k, Z80 & YM2612 */
|
/* Reset 68k, Z80 & YM2612 */
|
||||||
m68k_pulse_reset();
|
m68k_pulse_reset();
|
||||||
z80_reset();
|
z80_reset();
|
||||||
YM2612ResetChip();
|
fm_reset(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -458,9 +458,14 @@ static void gxSetAspectRatio(int *xscale, int *yscale)
|
|||||||
{
|
{
|
||||||
/* vertical borders */
|
/* vertical borders */
|
||||||
if (config.overscan & 1)
|
if (config.overscan & 1)
|
||||||
|
{
|
||||||
|
/* Genesis outputs 288(PAL) or 243(NTSC) lines */
|
||||||
|
/* Wii & Game Cube output 286/574(PAL50) or 240/480 (PAL60 & NTSC) lines */
|
||||||
*yscale = vdp_pal + ((gc_pal && !config.render) ? 143 : 120);
|
*yscale = vdp_pal + ((gc_pal && !config.render) ? 143 : 120);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
/* overscan is simulated (black) */
|
||||||
*yscale = bitmap.viewport.h / 2;
|
*yscale = bitmap.viewport.h / 2;
|
||||||
if (vdp_pal && (!gc_pal || config.render))
|
if (vdp_pal && (!gc_pal || config.render))
|
||||||
*yscale = *yscale * 240 / 288;
|
*yscale = *yscale * 240 / 288;
|
||||||
@ -470,29 +475,45 @@ static void gxSetAspectRatio(int *xscale, int *yscale)
|
|||||||
|
|
||||||
/* horizontal borders */
|
/* horizontal borders */
|
||||||
if (config.overscan & 2)
|
if (config.overscan & 2)
|
||||||
*xscale = 358 + ((reg[12] & 1)*2) - gc_pal;
|
{
|
||||||
|
/* max visible range is ~712 pixels, not 720 */
|
||||||
|
*xscale = 356;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
*xscale = 325 + ((reg[12] & 1)*2) - gc_pal;
|
{
|
||||||
|
/* overscan is simulated (black) */
|
||||||
|
*xscale = 327;
|
||||||
|
}
|
||||||
|
|
||||||
/* 16/9 correction */
|
/* 16/9 correction */
|
||||||
if (config.aspect & 2)
|
if (config.aspect & 2)
|
||||||
|
{
|
||||||
*xscale = (*xscale * 3) / 4;
|
*xscale = (*xscale * 3) / 4;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* manual aspect ratio (default is fullscreen) */
|
/* manual aspect ratio (default is unscaled raw) */
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* vertical borders */
|
/* vertical borders */
|
||||||
if (config.overscan & 1)
|
if (config.overscan & 1)
|
||||||
|
{
|
||||||
*yscale = (gc_pal && !config.render) ? (vdp_pal ? (268*144 / bitmap.viewport.h):143) : (vdp_pal ? (224*144 / bitmap.viewport.h):120);
|
*yscale = (gc_pal && !config.render) ? (vdp_pal ? (268*144 / bitmap.viewport.h):143) : (vdp_pal ? (224*144 / bitmap.viewport.h):120);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
*yscale = (gc_pal && !config.render) ? 134 : 112;
|
*yscale = (gc_pal && !config.render) ? 134 : 112;
|
||||||
|
}
|
||||||
|
|
||||||
/* horizontal borders */
|
/* horizontal borders */
|
||||||
if (config.overscan & 2)
|
if (config.overscan & 2)
|
||||||
*xscale = 352;
|
{
|
||||||
|
*xscale = 348;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
*xscale = 320;
|
*xscale = 320;
|
||||||
|
}
|
||||||
|
|
||||||
/* add user scaling */
|
/* add user scaling */
|
||||||
*xscale += config.xscale;
|
*xscale += config.xscale;
|
||||||
@ -565,9 +586,11 @@ static void gxResetScaler(u32 width)
|
|||||||
int xshift = (config.xshift * rmode->fbWidth) / rmode->viWidth;
|
int xshift = (config.xshift * rmode->fbWidth) / rmode->viWidth;
|
||||||
int yshift = (config.yshift * rmode->efbHeight) / rmode->viHeight;
|
int yshift = (config.yshift * rmode->efbHeight) / rmode->viHeight;
|
||||||
|
|
||||||
/* Configure GX vertical scaling (480i/576i/480p) */
|
/* Double Resolution modes (480i/576i/480p) */
|
||||||
if (config.render)
|
if (config.render)
|
||||||
|
{
|
||||||
yscale = yscale * 2;
|
yscale = yscale * 2;
|
||||||
|
}
|
||||||
|
|
||||||
/* Set GX scaler (Vertex Position matrix) */
|
/* Set GX scaler (Vertex Position matrix) */
|
||||||
square[6] = square[3] = xshift + xscale;
|
square[6] = square[3] = xshift + xscale;
|
||||||
@ -834,7 +857,7 @@ void gxCopyScreenshot(gx_texture *texture)
|
|||||||
GX_InvalidateTexAll();
|
GX_InvalidateTexAll();
|
||||||
|
|
||||||
/* scale texture to EFB width */
|
/* scale texture to EFB width */
|
||||||
s32 w = bitmap.viewport.x ? 704 : 640;
|
s32 w = bitmap.viewport.x ? 696 : 640;
|
||||||
s32 h = (bitmap.viewport.h + 2*bitmap.viewport.y) * 2;
|
s32 h = (bitmap.viewport.h + 2*bitmap.viewport.y) * 2;
|
||||||
s32 x = -w/2;
|
s32 x = -w/2;
|
||||||
s32 y = -(240+ 2*bitmap.viewport.y);
|
s32 y = -(240+ 2*bitmap.viewport.y);
|
||||||
|
149
source/hvc.h
149
source/hvc.h
@ -123,144 +123,6 @@
|
|||||||
#ifndef _HVC_H_
|
#ifndef _HVC_H_
|
||||||
#define _HVC_H_
|
#define _HVC_H_
|
||||||
|
|
||||||
/* V counter values for NTSC 192-line display */
|
|
||||||
const uint8 vc_ntsc_192[262] = {
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F,
|
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F,
|
|
||||||
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, 0x3E, 0x3F,
|
|
||||||
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F,
|
|
||||||
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, 0x5E, 0x5F,
|
|
||||||
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F,
|
|
||||||
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F,
|
|
||||||
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
||||||
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F,
|
|
||||||
0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
|
||||||
0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, 0xB6, 0xB7, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF,
|
|
||||||
0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF,
|
|
||||||
0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA,
|
|
||||||
0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF,
|
|
||||||
0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF,
|
|
||||||
0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF
|
|
||||||
};
|
|
||||||
|
|
||||||
/* V counter values for NTSC 224-line display */
|
|
||||||
const uint8 vc_ntsc_224[262] = {
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F,
|
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F,
|
|
||||||
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, 0x3E, 0x3F,
|
|
||||||
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F,
|
|
||||||
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, 0x5E, 0x5F,
|
|
||||||
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F,
|
|
||||||
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F,
|
|
||||||
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
||||||
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F,
|
|
||||||
0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
|
||||||
0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, 0xB6, 0xB7, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF,
|
|
||||||
0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF,
|
|
||||||
0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF,
|
|
||||||
0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA,
|
|
||||||
0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF,
|
|
||||||
0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF
|
|
||||||
};
|
|
||||||
|
|
||||||
/* V counter values for NTSC 240-line display (invalid mode) */
|
|
||||||
const uint8 vc_ntsc_240[262] = {
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F,
|
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F,
|
|
||||||
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, 0x3E, 0x3F,
|
|
||||||
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F,
|
|
||||||
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, 0x5E, 0x5F,
|
|
||||||
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F,
|
|
||||||
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F,
|
|
||||||
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
||||||
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F,
|
|
||||||
0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
|
||||||
0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, 0xB6, 0xB7, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF,
|
|
||||||
0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF,
|
|
||||||
0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF,
|
|
||||||
0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF,
|
|
||||||
0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF,
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05
|
|
||||||
};
|
|
||||||
|
|
||||||
/* V counter values for PAL 192-line display */
|
|
||||||
const uint8 vc_pal_192[313] = {
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F,
|
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F,
|
|
||||||
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, 0x3E, 0x3F,
|
|
||||||
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F,
|
|
||||||
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, 0x5E, 0x5F,
|
|
||||||
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F,
|
|
||||||
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F,
|
|
||||||
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
||||||
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F,
|
|
||||||
0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
|
||||||
0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, 0xB6, 0xB7, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF,
|
|
||||||
0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF,
|
|
||||||
0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF,
|
|
||||||
0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF,
|
|
||||||
0xF0, 0xF1, 0xF2,
|
|
||||||
0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF, 0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6,
|
|
||||||
0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF, 0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6,
|
|
||||||
0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF, 0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6,
|
|
||||||
0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF, 0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
|
||||||
0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF
|
|
||||||
};
|
|
||||||
|
|
||||||
/* V counter values for PAL 224-line display */
|
|
||||||
const uint8 vc_pal_224[313] = {
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F,
|
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F,
|
|
||||||
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, 0x3E, 0x3F,
|
|
||||||
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F,
|
|
||||||
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, 0x5E, 0x5F,
|
|
||||||
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F,
|
|
||||||
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F,
|
|
||||||
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
||||||
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F,
|
|
||||||
0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
|
||||||
0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, 0xB6, 0xB7, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF,
|
|
||||||
0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF,
|
|
||||||
0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF,
|
|
||||||
0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF,
|
|
||||||
0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF,
|
|
||||||
0x00, 0x01, 0x02,
|
|
||||||
0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF, 0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6,
|
|
||||||
0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF, 0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6,
|
|
||||||
0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF, 0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
|
||||||
0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF
|
|
||||||
};
|
|
||||||
|
|
||||||
/* V counter values for PAL 240-line display */
|
|
||||||
const uint8 vc_pal_240[313] = {
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F,
|
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, 0x2E, 0x2F,
|
|
||||||
0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, 0x3E, 0x3F,
|
|
||||||
0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, 0x4E, 0x4F,
|
|
||||||
0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, 0x5E, 0x5F,
|
|
||||||
0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, 0x6E, 0x6F,
|
|
||||||
0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F,
|
|
||||||
0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, 0x8E, 0x8F,
|
|
||||||
0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x97, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, 0x9E, 0x9F,
|
|
||||||
0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, 0xAE, 0xAF,
|
|
||||||
0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, 0xB6, 0xB7, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, 0xBE, 0xBF,
|
|
||||||
0xC0, 0xC1, 0xC2, 0xC3, 0xC4, 0xC5, 0xC6, 0xC7, 0xC8, 0xC9, 0xCA, 0xCB, 0xCC, 0xCD, 0xCE, 0xCF,
|
|
||||||
0xD0, 0xD1, 0xD2, 0xD3, 0xD4, 0xD5, 0xD6, 0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF,
|
|
||||||
0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6, 0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF,
|
|
||||||
0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6, 0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF,
|
|
||||||
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A,
|
|
||||||
0xD2, 0xD3, 0xD4, 0xD5, 0xD6,
|
|
||||||
0xD7, 0xD8, 0xD9, 0xDA, 0xDB, 0xDC, 0xDD, 0xDE, 0xDF, 0xE0, 0xE1, 0xE2, 0xE3, 0xE4, 0xE5, 0xE6,
|
|
||||||
0xE7, 0xE8, 0xE9, 0xEA, 0xEB, 0xEC, 0xED, 0xEE, 0xEF, 0xF0, 0xF1, 0xF2, 0xF3, 0xF4, 0xF5, 0xF6,
|
|
||||||
0xF7, 0xF8, 0xF9, 0xFA, 0xFB, 0xFC, 0xFD, 0xFE, 0xFF
|
|
||||||
};
|
|
||||||
|
|
||||||
/***************************************************************/
|
/***************************************************************/
|
||||||
/* */
|
/* */
|
||||||
/* H-counter timings in H40 & H32 modes (starts from HINT) */
|
/* H-counter timings in H40 & H32 modes (starts from HINT) */
|
||||||
@ -276,9 +138,9 @@ const uint8 vc_pal_240[313] = {
|
|||||||
/* */
|
/* */
|
||||||
/***************************************************************/
|
/***************************************************************/
|
||||||
|
|
||||||
const uint8 cycle2hc40[3420] =
|
static const uint8 cycle2hc40[3420] =
|
||||||
{
|
{
|
||||||
/* end of active display (16 pixels -> 128 Mcycles) , HINT triggered , Vcounter jump */
|
/* end of active display (16 pixels -> 128 Mcycles) , HINT triggered , Vcounter increment */
|
||||||
0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5,
|
0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5, 0xa5,
|
||||||
0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6,
|
0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6, 0xa6,
|
||||||
0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7,
|
0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7, 0xa7,
|
||||||
@ -513,9 +375,9 @@ const uint8 cycle2hc40[3420] =
|
|||||||
0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4,
|
0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4, 0xa4,
|
||||||
};
|
};
|
||||||
|
|
||||||
const uint8 cycle2hc32[3420] =
|
static const uint8 cycle2hc32[3420] =
|
||||||
{
|
{
|
||||||
/* end of active display (16 pixels -> 160 Mcycles) , HINT triggered ? , Vcounter jump */
|
/* end of active display (16 pixels -> 160 Mcycles) , HINT triggered ? , Vcounter increment */
|
||||||
0x85, 0x85, 0x85, 0x85, 0x85, 0x85,
|
0x85, 0x85, 0x85, 0x85, 0x85, 0x85,
|
||||||
0x85, 0x85, 0x85, 0x85, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86,
|
0x85, 0x85, 0x85, 0x85, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86,
|
||||||
0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x87, 0x87, 0x87, 0x87, 0x87, 0x87, 0x87, 0x87,
|
0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x86, 0x87, 0x87, 0x87, 0x87, 0x87, 0x87, 0x87, 0x87,
|
||||||
@ -752,8 +614,7 @@ const uint8 cycle2hc32[3420] =
|
|||||||
0x85, 0x85, 0x85, 0x85, 0x85, 0x85,
|
0x85, 0x85, 0x85, 0x85, 0x85, 0x85,
|
||||||
};
|
};
|
||||||
|
|
||||||
const uint8 *vctab;
|
static const uint8 *hctab;
|
||||||
const uint8 *hctab;
|
|
||||||
|
|
||||||
#endif /* _HVC_H_ */
|
#endif /* _HVC_H_ */
|
||||||
|
|
||||||
|
@ -862,7 +862,7 @@ void m68k_run (unsigned int cycles)
|
|||||||
CPU_INT_LEVEL = (irq_status & 6) << 8;
|
CPU_INT_LEVEL = (irq_status & 6) << 8;
|
||||||
|
|
||||||
/* IRQ was triggered during previous instruction */
|
/* IRQ was triggered during previous instruction */
|
||||||
if (irq_status & 0x40)
|
if (irq_status & 0x20)
|
||||||
{
|
{
|
||||||
/* one instruction latency */
|
/* one instruction latency */
|
||||||
REG_IR = m68ki_read_imm_16();
|
REG_IR = m68ki_read_imm_16();
|
||||||
|
@ -476,8 +476,7 @@ void ctrl_io_write_word(unsigned int address, unsigned int data)
|
|||||||
|
|
||||||
case 0x30: /* TIME */
|
case 0x30: /* TIME */
|
||||||
{
|
{
|
||||||
cart.hw.time_w(address & 0xfe, data >> 8);
|
cart.hw.time_w(address, data);
|
||||||
cart.hw.time_w(address, data & 0xff);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1754,12 +1754,7 @@ void render_line(int line)
|
|||||||
int width = bitmap.viewport.w;
|
int width = bitmap.viewport.w;
|
||||||
int x_offset = bitmap.viewport.x;
|
int x_offset = bitmap.viewport.x;
|
||||||
|
|
||||||
/* background color (blanked display or vertical borders) */
|
if ((reg[1] & 0x40) && (line < bitmap.viewport.h))
|
||||||
if (!(reg[1] & 0x40) || (status & 8))
|
|
||||||
{
|
|
||||||
memset(&lb[0x20 - x_offset], 0x40, width + 2*x_offset);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
/* update pattern generator */
|
/* update pattern generator */
|
||||||
if (bg_list_index)
|
if (bg_list_index)
|
||||||
@ -1826,6 +1821,11 @@ void render_line(int line)
|
|||||||
memset(&lb[0x20 + width], 0x40, x_offset);
|
memset(&lb[0x20 + width], 0x40, x_offset);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* background color (blanked display or vertical borders) */
|
||||||
|
memset(&lb[0x20 - x_offset], 0x40, width + 2*x_offset);
|
||||||
|
}
|
||||||
|
|
||||||
/* pixel color remapping */
|
/* pixel color remapping */
|
||||||
remap_buffer(line);
|
remap_buffer(line);
|
||||||
@ -1833,15 +1833,12 @@ void render_line(int line)
|
|||||||
|
|
||||||
void remap_buffer(int line)
|
void remap_buffer(int line)
|
||||||
{
|
{
|
||||||
/* display disabled */
|
|
||||||
if (reg[0] & 0x01) return;
|
|
||||||
|
|
||||||
int width = bitmap.viewport.w + 2*bitmap.viewport.x;
|
int width = bitmap.viewport.w + 2*bitmap.viewport.x;
|
||||||
|
|
||||||
/* get line offset from framebuffer */
|
/* get line offset from framebuffer */
|
||||||
line = (line + bitmap.viewport.y) % lines_per_frame;
|
line = (line + bitmap.viewport.y) % lines_per_frame;
|
||||||
|
|
||||||
/* double resolution mode */
|
/* interlaced modes */
|
||||||
if (config.render && interlaced)
|
if (config.render && interlaced)
|
||||||
line = (line << 1) + odd_frame;
|
line = (line << 1) + odd_frame;
|
||||||
|
|
||||||
|
@ -157,6 +157,35 @@ void sound_reset(void)
|
|||||||
psg_cycles_count = 0;
|
psg_cycles_count = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int sound_context_save(uint8 *state)
|
||||||
|
{
|
||||||
|
int bufferptr = 0;
|
||||||
|
|
||||||
|
save_param(YM2612GetContextPtr(),YM2612GetContextSize());
|
||||||
|
save_param(SN76489_GetContextPtr(),SN76489_GetContextSize());
|
||||||
|
save_param(&fm_cycles_count,sizeof(fm_cycles_count));
|
||||||
|
save_param(&psg_cycles_count,sizeof(psg_cycles_count));
|
||||||
|
|
||||||
|
return bufferptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
int sound_context_load(uint8 *state, char *version)
|
||||||
|
{
|
||||||
|
int bufferptr = 0;
|
||||||
|
|
||||||
|
YM2612Restore(&state[bufferptr]);
|
||||||
|
bufferptr += YM2612GetContextSize();
|
||||||
|
load_param(SN76489_GetContextPtr(),SN76489_GetContextSize());
|
||||||
|
|
||||||
|
if (version[15] > 0x30)
|
||||||
|
{
|
||||||
|
load_param(&fm_cycles_count,sizeof(fm_cycles_count));
|
||||||
|
load_param(&psg_cycles_count,sizeof(psg_cycles_count));
|
||||||
|
}
|
||||||
|
|
||||||
|
return bufferptr;
|
||||||
|
}
|
||||||
|
|
||||||
/* End of frame update, return the number of samples run so far. */
|
/* End of frame update, return the number of samples run so far. */
|
||||||
int sound_update(unsigned int cycles)
|
int sound_update(unsigned int cycles)
|
||||||
{
|
{
|
||||||
|
@ -27,6 +27,8 @@
|
|||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
extern void sound_init(void);
|
extern void sound_init(void);
|
||||||
extern void sound_reset(void);
|
extern void sound_reset(void);
|
||||||
|
extern int sound_context_save(uint8 *state);
|
||||||
|
extern int sound_context_load(uint8 *state, char *version);
|
||||||
extern int sound_update(unsigned int cycles);
|
extern int sound_update(unsigned int cycles);
|
||||||
extern void fm_reset(unsigned int cycles);
|
extern void fm_reset(unsigned int cycles);
|
||||||
extern void fm_write(unsigned int cycles, unsigned int address, unsigned int data);
|
extern void fm_write(unsigned int cycles, unsigned int address, unsigned int data);
|
||||||
|
@ -22,16 +22,6 @@
|
|||||||
|
|
||||||
#include "shared.h"
|
#include "shared.h"
|
||||||
|
|
||||||
#define STATE_VERSION "GENPLUS-GX 1.4.0"
|
|
||||||
|
|
||||||
#define load_param(param, size) \
|
|
||||||
memcpy(param, &state[bufferptr], size); \
|
|
||||||
bufferptr+= size;
|
|
||||||
|
|
||||||
#define save_param(param, size) \
|
|
||||||
memcpy(&state[bufferptr], param, size); \
|
|
||||||
bufferptr+= size;
|
|
||||||
|
|
||||||
int state_load(unsigned char *buffer)
|
int state_load(unsigned char *buffer)
|
||||||
{
|
{
|
||||||
/* buffer size */
|
/* buffer size */
|
||||||
@ -48,9 +38,10 @@ int state_load(unsigned char *buffer)
|
|||||||
uncompress ((Bytef *)state, &outbytes, (Bytef *)(buffer + 4), inbytes);
|
uncompress ((Bytef *)state, &outbytes, (Bytef *)(buffer + 4), inbytes);
|
||||||
|
|
||||||
/* version check */
|
/* version check */
|
||||||
char version[16];
|
char version[17];
|
||||||
load_param(version,16);
|
load_param(version,16);
|
||||||
if (strncmp(version,STATE_VERSION,16))
|
version[16] = 0;
|
||||||
|
if (strncmp(version,STATE_VERSION,15))
|
||||||
{
|
{
|
||||||
free(state);
|
free(state);
|
||||||
return -1;
|
return -1;
|
||||||
@ -58,41 +49,42 @@ int state_load(unsigned char *buffer)
|
|||||||
|
|
||||||
/* reset system */
|
/* reset system */
|
||||||
system_reset();
|
system_reset();
|
||||||
m68k_memory_map[0].base = cart.base;
|
|
||||||
|
|
||||||
// GENESIS
|
// GENESIS
|
||||||
load_param(work_ram, sizeof(work_ram));
|
load_param(work_ram, sizeof(work_ram));
|
||||||
load_param(zram, sizeof(zram));
|
load_param(zram, sizeof(zram));
|
||||||
load_param(&zstate, sizeof(zstate));
|
load_param(&zstate, sizeof(zstate));
|
||||||
load_param(&zbank, sizeof(zbank));
|
load_param(&zbank, sizeof(zbank));
|
||||||
|
if (zstate == 3)
|
||||||
|
{
|
||||||
|
m68k_memory_map[0xa0].read8 = z80_read_byte;
|
||||||
|
m68k_memory_map[0xa0].read16 = z80_read_word;
|
||||||
|
m68k_memory_map[0xa0].write8 = z80_write_byte;
|
||||||
|
m68k_memory_map[0xa0].write16 = z80_write_word;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m68k_memory_map[0xa0].read8 = m68k_read_bus_8;
|
||||||
|
m68k_memory_map[0xa0].read16 = m68k_read_bus_16;
|
||||||
|
m68k_memory_map[0xa0].write8 = m68k_unused_8_w;
|
||||||
|
m68k_memory_map[0xa0].write16 = m68k_unused_16_w;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (version[15] > 0x30)
|
||||||
|
{
|
||||||
|
/* extended state */
|
||||||
|
load_param(&mcycles_68k, sizeof(mcycles_68k));
|
||||||
|
load_param(&mcycles_z80, sizeof(mcycles_z80));
|
||||||
|
}
|
||||||
|
|
||||||
// IO
|
// IO
|
||||||
load_param(io_reg, sizeof(io_reg));
|
load_param(io_reg, sizeof(io_reg));
|
||||||
|
|
||||||
// VDP
|
// VDP
|
||||||
uint8 temp_reg[0x20];
|
bufferptr += vdp_context_load(&state[bufferptr], version);
|
||||||
load_param(sat, sizeof(sat));
|
|
||||||
load_param(vram, sizeof(vram));
|
|
||||||
load_param(cram, sizeof(cram));
|
|
||||||
load_param(vsram, sizeof(vsram));
|
|
||||||
load_param(temp_reg, sizeof(temp_reg));
|
|
||||||
load_param(&addr, sizeof(addr));
|
|
||||||
load_param(&addr_latch, sizeof(addr_latch));
|
|
||||||
load_param(&code, sizeof(code));
|
|
||||||
load_param(&pending, sizeof(pending));
|
|
||||||
load_param(&status, sizeof(status));
|
|
||||||
load_param(&dmafill, sizeof(dmafill));
|
|
||||||
load_param(&hint_pending, sizeof(hint_pending));
|
|
||||||
load_param(&vint_pending, sizeof(vint_pending));
|
|
||||||
load_param(&irq_status, sizeof(irq_status));
|
|
||||||
vdp_restore(temp_reg);
|
|
||||||
|
|
||||||
// FM
|
// SOUND
|
||||||
YM2612Restore(&state[bufferptr]);
|
bufferptr += sound_context_load(&state[bufferptr], version);
|
||||||
bufferptr+= YM2612GetContextSize();
|
|
||||||
|
|
||||||
// PSG
|
|
||||||
load_param(SN76489_GetContextPtr(),SN76489_GetContextSize());
|
|
||||||
|
|
||||||
// 68000
|
// 68000
|
||||||
uint16 tmp16;
|
uint16 tmp16;
|
||||||
@ -120,6 +112,9 @@ int state_load(unsigned char *buffer)
|
|||||||
// Z80
|
// Z80
|
||||||
load_param(&Z80, sizeof(Z80_Regs));
|
load_param(&Z80, sizeof(Z80_Regs));
|
||||||
|
|
||||||
|
// Cartridge HW
|
||||||
|
bufferptr += cart_hw_context_load(&state[bufferptr], version);
|
||||||
|
|
||||||
free(state);
|
free(state);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@ -143,31 +138,17 @@ int state_save(unsigned char *buffer)
|
|||||||
save_param(zram, sizeof(zram));
|
save_param(zram, sizeof(zram));
|
||||||
save_param(&zstate, sizeof(zstate));
|
save_param(&zstate, sizeof(zstate));
|
||||||
save_param(&zbank, sizeof(zbank));
|
save_param(&zbank, sizeof(zbank));
|
||||||
|
save_param(&mcycles_68k, sizeof(mcycles_68k));
|
||||||
|
save_param(&mcycles_z80, sizeof(mcycles_z80));
|
||||||
|
|
||||||
// IO
|
// IO
|
||||||
save_param(io_reg, sizeof(io_reg));
|
save_param(io_reg, sizeof(io_reg));
|
||||||
|
|
||||||
// VDP
|
// VDP
|
||||||
save_param(sat, sizeof(sat));
|
bufferptr += vdp_context_save(&state[bufferptr]);
|
||||||
save_param(vram, sizeof(vram));
|
|
||||||
save_param(cram, sizeof(cram));
|
|
||||||
save_param(vsram, sizeof(vsram));
|
|
||||||
save_param(reg, sizeof(reg));
|
|
||||||
save_param(&addr, sizeof(addr));
|
|
||||||
save_param(&addr_latch, sizeof(addr_latch));
|
|
||||||
save_param(&code, sizeof(code));
|
|
||||||
save_param(&pending, sizeof(pending));
|
|
||||||
save_param(&status, sizeof(status));
|
|
||||||
save_param(&dmafill, sizeof(dmafill));
|
|
||||||
save_param(&hint_pending, sizeof(hint_pending));
|
|
||||||
save_param(&vint_pending, sizeof(vint_pending));
|
|
||||||
save_param(&irq_status, sizeof(irq_status));
|
|
||||||
|
|
||||||
// FM
|
// SOUND
|
||||||
save_param(YM2612GetContextPtr(),YM2612GetContextSize());
|
bufferptr += sound_context_save(&state[bufferptr]);
|
||||||
|
|
||||||
// PSG
|
|
||||||
save_param(SN76489_GetContextPtr(),SN76489_GetContextSize());
|
|
||||||
|
|
||||||
// 68000
|
// 68000
|
||||||
uint16 tmp16;
|
uint16 tmp16;
|
||||||
@ -195,6 +176,9 @@ int state_save(unsigned char *buffer)
|
|||||||
// Z80
|
// Z80
|
||||||
save_param(&Z80, sizeof(Z80_Regs));
|
save_param(&Z80, sizeof(Z80_Regs));
|
||||||
|
|
||||||
|
// Cartridge HW
|
||||||
|
bufferptr += cart_hw_context_save(&state[bufferptr]);
|
||||||
|
|
||||||
/* compress state file */
|
/* compress state file */
|
||||||
unsigned long inbytes = bufferptr;
|
unsigned long inbytes = bufferptr;
|
||||||
unsigned long outbytes = STATE_SIZE;
|
unsigned long outbytes = STATE_SIZE;
|
||||||
|
@ -23,7 +23,16 @@
|
|||||||
#ifndef _STATE_H_
|
#ifndef _STATE_H_
|
||||||
#define _STATE_H_
|
#define _STATE_H_
|
||||||
|
|
||||||
#define STATE_SIZE 0x28000
|
#define STATE_SIZE 0x48100
|
||||||
|
#define STATE_VERSION "GENPLUS-GX 1.4.1"
|
||||||
|
|
||||||
|
#define load_param(param, size) \
|
||||||
|
memcpy(param, &state[bufferptr], size); \
|
||||||
|
bufferptr+= size;
|
||||||
|
|
||||||
|
#define save_param(param, size) \
|
||||||
|
memcpy(&state[bufferptr], param, size); \
|
||||||
|
bufferptr+= size;
|
||||||
|
|
||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
extern int state_load(unsigned char *buffer);
|
extern int state_load(unsigned char *buffer);
|
||||||
|
327
source/system.c
327
source/system.c
@ -288,7 +288,7 @@ void system_init (void)
|
|||||||
void system_reset (void)
|
void system_reset (void)
|
||||||
{
|
{
|
||||||
/* Cartridge Hardware */
|
/* Cartridge Hardware */
|
||||||
cart_hw_reset();
|
cart_hw_reset(1);
|
||||||
|
|
||||||
/* Genesis hardware */
|
/* Genesis hardware */
|
||||||
gen_hardreset();
|
gen_hardreset();
|
||||||
@ -317,12 +317,31 @@ void system_shutdown (void)
|
|||||||
/****************************************************************
|
/****************************************************************
|
||||||
* Virtual Genesis Frame emulation
|
* Virtual Genesis Frame emulation
|
||||||
****************************************************************/
|
****************************************************************/
|
||||||
|
|
||||||
void system_frame (int do_skip)
|
void system_frame (int do_skip)
|
||||||
{
|
{
|
||||||
int start = 0;
|
/* line counter */
|
||||||
int end = 0;
|
|
||||||
int line = 0;
|
int line = 0;
|
||||||
|
|
||||||
|
/* Z80 interrupt flag */
|
||||||
|
int zirq = 0;
|
||||||
|
|
||||||
|
/* reload H Counter */
|
||||||
|
int h_counter = reg[10];
|
||||||
|
|
||||||
|
/* reload V Counter */
|
||||||
|
v_counter = lines_per_frame - 1;
|
||||||
|
|
||||||
|
/* reset VDP FIFO */
|
||||||
|
fifo_write_cnt = 0;
|
||||||
|
fifo_lastwrite = 0;
|
||||||
|
|
||||||
|
/* reset line cycle count */
|
||||||
|
mcycles_vdp = 0;
|
||||||
|
|
||||||
|
/* update 6-Buttons & Lightguns */
|
||||||
|
input_refresh();
|
||||||
|
|
||||||
/* display changed during VBLANK */
|
/* display changed during VBLANK */
|
||||||
if (bitmap.viewport.changed & 2)
|
if (bitmap.viewport.changed & 2)
|
||||||
{
|
{
|
||||||
@ -339,32 +358,26 @@ void system_frame (int do_skip)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* active screen height */
|
/* active screen height */
|
||||||
if (reg[1] & 8)
|
bitmap.viewport.h = 224 + ((reg[1] & 8) << 1);
|
||||||
{
|
bitmap.viewport.y = (config.overscan & 1) * (8 - (reg[1] & 8) + vdp_pal*24);
|
||||||
bitmap.viewport.h = 240;
|
|
||||||
bitmap.viewport.y = (config.overscan & 1) ? (vdp_pal ? 24 : 0) : 0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bitmap.viewport.h = 224;
|
|
||||||
bitmap.viewport.y = (config.overscan & 1) ? (vdp_pal ? 32 : 8) : 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* active screen width */
|
/* active screen width */
|
||||||
if (reg[12] & 1)
|
bitmap.viewport.w = 256 + ((reg[12] & 1) << 6);
|
||||||
{
|
bitmap.viewport.x = (config.overscan & 2) * 7;
|
||||||
bitmap.viewport.w = 320;
|
|
||||||
bitmap.viewport.x = (config.overscan & 2) ? 16 : 0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bitmap.viewport.w = 256;
|
|
||||||
bitmap.viewport.x = (config.overscan & 2) ? 12 : 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Z80 interrupt flag */
|
/* render last line of overscan */
|
||||||
int zirq = 0;
|
if (!do_skip && bitmap.viewport.y)
|
||||||
|
{
|
||||||
|
render_line(v_counter);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* parse first line of sprites */
|
||||||
|
object_which = 1;
|
||||||
|
if (reg[1] & 0x40)
|
||||||
|
{
|
||||||
|
parse_satb(0x80);
|
||||||
|
}
|
||||||
|
|
||||||
/* clear VBLANK, DMA, FIFO FULL & field flags */
|
/* clear VBLANK, DMA, FIFO FULL & field flags */
|
||||||
status &= 0xFEE5;
|
status &= 0xFEE5;
|
||||||
@ -374,94 +387,151 @@ void system_frame (int do_skip)
|
|||||||
|
|
||||||
/* even/odd field flag (interlaced modes only) */
|
/* even/odd field flag (interlaced modes only) */
|
||||||
odd_frame ^= 1;
|
odd_frame ^= 1;
|
||||||
if (interlaced) status |= (odd_frame << 4);
|
if (interlaced)
|
||||||
|
|
||||||
/* reload HCounter */
|
|
||||||
int h_counter = reg[10];
|
|
||||||
|
|
||||||
/* reset VDP FIFO */
|
|
||||||
fifo_write_cnt = 0;
|
|
||||||
fifo_lastwrite = 0;
|
|
||||||
|
|
||||||
/* reset line cycle count */
|
|
||||||
mcycles_vdp = 0;
|
|
||||||
|
|
||||||
/* parse sprites on line zero */
|
|
||||||
object_which = 1;
|
|
||||||
if (reg[1] & 0x40) parse_satb(0x80);
|
|
||||||
|
|
||||||
/* process scanlines */
|
|
||||||
for (line = 0; line < lines_per_frame; line ++)
|
|
||||||
{
|
{
|
||||||
/* update VCounter */
|
status |= (odd_frame << 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update VDP DMA */
|
||||||
|
if (dma_length)
|
||||||
|
{
|
||||||
|
vdp_update_dma(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* run 68k & Z80 */
|
||||||
|
m68k_run(MCYCLES_PER_LINE);
|
||||||
|
if (zstate == 1)
|
||||||
|
{
|
||||||
|
z80_run(MCYCLES_PER_LINE);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mcycles_z80 = MCYCLES_PER_LINE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* run SVP chip */
|
||||||
|
if (svp)
|
||||||
|
{
|
||||||
|
ssp1601_run(SVP_cycles);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update line cycle count */
|
||||||
|
mcycles_vdp += MCYCLES_PER_LINE;
|
||||||
|
|
||||||
|
/* Active Display */
|
||||||
|
do
|
||||||
|
{
|
||||||
|
/* update V Counter */
|
||||||
v_counter = line;
|
v_counter = line;
|
||||||
|
|
||||||
/* update 6-Buttons & Lightguns */
|
/* update 6-Buttons & Lightguns */
|
||||||
input_refresh();
|
input_refresh();
|
||||||
|
|
||||||
/* update VDP DMA */
|
|
||||||
if (dma_length) vdp_update_dma();
|
|
||||||
|
|
||||||
/* vertical blanking */
|
|
||||||
if (status & 8)
|
|
||||||
{
|
|
||||||
/* render overscan */
|
|
||||||
if (!do_skip && ((line < end) || (line >= start)))
|
|
||||||
render_line(line);
|
|
||||||
|
|
||||||
if (zirq)
|
|
||||||
{
|
|
||||||
/* Z80 interrupt is asserted during one exact line */
|
|
||||||
m68k_run(mcycles_vdp + 788);
|
|
||||||
if (zstate == 1) z80_run(mcycles_vdp + 788);
|
|
||||||
else mcycles_z80 = mcycles_vdp + 788;
|
|
||||||
|
|
||||||
/* clear Z80 interrupt */
|
|
||||||
z80_set_irq_line(0, CLEAR_LINE);
|
|
||||||
zirq = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* active display */
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* H Interrupt */
|
/* H Interrupt */
|
||||||
if(--h_counter < 0)
|
if(--h_counter < 0)
|
||||||
{
|
{
|
||||||
|
/* reload H Counter */
|
||||||
h_counter = reg[10];
|
h_counter = reg[10];
|
||||||
|
|
||||||
|
/* interrupt level 4 */
|
||||||
hint_pending = 0x10;
|
hint_pending = 0x10;
|
||||||
if (reg[0] & 0x10)
|
if (reg[0] & 0x10)
|
||||||
irq_status = (irq_status & ~0x40) | 0x14;
|
{
|
||||||
|
irq_status = (irq_status & 2) | 0x14;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update VDP DMA */
|
||||||
|
if (dma_length)
|
||||||
|
{
|
||||||
|
vdp_update_dma(mcycles_vdp);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* swap sprite line buffers */
|
||||||
|
object_which ^= 1;
|
||||||
|
|
||||||
|
/* render scanline */
|
||||||
|
if (!do_skip)
|
||||||
|
{
|
||||||
|
render_line(line);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* parse next line of sprites */
|
||||||
|
if ((reg[1] & 0x40) && (line < (bitmap.viewport.h - 1)))
|
||||||
|
{
|
||||||
|
parse_satb(0x81 + line);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* run 68k & Z80 */
|
||||||
|
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
||||||
|
if (zstate == 1)
|
||||||
|
{
|
||||||
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* run SVP chip */
|
||||||
|
if (svp)
|
||||||
|
{
|
||||||
|
ssp1601_run(SVP_cycles);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update line cycle count */
|
||||||
|
mcycles_vdp += MCYCLES_PER_LINE;
|
||||||
|
}
|
||||||
|
while (++line < bitmap.viewport.h);
|
||||||
|
|
||||||
/* end of active display */
|
/* end of active display */
|
||||||
if (line == bitmap.viewport.h)
|
v_counter = line;
|
||||||
{
|
|
||||||
/* set border area */
|
|
||||||
start = lines_per_frame - bitmap.viewport.y;
|
|
||||||
end = bitmap.viewport.h + bitmap.viewport.y;
|
|
||||||
|
|
||||||
/* check viewport changes */
|
/* update 6-Buttons & Lightguns */
|
||||||
if (bitmap.viewport.h != bitmap.viewport.oh)
|
input_refresh();
|
||||||
|
|
||||||
|
/* H Interrupt */
|
||||||
|
if(--h_counter < 0)
|
||||||
{
|
{
|
||||||
bitmap.viewport.oh = bitmap.viewport.h;
|
/* reload H Counter */
|
||||||
bitmap.viewport.changed |= 1;
|
h_counter = reg[10];
|
||||||
|
|
||||||
|
/* interrupt level 4 */
|
||||||
|
hint_pending = 0x10;
|
||||||
|
if (reg[0] & 0x10)
|
||||||
|
{
|
||||||
|
irq_status = (irq_status & 2) | 0x14;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bitmap.viewport.w != bitmap.viewport.ow)
|
/* overscan area */
|
||||||
|
int start = lines_per_frame - bitmap.viewport.y;
|
||||||
|
int end = bitmap.viewport.h + bitmap.viewport.y;
|
||||||
|
|
||||||
|
/* check viewport changes */
|
||||||
|
if ((bitmap.viewport.w != bitmap.viewport.ow) || (bitmap.viewport.h != bitmap.viewport.oh))
|
||||||
{
|
{
|
||||||
bitmap.viewport.ow = bitmap.viewport.w;
|
bitmap.viewport.ow = bitmap.viewport.w;
|
||||||
|
bitmap.viewport.oh = bitmap.viewport.h;
|
||||||
bitmap.viewport.changed |= 1;
|
bitmap.viewport.changed |= 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* set VBLANK flag */
|
/* set VBLANK flag */
|
||||||
status |= 0x08;
|
status |= 0x08;
|
||||||
|
|
||||||
|
/* update VDP DMA */
|
||||||
|
if (dma_length)
|
||||||
|
{
|
||||||
|
vdp_update_dma(mcycles_vdp);
|
||||||
|
}
|
||||||
|
|
||||||
/* render overscan */
|
/* render overscan */
|
||||||
if (!do_skip && (line < end))
|
if (!do_skip && (line < end))
|
||||||
|
{
|
||||||
render_line(line);
|
render_line(line);
|
||||||
|
}
|
||||||
|
|
||||||
/* update inputs (doing this here fix Warriors of Eternal Sun) */
|
/* update inputs before VINT (Warriors of Eternal Sun) */
|
||||||
osd_input_Update();
|
osd_input_Update();
|
||||||
|
|
||||||
/* delay between VINT flag & V Interrupt (Ex-Mutants, Tyrant) */
|
/* delay between VINT flag & V Interrupt (Ex-Mutants, Tyrant) */
|
||||||
@ -470,46 +540,103 @@ void system_frame (int do_skip)
|
|||||||
|
|
||||||
/* delay between VBLANK flag & V Interrupt (Dracula, OutRunners, VR Troopers) */
|
/* delay between VBLANK flag & V Interrupt (Dracula, OutRunners, VR Troopers) */
|
||||||
m68k_run(mcycles_vdp + 788);
|
m68k_run(mcycles_vdp + 788);
|
||||||
if (zstate == 1) z80_run(mcycles_vdp + 788);
|
if (zstate == 1)
|
||||||
else mcycles_z80 = mcycles_vdp + 788;
|
{
|
||||||
|
z80_run(mcycles_vdp + 788);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mcycles_z80 = mcycles_vdp + 788;
|
||||||
|
}
|
||||||
|
|
||||||
/* V Interrupt */
|
/* V Interrupt */
|
||||||
vint_pending = 0x20;
|
vint_pending = 0x20;
|
||||||
if (reg[1] & 0x20)
|
if (reg[1] & 0x20)
|
||||||
irq_status = (irq_status & ~0x40) | 0x36;
|
{
|
||||||
|
irq_status = 0x16;
|
||||||
|
}
|
||||||
|
|
||||||
/* Z80 interrupt */
|
/* Z80 interrupt */
|
||||||
z80_set_irq_line(0, ASSERT_LINE);
|
z80_set_irq_line(0, ASSERT_LINE);
|
||||||
zirq = 1;
|
zirq = 1;
|
||||||
|
|
||||||
|
/* run 68k & Z80 */
|
||||||
|
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
||||||
|
if (zstate == 1)
|
||||||
|
{
|
||||||
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* swap sprite line buffers */
|
mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
||||||
object_which ^= 1;
|
}
|
||||||
|
|
||||||
/* render scanline */
|
/* run SVP chip */
|
||||||
if (!do_skip)
|
if (svp)
|
||||||
|
{
|
||||||
|
ssp1601_run(SVP_cycles);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update line cycle count */
|
||||||
|
mcycles_vdp += MCYCLES_PER_LINE;
|
||||||
|
|
||||||
|
/* increment line count */
|
||||||
|
line++;
|
||||||
|
|
||||||
|
/* Vertical Blanking */
|
||||||
|
do
|
||||||
|
{
|
||||||
|
/* update V Counter */
|
||||||
|
v_counter = line;
|
||||||
|
|
||||||
|
/* update 6-Buttons & Lightguns */
|
||||||
|
input_refresh();
|
||||||
|
|
||||||
|
/* render overscan */
|
||||||
|
if (!do_skip && ((line < end) || (line >= start)))
|
||||||
{
|
{
|
||||||
render_line(line);
|
render_line(line);
|
||||||
|
|
||||||
/* parse sprites on next line */
|
|
||||||
if ((reg[1] & 0x40) && (line < (bitmap.viewport.h - 1)))
|
|
||||||
parse_satb(0x81 + line);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* process line */
|
/* Z80 interrupt is asserted for one line */
|
||||||
|
if (zirq)
|
||||||
|
{
|
||||||
|
m68k_run(mcycles_vdp + 788);
|
||||||
|
if (zstate == 1)
|
||||||
|
{
|
||||||
|
z80_run(mcycles_vdp + 788);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mcycles_z80 = mcycles_vdp + 788;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* clear Z80 interrupt */
|
||||||
|
z80_set_irq_line(0, CLEAR_LINE);
|
||||||
|
zirq = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* run 68k & Z80 */
|
||||||
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
m68k_run(mcycles_vdp + MCYCLES_PER_LINE);
|
||||||
if (zstate == 1) z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
if (zstate == 1)
|
||||||
else mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
{
|
||||||
|
z80_run(mcycles_vdp + MCYCLES_PER_LINE);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mcycles_z80 = mcycles_vdp + MCYCLES_PER_LINE;
|
||||||
|
}
|
||||||
|
|
||||||
/* SVP chip */
|
/* run SVP chip */
|
||||||
if (svp) ssp1601_run(SVP_cycles);
|
if (svp)
|
||||||
|
{
|
||||||
|
ssp1601_run(SVP_cycles);
|
||||||
|
}
|
||||||
|
|
||||||
/* update line cycle count */
|
/* update line cycle count */
|
||||||
mcycles_vdp += MCYCLES_PER_LINE;
|
mcycles_vdp += MCYCLES_PER_LINE;
|
||||||
}
|
}
|
||||||
|
while (++line < (lines_per_frame - 1));
|
||||||
|
|
||||||
/* adjust cpu cycle count for next frame */
|
/* adjust cpu cycle count for next frame */
|
||||||
mcycles_68k -= mcycles_vdp;
|
mcycles_68k -= mcycles_vdp;
|
||||||
|
478
source/vdp.c
478
source/vdp.c
@ -42,15 +42,11 @@ uint8 vram[0x10000]; /* Video RAM (64Kx8) */
|
|||||||
uint8 cram[0x80]; /* On-chip color RAM (64x9) */
|
uint8 cram[0x80]; /* On-chip color RAM (64x9) */
|
||||||
uint8 vsram[0x80]; /* On-chip vertical scroll RAM (40x11) */
|
uint8 vsram[0x80]; /* On-chip vertical scroll RAM (40x11) */
|
||||||
uint8 reg[0x20]; /* Internal VDP registers (23x8) */
|
uint8 reg[0x20]; /* Internal VDP registers (23x8) */
|
||||||
uint16 addr; /* Address register */
|
|
||||||
uint16 addr_latch; /* Latched A15, A14 of address */
|
|
||||||
uint8 code; /* Code register */
|
|
||||||
uint8 pending; /* Pending write flag */
|
|
||||||
uint16 status; /* VDP status flags */
|
|
||||||
uint8 dmafill; /* next VDP Write is DMA Fill */
|
|
||||||
uint8 hint_pending; /* 0= Line interrupt is pending */
|
uint8 hint_pending; /* 0= Line interrupt is pending */
|
||||||
uint8 vint_pending; /* 1= Frame interrupt is pending */
|
uint8 vint_pending; /* 1= Frame interrupt is pending */
|
||||||
uint8 irq_status; /* 68K IRQ status */
|
uint8 irq_status; /* 68K IRQ status */
|
||||||
|
uint16 status; /* VDP status flags */
|
||||||
|
uint32 dma_length; /* DMA remaining length */
|
||||||
|
|
||||||
/* Global variables */
|
/* Global variables */
|
||||||
uint16 ntab; /* Name table A base address */
|
uint16 ntab; /* Name table A base address */
|
||||||
@ -66,16 +62,16 @@ uint8 bg_pattern_cache[0x80000]; /* Cached and flipped patterns */
|
|||||||
uint8 playfield_shift; /* Width of planes A, B (in bits) */
|
uint8 playfield_shift; /* Width of planes A, B (in bits) */
|
||||||
uint8 playfield_col_mask; /* Vertical scroll mask */
|
uint8 playfield_col_mask; /* Vertical scroll mask */
|
||||||
uint16 playfield_row_mask; /* Horizontal scroll mask */
|
uint16 playfield_row_mask; /* Horizontal scroll mask */
|
||||||
uint16 v_counter; /* VDP scanline counter */
|
|
||||||
uint32 hvc_latch; /* latched HVCounter (INT2) */
|
|
||||||
uint32 dma_length; /* Current DMA remaining bytes */
|
|
||||||
int32 fifo_write_cnt; /* VDP writes fifo count */
|
|
||||||
uint32 fifo_lastwrite; /* last VDP write cycle */
|
|
||||||
uint8 odd_frame; /* 1: odd field, 0: even field */
|
uint8 odd_frame; /* 1: odd field, 0: even field */
|
||||||
uint8 im2_flag; /* 1= Interlace mode 2 is being used */
|
uint8 im2_flag; /* 1= Interlace mode 2 is being used */
|
||||||
uint8 interlaced; /* 1: Interlaced mode 1 or 2 */
|
uint8 interlaced; /* 1: Interlaced mode 1 or 2 */
|
||||||
uint8 vdp_pal; /* 1: PAL , 0: NTSC (default) */
|
uint8 vdp_pal; /* 1: PAL , 0: NTSC (default) */
|
||||||
|
uint16 v_counter; /* VDP scanline counter */
|
||||||
uint16 lines_per_frame; /* PAL: 313 lines, NTSC: 262 lines */
|
uint16 lines_per_frame; /* PAL: 313 lines, NTSC: 262 lines */
|
||||||
|
int32 fifo_write_cnt; /* VDP writes fifo count */
|
||||||
|
uint32 fifo_lastwrite; /* last VDP write cycle */
|
||||||
|
uint32 hvc_latch; /* latched HVCounter (INT2) */
|
||||||
|
uint32 vc_max; /* Vertical Counter max value */
|
||||||
|
|
||||||
|
|
||||||
/* Tables that define the playfield layout */
|
/* Tables that define the playfield layout */
|
||||||
@ -84,11 +80,18 @@ static const uint8 col_mask_table[] = { 0x0F, 0x1F, 0x0F, 0x3F };
|
|||||||
static const uint16 row_mask_table[] = { 0x0FF, 0x1FF, 0x2FF, 0x3FF };
|
static const uint16 row_mask_table[] = { 0x0FF, 0x1FF, 0x2FF, 0x3FF };
|
||||||
|
|
||||||
static uint8 border; /* Border color index */
|
static uint8 border; /* Border color index */
|
||||||
|
static uint8 dma_type; /* DMA mode */
|
||||||
|
static uint8 dmafill; /* next VDP Write is DMA Fill */
|
||||||
|
static uint8 pending; /* Pending write flag */
|
||||||
|
static uint8 code; /* Code register */
|
||||||
|
static uint16 addr; /* Address register */
|
||||||
|
static uint16 addr_latch; /* Latched A15, A14 of address */
|
||||||
|
static uint16 fill_data; /* VRAM Fill data */
|
||||||
static uint16 sat_base_mask; /* Base bits of SAT */
|
static uint16 sat_base_mask; /* Base bits of SAT */
|
||||||
static uint16 sat_addr_mask; /* Index bits of SAT */
|
static uint16 sat_addr_mask; /* Index bits of SAT */
|
||||||
static uint32 dma_endCycles; /* 68k cycles to DMA end */
|
static uint32 dma_endCycles; /* 68k cycles to DMA end */
|
||||||
static uint32 dma_type; /* DMA mode */
|
|
||||||
static uint32 fifo_latency; /* CPU access latency */
|
static uint32 fifo_latency; /* CPU access latency */
|
||||||
|
static int cached_write; /* 2nd part of 32-bit CTRL port write */
|
||||||
|
|
||||||
/* DMA Timings
|
/* DMA Timings
|
||||||
|
|
||||||
@ -120,7 +123,7 @@ static uint32 fifo_latency; /* CPU access latency */
|
|||||||
static const uint8 dma_rates[16] = {
|
static const uint8 dma_rates[16] = {
|
||||||
8, 83, 9, 102, /* 68K to VRAM (1 word = 2 bytes) */
|
8, 83, 9, 102, /* 68K to VRAM (1 word = 2 bytes) */
|
||||||
16, 167, 18, 205, /* 68K to CRAM or VSRAM */
|
16, 167, 18, 205, /* 68K to CRAM or VSRAM */
|
||||||
15, 166, 17, 204, /* DMA fill */
|
16, 167, 18, 205, /* DMA fill (same as above, dma length is adjusted to take first write in account)*/
|
||||||
8, 83, 9, 102, /* DMA Copy */
|
8, 83, 9, 102, /* DMA Copy */
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -130,9 +133,9 @@ static const uint8 dma_rates[16] = {
|
|||||||
static void fifo_update(unsigned int cycles);
|
static void fifo_update(unsigned int cycles);
|
||||||
static void data_w(unsigned int data);
|
static void data_w(unsigned int data);
|
||||||
static void reg_w(unsigned int r, unsigned int d);
|
static void reg_w(unsigned int r, unsigned int d);
|
||||||
static void dma_copy(void);
|
static void dma_copy(int length);
|
||||||
static void dma_vbus (void);
|
static void dma_vbus (int length);
|
||||||
static void dma_fill(unsigned int data);
|
static void dma_fill(int length, unsigned int data);
|
||||||
|
|
||||||
/*--------------------------------------------------------------------------*/
|
/*--------------------------------------------------------------------------*/
|
||||||
/* Init, reset, shutdown functions */
|
/* Init, reset, shutdown functions */
|
||||||
@ -141,6 +144,7 @@ void vdp_init(void)
|
|||||||
{
|
{
|
||||||
/* PAL/NTSC timings */
|
/* PAL/NTSC timings */
|
||||||
lines_per_frame = vdp_pal ? 313: 262;
|
lines_per_frame = vdp_pal ? 313: 262;
|
||||||
|
status = (status & ~1) | vdp_pal;
|
||||||
}
|
}
|
||||||
|
|
||||||
void vdp_reset(void)
|
void vdp_reset(void)
|
||||||
@ -161,6 +165,7 @@ void vdp_reset(void)
|
|||||||
hvc_latch = 0;
|
hvc_latch = 0;
|
||||||
v_counter = 0;
|
v_counter = 0;
|
||||||
dmafill = 0;
|
dmafill = 0;
|
||||||
|
fill_data = 0;
|
||||||
dma_length = 0;
|
dma_length = 0;
|
||||||
dma_endCycles = 0;
|
dma_endCycles = 0;
|
||||||
dma_type = 0;
|
dma_type = 0;
|
||||||
@ -170,8 +175,11 @@ void vdp_reset(void)
|
|||||||
fifo_write_cnt = 0;
|
fifo_write_cnt = 0;
|
||||||
fifo_lastwrite = 0;
|
fifo_lastwrite = 0;
|
||||||
fifo_latency = 190;
|
fifo_latency = 190;
|
||||||
|
cached_write = -1;
|
||||||
|
|
||||||
status = vdp_pal | 0x0200; /* FIFO empty */
|
vc_max = 0xEA + 24*vdp_pal;
|
||||||
|
|
||||||
|
status = vdp_pal | 0x200; /* FIFO empty flag */
|
||||||
|
|
||||||
ntab = 0;
|
ntab = 0;
|
||||||
ntbb = 0;
|
ntbb = 0;
|
||||||
@ -190,27 +198,18 @@ void vdp_reset(void)
|
|||||||
memset ((char *) bg_name_list, 0, sizeof (bg_name_list));
|
memset ((char *) bg_name_list, 0, sizeof (bg_name_list));
|
||||||
memset ((char *) bg_pattern_cache, 0, sizeof (bg_pattern_cache));
|
memset ((char *) bg_pattern_cache, 0, sizeof (bg_pattern_cache));
|
||||||
|
|
||||||
playfield_shift = 6;
|
/* default HVC tables */
|
||||||
playfield_col_mask = 0x0F;
|
|
||||||
playfield_row_mask = 0x0FF;
|
|
||||||
|
|
||||||
/* reset HVC tables */
|
|
||||||
vctab = vdp_pal ? vc_pal_224 : vc_ntsc_224;
|
|
||||||
hctab = cycle2hc32;
|
hctab = cycle2hc32;
|
||||||
|
|
||||||
/* reset display area */
|
/* default display area */
|
||||||
bitmap.viewport.w = 256;
|
bitmap.viewport.w = 256;
|
||||||
bitmap.viewport.h = 224;
|
bitmap.viewport.h = 224;
|
||||||
bitmap.viewport.ow = 256;
|
bitmap.viewport.ow = 256;
|
||||||
bitmap.viewport.oh = 224;
|
bitmap.viewport.oh = 224;
|
||||||
|
|
||||||
/* reset overscan area */
|
/* default overscan area */
|
||||||
bitmap.viewport.x = 0;
|
bitmap.viewport.x = (config.overscan & 2) * 7;
|
||||||
bitmap.viewport.y = 0;
|
bitmap.viewport.y = (config.overscan & 1) * (8 + 24*vdp_pal);
|
||||||
if (config.overscan & 1)
|
|
||||||
bitmap.viewport.y = vdp_pal ? 32 : 8;
|
|
||||||
if (config.overscan & 2)
|
|
||||||
bitmap.viewport.x = 12;
|
|
||||||
|
|
||||||
/* initialize registers if OS ROM is not used */
|
/* initialize registers if OS ROM is not used */
|
||||||
if (config.tmss == 1)
|
if (config.tmss == 1)
|
||||||
@ -226,18 +225,76 @@ void vdp_reset(void)
|
|||||||
void vdp_shutdown(void)
|
void vdp_shutdown(void)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
void vdp_restore(uint8 *vdp_regs)
|
int vdp_context_save(uint8 *state)
|
||||||
{
|
{
|
||||||
int i;
|
int bufferptr = 0;
|
||||||
|
|
||||||
|
save_param(sat, sizeof(sat));
|
||||||
|
save_param(vram, sizeof(vram));
|
||||||
|
save_param(cram, sizeof(cram));
|
||||||
|
save_param(vsram, sizeof(vsram));
|
||||||
|
save_param(reg, sizeof(reg));
|
||||||
|
save_param(&addr, sizeof(addr));
|
||||||
|
save_param(&addr_latch, sizeof(addr_latch));
|
||||||
|
save_param(&code, sizeof(code));
|
||||||
|
save_param(&pending, sizeof(pending));
|
||||||
|
save_param(&status, sizeof(status));
|
||||||
|
save_param(&dmafill, sizeof(dmafill));
|
||||||
|
save_param(&hint_pending, sizeof(hint_pending));
|
||||||
|
save_param(&vint_pending, sizeof(vint_pending));
|
||||||
|
save_param(&irq_status, sizeof(irq_status));
|
||||||
|
|
||||||
|
/* extended state */
|
||||||
|
save_param(&dma_length, sizeof(dma_length));
|
||||||
|
save_param(&dma_type, sizeof(dma_type));
|
||||||
|
save_param(&fill_data, sizeof(fill_data));
|
||||||
|
save_param(&cached_write, sizeof(cached_write));
|
||||||
|
|
||||||
|
return bufferptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
int vdp_context_load(uint8 *state, char *version)
|
||||||
|
{
|
||||||
|
int i, bufferptr = 0;
|
||||||
|
uint8 temp_reg[0x20];
|
||||||
|
|
||||||
|
load_param(sat, sizeof(sat));
|
||||||
|
load_param(vram, sizeof(vram));
|
||||||
|
load_param(cram, sizeof(cram));
|
||||||
|
load_param(vsram, sizeof(vsram));
|
||||||
|
load_param(temp_reg, sizeof(temp_reg));
|
||||||
|
load_param(&addr, sizeof(addr));
|
||||||
|
load_param(&addr_latch, sizeof(addr_latch));
|
||||||
|
load_param(&code, sizeof(code));
|
||||||
|
load_param(&pending, sizeof(pending));
|
||||||
|
load_param(&status, sizeof(status));
|
||||||
|
load_param(&dmafill, sizeof(dmafill));
|
||||||
|
load_param(&hint_pending, sizeof(hint_pending));
|
||||||
|
load_param(&vint_pending, sizeof(vint_pending));
|
||||||
|
load_param(&irq_status, sizeof(irq_status));
|
||||||
|
|
||||||
|
/* extended state */
|
||||||
|
if (version[15] > 0x30)
|
||||||
|
{
|
||||||
|
load_param(&dma_length, sizeof(dma_length));
|
||||||
|
load_param(&dma_type, sizeof(dma_type));
|
||||||
|
load_param(&fill_data, sizeof(fill_data));
|
||||||
|
load_param(&cached_write, sizeof(cached_write));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* restore VDP registers */
|
||||||
for (i=0;i<0x20;i++)
|
for (i=0;i<0x20;i++)
|
||||||
{
|
{
|
||||||
reg_w(i, vdp_regs[i]);
|
reg_w(i, temp_reg[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* reinitialize HVC tables */
|
/* reinitialize HVC tables */
|
||||||
vctab = (vdp_pal) ? ((reg[1] & 8) ? vc_pal_240 : vc_pal_224) : vc_ntsc_224;
|
|
||||||
hctab = (reg[12] & 1) ? cycle2hc40 : cycle2hc32;
|
hctab = (reg[12] & 1) ? cycle2hc40 : cycle2hc32;
|
||||||
|
vc_max = 0xEA + 24*vdp_pal;
|
||||||
|
if (reg[1] & 8)
|
||||||
|
{
|
||||||
|
vc_max += (28 - 20*vdp_pal);
|
||||||
|
}
|
||||||
|
|
||||||
/* restore FIFO timings */
|
/* restore FIFO timings */
|
||||||
fifo_latency = (reg[12] & 1) ? 190 : 214;
|
fifo_latency = (reg[12] & 1) ? 190 : 214;
|
||||||
@ -256,6 +313,8 @@ void vdp_restore(uint8 *vdp_regs)
|
|||||||
color_update(0x00, *(uint16 *)&cram[border << 1]);
|
color_update(0x00, *(uint16 *)&cram[border << 1]);
|
||||||
for(i = 1; i < 0x40; i += 1)
|
for(i = 1; i < 0x40; i += 1)
|
||||||
color_update(i, *(uint16 *)&cram[i << 1]);
|
color_update(i, *(uint16 *)&cram[i << 1]);
|
||||||
|
|
||||||
|
return bufferptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -263,66 +322,125 @@ void vdp_restore(uint8 *vdp_regs)
|
|||||||
/* DMA update */
|
/* DMA update */
|
||||||
/*--------------------------------------------------------------------------*/
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
void vdp_update_dma()
|
void vdp_update_dma(unsigned int cycles)
|
||||||
{
|
{
|
||||||
int dma_cycles = 0;
|
int dma_cycles;
|
||||||
|
|
||||||
/* update DMA timings */
|
|
||||||
unsigned int index = dma_type;
|
|
||||||
if ((status & 8) || !(reg[1] & 0x40)) index++;
|
|
||||||
if (reg[12] & 1) index += 2;
|
|
||||||
|
|
||||||
/* DMA transfer rate (bytes per line) */
|
/* DMA transfer rate (bytes per line) */
|
||||||
unsigned int rate = dma_rates[index];
|
unsigned int rate = dma_type;
|
||||||
|
if ((status & 8) || !(reg[1] & 0x40)) rate++;
|
||||||
|
if (reg[12] & 1) rate += 2;
|
||||||
|
rate = dma_rates[rate];
|
||||||
|
|
||||||
/* 68k cycles left */
|
/* Remaining DMA cycles */
|
||||||
int left_cycles = (mcycles_vdp + MCYCLES_PER_LINE) - mcycles_68k;
|
if (status & 8)
|
||||||
if (left_cycles < 0) left_cycles = 0;
|
{
|
||||||
|
/* Process DMA until the end of VBLANK */
|
||||||
|
dma_cycles = (lines_per_frame * MCYCLES_PER_LINE) - cycles;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Process DMA until the end of current line */
|
||||||
|
dma_cycles = (mcycles_vdp + MCYCLES_PER_LINE) - cycles;
|
||||||
|
}
|
||||||
|
|
||||||
/* DMA bytes left */
|
/* Remaining DMA bytes */
|
||||||
unsigned int dma_bytes = (left_cycles * rate) / MCYCLES_PER_LINE;
|
int dma_bytes = (dma_cycles * rate) / MCYCLES_PER_LINE;
|
||||||
|
|
||||||
#ifdef LOGVDP
|
#ifdef LOGVDP
|
||||||
error("[%d(%d)][%d(%d)] DMA type %d (%d access/line)-> %d access (%d remaining) (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE,dma_type/4, rate, dma_length, dma_bytes, m68k_get_reg (NULL, M68K_REG_PC));
|
error("[%d(%d)][%d(%d)] DMA type %d (%d access/line)(%d cycles left)-> %d access (%d remaining) (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE,dma_type/4, rate, dma_cycles, dma_bytes, dma_length, m68k_get_reg (NULL, M68K_REG_PC));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* determinate DMA length in CPU cycles */
|
/* Check if DMA will be finished during current line */
|
||||||
if (dma_length < dma_bytes)
|
if (dma_length < dma_bytes)
|
||||||
{
|
{
|
||||||
/* DMA will be finished during this line */
|
/* Adjust remaining DMA */
|
||||||
dma_cycles = (dma_length * MCYCLES_PER_LINE) / rate;
|
dma_bytes = dma_length;
|
||||||
dma_length = 0;
|
dma_cycles = (dma_bytes * MCYCLES_PER_LINE) / rate;
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* DMA can not be finished until next scanline */
|
|
||||||
dma_cycles = left_cycles;
|
|
||||||
dma_length -= dma_bytes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* update 68k cycles counter */
|
/* Update DMA timings */
|
||||||
if (dma_type < 8)
|
if (dma_type < 8)
|
||||||
{
|
{
|
||||||
/* 68K to VRAM, CRAM, VSRAM */
|
|
||||||
/* 68K is frozen during DMA operation */
|
/* 68K is frozen during DMA operation */
|
||||||
mcycles_68k += dma_cycles;
|
mcycles_68k = cycles + dma_cycles;
|
||||||
|
#ifdef LOGVDP
|
||||||
#ifdef LOGVDP
|
|
||||||
error("-->CPU frozen for %d cycles\n", dma_cycles);
|
error("-->CPU frozen for %d cycles\n", dma_cycles);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* VRAM Fill or VRAM Copy */
|
/* Set DMA Busy flag */
|
||||||
/* set DMA end cyles count */
|
status |= 0x0002;
|
||||||
dma_endCycles = mcycles_68k + dma_cycles;
|
|
||||||
|
|
||||||
|
/* 68K is still running, set DMA end cycle */
|
||||||
|
dma_endCycles = cycles + dma_cycles;
|
||||||
#ifdef LOGVDP
|
#ifdef LOGVDP
|
||||||
error("-->DMA ends in %d cycles\n", dma_cycles);
|
error("-->DMA ends in %d cycles\n", dma_cycles);
|
||||||
#endif
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
/* set DMA Busy flag */
|
/* Process DMA */
|
||||||
status |= 0x0002;
|
if (dma_bytes > 0)
|
||||||
|
{
|
||||||
|
/* Update DMA length */
|
||||||
|
dma_length -= dma_bytes;
|
||||||
|
|
||||||
|
/* Select DMA operation */
|
||||||
|
switch (dma_type >> 2)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
|
/* 68K to VRAM, CRAM or VSRAM */
|
||||||
|
dma_vbus(dma_bytes);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
{
|
||||||
|
/* VRAM Fill */
|
||||||
|
if (dmafill)
|
||||||
|
{
|
||||||
|
/* process intial write */
|
||||||
|
dmafill = 0;
|
||||||
|
data_w(fill_data);
|
||||||
|
|
||||||
|
/* adjust remaining DMA length */
|
||||||
|
dma_bytes--;
|
||||||
|
|
||||||
|
/* fill MSB */
|
||||||
|
fill_data = (fill_data >> 8) & 0xff;
|
||||||
|
|
||||||
|
/* check remaining DMA length */
|
||||||
|
if (!dma_bytes) break;
|
||||||
|
}
|
||||||
|
|
||||||
|
dma_fill(dma_bytes, fill_data);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case 3:
|
||||||
|
{
|
||||||
|
/* VRAM Copy */
|
||||||
|
dma_copy(dma_bytes);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Check if DMA is finished */
|
||||||
|
if (!dma_length)
|
||||||
|
{
|
||||||
|
/* Reset DMA length registers */
|
||||||
|
reg[19] = reg[20] = 0;
|
||||||
|
|
||||||
|
/* Perform cached write, if any */
|
||||||
|
if (cached_write >= 0)
|
||||||
|
{
|
||||||
|
vdp_ctrl_w(cached_write);
|
||||||
|
cached_write = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -333,6 +451,14 @@ void vdp_ctrl_w(unsigned int data)
|
|||||||
{
|
{
|
||||||
if (pending == 0)
|
if (pending == 0)
|
||||||
{
|
{
|
||||||
|
/* check that DMA operation was not first triggered with the same instruction */
|
||||||
|
if (dma_length)
|
||||||
|
{
|
||||||
|
/* 2nd part of 32-bit write should be done after DMA completion (Formula One, Kawasaki Superbike Challenge) */
|
||||||
|
cached_write = data;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if ((data & 0xC000) == 0x8000)
|
if ((data & 0xC000) == 0x8000)
|
||||||
{
|
{
|
||||||
/* VDP register write */
|
/* VDP register write */
|
||||||
@ -364,18 +490,48 @@ void vdp_ctrl_w(unsigned int data)
|
|||||||
{
|
{
|
||||||
switch (reg[23] >> 6)
|
switch (reg[23] >> 6)
|
||||||
{
|
{
|
||||||
case 2: /* VRAM fill */
|
case 2:
|
||||||
|
{
|
||||||
|
/* VRAM write operation only (Williams Greatest Hits after soft reset) */
|
||||||
|
if ((code & 0x0F) == 1)
|
||||||
|
{
|
||||||
|
/* VRAM fill is triggered next time DATA port is written */
|
||||||
dmafill = 1;
|
dmafill = 1;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case 3: /* VRAM copy */
|
case 3:
|
||||||
dma_copy();
|
{
|
||||||
break;
|
/* VRAM read/write operation only */
|
||||||
|
if ((code & 0x1F) == 0x10)
|
||||||
|
{
|
||||||
|
/* retrieve DMA length */
|
||||||
|
dma_length = (reg[20] << 8 | reg[19]) & 0xFFFF;
|
||||||
|
if (!dma_length) dma_length = 0x10000;
|
||||||
|
|
||||||
default: /* V bus to VDP DMA */
|
/* VRAM copy */
|
||||||
dma_vbus();
|
dma_type = 12;
|
||||||
|
vdp_update_dma(mcycles_68k);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
/* retrieve DMA length */
|
||||||
|
dma_length = (reg[20] << 8 | reg[19]) & 0xFFFF;
|
||||||
|
if (!dma_length) dma_length = 0x10000;
|
||||||
|
|
||||||
|
/* SVP transfer latency */
|
||||||
|
if (svp && !(reg[23] & 0x60)) reg[21]--;
|
||||||
|
|
||||||
|
/* 68k to VDP DMA */
|
||||||
|
dma_type = (code & 0x06) ? 4 : 0;
|
||||||
|
vdp_update_dma(mcycles_68k);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -431,8 +587,7 @@ unsigned int vdp_ctrl_r(unsigned int cycles)
|
|||||||
temp |= 0x08;
|
temp |= 0x08;
|
||||||
|
|
||||||
/* HBLANK flag (Sonic 3 and Sonic 2 "VS Modes", Lemmings 2, Mega Turrican, V.R Troopers, Gouketsuji Ichizoku, ...) */
|
/* HBLANK flag (Sonic 3 and Sonic 2 "VS Modes", Lemmings 2, Mega Turrican, V.R Troopers, Gouketsuji Ichizoku, ...) */
|
||||||
if ((cycles % MCYCLES_PER_LINE) < 588)
|
if ((cycles % MCYCLES_PER_LINE) < 588) temp |= 0x04;
|
||||||
temp |= 0x04;
|
|
||||||
|
|
||||||
/* clear pending flag */
|
/* clear pending flag */
|
||||||
pending = 0;
|
pending = 0;
|
||||||
@ -455,16 +610,25 @@ unsigned int vdp_hvc_r(unsigned int cycles)
|
|||||||
uint8 hc = hctab[cycles%MCYCLES_PER_LINE];
|
uint8 hc = hctab[cycles%MCYCLES_PER_LINE];
|
||||||
|
|
||||||
/* Vertical Counter */
|
/* Vertical Counter */
|
||||||
uint8 vc = vctab[v_counter];
|
int16 vc = v_counter;
|
||||||
|
|
||||||
|
/* Check counter overflow */
|
||||||
|
if (vc > vc_max) vc -= lines_per_frame;
|
||||||
|
|
||||||
|
/* Interlaced Modes */
|
||||||
|
if (interlaced)
|
||||||
|
{
|
||||||
/* interlace mode 2 (Sonic the Hedgehog 2, Combat Cars) */
|
/* interlace mode 2 (Sonic the Hedgehog 2, Combat Cars) */
|
||||||
if (im2_flag)
|
if (im2_flag) vc = vc << 1;
|
||||||
vc = (vc << 1) | ((vc >> 7) & 1);
|
|
||||||
|
/* replace bit 0 with bit 8 */
|
||||||
|
vc = (vc & ~1) | ((vc >> 8) & 1);
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef LOGVDP
|
#ifdef LOGVDP
|
||||||
error("[%d(%d)][%d(%d)] VDP HVC Read -> 0x%04x (%x)\n", v_counter, cycles/MCYCLES_PER_LINE, cycles, cycles%MCYCLES_PER_LINE,(vc << 8) | hc, m68k_get_reg (NULL, M68K_REG_PC));
|
error("[%d(%d)][%d(%d)] VDP HVC Read -> 0x%04x (%x)\n", v_counter, cycles/MCYCLES_PER_LINE, cycles, cycles%MCYCLES_PER_LINE,(vc << 8) | hc, m68k_get_reg (NULL, M68K_REG_PC));
|
||||||
#endif
|
#endif
|
||||||
return ((vc << 8) | hc);
|
return (((vc & 0xff) << 8) | hc);
|
||||||
}
|
}
|
||||||
|
|
||||||
void vdp_test_w(unsigned int data)
|
void vdp_test_w(unsigned int data)
|
||||||
@ -479,9 +643,22 @@ void vdp_data_w(unsigned int data)
|
|||||||
/* Clear pending flag */
|
/* Clear pending flag */
|
||||||
pending = 0;
|
pending = 0;
|
||||||
|
|
||||||
|
/* DMA Fill */
|
||||||
if (dmafill)
|
if (dmafill)
|
||||||
{
|
{
|
||||||
dma_fill(data);
|
/* save fill data */
|
||||||
|
fill_data = data;
|
||||||
|
|
||||||
|
/* retrieve DMA length */
|
||||||
|
dma_length = (reg[20] << 8 | reg[19]) & 0xFFFF;
|
||||||
|
if (!dma_length) dma_length = 0x10000;
|
||||||
|
|
||||||
|
/* adjust DMA length (first write counts for timings) */
|
||||||
|
dma_length++;
|
||||||
|
|
||||||
|
/* perform DMA */
|
||||||
|
dma_type = 8;
|
||||||
|
vdp_update_dma(mcycles_68k);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -523,14 +700,14 @@ unsigned int vdp_data_r(void)
|
|||||||
switch (code & 0x0F)
|
switch (code & 0x0F)
|
||||||
{
|
{
|
||||||
case 0x00: /* VRAM */
|
case 0x00: /* VRAM */
|
||||||
temp = *(uint16 *) & vram[(addr & 0xFFFE)];
|
temp = *(uint16 *) &vram[(addr & 0xFFFE)];
|
||||||
#ifdef LOGVDP
|
#ifdef LOGVDP
|
||||||
error("[%d(%d)][%d(%d)] VRAM 0x%x read -> 0x%x (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE, addr, temp, m68k_get_reg (NULL, M68K_REG_PC));
|
error("[%d(%d)][%d(%d)] VRAM 0x%x read -> 0x%x (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE, addr, temp, m68k_get_reg (NULL, M68K_REG_PC));
|
||||||
#endif
|
#endif
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x08: /* CRAM */
|
case 0x08: /* CRAM */
|
||||||
temp = *(uint16 *) & cram[(addr & 0x7E)];
|
temp = *(uint16 *) &cram[(addr & 0x7E)];
|
||||||
temp = UNPACK_CRAM (temp);
|
temp = UNPACK_CRAM (temp);
|
||||||
#ifdef LOGVDP
|
#ifdef LOGVDP
|
||||||
error("[%d(%d)][%d(%d)] CRAM 0x%x read -> 0x%x (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE, addr, temp, m68k_get_reg (NULL, M68K_REG_PC));
|
error("[%d(%d)][%d(%d)] CRAM 0x%x read -> 0x%x (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE, addr, temp, m68k_get_reg (NULL, M68K_REG_PC));
|
||||||
@ -538,7 +715,7 @@ unsigned int vdp_data_r(void)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case 0x04: /* VSRAM */
|
case 0x04: /* VSRAM */
|
||||||
temp = *(uint16 *) & vsram[(addr & 0x7E)];
|
temp = *(uint16 *) &vsram[(addr & 0x7E)];
|
||||||
#ifdef LOGVDP
|
#ifdef LOGVDP
|
||||||
error("[%d(%d)][%d(%d)] VSRAM 0x%x read -> 0x%x (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE, addr, temp, m68k_get_reg (NULL, M68K_REG_PC));
|
error("[%d(%d)][%d(%d)] VSRAM 0x%x read -> 0x%x (%x)\n", v_counter, mcycles_68k/MCYCLES_PER_LINE, mcycles_68k, mcycles_68k%MCYCLES_PER_LINE, addr, temp, m68k_get_reg (NULL, M68K_REG_PC));
|
||||||
#endif
|
#endif
|
||||||
@ -569,7 +746,7 @@ int vdp_int_ack_callback(int int_level)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* VINT triggered ? */
|
/* VINT triggered ? */
|
||||||
if (irq_status&0x20)
|
if ((irq_status & 6) == 6)
|
||||||
{
|
{
|
||||||
vint_pending = 0;
|
vint_pending = 0;
|
||||||
status &= ~0x80; /* clear VINT flag */
|
status &= ~0x80; /* clear VINT flag */
|
||||||
@ -696,7 +873,7 @@ static void data_w(unsigned int data)
|
|||||||
color_update (0x00, data);
|
color_update (0x00, data);
|
||||||
|
|
||||||
/* CRAM modified during HBLANK (Striker, Zero the Kamikaze, etc) */
|
/* CRAM modified during HBLANK (Striker, Zero the Kamikaze, etc) */
|
||||||
if (!(status & 8) && (reg[1]& 0x40) && (mcycles_68k <= (mcycles_vdp + 860)))
|
if ((v_counter < bitmap.viewport.h) && (reg[1]& 0x40) && (mcycles_68k <= (mcycles_vdp + 860)))
|
||||||
{
|
{
|
||||||
/* remap current line */
|
/* remap current line */
|
||||||
remap_buffer(v_counter);
|
remap_buffer(v_counter);
|
||||||
@ -754,9 +931,9 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
if ((r & 0x10) && hint_pending)
|
if ((r & 0x10) && hint_pending)
|
||||||
{
|
{
|
||||||
/* update IRQ status */
|
/* update IRQ status */
|
||||||
irq_status = 0x50;
|
irq_status = 0x30;
|
||||||
if (vint_pending & reg[1])
|
if (vint_pending & reg[1])
|
||||||
irq_status |= 0x26;
|
irq_status |= 6;
|
||||||
else if (d & 0x10)
|
else if (d & 0x10)
|
||||||
irq_status |= 4;
|
irq_status |= 4;
|
||||||
}
|
}
|
||||||
@ -775,7 +952,7 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
if (r & 0x02)
|
if (r & 0x02)
|
||||||
{
|
{
|
||||||
if (reg[0] & 2) /* latch current HVC */
|
if (reg[0] & 2) /* latch current HVC */
|
||||||
hvc_latch = 0x10000 | (vctab[v_counter] << 8) | hctab[mcycles_68k%MCYCLES_PER_LINE];
|
hvc_latch = 0x10000 | vdp_hvc_r(mcycles_68k);
|
||||||
else /* free-running HVC */
|
else /* free-running HVC */
|
||||||
hvc_latch = 0;
|
hvc_latch = 0;
|
||||||
}
|
}
|
||||||
@ -792,9 +969,9 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
if ((r & 0x20) && vint_pending)
|
if ((r & 0x20) && vint_pending)
|
||||||
{
|
{
|
||||||
/* update IRQ status */
|
/* update IRQ status */
|
||||||
irq_status = 0x50;
|
irq_status = 0x30;
|
||||||
if (d & 0x20)
|
if (d & 0x20)
|
||||||
irq_status |= 0x26;
|
irq_status |= 6;
|
||||||
else if (hint_pending & reg[0])
|
else if (hint_pending & reg[0])
|
||||||
irq_status |= 4;
|
irq_status |= 4;
|
||||||
}
|
}
|
||||||
@ -802,35 +979,29 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
/* See if the viewport height has actually been changed */
|
/* See if the viewport height has actually been changed */
|
||||||
if (r & 0x08)
|
if (r & 0x08)
|
||||||
{
|
{
|
||||||
/* Update V Counter table */
|
if (v_counter < bitmap.viewport.h)
|
||||||
if (vdp_pal)
|
{
|
||||||
vctab = (d & 8) ? vc_pal_240 : vc_pal_224;
|
/* Update active display */
|
||||||
|
bitmap.viewport.h = 224 + ((d & 8) << 1);
|
||||||
/* Update viewport */
|
bitmap.viewport.y = (config.overscan & 1) * ( 8 - (d & 8) + vdp_pal*24);
|
||||||
if (status & 8)
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
/* changes should be applied on next frame */
|
/* changes should be applied on next frame */
|
||||||
bitmap.viewport.changed |= 2;
|
bitmap.viewport.changed |= 2;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
/* update vertical counter max value (see hvc.h) */
|
||||||
/* Update active display */
|
vc_max = 0xEA + 24*vdp_pal;
|
||||||
if (d & 8)
|
if (d & 8)
|
||||||
{
|
{
|
||||||
bitmap.viewport.h = 240;
|
vc_max += (28 - 20*vdp_pal);
|
||||||
bitmap.viewport.y = (config.overscan & 1) ? (vdp_pal ? 24 : 0) : 0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bitmap.viewport.h = 224;
|
|
||||||
bitmap.viewport.y = (config.overscan & 1) ? (vdp_pal ? 32 : 8) : 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Display status modified during active display (Legend of Galahad, Lemmings 2, */
|
/* Display status modified during active display (Legend of Galahad, Lemmings 2, */
|
||||||
/* Formula One Championship, Nigel Mansell's World Championship Racing, ...) */
|
/* Formula One Championship, Nigel Mansell's World Championship Racing, ...) */
|
||||||
if ((r & 0x40) && !(status & 8))
|
if ((r & 0x40) && (v_counter < bitmap.viewport.h))
|
||||||
{
|
{
|
||||||
int offset = mcycles_68k - mcycles_vdp - 860;
|
int offset = mcycles_68k - mcycles_vdp - 860;
|
||||||
if (offset <= 0)
|
if (offset <= 0)
|
||||||
@ -914,7 +1085,7 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
color_update(0x00, *(uint16 *)&cram[(d << 1)]);
|
color_update(0x00, *(uint16 *)&cram[(d << 1)]);
|
||||||
|
|
||||||
/* background color modified during Horizontal Blanking (Road Rash 1,2,3)*/
|
/* background color modified during Horizontal Blanking (Road Rash 1,2,3)*/
|
||||||
if (!(status & 8) && (mcycles_68k <= (mcycles_vdp + 860)))
|
if ((v_counter < bitmap.viewport.h) && (mcycles_68k <= (mcycles_vdp + 860)))
|
||||||
{
|
{
|
||||||
/* remap entire line */
|
/* remap entire line */
|
||||||
remap_buffer(v_counter);
|
remap_buffer(v_counter);
|
||||||
@ -962,29 +1133,34 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
/* Update HC table */
|
/* Update HC table */
|
||||||
hctab = cycle2hc32;
|
hctab = cycle2hc32;
|
||||||
|
|
||||||
/* Update fifo timings */
|
/* Update FIFO timings */
|
||||||
fifo_latency = 214;
|
fifo_latency = 214;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Adjust VRAM timings */
|
||||||
if ((code & 0x0F) == 0x01)
|
if ((code & 0x0F) == 0x01)
|
||||||
|
{
|
||||||
fifo_latency = fifo_latency * 2;
|
fifo_latency = fifo_latency * 2;
|
||||||
|
}
|
||||||
|
|
||||||
/* Update clipping */
|
/* Update clipping */
|
||||||
window_clip();
|
window_clip();
|
||||||
|
|
||||||
/* display width changed during HBLANK (Bugs Bunny Double Trouble) */
|
/* Display width switched during HBLANK (Bugs Bunny Double Trouble) */
|
||||||
if (!(status & 8))
|
if ((v_counter < bitmap.viewport.h) && (mcycles_68k <= (mcycles_vdp + 860)))
|
||||||
{
|
{
|
||||||
if (mcycles_68k <= (mcycles_vdp + 860))
|
/* Update active display */
|
||||||
{
|
bitmap.viewport.w = 256 + ((d & 1) << 6);
|
||||||
/* redraw entire line */
|
|
||||||
|
/* Redraw entire line */
|
||||||
render_line(v_counter);
|
render_line(v_counter);
|
||||||
}
|
}
|
||||||
}
|
else
|
||||||
|
{
|
||||||
/* changes should be applied on next frame */
|
/* Changes should only be applied on next frame (Golden Axe III intro) */
|
||||||
bitmap.viewport.changed |= 2;
|
bitmap.viewport.changed |= 2;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Interlaced modes */
|
/* Interlaced modes */
|
||||||
if (r & 0x06)
|
if (r & 0x06)
|
||||||
@ -1038,20 +1214,11 @@ static void reg_w(unsigned int r, unsigned int d)
|
|||||||
- see how source addr is affected
|
- see how source addr is affected
|
||||||
(can it make high source byte inc?)
|
(can it make high source byte inc?)
|
||||||
*/
|
*/
|
||||||
static void dma_copy(void)
|
static void dma_copy(int length)
|
||||||
{
|
{
|
||||||
int name;
|
int name;
|
||||||
unsigned int length = (reg[20] << 8 | reg[19]) & 0xFFFF;
|
|
||||||
unsigned int source = (reg[22] << 8 | reg[21]) & 0xFFFF;
|
unsigned int source = (reg[22] << 8 | reg[21]) & 0xFFFF;
|
||||||
|
|
||||||
if (!length)
|
|
||||||
length = 0x10000;
|
|
||||||
|
|
||||||
dma_type = 12;
|
|
||||||
dma_length = length;
|
|
||||||
vdp_update_dma();
|
|
||||||
|
|
||||||
/* proceed DMA */
|
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
vram[addr] = vram[source];
|
vram[addr] = vram[source];
|
||||||
@ -1060,29 +1227,18 @@ static void dma_copy(void)
|
|||||||
addr += reg[15];
|
addr += reg[15];
|
||||||
} while (--length);
|
} while (--length);
|
||||||
|
|
||||||
/* update length & source address registers */
|
/* update source address registers */
|
||||||
reg[19] = length & 0xFF;
|
|
||||||
reg[20] = (length >> 8) & 0xFF;
|
|
||||||
reg[21] = source & 0xFF; /* not sure */
|
reg[21] = source & 0xFF; /* not sure */
|
||||||
reg[22] = (source >> 8) & 0xFF;
|
reg[22] = (source >> 8) & 0xFF;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* 68K Copy to VRAM, VSRAM or CRAM */
|
/* 68K Copy to VRAM, VSRAM or CRAM */
|
||||||
static void dma_vbus (void)
|
static void dma_vbus (int length)
|
||||||
{
|
{
|
||||||
unsigned int source = ((reg[23] & 0x7F) << 17 | reg[22] << 9 | reg[21] << 1) & 0xFFFFFE;
|
unsigned int source = ((reg[23] & 0x7F) << 17 | reg[22] << 9 | reg[21] << 1) & 0xFFFFFE;
|
||||||
unsigned int base = source;
|
unsigned int base = source;
|
||||||
unsigned int length = (reg[20] << 8 | reg[19]) & 0xFFFF;
|
|
||||||
uint16 temp;
|
uint16 temp;
|
||||||
|
|
||||||
if (!length)
|
|
||||||
length = 0x10000;
|
|
||||||
|
|
||||||
/* DMA timings */
|
|
||||||
dma_type = (code & 0x06) ? 4 : 0;
|
|
||||||
dma_length = length;
|
|
||||||
vdp_update_dma();
|
|
||||||
|
|
||||||
/* DMA source */
|
/* DMA source */
|
||||||
if ((source >> 17) == 0x50)
|
if ((source >> 17) == 0x50)
|
||||||
{
|
{
|
||||||
@ -1105,7 +1261,9 @@ static void dma_vbus (void)
|
|||||||
|
|
||||||
/* All remaining locations access work RAM */
|
/* All remaining locations access work RAM */
|
||||||
else
|
else
|
||||||
|
{
|
||||||
temp = *(uint16 *)(work_ram + (source & 0xffff));
|
temp = *(uint16 *)(work_ram + (source & 0xffff));
|
||||||
|
}
|
||||||
|
|
||||||
source += 2;
|
source += 2;
|
||||||
source = ((base & 0xFE0000) | (source & 0x1FFFF));
|
source = ((base & 0xFE0000) | (source & 0x1FFFF));
|
||||||
@ -1115,12 +1273,6 @@ static void dma_vbus (void)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* SVP latency */
|
|
||||||
if (svp && (source < 0x400000))
|
|
||||||
{
|
|
||||||
source = (source - 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ROM & RAM */
|
/* ROM & RAM */
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
@ -1132,33 +1284,16 @@ static void dma_vbus (void)
|
|||||||
while (--length);
|
while (--length);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* update length & source address registers */
|
/* update source address registers */
|
||||||
reg[19] = length & 0xFF;
|
|
||||||
reg[20] = (length >> 8) & 0xFF;
|
|
||||||
reg[21] = (source >> 1) & 0xFF;
|
reg[21] = (source >> 1) & 0xFF;
|
||||||
reg[22] = (source >> 9) & 0xFF;
|
reg[22] = (source >> 9) & 0xFF;
|
||||||
reg[23] = (reg[23] & 0x80) | ((source >> 17) & 0x7F);
|
reg[23] = (reg[23] & 0x80) | ((source >> 17) & 0x7F);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* VRAM FILL */
|
/* VRAM FILL */
|
||||||
static void dma_fill(unsigned int data)
|
static void dma_fill(int length, unsigned int data)
|
||||||
{
|
{
|
||||||
int name;
|
int name;
|
||||||
unsigned int length = (reg[20] << 8 | reg[19]) & 0xFFFF;
|
|
||||||
|
|
||||||
if (!length)
|
|
||||||
length = 0x10000;
|
|
||||||
|
|
||||||
/* DMA timings */
|
|
||||||
dma_type = 8;
|
|
||||||
dma_length = length;
|
|
||||||
vdp_update_dma();
|
|
||||||
|
|
||||||
/* proceed DMA */
|
|
||||||
data_w(data);
|
|
||||||
|
|
||||||
/* write MSB */
|
|
||||||
data = (data >> 8) & 0xff;
|
|
||||||
|
|
||||||
/* intercept SAT writes */
|
/* intercept SAT writes */
|
||||||
if ((addr & sat_base_mask) == satb)
|
if ((addr & sat_base_mask) == satb)
|
||||||
@ -1183,9 +1318,4 @@ static void dma_fill(unsigned int data)
|
|||||||
}
|
}
|
||||||
while (--length);
|
while (--length);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* update length register */
|
|
||||||
reg[19] = length & 0xFF;
|
|
||||||
reg[20] = (length >> 8) & 0xFF;
|
|
||||||
dmafill = 0;
|
|
||||||
}
|
}
|
||||||
|
25
source/vdp.h
25
source/vdp.h
@ -30,15 +30,11 @@ extern uint8 vram[0x10000];
|
|||||||
extern uint8 cram[0x80];
|
extern uint8 cram[0x80];
|
||||||
extern uint8 vsram[0x80];
|
extern uint8 vsram[0x80];
|
||||||
extern uint8 reg[0x20];
|
extern uint8 reg[0x20];
|
||||||
extern uint16 addr;
|
|
||||||
extern uint16 addr_latch;
|
|
||||||
extern uint8 code;
|
|
||||||
extern uint8 pending;
|
|
||||||
extern uint16 status;
|
|
||||||
extern uint8 dmafill;
|
|
||||||
extern uint8 hint_pending;
|
extern uint8 hint_pending;
|
||||||
extern uint8 vint_pending;
|
extern uint8 vint_pending;
|
||||||
extern uint8 irq_status;
|
extern uint8 irq_status;
|
||||||
|
extern uint16 status;
|
||||||
|
extern uint32 dma_length;
|
||||||
|
|
||||||
/* Global variables */
|
/* Global variables */
|
||||||
extern uint16 ntab;
|
extern uint16 ntab;
|
||||||
@ -55,8 +51,6 @@ extern uint8 playfield_shift;
|
|||||||
extern uint8 playfield_col_mask;
|
extern uint8 playfield_col_mask;
|
||||||
extern uint16 playfield_row_mask;
|
extern uint16 playfield_row_mask;
|
||||||
extern uint16 v_counter;
|
extern uint16 v_counter;
|
||||||
extern uint32 hvc_latch;
|
|
||||||
extern uint32 dma_length;
|
|
||||||
extern int32 fifo_write_cnt;
|
extern int32 fifo_write_cnt;
|
||||||
extern uint32 fifo_lastwrite;
|
extern uint32 fifo_lastwrite;
|
||||||
extern uint8 im2_flag;
|
extern uint8 im2_flag;
|
||||||
@ -64,21 +58,16 @@ extern uint8 interlaced;
|
|||||||
extern uint8 odd_frame;
|
extern uint8 odd_frame;
|
||||||
extern uint8 vdp_pal;
|
extern uint8 vdp_pal;
|
||||||
extern uint16 lines_per_frame;
|
extern uint16 lines_per_frame;
|
||||||
|
extern uint32 hvc_latch;
|
||||||
extern const uint8 vc_ntsc_224[262];
|
extern uint32 vc_max;
|
||||||
extern const uint8 vc_pal_224[313];
|
|
||||||
extern const uint8 vc_pal_240[313];
|
|
||||||
extern const uint8 cycle2hc32[3420];
|
|
||||||
extern const uint8 cycle2hc40[3420];
|
|
||||||
extern const uint8 *vctab;
|
|
||||||
extern const uint8 *hctab;
|
|
||||||
|
|
||||||
/* Function prototypes */
|
/* Function prototypes */
|
||||||
extern void vdp_init(void);
|
extern void vdp_init(void);
|
||||||
extern void vdp_reset(void);
|
extern void vdp_reset(void);
|
||||||
extern void vdp_shutdown(void);
|
extern void vdp_shutdown(void);
|
||||||
extern void vdp_restore(uint8 *vdp_regs);
|
extern int vdp_context_save(uint8 *state);
|
||||||
extern void vdp_update_dma();
|
extern int vdp_context_load(uint8 *state, char *version);
|
||||||
|
extern void vdp_update_dma(unsigned int cycles);
|
||||||
extern void vdp_ctrl_w(unsigned int data);
|
extern void vdp_ctrl_w(unsigned int data);
|
||||||
extern unsigned int vdp_ctrl_r(unsigned int cycles);
|
extern unsigned int vdp_ctrl_r(unsigned int cycles);
|
||||||
extern void vdp_data_w(unsigned int data);
|
extern void vdp_data_w(unsigned int data);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user