kopia lustrzana https://github.com/Jean-MarcHarvengt/MCUME
3204 wiersze
88 KiB
C
3204 wiersze
88 KiB
C
#include <string.h>
|
|
|
|
#include "emuapi.h"
|
|
#include "iopins.h"
|
|
|
|
|
|
#include "shared.h"
|
|
#include "Z80.h" /* Z80 CPU emulation */
|
|
#include "MSX.h"
|
|
#include "Sound.h"
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
//#include <time.h>
|
|
|
|
#include "bios/cmos.h"
|
|
#include "bios/disk.h"
|
|
#include "bios/fmpac.h"
|
|
#include "bios/italic.h"
|
|
#include "bios/kanji.h"
|
|
#include "bios/msx.h"
|
|
#include "bios/msx2.h"
|
|
#include "bios/msx2ext.h"
|
|
#include "bios/msx2p.h"
|
|
#include "bios/msx2pext.h"
|
|
#include "bios/painter.h"
|
|
|
|
#include "flash_t.h"
|
|
|
|
static void RefreshLineTx80(byte Y);
|
|
static void RefreshLine0(byte Y);
|
|
static void RefreshLine1(byte Y);
|
|
static void RefreshLine2(byte Y);
|
|
static void RefreshLine3(byte Y);
|
|
static void RefreshLine4(byte Y);
|
|
static void RefreshLine5(byte Y);
|
|
static void RefreshLine6(byte Y);
|
|
static void RefreshLine7(byte Y);
|
|
static void RefreshLine8(byte Y);
|
|
static void RefreshLine10(byte Y);
|
|
static void RefreshLine12(byte Y);
|
|
|
|
|
|
/** Zero-terminated arrays of disk names for each drive ******/
|
|
extern char *Disks[2][MAXDISKS+1];
|
|
|
|
|
|
/**************************************
|
|
* Local macros/typedef
|
|
**************************************/
|
|
#define WIDTH 272
|
|
#define HEIGHT 228
|
|
|
|
#define RGB2INT(R,G,B) ((B)|((int)(G)<<8)|((int)(R)<<16))
|
|
|
|
typedef byte pixel;
|
|
|
|
|
|
/**************************************
|
|
* Local procedures
|
|
**************************************/
|
|
static void Sprites(byte Y,pixel *Line);
|
|
static void ColorSprites(byte Y,byte *ZBuf);
|
|
static pixel *RefreshBorder(byte Y,pixel C);
|
|
static void ClearLine(pixel *P,pixel C);
|
|
static pixel YJKColor(int Y,int J,int K);
|
|
|
|
/** Internal Functions ***************************************/
|
|
/** These functions are defined and internally used by the **/
|
|
/** code in MSX.c. **/
|
|
/*************************************************************/
|
|
byte *LoadROM(const char *Name,int Size,byte *Buf);
|
|
byte *LoadCARTROM(const char *Name,int Size,byte *Buf);
|
|
int LoadCart(const char *Name,int Slot);
|
|
int GuessROM(const byte *Buf,int Size);
|
|
void SetMegaROM(int Slot,byte P0,byte P1,byte P2,byte P3);
|
|
void MapROM(word A,byte V); /* Switch MegaROM banks */
|
|
void SSlot(byte V); /* Switch secondary slots */
|
|
void VDPOut(byte R,byte V); /* Write value into a VDP register */
|
|
void Printer(byte V); /* Send a character to a printer */
|
|
void PPIOut(byte New,byte Old); /* Set PPI bits (key click, etc.) */
|
|
void CheckSprites(void); /* Check collisions and 5th sprite */
|
|
byte RTCIn(byte R); /* Read RTC registers */
|
|
byte SetScreen(void); /* Change screen mode */
|
|
word SetIRQ(byte IRQ); /* Set/Reset IRQ */
|
|
word StateID(void); /* Compute emulation state ID */
|
|
|
|
static void RefreshLineF(register byte Y);
|
|
static void RefreshLine0(register byte Y);
|
|
static void RefreshLine1(register byte Y);
|
|
static void RefreshLine2(register byte Y);
|
|
static void RefreshLine3(register byte Y);
|
|
static void RefreshLine4(register byte Y);
|
|
static void RefreshLine5(register byte Y);
|
|
static void RefreshLine8(register byte Y);
|
|
static void RefreshLine10(register byte Y);
|
|
static void RefreshLine12(register byte Y);
|
|
//#ifdef NARROW
|
|
static void RefreshLine7(register byte Y);
|
|
static void RefreshLineTx80(register byte Y);
|
|
//#endif
|
|
|
|
/**************************************
|
|
* Local variables
|
|
**************************************/
|
|
static unsigned int BPal[256],XPal[80],XPal0;
|
|
|
|
static int FirstLine = 18; /* First scanline in the framebuffer */
|
|
|
|
/** User-defined parameters for fMSX *************************/
|
|
byte Verbose = 1; /* Debug msgs ON/OFF */
|
|
byte UPeriod = 2; /* Interrupts/scr. update */
|
|
int VPeriod = CPU_VPERIOD; /* CPU cycles per VBlank */
|
|
int HPeriod = CPU_HPERIOD; /* CPU cycles per HBlank */
|
|
byte SaveCMOS = 0; /* Save CMOS.ROM on exit */
|
|
byte SaveSRAM = 0; /* ~ GMASTER2.RAM on exit */
|
|
byte MSXVersion = 1; /* 0=MSX1,1=MSX2,2=MSX2+ */
|
|
byte JoyTypeA = 1; /* 0=None,1=Joystick, */
|
|
byte JoyTypeB = 0; /* 2=MouseAsJstk,3=Mouse */
|
|
byte ROMTypeA = MAXMAPPERS; /* MegaROM types */
|
|
byte ROMTypeB = MAXMAPPERS;
|
|
int RAMPages = 4; /* Number of RAM pages */
|
|
int VRAMPages = 2; /* Number of VRAM pages */
|
|
byte AutoFire = 0; /* Autofire on [SPACE] */
|
|
byte UseDrums = 0; /* Use drms for PSG noise */
|
|
byte ExitNow = 0; /* 1 = Exit the emulator */
|
|
|
|
/** Main hardware: CPU, RAM, VRAM, mappers *******************/
|
|
Z80 CPU; /* Z80 CPU state and regs */
|
|
|
|
byte *VRAM,*VPAGE; /* Video RAM */
|
|
|
|
byte *RAM[8]; /* Main RAM (8x8kB pages) */
|
|
byte *EmptyRAM; /* Empty RAM page (8kB) */
|
|
byte *SRAM; /* SRAM (battery backed) */
|
|
byte *MemMap[4][4][8]; /* Memory maps [PPage][SPage][Addr] */
|
|
|
|
byte *RAMData; /* RAM Mapper contents */
|
|
byte RAMMapper[4]; /* RAM Mapper state */
|
|
byte RAMMask; /* RAM Mapper mask */
|
|
|
|
byte *ROMData[2]; /* ROM Mapper contents */
|
|
byte ROMMapper[2][4]; /* ROM Mappers state */
|
|
byte ROMMask[2]; /* ROM Mapper masks */
|
|
|
|
byte EnWrite[4]; /* 1 if write enabled */
|
|
byte PSL[4],SSL[4]; /* Lists of current slots */
|
|
byte PSLReg,SSLReg; /* Storage for A8h port and (FFFFh) */
|
|
|
|
/** Memory blocks to free in TrashMSX() **********************/
|
|
byte *Chunks[256]; /* Memory blocks to free */
|
|
byte CCount; /* Number of memory blcks */
|
|
|
|
|
|
/** Cartridge files used by fMSX *****************************/
|
|
char CartA[128]; //"CARTA.ROM"; /* Cartridge A ROM file */
|
|
char *CartB = "CARTB.ROM"; /* Cartridge B ROM file */
|
|
|
|
/** Disk images used by fMSX *********************************/
|
|
char *DiskA = "DRIVEA.DSK"; /* Drive A disk image */
|
|
char *DiskB = "DRIVEB.DSK"; /* Drive B disk image */
|
|
|
|
/** Fixed font used by fMSX **********************************/
|
|
char *FontName = "DEFAULT.FNT"; /* Font file for text */
|
|
byte *FontBuf; /* Font for text modes */
|
|
byte UseFont = 0; /* Use ext. font when 1 */
|
|
|
|
/** Printer **************************************************/
|
|
#ifdef unused
|
|
char *PrnName = NULL; /* Printer redirect. file */
|
|
int *PrnStream;
|
|
#endif
|
|
|
|
/** Cassette tape ********************************************/
|
|
#ifdef unused
|
|
char *CasName = "DEFAULT.CAS"; /* Tape image file */
|
|
int *CasStream;
|
|
#endif
|
|
|
|
/** Serial port **********************************************/
|
|
#ifdef unused
|
|
char *ComName = NULL; /* Serial redirect. file */
|
|
int *ComIStream;
|
|
int *ComOStream;
|
|
#endif
|
|
|
|
/** Kanji font ROM *******************************************/
|
|
byte *Kanji; /* Kanji ROM 4096x32 */
|
|
int KanLetter; /* Current letter index */
|
|
byte KanCount; /* Byte count 0..31 */
|
|
|
|
/** Keyboard and mouse ***************************************/
|
|
byte KeyMap[16]; /* Keyboard map */
|
|
byte Buttons[2]; /* Mouse button states */
|
|
byte MouseDX[2],MouseDY[2]; /* Mouse offsets */
|
|
byte OldMouseX[2],OldMouseY[2]; /* Old mouse coordinates */
|
|
byte MCount[2]; /* Mouse nibble counter */
|
|
|
|
/** General I/O registers: i8255 *****************************/
|
|
I8255 PPI; /* i8255 PPI at A8h-ABh */
|
|
byte IOReg; /* Storage for AAh port */
|
|
|
|
/** Sound hardware: PSG, SCC, OPLL ***************************/
|
|
AY8910 PSG; /* PSG registers & state */
|
|
YM2413 OPLL; /* OPLL registers & state */
|
|
SCC SCChip; /* SCC registers & state */
|
|
byte SCCOn[2]; /* 1 = SCC page active */
|
|
|
|
/** Serial I/O hardware: i8251+i8253 *************************/
|
|
I8251 SIO; /* SIO registers & state */
|
|
|
|
/** Real-time clock ******************************************/
|
|
byte RTCReg,RTCMode; /* RTC register numbers */
|
|
byte RTC[4][13]; /* RTC registers */
|
|
|
|
/** Video processor ******************************************/
|
|
byte *ChrGen,*ChrTab,*ColTab; /* VDP tables (screen) */
|
|
byte *SprGen,*SprTab; /* VDP tables (sprites) */
|
|
int ChrGenM,ChrTabM,ColTabM; /* VDP masks (screen) */
|
|
int SprTabM; /* VDP masks (sprites) */
|
|
word VAddr; /* VRAM address in VDP */
|
|
byte VKey,PKey,WKey; /* Status keys for VDP */
|
|
byte FGColor,BGColor; /* Colors */
|
|
byte XFGColor,XBGColor; /* Second set of colors */
|
|
byte ScrMode; /* Current screen mode */
|
|
byte VDP[64],VDPStatus[16]; /* VDP registers */
|
|
byte IRQPending; /* Pending interrupts */
|
|
int ScanLine; /* Current scanline */
|
|
byte VDPData; /* VDP data buffer */
|
|
byte PLatch; /* Palette buffer */
|
|
byte ALatch; /* Address buffer */
|
|
int Palette[16]; /* Current palette */
|
|
|
|
/** Places in DiskROM to be patched with ED FE C9 ************/
|
|
const word DiskPatches[] = { 0x4010,0x4013,0x4016,0x401C,0x401F,0 };
|
|
|
|
/** Places in BIOS to be patched with ED FE C9 ***************/
|
|
const word BIOSPatches[] = { 0x00E1,0x00E4,0x00E7,0x00EA,0x00ED,0x00F0,0x00F3,0 };
|
|
|
|
/** Screen Mode Handlers [number of screens + 1] *************/
|
|
const void (*RefreshLine[MAXSCREEN+2])(byte Y) =
|
|
{
|
|
RefreshLine0, /* SCR 0: TEXT 40x24 */
|
|
RefreshLine1, /* SCR 1: TEXT 32x24 */
|
|
RefreshLine2, /* SCR 2: BLK 256x192 */
|
|
RefreshLine3, /* SCR 3: 64x48x16 */
|
|
RefreshLine4, /* SCR 4: BLK 256x192 */
|
|
RefreshLine5, /* SCR 5: 256x192x16 */
|
|
RefreshLine6, /* SCR 6: 512x192x4 */
|
|
RefreshLine7, /* SCR 7: 512x192x16 */
|
|
RefreshLine8, /* SCR 8: 256x192x256 */
|
|
0, /* SCR 9: NONE */
|
|
RefreshLine10, /* SCR 10: YAE 256x192 */
|
|
RefreshLine10, /* SCR 11: YAE 256x192 */
|
|
RefreshLine12, /* SCR 12: YJK 256x192 */
|
|
RefreshLineTx80 /* SCR 0: TEXT 80x24 */
|
|
};
|
|
|
|
/** VDP Address Register Masks *******************************/
|
|
const struct { byte R2,R3,R4,R5,M2,M3,M4,M5; } MSK[MAXSCREEN+2] =
|
|
{
|
|
{ 0x7F,0x00,0x3F,0x00,0x00,0x00,0x00,0x00 }, /* SCR 0: TEXT 40x24 */
|
|
{ 0x7F,0xFF,0x3F,0xFF,0x00,0x00,0x00,0x00 }, /* SCR 1: TEXT 32x24 */
|
|
{ 0x7F,0x80,0x3C,0xFF,0x00,0x7F,0x03,0x00 }, /* SCR 2: BLK 256x192 */
|
|
{ 0x7F,0x00,0x3F,0xFF,0x00,0x00,0x00,0x00 }, /* SCR 3: 64x48x16 */
|
|
{ 0x7F,0x80,0x3C,0xFC,0x00,0x7F,0x03,0x03 }, /* SCR 4: BLK 256x192 */
|
|
{ 0x60,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 5: 256x192x16 */
|
|
{ 0x60,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 6: 512x192x4 */
|
|
{ 0x20,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 7: 512x192x16 */
|
|
{ 0x20,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 8: 256x192x256 */
|
|
{ 0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00 }, /* SCR 9: NONE */
|
|
{ 0x20,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 10: YAE 256x192 */
|
|
{ 0x20,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 11: YAE 256x192 */
|
|
{ 0x20,0x00,0x00,0xFC,0x1F,0x00,0x00,0x03 }, /* SCR 12: YJK 256x192 */
|
|
{ 0x7C,0xF8,0x3F,0x00,0x03,0x07,0x00,0x00 } /* SCR 0: TEXT 80x24 */
|
|
};
|
|
|
|
/** MegaROM Mapper Names *************************************/
|
|
char *ROMNames[MAXMAPPERS+1] =
|
|
{
|
|
"GENERIC/8kB","GENERIC/16kB","KONAMI5/8kB",
|
|
"KONAMI4/8kB","ASCII/8kB","ASCII/16kB",
|
|
"GMASTER2/SRAM","UNKNOWN"
|
|
};
|
|
|
|
static byte JoyState;
|
|
static int ik; // joypad key
|
|
static int ihk; // I2C keyboard key
|
|
static int iusbhk; // USB keyboard key
|
|
static int prevhk; // previous keyboard key
|
|
|
|
|
|
static unsigned char linebuffer[WIDTH];
|
|
|
|
char *Disks[2][MAXDISKS+1]; /* Disk names for each drive */
|
|
|
|
|
|
void emu_KeyboardOnDown(int keymodifer, int key) {
|
|
if (key <= 0x7f) iusbhk = key;
|
|
else
|
|
iusbhk = 0;
|
|
}
|
|
|
|
void emu_KeyboardOnUp(int keymodifer, int key) {
|
|
iusbhk = 0;
|
|
}
|
|
|
|
void msx_Init(void)
|
|
{
|
|
/* Clear everything */
|
|
// CartCount=TypeCount=JoyCount=0;
|
|
// DiskCount[0]=DiskCount[1]=0;
|
|
|
|
/* Default disk images */
|
|
Disks[0][1]=Disks[1][1]=0;
|
|
Disks[0][0]=DiskA;
|
|
Disks[1][0]=DiskB;
|
|
/* Terminate disk lists and set initial disk names */
|
|
// if(DiskCount[0]) { Disks[0][DiskCount[0]]=0;DiskA=Disks[0][0]; }
|
|
// if(DiskCount[1]) { Disks[1][DiskCount[1]]=0;DiskB=Disks[1][0]; }
|
|
|
|
/* Start fMSX! */
|
|
// if(!InitMachine()) return(1);
|
|
// StartMSX();
|
|
// TrashMSX();
|
|
// TrashMachine();
|
|
emu_printf("Allocating MEM done");
|
|
}
|
|
|
|
|
|
void msx_Input(int click) {
|
|
ihk = emu_ReadI2CKeyboard();
|
|
ik = emu_ReadKeys();
|
|
}
|
|
|
|
|
|
|
|
|
|
void msx_Start(char * Cartridge)
|
|
{
|
|
emu_printf("init started");
|
|
|
|
|
|
|
|
/*** MSX versions: ***/
|
|
static const char *Versions[] = { "MSX","MSX2","MSX2+" };
|
|
|
|
/*** Joystick types: ***/
|
|
static const char *JoyTypes[] =
|
|
{
|
|
"nothing","normal joystick",
|
|
"mouse in joystick mode","mouse in real mode"
|
|
};
|
|
|
|
/*** CMOS ROM default values: ***/
|
|
static const byte RTCInit[4][13] =
|
|
{
|
|
{ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
|
|
{ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
|
|
{ 0, 0, 0, 0,40,80,15, 4, 4, 0, 0, 0, 0 },
|
|
{ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }
|
|
};
|
|
|
|
/*** VDP status register states: ***/
|
|
static const byte VDPSInit[16] = { 0x9F,0,0x6C,0,0,0,0,0,0,0,0,0,0,0,0,0 };
|
|
|
|
/*** VDP control register states: ***/
|
|
static const byte VDPInit[64] =
|
|
{
|
|
0x00,0x10,0xFF,0xFF,0xFF,0xFF,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
|
|
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00
|
|
};
|
|
|
|
/*** Initial palette: ***/
|
|
static const byte PalInit[16][3] =
|
|
{
|
|
{0x00,0x00,0x00},{0x00,0x00,0x00},{0x20,0xC0,0x20},{0x60,0xE0,0x60},
|
|
{0x20,0x20,0xE0},{0x40,0x60,0xE0},{0xA0,0x20,0x20},{0x40,0xC0,0xE0},
|
|
{0xE0,0x20,0x20},{0xE0,0x60,0x60},{0xC0,0xC0,0x20},{0xC0,0xC0,0x80},
|
|
{0x20,0x80,0x20},{0xC0,0x40,0xA0},{0xA0,0xA0,0xA0},{0xE0,0xE0,0xE0}
|
|
};
|
|
|
|
int *T,I,J,K;
|
|
byte *P;
|
|
word A;
|
|
//FILE *F;
|
|
|
|
#ifdef HAS_SND
|
|
emu_sndInit();
|
|
InitSound(22050,0);
|
|
#endif
|
|
/* Zero everyting */
|
|
#ifdef unused
|
|
CasStream=PrnStream=ComIStream=ComOStream=NULL;
|
|
#endif
|
|
ROMData[0]=ROMData[1]=NULL;
|
|
FontBuf = NULL;
|
|
VRAM = NULL;
|
|
SRAM = NULL;
|
|
Kanji = NULL;
|
|
SaveCMOS = 0;
|
|
SaveSRAM = 0;
|
|
ExitNow = 0;
|
|
CCount = 0;
|
|
|
|
/* Calculate IPeriod from VPeriod/HPeriod */
|
|
if(UPeriod<1) UPeriod=1;
|
|
if(HPeriod<CPU_HPERIOD) HPeriod=CPU_HPERIOD;
|
|
if(VPeriod<CPU_VPERIOD) VPeriod=CPU_VPERIOD;
|
|
CPU.TrapBadOps = Verbose&0x10;
|
|
CPU.IPeriod = CPU_H240;
|
|
CPU.IAutoReset = 0;
|
|
|
|
/* Check parameters for validity */
|
|
if(MSXVersion>2) MSXVersion=2;
|
|
if((RAMPages<(MSXVersion? 8:4))||(RAMPages>256))
|
|
RAMPages=MSXVersion? 8:4;
|
|
//RAMPages=MSXVersion? 4:4; // PATCH PICO
|
|
if((VRAMPages<(MSXVersion? 8:2))||(VRAMPages>8))
|
|
VRAMPages=MSXVersion? 8:2;
|
|
|
|
/* Number of RAM pages should be power of 2 */
|
|
/* Calculate RAMMask=(2^RAMPages)-1 */
|
|
for(J=1;J<RAMPages;J<<=1);
|
|
RAMPages=J;
|
|
RAMMask=J-1;
|
|
|
|
/* Number of VRAM pages should be a power of 2 */
|
|
for(J=1;J<VRAMPages;J<<=1);
|
|
VRAMPages=J;
|
|
|
|
/* Initialize ROMMasks to zeroes for now */
|
|
ROMMask[0]=ROMMask[1]=0;
|
|
|
|
/* Joystick types are in 0..3 range */
|
|
JoyTypeA&=3;
|
|
JoyTypeB&=3;
|
|
|
|
/* Allocate 16kB for the empty space (scratch RAM) */
|
|
if(Verbose)
|
|
emu_printf("Allocating 16kB for empty space...");
|
|
EmptyRAM=emu_Malloc(0x4000);
|
|
memset(EmptyRAM,NORAM,0x4000);
|
|
Chunks[CCount++]=EmptyRAM;
|
|
|
|
/* Reset memory map to the empty space */
|
|
for(I=0;I<4;I++)
|
|
for(J=0;J<4;J++)
|
|
for(K=0;K<8;K++)
|
|
MemMap[I][J][K]=EmptyRAM;
|
|
|
|
/* Allocate VRAMPages*16kB for VRAM */
|
|
/* 128k */
|
|
if(Verbose) {
|
|
emu_printi(VRAMPages*16);
|
|
emu_printf("kB for VRAM...");
|
|
}
|
|
VRAM=emu_Malloc(VRAMPages*0x4000);
|
|
memset(VRAM,0x00,VRAMPages*0x4000);
|
|
Chunks[CCount++]=VRAM;
|
|
|
|
/* Allocate RAMPages*16kB for RAM */
|
|
if(Verbose) {
|
|
emu_printi(RAMPages);
|
|
emu_printf("x16kB RAM pages...");
|
|
}
|
|
RAMData=emu_Malloc(RAMPages*0x4000);
|
|
memset(RAMData,NORAM,RAMPages*0x4000);
|
|
Chunks[CCount++]=RAMData;
|
|
|
|
|
|
if(Verbose) {
|
|
emu_printf("Loading ROMs for");
|
|
emu_printf(Versions[MSXVersion]);
|
|
}
|
|
|
|
|
|
/* Open/load system ROM(s) */
|
|
switch(MSXVersion)
|
|
{
|
|
case 0:
|
|
if(Verbose) emu_printf("Opening MSX.ROM...");
|
|
//P=LoadROM("msx/roms/MSX.ROM",0x8000,0);
|
|
P=&MSX[0];
|
|
if(!P) return(0);
|
|
MemMap[0][0][0]=P;
|
|
MemMap[0][0][1]=P+0x2000;
|
|
MemMap[0][0][2]=P+0x4000;
|
|
MemMap[0][0][3]=P+0x6000;
|
|
break;
|
|
|
|
case 1:
|
|
if(Verbose) emu_printf("Opening MSX2.ROM...");
|
|
//P=LoadROM("msx/roms/MSX2.ROM",0x8000,0);
|
|
P=&MSX2[0];
|
|
if(!P) return(0);
|
|
MemMap[0][0][0]=P;
|
|
MemMap[0][0][1]=P+0x2000;
|
|
MemMap[0][0][2]=P+0x4000;
|
|
MemMap[0][0][3]=P+0x6000;
|
|
if(Verbose) emu_printf("Opening MSX2EXT.ROM...");
|
|
//P=LoadROM("msx/roms/MSX2EXT.ROM",0x4000,0);
|
|
P=&MSX2EXT[0];
|
|
if(!P) return(0);
|
|
MemMap[3][1][0]=P;
|
|
MemMap[3][1][1]=P+0x2000;
|
|
break;
|
|
|
|
case 2:
|
|
if(Verbose) emu_printf("Opening MSX2P.ROM...");
|
|
//P=LoadROM("msx/roms/MSX2P.ROM",0x8000,0);
|
|
P=&MSX2P[0];
|
|
if(!P) return(0);
|
|
MemMap[0][0][0]=P;
|
|
MemMap[0][0][1]=P+0x2000;
|
|
MemMap[0][0][2]=P+0x4000;
|
|
MemMap[0][0][3]=P+0x6000;
|
|
if(Verbose) emu_printf("Opening MSX2PEXT.ROM...");
|
|
//P=LoadROM("msx/roms/MSX2PEXT.ROM",0x4000,0);
|
|
P=&MSX2PEXT[0];
|
|
if(!P) return(0);
|
|
MemMap[3][1][0]=P;
|
|
MemMap[3][1][1]=P+0x2000;
|
|
break;
|
|
}
|
|
|
|
/* Try loading DiskROM */
|
|
//P=LoadROM("msx/roms/DISK.ROM",0x4000,0);
|
|
P=&DISK[0];
|
|
if(P)
|
|
{
|
|
if(Verbose) emu_printf("Opening DISK.ROM...OK");
|
|
MemMap[3][1][2]=P;
|
|
MemMap[3][1][3]=P+0x2000;
|
|
}
|
|
|
|
/*
|
|
// Apply patches to BIOS
|
|
if(Verbose) emu_printf("Patching BIOS: ");
|
|
for(J=0;BIOSPatches[J];J++)
|
|
{
|
|
P=MemMap[0][0][0]+BIOSPatches[J];
|
|
P[0]=0xED;P[1]=0xFE;P[2]=0xC9;
|
|
}
|
|
if(Verbose) emu_printf("OK\n");
|
|
|
|
// Apply patches to BDOS
|
|
if(MemMap[3][1][2]!=EmptyRAM)
|
|
{
|
|
if(Verbose) emu_printf("Patching BDOS: ");
|
|
for(J=0;DiskPatches[J];J++)
|
|
{
|
|
P=MemMap[3][1][2]+DiskPatches[J]-0x4000;
|
|
P[0]=0xED;P[1]=0xFE;P[2]=0xC9;
|
|
}
|
|
if(Verbose) emu_printf("OK\n");
|
|
}
|
|
*/
|
|
if(Verbose) emu_printf("Loading other ROMs: ");
|
|
|
|
|
|
/* Try loading CMOS memory contents */
|
|
/*
|
|
if(LoadROM("msx/roms/CMOS.ROM",sizeof(RTC),(byte *)RTC))
|
|
{
|
|
if(Verbose) emu_printf("CMOS.ROM..");
|
|
}
|
|
else memcpy(RTC,RTCInit,sizeof(RTC));
|
|
*/
|
|
|
|
/* Try loading Kanji alphabet ROM */
|
|
/*
|
|
if(Kanji=LoadROM("msx/roms/KANJI.ROM",0x20000,0))
|
|
{
|
|
if(Verbose) emu_printf("KANJI.ROM..");
|
|
}
|
|
*/
|
|
|
|
/* Try loading RS232 support ROM */
|
|
//P=LoadROM("msx/roms/RS232.ROM..",0x4000,0);
|
|
/*
|
|
P=&RS232[0];
|
|
if(P)
|
|
{
|
|
if(Verbose) emu_printf("RS232.ROM..");
|
|
MemMap[3][0][2]=P;
|
|
MemMap[3][0][3]=P+0x2000;
|
|
}
|
|
*/
|
|
|
|
/* Try loading FM-PAC support ROM */
|
|
//P=LoadROM("msx/roms/FMPAC.ROM",0x4000,0);
|
|
P=&FMPAC[0];
|
|
if(P)
|
|
{
|
|
if(Verbose) emu_printf("FMPAC.ROM..");
|
|
MemMap[3][3][2]=P;
|
|
MemMap[3][3][3]=P+0x2000;
|
|
}
|
|
|
|
if(Verbose) emu_printf("OK\n");
|
|
|
|
|
|
|
|
|
|
/* Loading cartridge into slot A... */
|
|
strcpy(CartA, "CARTA.ROM");
|
|
if (Cartridge != NULL)
|
|
strcpy(CartA, Cartridge);
|
|
|
|
LoadCart(CartA,0);
|
|
J=(ROMMask[0]+1)*8192;
|
|
|
|
|
|
/* Guess mapper is not given */
|
|
if((ROMTypeA>=MAXMAPPERS)&&(J>0x8000))
|
|
{
|
|
ROMTypeA=GuessROM(ROMData[0],J);
|
|
if(Verbose) {
|
|
emu_printf("Cartridge A: Guessed");
|
|
emu_printf(ROMNames[ROMTypeA]);
|
|
}
|
|
}
|
|
|
|
/* For Generic/16kB carts, set ROM pages as 0:1:N-2:N-1 */
|
|
if((ROMTypeA==1)&&(J>0x8000))
|
|
SetMegaROM(0,0,1,ROMMask[0]-1,ROMMask[0]);
|
|
|
|
/* Loading cartridge into slot B... */
|
|
LoadCart(CartB,1);
|
|
J=(ROMMask[1]+1)*8192;
|
|
|
|
/* Guess mapper is not given */
|
|
if((ROMTypeB>=MAXMAPPERS)&&(J>0x8000))
|
|
{
|
|
ROMTypeB=GuessROM(ROMData[1],J);
|
|
if(Verbose) {
|
|
emu_printf("Cartridge B: Guessed");
|
|
emu_printf(ROMNames[ROMTypeB]);
|
|
}
|
|
}
|
|
|
|
|
|
/* For Generic/16kB carts, set ROM pages as 0:1:N-2:N-1 */
|
|
if((ROMTypeB==1)&&(J>0x8000))
|
|
SetMegaROM(1,0,1,ROMMask[1]-1,ROMMask[1]);
|
|
|
|
/* For GameMaster2, allocate and load SRAM */
|
|
if((ROMTypeA==6)||(ROMTypeB==6))
|
|
{
|
|
if(Verbose) emu_printf("Allocating 16kB for GameMaster2 SRAM...");
|
|
SRAM=emu_Malloc(0x4000);
|
|
/* Erase SRAM and add it to the list of mallocs */
|
|
memset(SRAM,NORAM,0x4000);
|
|
Chunks[CCount++]=SRAM;
|
|
#ifdef unused
|
|
/* Try loading SRAM data form disk */
|
|
if(!(F=fopen("GMASTER2.RAM","rb"))) {
|
|
emu_printf("failed opening");
|
|
}
|
|
else
|
|
{
|
|
if(Verbose) emu_printf("loading GMASTER2.RAM...");
|
|
if(fread(SRAM,1,0x2000,F)!=0x2000)
|
|
{
|
|
emu_printf("failed");
|
|
}
|
|
else
|
|
{
|
|
/* Mirror GameMaster2 SRAM as needed */
|
|
memcpy(SRAM+0x2000,SRAM+0x1000,0x1000);
|
|
memcpy(SRAM+0x3000,SRAM+0x1000,0x1000);
|
|
memcpy(SRAM+0x1000,SRAM,0x1000);
|
|
}
|
|
/* Done with the file */
|
|
fclose(F);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/* Load MSX2-dependent cartridge ROMs: MSXDOS2 and PAINTER */
|
|
if(MSXVersion>0)
|
|
{
|
|
/* Find an empty slot */
|
|
if(MemMap[1][0][2]==EmptyRAM) J=1;
|
|
else if(MemMap[2][0][2]==EmptyRAM) J=2;
|
|
else J=0;
|
|
|
|
/* Try loading MSXDOS2 cartridge, if slot */
|
|
/* found and DiskROM present */
|
|
if(J&&(MemMap[3][1][2]!=EmptyRAM))
|
|
{
|
|
if(J==2) ROMTypeB=1; else ROMTypeA=1;
|
|
if(LoadCart("MSXDOS2.ROM",J-1))
|
|
SetMegaROM(J-1,0,1,ROMMask[J-1]-1,ROMMask[J-1]);
|
|
}
|
|
|
|
/* Find an empty slot */
|
|
if(MemMap[1][0][2]==EmptyRAM) J=1;
|
|
else if(MemMap[2][0][2]==EmptyRAM) J=2;
|
|
else J=0;
|
|
|
|
/* Try loading PAINTER ROM if slot found */
|
|
if(J) LoadCart("PAINTER.ROM",J-1);
|
|
}
|
|
|
|
/* Try loading font */
|
|
if(Verbose) {
|
|
emu_printf("Loading font...");
|
|
emu_printf(FontName);
|
|
}
|
|
//FontBuf=LoadROM(FontName,0x800,0);
|
|
|
|
#ifdef unused
|
|
/* Open stream for a printer */
|
|
if(!PrnName) PrnStream=stdout;
|
|
else
|
|
{
|
|
if(Verbose) printf("Redirecting printer output to %s...",PrnName);
|
|
if(!(PrnStream=fopen(PrnName,"wb"))) PrnStream=stdout;
|
|
}
|
|
#endif
|
|
|
|
#ifdef unused
|
|
/* Open streams for serial IO */
|
|
if(!ComName) { ComIStream=stdin;ComOStream=stdout; }
|
|
else
|
|
{
|
|
if(Verbose) printf("Redirecting serial I/O to %s...",ComName);
|
|
if(!(ComOStream=ComIStream=fopen(ComName,"r+b")))
|
|
{ ComIStream=stdin;ComOStream=stdout; }
|
|
}
|
|
#endif
|
|
|
|
/* Open disk images */
|
|
#ifdef DISK
|
|
if(ChangeDisk(0,DiskA))
|
|
if(Verbose) {
|
|
emu_printf("Inserting floppy into drive A");
|
|
emu_printf(DiskA);
|
|
}
|
|
if(ChangeDisk(1,DiskB))
|
|
if(Verbose) {
|
|
emu_printf("Inserting floppy into drive B");
|
|
emu_printf(DiskB);
|
|
}
|
|
#endif /* DISK */
|
|
|
|
#ifdef unused
|
|
/* Open casette image */
|
|
if(CasName)
|
|
if(CasStream=fopen(CasName,"r+b"))
|
|
if(Verbose) {
|
|
emu_printf("Inserting tape");
|
|
emu_printf(CasName);
|
|
}
|
|
#endif
|
|
|
|
if(Verbose)
|
|
{
|
|
emu_printf("Attaching joystick port A");
|
|
emu_printf(JoyTypes[JoyTypeA]);
|
|
emu_printf("Attaching joystick port B");
|
|
emu_printf(JoyTypes[JoyTypeB]);
|
|
emu_printf("Initializing memory mappers...");
|
|
}
|
|
|
|
for(J=0;J<4;J++)
|
|
{
|
|
EnWrite[J]=0; /* Write protect ON for all slots */
|
|
PSL[J]=SSL[J]=0; /* PSL=0:0:0:0, SSL=0:0:0:0 */
|
|
MemMap[3][2][J*2] = RAMData+(3-J)*0x4000; /* RAMMap=3:2:1:0 */
|
|
MemMap[3][2][J*2+1] = MemMap[3][2][J*2]+0x2000;
|
|
RAMMapper[J] = 3-J;
|
|
RAM[J*2] = MemMap[0][0][J*2]; /* Setting RAM */
|
|
RAM[J*2+1] = MemMap[0][0][J*2+1];
|
|
}
|
|
|
|
if(Verbose) emu_printf("Initializing VDP, PSG, OPLL, SCC, and CPU...");
|
|
|
|
/* Initialize palette */
|
|
for(J=0;J<16;J++)
|
|
{
|
|
Palette[J]=RGB2INT(PalInit[J][0],PalInit[J][1],PalInit[J][2]);
|
|
SetColor(J,PalInit[J][0],PalInit[J][1],PalInit[J][2]);
|
|
}
|
|
|
|
/* Reset mouse coordinates/counters */
|
|
for(J=0;J<2;J++)
|
|
Buttons[J]=MouseDX[J]=MouseDY[J]=OldMouseX[J]=OldMouseY[J]=MCount[J]=0;
|
|
|
|
/* Initialize sound logging */
|
|
//InitMIDI("SndName");
|
|
|
|
/* Reset sound chips */
|
|
Reset8910(&PSG,0);
|
|
ResetSCC(&SCChip,AY8910_CHANNELS);
|
|
Reset2413(&OPLL,AY8910_CHANNELS);
|
|
Sync8910(&PSG,AY8910_SYNC);
|
|
SyncSCC(&SCChip,SCC_SYNC);
|
|
Sync2413(&OPLL,YM2413_SYNC);
|
|
|
|
/* Reset serial I/O */
|
|
#ifdef unused
|
|
Reset8251(&SIO,ComIStream,ComOStream);
|
|
#else
|
|
Reset8251(&SIO,NULL,NULL);
|
|
#endif
|
|
/* Reset PPI chips and slot selectors */
|
|
Reset8255(&PPI);
|
|
PPI.Rout[0]=PSLReg=0x00;
|
|
PPI.Rout[2]=IOReg=0x00;
|
|
SSLReg=0x00;
|
|
|
|
memcpy(VDP,VDPInit,sizeof(VDP));
|
|
memcpy(VDPStatus,VDPSInit,sizeof(VDPStatus));
|
|
memset(KeyMap,0xFF,16); /* Keyboard */
|
|
IRQPending=0x00; /* No IRQs pending */
|
|
SCCOn[0]=SCCOn[1]=0; /* SCCs off for now */
|
|
RTCReg=RTCMode=0; /* Clock registers */
|
|
KanCount=0;KanLetter=0; /* Kanji extension */
|
|
ChrTab=ColTab=ChrGen=VRAM; /* VDP tables */
|
|
SprTab=SprGen=VRAM;
|
|
ChrTabM=ColTabM=ChrGenM=SprTabM=~0; /* VDP addr. masks */
|
|
VPAGE=VRAM; /* VRAM page */
|
|
FGColor=BGColor=XFGColor=XBGColor=0; /* VDP colors */
|
|
ScrMode=0; /* Screen mode */
|
|
VKey=PKey=1;WKey=0; /* VDP keys */
|
|
VAddr=0x0000; /* VRAM access addr */
|
|
ScanLine=0; /* Current scanline */
|
|
VDPData=NORAM; /* VDP data buffer */
|
|
UseFont=0; /* Extern. font use */
|
|
|
|
/* Set "V9958" VDP version for MSX2+ */
|
|
if(MSXVersion>=2) VDPStatus[1]|=0x04;
|
|
|
|
/* Reset CPU */
|
|
ResetZ80(&CPU);
|
|
|
|
/* Done with initialization */
|
|
if(Verbose)
|
|
{
|
|
emu_printi(HPeriod);
|
|
emu_printf("CPU cycles per HBlank");
|
|
emu_printi(VPeriod);
|
|
emu_printf("CPU cycles per VBlank");
|
|
emu_printi(VPeriod/HPeriod);
|
|
emu_printf("scanlines");
|
|
}
|
|
|
|
|
|
/* Start execution of the code */
|
|
if(Verbose) emu_printf("RUNNING ROM CODE...\n");
|
|
A=RunZ80(&CPU);
|
|
|
|
|
|
emu_printf("init done");
|
|
}
|
|
|
|
#define INV_KEY {255,255}
|
|
|
|
struct { byte Pos,Mask; } Keys[] =
|
|
{
|
|
/* 0x00 */ INV_KEY,
|
|
/* 0x01 */ INV_KEY,
|
|
/* 0x02 */ INV_KEY,
|
|
/* 0x03 */ INV_KEY,
|
|
/* 0x04 */ INV_KEY,
|
|
/* 0x05 */ INV_KEY,
|
|
/* 0x06 */ INV_KEY,
|
|
/* 0x07 */ INV_KEY,
|
|
/* 0x08 */ INV_KEY,
|
|
/* 0x09 */ INV_KEY, // tab
|
|
/* 0x0A */ INV_KEY,
|
|
/* 0x0B */ INV_KEY,
|
|
/* 0x0C */ INV_KEY,
|
|
/* 0x0D */ {7,0x80}, // enter
|
|
/* 0x0E */ INV_KEY,
|
|
/* 0x0F */ INV_KEY,
|
|
/* 0x10 */ INV_KEY,
|
|
/* 0x11 */ INV_KEY,
|
|
/* 0x12 */ INV_KEY,
|
|
/* 0x13 */ INV_KEY,
|
|
/* 0x14 */ INV_KEY,
|
|
/* 0x15 */ INV_KEY,
|
|
/* 0x16 */ INV_KEY,
|
|
/* 0x17 */ INV_KEY,
|
|
/* 0x18 */ INV_KEY,
|
|
/* 0x19 */ INV_KEY,
|
|
/* 0x1A */ INV_KEY,
|
|
/* 0x1B */ INV_KEY, // esc
|
|
/* 0x1C */ INV_KEY,
|
|
/* 0x1D */ INV_KEY,
|
|
/* 0x1E */ INV_KEY,
|
|
/* 0x1F */ INV_KEY,
|
|
|
|
/* 0x20 */ {8,0x01}, // space
|
|
/* 0x21 */ INV_KEY, // ! exclamation mark
|
|
/* 0x22 */ INV_KEY, // " double quote
|
|
/* 0x23 */ INV_KEY, // # dies
|
|
/* 0x24 */ INV_KEY, // $ dollar
|
|
/* 0x25 */ INV_KEY, // % percent
|
|
/* 0x26 */ INV_KEY, // & ampercent
|
|
/* 0x27 */ INV_KEY, // ' singlequote
|
|
/* 0x28 */ INV_KEY, // ( bracket left
|
|
/* 0x29 */ INV_KEY, // ) bracket right
|
|
/* 0x2A */ INV_KEY, // * mult
|
|
/* 0x2B */ INV_KEY, // + plus
|
|
/* 0x2C */ {2,0x04}, // , comma
|
|
/* 0x2D */ INV_KEY, // - minus
|
|
/* 0x2E */ INV_KEY, // . period
|
|
/* 0x2F */ INV_KEY, // / slash
|
|
|
|
/* 0x30 */ {0,0x01}, //'0'
|
|
/* 0x31 */ {0,0x02}, //'1'
|
|
/* 0x32 */ {0,0x04}, //'2'
|
|
/* 0x33 */ {0,0x08}, //'3'
|
|
/* 0x34 */ {0,0x10}, //'4'
|
|
/* 0x35 */ {0,0x20}, //'5'
|
|
/* 0x36 */ {0,0x40}, //'6'
|
|
/* 0x37 */ {0,0x80}, //'7'
|
|
/* 0x38 */ {1,0x01}, //'8'
|
|
/* 0x39 */ {1,0x02}, //'9'
|
|
|
|
/* 0x3A */ INV_KEY, // : colon
|
|
/* 0x3B */ INV_KEY, // ; semi colon
|
|
/* 0x3C */ INV_KEY, // <
|
|
/* 0x3D */ INV_KEY, // = equal
|
|
/* 0x3E */ INV_KEY, // >
|
|
/* 0x3F */ INV_KEY, // ?
|
|
/* 0x40 */ INV_KEY, // @
|
|
|
|
/* 0x41 */ {2,0x40}, //'A'
|
|
/* 0x42 */ {2,0x80}, //'B'
|
|
/* 0x43 */ {3,0x01}, //'C'
|
|
/* 0x44 */ {3,0x02}, //'D'
|
|
/* 0x45 */ {3,0x04}, //'E'
|
|
/* 0x46 */ {3,0x08}, //'F'
|
|
/* 0x47 */ {3,0x10}, //'G'
|
|
/* 0x48 */ {3,0x20}, //'H'
|
|
/* 0x49 */ {3,0x40}, //'I'
|
|
/* 0x4A */ {3,0x80}, //'J'
|
|
/* 0x4B */ {4,0x01}, //'K'
|
|
/* 0x4C */ {4,0x02}, //'L'
|
|
/* 0x4D */ {4,0x04}, //'M'
|
|
/* 0x4E */ {4,0x08}, //'N'
|
|
/* 0x4F */ {4,0x10}, //'O'
|
|
/* 0x50 */ {4,0x20}, //'P'
|
|
/* 0x51 */ {4,0x40}, //'Q'
|
|
/* 0x52 */ {4,0x80}, //'R'
|
|
/* 0x53 */ {5,0x01}, //'S'
|
|
/* 0x54 */ {5,0x02}, //'T'
|
|
/* 0x55 */ {5,0x04}, //'U'
|
|
/* 0x56 */ {5,0x08}, //'V'
|
|
/* 0x57 */ {5,0x10}, //'W'
|
|
/* 0x58 */ {5,0x20}, //'X'
|
|
/* 0x59 */ {5,0x40}, //'Y'
|
|
/* 0x5A */ {5,0x80}, //'Z'
|
|
|
|
/* 0x5B */ INV_KEY, // square bracket open
|
|
/* 0x5C */ INV_KEY, // backslach
|
|
/* 0x5D */ INV_KEY, // square braquet close
|
|
/* 0x5E */ INV_KEY, // ^ circonflex
|
|
/* 0x5F */ INV_KEY, // _ undescore
|
|
/* 0x60 */ INV_KEY, // `backquote
|
|
|
|
/* 0x61 */ {2,0x40}, //'a'
|
|
/* 0x62 */ {2,0x80}, //'b'
|
|
/* 0x63 */ {3,0x01}, //'c'
|
|
/* 0x64 */ {3,0x02}, //'d'
|
|
/* 0x65 */ {3,0x04}, //'e'
|
|
/* 0x66 */ {3,0x08}, //'f'
|
|
/* 0x67 */ {3,0x10}, //'g'
|
|
/* 0x68 */ {3,0x20}, //'h'
|
|
/* 0x69 */ {3,0x40}, //'i'
|
|
/* 0x6A */ {3,0x80}, //'j'
|
|
/* 0x6B */ {4,0x01}, //'k'
|
|
/* 0x6C */ {4,0x02}, //'l'
|
|
/* 0x6D */ {4,0x04}, //'m'
|
|
/* 0x6E */ {4,0x08}, //'n'
|
|
/* 0x6F */ {4,0x10}, //'o'
|
|
/* 0x70 */ {4,0x20}, //'p'
|
|
/* 0x71 */ {4,0x40}, //'q'
|
|
/* 0x72 */ {4,0x80}, //'r'
|
|
/* 0x73 */ {5,0x01}, //'s'
|
|
/* 0x74 */ {5,0x02}, //'t'
|
|
/* 0x75 */ {5,0x04}, //'u'
|
|
/* 0x76 */ {5,0x08}, //'v'
|
|
/* 0x77 */ {5,0x10}, //'w'
|
|
/* 0x78 */ {5,0x20}, //'x'
|
|
/* 0x79 */ {5,0x40}, //'y'
|
|
/* 0x7A */ {5,0x80}, //'z'
|
|
|
|
/* 0x7B */ INV_KEY, // curly bracket open
|
|
/* 0x7C */ INV_KEY, // or
|
|
/* 0x7D */ INV_KEY, // curly bracket close
|
|
/* 0x7E */ INV_KEY, // tilt
|
|
/* 0x7F */ {7,0x20}, // backspace
|
|
|
|
/* 0xC0 */ INV_KEY,
|
|
/* 0xC1 */ INV_KEY,
|
|
/* 0xC2 */ INV_KEY, // F1
|
|
/* 0xC3 */ INV_KEY, // F2
|
|
/* 0xC4 */ INV_KEY, // F3
|
|
/* 0xC5 */ INV_KEY, // F4
|
|
/* 0xC6 */ INV_KEY, // F5
|
|
/* 0xC7 */ INV_KEY, // F6
|
|
/* 0xC8 */ INV_KEY, // F7
|
|
/* 0xC9 */ INV_KEY, // F8
|
|
/* 0xCA */ INV_KEY, // F9
|
|
/* 0xCB */ INV_KEY, // F10
|
|
/* 0xCC */ INV_KEY,
|
|
/* 0xCD */ INV_KEY,
|
|
/* 0xCE */ INV_KEY,
|
|
/* 0xCF */ INV_KEY,
|
|
/* 0xD0 */ INV_KEY,
|
|
/* 0xD1 */ INV_KEY,
|
|
/* 0xD2 */ INV_KEY,
|
|
/* 0xD3 */ INV_KEY,
|
|
/* 0xD4 */ INV_KEY, // DEL
|
|
/* 0xD5 */ INV_KEY,
|
|
/* 0xD6 */ INV_KEY,
|
|
/* 0xD7 */ INV_KEY, // U
|
|
/* 0xD8 */ INV_KEY, // L
|
|
/* 0xD9 */ INV_KEY, // R
|
|
/* 0xDA */ INV_KEY, // D
|
|
/* 0xDB */ INV_KEY,
|
|
/* 0xDC */ INV_KEY,
|
|
/* 0xDD */ INV_KEY,
|
|
/* 0xDE */ INV_KEY,
|
|
/* 0xDF */ INV_KEY
|
|
};
|
|
|
|
|
|
void msx_Step(void) {
|
|
int k = ik;
|
|
int hk = ihk;
|
|
if (iusbhk) hk = iusbhk;
|
|
|
|
JoyState = 0;
|
|
if (k & MASK_JOY2_DOWN) JoyState|=0x02;
|
|
if (k & MASK_JOY2_UP) JoyState|=0x01;
|
|
if (k & MASK_JOY2_RIGHT) JoyState|=0x04;
|
|
if (k & MASK_JOY2_LEFT) JoyState|=0x08;
|
|
if (k & MASK_JOY2_BTN) JoyState|=0x10;
|
|
if (k & MASK_KEY_USER2) JoyState|=0x20;
|
|
|
|
if (hk != 0) {
|
|
//emu_printh(hk);
|
|
KeyMap[Keys[hk].Pos] &=~ Keys[hk].Mask;
|
|
}
|
|
else {
|
|
memset(KeyMap,0xFF,16);
|
|
}
|
|
|
|
RunZ80(&CPU);
|
|
|
|
emu_DrawVsync();
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef HAS_SND
|
|
static int wave[256];
|
|
#endif
|
|
|
|
void SND_Process(void *stream, int len) {
|
|
#ifdef HAS_SND
|
|
audio_sample * snd_buf = (audio_sample *)stream;
|
|
memset(wave,0,256*sizeof(wave[0]));
|
|
RenderAudio(&wave[0], len);
|
|
for (int i = 0; i< len; i++ )
|
|
*snd_buf++ = (wave[i]/16+32767)>>8;
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void SetColor(byte N,byte R,byte G,byte B)
|
|
{
|
|
XPal[N] = N;
|
|
emu_SetPaletteEntry(R,G,B,N);
|
|
}
|
|
|
|
void Keyboard(void)
|
|
{
|
|
}
|
|
|
|
byte Joystick(register byte N) { return(JoyState); }
|
|
|
|
/** Mouse() **************************************************/
|
|
/** Query coordinates of a mouse connected to port N. **/
|
|
/** Returns F2.F1.Y.Y.Y.Y.Y.Y.Y.Y.X.X.X.X.X.X.X.X. **/
|
|
/*************************************************************/
|
|
int Mouse(register byte N)
|
|
{
|
|
return(0);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** ClearLine() **********************************************/
|
|
/** Clear 256 pixels from P with color C. **/
|
|
/*************************************************************/
|
|
static void ClearLine(register pixel *P,register pixel C)
|
|
{
|
|
register int J;
|
|
|
|
for(J=0;J<256;J++) P[J]=C;
|
|
}
|
|
|
|
/** YJKColor() ***********************************************/
|
|
/** Given a color in YJK format, return the corresponding **/
|
|
/** palette entry. **/
|
|
/*************************************************************/
|
|
static pixel YJKColor(register int Y,register int J,register int K)
|
|
{
|
|
register int R,G,B;
|
|
|
|
R=Y+J;
|
|
G=Y+K;
|
|
B=((5*Y-2*J-K)/4);
|
|
|
|
R=R<0? 0:R>31? 31:R;
|
|
G=G<0? 0:G>31? 31:G;
|
|
B=B<0? 0:B>31? 31:B;
|
|
|
|
return(BPal[(R&0x1C)|((G&0x1C)<<3)|(B>>3)]);
|
|
}
|
|
|
|
/** RefreshBorder() ******************************************/
|
|
/** This function is called from RefreshLine#() to refresh **/
|
|
/** the screen border. It returns a pointer to the start of **/
|
|
/** scanline Y in FB or 0 if scanline is beyond FB. **/
|
|
/*************************************************************/
|
|
pixel *RefreshBorder(register byte Y,register pixel C)
|
|
{
|
|
register pixel *P;
|
|
register int H;
|
|
|
|
/* First line number in the buffer */
|
|
if(!Y) FirstLine=(ScanLines212? 8:18)+VAdjust;
|
|
|
|
/* Return 0 if we've run out of the screen buffer due to overscan */
|
|
if(Y+FirstLine>=HEIGHT) return(0);
|
|
|
|
/* Set up the transparent color */
|
|
XPal[0]=(!BGColor||SolidColor0)? XPal0:XPal[BGColor];
|
|
|
|
/* Start of the buffer */
|
|
P=(pixel *)&linebuffer[0];
|
|
|
|
/* Paint top of the screen */
|
|
//if(!Y) for(H=WIDTH*FirstLine-1;H>=0;H--) P[H]=C;
|
|
|
|
/* Start of the line */
|
|
//P+=WIDTH*(FirstLine+Y);
|
|
|
|
/* Paint left/right borders */
|
|
for(H=(WIDTH-256)/2+HAdjust;H>0;H--) P[H-1]=C;
|
|
for(H=(WIDTH-256)/2-HAdjust;H>0;H--) P[WIDTH-H]=C;
|
|
|
|
/* Paint bottom of the screen */
|
|
//H=ScanLines212? 212:192;
|
|
//if(Y==H-1) for(H=WIDTH*(HEIGHT-H-FirstLine+1)-1;H>=WIDTH;H--) P[H]=C;
|
|
|
|
/* Return pointer to the scanline in FB */
|
|
return(P+(WIDTH-256)/2+HAdjust);
|
|
}
|
|
|
|
/** Sprites() ************************************************/
|
|
/** This function is called from RefreshLine#() to refresh **/
|
|
/** sprites in SCREENs 1-3. **/
|
|
/*************************************************************/
|
|
void Sprites(register byte Y,register pixel *Line)
|
|
{
|
|
register pixel *P,C;
|
|
register byte H,*PT,*AT;
|
|
register unsigned int M;
|
|
register int L,K;
|
|
|
|
/* Assign initial values before counting */
|
|
H=Sprites16x16? 16:8;
|
|
C=0;M=0;L=0;
|
|
AT=SprTab-4;
|
|
Y+=VScroll;
|
|
|
|
/* Count displayed sprites */
|
|
do
|
|
{
|
|
M<<=1;AT+=4;L++; /* Iterating through SprTab */
|
|
K=AT[0]; /* K = sprite Y coordinate */
|
|
if(K==208) break; /* Iteration terminates if Y=208 */
|
|
if(K>256-H) K-=256; /* Y coordinate may be negative */
|
|
|
|
/* Mark all valid sprites with 1s, break at MAXSPRITE1 sprites */
|
|
if((Y>K)&&(Y<=K+H)) { M|=1;C++;if(C==MAXSPRITE1) break; }
|
|
}
|
|
while(L<32);
|
|
|
|
/* Draw all marked sprites */
|
|
for(;M;M>>=1,AT-=4)
|
|
if(M&1)
|
|
{
|
|
C=AT[3]; /* C = sprite attributes */
|
|
L=C&0x80? AT[1]-32:AT[1]; /* Sprite may be shifted left by 32 */
|
|
C&=0x0F; /* C = sprite color */
|
|
|
|
if((L<256)&&(L>-H)&&C)
|
|
{
|
|
K=AT[0]; /* K = sprite Y coordinate */
|
|
if(K>256-H) K-=256; /* Y coordinate may be negative */
|
|
|
|
P=Line+L;
|
|
PT=SprGen+((int)(H>8? AT[2]&0xFC:AT[2])<<3)+Y-K-1;
|
|
C=XPal[C];
|
|
|
|
/* Mask 1: clip left sprite boundary */
|
|
K=L>=0? 0x0FFFF:(0x10000>>-L)-1;
|
|
/* Mask 2: clip right sprite boundary */
|
|
if(L>256-H) K^=((0x00200>>(H-8))<<(L-257+H))-1;
|
|
/* Get and clip the sprite data */
|
|
K&=((int)PT[0]<<8)|(H>8? PT[16]:0x00);
|
|
|
|
/* Draw left 8 pixels of the sprite */
|
|
if(K&0xFF00)
|
|
{
|
|
if(K&0x8000) P[0]=C;if(K&0x4000) P[1]=C;
|
|
if(K&0x2000) P[2]=C;if(K&0x1000) P[3]=C;
|
|
if(K&0x0800) P[4]=C;if(K&0x0400) P[5]=C;
|
|
if(K&0x0200) P[6]=C;if(K&0x0100) P[7]=C;
|
|
}
|
|
|
|
/* Draw right 8 pixels of the sprite */
|
|
if(K&0x00FF)
|
|
{
|
|
if(K&0x0080) P[8]=C; if(K&0x0040) P[9]=C;
|
|
if(K&0x0020) P[10]=C;if(K&0x0010) P[11]=C;
|
|
if(K&0x0008) P[12]=C;if(K&0x0004) P[13]=C;
|
|
if(K&0x0002) P[14]=C;if(K&0x0001) P[15]=C;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/** ColorSprites() *******************************************/
|
|
/** This function is called from RefreshLine#() to refresh **/
|
|
/** color sprites in SCREENs 4-8. The result is returned in **/
|
|
/** ZBuf, whose size must be 304 bytes (32+256+16). **/
|
|
/*************************************************************/
|
|
void ColorSprites(register byte Y,byte *ZBuf)
|
|
{
|
|
register byte C,H,J,OrThem;
|
|
register byte *P,*PT,*AT;
|
|
register int L,K;
|
|
register unsigned int M;
|
|
|
|
/* Clear ZBuffer and exit if sprites are off */
|
|
memset(ZBuf+32,0,256);
|
|
if(SpritesOFF) return;
|
|
|
|
/* Assign initial values before counting */
|
|
H=Sprites16x16? 16:8;
|
|
C=0;M=0;L=0;
|
|
AT=SprTab-4;
|
|
OrThem=0x00;
|
|
|
|
/* Count displayed sprites */
|
|
do
|
|
{
|
|
M<<=1;AT+=4;L++; /* Iterating through SprTab */
|
|
K=AT[0]; /* Read Y from SprTab */
|
|
if(K==216) break; /* Iteration terminates if Y=216 */
|
|
K=(byte)(K-VScroll); /* Sprite's actual Y coordinate */
|
|
if(K>256-H) K-=256; /* Y coordinate may be negative */
|
|
|
|
/* Mark all valid sprites with 1s, break at MAXSPRITE2 sprites */
|
|
if((Y>K)&&(Y<=K+H)) { M|=1;C++;if(C==MAXSPRITE2) break; }
|
|
}
|
|
while(L<32);
|
|
|
|
/* Draw all marked sprites */
|
|
for(;M;M>>=1,AT-=4)
|
|
if(M&1)
|
|
{
|
|
K=(byte)(AT[0]-VScroll); /* K = sprite Y coordinate */
|
|
if(K>256-H) K-=256; /* Y coordinate may be negative */
|
|
|
|
J=Y-K-1;
|
|
C=SprTab[-0x0200+((AT-SprTab)<<2)+J];
|
|
OrThem|=C&0x40;
|
|
|
|
if(C&0x0F)
|
|
{
|
|
PT=SprGen+((int)(H>8? AT[2]&0xFC:AT[2])<<3)+J;
|
|
P=ZBuf+AT[1]+(C&0x80? 0:32);
|
|
C&=0x0F;
|
|
J=PT[0];
|
|
|
|
if(OrThem&0x20)
|
|
{
|
|
if(J&0x80) P[0]|=C;if(J&0x40) P[1]|=C;
|
|
if(J&0x20) P[2]|=C;if(J&0x10) P[3]|=C;
|
|
if(J&0x08) P[4]|=C;if(J&0x04) P[5]|=C;
|
|
if(J&0x02) P[6]|=C;if(J&0x01) P[7]|=C;
|
|
if(H>8)
|
|
{
|
|
J=PT[16];
|
|
if(J&0x80) P[8]|=C; if(J&0x40) P[9]|=C;
|
|
if(J&0x20) P[10]|=C;if(J&0x10) P[11]|=C;
|
|
if(J&0x08) P[12]|=C;if(J&0x04) P[13]|=C;
|
|
if(J&0x02) P[14]|=C;if(J&0x01) P[15]|=C;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(J&0x80) P[0]=C;if(J&0x40) P[1]=C;
|
|
if(J&0x20) P[2]=C;if(J&0x10) P[3]=C;
|
|
if(J&0x08) P[4]=C;if(J&0x04) P[5]=C;
|
|
if(J&0x02) P[6]=C;if(J&0x01) P[7]=C;
|
|
if(H>8)
|
|
{
|
|
J=PT[16];
|
|
if(J&0x80) P[8]=C; if(J&0x40) P[9]=C;
|
|
if(J&0x20) P[10]=C;if(J&0x10) P[11]=C;
|
|
if(J&0x08) P[12]=C;if(J&0x04) P[13]=C;
|
|
if(J&0x02) P[14]=C;if(J&0x01) P[15]=C;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Update overlapping flag */
|
|
OrThem>>=1;
|
|
}
|
|
}
|
|
|
|
/** RefreshLineF() *******************************************/
|
|
/** Dummy refresh function called for non-existing screens. **/
|
|
/*************************************************************/
|
|
static void RefreshLineF(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
|
|
//if(Verbose>1)
|
|
// printf
|
|
// (
|
|
// "ScrMODE %d: ChrTab=%X ChrGen=%X ColTab=%X SprTab=%X SprGen=%X\n",
|
|
// ScrMode,ChrTab-VRAM,ChrGen-VRAM,ColTab-VRAM,SprTab-VRAM,SprGen-VRAM
|
|
// );
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(P) ClearLine(P,XPal[BGColor]);
|
|
}
|
|
|
|
/** RefreshLine0() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN0. **/
|
|
/*************************************************************/
|
|
static void RefreshLine0(register byte Y)
|
|
{
|
|
register pixel *P,FC,BC;
|
|
register byte X,*T,*G;
|
|
|
|
BC=XPal[BGColor];
|
|
P=RefreshBorder(Y,BC);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,BC);
|
|
else
|
|
{
|
|
P[0]=P[1]=P[2]=P[3]=P[4]=P[5]=P[6]=P[7]=P[8]=BC;
|
|
|
|
G=(UseFont&&FontBuf? FontBuf:ChrGen)+((Y+VScroll)&0x07);
|
|
T=ChrTab+40*(Y>>3);
|
|
FC=XPal[FGColor];
|
|
P+=9;
|
|
|
|
for(X=0;X<40;X++,T++,P+=6)
|
|
{
|
|
Y=G[(int)*T<<3];
|
|
P[0]=Y&0x80? FC:BC;P[1]=Y&0x40? FC:BC;
|
|
P[2]=Y&0x20? FC:BC;P[3]=Y&0x10? FC:BC;
|
|
P[4]=Y&0x08? FC:BC;P[5]=Y&0x04? FC:BC;
|
|
}
|
|
|
|
P[0]=P[1]=P[2]=P[3]=P[4]=P[5]=P[6]=BC;
|
|
}
|
|
}
|
|
|
|
/** RefreshLine1() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN1, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine1(register byte Y)
|
|
{
|
|
register pixel *P,FC,BC;
|
|
register byte K,X,*T,*G;
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor]);
|
|
else
|
|
{
|
|
Y+=VScroll;
|
|
G=(UseFont&&FontBuf? FontBuf:ChrGen)+(Y&0x07);
|
|
T=ChrTab+((int)(Y&0xF8)<<2);
|
|
|
|
for(X=0;X<32;X++,T++,P+=8)
|
|
{
|
|
K=ColTab[*T>>3];
|
|
FC=XPal[K>>4];
|
|
BC=XPal[K&0x0F];
|
|
K=G[(int)*T<<3];
|
|
P[0]=K&0x80? FC:BC;P[1]=K&0x40? FC:BC;
|
|
P[2]=K&0x20? FC:BC;P[3]=K&0x10? FC:BC;
|
|
P[4]=K&0x08? FC:BC;P[5]=K&0x04? FC:BC;
|
|
P[6]=K&0x02? FC:BC;P[7]=K&0x01? FC:BC;
|
|
}
|
|
|
|
if(!SpritesOFF) Sprites(Y,P-256);
|
|
}
|
|
}
|
|
|
|
/** RefreshLine2() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN2, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine2(register byte Y)
|
|
{
|
|
register pixel *P,FC,BC;
|
|
register byte K,X,*T;
|
|
register int I,J;
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor]);
|
|
else
|
|
{
|
|
Y+=VScroll;
|
|
T=ChrTab+((int)(Y&0xF8)<<2);
|
|
I=((int)(Y&0xC0)<<5)+(Y&0x07);
|
|
|
|
for(X=0;X<32;X++,T++,P+=8)
|
|
{
|
|
J=(int)*T<<3;
|
|
K=ColTab[(I+J)&ColTabM];
|
|
FC=XPal[K>>4];
|
|
BC=XPal[K&0x0F];
|
|
K=ChrGen[(I+J)&ChrGenM];
|
|
P[0]=K&0x80? FC:BC;P[1]=K&0x40? FC:BC;
|
|
P[2]=K&0x20? FC:BC;P[3]=K&0x10? FC:BC;
|
|
P[4]=K&0x08? FC:BC;P[5]=K&0x04? FC:BC;
|
|
P[6]=K&0x02? FC:BC;P[7]=K&0x01? FC:BC;
|
|
}
|
|
|
|
if(!SpritesOFF) Sprites(Y,P-256);
|
|
}
|
|
}
|
|
|
|
/** RefreshLine3() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN3, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine3(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
register byte X,K,*T,*G;
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor]);
|
|
else
|
|
{
|
|
Y+=VScroll;
|
|
T=ChrTab+((int)(Y&0xF8)<<2);
|
|
G=ChrGen+((Y&0x1C)>>2);
|
|
|
|
for(X=0;X<32;X++,T++,P+=8)
|
|
{
|
|
K=G[(int)*T<<3];
|
|
P[0]=P[1]=P[2]=P[3]=XPal[K>>4];
|
|
P[4]=P[5]=P[6]=P[7]=XPal[K&0x0F];
|
|
}
|
|
|
|
if(!SpritesOFF) Sprites(Y,P-256);
|
|
}
|
|
}
|
|
|
|
/** RefreshLine4() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN4, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine4(register byte Y)
|
|
{
|
|
register pixel *P,FC,BC;
|
|
register byte K,X,C,*T,*R;
|
|
register int I,J;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
Y+=VScroll;
|
|
T=ChrTab+((int)(Y&0xF8)<<2);
|
|
I=((int)(Y&0xC0)<<5)+(Y&0x07);
|
|
|
|
for(X=0;X<32;X++,R+=8,P+=8,T++)
|
|
{
|
|
J=(int)*T<<3;
|
|
K=ColTab[(I+J)&ColTabM];
|
|
FC=XPal[K>>4];
|
|
BC=XPal[K&0x0F];
|
|
K=ChrGen[(I+J)&ChrGenM];
|
|
|
|
C=R[0];P[0]=C? XPal[C]:(K&0x80)? FC:BC;
|
|
C=R[1];P[1]=C? XPal[C]:(K&0x40)? FC:BC;
|
|
C=R[2];P[2]=C? XPal[C]:(K&0x20)? FC:BC;
|
|
C=R[3];P[3]=C? XPal[C]:(K&0x10)? FC:BC;
|
|
C=R[4];P[4]=C? XPal[C]:(K&0x08)? FC:BC;
|
|
C=R[5];P[5]=C? XPal[C]:(K&0x04)? FC:BC;
|
|
C=R[6];P[6]=C? XPal[C]:(K&0x02)? FC:BC;
|
|
C=R[7];P[7]=C? XPal[C]:(K&0x01)? FC:BC;
|
|
}
|
|
}
|
|
}
|
|
|
|
/** RefreshLine5() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN5, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine5(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
register byte I,X,*T,*R;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
T=ChrTab+(((int)(Y+VScroll)<<7)&ChrTabM&0x7FFF);
|
|
|
|
for(X=0;X<16;X++,R+=16,P+=16,T+=8)
|
|
{
|
|
I=R[0];P[0]=XPal[I? I:T[0]>>4];
|
|
I=R[1];P[1]=XPal[I? I:T[0]&0x0F];
|
|
I=R[2];P[2]=XPal[I? I:T[1]>>4];
|
|
I=R[3];P[3]=XPal[I? I:T[1]&0x0F];
|
|
I=R[4];P[4]=XPal[I? I:T[2]>>4];
|
|
I=R[5];P[5]=XPal[I? I:T[2]&0x0F];
|
|
I=R[6];P[6]=XPal[I? I:T[3]>>4];
|
|
I=R[7];P[7]=XPal[I? I:T[3]&0x0F];
|
|
I=R[8];P[8]=XPal[I? I:T[4]>>4];
|
|
I=R[9];P[9]=XPal[I? I:T[4]&0x0F];
|
|
I=R[10];P[10]=XPal[I? I:T[5]>>4];
|
|
I=R[11];P[11]=XPal[I? I:T[5]&0x0F];
|
|
I=R[12];P[12]=XPal[I? I:T[6]>>4];
|
|
I=R[13];P[13]=XPal[I? I:T[6]&0x0F];
|
|
I=R[14];P[14]=XPal[I? I:T[7]>>4];
|
|
I=R[15];P[15]=XPal[I? I:T[7]&0x0F];
|
|
}
|
|
}
|
|
}
|
|
|
|
/** RefreshLine8() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN8, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine8(register byte Y)
|
|
{
|
|
static byte SprToScr[16] =
|
|
{
|
|
0x00,0x02,0x10,0x12,0x80,0x82,0x90,0x92,
|
|
0x49,0x4B,0x59,0x5B,0xC9,0xCB,0xD9,0xDB
|
|
};
|
|
register pixel *P;
|
|
register byte C,X,*T,*R;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,BPal[VDP[7]]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,BPal[VDP[7]]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
T=ChrTab+(((int)(Y+VScroll)<<8)&ChrTabM&0xFFFF);
|
|
|
|
for(X=0;X<32;X++,T+=8,R+=8,P+=8)
|
|
{
|
|
C=R[0];P[0]=BPal[C? SprToScr[C]:T[0]];
|
|
C=R[1];P[1]=BPal[C? SprToScr[C]:T[1]];
|
|
C=R[2];P[2]=BPal[C? SprToScr[C]:T[2]];
|
|
C=R[3];P[3]=BPal[C? SprToScr[C]:T[3]];
|
|
C=R[4];P[4]=BPal[C? SprToScr[C]:T[4]];
|
|
C=R[5];P[5]=BPal[C? SprToScr[C]:T[5]];
|
|
C=R[6];P[6]=BPal[C? SprToScr[C]:T[6]];
|
|
C=R[7];P[7]=BPal[C? SprToScr[C]:T[7]];
|
|
}
|
|
}
|
|
}
|
|
|
|
/** RefreshLine10() ******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN10/11, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine10(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
register byte C,X,*T,*R;
|
|
register int J,K;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,BPal[VDP[7]]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,BPal[VDP[7]]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
T=ChrTab+(((int)(Y+VScroll)<<8)&ChrTabM&0xFFFF);
|
|
|
|
/* Draw first 4 pixels */
|
|
C=R[0];P[0]=C? XPal[C]:BPal[VDP[7]];
|
|
C=R[1];P[1]=C? XPal[C]:BPal[VDP[7]];
|
|
C=R[2];P[2]=C? XPal[C]:BPal[VDP[7]];
|
|
C=R[3];P[3]=C? XPal[C]:BPal[VDP[7]];
|
|
R+=4;P+=4;
|
|
|
|
for(X=0;X<63;X++,T+=4,R+=4,P+=4)
|
|
{
|
|
K=(T[0]&0x07)|((T[1]&0x07)<<3);
|
|
if(K&0x20) K-=64;
|
|
J=(T[2]&0x07)|((T[3]&0x07)<<3);
|
|
if(J&0x20) J-=64;
|
|
|
|
C=R[0];Y=T[0]>>3;P[0]=C? XPal[C]:Y&1? XPal[Y>>1]:YJKColor(Y,J,K);
|
|
C=R[1];Y=T[1]>>3;P[1]=C? XPal[C]:Y&1? XPal[Y>>1]:YJKColor(Y,J,K);
|
|
C=R[2];Y=T[2]>>3;P[2]=C? XPal[C]:Y&1? XPal[Y>>1]:YJKColor(Y,J,K);
|
|
C=R[3];Y=T[3]>>3;P[3]=C? XPal[C]:Y&1? XPal[Y>>1]:YJKColor(Y,J,K);
|
|
}
|
|
}
|
|
}
|
|
|
|
/** RefreshLine12() ******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN12, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine12(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
register byte C,X,*T,*R;
|
|
register int J,K;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,BPal[VDP[7]]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,BPal[VDP[7]]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
T=ChrTab+(((int)(Y+VScroll)<<8)&ChrTabM&0xFFFF);
|
|
|
|
if(HScroll512&&(HScroll>255)) T=(byte *)((int)T^0x10000);
|
|
T+=HScroll&0xFC;
|
|
|
|
/* Draw first 4 pixels */
|
|
C=R[0];P[0]=C? XPal[C]:BPal[VDP[7]];
|
|
C=R[1];P[1]=C? XPal[C]:BPal[VDP[7]];
|
|
C=R[2];P[2]=C? XPal[C]:BPal[VDP[7]];
|
|
C=R[3];P[3]=C? XPal[C]:BPal[VDP[7]];
|
|
R+=4;P+=4;
|
|
|
|
for(X=1;X<64;X++,T+=4,R+=4,P+=4)
|
|
{
|
|
K=(T[0]&0x07)|((T[1]&0x07)<<3);
|
|
if(K&0x20) K-=64;
|
|
J=(T[2]&0x07)|((T[3]&0x07)<<3);
|
|
if(J&0x20) J-=64;
|
|
|
|
C=R[0];P[0]=C? XPal[C]:YJKColor(T[0]>>3,J,K);
|
|
C=R[1];P[1]=C? XPal[C]:YJKColor(T[1]>>3,J,K);
|
|
C=R[2];P[2]=C? XPal[C]:YJKColor(T[2]>>3,J,K);
|
|
C=R[3];P[3]=C? XPal[C]:YJKColor(T[3]>>3,J,K);
|
|
}
|
|
}
|
|
}
|
|
|
|
//#ifdef NARROW
|
|
|
|
/** RefreshLine6() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN6, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine6(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
register byte X,*T,*R,C;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor&0x03]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor&0x03]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
T=ChrTab+(((int)(Y+VScroll)<<7)&ChrTabM&0x7FFF);
|
|
|
|
for(X=0;X<32;X++)
|
|
{
|
|
C=R[0];P[0]=XPal[C? C:T[0]>>6];
|
|
C=R[1];P[1]=XPal[C? C:(T[0]>>2)&0x03];
|
|
C=R[2];P[2]=XPal[C? C:T[1]>>6];
|
|
C=R[3];P[3]=XPal[C? C:(T[1]>>2)&0x03];
|
|
C=R[4];P[4]=XPal[C? C:T[2]>>6];
|
|
C=R[5];P[5]=XPal[C? C:(T[2]>>2)&0x03];
|
|
C=R[6];P[6]=XPal[C? C:T[3]>>6];
|
|
C=R[7];P[7]=XPal[C? C:(T[3]>>2)&0x03];
|
|
R+=8;P+=8;T+=4;
|
|
}
|
|
}
|
|
}
|
|
|
|
/** RefreshLine7() *******************************************/
|
|
/** Refresh line Y (0..191/211) of SCREEN7, including **/
|
|
/** sprites in this line. **/
|
|
/*************************************************************/
|
|
static void RefreshLine7(register byte Y)
|
|
{
|
|
register pixel *P;
|
|
register byte C,X,*T,*R;
|
|
byte ZBuf[304];
|
|
|
|
P=RefreshBorder(Y,XPal[BGColor]);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,XPal[BGColor]);
|
|
else
|
|
{
|
|
ColorSprites(Y,ZBuf);
|
|
R=ZBuf+32;
|
|
T=ChrTab+(((int)(Y+VScroll)<<8)&ChrTabM&0xFFFF);
|
|
|
|
for(X=0;X<32;X++)
|
|
{
|
|
C=R[0];P[0]=XPal[C? C:T[0]>>4];
|
|
C=R[1];P[1]=XPal[C? C:T[1]>>4];
|
|
C=R[2];P[2]=XPal[C? C:T[2]>>4];
|
|
C=R[3];P[3]=XPal[C? C:T[3]>>4];
|
|
C=R[4];P[4]=XPal[C? C:T[4]>>4];
|
|
C=R[5];P[5]=XPal[C? C:T[5]>>4];
|
|
C=R[6];P[6]=XPal[C? C:T[6]>>4];
|
|
C=R[7];P[7]=XPal[C? C:T[7]>>4];
|
|
R+=8;P+=8;T+=8;
|
|
}
|
|
}
|
|
}
|
|
|
|
/** RefreshLineTx80() ****************************************/
|
|
/** Refresh line Y (0..191/211) of TEXT80. **/
|
|
/*************************************************************/
|
|
static void RefreshLineTx80(register byte Y)
|
|
{
|
|
register pixel *P,FC,BC;
|
|
register byte X,M,*T,*C,*G;
|
|
|
|
BC=XPal[BGColor];
|
|
P=RefreshBorder(Y,BC);
|
|
if(!P) return;
|
|
|
|
if(!ScreenON) ClearLine(P,BC);
|
|
else
|
|
{
|
|
P[0]=P[1]=P[2]=P[3]=P[4]=P[5]=P[6]=P[7]=P[8]=BC;
|
|
G=(UseFont&&FontBuf? FontBuf:ChrGen)+((Y+VScroll)&0x07);
|
|
T=ChrTab+((80*(Y>>3))&ChrTabM);
|
|
C=ColTab+((10*(Y>>3))&ColTabM);
|
|
P+=9;
|
|
|
|
for(X=0,M=0x00;X<80;X++,T++,P+=3)
|
|
{
|
|
if(!(X&0x07)) M=*C++;
|
|
if(M&0x80) { FC=XPal[XFGColor];BC=XPal[XBGColor]; }
|
|
else { FC=XPal[FGColor];BC=XPal[BGColor]; }
|
|
M<<=1;
|
|
Y=*(G+((int)*T<<3));
|
|
P[0]=Y&0xC0? FC:BC;
|
|
P[1]=Y&0x30? FC:BC;
|
|
P[2]=Y&0x0C? FC:BC;
|
|
}
|
|
|
|
P[0]=P[1]=P[2]=P[3]=P[4]=P[5]=P[6]=XPal[BGColor];
|
|
}
|
|
}
|
|
|
|
//#endif /* NARROW */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/** RdZ80() **************************************************/
|
|
/** Z80 emulation calls this function to read a byte from **/
|
|
/** address A of Z80 address space. Now moved to Z80.c and **/
|
|
/** made inlined to speed things up. **/
|
|
/*************************************************************/
|
|
byte RdZ80(word A)
|
|
{
|
|
if(A!=0xFFFF) return(RAM[A>>13][A&0x1FFF]);
|
|
else return(PSL[3]==3? ~SSLReg:RAM[7][0x1FFF]);
|
|
}
|
|
|
|
|
|
/** WrZ80() **************************************************/
|
|
/** Z80 emulation calls this function to write byte V to **/
|
|
/** address A of Z80 address space. **/
|
|
/*************************************************************/
|
|
void WrZ80(word A,byte V)
|
|
{
|
|
if(A!=0xFFFF)
|
|
{
|
|
if(EnWrite[A>>14]) RAM[A>>13][A&0x1FFF]=V;
|
|
else if((A>0x3FFF)&&(A<0xC000)) MapROM(A,V);
|
|
}
|
|
else
|
|
{
|
|
if(PSL[3]==3) SSlot(V);
|
|
else if(EnWrite[3]) RAM[7][A&0x1FFF]=V;
|
|
}
|
|
}
|
|
|
|
/** InZ80() **************************************************/
|
|
/** Z80 emulation calls this function to read a byte from **/
|
|
/** a given I/O port. **/
|
|
/*************************************************************/
|
|
byte InZ80(word Port)
|
|
{
|
|
/* MSX only uses 256 IO ports */
|
|
Port&=0xFF;
|
|
|
|
/* Return an appropriate port value */
|
|
switch(Port)
|
|
{
|
|
|
|
case 0x90: return(0xFD); /* Printer READY signal */
|
|
case 0xB5: return(RTCIn(RTCReg)); /* RTC registers */
|
|
|
|
case 0xA8: /* Primary slot state */
|
|
case 0xA9: /* Keyboard port */
|
|
case 0xAA: /* General IO register */
|
|
case 0xAB: /* PPI control register */
|
|
PPI.Rin[1]=KeyMap[PPI.Rout[2]&0x0F];
|
|
return(Read8255(&PPI,Port-0xA8));
|
|
|
|
case 0xFC: /* Mapper page at 0000h */
|
|
case 0xFD: /* Mapper page at 4000h */
|
|
case 0xFE: /* Mapper page at 8000h */
|
|
case 0xFF: /* Mapper page at C000h */
|
|
return(RAMMapper[Port-0xFC]|~RAMMask);
|
|
|
|
case 0xD9: /* Kanji support */
|
|
Port=Kanji? Kanji[KanLetter+KanCount]:NORAM;
|
|
KanCount=(KanCount+1)&0x1F;
|
|
return(Port);
|
|
|
|
case 0x80: /* SIO data */
|
|
case 0x81:
|
|
case 0x82:
|
|
case 0x83:
|
|
case 0x84:
|
|
case 0x85:
|
|
case 0x86:
|
|
case 0x87:
|
|
return(NORAM);
|
|
/*return(Rd8251(&SIO,Port&0x07));*/
|
|
|
|
case 0x98: /* VRAM read port */
|
|
/* Read from VRAM data buffer */
|
|
Port=VDPData;
|
|
/* Reset VAddr latch sequencer */
|
|
VKey=1;
|
|
/* Fill data buffer with a new value */
|
|
VDPData=VPAGE[VAddr];
|
|
/* Increment VRAM address */
|
|
VAddr=(VAddr+1)&0x3FFF;
|
|
/* If rolled over, modify VRAM page# */
|
|
if(!VAddr&&(ScrMode>3))
|
|
{
|
|
VDP[14]=(VDP[14]+1)&(VRAMPages-1);
|
|
VPAGE=VRAM+((int)VDP[14]<<14);
|
|
}
|
|
return(Port);
|
|
|
|
case 0x99: /* VDP status registers */
|
|
/* Read an appropriate status register */
|
|
Port=VDPStatus[VDP[15]];
|
|
/* Reset VAddr latch sequencer */
|
|
VKey=1;
|
|
/* Update status register's contents */
|
|
switch(VDP[15])
|
|
{
|
|
case 0: VDPStatus[0]&=0x5F;SetIRQ(~INT_IE0);break;
|
|
case 1: VDPStatus[1]&=0xFE;SetIRQ(~INT_IE1);break;
|
|
case 7: VDPStatus[7]=VDP[44]=VDPRead();break;
|
|
}
|
|
/* Return the status register value */
|
|
return(Port);
|
|
|
|
case 0xA2: /* PSG input port */
|
|
/* PSG[14] returns joystick/mouse data */
|
|
if(PSG.Latch==14)
|
|
{
|
|
int DX,DY,L;
|
|
|
|
/* Number of a joystick port */
|
|
Port=PSG.R[15]&0x40? 1:0;
|
|
L=Port? JoyTypeB:JoyTypeA;
|
|
|
|
/* If no joystick, return dummy value */
|
|
if(!L) return(0x7F);
|
|
|
|
/* @@@ For debugging purposes */
|
|
/*printf("Reading from PSG[14]: MCount[%d]=%d PSG[15]=%02Xh Value=",Port,MCount[Port],PSG.R[15]);*/
|
|
|
|
/* Poll mouse position, if needed */
|
|
if((L==2)||(MCount[Port]==1))
|
|
{
|
|
/* Read new mouse coordinates */
|
|
L=Mouse(Port);
|
|
Buttons[Port]=(~L>>12)&0x30;
|
|
DY=(L>>8)&0xFF;
|
|
DX=L&0xFF;
|
|
|
|
/* Compute offsets and store coordinates */
|
|
L=OldMouseX[Port]-DX;OldMouseX[Port]=DX;DX=L;
|
|
L=OldMouseY[Port]-DY;OldMouseY[Port]=DY;DY=L;
|
|
|
|
/* Adjust offsets */
|
|
MouseDX[Port]=(DX>127? 127:(DX<-127? -127:DX))&0xFF;
|
|
MouseDY[Port]=(DY>127? 127:(DY<-127? -127:DY))&0xFF;
|
|
}
|
|
|
|
/* Determine return value */
|
|
switch(MCount[Port])
|
|
{
|
|
case 0: /* Normal joystick */
|
|
if(PSG.R[15]&(Port? 0x20:0x10)) Port=0x3F;
|
|
else
|
|
if((Port? JoyTypeB:JoyTypeA)<2) Port=~Joystick(Port)&0x3F;
|
|
else Port=Buttons[Port]
|
|
|(MouseDX[Port]? (MouseDX[Port]<128? 0x08:0x04):0x0C)
|
|
|(MouseDY[Port]? (MouseDY[Port]<128? 0x02:0x01):0x03);
|
|
break;
|
|
|
|
case 1: Port=(MouseDX[Port]>>4)|Buttons[Port];break;
|
|
case 2: Port=(MouseDX[Port]&0x0F)|Buttons[Port];break;
|
|
case 3: Port=(MouseDY[Port]>>4)|Buttons[Port];break;
|
|
case 4: Port=(MouseDY[Port]&0x0F)|Buttons[Port];break;
|
|
}
|
|
|
|
/* @@@ For debugging purposes */
|
|
/*printf("%02Xh\n",Port|0x40);*/
|
|
|
|
/* 6th bit is always 1 */
|
|
return(Port|0x40);
|
|
}
|
|
|
|
/* PSG[15] resets mouse counters */
|
|
if(PSG.Latch==15)
|
|
{
|
|
/* @@@ For debugging purposes */
|
|
/*printf("Reading from PSG[15]\n");*/
|
|
|
|
/*MCount[0]=MCount[1]=0;*/
|
|
return(PSG.R[15]&0xF0);
|
|
}
|
|
|
|
/* Return PSG[0-13] as they are */
|
|
return(RdData8910(&PSG));
|
|
}
|
|
|
|
/* Return NORAM for non-existing ports */
|
|
return(NORAM);
|
|
}
|
|
|
|
/** OutZ80() *************************************************/
|
|
/** Z80 emulation calls this function to write byte V to a **/
|
|
/** given I/O port. **/
|
|
/*************************************************************/
|
|
void OutZ80(word Port,byte Value)
|
|
{
|
|
register byte I,J,K;
|
|
|
|
Port&=0xFF;
|
|
switch(Port)
|
|
{
|
|
|
|
case 0x7C: WrCtrl2413(&OPLL,Value);return; /* OPLL Register# */
|
|
case 0x7D: WrData2413(&OPLL,Value);return; /* OPLL Data */
|
|
case 0x91: Printer(Value);return; /* Printer Data */
|
|
case 0xA0: WrCtrl8910(&PSG,Value);return; /* PSG Register# */
|
|
case 0xB4: RTCReg=Value&0x0F;return; /* RTC Register# */
|
|
|
|
case 0xD8: /* Upper bits of Kanji ROM address */
|
|
KanLetter=(KanLetter&0x1F800)|((int)(Value&0x3F)<<5);
|
|
KanCount=0;
|
|
return;
|
|
|
|
case 0xD9: /* Lower bits of Kanji ROM address */
|
|
KanLetter=(KanLetter&0x007E0)|((int)(Value&0x3F)<<11);
|
|
KanCount=0;
|
|
return;
|
|
|
|
case 0x80: /* SIO data */
|
|
case 0x81:
|
|
case 0x82:
|
|
case 0x83:
|
|
case 0x84:
|
|
case 0x85:
|
|
case 0x86:
|
|
case 0x87:
|
|
return;
|
|
/*Wr8251(&SIO,Port&0x07,Value);
|
|
return;*/
|
|
|
|
case 0x98: /* VDP Data */
|
|
VKey=1;
|
|
if(WKey)
|
|
{
|
|
/* VDP set for writing */
|
|
VDPData=VPAGE[VAddr]=Value;
|
|
VAddr=(VAddr+1)&0x3FFF;
|
|
}
|
|
else
|
|
{
|
|
/* VDP set for reading */
|
|
VDPData=VPAGE[VAddr];
|
|
VAddr=(VAddr+1)&0x3FFF;
|
|
VPAGE[VAddr]=Value;
|
|
}
|
|
/* If VAddr rolled over, modify VRAM page# */
|
|
if(!VAddr&&(ScrMode>3))
|
|
{
|
|
VDP[14]=(VDP[14]+1)&(VRAMPages-1);
|
|
VPAGE=VRAM+((int)VDP[14]<<14);
|
|
}
|
|
return;
|
|
|
|
case 0x99: /* VDP Address Latch */
|
|
if(VKey) { ALatch=Value;VKey=0; }
|
|
else
|
|
{
|
|
VKey=1;
|
|
switch(Value&0xC0)
|
|
{
|
|
case 0x80:
|
|
/* Writing into VDP registers */
|
|
VDPOut(Value&0x3F,ALatch);
|
|
break;
|
|
case 0x00:
|
|
case 0x40:
|
|
/* Set the VRAM access address */
|
|
VAddr=(((word)Value<<8)+ALatch)&0x3FFF;
|
|
/* WKey=1 when VDP set for writing into VRAM */
|
|
WKey=Value&0x40;
|
|
/* When set for reading, perform first read */
|
|
if(!WKey)
|
|
{
|
|
VDPData=VPAGE[VAddr];
|
|
VAddr=(VAddr+1)&0x3FFF;
|
|
if(!VAddr&&(ScrMode>3))
|
|
{
|
|
VDP[14]=(VDP[14]+1)&(VRAMPages-1);
|
|
VPAGE=VRAM+((int)VDP[14]<<14);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return;
|
|
|
|
case 0x9A: /* VDP Palette Latch */
|
|
if(PKey) { PLatch=Value;PKey=0; }
|
|
else
|
|
{
|
|
byte R,G,B;
|
|
/* New palette entry written */
|
|
PKey=1;
|
|
J=VDP[16];
|
|
/* Compute new color components */
|
|
R=(PLatch&0x70)*255/112;
|
|
G=(Value&0x07)*255/7;
|
|
B=(PLatch&0x07)*255/7;
|
|
/* Set new color for palette entry J */
|
|
Palette[J]=RGB2INT(R,G,B);
|
|
SetColor(J,R,G,B);
|
|
/* Next palette entry */
|
|
VDP[16]=(J+1)&0x0F;
|
|
}
|
|
return;
|
|
|
|
case 0x9B: /* VDP Register Access */
|
|
J=VDP[17]&0x3F;
|
|
if(J!=17) VDPOut(J,Value);
|
|
if(!(VDP[17]&0x80)) VDP[17]=(J+1)&0x3F;
|
|
return;
|
|
|
|
case 0xA1: /* PSG Data */
|
|
/* PSG[15] is responsible for joystick/mouse */
|
|
if(PSG.Latch==15)
|
|
{
|
|
/* @@@ For debugging purposes */
|
|
/*printf("Writing PSG[15] <- %02Xh\n",Value);*/
|
|
|
|
/* For mouse, update nibble counter */
|
|
/* For joystick, set nibble counter to 0 */
|
|
if((Value&0x0C)==0x0C) MCount[1]=0;
|
|
else if((JoyTypeB>2)&&((Value^PSG.R[15])&0x20))
|
|
MCount[1]+=MCount[1]==4? -3:1;
|
|
|
|
/* For mouse, update nibble counter */
|
|
/* For joystick, set nibble counter to 0 */
|
|
if((Value&0x03)==0x03) MCount[0]=0;
|
|
else if((JoyTypeA>2)&&((Value^PSG.R[15])&0x10))
|
|
MCount[0]+=MCount[0]==4? -3:1;
|
|
}
|
|
|
|
/* Put value into a register */
|
|
WrData8910(&PSG,Value);
|
|
return;
|
|
|
|
case 0xB5: /* RTC Data */
|
|
if(RTCReg<13)
|
|
{
|
|
/* J = register bank# now */
|
|
J=RTCMode&0x03;
|
|
/* Store the value */
|
|
RTC[J][RTCReg]=Value;
|
|
/* If CMOS modified, we need to save it */
|
|
if(J>1) SaveCMOS=1;
|
|
return;
|
|
}
|
|
/* RTC[13] is a register bank# */
|
|
if(RTCReg==13) RTCMode=Value;
|
|
return;
|
|
|
|
case 0xA8: /* Primary slot state */
|
|
case 0xA9: /* Keyboard port */
|
|
case 0xAA: /* General IO register */
|
|
case 0xAB: /* PPI control register */
|
|
/* Write to PPI */
|
|
Write8255(&PPI,Port-0xA8,Value);
|
|
/* If general I/O register has changed... */
|
|
if(PPI.Rout[2]!=IOReg) { PPIOut(PPI.Rout[2],IOReg);IOReg=PPI.Rout[2]; }
|
|
/* If primary slot state has changed... */
|
|
if(PPI.Rout[0]!=PSLReg)
|
|
for(J=0,PSLReg=Value=PPI.Rout[0];J<4;J++,Value>>=2)
|
|
{
|
|
PSL[J]=Value&3;I=J<<1;
|
|
K=PSL[J]==3? SSL[J]:0;
|
|
EnWrite[J]=(K==2)&&(MemMap[3][2][I]!=EmptyRAM);
|
|
RAM[I]=MemMap[PSL[J]][K][I];
|
|
RAM[I+1]=MemMap[PSL[J]][K][I+1];
|
|
}
|
|
/* Done */
|
|
return;
|
|
|
|
case 0xFC: /* Mapper page at 0000h */
|
|
case 0xFD: /* Mapper page at 4000h */
|
|
case 0xFE: /* Mapper page at 8000h */
|
|
case 0xFF: /* Mapper page at C000h */
|
|
J=Port-0xFC;
|
|
Value&=RAMMask;
|
|
if(RAMMapper[J]!=Value)
|
|
{
|
|
if(Verbose&0x08) {
|
|
//printf("RAM-MAPPER: block %d at %Xh\n",Value,J*0x4000);
|
|
emu_printf("RAM-MAPPER: block");
|
|
emu_printi(Value);
|
|
emu_printi(J*0x4000);
|
|
}
|
|
I=J<<1;
|
|
RAMMapper[J] = Value;
|
|
MemMap[3][2][I] = RAMData+((int)Value<<14);
|
|
MemMap[3][2][I+1] = MemMap[3][2][I]+0x2000;
|
|
if((PSL[J]==3)&&(SSL[J]==2))
|
|
{
|
|
EnWrite[J] = 1;
|
|
RAM[I] = MemMap[3][2][I];
|
|
RAM[I+1] = MemMap[3][2][I+1];
|
|
}
|
|
}
|
|
return;
|
|
|
|
}
|
|
}
|
|
|
|
/** MapROM() *************************************************/
|
|
/** Switch ROM Mapper pages. This function is supposed to **/
|
|
/** be called when ROM page registers are written to. **/
|
|
/*************************************************************/
|
|
void MapROM(register word A,register byte V)
|
|
{
|
|
byte I,J,K,*P;
|
|
|
|
/* @@@ For debugging purposes
|
|
printf("(%04Xh) = %02Xh at PC=%04Xh\n",A,V,CPU.PC.W);
|
|
*/
|
|
|
|
/* J contains 16kB page number 0-3 */
|
|
J=A>>14;
|
|
|
|
/* I contains slot number 0/1 */
|
|
if(PSL[J]==1) I=0;
|
|
else if(PSL[J]==2) I=1;
|
|
else return;
|
|
|
|
/* K contains MegaROM type */
|
|
K=I? ROMTypeB:ROMTypeA;
|
|
|
|
/* SCC: enable/disable for no cart */
|
|
if(!ROMData[I]&&(A==0x9000)) SCCOn[I]=(V==0x3F)? 1:0;
|
|
|
|
/* SCC: types 0, 2, or no cart */
|
|
if(((A&0xFF00)==0x9800)&&SCCOn[I])
|
|
{
|
|
/* Compute SCC register number */
|
|
J=A&0x00FF;
|
|
|
|
/* When no MegaROM present, we allow the program */
|
|
/* to write into SCC wave buffer using EmptyRAM */
|
|
/* as a scratch pad. */
|
|
if(!ROMData[I]&&(J<0x80)) EmptyRAM[0x1800+J]=V;
|
|
|
|
/* Output data to SCC chip */
|
|
WriteSCC(&SCChip,J,V);
|
|
return;
|
|
}
|
|
|
|
/* SCC+: types 0, 2, or no cart */
|
|
if(((A&0xFF00)==0xB800)&&SCCOn[I])
|
|
{
|
|
/* Compute SCC register number */
|
|
J=A&0x00FF;
|
|
|
|
/* When no MegaROM present, we allow the program */
|
|
/* to write into SCC wave buffer using EmptyRAM */
|
|
/* as a scratch pad. */
|
|
if(!ROMData[I]&&(J<0xA0)) EmptyRAM[0x1800+J]=V;
|
|
|
|
/* Output data to SCC chip */
|
|
WriteSCCP(&SCChip,J,V);
|
|
return;
|
|
}
|
|
|
|
/* If no cartridge or no mapper, exit */
|
|
if(!ROMData[I]||!ROMMask[I]) return;
|
|
|
|
switch(K)
|
|
{
|
|
case 0: /*** Generic 8kB cartridges (Konami, etc.) ***/
|
|
if((A<0x4000)||(A>0xBFFF)) return;
|
|
J=(A-0x4000)>>13;
|
|
/* Turn SCC on/off on writes to 8000h-9FFFh */
|
|
if(J==2) SCCOn[I]=(V==0x3F)? 1:0;
|
|
/* Switch ROM pages */
|
|
V&=ROMMask[I];
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
RAM[J+2]=MemMap[I+1][0][J+2]=ROMData[I]+((int)V<<13);
|
|
ROMMapper[I][J]=V;
|
|
}
|
|
break;
|
|
|
|
case 1: /*** Generic 16kB cartridges (MSXDOS2, HoleInOneSpecial) ***/
|
|
if((A<0x4000)||(A>0xBFFF)) return;
|
|
J=(A&0x8000)>>14;
|
|
V=(V<<1)&ROMMask[I];
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
RAM[J+2]=MemMap[I+1][0][J+2]=ROMData[I]+((int)V<<13);
|
|
RAM[J+3]=MemMap[I+1][0][J+3]=RAM[2]+0x2000;
|
|
ROMMapper[I][J]=V;
|
|
}
|
|
break;
|
|
|
|
case 2: /*** KONAMI5 8kB cartridges ***/
|
|
if((A<0x5000)||(A>0xB000)||((A&0x1FFF)!=0x1000)) return;
|
|
J=(A-0x5000)>>13;
|
|
/* Turn SCC on/off on writes to 9000h */
|
|
if(J==2) SCCOn[I]=(V==0x3F)? 1:0;
|
|
/* Switch ROM pages */
|
|
V&=ROMMask[I];
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
RAM[J+2]=MemMap[I+1][0][J+2]=ROMData[I]+((int)V<<13);
|
|
ROMMapper[I][J]=V;
|
|
}
|
|
break;
|
|
|
|
case 3: /*** KONAMI4 8kB cartridges ***/
|
|
/* Page at 4000h is fixed */
|
|
if((A<0x6000)||(A>0xA000)||(A&0x1FFF)) return;
|
|
J=(A-0x4000)>>13;
|
|
V&=ROMMask[I];
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
RAM[J+2]=MemMap[I+1][0][J+2]=ROMData[I]+((int)V<<13);
|
|
ROMMapper[I][J]=V;
|
|
}
|
|
break;
|
|
|
|
case 4: /*** ASCII 8kB cartridges ***/
|
|
if((A<0x6000)||(A>0x7FFF)) return;
|
|
J=(A&0x1800)>>11;
|
|
V&=ROMMask[I];
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
P=ROMData[I]+((int)V<<13);
|
|
MemMap[I+1][0][J+2]=P;
|
|
ROMMapper[I][J]=V;
|
|
/* Only update memory when cartridge's slot selected */
|
|
if(PSL[(J>>1)+1]==I+1) RAM[J+2]=P;
|
|
}
|
|
break;
|
|
|
|
case 5: /*** ASCII 16kB cartridges ***/
|
|
if((A<0x6000)||(A>0x7FFF)) return;
|
|
J=(A&0x1000)>>11;
|
|
V=(V<<1)&ROMMask[I];
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
P=ROMData[I]+((int)V<<13);
|
|
MemMap[I+1][0][J+2]=P;
|
|
MemMap[I+1][0][J+3]=P+0x2000;
|
|
ROMMapper[I][J]=V;
|
|
/* Only update memory when cartridge's slot selected */
|
|
if(PSL[(J>>1)+1]==I+1)
|
|
{
|
|
RAM[J+2]=P;
|
|
RAM[J+3]=P+0x2000;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 6: /*** GameMaster2+SRAM cartridge ***/
|
|
/* Switch ROM and SRAM pages, page at 4000h is fixed */
|
|
if((A>=0x6000)&&(A<=0xA000)&&!(A&0x1FFF))
|
|
{
|
|
/* If changing SRAM page... */
|
|
if(V&0x10)
|
|
{
|
|
/* Select SRAM page */
|
|
RAM[5]=MemMap[I+1][0][5]=(SRAM? SRAM:EmptyRAM)+(V&0x20? 0x2000:0);
|
|
/* SRAM is now on */
|
|
ROMMapper[I][3]=0xFF;
|
|
}
|
|
else
|
|
{
|
|
/* Figure out which ROM page gets switched */
|
|
J=(A-0x4000)>>13;
|
|
/* Compute new ROM page number */
|
|
V&=ROMMask[I];
|
|
/* If ROM page number has changed... */
|
|
if(V!=ROMMapper[I][J])
|
|
{
|
|
RAM[J+2]=MemMap[I+1][0][J+2]=ROMData[I]+((int)V<<13);
|
|
ROMMapper[I][J]=V;
|
|
}
|
|
}
|
|
/* Done with page switch */
|
|
break;
|
|
}
|
|
/* Write to SRAM */
|
|
if((A>=0xB000)&&(A<0xC000)&&(ROMMapper[I][3]==0xFF))
|
|
{
|
|
RAM[5][(A&0x0FFF)|0x1000]=RAM[5][A&0x0FFF]=V;
|
|
SaveSRAM=1;
|
|
break;
|
|
}
|
|
/* Done */
|
|
return;
|
|
|
|
default: /*** No MegaROM mapper by default ***/
|
|
return;
|
|
}
|
|
|
|
if(Verbose&0x08)
|
|
{
|
|
/* K=1 for 16kB pages */
|
|
K=(K==1)||(K==5);
|
|
|
|
//printf
|
|
//(
|
|
// "ROM-MAPPER %c: %dkB block %d at %04Xh\n",
|
|
// I+'A',8*(K+1),K? (V>>1):V,J*0x2000+0x4000
|
|
//);
|
|
}
|
|
}
|
|
|
|
/** SSlot() **************************************************/
|
|
/** Switch secondary memory slots. This function is called **/
|
|
/** when value in (FFFFh) changes. **/
|
|
/*************************************************************/
|
|
void SSlot(register byte V)
|
|
{
|
|
register byte I,J;
|
|
|
|
if(SSLReg!=V)
|
|
{
|
|
SSLReg=V;
|
|
|
|
for(J=0;J<4;J++,V>>=2)
|
|
{
|
|
SSL[J]=V&3;
|
|
if(PSL[J]==3)
|
|
{
|
|
I=J<<1;
|
|
EnWrite[J]=(SSL[J]==2)&&(MemMap[3][2][I]!=EmptyRAM);
|
|
RAM[I]=MemMap[3][SSL[J]][I];
|
|
RAM[I+1]=MemMap[3][SSL[J]][I+1];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/** LoadROM() ************************************************/
|
|
/** Load a file, allocating memory as needed. Returns addr. **/
|
|
/** of the alocated space or 0 if failed. **/
|
|
/*************************************************************/
|
|
byte *LoadROM(const char *Name,int Size,byte *Buf)
|
|
{
|
|
byte *P;
|
|
|
|
/* Can't give address without size! */
|
|
if(Buf&&!Size) return(0);
|
|
|
|
Size = emu_FileSize(Name);
|
|
emu_printi(Size);
|
|
|
|
if (Size<=0) return (0);
|
|
|
|
P=Buf? Buf:emu_Malloc(Size);
|
|
emu_LoadFile(Name, P, Size);
|
|
|
|
//Add a new chunk to free
|
|
if(!Buf) Chunks[CCount++]=P;
|
|
|
|
return(P);
|
|
}
|
|
|
|
/** LoadCARTROM() ********************************************/
|
|
/** Load a file. Returns addr where loaded. **/
|
|
/*************************************************************/
|
|
byte *LoadCARTROM(const char *Name,int Size,byte *Buf)
|
|
{
|
|
byte *P;
|
|
|
|
Size = emu_FileSize(Name);
|
|
emu_printi(Size);
|
|
|
|
if (Size<=0) return (0);
|
|
|
|
if (Buf)
|
|
{
|
|
P=Buf;
|
|
emu_printf("Loading to ram");
|
|
emu_LoadFile(Name, P, Size);
|
|
//byte * FP = (byte *)flash_load(Name);
|
|
//flash_verify(P,Size);
|
|
return P;
|
|
}
|
|
else
|
|
{
|
|
emu_printf("Loading to flash");
|
|
flash_load(Name);
|
|
P = (byte *)flash_start;
|
|
return P;
|
|
}
|
|
}
|
|
|
|
/** LoadCart() ***********************************************/
|
|
/** Load a cartridge ROM from .ROM file into given slot. **/
|
|
/*************************************************************/
|
|
int LoadCart(const char *Name,int Slot)
|
|
{
|
|
char C1,C2;
|
|
int C3,ROM64,LastFirst;
|
|
|
|
/* Check slot #, try to open file */
|
|
if((Slot!=0)&&(Slot!=1)) return(0);
|
|
|
|
int Size = emu_FileSize(Name);
|
|
|
|
int f = emu_FileOpen(Name,"r+b");
|
|
if(!f) return(0);
|
|
if(Verbose) emu_printf(Name);
|
|
if(Verbose) emu_printi(Size);
|
|
|
|
/* Check "AB" signature in a file */
|
|
ROM64=LastFirst=0;
|
|
emu_FileRead(&C1,1,f);
|
|
emu_FileRead(&C2,1,f);
|
|
|
|
/* Maybe this is a flat 64kB ROM? */
|
|
if((C1!='A')||(C2!='B')) {
|
|
if(Verbose) emu_printf("check next");
|
|
if(emu_FileSeek(f,0x4000,0)>=0)
|
|
{
|
|
emu_FileRead(&C1,1,f);
|
|
emu_FileRead(&C2,1,f);
|
|
ROM64=(C1=='A')&&(C2=='B');
|
|
}
|
|
}
|
|
|
|
/* Maybe it is the last page that contains "AB" signature? */
|
|
if((C1!='A')||(C2!='B')) {
|
|
if(Verbose) emu_printf("check next");
|
|
if(emu_FileSeek(f,Size-0x4000,0)>=0)
|
|
{
|
|
emu_FileRead(&C1,1,f);
|
|
emu_FileRead(&C2,1,f);
|
|
LastFirst=(C1=='A')&&(C2=='B');
|
|
}
|
|
}
|
|
|
|
/* If we can't find "AB" signature, drop out */
|
|
if((C1!='A')||(C2!='B'))
|
|
{
|
|
if(Verbose) emu_printf(" Not a valid cartridge ROM");
|
|
emu_FileClose(f);
|
|
return(0);
|
|
}
|
|
|
|
if(Verbose) {
|
|
emu_printf(" Cartridge");
|
|
char slot[2] = {0,0};
|
|
slot[0] = 'A'+Slot;
|
|
emu_printf(slot);
|
|
}
|
|
|
|
|
|
|
|
/* Length must be a multiple of 8kB */
|
|
/* Flat 64kB ROM must be 40..64kB */
|
|
//if(Size&0x1FFF) { emu_printf("failed");return(0); }
|
|
//if(ROM64&&(Size<0xA000)) { emu_printf("failed");return(0); }
|
|
//if(ROM64&&(Size>0x10000)) { emu_printf("failed");return(0); }
|
|
|
|
/* Show ROM type and size */
|
|
//if(Verbose)
|
|
// printf
|
|
// (
|
|
// "%dkB %s ROM...",C1/1024,
|
|
// !ROM64&&(C1>0x8000)? ROMNames[Slot? ROMTypeB:ROMTypeA]:"NORMAL"
|
|
// );
|
|
|
|
/* Compute size in 8kB pages */
|
|
Size >>= 13;
|
|
|
|
/* Calculate 2^n closest to number of pages */
|
|
for(C3=1;C3<Size;C3<<=1);
|
|
|
|
/* Assign ROMMask for MegaROMs */
|
|
ROMMask[Slot]=!ROM64&&(Size>4)? C3-1:0x00;
|
|
|
|
if ((Size*0x2000) <= 32768)
|
|
{
|
|
// Allocate space for the ROM
|
|
ROMData[Slot]=emu_Malloc(C3*0x2000);
|
|
if(!ROMData[Slot]) {
|
|
emu_printf("failed");
|
|
return(0);
|
|
}
|
|
Chunks[CCount++]=ROMData[Slot];
|
|
|
|
// Try loading ROM
|
|
byte * P = LoadCARTROM(Name,Size*0x2000,ROMData[Slot]);
|
|
if(!P) {
|
|
emu_printf("failed");
|
|
return(0);
|
|
}
|
|
ROMData[Slot] = P;
|
|
// Mirror ROM if it is smaller than 2^n pages
|
|
if(Size<C3) {
|
|
emu_printf("mirroring rom");
|
|
memcpy
|
|
(
|
|
ROMData[Slot]+Size*0x2000,
|
|
ROMData[Slot]+(Size-C3/2)*0x2000,
|
|
(C3-Size)*0x2000
|
|
);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ROMData[Slot] = LoadCARTROM(Name,Size*0x2000,NULL);
|
|
}
|
|
|
|
|
|
/* Set memory map depending on the ROM size */
|
|
switch(Size)
|
|
{
|
|
case 1:
|
|
/* 8kB ROMs are mirrored 8 times: 0:0:0:0:0:0:0:0 */
|
|
MemMap[Slot+1][0][0]=ROMData[Slot];
|
|
MemMap[Slot+1][0][1]=ROMData[Slot];
|
|
MemMap[Slot+1][0][2]=ROMData[Slot];
|
|
MemMap[Slot+1][0][3]=ROMData[Slot];
|
|
MemMap[Slot+1][0][4]=ROMData[Slot];
|
|
MemMap[Slot+1][0][5]=ROMData[Slot];
|
|
MemMap[Slot+1][0][6]=ROMData[Slot];
|
|
MemMap[Slot+1][0][7]=ROMData[Slot];
|
|
break;
|
|
|
|
case 2:
|
|
/* 16kB ROMs are mirrored 4 times: 0:1:0:1:0:1:0:1 */
|
|
MemMap[Slot+1][0][0]=ROMData[Slot];
|
|
MemMap[Slot+1][0][1]=ROMData[Slot]+0x2000;
|
|
MemMap[Slot+1][0][2]=ROMData[Slot];
|
|
MemMap[Slot+1][0][3]=ROMData[Slot]+0x2000;
|
|
MemMap[Slot+1][0][4]=ROMData[Slot];
|
|
MemMap[Slot+1][0][5]=ROMData[Slot]+0x2000;
|
|
MemMap[Slot+1][0][6]=ROMData[Slot];
|
|
MemMap[Slot+1][0][7]=ROMData[Slot]+0x2000;
|
|
break;
|
|
|
|
case 3:
|
|
case 4:
|
|
/* 24kB and 32kB ROMs are mirrored twice: 0:1:0:1:2:3:2:3 */
|
|
MemMap[Slot+1][0][0]=ROMData[Slot];
|
|
MemMap[Slot+1][0][1]=ROMData[Slot]+0x2000;
|
|
MemMap[Slot+1][0][2]=ROMData[Slot];
|
|
MemMap[Slot+1][0][3]=ROMData[Slot]+0x2000;
|
|
MemMap[Slot+1][0][4]=ROMData[Slot]+0x4000;
|
|
MemMap[Slot+1][0][5]=ROMData[Slot]+0x6000;
|
|
MemMap[Slot+1][0][6]=ROMData[Slot]+0x4000;
|
|
MemMap[Slot+1][0][7]=ROMData[Slot]+0x6000;
|
|
break;
|
|
|
|
default:
|
|
if(ROM64)
|
|
{
|
|
/* 64kB ROMs are loaded to fill slot: 0:1:2:3:4:5:6:7 */
|
|
MemMap[Slot+1][0][0]=ROMData[Slot];
|
|
MemMap[Slot+1][0][1]=ROMData[Slot]+0x2000;
|
|
MemMap[Slot+1][0][2]=ROMData[Slot]+0x4000;
|
|
MemMap[Slot+1][0][3]=ROMData[Slot]+0x6000;
|
|
MemMap[Slot+1][0][4]=ROMData[Slot]+0x8000;
|
|
MemMap[Slot+1][0][5]=ROMData[Slot]+0xA000;
|
|
MemMap[Slot+1][0][6]=ROMData[Slot]+0xC000;
|
|
MemMap[Slot+1][0][7]=ROMData[Slot]+0xE000;
|
|
}
|
|
else
|
|
{
|
|
/* MegaROMs are switched into 4000h-BFFFh */
|
|
if(!LastFirst) SetMegaROM(Slot,0,1,2,3);
|
|
else SetMegaROM(Slot,Size-2,Size-1,Size-2,Size-1);
|
|
}
|
|
break;
|
|
}
|
|
|
|
/* Show starting address */
|
|
if(Verbose) {
|
|
emu_printf("starts at");
|
|
emu_printi(MemMap[Slot+1][0][2][2]+256*MemMap[Slot+1][0][2][3]);
|
|
}
|
|
|
|
|
|
/* Done loading cartridge */
|
|
return(1);
|
|
}
|
|
|
|
|
|
|
|
/** SetIRQ() *************************************************/
|
|
/** Set or reset IRQ. Returns IRQ vector assigned to **/
|
|
/** CPU.IRequest. When upper bit of IRQ is 1, IRQ is reset. **/
|
|
/*************************************************************/
|
|
word SetIRQ(register byte IRQ)
|
|
{
|
|
if(IRQ&0x80) IRQPending&=IRQ; else IRQPending|=IRQ;
|
|
CPU.IRequest=IRQPending? INT_IRQ:INT_NONE;
|
|
return(CPU.IRequest);
|
|
}
|
|
|
|
/** SetScreen() **********************************************/
|
|
/** Change screen mode. Returns new screen mode. **/
|
|
/*************************************************************/
|
|
byte SetScreen(void)
|
|
{
|
|
register byte I,J;
|
|
|
|
switch(((VDP[0]&0x0E)>>1)|(VDP[1]&0x18))
|
|
{
|
|
case 0x10: J=0;break;
|
|
case 0x00: J=1;break;
|
|
case 0x01: J=2;break;
|
|
case 0x08: J=3;break;
|
|
case 0x02: J=4;break;
|
|
case 0x03: J=5;break;
|
|
case 0x04: J=6;break;
|
|
case 0x05: J=7;break;
|
|
case 0x07: J=8;break;
|
|
case 0x12: J=MAXSCREEN+1;break;
|
|
default: J=ScrMode;break;
|
|
}
|
|
|
|
/* Recompute table addresses */
|
|
I=(J>6)&&(J!=MAXSCREEN+1)? 11:10;
|
|
ChrTab = VRAM+((int)(VDP[2]&MSK[J].R2)<<I);
|
|
ChrGen = VRAM+((int)(VDP[4]&MSK[J].R4)<<11);
|
|
ColTab = VRAM+((int)(VDP[3]&MSK[J].R3)<<6)+((int)VDP[10]<<14);
|
|
SprTab = VRAM+((int)(VDP[5]&MSK[J].R5)<<7)+((int)VDP[11]<<15);
|
|
SprGen = VRAM+((int)VDP[6]<<11);
|
|
ChrTabM = ((int)(VDP[2]|~MSK[J].M2)<<I)|((1<<I)-1);
|
|
ChrGenM = ((int)(VDP[4]|~MSK[J].M4)<<11)|0x007FF;
|
|
ColTabM = ((int)(VDP[3]|~MSK[J].M3)<<6)|0x1C03F;
|
|
SprTabM = ((int)(VDP[5]|~MSK[J].M5)<<7)|0x1807F;
|
|
|
|
/* Return new screen mode */
|
|
ScrMode=J;
|
|
return(J);
|
|
}
|
|
|
|
/** SetMegaROM() *********************************************/
|
|
/** Set MegaROM pages for a given slot. SetMegaROM() always **/
|
|
/** assumes 8kB pages. **/
|
|
/*************************************************************/
|
|
void SetMegaROM(int Slot,byte P0,byte P1,byte P2,byte P3)
|
|
{
|
|
P0&=ROMMask[Slot];
|
|
P1&=ROMMask[Slot];
|
|
P2&=ROMMask[Slot];
|
|
P3&=ROMMask[Slot];
|
|
MemMap[Slot+1][0][2]=ROMData[Slot]+P0*0x2000;
|
|
MemMap[Slot+1][0][3]=ROMData[Slot]+P1*0x2000;
|
|
MemMap[Slot+1][0][4]=ROMData[Slot]+P2*0x2000;
|
|
MemMap[Slot+1][0][5]=ROMData[Slot]+P3*0x2000;
|
|
ROMMapper[Slot][0]=P0;
|
|
ROMMapper[Slot][1]=P1;
|
|
ROMMapper[Slot][2]=P2;
|
|
ROMMapper[Slot][3]=P3;
|
|
}
|
|
|
|
/** VDPOut() *************************************************/
|
|
/** Write value into a given VDP register. **/
|
|
/*************************************************************/
|
|
void VDPOut(register byte R,register byte V)
|
|
{
|
|
register byte J;
|
|
|
|
switch(R)
|
|
{
|
|
case 0: /* Reset HBlank interrupt if disabled */
|
|
if((VDPStatus[1]&0x01)&&!(V&0x10))
|
|
{
|
|
VDPStatus[1]&=0xFE;
|
|
SetIRQ(~INT_IE1);
|
|
}
|
|
/* Set screen mode */
|
|
if(VDP[0]!=V) { VDP[0]=V;SetScreen(); }
|
|
break;
|
|
case 1: /* Set/Reset VBlank interrupt if enabled or disabled */
|
|
if(VDPStatus[0]&0x80) SetIRQ(V&0x20? INT_IE0:~INT_IE0);
|
|
/* Set screen mode */
|
|
if(VDP[1]!=V) { VDP[1]=V;SetScreen(); }
|
|
break;
|
|
case 2: J=(ScrMode>6)&&(ScrMode!=MAXSCREEN+1)? 11:10;
|
|
ChrTab = VRAM+((int)(V&MSK[ScrMode].R2)<<J);
|
|
ChrTabM = ((int)(V|~MSK[ScrMode].M2)<<J)|((1<<J)-1);
|
|
break;
|
|
case 3: ColTab = VRAM+((int)(V&MSK[ScrMode].R3)<<6)+((int)VDP[10]<<14);
|
|
ColTabM = ((int)(V|~MSK[ScrMode].M3)<<6)|0x1C03F;
|
|
break;
|
|
case 4: ChrGen = VRAM+((int)(V&MSK[ScrMode].R4)<<11);
|
|
ChrGenM = ((int)(V|~MSK[ScrMode].M4)<<11)|0x007FF;
|
|
break;
|
|
case 5: SprTab = VRAM+((int)(V&MSK[ScrMode].R5)<<7)+((int)VDP[11]<<15);
|
|
SprTabM = ((int)(V|~MSK[ScrMode].M5)<<7)|0x1807F;
|
|
break;
|
|
case 6: V&=0x3F;SprGen=VRAM+((int)V<<11);break;
|
|
case 7: FGColor=V>>4;BGColor=V&0x0F;break;
|
|
case 10: V&=0x07;
|
|
ColTab=VRAM+((int)(VDP[3]&MSK[ScrMode].R3)<<6)+((int)V<<14);
|
|
break;
|
|
case 11: V&=0x03;
|
|
SprTab=VRAM+((int)(VDP[5]&MSK[ScrMode].R5)<<7)+((int)V<<15);
|
|
break;
|
|
case 14: V&=VRAMPages-1;VPAGE=VRAM+((int)V<<14);
|
|
break;
|
|
case 15: V&=0x0F;break;
|
|
case 16: V&=0x0F;PKey=1;break;
|
|
case 17: V&=0xBF;break;
|
|
case 25: VDP[25]=V;
|
|
SetScreen();
|
|
break;
|
|
case 44: VDPWrite(V);break;
|
|
case 46: VDPDraw(V);break;
|
|
}
|
|
|
|
/* Write value into a register */
|
|
VDP[R]=V;
|
|
}
|
|
|
|
/** Printer() ************************************************/
|
|
/** Send a character to the printer. **/
|
|
/*************************************************************/
|
|
#ifdef unused
|
|
void Printer(byte V) { fputc(V,PrnStream); }
|
|
#else
|
|
void Printer(byte V) { }
|
|
#endif
|
|
|
|
/** PPIOut() *************************************************/
|
|
/** This function is called on each write to PPI to make **/
|
|
/** key click sound, motor relay clicks, and so on. **/
|
|
/*************************************************************/
|
|
void PPIOut(register byte New,register byte Old)
|
|
{
|
|
/* Keyboard click bit */
|
|
if((New^Old)&0x80) Drum(DRM_CLICK,64);
|
|
/* Motor relay bit */
|
|
if((New^Old)&0x10) Drum(DRM_CLICK,255);
|
|
}
|
|
|
|
byte RTCIn(register byte R)
|
|
{
|
|
register byte J;
|
|
|
|
/* Only 16 registers/mode */
|
|
R&=0x0F;
|
|
|
|
/* Bank mode 0..3 */
|
|
J=RTCMode&0x03;
|
|
|
|
if(R>12) J=R==13? RTCMode:NORAM;
|
|
else
|
|
if(J) J=RTC[J][R];
|
|
else
|
|
{
|
|
#ifdef unused
|
|
static time_t PrevTime;
|
|
static struct tm TM;
|
|
|
|
/* Retrieve system time if any time passed */
|
|
time_t CurTime=time(NULL);
|
|
if(CurTime!=PrevTime)
|
|
{
|
|
TM=*localtime(&CurTime);
|
|
PrevTime=CurTime;
|
|
}
|
|
|
|
/* Parse contents of last retrieved TM */
|
|
switch(R)
|
|
{
|
|
case 0: J=TM.tm_sec%10;break;
|
|
case 1: J=TM.tm_sec/10;break;
|
|
case 2: J=TM.tm_min%10;break;
|
|
case 3: J=TM.tm_min/10;break;
|
|
case 4: J=TM.tm_hour%10;break;
|
|
case 5: J=TM.tm_hour/10;break;
|
|
case 6: J=TM.tm_wday;break;
|
|
case 7: J=TM.tm_mday%10;break;
|
|
case 8: J=TM.tm_mday/10;break;
|
|
case 9: J=(TM.tm_mon+1)%10;break;
|
|
case 10: J=(TM.tm_mon+1)/10;break;
|
|
case 11: J=(TM.tm_year-80)%10;break;
|
|
case 12: J=((TM.tm_year-80)/10)%10;break;
|
|
default: J=0x0F;break;
|
|
}
|
|
#endif
|
|
}
|
|
/* Four upper bits are always high */
|
|
return(J|0xF0);
|
|
}
|
|
|
|
/** LoopZ80() ************************************************/
|
|
/** Refresh screen, check keyboard and sprites. Call this **/
|
|
/** function on each interrupt. **/
|
|
/*************************************************************/
|
|
word LoopZ80(register Z80 *R, int * ras)
|
|
//word LoopZ80(Z80 *R)
|
|
{
|
|
static byte BFlag=0;
|
|
static byte BCount=0;
|
|
static int UCount=1;
|
|
static byte ACount=0;
|
|
static byte Drawing=0;
|
|
register int J;
|
|
|
|
/* Flip HRefresh bit */
|
|
VDPStatus[2]^=0x20;
|
|
|
|
/* If HRefresh is now in progress... */
|
|
if(!(VDPStatus[2]&0x20))
|
|
{
|
|
/* HRefresh takes most of the scanline */
|
|
R->IPeriod=!ScrMode||(ScrMode==MAXSCREEN+1)? CPU_H240:CPU_H256;
|
|
|
|
/* New scanline */
|
|
ScanLine=ScanLine<(PALVideo? 312:261)? ScanLine+1:0;
|
|
|
|
/* If first scanline of the screen... */
|
|
if(!ScanLine)
|
|
{
|
|
/* Drawing now... */
|
|
Drawing=1;
|
|
|
|
/* Reset VRefresh bit */
|
|
VDPStatus[2]&=0xBF;
|
|
|
|
/* Refresh display */
|
|
if(UCount) UCount--;
|
|
else
|
|
{
|
|
UCount=UPeriod-1;
|
|
//RefreshScreen();
|
|
}
|
|
|
|
/* Blinking for TEXT80 */
|
|
if(BCount) BCount--;
|
|
else
|
|
{
|
|
BFlag=!BFlag;
|
|
if(!VDP[13]) { XFGColor=FGColor;XBGColor=BGColor; }
|
|
else
|
|
{
|
|
BCount=(BFlag? VDP[13]&0x0F:VDP[13]>>4)*10;
|
|
if(BCount)
|
|
{
|
|
if(BFlag) { XFGColor=FGColor;XBGColor=BGColor; }
|
|
else { XFGColor=VDP[12]>>4;XBGColor=VDP[12]&0x0F; }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Line coincidence is active at 0..255 */
|
|
/* in PAL and 0..234/244 in NTSC */
|
|
J=PALVideo? 256:ScanLines212? 245:235;
|
|
|
|
/* When reaching end of screen, reset line coincidence */
|
|
if(ScanLine==J)
|
|
{
|
|
VDPStatus[1]&=0xFE;
|
|
SetIRQ(~INT_IE1);
|
|
}
|
|
|
|
/* When line coincidence is active... */
|
|
if(ScanLine<J)
|
|
{
|
|
/* Line coincidence processing */
|
|
J=(((ScanLine+VScroll)&0xFF)-VDP[19])&0xFF;
|
|
if(J==2)
|
|
{
|
|
/* Set HBlank flag on line coincidence */
|
|
VDPStatus[1]|=0x01;
|
|
/* Generate IE1 interrupt */
|
|
if(VDP[0]&0x10) SetIRQ(INT_IE1);
|
|
}
|
|
else
|
|
{
|
|
/* Reset flag immediately if IE1 interrupt disabled */
|
|
if(!(VDP[0]&0x10)) VDPStatus[1]&=0xFE;
|
|
}
|
|
}
|
|
|
|
/* Return whatever interrupt is pending */
|
|
R->IRequest=IRQPending? INT_IRQ:INT_NONE;
|
|
return(R->IRequest);
|
|
}
|
|
|
|
/*********************************/
|
|
/* We come here for HBlanks only */
|
|
/*********************************/
|
|
|
|
/* HBlank takes HPeriod-HRefresh */
|
|
R->IPeriod=!ScrMode||(ScrMode==MAXSCREEN+1)? CPU_H240:CPU_H256;
|
|
R->IPeriod=HPeriod-R->IPeriod;
|
|
|
|
/* If last scanline of VBlank, see if we need to wait more */
|
|
J=PALVideo? 313:262;
|
|
if(ScanLine>=J-1)
|
|
{
|
|
J*=CPU_HPERIOD;
|
|
if(VPeriod>J) R->IPeriod+=VPeriod-J;
|
|
}
|
|
|
|
/* If first scanline of the bottom border... */
|
|
if(ScanLine==(ScanLines212? 212:192)) Drawing=0;
|
|
|
|
/* If first scanline of VBlank... */
|
|
J=PALVideo? (ScanLines212? 212+42:192+52):(ScanLines212? 212+18:192+28);
|
|
if(!Drawing&&(ScanLine==J))
|
|
{
|
|
/* Set VBlank bit, set VRefresh bit */
|
|
VDPStatus[0]|=0x80;
|
|
VDPStatus[2]|=0x40;
|
|
|
|
/* Generate VBlank interrupt */
|
|
if(VDP[1]&0x20) SetIRQ(INT_IE0);
|
|
}
|
|
|
|
/* Run V9938 engine */
|
|
LoopVDP();
|
|
|
|
/* Refresh scanline, possibly with the overscan */
|
|
if(!UCount&&Drawing&&(ScanLine<256))
|
|
{
|
|
if(!ModeYJK||(ScrMode<7)||(ScrMode>8))
|
|
(RefreshLine[ScrMode])(ScanLine);
|
|
else
|
|
if(ModeYAE) RefreshLine10(ScanLine);
|
|
else RefreshLine12(ScanLine);
|
|
|
|
emu_DrawLinePal16(&linebuffer[0], WIDTH , HEIGHT, ScanLine+FirstLine);
|
|
}
|
|
|
|
/* Keyboard, sound, and other stuff always runs at line 192 */
|
|
/* This way, it can't be shut off by overscan tricks (Maarten) */
|
|
if(ScanLine==192)
|
|
{
|
|
/* Check sprites and set Collision, 5Sprites, 5thSprite bits */
|
|
if(!SpritesOFF&&ScrMode&&(ScrMode<MAXSCREEN+1)) CheckSprites();
|
|
|
|
/* Count MIDI ticks */
|
|
#ifdef unused
|
|
MIDITicks(VPeriod/CPU_CLOCK);
|
|
#endif
|
|
/* Update AY8910 state every VPeriod/CPU_CLOCK milliseconds */
|
|
Loop8910(&PSG,VPeriod/CPU_CLOCK);
|
|
|
|
/* Flush changes to the sound channels */
|
|
Sync8910(&PSG,AY8910_FLUSH|(UseDrums? AY8910_DRUMS:0));
|
|
SyncSCC(&SCChip,SCC_FLUSH);
|
|
Sync2413(&OPLL,YM2413_FLUSH);
|
|
|
|
/* Check keyboard */
|
|
Keyboard();
|
|
|
|
/* Exit emulation if requested */
|
|
//if(ExitNow) return(INT_QUIT);
|
|
|
|
*ras = 1;
|
|
|
|
/* Autofire emulation */
|
|
ACount=(ACount+1)&0x07;
|
|
if((ACount>3)&&AutoFire) KeyMap[8]|=0x01;
|
|
}
|
|
|
|
/* Return whatever interrupt is pending */
|
|
R->IRequest=IRQPending? INT_IRQ:INT_NONE;
|
|
return(R->IRequest);
|
|
}
|
|
|
|
/** CheckSprites() *******************************************/
|
|
/** Check for sprite collisions and 5th/9th sprite in a **/
|
|
/** row. **/
|
|
/*************************************************************/
|
|
void CheckSprites(void)
|
|
{
|
|
register word LS,LD;
|
|
register byte DH,DV,*PS,*PD,*T;
|
|
byte I,J,N,M,*S,*D;
|
|
|
|
/* Clear 5Sprites, Collision, and 5thSprite bits */
|
|
VDPStatus[0]=(VDPStatus[0]&0x9F)|0x1F;
|
|
|
|
for(N=0,S=SprTab;(N<32)&&(S[0]!=208);N++,S+=4);
|
|
M=SolidColor0;
|
|
|
|
if(Sprites16x16)
|
|
{
|
|
for(J=0,S=SprTab;J<N;J++,S+=4)
|
|
if((S[3]&0x0F)||M)
|
|
for(I=J+1,D=S+4;I<N;I++,D+=4)
|
|
if((D[3]&0x0F)||M)
|
|
{
|
|
DV=S[0]-D[0];
|
|
if((DV<16)||(DV>240))
|
|
{
|
|
DH=S[1]-D[1];
|
|
if((DH<16)||(DH>240))
|
|
{
|
|
PS=SprGen+((int)(S[2]&0xFC)<<3);
|
|
PD=SprGen+((int)(D[2]&0xFC)<<3);
|
|
if(DV<16) PD+=DV; else { DV=256-DV;PS+=DV; }
|
|
if(DH>240) { DH=256-DH;T=PS;PS=PD;PD=T; }
|
|
while(DV<16)
|
|
{
|
|
LS=((word)*PS<<8)+*(PS+16);
|
|
LD=((word)*PD<<8)+*(PD+16);
|
|
if(LD&(LS>>DH)) break;
|
|
else { DV++;PS++;PD++; }
|
|
}
|
|
if(DV<16) { VDPStatus[0]|=0x20;return; }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for(J=0,S=SprTab;J<N;J++,S+=4)
|
|
if((S[3]&0x0F)||M)
|
|
for(I=J+1,D=S+4;I<N;I++,D+=4)
|
|
if((D[3]&0x0F)||M)
|
|
{
|
|
DV=S[0]-D[0];
|
|
if((DV<8)||(DV>248))
|
|
{
|
|
DH=S[1]-D[1];
|
|
if((DH<8)||(DH>248))
|
|
{
|
|
PS=SprGen+((int)S[2]<<3);
|
|
PD=SprGen+((int)D[2]<<3);
|
|
if(DV<8) PD+=DV; else { DV=256-DV;PS+=DV; }
|
|
if(DH>248) { DH=256-DH;T=PS;PS=PD;PD=T; }
|
|
while((DV<8)&&!(*PD&(*PS>>DH))) { DV++;PS++;PD++; }
|
|
if(DV<8) { VDPStatus[0]|=0x20;return; }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/** GuessROM() ***********************************************/
|
|
/** Guess MegaROM mapper of a ROM. **/
|
|
/*************************************************************/
|
|
int GuessROM(const byte *Buf,int Size)
|
|
{
|
|
int J,I,ROMCount[6];
|
|
|
|
/* Clear all counters */
|
|
for(J=0;J<6;J++) ROMCount[J]=1;
|
|
ROMCount[0]+=1; /* Mapper #0 is default */
|
|
ROMCount[4]-=1; /* #5 preferred over #4 */
|
|
|
|
/* Count occurences of characteristic addresses */
|
|
for(J=0;J<Size-2;J++)
|
|
{
|
|
I=Buf[J]+((int)Buf[J+1]<<8)+((int)Buf[J+2]<<16);
|
|
switch(I)
|
|
{
|
|
case 0x500032: ROMCount[2]++;break;
|
|
case 0x900032: ROMCount[2]++;break;
|
|
case 0xB00032: ROMCount[2]++;break;
|
|
case 0x400032: ROMCount[3]++;break;
|
|
case 0x800032: ROMCount[3]++;break;
|
|
case 0xA00032: ROMCount[3]++;break;
|
|
case 0x680032: ROMCount[4]++;break;
|
|
case 0x780032: ROMCount[4]++;break;
|
|
case 0x600032: ROMCount[3]++;ROMCount[4]++;ROMCount[5]++;break;
|
|
case 0x700032: ROMCount[2]++;ROMCount[4]++;ROMCount[5]++;break;
|
|
case 0x77FF32: ROMCount[5]++;break;
|
|
}
|
|
}
|
|
|
|
/* Find which mapper type got more hits */
|
|
for(I=0,J=0;J<6;J++)
|
|
if(ROMCount[J]>ROMCount[I]) I=J;
|
|
|
|
/* Return the most likely mapper type */
|
|
return(I);
|
|
}
|