kolibrios/programs/emulator/dgen-sdl-1.33/cz80/cz80.c

505 lines
10 KiB
C
Raw Normal View History

/********************************************************************************/
/* */
/* CZ80 (Z80 CPU emulator) version 0.91 */
/* Compiled with Dev-C++ */
/* Copyright 2004-2005 Stephane Dallongeville */
/* */
/********************************************************************************/
//#ifdef CPUZ80_CZ80_CORE
#include <stdio.h>
#include <string.h>
#include <stddef.h>
#include "cz80.h"
// include macro file
//////////////////////
#include "cz80.inc"
// shared global variable
//////////////////////////
cz80_struc CZ80;
static uint8_t SZXY[256]; // zero and sign flags
static uint8_t SZXYP[256]; // zero, sign and parity flags
static uint8_t SZXY_BIT[256]; // zero, sign and parity/overflow (=zero) flags for BIT opcode
static uint8_t SZXYHV_inc[256]; // zero, sign, half carry and overflow flags INC R8
static uint8_t SZXYHV_dec[256]; // zero, sign, half carry and overflow flags DEC R8
// prototype
/////////////
uint8_t FASTCALL Cz80_Read_Dummy(void *ctx, const uint16_t adr);
void FASTCALL Cz80_Write_Dummy(void *ctx, const uint16_t adr, uint8_t data);
uint8_t FASTCALL Cz80_Interrupt_Ack_Dummy(void *ctx, uint8_t param);
void FASTCALL Cz80_RetI_Dummy(void *ctx);
// core main functions
///////////////////////
void Cz80_Init(cz80_struc *cpu)
{
unsigned int i, j, p;
memset(cpu, 0, sizeof(cz80_struc));
// flags tables initialisation
for (i = 0; i < 256; i++)
{
SZXY[i] = i & (CZ80_SF | CZ80_YF | CZ80_XF);
if (!i) SZXY[i] |= CZ80_ZF;
SZXY_BIT[i] = i & (CZ80_SF | CZ80_YF | CZ80_XF);
if (!i) SZXY_BIT[i] |= CZ80_ZF | CZ80_PF;
for (j = 0, p = 0; j < 8; j++) if (i & (1 << j)) p++;
SZXYP[i] = SZXY[i];
if (!(p & 1)) SZXYP[i] |= CZ80_PF;
SZXYHV_inc[i] = SZXY[i];
if(i == 0x80) SZXYHV_inc[i] |= CZ80_VF;
if((i & 0x0F) == 0x00) SZXYHV_inc[i] |= CZ80_HF;
SZXYHV_dec[i] = SZXY[i] | CZ80_NF;
if (i == 0x7F) SZXYHV_dec[i] |= CZ80_VF;
if ((i & 0x0F) == 0x0F) SZXYHV_dec[i] |= CZ80_HF;
}
Cz80_Set_Fetch(cpu, 0x0000, 0xFFFF, NULL);
Cz80_Set_ReadB(cpu, Cz80_Read_Dummy);
Cz80_Set_WriteB(cpu, Cz80_Write_Dummy);
cpu->Interrupt_Ack = Cz80_Interrupt_Ack_Dummy;
cpu->RetI = Cz80_RetI_Dummy;
}
uint8_t Cz80_Reset(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
memset(CPU, 0, offsetof(cz80_struc, CycleSup));
Cz80_Set_PC(CPU, 0);
zIX = 0xFFFF;
zIY = 0xFFFF;
#if CZ80_DEBUG
zF = CZ80_ZF;
#else
zSP = 0xFFFF;
zFA = 0xFFFF;
#endif
return CPU->Status;
}
/////////////////////////////////
#include "cz80exec.inc"
/////////////////////////////////
void FASTCALL Cz80_Set_IRQ(cz80_struc *cpu, uint8_t vector)
{
cpu->IntVect = vector;
cpu->Status |= CZ80_HAS_INT;
cpu->CycleSup = cpu->CycleIO;
cpu->CycleIO = 0;
}
void FASTCALL Cz80_Set_NMI(cz80_struc *cpu)
{
cpu->Status |= CZ80_HAS_NMI;
cpu->CycleSup = cpu->CycleIO;
cpu->CycleIO = 0;
}
void FASTCALL Cz80_Clear_IRQ(cz80_struc *cpu)
{
cpu->Status &= ~CZ80_HAS_INT;
}
void FASTCALL Cz80_Clear_NMI(cz80_struc *cpu)
{
cpu->Status &= ~CZ80_HAS_NMI;
}
/////////////////////////////////
int FASTCALL Cz80_Get_CycleToDo(cz80_struc *cpu)
{
if (!(cpu->Status & CZ80_RUNNING)) return -1;
return cpu->CycleToDo;
}
int FASTCALL Cz80_Get_CycleRemaining(cz80_struc *cpu)
{
if (!(cpu->Status & CZ80_RUNNING)) return -1;
return (cpu->CycleIO + cpu->CycleSup);
}
int FASTCALL Cz80_Get_CycleDone(cz80_struc *cpu)
{
if (!(cpu->Status & CZ80_RUNNING)) return -1;
return (cpu->CycleToDo - (cpu->CycleIO + cpu->CycleSup));
}
void FASTCALL Cz80_Release_Cycle(cz80_struc *cpu)
{
if (cpu->Status & CZ80_RUNNING) cpu->CycleIO = cpu->CycleSup = 0;
}
void FASTCALL Cz80_Add_Cycle(cz80_struc *cpu, unsigned int cycle)
{
if (cpu->Status & CZ80_RUNNING) cpu->CycleIO -= cycle;
}
// Read / Write dummy functions
////////////////////////////////
uint8_t FASTCALL Cz80_Read_Dummy(void *ctx, const uint16_t adr)
{
(void)ctx;
(void)adr;
return 0;
}
void FASTCALL Cz80_Write_Dummy(void *ctx, const uint16_t adr, uint8_t data)
{
(void)ctx;
(void)adr;
(void)data;
}
uint8_t FASTCALL Cz80_Interrupt_Ack_Dummy(void *ctx, uint8_t param)
{
(void)ctx;
(void)param;
// return vector
return -1;
}
void FASTCALL Cz80_RetI_Dummy(void *ctx)
{
(void)ctx;
}
// Read / Write core functions
///////////////////////////////
void Cz80_Set_Ctx(cz80_struc *cpu, void *ctx)
{
cpu->ctx = ctx;
}
uint8_t Cz80_Read_Byte(cz80_struc *cpu, uint16_t adr)
{
return cpu->Read_Byte(cpu->ctx, adr);
}
uint16_t Cz80_Read_Word(cz80_struc *cpu, uint16_t adr)
{
#if CZ80_USE_WORD_HANDLER
return cpu->Read_Word(cpu->ctx, adr);
#elif CZ80_LITTLE_ENDIAN
return (cpu->Read_Byte(cpu->ctx, adr) |
(cpu->Read_Byte(cpu->ctx, (adr + 1)) << 8));
#else
return ((cpu->Read_Byte(cpu->ctx, adr) << 8) |
cpu->Read_Byte(cpu->ctx, (adr + 1)));
#endif
}
void Cz80_Write_Byte(cz80_struc *cpu, uint16_t adr, uint8_t data)
{
cpu->Write_Byte(cpu->ctx, adr, data);
}
void Cz80_Write_Word(cz80_struc *cpu, uint16_t adr, uint16_t data)
{
#if CZ80_USE_WORD_HANDLER
cpu->Write_Word(cpu->ctx, adr, data);
#elif CZ80_LITTLE_ENDIAN
cpu->Write_Byte(cpu->ctx, adr, (data & 0xFF));
cpu->Write_Byte(cpu->ctx, (adr + 1), (data >> 8));
#else
cpu->Write_Byte(cpu->ctx, adr, (data >> 8));
cpu->Write_Byte(cpu->ctx, (adr + 1), (data & 0xFF));
#endif
}
// setting core functions
//////////////////////////
void Cz80_Set_Fetch(cz80_struc *cpu, uint16_t low_adr, uint16_t high_adr, void *fetch_adr)
{
uint16_t i, j;
i = low_adr >> CZ80_FETCH_SFT;
j = high_adr >> CZ80_FETCH_SFT;
fetch_adr = (void *)((uintptr_t)fetch_adr - (i << CZ80_FETCH_SFT));
while (i <= j) cpu->Fetch[i++] = (uint8_t*) fetch_adr;
}
void Cz80_Set_ReadB(cz80_struc *cpu, CZ80_READ *Func)
{
cpu->Read_Byte = Func;
}
void Cz80_Set_WriteB(cz80_struc *cpu, CZ80_WRITE *Func)
{
cpu->Write_Byte = Func;
}
#if CZ80_USE_WORD_HANDLER
void Cz80_Set_ReadW(cz80_struc *cpu, CZ80_READ_WORD *Func)
{
cpu->Read_Word = Func;
}
void Cz80_Set_WriteW(cz80_struc *cpu, CZ80_WRITE_WORD *Func)
{
cpu->Write_Word = Func;
}
#endif
void Cz80_Set_INPort(cz80_struc *cpu, CZ80_READ *Func)
{
cpu->IN_Port = Func;
}
void Cz80_Set_OUTPort(cz80_struc *cpu, CZ80_WRITE *Func)
{
cpu->OUT_Port = Func;
}
void Cz80_Set_IRQ_Callback(cz80_struc *cpu, CZ80_INT_CALLBACK *Func)
{
cpu->Interrupt_Ack = Func;
}
void Cz80_Set_RETI_Callback(cz80_struc *cpu, CZ80_RETI_CALLBACK *Func)
{
cpu->RetI = Func;
}
// externals main functions
////////////////////////////
uint16_t FASTCALL Cz80_Get_BC(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zBC;
}
uint16_t FASTCALL Cz80_Get_DE(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zDE;
}
uint16_t FASTCALL Cz80_Get_HL(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zHL;
}
uint16_t FASTCALL Cz80_Get_AF(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return (zF | (zA << 8));
}
uint16_t FASTCALL Cz80_Get_BC2(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zBC2;
}
uint16_t FASTCALL Cz80_Get_DE2(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zDE2;
}
uint16_t FASTCALL Cz80_Get_HL2(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zHL2;
}
uint16_t FASTCALL Cz80_Get_AF2(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return (zF2 | (zA2 << 8));
}
uint16_t FASTCALL Cz80_Get_IX(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zIX;
}
uint16_t FASTCALL Cz80_Get_IY(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zIY;
}
uint16_t FASTCALL Cz80_Get_SP(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zSP;
}
uint16_t FASTCALL Cz80_Get_PC(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return CPU->PC;
}
uint16_t FASTCALL Cz80_Get_R(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zR;
}
uint16_t FASTCALL Cz80_Get_IFF(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
uint16_t value = 0;
if (zIFF1 & CZ80_IFF) value |= 1;
if (zIFF2 & CZ80_IFF) value |= 2;
return value;
}
uint8_t FASTCALL Cz80_Get_IM(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zIM;
}
uint8_t FASTCALL Cz80_Get_I(cz80_struc *cpu)
{
cz80_struc *CPU = cpu;
return zI;
}
void FASTCALL Cz80_Set_BC(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zBC = value;
}
void FASTCALL Cz80_Set_DE(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zDE = value;
}
void FASTCALL Cz80_Set_HL(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zHL = value;
}
void FASTCALL Cz80_Set_AF(cz80_struc *cpu, uint16_t val)
{
cz80_struc *CPU = cpu;
zF = val;
zA = val >> 8;
}
void FASTCALL Cz80_Set_BC2(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zBC2 = value;
}
void FASTCALL Cz80_Set_DE2(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zDE2 = value;
}
void FASTCALL Cz80_Set_HL2(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zHL2 = value;
}
void FASTCALL Cz80_Set_AF2(cz80_struc *cpu, uint16_t val)
{
cz80_struc *CPU = cpu;
zF2 = val;
zA2 = val >> 8;
}
void FASTCALL Cz80_Set_IX(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zIX = value;
}
void FASTCALL Cz80_Set_IY(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zIY = value;
}
void FASTCALL Cz80_Set_SP(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zSP = value;
}
void FASTCALL Cz80_Set_PC(cz80_struc *cpu, uint16_t val)
{
cpu->BasePC = (uintptr_t) cpu->Fetch[val >> CZ80_FETCH_SFT];
cpu->PC = val;
}
void FASTCALL Cz80_Set_R(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zR = value & 0xFF;
zR2 = value & 0x80;
}
void FASTCALL Cz80_Set_IFF(cz80_struc *cpu, uint16_t value)
{
cz80_struc *CPU = cpu;
zIFF = 0;
if (value & 1) zIFF1 = CZ80_IFF;
if (value & 2) zIFF2 = CZ80_IFF;
}
void FASTCALL Cz80_Set_IM(cz80_struc *cpu, uint8_t value)
{
cz80_struc *CPU = cpu;
zIM = value & 3;
}
void FASTCALL Cz80_Set_I(cz80_struc *cpu, uint8_t value)
{
cz80_struc *CPU = cpu;
zI = value & 0xFF;
}
//#endif // CPUZ80_CZ80_CORE