aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Makefile17
-rw-r--r--Makefile.common9
-rw-r--r--jni/Android.mk1
-rw-r--r--libretro.c128
-rw-r--r--source/apu.c47
-rw-r--r--source/apu.h4
-rw-r--r--source/c4.h1
-rw-r--r--source/c4emu.c35
-rw-r--r--source/catsfc_griffin.c35
-rw-r--r--source/changes.txt2155
-rw-r--r--source/cheats.c271
-rw-r--r--source/cheats.h6
-rw-r--r--source/cheats2.c29
-rw-r--r--source/clip.c157
-rw-r--r--source/cpuaddr.h13
-rw-r--r--source/cpuexec.c108
-rw-r--r--source/cpuexec.h7
-rw-r--r--source/cpumacro.h128
-rw-r--r--source/cpuops.c302
-rw-r--r--source/display.h1
-rw-r--r--source/dma.c133
-rw-r--r--source/dma.h1
-rw-r--r--source/dsp1.c196
-rw-r--r--source/dsp1.h1
-rw-r--r--source/dsp1emu.c138
-rw-r--r--source/dsp2emu.c58
-rw-r--r--source/dsp4.h109
-rw-r--r--source/dsp4emu.c325
-rw-r--r--source/fxemu.c63
-rw-r--r--source/fxemu.h26
-rw-r--r--source/fxinst.c48
-rw-r--r--source/fxinst.h56
-rw-r--r--source/getset.h108
-rw-r--r--source/gfx.c1382
-rw-r--r--source/gfx.h18
-rw-r--r--source/globals.c18
-rw-r--r--source/hardware.txt502
-rw-r--r--source/memmap.c266
-rw-r--r--source/memmap.h4
-rw-r--r--source/obc1.c1
-rw-r--r--source/obc1.h3
-rw-r--r--source/port.h7
-rw-r--r--source/ppu.c396
-rw-r--r--source/ppu.h36
-rw-r--r--source/problems.txt459
-rw-r--r--source/sa1.c85
-rw-r--r--source/sar.h2
-rw-r--r--source/sdd1.c2
-rw-r--r--source/sdd1.h5
-rw-r--r--source/sdd1emu.c70
-rw-r--r--source/sdd1emu.h1
-rw-r--r--source/seta.h3
-rw-r--r--source/seta010.c64
-rw-r--r--source/seta011.c50
-rw-r--r--source/seta018.c33
-rw-r--r--source/snes9x.h43
-rw-r--r--source/soundux.c23
-rw-r--r--source/spc700.c627
-rw-r--r--source/spc700.h6
-rw-r--r--source/spc7110.c1333
-rw-r--r--source/spc7110.h19
-rw-r--r--source/spc7110dec.c611
-rw-r--r--source/spc7110dec.h28
-rw-r--r--source/srtc.c143
-rw-r--r--source/srtc.h7
-rw-r--r--source/tile.c256
-rw-r--r--source/tile.h28
67 files changed, 2883 insertions, 8364 deletions
diff --git a/Makefile b/Makefile
index 0d7bd9a..9d6b5c4 100644
--- a/Makefile
+++ b/Makefile
@@ -1,6 +1,5 @@
DEBUG = 0
PERF_TEST = 0
-HAVE_GRIFFIN = 0
LOAD_FROM_MEMORY_TEST = 1
USE_BLARGG_APU = 0
@@ -97,15 +96,13 @@ else
endif
# Theos iOS
else ifeq ($(platform), theos_ios)
-DEPLOYMENT_IOSVERSION = 5.0
-TARGET = iphone:latest:$(DEPLOYMENT_IOSVERSION)
-ARCHS = armv7 armv7s
-TARGET_IPHONEOS_DEPLOYMENT_VERSION=$(DEPLOYMENT_IOSVERSION)
-THEOS_BUILD_DIR := objs
-include $(THEOS)/makefiles/common.mk
-
-LIBRARY_NAME = $(TARGET_NAME)_libretro_ios
-
+ DEPLOYMENT_IOSVERSION = 5.0
+ TARGET = iphone:latest:$(DEPLOYMENT_IOSVERSION)
+ ARCHS = armv7 armv7s
+ TARGET_IPHONEOS_DEPLOYMENT_VERSION=$(DEPLOYMENT_IOSVERSION)
+ THEOS_BUILD_DIR := objs
+ include $(THEOS)/makefiles/common.mk
+ LIBRARY_NAME = $(TARGET_NAME)_libretro_ios
else ifeq ($(platform), qnx)
TARGET := $(TARGET_NAME)_libretro_$(platform).so
fpic := -fPIC
diff --git a/Makefile.common b/Makefile.common
index e62fabb..21e319e 100644
--- a/Makefile.common
+++ b/Makefile.common
@@ -2,13 +2,6 @@ LIBRETRO_COMM_DIR := $(LIBRETRO_DIR)/libretro-common
INCFLAGS := -I$(CORE_DIR) -I$(LIBRETRO_DIR) -I$(LIBRETRO_COMM_DIR)/include
-ifeq ($(HAVE_GRIFFIN), 1)
-SOURCES_C := $(CORE_DIR)/catsfc_griffin.c
-SOURCES_C += \
- $(CORE_DIR)/sa1.c \
- $(CORE_DIR)/sa1cpu.c \
- $(CORE_DIR)/apu_blargg.c
-else
SOURCES_C := \
$(CORE_DIR)/apu.c \
$(CORE_DIR)/c4.c \
@@ -40,11 +33,11 @@ SOURCES_C := \
$(CORE_DIR)/soundux.c \
$(CORE_DIR)/spc700.c \
$(CORE_DIR)/spc7110.c \
+ $(CORE_DIR)/spc7110dec.c \
$(CORE_DIR)/srtc.c \
$(CORE_DIR)/tile.c \
$(CORE_DIR)/apu_blargg.c \
$(LIBRETRO_DIR)/libretro.c
-endif
ifeq ($(USE_BLARGG_APU),1)
FLAGS += -DUSE_BLARGG_APU
diff --git a/jni/Android.mk b/jni/Android.mk
index 21da6bb..e832e41 100644
--- a/jni/Android.mk
+++ b/jni/Android.mk
@@ -12,7 +12,6 @@ LIBRETRO_DIR := ..
DEBUG = 0
PERF_TEST = 0
-HAVE_GRIFFIN = 0
LOAD_FROM_MEMORY_TEST = 1
USE_BLARGG_APU = 0
diff --git a/libretro.c b/libretro.c
index 1cb7b30..107dbfd 100644
--- a/libretro.c
+++ b/libretro.c
@@ -54,7 +54,9 @@ static int32_t samplerate = (((SNES_CLOCK_SPEED * 6) / (32 * ONE_APU_CYCLE)));
perf_cb.perf_register(&(name)); \
current_ticks = name.total
-#define RETRO_PERFORMANCE_START(name) perf_cb.perf_start(&(name))
+#define RETRO_PERFORMANCE_START(name) \
+ perf_cb.perf_start(&(name))
+
#define RETRO_PERFORMANCE_STOP(name) \
perf_cb.perf_stop(&(name)); \
current_ticks = name.total - current_ticks;
@@ -67,7 +69,6 @@ static int32_t samplerate = (((SNES_CLOCK_SPEED * 6) / (32 * ONE_APU_CYCLE)));
void retro_set_environment(retro_environment_t cb)
{
struct retro_log_callback log;
-
environ_cb = cb;
if (environ_cb(RETRO_ENVIRONMENT_GET_LOG_INTERFACE, &log))
@@ -166,11 +167,6 @@ void S9xInitDisplay(void)
GFX.Delta = (GFX.SubScreen - GFX.Screen) >> 1;
}
-bool S9xInitUpdate()
-{
- return (true);
-}
-
#ifndef __WIN32__
void _makepath(char* path, const char* drive, const char* dir, const char* fname, const char* ext)
{
@@ -265,8 +261,6 @@ void init_sfc_setting(void)
Settings.APUEnabled = true;
Settings.H_Max = SNES_CYCLES_PER_SCANLINE;
- Settings.SkipFrames = AUTO_FRAMERATE;
- Settings.ShutdownMaster = true;
Settings.FrameTimePAL = 20000;
Settings.FrameTimeNTSC = 16667;
Settings.DisableMasterVolume = false;
@@ -274,14 +268,10 @@ void init_sfc_setting(void)
Settings.SuperScope = true;
Settings.MultiPlayer5 = true;
Settings.ControllerOption = SNES_JOYPAD;
-
- Settings.Transparency = true;
#ifdef USE_BLARGG_APU
Settings.SoundSync = false;
#endif
Settings.ApplyCheats = true;
- Settings.StretchScreenshots = 1;
-
Settings.HBlankStart = (256 * Settings.H_Max) / SNES_HCOUNTER_MAX;
}
@@ -303,7 +293,9 @@ void retro_init(void)
{
struct retro_log_callback log;
enum retro_pixel_format rgb565;
- static const struct retro_variable vars[] = {
+
+ static const struct retro_variable vars[] =
+ {
{ "catsfc_VideoMode", "Video Mode; auto|NTSC|PAL" },
{ NULL, NULL },
};
@@ -319,8 +311,7 @@ void retro_init(void)
rgb565 = RETRO_PIXEL_FORMAT_RGB565;
if (environ_cb(RETRO_ENVIRONMENT_SET_PIXEL_FORMAT, &rgb565) && log_cb)
- log_cb(RETRO_LOG_INFO,
- "Frontend supports RGB565 - will use that instead of XRGB1555.\n");
+ log_cb(RETRO_LOG_INFO, "Frontend supports RGB565 - will use that instead of XRGB1555.\n");
init_sfc_setting();
S9xInitMemory();
@@ -340,7 +331,7 @@ void retro_init(void)
void retro_deinit(void)
{
if (Settings.SPC7110)
- (*CleanUp7110)();
+ Del7110Gfx();
S9xDeinitGFX();
S9xDeinitDisplay();
@@ -419,7 +410,6 @@ void retro_run(void)
#ifndef USE_BLARGG_APU
static int16_t audio_buf[2048];
-
samples_to_play += samples_per_frame;
if (samples_to_play > 512)
@@ -441,32 +431,19 @@ void retro_run(void)
#ifdef PSP
static unsigned int __attribute__((aligned(16))) d_list[32];
- void* const texture_vram_p = (void*)(0x44200000 - (512 *
- 512)); // max VRAM address - frame size
-
- sceKernelDcacheWritebackRange(GFX.Screen,
- GFX.Pitch * IPPU.RenderedScreenHeight);
-
+ void* const texture_vram_p = (void*)(0x44200000 - (512 * 512)); // max VRAM address - frame size
+ sceKernelDcacheWritebackRange(GFX.Screen, GFX.Pitch * IPPU.RenderedScreenHeight);
sceGuStart(GU_DIRECT, d_list);
-
- sceGuCopyImage(GU_PSM_4444, 0, 0, IPPU.RenderedScreenWidth,
- IPPU.RenderedScreenHeight, GFX.Pitch >> 1, GFX.Screen, 0,
- 0,
- 512, texture_vram_p);
-
+ sceGuCopyImage(GU_PSM_4444, 0, 0, IPPU.RenderedScreenWidth, IPPU.RenderedScreenHeight, GFX.Pitch >> 1, GFX.Screen, 0, 0, 512, texture_vram_p);
sceGuTexSync();
sceGuTexImage(0, 512, 512, 512, texture_vram_p);
sceGuTexMode(GU_PSM_5551, 0, 0, GU_FALSE);
sceGuTexFunc(GU_TFX_REPLACE, GU_TCC_RGB);
sceGuDisable(GU_BLEND);
-
sceGuFinish();
-
- video_cb(texture_vram_p, IPPU.RenderedScreenWidth, IPPU.RenderedScreenHeight,
- GFX.Pitch);
+ video_cb(texture_vram_p, IPPU.RenderedScreenWidth, IPPU.RenderedScreenHeight, GFX.Pitch);
#else
- video_cb(GFX.Screen, IPPU.RenderedScreenWidth, IPPU.RenderedScreenHeight,
- GFX.Pitch);
+ video_cb(GFX.Screen, IPPU.RenderedScreenWidth, IPPU.RenderedScreenHeight, GFX.Pitch);
#endif
#ifdef FRAMESKIP
@@ -482,67 +459,23 @@ void retro_run(void)
bool S9xReadMousePosition(int32_t which1, int32_t* x, int32_t* y, uint32_t* buttons)
{
- return (false);
+ return false;
}
bool S9xReadSuperScopePosition(int32_t* x, int32_t* y, uint32_t* buttons)
{
- return (true);
+ return true;
}
bool JustifierOffscreen()
{
- return (false);
+ return false;
}
void JustifierButtons(uint32_t* justifiers)
{
}
-char* osd_GetPackDir()
-{
- static char filename[_MAX_PATH];
- memset(filename, 0, _MAX_PATH);
-
- char dir [_MAX_DIR + 1];
- char drive [_MAX_DRIVE + 1];
- char name [_MAX_FNAME + 1];
- char ext [_MAX_EXT + 1];
- _splitpath(Memory.ROMFilename, drive, dir, name, ext);
- _makepath(filename, drive, dir, NULL, NULL);
-
- if (!strncmp((char*)&Memory.ROM [0xffc0], "SUPER POWER LEAG 4 ", 21))
- {
- if (getenv("SPL4PACK"))
- return getenv("SPL4PACK");
- else
- strcat(filename, "/SPL4-SP7");
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "MOMOTETSU HAPPY ", 21))
- {
- if (getenv("MDHPACK"))
- return getenv("MDHPACK");
- else
- strcat(filename, "/SMHT-SP7");
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "HU TENGAI MAKYO ZERO ", 21))
- {
- if (getenv("FEOEZPACK"))
- return getenv("FEOEZPACK");
- else
- strcat(filename, "/FEOEZSP7");
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "JUMP TENGAIMAKYO ZERO", 21))
- {
- if (getenv("SJNSPACK"))
- return getenv("SJNSPACK");
- else
- strcat(filename, "/SJUMPSP7");
- }
- else strcat(filename, "/MISC-SP7");
- return filename;
-}
-
unsigned retro_get_region()
{
return Settings.PAL ? RETRO_REGION_PAL : RETRO_REGION_NTSC;
@@ -577,11 +510,9 @@ void retro_get_system_av_info(struct retro_system_av_info* info)
info->geometry.aspect_ratio = 4.0 / 3.0;
if (!Settings.PAL)
- info->timing.fps = (SNES_CLOCK_SPEED * 6.0 / (SNES_CYCLES_PER_SCANLINE *
- SNES_MAX_NTSC_VCOUNTER));
+ info->timing.fps = (SNES_CLOCK_SPEED * 6.0 / (SNES_CYCLES_PER_SCANLINE * SNES_MAX_NTSC_VCOUNTER));
else
- info->timing.fps = (SNES_CLOCK_SPEED * 6.0 / (SNES_CYCLES_PER_SCANLINE *
- SNES_MAX_PAL_VCOUNTER));
+ info->timing.fps = (SNES_CLOCK_SPEED * 6.0 / (SNES_CYCLES_PER_SCANLINE * SNES_MAX_PAL_VCOUNTER));
info->timing.sample_rate = samplerate;
}
@@ -597,18 +528,16 @@ size_t retro_serialize_size(void)
return sizeof(CPU) + sizeof(ICPU) + sizeof(PPU) + sizeof(DMA) +
0x10000 + 0x20000 + 0x20000 + 0x8000 +
#ifndef USE_BLARGG_APU
- sizeof(APU) + sizeof(IAPU) + 0x10000
+ sizeof(APU) + sizeof(IAPU) + 0x10000 +
#else
- SPC_SAVE_STATE_BLOCK_SIZE
+ SPC_SAVE_STATE_BLOCK_SIZE +
#endif
- + sizeof(SA1) +
- sizeof(s7r) + sizeof(rtc_f9);
+ sizeof(SA1) + sizeof(s7r) + sizeof(rtc_f9);
}
bool retro_serialize(void* data, size_t size)
{
int32_t i;
-
S9xUpdateRTC();
S9xSRTCPreSaveState();
uint8_t* buffer = data;
@@ -662,7 +591,7 @@ bool retro_unserialize(const void* data, size_t size)
S9xReset();
#ifndef USE_BLARGG_APU
uint8_t* IAPU_RAM_current = IAPU.RAM;
- uint32_t IAPU_RAM_offset;
+ uintptr_t IAPU_RAM_offset;
#endif
memcpy(&CPU, buffer, sizeof(CPU));
buffer += sizeof(CPU);
@@ -721,7 +650,6 @@ bool retro_unserialize(const void* data, size_t size)
S9xUnpackStatus();
S9xFixCycles();
S9xReschedule();
-
return true;
}
@@ -741,12 +669,8 @@ void retro_cheat_set(unsigned index, bool enabled, const char* code)
bool sram;
uint8_t bytes[3];//used only by GoldFinger, ignored for now
- if (S9xGameGenieToRaw(code, &address, &val)!=NULL &&
- S9xProActionReplayToRaw(code, &address, &val)!=NULL &&
- S9xGoldFingerToRaw(code, &address, &sram, &val, bytes)!=NULL)
- { // bad code, ignore
- return;
- }
+ if (S9xGameGenieToRaw(code, &address, &val) && S9xProActionReplayToRaw(code, &address, &val) && S9xGoldFingerToRaw(code, &address, &sram, &val, bytes))
+ return; // bad code, ignore
if (index > Cheat.num_cheats)
return; // cheat added in weird order, ignore
if (index == Cheat.num_cheats)
@@ -755,7 +679,6 @@ void retro_cheat_set(unsigned index, bool enabled, const char* code)
Cheat.c[index].address = address;
Cheat.c[index].byte = val;
Cheat.c[index].enabled = enabled;
-
Cheat.c[index].saved = false; // it'll be saved next time cheats run anyways
Settings.ApplyCheats = true;
@@ -764,7 +687,8 @@ void retro_cheat_set(unsigned index, bool enabled, const char* code)
static void init_descriptors(void)
{
- struct retro_input_descriptor desc[] = {
+ struct retro_input_descriptor desc[] =
+ {
{ 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_LEFT, "D-Pad Left" },
{ 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_UP, "D-Pad Up" },
{ 0, RETRO_DEVICE_JOYPAD, 0, RETRO_DEVICE_ID_JOYPAD_DOWN, "D-Pad Down" },
diff --git a/source/apu.c b/source/apu.c
index 9d18242..b051744 100644
--- a/source/apu.c
+++ b/source/apu.c
@@ -37,9 +37,7 @@ uint8_t APUROM [64];
void S9xResetAPU()
{
int32_t i, j;
-
Settings.APUEnabled = true;
-
memset(IAPU.RAM, 0, 0x100);
memset(IAPU.RAM + 0x20, 0xFF, 0x20);
memset(IAPU.RAM + 0x60, 0xFF, 0x20);
@@ -49,7 +47,7 @@ void S9xResetAPU()
for (i = 1; i < 256; i++)
memcpy(IAPU.RAM + (i << 8), IAPU.RAM, 0x100);
- memset(APU.OutPorts, 0, 4);
+ memset(APU.OutPorts, 0, sizeof(APU.OutPorts));
IAPU.DirectPage = IAPU.RAM;
// memmove converted: Different mallocs [Neb]
// DS2 DMA notes: The APU ROM is not 32-byte aligned [Neb]
@@ -61,14 +59,14 @@ void S9xResetAPU()
APU.Cycles = 0;
IAPU.Registers.YA.W = 0;
IAPU.Registers.X = 0;
- IAPU.Registers.S = 0xff;
- IAPU.Registers.P = 0;
+ IAPU.Registers.S = 0xef;
+ IAPU.Registers.P = 0x02;
S9xAPUUnpackStatus();
IAPU.Registers.PC = 0;
IAPU.APUExecuting = Settings.APUEnabled;
IAPU.WaitAddress1 = NULL;
IAPU.WaitAddress2 = NULL;
- IAPU.WaitCounter = 0;
+ IAPU.WaitCounter = 1;
APU.ShowROM = true;
IAPU.RAM [0xf1] = 0x80;
@@ -89,7 +87,7 @@ void S9xResetAPU()
APU.DSP [APU_ENDX] = 0;
APU.DSP [APU_KOFF] = 0;
APU.DSP [APU_KON] = 0;
- APU.DSP [APU_FLG] = APU_MUTE | APU_ECHO_DISABLED;
+ APU.DSP[APU_FLG] = APU_SOFT_RESET | APU_MUTE;
APU.KeyedChannels = 0;
S9xResetSound(true);
@@ -211,7 +209,8 @@ void S9xSetAPUDSP(uint8_t byte)
APU.DSP [APU_ENDX] &= ~mask;
S9xPlaySample(c);
}
- else KeyOn |= mask;
+ else
+ KeyOn |= mask;
}
}
}
@@ -244,7 +243,7 @@ void S9xSetAPUDSP(uint8_t byte)
case APU_P_LOW + 0x50:
case APU_P_LOW + 0x60:
case APU_P_LOW + 0x70:
- S9xSetSoundHertz(reg >> 4, (((int16_t) byte + ((int16_t) APU.DSP [reg + 1] << 8)) & FREQUENCY_MASK) / 8);
+ S9xSetSoundHertz(reg >> 4, ((((int16_t) byte + ((int16_t) APU.DSP [reg + 1] << 8)) & FREQUENCY_MASK) * 32000) >> 12);
break;
case APU_P_HIGH + 0x00:
case APU_P_HIGH + 0x10:
@@ -334,7 +333,7 @@ void S9xFixEnvelope(int32_t channel, uint8_t gain, uint8_t adsr1, uint8_t adsr2)
if(S9xSetSoundMode(channel, MODE_ADSR))
S9xSetSoundADSR(channel, adsr1 & 0xf, (adsr1 >> 4) & 7, adsr2 & 0x1f, (adsr2 >> 5) & 7, 8);
} // Gain mode
- else if ((gain & 0x80) == 0)
+ else if (!(gain & 0x80))
{
if (S9xSetSoundMode(channel, MODE_GAIN))
{
@@ -345,9 +344,7 @@ void S9xFixEnvelope(int32_t channel, uint8_t gain, uint8_t adsr1, uint8_t adsr2)
else if (gain & 0x40)
{
// Increase mode
- if(S9xSetSoundMode(channel, (gain & 0x20) ?
- MODE_INCREASE_BENT_LINE :
- MODE_INCREASE_LINEAR))
+ if(S9xSetSoundMode(channel, (gain & 0x20) ? MODE_INCREASE_BENT_LINE : MODE_INCREASE_LINEAR))
S9xSetEnvelopeRate(channel, gain, 1, 127, (3 << 28) | gain);
}
else if (gain & 0x20)
@@ -364,30 +361,30 @@ void S9xFixEnvelope(int32_t channel, uint8_t gain, uint8_t adsr1, uint8_t adsr2)
void S9xSetAPUControl(uint8_t byte)
{
- if ((byte & 1) != 0 && !APU.TimerEnabled [0])
+ if ((byte & 1) && !APU.TimerEnabled [0])
{
APU.Timer [0] = 0;
IAPU.RAM [0xfd] = 0;
if ((APU.TimerTarget [0] = IAPU.RAM [0xfa]) == 0)
APU.TimerTarget [0] = 0x100;
}
- if ((byte & 2) != 0 && !APU.TimerEnabled [1])
+ if ((byte & 2) && !APU.TimerEnabled [1])
{
APU.Timer [1] = 0;
IAPU.RAM [0xfe] = 0;
if ((APU.TimerTarget [1] = IAPU.RAM [0xfb]) == 0)
APU.TimerTarget [1] = 0x100;
}
- if ((byte & 4) != 0 && !APU.TimerEnabled [2])
+ if ((byte & 4) && !APU.TimerEnabled [2])
{
APU.Timer [2] = 0;
IAPU.RAM [0xff] = 0;
if ((APU.TimerTarget [2] = IAPU.RAM [0xfc]) == 0)
APU.TimerTarget [2] = 0x100;
}
- APU.TimerEnabled [0] = (bool) (byte & 1);
- APU.TimerEnabled [1] = (bool) (byte & 2);
- APU.TimerEnabled [2] = (bool) (byte & 4);
+ APU.TimerEnabled [0] = !!(byte & 1);
+ APU.TimerEnabled [1] = !!(byte & 2);
+ APU.TimerEnabled [2] = !!(byte & 4);
if (byte & 0x10)
IAPU.RAM [0xF4] = IAPU.RAM [0xF5] = 0;
@@ -431,9 +428,8 @@ uint8_t S9xGetAPUDSP()
case APU_OUTX + 0x60:
case APU_OUTX + 0x70:
if(SoundData.channels [reg >> 4].state == SOUND_SILENT)
- return (0);
- return ((SoundData.channels [reg >> 4].sample >> 8) |
- (SoundData.channels [reg >> 4].sample & 0xff));
+ return 0;
+ return (SoundData.channels [reg >> 4].sample >> 8) | (SoundData.channels [reg >> 4].sample & 0xff);
case APU_ENVX + 0x00:
case APU_ENVX + 0x10:
case APU_ENVX + 0x20:
@@ -442,11 +438,14 @@ uint8_t S9xGetAPUDSP()
case APU_ENVX + 0x50:
case APU_ENVX + 0x60:
case APU_ENVX + 0x70:
- return 0;
+ {
+ int32_t eVal = SoundData.channels [reg >> 4].envx;
+ return (eVal > 0x7F) ? 0x7F : (eVal < 0 ? 0 : eVal);
+ }
default:
break;
}
- return (byte);
+ return byte;
}
#endif
diff --git a/source/apu.h b/source/apu.h
index 8944377..0197a20 100644
--- a/source/apu.h
+++ b/source/apu.h
@@ -78,8 +78,8 @@ void S9xSetAPUDSP(uint8_t byte);
uint8_t S9xGetAPUDSP(void);
bool S9xInitSound(void);
void S9xPrintAPUState(void);
-extern uint16_t S9xAPUCycles [256]; // Scaled cycle lengths
-extern uint16_t S9xAPUCycleLengths [256]; // Raw data.
+extern uint8_t S9xAPUCycles [256]; // Scaled cycle lengths
+extern uint8_t S9xAPUCycleLengths [256]; // Raw data.
extern void (*S9xApuOpcodes [256])(void);
#define APU_VOL_LEFT 0x00
diff --git a/source/c4.h b/source/c4.h
index 7ab3964..3c47d5f 100644
--- a/source/c4.h
+++ b/source/c4.h
@@ -29,5 +29,4 @@ int16_t _atan2(int16_t x, int16_t y);
extern int16_t C4CosTable[];
extern int16_t C4SinTable[];
-
#endif
diff --git a/source/c4emu.c b/source/c4emu.c
index 7805dc5..300e421 100644
--- a/source/c4emu.c
+++ b/source/c4emu.c
@@ -15,7 +15,7 @@ uint8_t S9xGetC4(uint16_t Address)
{
if (Address == 0x7f5e)
return 0;
- return (Memory.C4RAM [Address - 0x6000]);
+ return Memory.C4RAM [Address - 0x6000];
}
static uint8_t C4TestPattern [12 * 4] =
@@ -68,8 +68,7 @@ static void C4ConvOAM()
SprY = READ_WORD(srcptr + 2) - globalY;
SprName = srcptr[5];
SprAttr = srcptr[4] | srcptr[0x06]; // XXX: mask bits?
-
- sprptr = S9xGetMemPointer(READ_3WORD(srcptr + 7));
+ sprptr = S9xGetMemPointer(READ_3WORD(srcptr + 7));
if (*sprptr != 0)
{
int32_t SprCnt;
@@ -314,7 +313,6 @@ static void C4DrawWireFrame()
int16_t X1, Y1, Z1;
int16_t X2, Y2, Z2;
uint8_t Color;
-
int32_t i;
for (i = Memory.C4RAM[0x0295]; i > 0; i--, line += 5)
{
@@ -345,7 +343,6 @@ static void C4TransformLines()
int32_t i;
uint8_t *ptr;
uint8_t* ptr2;
-
C4WFX2Val = Memory.C4RAM[0x1f83];
C4WFY2Val = Memory.C4RAM[0x1f86];
C4WFDist = Memory.C4RAM[0x1f89];
@@ -353,18 +350,17 @@ static void C4TransformLines()
// transform vertices
ptr = Memory.C4RAM;
+
+ for (i = READ_WORD(Memory.C4RAM + 0x1f80); i > 0; i--, ptr += 0x10)
{
- for (i = READ_WORD(Memory.C4RAM + 0x1f80); i > 0; i--, ptr += 0x10)
- {
- C4WFXVal = READ_WORD(ptr + 1);
- C4WFYVal = READ_WORD(ptr + 5);
- C4WFZVal = READ_WORD(ptr + 9);
- C4TransfWireFrame();
-
- // displace
- WRITE_WORD(ptr + 1, C4WFXVal + 0x80);
- WRITE_WORD(ptr + 5, C4WFYVal + 0x50);
- }
+ C4WFXVal = READ_WORD(ptr + 1);
+ C4WFYVal = READ_WORD(ptr + 5);
+ C4WFZVal = READ_WORD(ptr + 9);
+ C4TransfWireFrame();
+
+ // displace
+ WRITE_WORD(ptr + 1, C4WFXVal + 0x80);
+ WRITE_WORD(ptr + 5, C4WFYVal + 0x50);
}
WRITE_WORD(Memory.C4RAM + 0x600, 23);
WRITE_WORD(Memory.C4RAM + 0x602, 0x60);
@@ -403,7 +399,6 @@ static void C4BitPlaneWave()
uint32_t waveptr = Memory.C4RAM[0x1f83];
uint16_t mask1 = 0xc0c0;
uint16_t mask2 = 0x3f3f;
-
int32_t i, j;
for (j = 0; j < 0x10; j++)
{
@@ -426,8 +421,7 @@ static void C4BitPlaneWave()
waveptr = (waveptr + 1) & 0x7f;
mask1 = (mask1 >> 2) | (mask1 << 6);
mask2 = (mask2 >> 2) | (mask2 << 6);
- }
- while (mask1 != 0xc0c0);
+ } while (mask1 != 0xc0c0);
dst += 16;
do
@@ -450,8 +444,7 @@ static void C4BitPlaneWave()
waveptr = (waveptr + 1) & 0x7f;
mask1 = (mask1 >> 2) | (mask1 << 6);
mask2 = (mask2 >> 2) | (mask2 << 6);
- }
- while (mask1 != 0xc0c0);
+ } while (mask1 != 0xc0c0);
dst += 16;
}
}
diff --git a/source/catsfc_griffin.c b/source/catsfc_griffin.c
deleted file mode 100644
index 8b4666c..0000000
--- a/source/catsfc_griffin.c
+++ /dev/null
@@ -1,35 +0,0 @@
-#include "../copyright"
-
-#include "apu.c"
-#include "c4.c"
-#include "c4emu.c"
-#include "cheats2.c"
-#include "cheats.c"
-#include "clip.c"
-#include "cpu.c"
-#include "cpuexec.c"
-#include "cpuops.c"
-#include "data.c"
-#include "dma.c"
-#include "dsp1.c"
-#include "fxdbg.c"
-#include "fxemu.c"
-#include "fxinst.c"
-#include "gfx.c"
-#include "globals.c"
-#include "memmap.c"
-#include "obc1.c"
-#include "ppu.c"
-#include "sdd1.c"
-#include "sdd1emu.c"
-#include "seta010.c"
-#include "seta011.c"
-#include "seta018.c"
-#include "seta.c"
-#include "soundux.c"
-#include "spc700.c"
-#include "spc7110.c"
-#include "srtc.c"
-#include "tile.c"
-
-#include "../libretro.c"
diff --git a/source/changes.txt b/source/changes.txt
deleted file mode 100644
index 3aabb22..0000000
--- a/source/changes.txt
+++ /dev/null
@@ -1,2155 +0,0 @@
-Snes9x 1.43 (WIP1)
-- Win32: added .avi output feature (blip)
-- Win32: fixed frame timings >100ms, added frame advance (blip)
-- Rewrote Unfreeze, renamed it S9xUnfreezeFromStream,
- failing to load a freeze file no longer resets emulation (blip)
-- Fixed Unfreeze to restore IPPU.HDMA properly (blip)
-- Rewrote OBC1 code to match the real chip (Overload)
-- More updates the to DSP-1 code, fixes to projection (Overload, Andreas Naive)
-- Unix/X11: Rewrote keyboard setup code (Bisqwit)
-- Added movie recording+rerecording support (blip, Bisqwit)
-- Added -hidemenu CLI switch (funkyass)
-- fixed broken Win32 filters (lantus)
-- Added internal support for emulating the new-style SNES (MKendora)
-- Cleaned up many quirks of the cheat search engine (MKendora, Don Vincenzo)
-- Fix mosaic in hires SNES modes (Tokimeki Memorial) (MKendora, zones)
-- Rewrote Legend's hack, added another game to it (MKendora)
-- Optimized the Open ROM dialog (MKendora)
-- Rewrote the Seta DSP map (The Dumper, MKendora)
-- Began string isolation for the UI, eases translation (funkyass)
-- added -nopatch -nocheat, and -cheat CLI items (MKendora)
-- fixed a UI typo (funkyass)
-- fixed several C core stack ops in emulation mode (MKendora)
-- split emulation mode ops from native mode ops (MKendora)
-- Seta special chip emulation enhancements (Feather, The Dumper, Overload, MKendora)
-- code tweaks to the ST010 (Nach, pagefault)
-- fix some C/asm quirks and HDMA quirks (all my fault) (MKendora)
-- several timing hacks to fix games (lantus)
-- improved checksumming for odd mirrorings (MKendora)
-- Snes9x uses a standard zlib instead of a packaged one (PBortas)
-- Exhaust Heat 2 and regional ports are playable (Feather, The Dumper, Overload, MKendora)
-- Game Doctor dumps that are 24 Mbit are now supported by
- a force option (MKendora, Nach)
-- SuperFx interleave format is now considered deprecated.
- Support will be removed in future versions (Team decision)
-- made SuperFx interleave detection a compile option (MKendora)
-- added memory maps for slotted games (MKendora)
-- fixed a typo in the usage messages (MKendora)
-- fixed the bug that had nuked optimizations (The Dumper)
-- restored full speed optimizations in release builds (funkyass)
-- Added non-speed-hack version of color subtraction. (zones)
-- Fixed mouse offset in resized X11 window. (PhaethonH)
-- Fixed a (presumably) long-standing bug: Mode 6's BG is
- depth 4, not depth 8! (anomie)
-- Unix: unmap all joystick buttons before applying -joymapX (anomie)
-- Win32: added a define to disable pausing when focus is lost, NOPAUSE (funkyass)
-- Win32: Changed the default for Auto-save SRAM to 15 sec (funkyass)
-- Dreamcast: Added SH4 assembler (PBortas, Marcus Comstedt, Per Hedbor)
-
-
-Snes9x 1.42
-- Added 8-bit rendering filters (funkyass)
-- Added Sanity Checks for the Display Dialog (funkyass)
-- New Layout for the Joypad Dialog, (funkyass)
-- Fixed that anoying Joypad dialog bug. Now check to see
- if the axis exists before asking for the info form it (funkyass)
-- Added full POV support. (funkyass)
-- Fixed sram sizes for SuperFx games (Nach, MKendora)
-- Stopped saving sram for games with no battery (Nach, Mkendora)
-- Killed the gray line and slightly optimized Win32 GL (MKendora)
-- stack wrapping fix in C core (MKendora)
-- removed some dead hacks (Oda Nobunaga and Dezaemon) (MKendora)
-- fixed some DMA and HDMA modes (anomie, MKendora)
-- improved HDMA timing (anomie)
-- cleaned up load and deinterleave code (MKendora)
-- removed old UI DLL (MKendora)
-- new cheat dialogs (MKendora)
-- started Unicode preparation in Win32 UI (MKendora)
-- Implement odd sprite sizes, sprite priority rotation. (anomie)
-- RTO code that hopefully works. MK's #define is
- "MK_DEBUG_RTO" to try to debug the RTO code. (anomie)
-- SDD1 decompression support for Linux. Also added a new
- command line option -sdd1-pack. (anomie)
-- Added correct VRAM read logic. #define CORRECT_VRAM_READS
- if you want it. (anomie)
-- removed the non-VAR_CYCLES path (MKendora)
-- changed access timing map to be address-based. (MKendora, anomie)
-- DSP-1 updates (Overload, Andreas Naive)
-- S-DD1 decompression support (Andreas Naive)
-- optimized S-DD1 code (anomie)
-- S-DD1 can use packs or decompression (MKendora)
-- More work on Exhaust Heat 2 (MKendora, Overload, The Dumper)
-- separated ROM detection from file reading (lantus)
-- fixed a mirroring bug in LoROMs (MKendora)
-- cleaned up some mapping issues (MKendora)
-- ST018 games now boot before locking up (Mkendora, Overload)
-- SA-1 state was not completely reset, crashed Marvelous (zones)
-- Removed sample caching. It caused problems, and was not
- noticably faster. (MKendora)
-- Fixed interlace without breaking the displays for MK (anomie)
-- Fixed a PPU OpenBus hack (anomie)
-- Moved SPC7110 and S-DD1 regs to speed up the general case
- of reading the $4xxx registers (MKendora)
-- altered Hi/Lo ROM detection to fix a few misdetects. (MKendora)
-- Implemented RTO flags. With MK's implementation of $213F's
- interlace bit, we now pass the SNES Test Cart's
- Electronics Test (anomie)
-- Fix sprite windowing bug (anomie)
-- Way back in 1.40 MK changed the Windows port to default
- to a plain old joypad instead of the MP5. And then we
- removed the hacks for games that dislike the MP5. So
- we need to change the defaults elsewhere too... (anomie)
-- cleaned up the hacks section somewhat (MKendora)
-- removed some interleave hacks (MKendora)
-- fixed a bug in KartContents (MKendora)
-- transparency fix for Jurassic Park (lantus)
-- A hidden Win32 feature (MKendora)
-- Kludged Mark Davis until I get stable APU timing (MKendora)
-- Win32 renders overscan always, fixes some jumpy games (MKendora, lantus)
-- Fixed an FMOD bug (MKendora)
-- cosmetic tweaks (Everyone)
-- Fixed 2 special chip bugs in the C core (zones)
-- Added some sanity fixes to the C core, fixes MLBPA
- Baseball for C core users (zones)
-- updated zlib source (includes 1.1.4-1 patch) (MKendora)
-- compiler warning fixes (PBortas)
-- Updated the SuperFx asm core (pagefault)
-- Kludged Unix compilation to produce working SuperFx (PBortas)
- with the asm core.
-- Kludged VC to deal with optimization weirdness (MKendora)
-- Hacked Robocop vs. Terminator using Daffy Duck hack. Stops
- flashing. (MKendora)
-- Added some defines to the asm core (MKendora)
-- Added possibility to take screenshots on Unix (PBortas)
-- Initialize the C SuperFx core better (PBortas)
-- Kludge a Japanese golf game until the APU timing is fixed (MKendora)
-
-
-Snes9x 1.41-1
-
-- Oops, in the asm CPU core i was stomping on %eax too
- early, so register $4210 wasn't getting set properly. (anomie)
-
-
-Snes9x 1.41
-
-- Win32 controllers now stay the same between games (MKendora)
-- Win 32 Open ROM dialog fixes (MKendora)
-- Win32 Display dialog fixes (funkyass)
-- Win32 OpenGL ratio tweaking. (Reduces the gray line) (kode54)
-- Fixed Win32 superscope for those having issues (MKendora)
-- Generic accuracy fix in main SUperscope emulation (MKendora)
-- sprite bug fixed (gah! How'd we miss that) (anomie)
-- SPC saving compatibility fix (Caz and zones)
-- Window clipping update (anomie)
-- Mode 7 clipping fix (TRAC)
-- latching fix (anomie)
-- BS BIOS checksum and mapping fix (MKendora)
-- Working Uniracers hack (dma.cpp) (anomie)
-- HDMA Indirect Address fix for Romancing Saga 2 (anomie)
-- Better savestate hack, does it break anything? (anomie)
-- C4 C core fixes. Mostly Trapezoid (thanks Nach),
- some s/short/int16/, some indentation. (anomie)
-- Damn, but the indentation in ppu.cpp was screwed up.
- Killed some dead code too (twas commented forevermore). (anomie)
-- fixed a potential crash in S-DD1 logging (MKendora)
-- Improved accuracy of Hi/LoROM detection (~500 ROM test) (MKendora)
-- Hack for Moryou Senki Madara 2, don't call
- SelectTileRenderer from DrawOBJS if BGMode is 5 or 6. A
- real fix requires at least rewriting SelectTileRenderer,
- or inlining a special version in DrawOBJS. (anomie)
-- DMA traces: add additional address info to reads too. (anomie)
-- Killed the old Borland Joypad dialog (funkyass)
-- Fixed issues with Dezaemon and CT, maybe others (anomie, MKendora)
-- Changed the internal snapshot key from \ to VK_F12 (funkyass)
- Fixes issues with non-US keyboard layouts.
-- Fixed OAM reset to not occur during forced blank. (anomie)
-- Killed some dead OAM reset code that doesn't need saving. (anomie)
-- Unix/X11: Fixed screen jumping. CT enables overscan mid-
- frame for only one frame, and we now update the rendered
- screen height accordingly. Other ports are still broken. (anomie)
-- Unix/X11: Fixed possible TV mode crash. (anomie)
-- Fixed OAM reset timing (beginning of V-Blank rather than
- end) for R-TYPE 3 (J). (anomie)
-- Unix/X11: Fixed OpenGL target (PBortas)
-- Unix/OSS: Fixed big endian sound (PBortas/ernstp)
-- Tweaked the About Dialog so its read-only and no scroll (funkyass)
-
-
-Snes9x 1.40
-
-- cleaned up a sound skipping code issue. Same as the
- RTC issue (lantus)
-- re-fixed the invalid BRR header behavior twice (Lord Nightmare, FatlXception, Mkendora)
-- More BS mapping fixes. (The Dumper, MKendora)
-- Fixed Ranma Bun no 1 - Chonai Gekitou Hen (J) and
- Street Combat (U). Interlace is not supported in the
- non-Hi-res modes, as far as I can tell. (MKendora)
-- Also fixes Maka Maka (J). Frank Yang's report, and
- anomie's code both provided clues to this one.
-- Removed special casing on setting 5c77 version to one.
- This seems to be true for U and J units always. I need
- it checked out on PAL... (neviksti)
-- Using SNEeSe's values for 5c78 and 5A22. Note we know
- that the 5c78 version can also be 1 or 2, instead of 3. (TRAC, neviksti)
-- Added turbo buttons. Credit/blame for the design goes
- to slack, Nave, Gogo, and myself. (MKendora)
-- fixed a bug in turbo (slack, MKendora)
-- Tried merging the behavior of Old $4200 with new $4200 (MKendora)
-- Made $4200's return value match what VSMC Explorer
- showed on Fancia's SNES (MKendora)
-- Fixed a matrix multiplcation bug in ZSNES state loads (MKendora)
-- Fixed Dezaemon and Ys3 mode 7 (lantus)
-- Fixed H-DMA modes 5-7. Thanks to The Dumper for the
- extra motivation needed. GunForce and Genocide 2 work. (The Dumper, MKendora)
-- Fixed BG3 Priority. I'm stupid. anomie had fixed it,
- but lantus fixed it again, because I didn't use it. (anomie, lantus)
-- Added a Star Fox 2 hack, and an interleave skip (The Dumper, lantus, MKendora)
-- Cleared BS setting on load (lantus)
-- Fix for Mode 7 priorities. fixes F-1 Grand Prix (all 3) (anomie)
-- JANJYU GAKUEN 2 needs Multi-tap 5 off. (Frank Yang, MKendora)
-- HONKAKUHA IGO GOSEI: No multi-tap 5, allow mouse (lantus, MKendora)
-- Added a few missed conditional compiles (Nach)
-- disabled multitap 5 by default, added menu to enable (MKendora)
-- special thanks to anomie and lantus. One of them is
- responsible for a bug fix I forgot already. (anomie, lantus)
-- Removed several Multitap5 disable hacks. (MKendora)
-- Added an SPC dumping upgrade from kode54 (kode54)
-- cleaned up some resource leaks (MKendora)
-- I forgot this since 1.39mk, but SPC700 flag fixes (anomie)
-- Mode 7 interpolation screen flip fix (anomie)
-- Updated SPC7110 code a bit, for compatibility (Daniel, anomie)
-- Changed RTC saving. (Byte exact to old format on Win32)
- The submitted patch for "safety" doubled the file size,
- so I had to write it in explicitly little-endian. (MKendora)
-- Removed the old hidden cursor (MKendora)
-- Applied a WAI correction from anomie. (anomie)
-- Added a patch for Pseudo hi-res (anomie)
-- Hacked around Word writes to $7F:FFFF. Thanks to lantus
- and The Dumper for verification. (MKendora)
-- PPC compile fix? and debugger reversion (anomie)
-- Set defaults differently to improve sound quality. (MKendora)
-- Clear Force load settings after Init (lantus)
-- Made menu reset a soft reset. Fixed BL Sound Test & more (CaitSith2)
-- Fixed word writes to block bounds in asm core. (MKendora)
-- redone version of my bounds fix, only this one WORKS! (TRAC)
-- Thanks to TRAC for the AT&T syntax refresher! (TRAC)
-- Fixed screen saver disable (kode54)
-- Fixed OAM and sprite priority in the asm core (anomie)
-- Proper Interlace fix for mid-frame changes (anomie)
-- Fixed OpenGL to accomodate previous patch (MKendora)
-- Ported the "Settings" dialog to VC (MKendora)
-- Fixed ROM Info bugs (_pentium_five, MKendora)
-- Fixed non-stretched interlacing, but it's s.l.o.w. (anomie)
-- Superscope and Mouse need to be enabled by the menu. (MKendora)
-- Fixed HiROM sram reads in asm and C cores (anomie, MKendora)
-- Added Company 48 to the list. Thanks to _pentium_five_ (StatMat)
-- Set Super Drift Out's S-ram correctly. (Snes9xppSE Team)
-- Fixed NTSC timing. Helps ToP Intro greatly (kode54)
-- Added several entries to the company list, from uCON64 (Nach)
-- Lots more companies (StatMat, Nach)
-- Fixed Win32 Superscope support (NT kernel only?) (MKendora)
-- Added ZSNES OBC1 code ported from asm to C (sanmaiwashi)
-- Implemented Justifier emulation (neviksti, MKendora)
-- Fixed Rudora no Hihou's clip window bug (anomie)
-- Fixed Flintstones sprite issue (lantus)
-- Fixed sram mappings for Big Sky Troopers and
- Taikyoku - IGO Goliath. Both map in bank F0 (MKendora)
-- Fixed a possible crash when switching audio settings (MKendora)
-- Added per-pack gfx pack configuration (MKendora)
-- Fixed glitches in DSP-1 games (Flintstones fix) (lantus)
-- Added delay to Superscope latching. Fixes X-Zone. (neviksti, MKendora, zones)
-- Added DSP-2 support (Overload, The Dumper, Lord Nightmare,
- MKendora, neviksti)
-- Fixed Super Bases Loaded 2 (and J/K ports) DSP-1 seems
- to ignore the A15 line in LoROM maps (MKendora)
-- Corrected $4200 again (The Dumper)
-- Corrected $2100, $2102, and $2102 read behavior (anomie)
-- Fixed Cancel on the Sound Options dialog. (MKendora)
-- Fixed the sound options dialog (Thanks, Quattro) (MKendora)
-- updated DSP-1 support to match chip better (Overload, neviksti, The Dumper)
-- added a few Ops to the DSP-4 routine (Nothing plays yet) (neviksti, The Dumper, Overload, MKendora)
-- added screenshot support (anomie, sanmaiwashi)
-- stubbed the ST010 chip in Exhaust Heat 2 (Overload, MKendora)
-- hacked around War 2410's lockup (pagefault, _Demo_, MKendora)
-- updated tests for type 1 ROMs (based on reset vector) (MKendora)
-- Emulation mode CPU fix (The Dumper)
-- Open Bus fixes (anomie)
-- Better Expansion port emulation (anomie)
-- More Open Bus fixes (Overload, anomie)
-- HDMA fixes (fix colors only in Full Throttle Racing) (anomie)
-- Migrated DKJM2 onto the Tales map (MKendora)
-- Tried to remove Dragon Knight 4 hack (LoROM sram fix) (MKendora)
-- Fixed ROM Mirroring for LoROMs (<= 32 Mbit) (MKendora, TRAC)
-- blocked wram to wram DMAs (neviksti)
-- fixed HiROM mirroring, too. Thanks TRAC! (MKendora, TRAC)
-- fixed C core RMW and Push ops to write in the correct
- order, fixes Michael Jordan gfx. (anomie, Overload, MKendora)
-- set RDIO to start as 0xFF, fixes SuperFx games. (anomie, Overload)
-- New connect dialog (funkyass)
-- better conditional compile of FMOD (funkyass)
-- fixed screenshot code when libpng is not used (funkyass)
-- added portability fixes (zones)
-- fixed asm Pushes (anomie)
-- fixed asm LoROM s-ram decode (MKendora)
-- migrated DEZAEMON to standard LoROM map (MKendora)
-- fixed the Madara 2 OpenGL bug (key found in Rudra) (MKendora)
-- fixed asm RMW instructions (MKendora)
-- fixed ADC opcode (The Dumper)
-- added DSP-2 Op09 (The Dumper)
-- updated C4 C code (anomie)
-- updated C4 asm code (Nach)
-- Keep OpenGL in ratio (kode54)
-- Replaced many more Borland dialogs (funkyass, MKendora, Nach)
-- Added CRC32 to displayed ROM Info (Nach, MKendora)
-- Fix cheat support (The Dumper)
-- improved DMA timing (MKendora, Overload, The Dumper)
-- Fixed Mode 7 math, removed Dezaemon, Gaia, Ys 3 hacks (TRAC, MKendora)
-- Mode 7 flip fix (TRAC)
-- Multiple safety and initialization fixes (zones)
-- Platform safety fixes (PBortas)
-- Memmap cleanups (MKendora)
-- More preliminary work on special chips (The Dumper, Overload, MKendora)
-- Added color coding (MKendora)
-- Another HDMA fix (anomie)
-- added another known hack to the hacked games list (Nach)
-- ToP memmap changes (MKendora)
-- Checksum calculation changes (MKendora)
-- Special cased a few games for OAM issues (MKendora)
-- Reverted OAM reset to 1.39 timing (MKendora)
-- Reworked vram wrapping (zones, Mkendora)
-- Fixed $4210 and Super Professional Baseball 2 (Overload, MKendora)
-- Fixed APU RAM init (Overload, MKendora)
-- More support for Exhaust Heat 2 (not playable) (The Dumper, Overload, neviksti)
-- removed some debris from save states (MKendora)
-- fixed? Doom's save state bug (MKendora)
-- simple overdump detection warning (MKendora)
-
-
-1.39mk3b
-
-- Fixed the RTC detection. FINALLY done correctly (lantus, MKendora)
-
-
-1.39mk3a
-
-- neatened up the company table. (MKendora)
-- fixed a mistake in the ROM Info box (MKendora)
-- Added a Calulcated Size field to ROM INfo. (MKendora)
-- Added 3 more companies to the ROM Info table (MKendora)
-- Fixed BS detection (The Dumper)
-- Added a Legend-specific hack to get sound. I remembered
- it being mentioned in the changelog. (Gary Henderson)
-- Unbroke the Star Ocean special cases (Trigger of Time, MKendora)
-- Company 255 is not Hudson-ZFE detects all Hudson games
- without it, except a corrupt dump (StatMat, MKendora)
-- fixed a bug in the redone detection for the SPC7110 (CaitSith2)
-- 44Khz sound should be 44.1Kz. Changed, though you'll
- need to re-set 44.1Khz to make it take effect. Not sure
- if this affects non-Windows ports. (MKendora)
-- Added 32Khz playback (MKendora)
-- Inproved BS ROM mapping (_Demo_, The Dumper, MKendora)
-
-
-1.39mk3
-
-- Honkaku Syogi Fuunji Ryuou (J) fixed (force no multitap) (Frank Yang)
- Also Fixed Super Castles (j).
- Also fixed a bunch more. This dude e-mailed like 100 bugs
- to my hosts, some already fixed in Snes9x1.39mk2, but
- about 7 were clearly multi-tap5.
-- also fixed Dekitate High School. Error was in Japanese (Frank Yang, Tomato)
-- fixed 2 memory leaks (Aaron)
-- Dai Kaiju Monogotari 2 works as a 40 Mbit ROM. (MKendora, The Dumper)
-- Fixed the Flashback bug. Lots of info led to this. (neviksti, MKendora)
- Thanks neviksti, The Dumper, TRAC, and FatlXception
- for clarifying the behavior.
-- Fixed Sailor Moon Fuwa Fuwa Panic 2 to work with (neviksti, MKendora)
- previous fix. It's a total hack, but it should sound
- just like the old Snes9x did. neviksti strikes again!
-- Dirty hack to make 3 games deinterleave properly: (MKendora)
- Wizardry 4, Mark Davis, and Honkakuha Igo Gosei(FX)
- all work as well as the deinterleaved counterparts.
- (The last is a hacked game, and you should get the
- non-FX version)
-- Fixed Seima Jyuden Beasts and Blades. Another Multitap, (Frank Yang)
- but for some reason, the hack requires the C cpu core.
- Thanks to Tomato for taking a stab at the error message,
- as well. It was too vague to be of use, he said. I
- just tried it because it worked on other games.
-- Res Arcana fixed. Another Frank Yang report, another J (Frank Yang, MKendora)
- error, but I can read kana well enough with a table!
-- Removed a Terranigma specific hack. Not sure, but the (anomie)
- new behavior might have fixed Tin-Tin in Tibet's colors.
-- Dirty hack to work around a dirty hack. Both Yoshi's (MKendora)
- Island (E) dumps should work now
-- Added the JumboLoROM memory map, Extends LoROM support (The Dumper, neviksti, MKendora)
- to 48+ Megabits.
-- added an EXTBG fix, since iirc, TRAC is using it as well (anomie)
- Does it actually fix anything?
-- Fixed crash in DSP Op06 (The Dumper)
-- Fixed a GUI error on my part (Trigger of Time)
-- Cleaned up some of the SPC7110 detection/size code. (MKendora)
-- Merged in XBox port changes to SPC7110 code (lantus)
-- Added a call to Memory.Deinit when exiting. (lantus, MKendora)
-- Many memory leaks fixed while chatting with lantus (lantus, MKendora)
-- Fixed that stubborn open/close leak (lantus)
-
-
-1.39mk2
-
-- hacked in Shien's Revenge (anomie)
-- fixed Orge Battle's green lines. (CPU source for DMA) (anomie)
- - Looks interesting, and might apply to other DMA cases?
-- maybe "fixed" DKC's barrels? by treating $2001
- as unmapped. The game worked before with a hack. (MKendora)
-- optimized SPC7110 slightly by removing extra setup work (MKendora)
-- Fixed DBZ 3 (Korean). S. Korea is, in fact, NTSC. (MKendora)
-- Fixed a hard-coded value in the SPC7110 (MKendora)
-- Added a Win port ROM Info dialog (MKendora)
- - some companies aren't in the table I used.
- If you encounter an Unimplemented company,
- report it the the Snes9x development forum, with
- the correct company and the number.
-
-
-1.39mk
-- SPC7110 support based on Dark Force's docs. (Dark Force, zsKnight,
- The Dumper, MKendora)
- Trust me when I say those guys deserve the credit more
- than me. From what I'm told, Dark Force is the man
- behind most of the reverse engineering, but they all
- did a much harder bunch of work than I did following
- their specs. It's plain and simple that these three
- are the masterminds behind all SPC7110 support.
-
- Dark Force for reverse engineering the chip (Extremely tough work!)
- zsKnight for the original core, and probably other things
- The Dumper for dumping the packs and doing hardware tests.
-
- Also thanks to CaitSith2 for numerous bug reports
- and a lot of bug fixes.
-
-- Theme Park hack removed, fixed via PPU latching (anomie, MKendora, TRAC)
-- WWF Wrestlemania hack removed (anomie, TRAC)
-- Strike Gunner hack fixed (anomie, MKendora, TRAC)
-- FF:MQ text fixed. May help other sprite issues. (TRAC)
-- Umi Hara Kawa Se timing corrected. (anomie)
-- S-DD1 packs load by the same rules as ZSNES (MKendora)
-- SPC7110 code builds in linux (Lord Nightmare, zinx)
-- Added The Dumper's DSP-1 updates (The Dumper)
-- SPC7110 is correctly displayed on load, RTC also noted. (MKendora)
-- Fixed a potential graphics problem (TRAC)
- no known games fixed, but who knows?
-- Fixed Ballz3D (pagefault)
-- Re-fixed Ballz3D, via DSP op 0F (The Dumper)
-- included some of anomie's fixes. Many caused me grief,
- so only Marko's Magic Football is intentionally fixed. (anomie)
-- finished zsnes save support, though I don't know how
- well it will work with SPC7110 games (MKendora)
-- Added a new soundux.cpp again to fix some noise.
- (Fixes the GW "fart track") (Lord Nightmare, info from Anti-Res)
-- Added 3 cache modes for SPC7110 games (MKendora)
-- Added new BRR decoder. Requires sample caching
- and the Anti-Res decoder be disabled. (FatlXception, port by Lord Nightmare)
-- Added CaitSith2's RTC debugger. define RTC_DEBUGGER in
- project settings to enable it. (CaitSith2)
-- SPC7110 per-game cumulative logging (MKendora)
-- other fixes that I've forgotten (sanma iwashi, TRAC, anomie, ????)
-
-- "I'm not worthy" thanks to the original SPC7110 crew (DF, zsKnight, and the Dumper)
-- Thanks again to the same people, because they deserve it!
-- thanks to The Dumper, Dejap, TRAC, and all the ZSNES crew for technical assistance
-- Thanks to most of the Snes9x mods for testing (no thanks to you, Raptor ;)
-- and thanks to TRAC and #mkendora for letting me vent at you.
-
-1.39
-- Added SDD-1 unknown graphics data logging at the dumper's request. A bit late
- but might help with Street Fighter 2 Alpha's data dumping. Creates a
- romname.dat file in the freeze file folder.
-- Implemented 16-bit texture support for OpenGL modes in Windows and Linux.
- Had to support a new pixel format type to do it - RGB5551 (one bit of alpha)
- which caused me some major problems - black was no longer always pixel value
- zero!
-- Removed the Bump map OpenGL mode from the Windows port (didn't look so good
- anyway and was slow).
-- Added a hidden novelty OpenGL mode (clue: a keyboard shortcut activates it)
-- Reverted back to FMod version 3.20 after reports that version 3.33 broke
- AD3 support.
-- Implemented a better work-around for the broken select system call in the
- Linux kernel - the original work-around was long-winded and stopped working
- when I implemented OpenGL support under Linux.
-- Added the same speed-up hack to the OpenGL code that the Glide code already
- supported. Basically, if your OpenGL implementation supports 16-bit textures
- then OpenGL mode should be as fast, or faster than the 3dfx Glide mode.
-- Hopefully fixed Glide support.
-- Reverted back to the original colour blending code. The newer code, although
- more accurate in most cases, had too many glitches and was slower.
-- Included multiple Japanese games fixes from Iswashi San.
-- Fixed a timing problem caused by a speed up hack that was affecting Top Gear
- 300. No the game still isn't playable yet, but I noticed the problem while
- investigating the DSP-4 chip used by the game.
-1.38
-- Added support for Star Ocean and Street Fighter 2 Alpha decompressed graphics
- packs from dejap. Used a binary chop search rather than a linear search to
- locate correct decompressed graphics more quickly - should help emulation
- speed during later stages of the game.
-- Included OpenGL support into the Linux port and speeded up the Windows OpenGL
- implementation slightly. The real speed up would occur if I could figure out
- how/if 16-bit textures are supported in OpenGL because at the moment the
- 16-bit software rendered SNES image must be converted to 24-bit before being
- uploaded as a texture...
-- Included the latest ZSNES DSP-1 code. Now Pilotwings, SD Racer and Suzuka 8
- Hours are playable. Aim For The Ace, Super Air Diver 1 & 2 and Syutoko Battle 94
- are also playable, but with bugs. Thanks to zsKnight, _demo_, et al for all
- their hard work.
-- Another Daffy Duck: Marvin Missions screen flicker problem worked around -
- writing to the IRQ enable register shouldn't clear any pending IRQs, but
- Sieken 3 seems to require this or else the game hangs. Special-cased Daffy
- Duck for now.
-- An NMI emulation bug was triggering a Panic Bomberman World game bug,
- crashing it. Basically, if a game enables NMIs after the normal trigger
- point, the NMI should not trigger if the game has already read the NMI clear
- register.
-- Panic Bomberman World requires SPC700 memory to be initialised to zero on
- reset otherwise the game hangs when a tune finishes and another one should
- start.
-- Added mouse pointer auto-hide to the Windows port. Much better than the turn
- the mouse pointer into a black dot method I was using before.
-- Included the latest ZSNES Super FX code. Not sure if it fixes actually fixes
- any games.
-- Added an offset hack for Strike Gunner to get the scrolling ground layer
- to line up correctly - another offset-per-tile bug hacked around for now.
-- Arrr! Left in some debugging code in the last release that prevented all
- games that need the slower SPC700 timing from working. Removed it.
-- Hmm. The broken cut-scenes in Deep Space 9 seem to indicate that I haven't
- got the emulated clock speed of the 65c816 CPU correct yet. And not by a
- little bit - a 9% too slow error. Hacked special timing for the game for now.
-- Added triple-buffering to Windows port - enabling double-buffering actually
- enables triple-buffering if you have enough free video RAM, defaulting to
- double-buffering if you don't.
-- Fixed another crash bug in the interpolated mode 7 code - if no scaling
- was being used (either up or down) and screen repeat was enabled and the
- screen was flipped horizontally, the routine would crash Snes9x. Was causing
- Snes9x to crash during rock monster boss stage of Castlevania 4.
-- Oops. Got the initialisation of the default SNES screen width and height
- round the wrong way - could cause a X Windows System error message on the
- UNIX port after loading a ZSNES freeze file.
-- Included the unofficial Windows port emulation fixes for several games including
- Kentouou World championship and TKO Super Championship.
-- Included Iwashi San's improved Anti Res. sound sample decoding routine and
- updated the C version to match.
-- Included Anti Res. improved sample decompression code he sent me ages ago,
- but for some reason I didn't include. Sorry. This version seems good enough
- to leave enabled all the time.
-1.37
-- Added fix for Captain America's corrupt graphics - a ROM bug causes it to
- read from what I thought should be an unmapped memory area, but it expects
- the value returned to be zero.
-- Added code to support games that switch to the hi-res. SNES screen mode part
- way down the screen while using the 3dfx bi-linear filter mode. The code
- basically has to back out of the speed up hack it was using when the game
- switches resolutions.
-- Fixed support for games that have mixed lo-res. (256x224), medium res.
- (512x224) and hi-res. (512x448) all on the same screen - corrects the display
- of Majin Tensei 2.
-- Added support for games that use sub-screen addition to the back-drop layer
- while displaying hi-res. graphics - something I thought the SNES couldn't do
- but the game Marvelous uses this.
-- Reworked the UNIX/Linux output image handling code: the image doesn't always
- have to be scaled when hi-res. support is enabled, the PutImage operation
- only updates the area of the screen it has to, the SNES image is now always
- centred in the window/full-screen area and if the SNES image changes size
- between frames, the old screen areas are now correctly cleared.
-- Fixed the corrupt graphics problem during the battle scene of Last Bible 3 -
- it requires that previously unknown DMA mode 5 should just act the same as
- DMA mode 1.
-- Fixed a nasty bug when H-IRQs were being reused on the same scanline - a logic
- bug could cause H-DMA processing for that line to be skipped. Was causing
- the bridge and the start banners to be the wrong colours in Top Gear 2.
-- Added Kreed's display processing modes to the Linux port, including his new
- asm version of the Super2xSaI mode and the new software bi-linear filtering
- mode.
-- Think I might have figured out the odd Mode 7 glitch problems the games
- Illusion and Gaia and Chase HQ were having. My original fix was to mod the
- centre X & Y values with 1024, but looks like the true fix is to mod
- X + horizontal offset and Y + vertical offset with 1024 when screen wrapping
- is enabled.
-- Disabled H-DMA'ing into V-RAM via registers 2118/2119. The game Hook
- deliberately does this causing graphic corruption while dialog boxes are
- displayed. Maybe the real SNES disallowed this and it was left in the game by
- mistake? Not sure what effect the game was trying to produce because
- disabling the emulation of this feature doesn't seem to affect the game at
- all, other than stopping the corruption.
- + Also fixes graphics junk problem on first screen of Bugs Bunny.
-- Added a 'region-free' timing hack for Power Rangers Fight - without it the
- NTSC version was displaying badly glitching graphics; I'd already fixed the
- PAL version.
-- Added true priority-per-pixel mode 7 support (the previous support was just
- a hack to get the colours correct) - level 2 of Contra 3 used this feature.
-- The Japanese, German, French and Spanish version of Illusion of Gaia needs the
- slow SPC700 timing.
-- Deleted the Breath of Fire 2 S-RAM hack for the hacker intro version -
- according to reports it was causing problems for the non-hacked version.
-- Legend, the PAL version, never sets the sound master volume control - Snes9x
- was defaulting this to off, I guess the real SNES must default it to full
- volume; changed Snes9x. The NTSC version of Legend does set the master
- volume level, but sets it to off just after the title screen. Hmm. The -nmv
- command-line switch allows you to hear sound in this version.
-- Panic Bomber World was tripping an SA-1 emulation bug - the WAI instruction
- emulation code was setting the 'waiting for interrupt' flag on the wrong CPU
- causing the main SNES to skip an instruction when the next interrupt occurred.
-- Panic Bomber World, Bomberman 4 and UFO Kamen Yakisoban all need the slower
- SPC700 timing.
-- Oops! The Super Formation Soccer 95 fix was causing Aero 2 to lock up. This
- means I have no no idea what value the DMA in progress register should
- represent. I've hacked it and made it toggle between 0 and $ff on each read
- which gets both games working, for now...
-- The ROM de-interleaving code always assumed the blocks were rearranged based
- on a power of two, but Francois found a copy of Soldiers of Fortune where
- this was not the case. Corrected the code.
-1.36
-- Finally worked out why the menu items weren't being highlighted in several
- ROMs, including Battletoads, U.N. Squadron and All Japan Pro Wrestling.
- Two problems: its seems the SNES does halve the colour value result when
- blending colours when only the fixed colour addition/subtraction is enabled,
- but doesn't halve the result when sub-screen is being blended and its a clear
- part of the sub-screen. The second problem was that I had an optimisation
- that prevented the time consuming colour blending code from being called if
- the colour being added/subtracted was black - adding zero to a number doesn't
- affect the result, but not performing the side-effect of halving the result
- does affect the final value...
-- Super Formation Soccer 95 requires that the DMA enabled register doesn't
- always return zero, otherwise the game locks up.
-- Thanks to several people reporting a screen flickering problem in the
- pseudo 3-d section of Jurassic Park 2 I've fixed a nasty problem in H-IRQ
- handling code which could cause double-triggers or skip IRQs altogether.
- With this fix I can now remove the special hacks for Ninja Warriors Again,
- Chuck Rock and F-1 Grand Prix.
-- More games needing the slow SPC700 timing:
- Zennihon Puroresu 2, Soulblazer and Robotrek.
-- The CPU idle time skipping code was skipping cycles during a software delay
- loop in Itchy and Scratchy, causing screen flicker.
-- Looks like reading the value of register $2137 shouldn't clear a pending
- IRQ - was causing screen flicker on Yoshi's Island.
-- Actraiser 1 & 2 both need the slow SPC700 timing.
-- Terranigma reads a sound channel's current sample output value and waits for
- it to be zero before preceeding. I forgot to always return zero when a
- channel was silent. This mistake was causing the game to lock up.
- + Itchy and Scratchy and was causing the music to stop and samples to be cut
- short in the Mario Early Years series.
-- Added a hack for Secret of the Evermore - at several points in the game, just
- as the plane is about to land, it reads from unknown registers $4000 and
- $4001 and, if it doesn't get the value its looking for, the game hangs or
- displays corrupt graphics.
-- Silva Saga 2 was accidentally triggering a colour blending hack I put in
- place Kirby Dreamland 3 and Kirby Superstar.
-- The ZSNES freeze-file loading code could leave a file open if the file wasn't
- a valid ZSNES freeze file.
-- Super Punch-out requires certain DMA registers to be updated after the DMA
- completes. Snes9x used to do that, but I must have accidentally left the code
- commented out whilst investigating a different problem in another game.
-1.35
-- Added a recently played game list to the Windows port File menu so you can
- quickly load up your favourite games.
-- Included IPS patching support based on code from Neill Corlett - just rename
- the patch file to match your ROM image name but with a .ips extension and
- copy it into your ROM or freeze-file folder.
-- Added John Weidman's and Darkforce's S-RTC, (Real Time Clock) emulation code.
- The only game that seems to use it is Dai Kaijyu Monogatari II.
-- Included code from Nose000 for games with 128Kbytes of S-RAM. Now
- Sound Novel-Tcool, Thoroughbred Breeder 3, RPG-Tcool 2 and Dezaemon are
- supported.
-- The Windows port now has an option to make the 'turbo speed' button a toggle
- button.
-- The optimised fixed colour addition/subtraction code was ignoring the colour
- window. Thanks to John Weidman for pointing this out.
-- Added mode 7 and hi-res. hack for Dezaemon from Nose000 - the mode 7 hack
- looks interesting (to me); I wonder if some other games would benefit?
-- Both Tales of Phantasia and Star Ocean need custom sound CPU timing. Hmm.
- That's 4 ROMs now, there will be more... That means I still haven't
- discovered all the major SNES timing quirks. :-(
-- Windows port now has an option to save the S-RAM data at any time.
-- Windows port saving SPC dumps now auto-increments the filename.
-- Added work-around for a Super Robot Wars Ex ROM bug - the game was checking
- the wrong PPU register for end of h-blank. The game must have only worked by
- chance rather than by design on a real SNES.
-1.34
-- Corrected the colour addition/subtraction and halve the result code not to
- halve the result when only the fixed colour is used, i.e. the sub-screen is
- clear. Discovered and fixed this awhile ago, but I accidentally reintroduced
- the bug when adding some optimisations a few versions back.
-- Finally cleared the last of the offset per tile background mode bugs. There
- was something odd about the tile at the left-hand edge of the screen that I
- couldn't figure out - well now I have. Yoshi's Island level 6 boss screen,
- Mario RPG mine cart screen and Jim Power title screen now all display
- correctly.
-- Made reading blank areas of the SNES memory map return the middle byte of
- the address - fixes Home Alone which tries to execute code in an empty part
- of its memory map but only works because the real SNES seems to return the
- middle byte of the address - $60 in this case, which corresponds to the
- ReTurn from Subroutine instruction.
-- Added auto-cycle skipping disable for Earth Worm Jim 2 and several other
- games that spool sample data using H-DMA as the sample is being played.
- Improves some sound effects in these games.
-- Fixed joy-pad routines to only report up or left if down or right are also
- pressed respectively. Works around a game bug in Empire Strikes Back in the
- asteroid stage where the game crashes if both left and right are pressed -
- something impossible to do on the original SNES game-pad.
-- Added custom SPC700 timing for Rendering Ranger R2 - the game now works with
- full sound. No idea why it needs custom SPC700 timing.
-- The ROM type detection was broken for Treasure Hunter G and Test Drive 2 -
- fixed the code so type 2 ROMs can be LoROM.
-- Adjusted the main CPU cycles per scan-line from 341 to 342 to give an exact
- match for the timing required for Earth Worm Jim 2. All EWJ2 needs now
- for perfect sound emulation is a method of synchronising the emulation
- speed to the host hardware's sound card playback rate, oh, and a fast CPU!
- The Linux port already has this but seems to be broken because games
- play at double-speed when this option is enabled.
-- Some SPC700 code in Earth Worm Jim 2 seemed to prove that I had guessed the
- clock speed of the SPC700 sound CPU incorrectly - out by almost a factor of
- two, in fact. Changed the relative emulated clock speed of SPC700. Now
- Chrono Trigger doesn't lock up at certain points anymore, the special SPC700
- timing for games written by the Human Software company isn't required and
- you can hear some more of the sound samples in Earth Worm Jim 2, etc.
-- H-IRQ triggering code was broken - if a ROM turned on H-IRQ but later turned
- it off, Snes9x could continued to generate H-IRQs, crashing some games.
-- Added a generic test for Human Entertainment games - they need special
- sound CPU timing to work. Gets Taekwon-Do working.
-- Disabled offset-per-tile mode for Theme Park; the world map screen is corrupt
- with it enabled.
-- Yet more changes to the offset-per-tile backgrounds modes 2 and 4. Added
- 64 tile wide screen support for Mario RPG's mine cart ride and fixed multiple
- bugs with the handling of horizontal offset-per-tile used in Chrono Trigger's
- fade in of the space ship.
-- New feature: Snes9x can now load ZSNES freeze state files! Just copy them
- into the freeze file folder and Snes9x will load them when you load a freeze
- file, but only if the corresponding native format Snes9x freeze file doesn't
- exist.
-- Added memory map hack for Batman - Revenge of the Joker: its ROM header block
- is in the wrong location and Snes9x incorrectly detected its ROM type.
-- Fixed an off-by-one-pixel clip window 2 bug when the window was set to clip
- outside the window area; clip window 1 was already correct. Removed the bright
- line bug at the left edge when the combat screen is appearing in Starfox and
- the clip problem when text boxes zoom-out in Yoshi's Island.
-- Jim Power's title screen seems to prove that the per-tile offset data on
- mode 2 isn't ignored for the left most tile as I originally thought.
- Modified the code.
-- The recent timing changes highlighted another problem with Daffy Duck -
- changed IRQ enable register to only clear pending IRQs if one has been pending
- for several microseconds.
-- Speeded up the sprite data register handling slightly.
-- Finally got Aero the AcroBat 2 working, after many hours of investigation,
- spread over several years - literally! Two problems. The SNES doesn't seem
- to consider scan-line line zero to be part of the v-blank period even though
- the line is never drawn and V-IRQs at the start of the scan-line have to be
- delayed until a few microseconds into the line - Traverse: Starlight & Prairie
- required this as well, so I removed the original, Traverse specific hack.
- There's a problem with the in-game music that I'll investigate at a later
- date.
- - The in-game music problem just required ENVX emulation to be switched on,
- off by default on the Linux port, on by default on the Windows port.
-- Fixed the mode 7 corruption problem on the title screen of Chase HQ using the
- same trick as Illusion of Gaia - i.e. mod the mode 7 centre X & Y values with
- 1024.
-- Fixed another crash bug in the interpolated mode 7 code - a portion of
- the code was ignoring the screen flip value and the fact that X render
- direction reversed if the screen was flipped horizontally. Was causing a
- crash on the whale boss screen of Kirby Superstar.
-- Mortal Kombat 3 now auto-adjusts emulated cycles per scan-line work-around
- a speech sample being cut short.
-- Added sample data register reading support to the sound DSP - somehow I
- seem to have missed implementing this. Not sure if any ROM actually reads
- the value.
-- Followed Sumire Kinoshita's suggestion and stopped clearing the ENDX flags
- when the value is read, against my better judgement, and it does actually
- improve speech samples in several games. Ooops! The Mortal Kombat series,
- Magical Drop 2 and Metal Combat are the ones I've discovered so far.
-- WWF Arcade now auto-adjusts the cycles per scan-line value to work-around
- a sound sample repeat problem.
-- Hmm. There's something about offset-per-tile mode I don't understand - WWF
- Wrestlemania Arcade is getting corrupt graphics; not sure what effect the
- ROM is trying to produce. Disabled offset-per-tile mode for the game for now.
-- Fixed Street Racer player 1 wobble problem during the soccer game by auto-
- adjusting the cycles per scan-line value slightly.
-- Made Power Rangers Fight auto-adjust emulated cycles per scan-line to work
- around a slight timing problem that causes an NMI to corrupt register
- values that an IRQ handler is trying to update. Without it the scrolling
- back-drop and fighter graphics are corrupt.
-- Illusion of Gaia seems to need the mode 7 centre X & Y values to be mod 1024
- if the screen repeat flag is set. Fixes the island fly-over bug right at
- the end of the intro but breaks a few other games. Hmm. Made it auto-switch
- on for this game only.
-- Added memory map support for Radical Dreamers. Thanks to satellite hut master
- for the information.
-- Made updates to the top bit of the sprite write address register be ignored
- unless the low byte had been written to first. A ROM coding bug in
- James Pond II requires this, otherwise it writes a junk byte value into the
- main character's X position and Robocod wobbles around all over the place.
-- Reverted back to pre 1.31 way of initialising unknown register values -
- Rock and Roll Racing was reading a junk register value and using the value
- to set up DMA, which in turn was causing corruption on the player select
- screen.
-- Added Star Ocean memory map - thanks zsKnight! The original ROM I was testing
- was corrupt, no wonder I couldn't figure out the memory map myself! The game
- still isn't playable, though, due to missing S-DD1 graphics decompression
- (+ encryption?) emulation.
-- Started to dump some compressed data values from Street Fighter 2 Alpha in
- the hope that one day someone will be able to crack the S-DD1's compression
- algorithm.
-1.33a
-- C4 emulation wasn't being automatically enabled for Rockman X2 / X3 - the
- Japanese versions of Megaman X2 / X3.
-- Fixed the Super FX plot table pointer that I accidentally broke while saving
- 1Mb of workspace RAM - it was stopping all Super FX games from working.
-1.33
-- Noticed another problem with the CPU_SHUTDOWN code - Chrono Trigger locked
- up during the intro but only when using the asm code CPU core. Found the
- algorithm difference between the code and made the CPU match what the C
- version was doing. Still not sure why it caused a problem in the first place.
-- Changed colour subtraction code to use Lindsey Dubb's newer version he sent
- me some time ago but I 'forgot' to include. I say forgot, but I really put
- off including it because, although it improves most games that use the
- effect, it does result in one or two slight visual glitches.
-- Hacked in zsKnight's C4 emulation asm code - now both Megaman X2 and X3 are
- playable. Still got to complete the reverse engineering of the i386 asm code
- to C so other, non-Intel ports can have C4 emulation.
-- Shuffled the keyboard mapping a bit on the Linux port so now Tab key acts as
- an emulation speed turbo button, `, # and ~ act as superscope turbo and
- / acts as the superscope pause button.
-- Fixed asm CPU_SHUTDOWN code that I accidentally broke while trying to
- optimise it! Thanks to all the people who noticed Snes9x's frame skipping
- had changed between releases. Frames rates should be improved again for more
- than 50% of games.
-- Re-enabled in-lining of the C SNES memory access routines, improves frame
- rate by one or two on slower machines.
-- Optimised the asm 65c816 addressing mode emulation code a little.
-- Included some code changes making life easier for the Mac porter, John Stiles.
-- Added memory map support for Sufami Turbo using information supplied by
- Nose0000. No idea if it works because I don't have the ROM.
-- Spent a few minutes trying to figure out the Star Ocean memory map so at
- least the sound effects could be heard. But gave up after a couple of hours
- due to laziness. If anyone knows the memory map details, let me know please!
-1.32a
-- The delay loading of the OpenGL DLLs on the Windows port was causing the
- OpenGL initialisation code to fail. Reverted back to normal DDL loading but
- with the side effect that Windows 95 users must visit the Microsoft web site
- and download the OpenGL add-on before Snes9x will work for them.
-- Corrected the OpenGL bump-map display option - my attempt to get the
- bi-linear OpenGL display option to work with Voodoo card's limited texture
- size had broken the bump-map mode.
-1.32
-- Changed the Windows port to delay load the two OpenGL DLLs, so now they're
- only loaded if you switch to OpenGL mode. The original version of Windows 95
- didn't include the OpenGL DDLs, so Snes9x wouldn't even start on that
- platform; now it should.
-- Added yet another sound buffer option to the Windows port - this time the
- block size of sound data to mix. Some DirectSound sound card drivers only
- report the play position moving in steps rather than continuous amounts and
- Snes9x's default mix block size turned out to be smaller than this step
- value on several cards.
- Snes9x couldn't work out out where the true play position was accurately
- enough resulting in broken, noisy sound output.
-- Modified the Windows frame timer code to use semaphores rather than events -
- they should make Snes9x more reliable at not missing frame sync pulses when
- Windows is busy doing background tasks.
-- Added SA-1 shutdown code - basically, Snes9x now stops emulating SA-1 CPU
- instructions when the SA-1 enters an idle loop waiting for the main SNES
- CPU to give it something to do. All SA-1 run much faster and smoother now.
-- Added multi-axis joystick/game controller support to the Windows port and
- tweaked the dead-zone threshold position a little.
-- It looks like the SNES PPU was designed to support 128K of V-RAM but only
- 64K was fitted; Snes9x wasn't wrapping all V-RAM address to stay within the
- 64K limit causing a corrupt title screen on ReX Ronan - there will be others.
-- Added amend functionality to the Windows Cheat Entry dialog and added extra
- text boxes for direct address and cheat value input rather than only being
- able to type in a Game Genie or Pro-Action Reply code.
-- BS Suttehakkun2 was crashing just before start of play - the ROM was
- performing a junk DMA that was corrupting RAM, crashing the game when it
- went searching for a particular value.
-- F-1 Grand Prix requires IRQ triggering when IRQ scan-line register set to
- current scan line, but Chuck Rock objects. Hmm. Chuck Rock seems to indicate
- the CPU emulation is running too fast, but I can't see where the mistake is.
- Special-cased Chuck Rock for now.
-- Optimised SNES DMA handling slightly - copying data to SNES V-RAM is now
- significantly faster.
-- Windows Cheat search dialog was ignoring data type parameter in various
- places which was causing problems when larger numbers were being searched
- for.
-- Forced unknown PPU register reads to always return 0 - a coding bug in
- Equinox shows that this is required. An earlier fix didn't work.
-- Puya Puya 2 & remix were objecting to an NMI being triggered when enabling
- NMIs after scan-line 226, but Ys 5 seems to require this. Hmm. Added a hack
- to support both games.
-1.31
-- Snes9x DirectSound code modified - the mixing block size is now always 10ms
- for Windows 95/98/2000 and 20ms for NT 4.x, now there should be no need to
- enable Sync Sound when a large sound buffer is required (helps emulation
- speed). The maximum sound buffer length values have been updated to reflect
- the smaller mixing block size.
-- Changed the DirectSound code back to use an offset from the play position
- as the place to write new sample data into the sound buffer - on NT 4.x the
- write position seems to vary randomly rather than being a fixed distance
- in front of the play position as documented. Now I know why I used the play
- position originally!
-- Changed the DirectSound code to fill the sound buffer at the write position
- supplied by DirectSound, rather than just before the current play position -
- should help reduce latency.
-- Added an auto-detect method for interleaved mode 2 Super FX ROM images -
- well, not really auto-detect: if the game crashes and its a Super FX game,
- Snes9x assumes its in interleaved mode 2, de-mangles the ROM image and tries
- to run the game again.
-- Had to update the Snes9x Windows registry version number as the additional
- diagonal settings make old registry settings incompatible.
-- Added diagonal keyboard controls to the Windows port, as requested by
- several users.
-- Changed PPU code to return zero when reading non-existent registers - the
- game Equinox relies on this due to an original game coding bug.
-- Included FMOD sound driver support to Windows port - people experiencing
- broken sound or delayed sound, etc, might want to give it a try.
-- Tales of Phantasia - un-interleaved format ROM memory map changes to match
- odd ZSNES format, now the hacked ROM works.
-- Changed NMI again. Made reading or writing to PPU register 0x4210
- clear NMI pending flag again, without this Super Tennis does not work.
-- Changed NMI timing back to be the same as several versions ago and just
- special cased Cacoma Knight instead - although kept the code to prevent
- the re-triggering of an NNI more than once in the same frame.
-1.30
-- Forgot to force GUI surface to be displayed when some dialogs where popped
- up - problem only happened on full-screen mode with triple or double
- buffering enabled, or when using 3dfx mode. It appeared as if Snes9x had
- locked up, but pressing Esc would pop down the hidden dialog.
-- Added a couple of options to the Settings dialog. Now its possible to
- disable S-RAM auto-save which was causing Snes9x to write to the hard disk
- every 30 seconds on some games, causing the occasional skipped frame.
-- Fixed Reset option which was accidentally broken when Netplay support was
- added.
-- Added support for Dirt Racer - it leaves the Super FX chip running all the
- time, so the default CPU emulation method never allocated any time to other
- CPUs and the emulation seemed to lock up.
-- NMI timing changed again. Now an NMI can only be triggered once per
- frame and enabling an NMI after the normal trigger scan line triggers
- an NMI immediately. This fixes display glitches in Ys 5, Stargate and
- Daffy Duck.
-- Fixed the WAI instruction to only 'wake up' once an actual NMI has
- triggered, rather than just waking up when it should have triggered.
- This fixes Battletoads, broken since version 1.29(ish).
-- Changed NMI again. Made reading or writing to PPU register 0x4210 not
- clear NMI pending flag. Seems to allow all the NMI timing sensitive ROMs
- I had on my list to now work without any special hacks. Illusion of
- Gaia now works again.
-- Another NMI fix - cleared the CPU pending NMI flag at start of frame;
- Battletoads intro was crashing without this. A long DMA was stopping the
- SNES CPU so it couldn't and shouldn't respond to the NMI signal from the PPU.
-- Fixed Netplay problem when game didn't have any S-RAM and Sync Using Reset
- was being used. An error dialog was displayed and the client would disconnect
- from the server.
-1.30b#1
-- The Windows auto-frame skip code was broken - badly. It didn't re-sync a
- timer value with timer events being generated, causing Snes9x to deliberately
- stop and wait for an event when it didn't need to, slowing down the overall
- emulation speed and increasing the number of frames skipped.
-- Improved the Windows cheat search dialog - its now possible to compare
- against a value and more comparison functions are available.
-- Finally worked out why Voodoo 3 support was so buggy in Snes9x - the Voodoo 3
- card generates a WM_DISPLAYCHANGE message when switching to Voodoo mode (the
- Voodoo 1 and 2 cards don't); Snes9x thought that some other application had
- changed the screen depth or resolution and tried to adjust its window to
- match - triggering another WM_DISPLAYCHANGE message. No idea how the code
- worked at all; it must have been only by chance and very dependant on the
- driver version you were using!
-- Implemented Netplay on the Windows port - but its buggy as hell. I seem to
- be having major Windows multi-threading problems. Comments I've seen seem to
- suggest that Windows 95/98 don't implement true multi-threading; hmm...
-- Not happy with the current Netplay, so I scrapped it and tried again;
- the protocol is much improved and not using select to control game timing
- seems to have removed lots of the threading-type problems I was having.
-- Attempted to switch to just using Borland's C++ Builder to build the Windows
- port - and failed, again. Although C++ Builder can build Snes9x from sources,
- it can't then link in the asm CPU cores. I had hoped Borland might have
- fixed this with their latest release - they haven't.
-- Several attempts to get Anti Resonance's super-fast sound CPU and sound DSP
- code working in Snes9x, but all failed. Part of the problem was his code was
- written using TASM and the object files it generated would only work under
- Windows - but all my SNES debugging code was in the Linux port. Anti' fixed
- that, and I then had some success getting his code working, but its just too
- unstable at the moment for a main-stream release.
-- Included an option to use Anti Resonance's alternate sample decoding routine;
- it can approximate the wind and noise sound effects heard in several Square
- Soft games.
-- Thanks to Lindsey Dubb for the mode 7 bi-linear filtering code - it
- generates a nice smooth image when a game scales the screen using the SNES'
- mode 7, but you'll a fast machine if you don't want the frame rate to drop.
-- Thanks again to Lindsey Dubb, he improved the colour addition/subtraction
- subtraction routines - they are just a little slower but now mostly perform
- full 15-bit precision addition and subtraction rather than the previous
- 13-bits of precision. Many more colour shades can be seen - look at the
- improved shading on the Mario Kart or F-Zero track for example.
-- Added a reverse stereo option, for people with sound cards that swap the two
- channels.
-- Added a sound config dialog to the Windows port - now you can access extra
- sound options that have always been there, but just no GUI interface to
- access them.
-- Fixed the 32-bit windowed support on the Windows port.
-- Adjusted the NMI timing by a few microseconds to get Metal Warriors working
- again.
-- Added a few more sound playback rate choices. Most modern sound cards allow
- any value to be used from a large range, rather than just a select few, may
- be I ought to add text field so you could just type a value in?
-- Used Factory Setup 4 to build a new installer package for the Windows port -
- just shipping a zip file was confusing novice users and many (mostly AOL
- users) seemed to have an odd program mapped to .zip files, further confusing
- the issue.
-1.29
-- Disabled the SPC700 noise feature simulation used by Chrono Trigger and
- Final Fantasy 3 until I work out why its being triggered by sound effects
- that don't use it.
-- Rewrote/reorganised the DirectX and 3D/fx handling code, now both are never
- enabled at the same time in Snes9X. It might fix the crashing problems some
- Window port users are seeing. Changing between DirectX and Voodoo 3D/fx
- modes now requires Snes9X to be restarted.
-- Tracked down and fixed the Chrono Trigger black screen problem on the Windows
- port: a rogue asm instruction was left in by mistake after some code edits -
- it was only by chance that the code worked on the Linux port.
-- Added some SNES debug options to the Windows port, but disabled by default,
- on the shipped version.
-- Clicking on the column headings in the OpenROM dialog in the Windows port
- now sorts by that column; plus added some slight screen update optimisations.
-- Added an optimisation to graphics rendering: don't add or subtract
- sub-screen from background layers, or clear the sub-screen, if SNES fixed
- colour is black and no background layers are enabled on sub-screen, even if
- ROM tries to enable translucency effects for every background layer.
- Discovered Sonic was doing this, there will be others.
-- Forgot to enable auto S-RAM save on Windows port, oops!
-1.28
-- Warning dialog added to the Windows port - if a ROM is loaded from a
- read-only directory, e.g. a CD, and the freeze file folder is set to be the
- same as the ROM image folder, then a warning is displayed when the game first
- starts.
-- The Windows port now supports 5 joy-pads - Snes9x always did support 5 but
- the Windows port lacked the GUI option to enable and configure it.
-- Added an about dialog to the Windows port.
-- The Windows port now has a simple settings dialog, only one option so far -
- changing the freeze file and S-RAM save directory; much better than having to
- use regedit at least.
-- Added a new cheat search dialog, you can use it to find where games are
- storing life counters, health levels, etc. and then add cheats that stop the
- values from changing.
-- Added a cheat code entry dialog to the Windows port; now Game Genie,
- Pro-Action Replay and Gold Finger codes can be graphically entered and
- edited.
-- Added a master cheat codes on/off toggle, available from the Cheats menu
- on the Windows port.
-- Extended the number of cheats per game from 10 to 75.
-- Changed cheat code to reapply cheat every emulated frame so if RAM is being
- patched the cheat value is continuously applied.
-- Wrote some new cheat search code, the code won't be useful until I get around
- to writing a cheat search dialog.
-- Added automatic cheat code loading and saving using the same file format as
- ZSNES.
-- Rewrote large parts of the Snes9x cheat handling code ready for adding
- cheat dialogs to the Windows port.
-1.27
-- Added a flag to only enable SPC700 noise 'feature' when Chrono Trigger or
- Final Fantasy 3 are loaded - the conditions that I thought were necessary to
- trigger the feature where sometimes being met by other games.
-- Added a simulation of the SPC700 noise 'feature' where some games, notably
- Chrono Trigger and Final Fantasy 3, play samples that deliberately overrun
- outside a 16-bit value, the SPC700 sound DSP then for some reason starts to
- generate a type of noise sound which the games use to generate wind and
- swish type sound effects. Thanks to ZSNES for some of the information.
-- Fixed another sound interpolation problem, thanks to Mikael Bouillot -
- the initial value of the sample byte being played was not being set correctly
- when processing fractional offsets.
-- Added auto S-RAM save option; S-RAM is automatically written to a .srm file
- a few seconds (30 by default) after a ROM writes to it - useful for people
- who were playing games long into to night, only to lose their progress
- after a power cut or machine crash.
-- NMI delay code changed again - the fix for Cacoma Knight was breaking
- Tuff E Nuff; it would seem delaying NMI until the start of h-blank to too
- long, added a cycle counter instead.
-- Fixed yet another clip window bug - clip window was being incorrectly set
- at no range if colour window was enabled but background layer clip window
- was disabled (meaning layer should not be clipped).
- Fixes the sunken ship level on FF5.
-- Worked out (by example) how to add keyboard accelerators to the Windows port,
- now toggling full screen using ALT+Return works.
-- Added mouse-warp to the Windows port so the the cursor doesn't wonder off the
- Window while SNES mouse emulation is enabled.
-- Improved 3dfx support on Windows port - load dialog doesn't drop out of
- bi-linear mode and underlying window zooms to full-screen so its easy to find
- and click on the menu bar with the mouse.
-- Added Mouse and Superscope SNES emulation support to the Windows port, use
- '7' on the keyboard to select.
-- Windows cursor now hidden unless super scope emulation is enabled.
-- Windows port now has command line parsing - cheapo way of adding Game Genie,
- Pro Action Replay cheat codes, disabling sound CPU emulation for the
- corrupt copy of Star Fox 2, etc. Also allows ROM images to be dropped onto
- the Snes9x icon.
-- Cacoma Knight seems to provide proof that Snes9x triggers the SNES
- non-maskable interrupt (NMI) too early. Changed interrupt to trigger at the
- start of the next horizontal blank period. Will have to watch for it
- causing problems for other ROMs.
-- Added a translucency hack - when a ROM tries to create a stipple background
- pattern by enabling pseudo hi-res. and not enabling a background layer on
- one of the screens, Snes9x changes the effect to use transparency effects
- instead (the real SNES can't do transparency effects with pseudo hi-res.
- enabled). Now the water in Kirby 3 is translucent.
-- SA-1 CPU reset bug fixed, now Jumpin' Derby boots and plays but with major
- graphics problems.
-- Fixed nasty asm SA-1 custom hardware read/write bug that was causing the
- course map not to be displayed on Augusta Masters and Pebble Beach.
-- Added SA-1 character conversion DMA support for all SNES depths, now
- Augusta Masters and Pebble Beach work.
-- Merged in minor code changes for Linux running on the Alpha processor. Thanks
- to Sadruddin Rejeb for the changes.
-- Added four more auto-multi-player-adaptor-emulation-off presets based on
- code from Mystagogus.
-- Added DirectX3D output image processing support to the Windows port... and
- removed it again because it causes my desktop machine to lock up. Back to
- the drawing board...
-1.26
-- Fixed memory leak that crept in when SA-1 support was added when loading a
- game freeze file.
-- Added SPC dumping option based on code from Cyber Warrior X that he sent me
- ages ago but I've just found again while looking for something else!
-- Merged in most of the Amiga PPC port source code changes into the main
- source code tree.
-- Keying on a sound channel seems to clear its last-sound-sample-block-just-
- played flag. Chaos Engine/Soldiers of Fortune needs this.
-- Add multi-thread support to the UNIX ports for sound playing - required in
- the Linux port to work around a Sound Blaster Live driver bug and useful if
- you have multiple CPUs in your machine to help spread the emulation workload.
-1.25
-- Added BS 24Mbit ROM memory map, for Derby Stallion 96 and Sound Novel-TCool.
- No idea if it works. Thanks to Nose0000 for the info and code.
-- Corrected unzip code not to loop forever if an encrypted zip file is loaded -
- an error is generated instead.
-- Changed relative SPC700 cycle length for Mortal Kombat 3 to fix sample
- repeat problems - I wish I knew exactly how fast the SPC700 is clocked.
- Maybe I should write a test ROM and run it on a real SNES?
-1.24
-- 3dfx speed hack back again, only disabled when Seiken 3 is loaded.
-- Some minor SA-1 speed ups added - the SA-1 instruction skipping code will
- have to wait until I have more time.
-1.23
-- Corrected a SA-1 reset bug that reset the SA-1 RAM bank pointer back to block
- zero but didn't clear the RAM bank register. Was causing Kirby 3 to crash.
-- Fixed a wave clipping problem with interpolated sound that was causing noise
- on sound output when certain sound samples were played.
-- Fixed a bug in the sync-sound code that could overrun the sound buffer by a
- few bytes causing clicks on the sound output.
-- The sound sample repeat bug that has plagued Snes9x ever since is was called
- Snes96 finally bit the dust - Snes9x continued to play sample loops
- even if the game dynamically updated the sample not to loop. Fixes the
- stutter in the Mortal Kombat series and improves the sound from several games
- that download sound samples in real-time as they are played.
-- Rewrote the code the handled the SPC700's 64 byte shadow RAM area to fix a
- possible sample corruption problem with ROMs that stored samples that
- cross the 64 byte start area.
-- Added code to allow ROMs to change the sample being played the next time the
- channel loops or is keyed on - not sure if it fixes anything but seems more
- correct.
-- Added a zero-frequency fix to the stereo sound mixing code that I'd already
- added to the mono code some time ago.
-- Changed the code to set the end-of-sample flag just before the last block is
- played, rather than just after. Seems to help improve the sound on some
- games.
-- Sound sample start code now doesn't reset the channel's envelope volume level
- to zero before starting the sample - helps reduce the clicks being heard when
- a channel envelope volume level hadn't reached zero before being keyed on
- again.
-- Changed initialisation of sample-end-register to 0 rather than 255 - seems
- more logical now I've thought about it. Not sure if it helps anything.
-1.22
-- Finally fixed the corrupt copy of Donkey Kong Country not working problem -
- Snes9x thought the ROM used the same memory map as Street Fighter Alpha 2.
-- Added explode, un-shrink and un-reduce decompression modes support to the
- unzip code.
-- Fixed offset per tile bug that crept in after me trying to fix the Starfox
- on-tilt bug.
-- Made some fixes to the C Super FX emulation code, enough to get most 'FX
- games playable on the Mac port.
-1.21
-- Finally worked out how character DMA worked on the SA-1 and implemented a
- hacky, slow version, but its enough to get the level up screens displaying
- correctly on Mario RPG.
-- Incorporated ZSNES' new optimised Super FX asm code - had to track down and
- fix a nasty memory overwrite bug in the code first to get it to work.
-- Changed sample mixing code to not automatically wrap offsets to
- keep inside the sound buffer, external port code is now expected to do that.
- Helped me fix a problem in the Windows port that prevented very large sound
- buffers from working, which are required for some badly written sound card
- drivers.
-- Corrected a bug in the SA-1 C code where incorrect processor emulation
- functions where called if the code was compiled with in-lining turned off.
-- Fixed crash bug in Super Mario RPG on the level up screen - forgot to mask
- the enable bit from the RAM bank register. Thanks to Christian Wolf for
- sending me a freeze file which made it easy to find the problem.
-- Fixed a lockup bug in the window clipping code, if the ROM ever turned off
- the sub-screen completely the clipping code would enter an infinite loop.
- Fixes The Cartoon Addams.
-- Made the Daffy Duck NMI fix only enable when Daffy Duck is loaded - fix was
- causing problems for Breath Of Fire 1 and 2.
-1.20
-- Windows port no longer sets DirectSound to exclusive mode, so its now
- possible to hear sound output from Windows apps while Snes9x has focus.
-- Fixed the freeze file loading and saving on the Windows port.
-- More GUI settings are saved in the registry on the Windows port now.
-- Added 3D/FX image scaling/filtering support to the Windows port.
-- Added the TV mode from the Mac/Linux ports to the Windows port.
-- Incorporated Kreed's new output image routines into the Windows port that
- fixes RGB555 display colour problems. Many thanks to Kreed.
-- New auto-frame rate timing code on the Windows port, stops the silly speed
- up problems when the old code tried to 'catch up' after the emulator had
- been paused.
-- Increased the DirectSound secondary buffer length on the Windows port to
- hopefully fix all the static/broken sound output problems some people were
- experiencing.
-- Altered the ZSNES Super FX asm code so the Windows port could use it - all
- previous versions of the Windows port were shipped using the C Super FX
- emulation code which is a lot slower.
-- Implemented interpolated and sync-sound options on the Windows port.
-- Added an image stretch option to the Windows port - stretches the SNES image
- to fill the whole screen or the Window. Looks really good on my TNT card
- since that chips seems to filter the image as it scales it.
-- Implemented Windowed mode on the Windows port.
-- Added special SPC700 cycle timing for Empire Strikes Back.
-- Fixed the missing polygon problem for Super FX games - thanks to zsknight
- for the information.
-- Implemented SA-1 support required for Mario RPG, Kirby Superstar,
- Paradius 3, etc. but since only a good image of Mario RPG exists, I could
- only test that game.
-- Fixed a graphics clip window bug: inverting the area of a clip area that
- only consisted of empty bands should become the full width of the screen;
- Mario Kart's rear-view mirror display needs it.
-- Fixed mode 7 render code to use correct z-buffer when rendering onto the
- sub-screen. Fixes Final Fantasy V title screen.
-- Added horizontal offset per tile support in the offset per tile modes 2
- and 6, and switchable horizontal/vertical offset in mode 4. Fixes Chrono
- Trigger in several places and Mario All Stars title screens.
-- Changed SPC700 relative cycle length to 14, needed for Stunt Car Racer.
-- Enabled immediate triggering of NMI if NMI enable flag set while scan-line
- was on first line of v-blank. Needed to fix a background jitter bug in
- Daffy Duck: The Marvin Missions.
-- Altered ROM load code to ignore corrupt ROM map type byte in ROM header,
- preventing the code erroneously detecting what it thinks are interleaved
- ROMs. Fixes EEK! The cat, Formation Soccer, the corrupt copy of Donkey
- Kong Country, ...
-- Disabled IRQ re-triggering if V-IRQ registers set to the current line. Fixes
- Chuck Rock.
-- Fixed missing sprites in Andre Agassi Tennis - writing to low byte only of
- the sprite write address register seems to also clear the hi-byte.
-1.19
-- Games written by the Japanese software company Human seem to need special
- SPC700 sound CPU timing, so the ROM load and reset routines now check the
- software author company and adjust the CPU cycle length accordingly.
- It gets Clock Tower, Super Fire Pro-wrestling Premium, etc working.
-- Added ROM check sum calculation and testing code - Snes9x can now detect
- pure, corrupt or hacked ROMs.
-- Noticed a fast way to implement the SNES 4096 colour mode, so I implemented
- it. Now the colours in ActRaiser 2 look correct.
-- Corrected a noise frequency error thanks to information from Takehiro.
-- Added a 'start in full screen mode' flag to the Linux port.
-- While debugging the new graphics code I thought of a fast way to implement
- the SNES direct colour mode, tried it out and now the colours in Actraiser 2
- are correct.
-- Blast, forgot about the colour window and fixed colour effects. The separate
- sub-screen is back again, but all the other graphics speed ups are there.
-- Now I've got a z-buffer I keep finding other ways to optimise the SNES
- graphics rendering - no need for a separate sub-screen, no need to clear
- the sub-screen to the fixed colour, no need to waste CPU time on translucency
- effects on hidden pixels, no need to completely clear the main-screen to the
- back drop colour, etc., etc.
-- Implemented a software z-buffer and changed the SNES graphics rendering to
- use it (required change for future 3D card support). Finally fixes the
- sprite-to-sprite priority bug that some games suffer from. Also a big speed
- increasing for some games (10 fps+), others are slight losers.
-- Added code to skip the rendering of completely transparent graphic tiles
- rather than comparing each pixel to see if it is transparent; helps the
- frame rate a bit on some games.
-- Added a fixed for Tetris & Dr. Mario - the game didn't like a multi-player 5
- adaptor plugged in to the real SNES when being played, so turned off the
- adaptor emulation for this game.
-- Added hack for Final Fantasy II - if sync sound isn't on, make attack rate of
- 1ms actually 0ms (old v1.16 behaviour). Causes a slight click but its better
- than samples being cut short.
-- Fixed a clip window area invert bug if the colour window was enabled on
- on one window and the other window was being used to clip a background layer.
- Fixes the finial (I hope) display problem with Gun Hazard.
-- Added code to intersect the clip window areas if both a colour window and
- a background layer clip window were enabled at the same time. Required by
- Gun Hazard.
-- Forgot to mark graphic clip windows as needing recomputing when the master
- colour window inside/outside/on/off/main-screen/sub-screen PPU register was
- updated. Was causing display problems for Gun Hazard.
-- Internal H-DMA execution accelerator pointer variables where not always
- being recomputed when started H-DMA part way into a frame. Was causing
- display problems for Gun Hazard.
-- Made H-DMA continue for one extra scan-line to fix a disappearing monster
- problem in Dragon Quest 5. Thanks to Alex Jackson for the bug report.
-- Zoop seems to require volume envelope height reading by the sound CPU to
- always return 0 when the channel is in gain mode.
-- The sound code was ignoring updates to the ADSR volume envelope rates while
- one was in progress. Fixed that and now the bird song at the start of
- Chrono Trigger sounds correct.
-- Had to disable the CPU shutdown code for loops reading the horizontal beam
- position, it was causing problems for Star Fox. Still no polygons though.
-- Oops, sound DSP noise output was broken - accidentally deleted an important
- line while removing debug code ready for the last release.
-- Added initial 3Dfx support to the Linux port - basically using the Voodoo
- card as a bi-linear filtering, scaling blitter. Actually slightly slower than
- TV mode, for non-scrolling images due to poor texture upload speeds to the
- card, but the full-screen feature is nice and the speed doesn't drop as more
- of the screen changes.
-1.18
-- Implemented a sync-sound mode where sound data gets generated in sync with
- SPC700 instructions being executed. Finally the sound Williams Arcade
- classics can be heard. Also helps slight sound timing problems in many other
- games but doesn't fix Mortal Kombat 2 like I thought it would - its
- sound routine programmers must have been on drugs or something!
-- Added interpolated sound - gives low frequency sounds much more bass similar
- to a real SNES especially with the playback rate ramped up to 44KHz.
-- Added on-screen messages as various emulation options are toggled on and off
- using the in-game keys.
-- Fixed a PPU register read bug with the sprite register write position. Thanks
- to Takehiro TOMINAGA for the bug report.
-- Altered the auto-frame skip timing code to only wait and re-sync to the end
- of frame when frames haven't been skipped. Again thanks to Takehiro.
-- Speeded up the colour addition and subtraction code using ideas from
- Takehiro.
-1.17
-- Linux and UNIX sound code now driven directly from signal timer handler
- rather than the timer handler just setting a flag which had to be polled in
- the main emulation code. Slightly faster execution.
-- Fixed the crash bug in the ZSNES Super FX asm code with Vortex - the game's
- polygons still aren't visible though.
-- Implemented bent-line increase and exponential decay and sustain volume
- envelopes - they should match, or at least be very similar to the real SNES
- sound DSP chip now.
-- It would seem ROMs can key on sound channels even if the channel hasn't
- been keyed-off, Pac-In-Time requires it. Changed code to allow it.
-- Quick mod to ZSNES Super FX code to get Winter Gold working - it was already
- working with the C Super FX code.
-- Added emulation of the extra 1/2 scan-line per frame on PAL and NTSC -
- should help improve music speed emulation.
-- Worked around the click sound heard when ROMs use 0 volume envelope attack
- rate.
-- Removed the 'check for IRQ already happened' H-IRQ position register setting
- code - it was causing problems for Ninja Warriors and was not required by
- F1 Grand Prix.
-- Fixed a bug in the new sound code - the sustain part of the
- attack-decay-sustain-release volume envelope was being skipped if the
- sustain level wasn't at 100%. The fix has helped some music notes from
- being cut off early in a few games.
-- Added fix to Pro Action Reply support (again). Thanks to Paul Shoener III for
- the original fix and Gil Pedersen for reminding me to apply it!
-- Finally fixed the Tales of Phantasia 'bum note' problem! The ROM set its
- sample directory to the upper-most page and I forget to code for the hidden
- 64 bytes of RAM, that appear when the boot ROM is switched off, when fetching
- sample addresses.
-- Adjusted the relative cycle length between the 65c816 and the SPC700 slightly
- to get Terranigma working again.
-- Oops, the emulated joypads 3 and 4 via the emulated Multi-player 5 interface
- weren't working. Thanks to Steffen Schwenke for the bug report.
-- Optimised the echo sound code - by-passed the the FIR filter code if only
- a pass-through FIR filter was defined by the ROM.
-- Modified V and H-IRQ register changing code to trigger an IRQ immediately if
- V-IRQ is enabled and the scan-lines match and either H-IRQ is not enabled or
- the electron beam position has already gone past the trigger point. Fixes
- the screen flicker in F1 Grand Prix.
-- Modified the priority-per-pixel mode 7 code to use BG#1's clipping data if
- the top bit of the mode 7 pixel is set. Fixes initial track drive-through
- display in F1 Grand Prix.
-- Modified the sprite priority levels for the priority-per-pixel mode 7
- display. Now the car can be seen in F1 Grand Prix.
-- Wrote a sound DSP register recording scheme which 'plays back' the register
- changes in sync with the sound generation code. I'm bit disappointed, it
- only improves the sound in a very few games... Scrapped the code, it actually
- causes more problems than it fixes. Oh, well, another 3 weeks work wasted...
-- Fixed a SPC700 wake up problem for Lufia I - made the SPC700 also wake up
- when the 65c816 read from one of the four comm ports.
-- Included lots of sound code speed ups and sound quality improvements
- from Takehiro TOMINAGA - many thanks go to him.
-1.16
-- Fixed a case where the -forcelorom option didn't work - the case was
- required for Formation Soccer which claims in its ROM header to use the
- same memory map as Super FX ROM, it doesn't.
-- Pulled apart a real SNES using a crowbar (great fun), just to look at what
- speed the SPC700 is actually clocked at for more accurate relative emulation
- speed.
-- Implemented SPC700 cycle counting in the hope the improved timing would fix
- Tales'; no such luck but at least the -ratio option is obsolete now.
-- Implemented executing SPC700 instructions during DMA, fixes BSZelda and
- Goal lock up at start and music pausing briefly when ROMs do lots of DMA,
- usually between game screens.
-- Scrapped the i386 asm SPC700 code - it was the cause of the music not
- restarting after a battle in Chrono Trigger and FF3 and I didn't realise
- because the bug had already occurred in the test freeze-file I had.
- Thanks to John Stiles for pointing out that the Mac port didn't have the
- missing music problem.
-- Fixed RGB subtraction bug on displays with only 5 bits for green, e.g. RGB555
- displays. The GREEN_HI_BIT variable was always set to a value for 6 bit
- green displays.
-- Added the SA-1 memory map, still a long way to go before any SA-1 game will
- run.
-1.15
-- Jumped versions to keep in sync with the DOS port release.
-1.14
-- Improved 8-bit sound generation slightly, but it still sounds very poor
- compared to 16-bit sound.
-1.13
-- Implemented the Tales of Phantasia memory map using the information supplied
- by zsKnight. Had to also implement a de-interleave routine to work around
- a ROM feature and Snes9x CPU instruction fetching implementation detail.
-- Added a frames-per-second on-screen display option.
-- Fixed the final glitch bug with the Mario Kart track display - the byte code
- for the termination of the DSP1 raster command wasn't been recognised.
-- Disabled a NMI/DMA hack for Rise of the Robots, was causing problems for
- Mario Kart and 'Robots wasn't working correctly anyway.
-- Optimised the mode 7 rendering a little.
-- Changed tile rendering code to use offsets into screen buffer rather than
- direct pointers ready for z-buffer implementation.
-1.12
-- Changed V-blank NMI to occur immediately after a WAI instruction, Toy Story
- required this.
-- Fixed reading of H-DMA line counter register, Top Gear 3000 needed this.
-- Ripped off large parts of ZSNES's DSP1 code (with _Demo_'s and zsKnight's
- approval). Now Mario Kart works almost 100%.
-- Added a check to see if a vertical scan-line IRQ register change will cause
- a H-IRQ later on the current scan-line. Pilot Wings needed this.
-- Fixed possible crash bug in clip window code when both windows had two
- spans. Could actually cause Chrono Trigger to crash the emulator.
-- Fixed a lock-up problem with the C Super FX code, Star Fox and executing
- a few 'FX instructions per scan-line (required for Winter Gold).
-1.11
-- Partially fixed the DOS netplay server - the server timer is running too
- slowly and it doesn't deal with disconnects correctly yet.
-- Corrected the sound echo delay - it was varying with the sound playback
- rate chosen by the user - it shouldn't have been.
-- Implemented DOS netplay code - DOS server code still not working though.
-- Removed all floating point calculations from the sound generation code.
-- Fiddled with the pitch modulation code - my guess is the output of a
- channel that is used to modulate the frequency of another channel is
- automatically muted by the SPC700 chip. Just a guess, but the wind from
- FF3 sounds 'better' but far from perfect.
-- Optimised the tile palette index calculation.
-- Optimised the planar to chunky tile conversion code.
-- Fixed X11 port to always scale SNES image if hi-res. only (no interpolation)
- support is enabled.
-- Added zipped ROM image support using Gilles Vollant unzip code and
- some code that Ivar (Lestat) sent me a long time ago.
-- 65c816 asm RTI instruction was destroying the program bank in emulation mode,
- the C code was already correct. Caused C64E to break.
-1.10
-- Finished NetPlay v1 - allows up to five networked machines to play
- multi-player SNES games, one player on each machine.
-- Switchable full-screen mode added to Linux X11 port, some code and ideas
- nicked from Maciej Babinski's original Snes9x XFree86 DGA Linux port, the
- UAE Amiga emulator, plus lots of my own code.
-1.08
-- Bug fixes to C Super FX emulation - now Winter Gold works correctly again.
-1.07
-- More DSP1 work. Mario Kart is now playable! The character projection code
- is still broken so the opponents and obstacles aren't always positioned
- correctly on screen and you keep bumping into them, but I can still keep
- coming first!
-- Started work on NetPlay support.
-- Decreased sound card DMA buffer size on DOS port to improve sound generation
- and sound CPU synchronisation in some games.
-- Included Linux joystick driver patches from Vojtech Pavlik so the port can
- use the new v1.x joystick drivers, again written by Vojtech Pavlik. Allows
- use of Micro$oft Sidewinder pads, NES and SNES pads, PlayStation pads,
- Gamepad Pros, etc.
-- Added halve-the-result colour subtraction.
-1.06
-- Extended code to allow support for multiple 16-bit screen formats,
- switchable at run-time, rather just supporting one, selectable at compile
- time.
-- Added XFree86 DGA Linux port - code from Maciej Babinski.
-- More fixes to the X11 image format conversion and setup code.
-- The asm SetByte routine wasn't wrapping writes to S-RAM correctly, allowing
- some ROMs to think they were running on a copier and put up an error
- screen. Thanks to Nu of #rom for the report.
-- Added 'TV-Mode' support (interpolation and scan-lines) to the DOS and
- UNIX ports from code based on John Stiles work.
-- Added v-sync option to the DOS port.
-- Added fix to Pro Action Reply support, thanks to Paul Shoener III.
-- Added ggi support (untested) to Linux port using patches from
- Alexander Larsson (alla@lysator.liu.se).
-- Added 16 to 24/32 bit image conversion routines to the UNIX X11 code.
-- The SPC700 OR1 instruction was broken. Thanks to Pyrgopolinices for the
- report.
-- DOS port was having trouble splitting and joining path names - caused
- problems when specifying the full path name of a ROM when the ROM image
- was on another drive.
-- If a ROM reset the sound DSP and then turned on echo effects but kept
- the same echo delay setting, then the echo effects could not be heard.
- Thanks to madec@mclink.it for the bug report and freeze file that made it
- easy to find the problem.
-- DOS port was always using stereo sound setting, if sound card
- supported it, regardless of the user preference.
-- Linux port X11 port could crash if window was resized while transparency
- effects were enabled.
-- The colour subtraction accelerator look-up table was slightly wrong, causing
- one bit of red, green blue values to 'spill' into the next field.
-- Allowed colour window to cut a hole in the main-screen and show the sub-
- screen underneath. The effect is used by Illusion of Gaia.
-- Added support for colour subtraction, with the halve-the-result flag
- set.
-- Included DSP1 code from _Demo_. Now you can see the track in Mario Kart and
- the ground in Pilot Wings - still can't play the games though due to other
- missing commands.
-- Added an NMI hack to work around a code bug in Battle Toads: BATTLEMANIACS,
- its only by chance that the game works on a real SNES - And disabled it
- again because it causes problems for Chrono Trigger.
-- A frame skip of zero was actually still skipping one frame. Thanks to
- Marius Fodor for the info.
-- And yet more X-OR window bug fixes - now the effects during some of the more
- 'posh' spells look correct in Chrono Trigger.
-- Yet another window area inversion bug - off by one pixel on right-hand edge.
-- Forgot to put dummy start and end points for XOR window combination modes -
- now Uniracers looks correct and Sailor Moon looks like it does on a real
- SNES.
-- Window clip code was using wrong index into a 2-dimensional array when
- the whole of the main or sub-screens were clipped.
-1.05
-- The master volume disable code was looking that the wrong variable!
-- Fixed crash bug in newer sound code if a ROM tried to start a sample
- playing who's data went past the end of SPC700 memory. (Cannon Fodder)
-1.04
-- Fixed DSP1 ROM header detection bug.
-- More DSP1 work; still nothing works, although I know the multiply command
- is correct because I've compared the results against a real DSP1.
-1.03
-- Oops, the multi-player 5 disable code change broke the multi-player 5 being
- the default controller.
-- Implemented the colour window on the main screen - now Zelda's oval zoom
- window displays correctly and Krusty's Super Fun House clips the left-most
- 8 pixels as it does on the real SNES.
-- TERRANIGMA didn't like me returning a random value when it attempted to
- read a channel's the current sample byte.
-- Hacked in initial support for mode 7 priority-per-pixel - the priority bit
- doesn't actually change the priority of the pixel but the two games that I
- know of that use the feature look OK. (Winter Extreme Skiing and the
- intro of Tiny Toons Adventures).
-- Colour addition/subtraction code now uses RGB565 rather than RGB555
- calculations - helps a little with the loss of the bottom bit of SNES
- colour data.
-- DSP1 emulation started - nothing works yet.
-1.02
-- Switched to adding back drop colour rather than fixed colour when
- sub-screen addition is enabled but there's nothing on the sub-screen.
- Uniracers seems to need it. - DISABLED it again. Causes problems for
- other ROMs and Uniracers itself on later screens.
-- Fixed XOR window logic combination mode and area inversion code, now
- Uniracers works correctly.
-- Oops, if colour window and half colour addition/subtraction were both
- switched on, area outside colour window was still being halved, it shouldn't.
- Hacky fix at the moment until I implement the correct fix.
-- Fixed several bugs with the mosaic effect and 16x16 tiles and a few
- possible background scroll offset bugs and the mosaic effect.
-- Optimised the sound sample generation code for cases when the SNES
- sample playback frequency was higher than the sound card playback rate.
-- Fixed possible click sound when a sample was first started to be played.
-1.01
-- Corrected scan-line count for PAL games - should be 312 lines verses 262 for
- NTSC. Was causing slow music on PAL games.
-- Added error correction code to the SPC700 timer update code - the
- SPC700 timers are updated using the emulated h-blank handler which is
- called every emulated 63.6 microseconds (15.720KHz) but the SPC700 timers
- need to be updated at multiples of 8KHz, hence the error. Was causing
- music to be played slightly too fast.
-- Switched back to using C SPC700 code - the old SPC700 asm code was lacking
- several optimisations that the C version had. It also had multiple
- speed hack cycle skipping bugs. Plus I hadn't even finished optimising
- all the code from the last time I converted the C compiler output.
-- Optimised SPC700 memory access routines a little.
-- Disabled code that prevented ROMs updating SPC700 timer values while the
- timer was running - it seems like it is allowed, even though docs on the
- 'net I've seen say its not.
-1.0
-- Fixed SuperScope support.
-- Added hi-res. option to my DOS port.
-- Fixed 4, 6, and 8 button standard PC joystick support.
-- Changed some types the source code was using BYTE -> uint8, WORD -> uint16,
- DWORD -> uint32 and BOOL -> bool8, types were clashing Windows typedefs
- but sizes didn't always match.
-0.99
-- 8-bit double height and/or width tile rendering was missing every other
- group of 4 pixels - screen pointer advance count was wrong.
-- Asm SPC700 emulation was ignoring the Shutdown flag - the result is its
- not possible to turn off cycle skipping for the SPC700 emulation.
-0.98
-- CPU to ROM address decoding code rewritten - used by Game Genie cheat codes,
- orginal code might have been the cause of some Game Genie codes not working.
-- Started to remove printf calls and replace them with calls to S9xMessage,
- port code can then dicide what to do with message.
-0.97
-- Re-enabled decompressed sample caching, still has a possible click problem
- but the sound code is a lot faster with it enabled. Added command line option
- to disable it if required.
-- Added '7' key support to rotate through available controller options, in
- the order multi-player5, mouse on #1, mouse on #2, superscope,
- standard controller and then back to multi-player5.
-- Hi-res. (512x448) support fixed.
-- Mouse support completed - Lemmings 2 and Mario Paint working a treat.
-- More colour window fixes.
-- Fixed freeze game problem when ZSNES SuperFX code is being used -
- ZSNES 'FX state was not being saved and restored.
-- ZSNES SuperFX asm emulation code plugged in to Snes9x.
-0.96
-- Looks like if the colour window is not enabled at all and the colour
- window selector is defined to only allow colour effects inside the colour
- window, then no effects should be visible.
-- Offset-per-tile rendering code didn't support width 64 screen size, which
- Chrono Trigger used on its title screen.
-- Contra 3 seems to prove that defining the clip window area to be 'outside'
- a window that covers the whole screen is not an area with no range.
- - No it doesn't. It proves that I shouldn't have initialised the right
- window edges to 255! Contra 3 enables clipping windows without first
- defining their range.
-- Debug frame advance feature was being prevented from forcing the next
- frame to be rendered by SyncSpeed which was being called after the
- debugger returned to the main loop.
-- H-DMA code was allowing ROMs to manually start H-DMA during the v-blank
- period, ROMs shouldn't be allowed to do this.
-- Asm code would not push the correct CPU status onto the emulated stack if
- returning from an NMI immediately triggered an IRQ - fixes Mortal Kombat 1
- and War of the Gems.
-- 'd' dump memory debug command was not preserving the CYCLES count.
-- C versions of SNES memory access code had same problem as asm code on the DOS
- port except it didn't cause a crash just ROMs failed to work correctly.
-- Asm i386 code was using signed compares to check for special case memory
- areas - it was causing crash problems on the DOS port which was sometimes
- returning valid address values with the top bit set - i.e. they seemed
- like negative values!
-- Changed event reschedule code to always allow h-blank start events, used to
- disable them during v-blank period.
-- Added code to HDMA when end of visible lines reached.
-- Changed register 4212 code not to always return h-blank when in v-blank.
-- Clipping fixed colour addition to background area was off by one pixel on
- the right-hand edge.
-- HDMA: Finally worked out how the real SNES operates when ROMs manual
- start H-DMA during the frame - ROMs must set up the H-DMA line count
- and address values before H-DMA is started.
-- Fixed the asm code to remove all hard-wired structure offsets - one offset
- into the IPPU structure was wrong in the code because the structure had
- changed size.
-- Added colour window support and allowed graphic window settings to be
- different on the main screen and sub screen, just like a real SNES.
-- SuperFX LJMP instruction had bank and address values swapped.
-- Fixed possible memory overwrite problem because OBJList array was one
- element too short.
-- Added AND multi-graphic window combo support.
-- ROM image memory allocation allocates an extra 32K of RAM, then moves the
- pointer forward by that amount - stops the SuperFX emulation from accessing
- unallocated memory, possibly causing a crash.
-- SuperFX emulation now stores sign and zero flags in separate variables so
- the MERGE instruction can set flags correctly.
-- Added 65c816 instruction skipping to i386 asm code when 65c816 waiting in
- a simple loop for some 'event' to happen e.g. end of frame NMI.
-- Finally fixed the APU instruction skipping problem with the i386 asm
- code when the WAI instruction is used - caused slow music on some ROMs.
-- Offset-per-tile modes don't seem to support screen size - Mario All Stars
- Super Mario 2 requires this on title screen. Doesn't seem to effect
- Tetris Attack or Puzzle Bobble.
-- Changed SNES select and start keys from shift and control to space and
- enter - allows shift-fn key to save game positions without the SNES ROM
- also getting a select joypad button press.
-- Multiplayer5 support for controllers 3+ was broken for ROMs that used
- automatic hardware joypad reading rather than reading joypads serially.
-- ResetPPU was not clearing tile caches and marking OBJ as need recomputing.
-- Cached OBJ positions and sizes were not being recomputed if ROM changed
- global OBJ sizes during frame.
-- Fixed brightness multiplication problem on 16-bit code for green.
-- SPC700 emulation now uses one variable to store ZERO and NEGATIVE flags.
-- SPC700 emulation now only increments PC once at end of instruction.
-- New ROM type and interleaved detection code.
-- Reading sound DSP register ENDX also clears the value. The docs on the
- 'net said that only writing to the register cleared its value. Fixes
- sound in Zoop.
-- Fixed mode 4 colour palette problem on background #2 in tile-based graphics
- code.
-- Fixed graphics mode 4, offset-per-tile support. Only one set of offset data
- that is switchable between horizontal and vertical, unlike modes 2 and 6
- which allow separate horizontal and vertical offsets per tile.
-- Modified the APU timer code again, if the timer is enabled, a write to the
- timer target register is only allowed if a value hasn't been written yet.
- Fixed Donkey Kong Country 1 and Earth Worm Jim 1 & 2.
-- Attack rate of 0ms changed from 1ms back to 0ms because of a group of ROMs
- that change from attack mode to decay mode in real-time. Will change back
- when I've added better SPC700 CPU and sound generation sync code.
-- Added support for ROMs set a new sound timer value while the timer is
- enabled (EWJ 1 & 2).
-- Added support for ROMs that read the sound envelope height (MK1, MK2, etc).
-- ROMs writing to the H-DMA enable register during visible scan-lines were
- restarting H-DMA for that frame causing random screen effect corruption.
-- Echo feedback seems to be after the FIR filter, not before as a diagram I've
- seen suggests.
-- Sound pitch modulation added.
-- Memory access routines changed to pass a single 24-bit address rather than
- the previous separate 8-bit bank and 16-bit address parameters.
-0.3
-- Updates to A-Bus address during a frame must not update H-DMA address.
- Fixes Actraiser 2 and Pacman 2.
-- Removed sound volume mangling - with echo support enabled it doesn't seem to
- be required.
-- Attack rate of 0ms changed to 1ms to help prevent click sound with sudden
- start of a sample playing.
-- Sample caching of samples that looped using part of the original sample
- created a click on the sound output. Caching disabled for the moment. Would
- require 512K of cache RAM to fix sample caching.
-- Colour addition/subtraction support added - but still a little buggy in
- places and very slow.
-- 16-bit colour support added.
-- Sustain sound volume was not being set if a sample using ADSR was started
- with both the attack rate and decay rate set to zero - resulted in missing
- sound samples on with some games.
-- Sound echo support added.
-- Sound channel mixing code was not completely clearing a channel's sound
- buffer when a channel finished playing a sample.
-- Sound mixing code rewritten to use one buffer, rather than writing each
- channel into a separate buffer then combining them into one buffer.
-- Memory access routines rewritten to use an 8K block lookup table rather than
- dedicated code for each ROM memory map - it was getting difficult to support
- the new types of SNES ROM memory maps becoming apparent.
-- Sound sample decoding wasn't decoding sound samples correctly if a
- previously cached sample was only partially overwritten by the ROM as
- opposed to being completely replaced.
-- Sound sample decoding wasn't clipping generated sample values correctly.
-- Changed H-DMA to start in the current frame only if enable register is
- written to during v-blank, h-blank or while the screen is blanked.
-- The SPC700 seems to start executing instructions before the 65c816 -
- shorter reset pulse? (NO - forgot the SPC700 executes instructions while DMA
- is taking place).
-- ROMs that reset the H-IRQ position so another IRQ would be triggered on the
- same scan-line where not supported - Super Off-Road: The Baj needs it.
-- $4212 bit 7 needs to go high at the end of h-blank at line 224 not at the
- start of h-blank - Bubsy needs it.
-- Sample decoding routine could write to memory outside sample cache area if
- address of block to decode was greater than $0x10000 - 9.
-- Walking mario can be seen on map screen of MarioWorld - needed sprite
- priority rotation working. ROM sets bit 7 of $2103 then sets rotation in
- $2102. Reset rotation at start of v-blank not at end.
-0.24
-- Fixed reading of DMA register values - now Ms Pacman works.
-- Saved sprite memory address being restored on the wrong scan-line - caused
- corrupt sprites on at least one game (GANBARE GOEMON 2).
-- Screen colour palette not being updated if ROM only wrote to low byte of
- palette register.
-- Possible memory corruption fixed if a ROM tried to write to an invalid
- sprite address via PPU registers.
-- X11 port support quick load and save by pressing function keys to load or
- shift + function keys to save.
-0.23
-- Added option to disable graphic window effects - T2: The Arcade Game doesn't
- seem to like them.
-- Mode 7 "outside screen area" register interpretation fixed - now the
- Actraiser map screen looks a lot better.
-- Old DMA code hack for Battle Toads: Double Dragon removed as it was no
- longer required and it was causing problems for Ys III.
-- Lowered max volume level of 16-bit sound mixing code to help with sound
- clipping problems is lots of SNES sound channels are playing.
-0.22
-- Crash bug fixed in mode 7 graphics windows code
-0.21
-- Fixed a noise channel volume bug - noise waveform was getting clipped.
-- Fixed 24bit X Window System server support on the Solaris port.
-- Sprites in priority level 1 on mode 7 were being drawn incorrectly behind
- graphics screen.
-- BG 3 priority 1 tiles sometimes not drawn dependent on the $2105 bit 3
- setting.
-- Added graphic window support the tile redraw code.
-- Added mosaic support to tile redraw code.
-- Tile redraw code was drawing one line too many on screen-splits.
-- Tile-based redraw code made more intelligent about when a background should
- be displayed or not.
-- Added wrap within bank support to large DMAs just to support Rock 'n' Roll
- racing.
-0.20
-- DMA routines added lots of special cases and removed most calls to GetByte,
- using a pointer instead.
-- Multiple using PPU registers is now only computed when first byte of result
- is actually read.
-- Sound enabled by default if compiled without DEBUGGER defined.
-- Tile redraw method made the default.
-- Fixed CPU_SHUTDOWN so SPC700 continues to execute even if main CPU is
- "skipping" cycles waiting for an event to happen.
-- More command line options added.
-- Default cycles-per-scan-line to execute lowered to 90% from 100%.
-- +/- keys now work even if auto-frame rate adjust was enabled.
-- SPC700 emulation partially rewritten in assembler.
-- Asm 65c816 code change to use same speed up techniques as the C++ code.
-- Minor speed tweaks to the sound decoding and mixing code.
-- C++ SPC700 emulation changed to use same method as 65c816 emulation for
- computing and storing emulated CPU flags.
-- Mode 7 code rewritten and several scrolling offset bugs fixed.
-- Lo-ROM S-RAM memory map bug fixed - now Uniracers works.
-- Multiple speed ups and changes to the tile and line-based redraw code.
-- Tile and line redraw code changed to cache converted tiles between frames.
-- Variable cycle length timing made compile-tile switchable.
-- C++ 65c816 emulation changed to use several opcode jump tables to avoid
- a register size comparison test on most emulated instructions.
-- C++ 65c816 emulation changed how is computes and stores emulated CPU flags.
-- Fixed high frequency sound playback bug - the sample rate calculation was
- blowing the range of an unsigned long.
-- Fixed V-RAM reading so DKC3, Addams Family, Aladdin and Pacman all work.
-- Fixed sound code so ROMs can change from ADSR mode to decrease mode - fixes
- lots of ROMs.
-0.12 released
-- Added dynamic speed regulation.
-- TCALL vector calculation change from n to 15 - n.
-- Fixed crash bug if ROM writes to sound DSP register numbers greater than
- 127.
-- Fixed DOS memory locked for interrupt code.
-- Added long name versions of command line switches.
-- Added command line switch for SPC700_SHUTDOWN code and WAI cycle skipping
- code.
-0.1 released
-- All DOS memory is now locked from being swapped.
-- Fixed DOS port keyboard polling code - could get confused if a keyboard
- interrupt happened while keys were being checked.
-- SPC700 ADC instruction never cleared Overflow or Carry flags!
-- Changed selection of playback speeds for Solaris port.
-- Sample caching code was broken - cached samples were never used.
-- Added code speed ups for ROMs that use a lot of DMA to VRAM.
-- More cpu code asm speed up.
-- Fixed 16x16 size tiles on tile-based redraw code.
-- Fixed sound gain-mode increase and decrease volume envelopes.
-- Added code to support ROMs that reuse sprites in the same frame.
-- Fixed processing of negative volume levels.
-- Fixed SPC700 EOR1 instruction.
-- Added SPC700 shutdown code to stop executing SPC700 instructions if in
- a tight loop waiting for a timer or for the 65C816 to respond.
-- DOS playback rate was being forced to 16KHz by Allegro - fixed.
-- Fixed bug in SPC700 MOV1 C,bit, address.
-- Fixed a off-by-one loop sample pointer bug in MixSamples.
-- Added command line flags for cached-tile based drawing and sub-screen
- background layers priority swapping.
-- NOPE, got encoding of the OR1/EOR1,AND1 range of correct originally -
- got duff information from an "SPC700" programmer.
-- More SPC700 fixes: got the encoding of the OR1/EOR1,AND1 range of
- instructions wrong - I guessed wrong originally.
-- Sample looping bug fix on mono sound mixing code.
-- Sound pitch value no-longer clipped to 14 bits - apparently FF3 needs this.
-- Followed Paradox's suggestion and changed graphics code to place sub-screen
- background layers below main-screen background layers. Helps lots of games
- that use sub-screen addition/subtraction - now you don't have to toggle
- background layers on and off so often just to see hidden text, characters,
- or maps, etc. Made it switchable.
- Acts as a good intermediate solution until sub-screen addition/subtraction
- is actually implemented.
-- Modified sound skipper code to return random values when ROM is stuck
- waiting for the SPC700 CPU to respond - helps several ROMs that previously
- don't work with the currently selection of APU skippers.
-- Improved sound mixing code so volume is not attenuated so much, giving
- better results on 8bit sound cards.
-- Changed the frequency at which the joystick polling routine is called - now
- called every-other frame rather than every 3rd frame.
-- Recompiled Linux and DOS ports with the Pentium optimising version of gcc -
- gives a few percent speed increase.
-- Changed V-RAM increment count from 64 to 128 - apparently Final Fantasy 3
- needs this as well.
-- Fixed sprite priority bug with Mode 7 - apparently Final Fantasy 3 needs
- this.
-- Fixed a screen clipping problem with the S-VGA mode.
-- Fixed bug that had crept in with -m 2 S-VGA mode (Linux version).
-- Fixed S-VGA Linux version with sound enabled.
-- The SPC700 ADC (X),(Y) instruction was broken - with all these SPC700 fixes
- now many more ROMs work with sound enabled.
-- The SPC700 Pop PSW instruction was not resetting the direct page location.
-- The SPC700 instruction MOV A,[DP+X] was incorrectly doing a MOV A,DP+X.
-- Got the SPC700 SETx and CLRx instruction encoding swapped around.
-- Fixed #define problem that was stopping DOS snapshot saving from working.
-0.72 released
-- Fixed the DOS filename handling - old Unix code was screwing up with ROM
- filenames that contained backslashes (\) - the ROM would load but S-RAM
- loading and saving would fail and the default filename for snapshots
- wouldn't work.
-- This time really fixed Allegro library keyboard handling (DOS port); it
- was missing key some presses/releases (was stopping Chrono Trigger
- Left + Right + A button combo from working).
-- Added code to automatically remove headers off S-RAM save files with
- 512 byte headers.
-- 32Mbit ROMs in interleaved format are now automatically detected and
- converted.
-- Added -ss 3 sound skip method support to the asm version - now NBA Live '96
- works again.
-- Added support for multi-part ROM images.
-0.71 released
-- Made libgz.so statically linked (again) on Linux port - sorry.
-- Made writing to $4200 also clear any pending IRQs. This finally allows
- Battle Toads: Double Dragon, Spawn and Sieken 3 all the work with the same
- IRQ logic (but Sieken 3 still gets stuck in sound download code).
-- Fixed a H-DMA wobble bug - some frames could randomly miss a line of
- H-DMA causing the F-Zero screen to wobble, and slight text character
- corruption on games like DKC3.
-- Interleaved format ROM images are now swapped in-place, without the need
- for a temp 4Mb buffer (saves lots of disk swapping on a 16Mb Windows 95
- machine).
-0.7 released
-- Fixed Allegro library keyboard handling (DOS port); it was missing key
- some presses/releases.
-- DOS port had a different MAX_PATH value which moved the location of the
- SRAM size variable when using the asm CPU emulation core. This, in turn,
- caused the SRAM emulation to fail on the DOS port. Donkey Kong County 2 & 3
- were reporting a ROM copier was connected to the SNES and refused to run.
-- Fixed assembler version of XCE - it was always leaving the carry flag
- clear - caused Killer Instinct and Super Punchout to think a ROM
- copier was fitted to the SNES and they all refused to run.
-- Fixed assembler versions of MVN/MVP - they weren't setting the data bank
- register to the destination bank of the instruction.
-- Fixed joystick detection on MS-DOS port - a single 2 or 4 button joystick in
- port 1 was being ignored if a second joystick was not present in port 2.
-- Fixed an uninitialised variable in graphics code - was causing random
- missing scan lines on Mode 7 screens.
-- Joysticks now scanned every 3rd frame (joystick scanning is slow in the PC).
-- Double-whoops, Metriod 3 had stopped working in v0.6 - fixed it
- (memory map bug).
-- Made bit 6 of $4211 set if v-counter == v-timer-position.
-- Made reading of $4200 read $4212 instead.
-- Adjusted DMA timing to always access ROM memory at slow speed - this seems
- to fix Battle Toads.
-- Added code to automatically clear pending IRQs when the horizontal line
- is no longer equal to the horizontal timer line - this fixes Seiken 3, it
- now just gets stuck in the sound CPU wait code - oh well.
-- Moved NMI back to its original pre-0.65 behaviour, now Puzzle Bobble works.
-- More graphics speed ups - the code to render background tiles with their
- priority bits set is only called if there are actual priority-bit tiles.
-- Changed default frame skip rate from 1 to 2 - its seems most people don't
- bother to read the docs, so I thought I'll help them out a bit!
-- Speeded up Mode 7 graphics on games like F-Zero that rewrite the matrix
- registers on each scan line using H-DMA.
-- Reorganised the graphics code and did a slight speed up - graphics code
- will be the next thing to rewrite in assembler.
-- Rewrote CPU core in assembler for Intel platforms - gives a very noticeable
- speed increase.
-- Fixed several problems with the APU sound CPU emulation - its now getting
- stable enough to try and implement sound.
-- Fixed bug that caused 1 byte of S-RAM to be emulated when ROM didn't
- expect any - it was enough to stop Street Fighter 2 and others from
- working - thanks Lord ESNES.
-- The TXS and TCS instructions shouldn't set the Z and N flags.
-- Looks like MVP/MVN instructions should ignore accumulator size - change
- code to always use all 16 bits and exit with accumulator set to 0xffff.
-- Whoops, accidently left some test code in which was causing the V-BLANK
- flag, bit 8 in register $4212, to be miss-calculated.
-- Fixed palette in mode 0.
-- Speeded up graphics drawing a little by skipping groups of 4 pixels that
- were all transparent.
-0.65 released
-- S-VGA and MS-DOS ports now have a VGA mode command line flag.
-- Improved the fading code - should be much more smooth now.
-- Fixed second joy-pad support and re-mapped keys and joysticks to actually
- make a match between what my docs said and a real SNES (SNES docs I'd
- seen were wrong!).
-- Fixed a bug in Relative Long CPU addressing mode.
-- Ported Snes96 to MS-DOS.
-- Snapshot loading and saving no longer uses external gzip binary.
-- Added support for registers at $21c2 and $21c3.
-- Made reading the software latch for the horizontal and vertical counters also
- clear any pending IRQ.
-- Added sprite priority rotation.
-- Rewrote parts of the graphics routines to fix a sprite-to-sprite priority
- bug.
-- NMI flag changed again - now back to being reset by reading $4210 but
- actual NMI is delayed.
-- Made mode 7 background colour 0 transparent - this fixed several sprite
- priority problems a few games where having.
-- Finally worked out how sprite "Object Name Select" works and emulated it -
- this fixes many (if not all) of the corrupted sprites some games
- experienced.
-- Delayed NMI activation for one instruction to give time for loops that
- wait for bit 7 of $4210 to go high.
-- Special-cased line count of 128 on H-DMA to mean repeat previous data with
- a line count of 128 and not just terminate H-DMA on that channel.
-- APU sound CPU emulation added - just need to debug the thing.
-- Fixed Overflow flag setting in ADC and SBC instructions - it was never
- being set.
-- Rewrote how CPU instructions are fetched and how values are pushed and pulled
- from the stack - it gave a very large increase in emulation speed.
-- H-DMA was being started one scan-line too late.
-- Added CG-RAM reading support.
-- Added "Full Graphic" V-RAM reading.
-- Speeded up C version of CPU emulation quite a bit - could speed it up a
- little more before rewriting in assembler.
-- Fixed bugs in 16x16 tile drawing on 2bit and 8bit deep screens.
-0.6 released
-- Speeded up 16x16 tile background rendering by removing a temp tile buffer
- it was using. The speed up also fixed a vertical scroll bug.
-- Fixed slight window clipping on 16x16 tile backgrounds.
-- Added automatic PAL/NTSC mode switching.
-- Fixed background and sprites so only visible if on main-screen or
- on sub-screen under correct circumstance.
-- Fixed lockup bug in DMA.
-- Stopped NMI flag from being reset by reading $4210 - was causing a couple
- of games to get stuck.
-- Whoops, got horizontal and vertical Mode 7 flip bits around the wrong way!
-- Fixed MIT shared memory pixmap support for X11 version (it was always turned
- off).
-- Fixed minor bug - first sprite in priority group was drawn twice. Didn't
- cause any visual bugs, it just slowed down redrawing a little.
-- Fixed DMA bug - transfer byte count should be 0 after DMA has finished.
-- Fixed a scaling bug if width < height.
-- Interleaved ROM image support added.
-- 16bit and 24bit X11 server support added - with scaling.
-- Added window scaling on X11 version.
-- Partial clip windows added - the only window overlap option implemented at
- the moment is OR, it seems it good enough for all the ROMs I've tested
- it with.
-- Partial Mosaic effect added (pixels only growing vertically).
-- Missing Mode 7 "outside screen area" option added.
-- Fixed mode 7 screen wrap "outside screen area" option.
-- Used new event processing to finally fix H-IRQ so it triggers at the
- correct position on the scan line.
-- New event processing added.
-- Linux version now statically links libgz.so (sorry).
-0.5 released
-- Linux S-VGA version changed from using a 320x240 ModeX screen (slow) to a
- 256x256 chunky screen (faster) - thanks to Phillip Ezolt (pe28+@andrew.cmu.edu)
- for information on how to do this.
-- Mode 7 screen flipping added.
-- Included Snes97's CPU emulation code into Snes96. Didn't fix any bugs but
- slowed down the emulation some what and I couldn't compile it optimised
- because it was so large - so I removed it again.
-- Added a few extra features available via the keyboard.
-- Fixed a H-DMA transfer mode - bad documentation.
-- Fixed H-DMA indirect addressing (it was using the wrong CPU memory bank).
-- The Linux slow down bug is my crappy laptop enabling battery saving features !
-- Changed graphics code to perform true line-by-line screen updates.
-- Fixed sprite drawing bugs.
-- Ported Snes97's graphics code to Snes96.
-- Fixed memory map for HiROM save RAM area.
-- Fixed HiROM memory map - now Killer Instinct and Donkey Kong County work !
-- OK the slow down bug is just actually my laptop trying to save battery
- power by slowing the CPU clock!
-- The Linux slow down bug shows itself on DOS emulators running under DOSEMU
- so it must be a kernel problem (or feature).
-- Fixed H-DMA (again) to be complete emulation - all I need now is line-by-line
- screen update...
-- Fixed DMA to not copy too many bytes if byte count was not a multiple of
- the transfer mode quantity (caused corruption on Super Mario World map screen).
-- Changed mapping of keyboard to joy-pad buttons and added additional
- direction keys for joy-pad one so player one's right hand doesn't have to
- obscure player two's keyboard joy-pad buttons.
-- Changed joystick button layout to match SNES if using a 6 button joy-pad.
-- Changed snapshot format so I can easily use libgz on Linux.
-- Added few speed up tweaks that will be lost again when I add line-by-line
- screen update.
-- First visible scan-line changed from 8 to 1 to match with new docs.
-- New SNES information source found; fixed partial H-DMA emulation to include
- indirect addressing support.
-- Snapshot files are now compressed.
-- Compressed ROM images now supported on Linux.
-- Snapshot loading and saving added.
-- Joystick support for Linux added. One 2, 4 or 6 button joystick, or two 2
- button joysticks supported (PC hardware limitation).
-- SVGA full screen support added for Linux. Still has the X11 slow down bug so
- can't blame the X11 server any more! Must be a kernel bug or a very odd
- emulator bug.
-- Added emulation of two joy-pads on the PC/Sun keyboard.
-- Removed -i command line flag as it is no longer used. -h value range has also
- changed: now 1 - 100 (percentage).
-- Actuate cycle counting rather than instruction counting now added including
- fast and slow ROM timing - should give much better timing information when
- line-by-line screen update added.
-- Bug fixed old-style joy-pad access used by some ROMs - Mario All Stars still
- gives problems if enabled and I don't know why; but at least Super Bomberman
- now works !
-- Looks like if both horizontal and vertical IRQ are enabled then IRQ should
- only be triggered once per frame and not once per scan line - looking at the
- IRQ handler of a couple of ROMs seems to confirm this.
-- Added initial cycle counting - not accurate enough for some ROMs though.
-- Finally worked out how the odd VRAM address increments should work but only
- found one ROM, so far, that actually uses it.
-- Debugged the odd slow down problem with the Linux port - it seems to be a
- bug in the X Window System server - starve the X server of keyboard presses
- or mouse clicks or movement and the X server slows down, slowing down the
- emulator with it !
-0.4 released
-- Fixed sprite vertical clipping at top of screen.
-- No need to invert the Mode 7 transformation matrix before use - the
- ROM coder already had to!
-- Fixed Mode 7 scrolling offset when using special effects.
-- Added Mode 7 rotation, enlargement and reduction emulation.
-- DMA shouldn't zero the byte count value after a DMA has completed.
-- Added DMA reading (Addams Family was using it)
-- Fixed V-RAM read function - returned data should lag behind the V-RAM
- address by one byte/word.
-- Added mode 7 graphics only.
-0.3 released
-- Speeded up the main CPU loop a bit.
-- Add more command line options:
- -f <frame skip> (default 1)
- -i <no instructions between polling X> (default 32768)
- -h <number instructions per scan line> (default 45, some games allow a lower
- setting resulting in a increased
- emulated frame rate)
- -t enable CPU tracing
- -ss <sound CPU skip wait method> (default 0, more methods to be added)
- -H disable H-DMA emulation
- -F Force Hi-ROM memory map
-- Modified planar to chunky conversion to use look up tables.
-- But now Mario All Stars won't start. Made emulation of $4016 optional with
- -o command line switch.
-- Thanks to Carlos (calb) of ESNES fame, I've added correct $4016 & $4017
- joy-pad register processing - now several more ROMs will start once a
- button is pressed and can be controlled.
-- DMA wasn't updating DMA registers with the final CPU address used after the
- DMA had completed (caused sprite and background corruption with some ROMs).
- Still suspect another DMA side effect isn't being emulated correctly though.
-- Fixed setting of CPU overflow flag in ADC and SBC instructions in decimal
- mode.
-- Fixed MVP/MVN CPU instructions to leave X and Y values correct at end of
- loop - several more ROMs now work. Still don't know if MVP/MVN instructions
- should ignore the accumulator size flag or not.
-- Rewrote background drawing code - gives a large increase in speed.
-- Flag to only update X Windows colour palette when necessary was missing a
- case - caused some ROMs to start with a black screen.
-- Code to only update background tiles when changed wasn't working so I
- disabled it.
-- CPU WAI instruction needed to trigger on hardware IRQ even when interrupt
- enable flag was false.
-- DMA was not transferring 65536 bytes when byte count was 0.
-- Fixed matrix 16bit x 8bit multiplication (old debug code was causing junk
- value to be returned).
-- Fixed Makefile so version.h header file change recompiles file that shows
- version number in window title.
-- Added more reporting of used but unimplemented missing hardware features to
- debug command.
-- New ROM loading code from Jerremy included, can now cope with ROM images
- with no 512 byte header.
-- Speeded up emulated memory access a little bit.
-0.2 released
-- Added matrix 16bit x 8bit multiplication for Super Off-Road Racer.
-- Added initial H-DMA emulation - visual effects using it will not be seen
- correctly until screen is updated line-by-line rather than the whole screen
- at end-of-frame.
-- Fixed horizontal sprite clipping (vertical clipping still has a problem).
-- Integrated large sprite bug fixes and new background drawing code from
- Jerremy.
-- Fixed large size per-sprite flag; always stayed true after sprite size was
- changed to large.
-- Rewrote the planar to chunky pixel conversion routines (still need more
- work).
-- Made registers $4016 & $4017 always return $ff - lots of ROMs that previously
- wouldn't go beyond the title screen thought old-style joy-pads were
- connected and were waiting for the user to press a button on them.
-- Frame skip rate now set to 1 instead of 5 on my P166 laptop!
-- Fixed NMI v-blank flag being incorrect set, caused some ROMs to lock.
-- X keyboard autorepeat now switched off when emulator has keyboard focus.
-- Added number key options to toggle backgrounds 1 to 4 and objs (sprites) on
- and off.
-- Fixed sprite clipping problems at edge of left hand side of screen.
-- Corrected Hi-ROM memory map (I think) (no I didn't)
-- Fixed most of the sprite-to-sprite priority problems.
-- Added sprite debug command, 'S'.
-- Added a debug command to show what missing hardware features a ROM was using.
-- Added horizontal and vertical beam position IRQ - horizontal always triggers
- at start of line at the moment.
-- Fixed SBC instruction to set carry flag the correct way around.
-Initial release 0.1
-- Ported Windows 95 version of Snes96 to Linux on a PC and Solaris on a
- SparcStation.
-- Corrected work RAM memory map.
diff --git a/source/cheats.c b/source/cheats.c
index 1a5e833..572fd55 100644
--- a/source/cheats.c
+++ b/source/cheats.c
@@ -22,8 +22,7 @@ static bool S9xAllHex(const char* code, int32_t len)
const char* S9xProActionReplayToRaw(const char* code, uint32_t* address, uint8_t* byte)
{
uint32_t data = 0;
- if (strlen(code) != 8 || !S9xAllHex(code, 8) ||
- sscanf(code, "%x", &data) != 1)
+ if (strlen(code) != 8 || !S9xAllHex(code, 8) || sscanf(code, "%x", &data) != 1)
return "Invalid Pro Action Replay code - should be 8 hex digits in length.";
*address = data >> 8;
@@ -100,271 +99,3 @@ const char* S9xGameGenieToRaw(const char* code, uint32_t* address, uint8_t* byte
return NULL;
}
-
-void S9xStartCheatSearch(SCheatData* d)
-{
- // memmove may be required: Source is usually a different malloc, but could be pointed to d->CWRAM [Neb]
- memmove(d->CWRAM, d->RAM, 0x20000);
- // memmove may be required: Source is usually a different malloc, but could be pointed to d->CSRAM [Neb]
- memmove(d->CSRAM, d->SRAM, 0x10000);
- // memmove may be required: Source is usually a different malloc, but could be pointed to d->CIRAM [Neb]
- memmove(d->CIRAM, &d->FillRAM [0x3000], 0x2000);
- memset(d->WRAM_BITS, 0xff, 0x20000 >> 3);
- memset(d->SRAM_BITS, 0xff, 0x10000 >> 3);
- memset(d->IRAM_BITS, 0xff, 0x2000 >> 3);
-}
-
-#define BIT_CLEAR(a,v) \
-(a)[(v) >> 5] &= ~(1 << ((v) & 31))
-
-#define BIT_SET(a,v) \
-(a)[(v) >> 5] |= 1 << ((v) & 31)
-
-#define TEST_BIT(a,v) \
-((a)[(v) >> 5] & (1 << ((v) & 31)))
-
-#define CHEATS_C(c,a,b) \
-((c) == S9X_LESS_THAN ? (a) < (b) : \
- (c) == S9X_GREATER_THAN ? (a) > (b) : \
- (c) == S9X_LESS_THAN_OR_EQUAL ? (a) <= (b) : \
- (c) == S9X_GREATER_THAN_OR_EQUAL ? (a) >= (b) : \
- (c) == S9X_EQUAL ? (a) == (b) : \
- (a) != (b))
-
-#define _D(s,m,o) \
-((s) == S9X_8_BITS ? (uint8_t) (*((m) + (o))) : \
- (s) == S9X_16_BITS ? ((uint16_t) (*((m) + (o)) + (*((m) + (o) + 1) << 8))) : \
- (s) == S9X_24_BITS ? ((uint32_t) (*((m) + (o)) + (*((m) + (o) + 1) << 8) + (*((m) + (o) + 2) << 16))) : \
-((uint32_t) (*((m) + (o)) + (*((m) + (o) + 1) << 8) + (*((m) + (o) + 2) << 16) + (*((m) + (o) + 3) << 24))))
-
-#define _DS(s,m,o) \
-((s) == S9X_8_BITS ? ((int8_t) *((m) + (o))) : \
- (s) == S9X_16_BITS ? ((int16_t) (*((m) + (o)) + (*((m) + (o) + 1) << 8))) : \
- (s) == S9X_24_BITS ? (((int32_t) ((*((m) + (o)) + (*((m) + (o) + 1) << 8) + (*((m) + (o) + 2) << 16)) << 8)) >> 8): \
- ((int32_t) (*((m) + (o)) + (*((m) + (o) + 1) << 8) + (*((m) + (o) + 2) << 16) + (*((m) + (o) + 3) << 24))))
-
-void S9xSearchForChange(SCheatData* d, S9xCheatComparisonType cmp,
- S9xCheatDataSize size, bool is_signed, bool update)
-{
- int32_t l;
- int32_t i;
-
- switch (size)
- {
- case S9X_8_BITS:
- l = 0;
- break;
- case S9X_16_BITS:
- l = 1;
- break;
- case S9X_24_BITS:
- l = 2;
- break;
- default:
- case S9X_32_BITS:
- l = 3;
- break;
- }
-
- if (is_signed)
- {
- for (i = 0; i < 0x20000 - l; i++)
- {
- if (TEST_BIT(d->WRAM_BITS, i) &&
- CHEATS_C(cmp, _DS(size, d->RAM, i), _DS(size, d->CWRAM, i)))
- {
- if (update)
- d->CWRAM [i] = d->RAM [i];
- }
- else
- BIT_CLEAR(d->WRAM_BITS, i);
- }
-
- for (i = 0; i < 0x10000 - l; i++)
- {
- if (TEST_BIT(d->SRAM_BITS, i) &&
- CHEATS_C(cmp, _DS(size, d->SRAM, i), _DS(size, d->CSRAM, i)))
- {
- if (update)
- d->CSRAM [i] = d->SRAM [i];
- }
- else
- BIT_CLEAR(d->SRAM_BITS, i);
- }
-
- for (i = 0; i < 0x2000 - l; i++)
- {
- if (TEST_BIT(d->IRAM_BITS, i) &&
- CHEATS_C(cmp, _DS(size, d->FillRAM + 0x3000, i), _DS(size, d->CIRAM, i)))
- {
- if (update)
- d->CIRAM [i] = d->FillRAM [i + 0x3000];
- }
- else
- BIT_CLEAR(d->IRAM_BITS, i);
- }
- }
- else
- {
- for (i = 0; i < 0x20000 - l; i++)
- {
- if (TEST_BIT(d->WRAM_BITS, i) &&
- CHEATS_C(cmp, _D(size, d->RAM, i), _D(size, d->CWRAM, i)))
- {
- if (update)
- d->CWRAM [i] = d->RAM [i];
- }
- else
- BIT_CLEAR(d->WRAM_BITS, i);
- }
-
- for (i = 0; i < 0x10000 - l; i++)
- {
- if (TEST_BIT(d->SRAM_BITS, i) &&
- CHEATS_C(cmp, _D(size, d->SRAM, i), _D(size, d->CSRAM, i)))
- {
- if (update)
- d->CSRAM [i] = d->SRAM [i];
- }
- else
- BIT_CLEAR(d->SRAM_BITS, i);
- }
-
- for (i = 0; i < 0x2000 - l; i++)
- {
- if (TEST_BIT(d->IRAM_BITS, i) &&
- CHEATS_C(cmp, _D(size, d->FillRAM + 0x3000, i), _D(size, d->CIRAM, i)))
- {
- if (update)
- d->CIRAM [i] = d->FillRAM [i + 0x3000];
- }
- else
- BIT_CLEAR(d->IRAM_BITS, i);
- }
- }
-}
-
-void S9xSearchForValue(SCheatData* d, S9xCheatComparisonType cmp,
- S9xCheatDataSize size, uint32_t value,
- bool is_signed, bool update)
-{
- int32_t l;
- int32_t i;
-
- switch (size)
- {
- case S9X_8_BITS:
- l = 0;
- break;
- case S9X_16_BITS:
- l = 1;
- break;
- case S9X_24_BITS:
- l = 2;
- break;
- default:
- case S9X_32_BITS:
- l = 3;
- break;
- }
-
-
- if (is_signed)
- {
- for (i = 0; i < 0x20000 - l; i++)
- {
- if (TEST_BIT(d->WRAM_BITS, i) &&
- CHEATS_C(cmp, _DS(size, d->RAM, i), (int32_t) value))
- {
- if (update)
- d->CWRAM [i] = d->RAM [i];
- }
- else
- BIT_CLEAR(d->WRAM_BITS, i);
- }
-
- for (i = 0; i < 0x10000 - l; i++)
- {
- if (TEST_BIT(d->SRAM_BITS, i) &&
- CHEATS_C(cmp, _DS(size, d->SRAM, i), (int32_t) value))
- {
- if (update)
- d->CSRAM [i] = d->SRAM [i];
- }
- else
- BIT_CLEAR(d->SRAM_BITS, i);
- }
-
- for (i = 0; i < 0x2000 - l; i++)
- {
- if (TEST_BIT(d->IRAM_BITS, i) &&
- CHEATS_C(cmp, _DS(size, d->FillRAM + 0x3000, i), (int32_t) value))
- {
- if (update)
- d->CIRAM [i] = d->FillRAM [i + 0x3000];
- }
- else
- BIT_CLEAR(d->IRAM_BITS, i);
- }
- }
- else
- {
- for (i = 0; i < 0x20000 - l; i++)
- {
- if (TEST_BIT(d->WRAM_BITS, i) &&
- CHEATS_C(cmp, _D(size, d->RAM, i), value))
- {
- if (update)
- d->CWRAM [i] = d->RAM [i];
- }
- else
- BIT_CLEAR(d->WRAM_BITS, i);
- }
-
- for (i = 0; i < 0x10000 - l; i++)
- {
- if (TEST_BIT(d->SRAM_BITS, i) &&
- CHEATS_C(cmp, _D(size, d->SRAM, i), value))
- {
- if (update)
- d->CSRAM [i] = d->SRAM [i];
- }
- else
- BIT_CLEAR(d->SRAM_BITS, i);
- }
-
- for (i = 0; i < 0x2000 - l; i++)
- {
- if (TEST_BIT(d->IRAM_BITS, i) &&
- CHEATS_C(cmp, _D(size, d->FillRAM + 0x3000, i), value))
- {
- if (update)
- d->CIRAM [i] = d->FillRAM [i + 0x3000];
- }
- else
- BIT_CLEAR(d->IRAM_BITS, i);
- }
- }
-}
-
-void S9xOutputCheatSearchResults(SCheatData* d)
-{
- int32_t i;
- for (i = 0; i < 0x20000; i++)
- {
- if (TEST_BIT(d->WRAM_BITS, i))
- printf("WRAM: %05x: %02x\n", i, d->RAM [i]);
- }
-
- for (i = 0; i < 0x10000; i++)
- {
- if (TEST_BIT(d->SRAM_BITS, i))
- printf("SRAM: %04x: %02x\n", i, d->SRAM [i]);
- }
-
- for (i = 0; i < 0x2000; i++)
- {
- if (TEST_BIT(d->IRAM_BITS, i))
- printf("IRAM: %05x: %02x\n", i, d->FillRAM [i + 0x3000]);
- }
-}
diff --git a/source/cheats.h b/source/cheats.h
index 42e2b47..df8a9a6 100644
--- a/source/cheats.h
+++ b/source/cheats.h
@@ -59,10 +59,4 @@ void S9xDeleteCheats();
void S9xDeleteCheat(uint32_t which1);
bool S9xLoadCheatFile(const char* filename);
bool S9xSaveCheatFile(const char* filename);
-
-void S9xStartCheatSearch(SCheatData* cheats);
-void S9xSearchForChange(SCheatData* cheats, S9xCheatComparisonType cmp, S9xCheatDataSize size, bool is_signed, bool update);
-void S9xSearchForValue(SCheatData* cheats, S9xCheatComparisonType cmp, S9xCheatDataSize size, uint32_t value, bool is_signed, bool update);
-void S9xOutputCheatSearchResults(SCheatData* cheats);
-
#endif
diff --git a/source/cheats2.c b/source/cheats2.c
index 3d6519e..f113cd9 100644
--- a/source/cheats2.c
+++ b/source/cheats2.c
@@ -112,11 +112,9 @@ void S9xApplyCheats(void)
{
uint32_t i;
if (Settings.ApplyCheats)
- {
for (i = 0; i < Cheat.num_cheats; i++)
if (Cheat.c [i].enabled)
S9xApplyCheat(i);
- }
}
void S9xRemoveCheats(void)
@@ -130,27 +128,22 @@ void S9xRemoveCheats(void)
bool S9xLoadCheatFile(const char* filename)
{
uint8_t data [8 + MAX_SFCCHEAT_NAME];
- FILE* fs = NULL;
-
+ FILE* fs = fopen(filename, "rb");
Cheat.num_cheats = 0;
- fs = fopen(filename, "rb");
-
if (!fs)
- return (false);
+ return false;
- while (fread((void*) data, 1, 8 + MAX_SFCCHEAT_NAME,
- fs) == 8 + MAX_SFCCHEAT_NAME)
+ while (fread((void*) data, 1, 8 + MAX_SFCCHEAT_NAME, fs) == 8 + MAX_SFCCHEAT_NAME)
{
if (data[6] != 254 || data[7] != 252)
{
fclose(fs);
- return (false);
+ return false;
}
Cheat.c [Cheat.num_cheats].enabled = (data [0] & 4) == 0;
Cheat.c [Cheat.num_cheats].byte = data [1];
- Cheat.c [Cheat.num_cheats].address = data [2] | (data [3] << 8) |
- (data [4] << 16);
+ Cheat.c [Cheat.num_cheats].address = data [2] | (data [3] << 8) | (data [4] << 16);
Cheat.c [Cheat.num_cheats].saved_byte = data [5];
Cheat.c [Cheat.num_cheats].saved = (data [0] & 8) != 0;
memcpy(Cheat.c [Cheat.num_cheats].name, &data [8], MAX_SFCCHEAT_NAME - 1);
@@ -158,25 +151,25 @@ bool S9xLoadCheatFile(const char* filename)
}
fclose(fs);
- return (true);
+ return true;
}
bool S9xSaveCheatFile(const char* filename)
{
uint32_t i;
uint8_t data [8 + MAX_SFCCHEAT_NAME];
- FILE* fs = NULL;
+ FILE* fs;
if (Cheat.num_cheats == 0)
{
(void) remove(filename);
- return (true);
+ return true;
}
fs = fopen(filename, "wb");
if (!fs)
- return (false);
+ return false;
for (i = 0; i < Cheat.num_cheats; i++)
{
@@ -199,10 +192,10 @@ bool S9xSaveCheatFile(const char* filename)
if (fwrite(data, 8 + MAX_SFCCHEAT_NAME, 1, fs) != 1)
{
fclose(fs);
- return (false);
+ return false;
}
}
fclose(fs);
- return (true);
+ return true;
}
diff --git a/source/clip.c b/source/clip.c
index c579b77..7cafc10 100644
--- a/source/clip.c
+++ b/source/clip.c
@@ -13,24 +13,27 @@ typedef struct
} Band;
#define BAND_EMPTY(B) (B.Left >= B.Right)
-#define BANDS_INTERSECT(A,B) ((A.Left >= B.Left && A.Left < B.Right) || \
- (B.Left >= A.Left && B.Left < A.Right))
-#define OR_BANDS(R,A,B) {\
+#define BANDS_INTERSECT(A,B) ((A.Left >= B.Left && A.Left < B.Right) || (B.Left >= A.Left && B.Left < A.Right))
+#define OR_BANDS(R,A,B) \
+{ \
R.Left = MIN(A.Left, B.Left); \
- R.Right = MAX(A.Right, B.Right);}
+ R.Right = MAX(A.Right, B.Right); \
+}
-#define AND_BANDS(R,A,B) {\
+#define AND_BANDS(R,A,B) \
+{ \
R.Left = MAX(A.Left, B.Left); \
- R.Right = MIN(A.Right, B.Right);}
+ R.Right = MIN(A.Right, B.Right); \
+}
static int32_t IntCompare(const void* d1, const void* d2)
{
- return (*(uint32_t*) d1 - * (uint32_t*) d2);
+ return *(uint32_t*) d1 - *(uint32_t*) d2;
}
static int32_t BandCompare(const void* d1, const void* d2)
{
- return (((Band*) d1)->Left - ((Band*) d2)->Left);
+ return ((Band*) d1)->Left - ((Band*) d2)->Left;
}
void ComputeClipWindows()
@@ -53,8 +56,7 @@ void ComputeClipWindows()
{
if ((Memory.FillRAM [0x2130] & 0xc0) == 0xc0)
{
- // The whole of the main screen is switched off,
- // completely clip everything.
+ // The whole of the main screen is switched off, completely clip everything.
for (i = 0; i < 6; i++)
{
IPPU.Clip [c].Count [i] = 1;
@@ -66,37 +68,29 @@ void ComputeClipWindows()
else if ((Memory.FillRAM [0x2130] & 0xc0) == 0x00)
continue;
}
- else
+ else if ((Memory.FillRAM [0x2130] & 0x30) == 0x30) // .. colour window on the sub-screen.
{
- // .. colour window on the sub-screen.
- if ((Memory.FillRAM [0x2130] & 0x30) == 0x30)
+ // The sub-screen is switched off, completely clip everything.
+ int32_t i;
+ for (i = 0; i < 6; i++)
{
- // The sub-screen is switched off, completely
- // clip everything.
- int32_t i;
- for (i = 0; i < 6; i++)
- {
- IPPU.Clip [1].Count [i] = 1;
- IPPU.Clip [1].Left [0][i] = 1;
- IPPU.Clip [1].Right [0][i] = 0;
- }
- return;
+ IPPU.Clip [1].Count [i] = 1;
+ IPPU.Clip [1].Left [0][i] = 1;
+ IPPU.Clip [1].Right [0][i] = 0;
}
- else if ((Memory.FillRAM [0x2130] & 0x30) == 0x00)
- continue;
+ return;
}
+ else if ((Memory.FillRAM [0x2130] & 0x30) == 0x00)
+ continue;
}
- if (w == 5 || pClip->Count [5] ||
- (Memory.FillRAM [0x212c + c] & Memory.FillRAM [0x212e + c] & (1 << w)))
+ if (w == 5 || pClip->Count [5] || (Memory.FillRAM [0x212c + c] & Memory.FillRAM [0x212e + c] & (1 << w)))
{
Band Win1[3];
Band Win2[3];
uint32_t Window1Enabled = 0;
uint32_t Window2Enabled = 0;
- bool invert = (w == 5 &&
- ((c == 1 && (Memory.FillRAM [0x2130] & 0x30) == 0x10) ||
- (c == 0 && (Memory.FillRAM [0x2130] & 0xc0) == 0x40)));
+ bool invert = (w == 5 && ((c == 1 && (Memory.FillRAM [0x2130] & 0x30) == 0x10) || (c == 0 && (Memory.FillRAM [0x2130] & 0xc0) == 0x40)));
if (w == 5 || (Memory.FillRAM [0x212c + c] & Memory.FillRAM [0x212e + c] & (1 << w)))
{
@@ -107,33 +101,30 @@ void ComputeClipWindows()
Win1[Window1Enabled].Left = PPU.Window1Left;
Win1[Window1Enabled++].Right = PPU.Window1Right + 1;
}
- else
+ else if (PPU.Window1Left <= PPU.Window1Right)
{
- if (PPU.Window1Left <= PPU.Window1Right)
+ if (PPU.Window1Left > 0)
{
- if (PPU.Window1Left > 0)
- {
- Win1[Window1Enabled].Left = 0;
- Win1[Window1Enabled++].Right = PPU.Window1Left;
- }
- if (PPU.Window1Right < 255)
- {
- Win1[Window1Enabled].Left = PPU.Window1Right + 1;
- Win1[Window1Enabled++].Right = 256;
- }
- if (Window1Enabled == 0)
- {
- Win1[Window1Enabled].Left = 1;
- Win1[Window1Enabled++].Right = 0;
- }
+ Win1[Window1Enabled].Left = 0;
+ Win1[Window1Enabled++].Right = PPU.Window1Left;
}
- else
+ if (PPU.Window1Right < 255)
{
- // 'outside' a window with no range -
- // appears to be the whole screen.
- Win1[Window1Enabled].Left = 0;
+ Win1[Window1Enabled].Left = PPU.Window1Right + 1;
Win1[Window1Enabled++].Right = 256;
}
+ if (Window1Enabled == 0)
+ {
+ Win1[Window1Enabled].Left = 1;
+ Win1[Window1Enabled++].Right = 0;
+ }
+ }
+ else
+ {
+ // 'outside' a window with no range -
+ // appears to be the whole screen.
+ Win1[Window1Enabled].Left = 0;
+ Win1[Window1Enabled++].Right = 256;
}
}
if (PPU.ClipWindow2Enable [w])
@@ -198,48 +189,42 @@ void ComputeClipWindows()
{
if (BAND_EMPTY(Win2[0]))
Bands[B++] = Win1[0];
- else
- {
- if (BANDS_INTERSECT(Win1[0], Win2[0]))
- {
- OR_BANDS(Bands[0], Win1[0], Win2[0])
- B = 1;
- }
- else
- {
- Bands[B++] = Win1[0];
- Bands[B++] = Win2[0];
- }
- }
- }
- else
- {
- if (BANDS_INTERSECT(Win1[0], Win2[0]))
+ else if (BANDS_INTERSECT(Win1[0], Win2[0]))
{
OR_BANDS(Bands[0], Win1[0], Win2[0])
- if (BANDS_INTERSECT(Win1[0], Win2[1]))
- OR_BANDS(Bands[1], Win1[0], Win2[1])
- else
- Bands[1] = Win2[1];
B = 1;
- if (BANDS_INTERSECT(Bands[0], Bands[1]))
- OR_BANDS(Bands[0], Bands[0], Bands[1])
- else
- B = 2;
}
- else if (BANDS_INTERSECT(Win1[0], Win2[1]))
+ else
{
+ Bands[B++] = Win1[0];
Bands[B++] = Win2[0];
- OR_BANDS(Bands[B], Win1[0], Win2[1]);
- B++;
}
+ }
+ else if (BANDS_INTERSECT(Win1[0], Win2[0]))
+ {
+ OR_BANDS(Bands[0], Win1[0], Win2[0])
+ if (BANDS_INTERSECT(Win1[0], Win2[1]))
+ OR_BANDS(Bands[1], Win1[0], Win2[1])
else
- {
- Bands[0] = Win2[0];
- Bands[1] = Win1[0];
- Bands[2] = Win2[1];
- B = 3;
- }
+ Bands[1] = Win2[1];
+ B = 1;
+ if (BANDS_INTERSECT(Bands[0], Bands[1]))
+ OR_BANDS(Bands[0], Bands[0], Bands[1])
+ else
+ B = 2;
+ }
+ else if (BANDS_INTERSECT(Win1[0], Win2[1]))
+ {
+ Bands[B++] = Win2[0];
+ OR_BANDS(Bands[B], Win1[0], Win2[1]);
+ B++;
+ }
+ else
+ {
+ Bands[0] = Win2[0];
+ Bands[1] = Win1[0];
+ Bands[2] = Win2[1];
+ B = 3;
}
}
}
@@ -426,7 +411,6 @@ void ComputeClipWindows()
{
j = 0;
// Easy case to deal with, so special case it.
-
if (Bands[0].Left > 0)
{
pClip->Left[j][w] = 0;
@@ -500,7 +484,6 @@ void ComputeClipWindows()
{
// Only one window enabled so no need to perform
// complex overlap logic...
-
if (Window1Enabled)
{
if (invert)
diff --git a/source/cpuaddr.h b/source/cpuaddr.h
index 4c0fb84..bdeb242 100644
--- a/source/cpuaddr.h
+++ b/source/cpuaddr.h
@@ -13,13 +13,13 @@ static INLINE void Immediate8(void)
CPU.PC++;
}
-static INLINE void Immediate16()
+static INLINE void Immediate16(void)
{
OpAddress = ICPU.ShiftedPB + CPU.PC - CPU.PCBase;
CPU.PC += 2;
}
-static INLINE void Relative()
+static INLINE void Relative(void)
{
int8_t Int8 = *CPU.PC++;
#ifndef SA1_OPCODES
@@ -28,7 +28,7 @@ static INLINE void Relative()
OpAddress = ((int32_t)(CPU.PC - CPU.PCBase) + Int8) & 0xffff;
}
-static INLINE void RelativeLong()
+static INLINE void RelativeLong(void)
{
#ifdef FAST_LSB_WORD_ACCESS
OpAddress = *(uint16_t*) CPU.PC;
@@ -152,9 +152,6 @@ static INLINE void DirectIndirectIndexed(bool read)
if (read)
OpenBus = (uint8_t)(OpAddress >> 8);
OpAddress += ICPU.ShiftedDB + ICPU.Registers.Y.W;
-
- // XXX: always add one if STA
- // XXX: else Add one cycle if crosses page boundary
}
static INLINE void DirectIndirectIndexedLong(bool read)
@@ -221,8 +218,6 @@ static INLINE void AbsoluteIndexedX(bool read)
#ifndef SA1_OPCODES
CPU.Cycles += CPU.MemSpeedx2;
#endif
- // XXX: always add one cycle for ROL, LSR, etc
- // XXX: else is cross page boundary add one cycle
}
static INLINE void AbsoluteIndexedY(bool read)
@@ -238,8 +233,6 @@ static INLINE void AbsoluteIndexedY(bool read)
#ifndef SA1_OPCODES
CPU.Cycles += CPU.MemSpeedx2;
#endif
- // XXX: always add cycle for STA
- // XXX: else is cross page boundary add one cycle
}
static INLINE void AbsoluteLongIndexedX(bool read)
diff --git a/source/cpuexec.c b/source/cpuexec.c
index ef96339..fedb318 100644
--- a/source/cpuexec.c
+++ b/source/cpuexec.c
@@ -367,7 +367,6 @@ void S9xDoHBlankProcessing_SFX()
if (!PPU.ForcedBlanking)
{
uint8_t tmp = 0;
-
PPU.OAMAddr = PPU.SavedOAMAddr;
if (PPU.OAMPriorityRotation)
@@ -399,45 +398,42 @@ void S9xDoHBlankProcessing_SFX()
CPU.Flags &= ~NMI_FLAG;
S9xStartScreenRefresh();
}
- if (CPU.V_Counter >= FIRST_VISIBLE_LINE &&
- CPU.V_Counter < PPU.ScreenHeight + FIRST_VISIBLE_LINE)
+ if (CPU.V_Counter >= FIRST_VISIBLE_LINE && CPU.V_Counter < PPU.ScreenHeight + FIRST_VISIBLE_LINE)
RenderLine(CPU.V_Counter - FIRST_VISIBLE_LINE);
#ifndef USE_BLARGG_APU
+ if (APU.TimerEnabled [2])
{
- if (APU.TimerEnabled [2])
+ APU.Timer [2] += 4;
+ while (APU.Timer [2] >= APU.TimerTarget [2])
{
- APU.Timer [2] += 4;
- while (APU.Timer [2] >= APU.TimerTarget [2])
+ IAPU.RAM [0xff] = (IAPU.RAM [0xff] + 1) & 0xf;
+ APU.Timer [2] -= APU.TimerTarget [2];
+ IAPU.WaitCounter++;
+ IAPU.APUExecuting = true;
+ }
+ }
+ if (CPU.V_Counter & 1)
+ {
+ if (APU.TimerEnabled [0])
+ {
+ APU.Timer [0]++;
+ if (APU.Timer [0] >= APU.TimerTarget [0])
{
- IAPU.RAM [0xff] = (IAPU.RAM [0xff] + 1) & 0xf;
- APU.Timer [2] -= APU.TimerTarget [2];
+ IAPU.RAM [0xfd] = (IAPU.RAM [0xfd] + 1) & 0xf;
+ APU.Timer [0] = 0;
IAPU.WaitCounter++;
IAPU.APUExecuting = true;
}
}
- if (CPU.V_Counter & 1)
+ if (APU.TimerEnabled [1])
{
- if (APU.TimerEnabled [0])
+ APU.Timer [1]++;
+ if (APU.Timer [1] >= APU.TimerTarget [1])
{
- APU.Timer [0]++;
- if (APU.Timer [0] >= APU.TimerTarget [0])
- {
- IAPU.RAM [0xfd] = (IAPU.RAM [0xfd] + 1) & 0xf;
- APU.Timer [0] = 0;
- IAPU.WaitCounter++;
- IAPU.APUExecuting = true;
- }
- }
- if (APU.TimerEnabled [1])
- {
- APU.Timer [1]++;
- if (APU.Timer [1] >= APU.TimerTarget [1])
- {
- IAPU.RAM [0xfe] = (IAPU.RAM [0xfe] + 1) & 0xf;
- APU.Timer [1] = 0;
- IAPU.WaitCounter++;
- IAPU.APUExecuting = true;
- }
+ IAPU.RAM [0xfe] = (IAPU.RAM [0xfe] + 1) & 0xf;
+ APU.Timer [1] = 0;
+ IAPU.WaitCounter++;
+ IAPU.APUExecuting = true;
}
}
}
@@ -501,7 +497,6 @@ void S9xDoHBlankProcessing_NoSFX()
if (!PPU.ForcedBlanking)
{
uint8_t tmp = 0;
-
PPU.OAMAddr = PPU.SavedOAMAddr;
if (PPU.OAMPriorityRotation)
@@ -533,45 +528,42 @@ void S9xDoHBlankProcessing_NoSFX()
CPU.Flags &= ~NMI_FLAG;
S9xStartScreenRefresh();
}
- if (CPU.V_Counter >= FIRST_VISIBLE_LINE &&
- CPU.V_Counter < PPU.ScreenHeight + FIRST_VISIBLE_LINE)
+ if (CPU.V_Counter >= FIRST_VISIBLE_LINE && CPU.V_Counter < PPU.ScreenHeight + FIRST_VISIBLE_LINE)
RenderLine(CPU.V_Counter - FIRST_VISIBLE_LINE);
#ifndef USE_BLARGG_APU
+ if (APU.TimerEnabled [2])
{
- if (APU.TimerEnabled [2])
+ APU.Timer [2] += 4;
+ while (APU.Timer [2] >= APU.TimerTarget [2])
{
- APU.Timer [2] += 4;
- while (APU.Timer [2] >= APU.TimerTarget [2])
+ IAPU.RAM [0xff] = (IAPU.RAM [0xff] + 1) & 0xf;
+ APU.Timer [2] -= APU.TimerTarget [2];
+ IAPU.WaitCounter++;
+ IAPU.APUExecuting = true;
+ }
+ }
+ if (CPU.V_Counter & 1)
+ {
+ if (APU.TimerEnabled [0])
+ {
+ APU.Timer [0]++;
+ if (APU.Timer [0] >= APU.TimerTarget [0])
{
- IAPU.RAM [0xff] = (IAPU.RAM [0xff] + 1) & 0xf;
- APU.Timer [2] -= APU.TimerTarget [2];
+ IAPU.RAM [0xfd] = (IAPU.RAM [0xfd] + 1) & 0xf;
+ APU.Timer [0] = 0;
IAPU.WaitCounter++;
IAPU.APUExecuting = true;
}
}
- if (CPU.V_Counter & 1)
+ if (APU.TimerEnabled [1])
{
- if (APU.TimerEnabled [0])
+ APU.Timer [1]++;
+ if (APU.Timer [1] >= APU.TimerTarget [1])
{
- APU.Timer [0]++;
- if (APU.Timer [0] >= APU.TimerTarget [0])
- {
- IAPU.RAM [0xfd] = (IAPU.RAM [0xfd] + 1) & 0xf;
- APU.Timer [0] = 0;
- IAPU.WaitCounter++;
- IAPU.APUExecuting = true;
- }
- }
- if (APU.TimerEnabled [1])
- {
- APU.Timer [1]++;
- if (APU.Timer [1] >= APU.TimerTarget [1])
- {
- IAPU.RAM [0xfe] = (IAPU.RAM [0xfe] + 1) & 0xf;
- APU.Timer [1] = 0;
- IAPU.WaitCounter++;
- IAPU.APUExecuting = true;
- }
+ IAPU.RAM [0xfe] = (IAPU.RAM [0xfe] + 1) & 0xf;
+ APU.Timer [1] = 0;
+ IAPU.WaitCounter++;
+ IAPU.APUExecuting = true;
}
}
}
diff --git a/source/cpuexec.h b/source/cpuexec.h
index 9654d86..3a5075b 100644
--- a/source/cpuexec.h
+++ b/source/cpuexec.h
@@ -116,10 +116,8 @@ static INLINE void S9xReschedule(void)
}
if (PPU.HTimerEnabled &&
- (int32_t) PPU.HTimerPosition < max &&
- (int32_t) PPU.HTimerPosition > CPU.NextEvent &&
- (!PPU.VTimerEnabled ||
- (PPU.VTimerEnabled && CPU.V_Counter == PPU.IRQVBeamPos)))
+ (int32_t) PPU.HTimerPosition < max && (int32_t) PPU.HTimerPosition > CPU.NextEvent &&
+ (!PPU.VTimerEnabled || (PPU.VTimerEnabled && CPU.V_Counter == PPU.IRQVBeamPos)))
{
which = (int32_t) PPU.HTimerPosition < Settings.HBlankStart ? HTIMER_BEFORE_EVENT : HTIMER_AFTER_EVENT;
max = PPU.HTimerPosition;
@@ -127,5 +125,4 @@ static INLINE void S9xReschedule(void)
CPU.NextEvent = max;
CPU.WhichEvent = which;
}
-
#endif
diff --git a/source/cpumacro.h b/source/cpumacro.h
index 313e920..77d569f 100644
--- a/source/cpumacro.h
+++ b/source/cpumacro.h
@@ -19,7 +19,7 @@ static INLINE void SetZN8(uint8_t Work)
ICPU._Negative = Work;
}
-static INLINE void ADC8()
+static INLINE void ADC8(void)
{
uint8_t Work8 = S9xGetByte(OpAddress);
@@ -67,7 +67,7 @@ static INLINE void ADC8()
SetZN8(ICPU.Registers.AL);
}
-static INLINE void ADC16()
+static INLINE void ADC16(void)
{
uint16_t Work16 = S9xGetWord(OpAddress);
@@ -134,19 +134,19 @@ static INLINE void ADC16()
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void AND16()
+static INLINE void AND16(void)
{
ICPU.Registers.A.W &= S9xGetWord(OpAddress);
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void AND8()
+static INLINE void AND8(void)
{
ICPU.Registers.AL &= S9xGetByte(OpAddress);
SetZN8(ICPU.Registers.AL);
}
-static INLINE void A_ASL16()
+static INLINE void A_ASL16(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -156,7 +156,7 @@ static INLINE void A_ASL16()
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void A_ASL8()
+static INLINE void A_ASL8(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -166,7 +166,7 @@ static INLINE void A_ASL8()
SetZN8(ICPU.Registers.AL);
}
-static INLINE void ASL16()
+static INLINE void ASL16(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -180,7 +180,7 @@ static INLINE void ASL16()
SetZN16(Work16);
}
-static INLINE void ASL8()
+static INLINE void ASL8(void)
{
uint8_t Work8;
#ifndef SA1_OPCODES
@@ -193,7 +193,7 @@ static INLINE void ASL8()
SetZN8(Work8);
}
-static INLINE void BIT16()
+static INLINE void BIT16(void)
{
uint16_t Work16 = S9xGetWord(OpAddress);
ICPU._Overflow = (Work16 & 0x4000) != 0;
@@ -201,7 +201,7 @@ static INLINE void BIT16()
ICPU._Zero = (Work16 & ICPU.Registers.A.W) != 0;
}
-static INLINE void BIT8()
+static INLINE void BIT8(void)
{
uint8_t Work8 = S9xGetByte(OpAddress);
ICPU._Overflow = (Work8 & 0x40) != 0;
@@ -209,49 +209,49 @@ static INLINE void BIT8()
ICPU._Zero = Work8 & ICPU.Registers.AL;
}
-static INLINE void CMP16()
+static INLINE void CMP16(void)
{
int32_t Int32 = (int32_t) ICPU.Registers.A.W - (int32_t) S9xGetWord(OpAddress);
ICPU._Carry = Int32 >= 0;
SetZN16((uint16_t) Int32);
}
-static INLINE void CMP8()
+static INLINE void CMP8(void)
{
int16_t Int16 = (int16_t) ICPU.Registers.AL - (int16_t) S9xGetByte(OpAddress);
ICPU._Carry = Int16 >= 0;
SetZN8((uint8_t) Int16);
}
-static INLINE void CMX16()
+static INLINE void CMX16(void)
{
int32_t Int32 = (int32_t) ICPU.Registers.X.W - (int32_t) S9xGetWord(OpAddress);
ICPU._Carry = Int32 >= 0;
SetZN16((uint16_t) Int32);
}
-static INLINE void CMX8()
+static INLINE void CMX8(void)
{
int16_t Int16 = (int16_t) ICPU.Registers.XL - (int16_t) S9xGetByte(OpAddress);
ICPU._Carry = Int16 >= 0;
SetZN8((uint8_t) Int16);
}
-static INLINE void CMY16()
+static INLINE void CMY16(void)
{
int32_t Int32 = (int32_t) ICPU.Registers.Y.W - (int32_t) S9xGetWord(OpAddress);
ICPU._Carry = Int32 >= 0;
SetZN16((uint16_t) Int32);
}
-static INLINE void CMY8()
+static INLINE void CMY8(void)
{
int16_t Int16 = (int16_t) ICPU.Registers.YL - (int16_t) S9xGetByte(OpAddress);
ICPU._Carry = Int16 >= 0;
SetZN8((uint8_t) Int16);
}
-static INLINE void A_DEC16()
+static INLINE void A_DEC16(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -261,7 +261,7 @@ static INLINE void A_DEC16()
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void A_DEC8()
+static INLINE void A_DEC8(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -271,10 +271,9 @@ static INLINE void A_DEC8()
SetZN8(ICPU.Registers.AL);
}
-static INLINE void DEC16()
+static INLINE void DEC16(void)
{
uint16_t Work16;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -285,10 +284,9 @@ static INLINE void DEC16()
SetZN16(Work16);
}
-static INLINE void DEC8()
+static INLINE void DEC8(void)
{
uint8_t Work8;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -298,19 +296,19 @@ static INLINE void DEC8()
SetZN8(Work8);
}
-static INLINE void EOR16()
+static INLINE void EOR16(void)
{
ICPU.Registers.A.W ^= S9xGetWord(OpAddress);
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void EOR8()
+static INLINE void EOR8(void)
{
ICPU.Registers.AL ^= S9xGetByte(OpAddress);
SetZN8(ICPU.Registers.AL);
}
-static INLINE void A_INC16()
+static INLINE void A_INC16(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -320,7 +318,7 @@ static INLINE void A_INC16()
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void A_INC8()
+static INLINE void A_INC8(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -330,7 +328,7 @@ static INLINE void A_INC8()
SetZN8(ICPU.Registers.AL);
}
-static INLINE void INC16()
+static INLINE void INC16(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -343,10 +341,9 @@ static INLINE void INC16()
SetZN16(Work16);
}
-static INLINE void INC8()
+static INLINE void INC8(void)
{
uint8_t Work8;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -356,43 +353,43 @@ static INLINE void INC8()
SetZN8(Work8);
}
-static INLINE void LDA16()
+static INLINE void LDA16(void)
{
ICPU.Registers.A.W = S9xGetWord(OpAddress);
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void LDA8()
+static INLINE void LDA8(void)
{
ICPU.Registers.AL = S9xGetByte(OpAddress);
SetZN8(ICPU.Registers.AL);
}
-static INLINE void LDX16()
+static INLINE void LDX16(void)
{
ICPU.Registers.X.W = S9xGetWord(OpAddress);
SetZN16(ICPU.Registers.X.W);
}
-static INLINE void LDX8()
+static INLINE void LDX8(void)
{
ICPU.Registers.XL = S9xGetByte(OpAddress);
SetZN8(ICPU.Registers.XL);
}
-static INLINE void LDY16()
+static INLINE void LDY16(void)
{
ICPU.Registers.Y.W = S9xGetWord(OpAddress);
SetZN16(ICPU.Registers.Y.W);
}
-static INLINE void LDY8()
+static INLINE void LDY8(void)
{
ICPU.Registers.YL = S9xGetByte(OpAddress);
SetZN8(ICPU.Registers.YL);
}
-static INLINE void A_LSR16()
+static INLINE void A_LSR16(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -402,7 +399,7 @@ static INLINE void A_LSR16()
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void A_LSR8()
+static INLINE void A_LSR8(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
@@ -412,10 +409,9 @@ static INLINE void A_LSR8()
SetZN8(ICPU.Registers.AL);
}
-static INLINE void LSR16()
+static INLINE void LSR16(void)
{
uint16_t Work16;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -427,10 +423,9 @@ static INLINE void LSR16()
SetZN16(Work16);
}
-static INLINE void LSR8()
+static INLINE void LSR8(void)
{
uint8_t Work8;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -441,19 +436,19 @@ static INLINE void LSR8()
SetZN8(Work8);
}
-static INLINE void ORA16()
+static INLINE void ORA16(void)
{
ICPU.Registers.A.W |= S9xGetWord(OpAddress);
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void ORA8()
+static INLINE void ORA8(void)
{
ICPU.Registers.AL |= S9xGetByte(OpAddress);
SetZN8(ICPU.Registers.AL);
}
-static INLINE void A_ROL16()
+static INLINE void A_ROL16(void)
{
uint32_t Work32;
#ifndef SA1_OPCODES
@@ -465,7 +460,7 @@ static INLINE void A_ROL16()
SetZN16((uint16_t) Work32);
}
-static INLINE void A_ROL8()
+static INLINE void A_ROL8(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -479,10 +474,9 @@ static INLINE void A_ROL8()
SetZN8((uint8_t) Work16);
}
-static INLINE void ROL16()
+static INLINE void ROL16(void)
{
uint32_t Work32;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -495,7 +489,7 @@ static INLINE void ROL16()
SetZN16((uint16_t) Work32);
}
-static INLINE void ROL8()
+static INLINE void ROL8(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -509,7 +503,7 @@ static INLINE void ROL8()
SetZN8((uint8_t) Work16);
}
-static INLINE void A_ROR16()
+static INLINE void A_ROR16(void)
{
uint32_t Work32;
#ifndef SA1_OPCODES
@@ -523,7 +517,7 @@ static INLINE void A_ROR16()
SetZN16((uint16_t) Work32);
}
-static INLINE void A_ROR8()
+static INLINE void A_ROR8(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -536,10 +530,9 @@ static INLINE void A_ROR8()
SetZN8((uint8_t) Work16);
}
-static INLINE void ROR16()
+static INLINE void ROR16(void)
{
uint32_t Work32;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -552,7 +545,7 @@ static INLINE void ROR16()
SetZN16((uint16_t) Work32);
}
-static INLINE void ROR8()
+static INLINE void ROR8(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -566,7 +559,7 @@ static INLINE void ROR8()
SetZN8((uint8_t) Work16);
}
-static INLINE void SBC16()
+static INLINE void SBC16(void)
{
uint16_t Work16 = S9xGetWord(OpAddress);
@@ -633,7 +626,7 @@ static INLINE void SBC16()
SetZN16(ICPU.Registers.A.W);
}
-static INLINE void SBC8()
+static INLINE void SBC8(void)
{
uint8_t Work8 = S9xGetByte(OpAddress);
if (CheckDecimal())
@@ -681,50 +674,49 @@ static INLINE void SBC8()
SetZN8(ICPU.Registers.AL);
}
-static INLINE void STA16()
+static INLINE void STA16(void)
{
S9xSetWord(ICPU.Registers.A.W, OpAddress);
}
-static INLINE void STA8()
+static INLINE void STA8(void)
{
S9xSetByte(ICPU.Registers.AL, OpAddress);
}
-static INLINE void STX16()
+static INLINE void STX16(void)
{
S9xSetWord(ICPU.Registers.X.W, OpAddress);
}
-static INLINE void STX8()
+static INLINE void STX8(void)
{
S9xSetByte(ICPU.Registers.XL, OpAddress);
}
-static INLINE void STY16()
+static INLINE void STY16(void)
{
S9xSetWord(ICPU.Registers.Y.W, OpAddress);
}
-static INLINE void STY8()
+static INLINE void STY8(void)
{
S9xSetByte(ICPU.Registers.YL, OpAddress);
}
-static INLINE void STZ16()
+static INLINE void STZ16(void)
{
S9xSetWord(0, OpAddress);
}
-static INLINE void STZ8()
+static INLINE void STZ8(void)
{
S9xSetByte(0, OpAddress);
}
-static INLINE void TSB16()
+static INLINE void TSB16(void)
{
uint16_t Work16;
-
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
@@ -735,7 +727,7 @@ static INLINE void TSB16()
S9xSetByte(Work16 & 0xFF, OpAddress);
}
-static INLINE void TSB8()
+static INLINE void TSB8(void)
{
uint8_t Work8;
#ifndef SA1_OPCODES
@@ -747,7 +739,7 @@ static INLINE void TSB8()
S9xSetByte(Work8, OpAddress);
}
-static INLINE void TRB16()
+static INLINE void TRB16(void)
{
uint16_t Work16;
#ifndef SA1_OPCODES
@@ -760,7 +752,7 @@ static INLINE void TRB16()
S9xSetByte(Work16 & 0xFF, OpAddress);
}
-static INLINE void TRB8()
+static INLINE void TRB8(void)
{
uint8_t Work8;
#ifndef SA1_OPCODES
diff --git a/source/cpuops.c b/source/cpuops.c
index 2254713..9a15852 100644
--- a/source/cpuops.c
+++ b/source/cpuops.c
@@ -20,7 +20,7 @@
int32_t OpAddress;
-/* ADC *************************************************************************************** */
+/* ADC */
static void Op69M1(void)
{
Immediate8();
@@ -201,9 +201,7 @@ static void Op73M0(void)
ADC16();
}
-/**********************************************************************************************/
-
-/* AND *************************************************************************************** */
+/* AND */
static void Op29M1(void)
{
ICPU.Registers.AL &= *CPU.PC++;
@@ -394,9 +392,8 @@ static void Op33M0(void)
StackRelativeIndirectIndexed(true);
AND16();
}
-/**********************************************************************************************/
-/* ASL *************************************************************************************** */
+/* ASL */
static void Op0AM1(void)
{
A_ASL8();
@@ -454,9 +451,8 @@ static void Op1EM0(void)
AbsoluteIndexedX(false);
ASL16();
}
-/**********************************************************************************************/
-/* BIT *************************************************************************************** */
+/* BIT */
static void Op89M1(void)
{
ICPU._Zero = ICPU.Registers.AL & *CPU.PC++;
@@ -525,9 +521,8 @@ static void Op3CM0(void)
AbsoluteIndexedX(true);
BIT16();
}
-/**********************************************************************************************/
-/* CMP *************************************************************************************** */
+/* CMP */
static void OpC9M1(void)
{
int32_t Int32 = (int32_t) ICPU.Registers.AL - (int32_t) *CPU.PC++;
@@ -721,9 +716,7 @@ static void OpD3M0(void)
CMP16();
}
-/**********************************************************************************************/
-
-/* CMX *************************************************************************************** */
+/* CMX */
static void OpE0X1(void)
{
int32_t Int32 = (int32_t) ICPU.Registers.XL - (int32_t) *CPU.PC++;
@@ -773,9 +766,7 @@ static void OpECX0(void)
CMX16();
}
-/**********************************************************************************************/
-
-/* CMY *************************************************************************************** */
+/* CMY */
static void OpC0X1(void)
{
int32_t Int32 = (int32_t) ICPU.Registers.YL - (int32_t) *CPU.PC++;
@@ -825,9 +816,7 @@ static void OpCCX0(void)
CMY16();
}
-/**********************************************************************************************/
-
-/* DEC *************************************************************************************** */
+/* DEC */
static void Op3AM1(void)
{
A_DEC8();
@@ -886,9 +875,7 @@ static void OpDEM0(void)
DEC16();
}
-/**********************************************************************************************/
-
-/* EOR *************************************************************************************** */
+/* EOR */
static void Op49M1(void)
{
ICPU.Registers.AL ^= *CPU.PC++;
@@ -1080,9 +1067,7 @@ static void Op53M0(void)
EOR16();
}
-/**********************************************************************************************/
-
-/* INC *************************************************************************************** */
+/* INC */
static void Op1AM1(void)
{
A_INC8();
@@ -1141,8 +1126,7 @@ static void OpFEM0(void)
INC16();
}
-/**********************************************************************************************/
-/* LDA *************************************************************************************** */
+/* LDA */
static void OpA9M1(void)
{
ICPU.Registers.AL = *CPU.PC++;
@@ -1335,9 +1319,7 @@ static void OpB3M0(void)
LDA16();
}
-/**********************************************************************************************/
-
-/* LDX *************************************************************************************** */
+/* LDX */
static void OpA2X1(void)
{
ICPU.Registers.XL = *CPU.PC++;
@@ -1408,9 +1390,8 @@ static void OpBEX0(void)
AbsoluteIndexedY(true);
LDX16();
}
-/**********************************************************************************************/
-/* LDY *************************************************************************************** */
+/* LDY */
static void OpA0X1(void)
{
ICPU.Registers.YL = *CPU.PC++;
@@ -1482,9 +1463,8 @@ static void OpBCX0(void)
AbsoluteIndexedX(true);
LDY16();
}
-/**********************************************************************************************/
-/* LSR *************************************************************************************** */
+/* LSR */
static void Op4AM1(void)
{
A_LSR8();
@@ -1543,9 +1523,7 @@ static void Op5EM0(void)
LSR16();
}
-/**********************************************************************************************/
-
-/* ORA *************************************************************************************** */
+/* ORA */
static void Op09M1(void)
{
ICPU.Registers.AL |= *CPU.PC++;
@@ -1737,9 +1715,7 @@ static void Op13M0(void)
ORA16();
}
-/**********************************************************************************************/
-
-/* ROL *************************************************************************************** */
+/* ROL */
static void Op2AM1(void)
{
A_ROL8();
@@ -1797,9 +1773,8 @@ static void Op3EM0(void)
AbsoluteIndexedX(false);
ROL16();
}
-/**********************************************************************************************/
-/* ROR *************************************************************************************** */
+/* ROR */
static void Op6AM1(void)
{
A_ROR8();
@@ -1857,9 +1832,8 @@ static void Op7EM0(void)
AbsoluteIndexedX(false);
ROR16();
}
-/**********************************************************************************************/
-/* SBC *************************************************************************************** */
+/* SBC */
static void OpE9M1(void)
{
Immediate8();
@@ -2039,9 +2013,8 @@ static void OpF3M0(void)
StackRelativeIndirectIndexed(true);
SBC16();
}
-/**********************************************************************************************/
-/* STA *************************************************************************************** */
+/* STA */
static void Op85M1(void)
{
Direct(false);
@@ -2217,9 +2190,8 @@ static void Op93M0(void)
StackRelativeIndirectIndexed(false);
STA16();
}
-/**********************************************************************************************/
-/* STX *************************************************************************************** */
+/* STX */
static void Op86X1(void)
{
Direct(false);
@@ -2255,9 +2227,8 @@ static void Op8EX0(void)
Absolute(false);
STX16();
}
-/**********************************************************************************************/
-/* STY *************************************************************************************** */
+/* STY */
static void Op84X1(void)
{
Direct(false);
@@ -2293,9 +2264,8 @@ static void Op8CX0(void)
Absolute(false);
STY16();
}
-/**********************************************************************************************/
-/* STZ *************************************************************************************** */
+/* STZ */
static void Op64M1(void)
{
Direct(false);
@@ -2344,9 +2314,7 @@ static void Op9EM0(void)
STZ16();
}
-/**********************************************************************************************/
-
-/* TRB *************************************************************************************** */
+/* TRB */
static void Op14M1(void)
{
Direct(false);
@@ -2370,9 +2338,8 @@ static void Op1CM0(void)
Absolute(false);
TRB16();
}
-/**********************************************************************************************/
-/* TSB *************************************************************************************** */
+/* TSB */
static void Op04M1(void)
{
Direct(false);
@@ -2397,9 +2364,7 @@ static void Op0CM0(void)
TSB16();
}
-/**********************************************************************************************/
-
-/* Branch Instructions *********************************************************************** */
+/* Branch Instructions */
#ifndef SA1_OPCODES
#define BranchCheck() \
if(CPU.BranchSkip) \
@@ -2421,7 +2386,6 @@ static INLINE void CPUShutdown(void)
// interrupt. Interrupts are delayed for a few cycles already, but
// the delay could allow the shutdown code to cycle skip again.
// Was causing screen flashing on Top Gear 3000.
-
if (CPU.WaitCounter == 0 && !(CPU.Flags & (IRQ_PENDING_FLAG | NMI_FLAG)))
{
CPU.WaitAddress = NULL;
@@ -2433,8 +2397,7 @@ static INLINE void CPUShutdown(void)
do
{
APU_EXECUTE1();
- }
- while (APU.Cycles < CPU.NextEvent);
+ } while (APU.Cycles < CPU.NextEvent);
ICPU.CPUExecuting = true;
}
#endif
@@ -2474,8 +2437,7 @@ static INLINE void ForceShutdown(void)
do
{
APU_EXECUTE1();
- }
- while (APU.Cycles < CPU.NextEvent);
+ } while (APU.Cycles < CPU.NextEvent);
ICPU.CPUExecuting = true;
}
#endif
@@ -2615,9 +2577,8 @@ static void Op70(void)
CPUShutdown();
}
}
-/**********************************************************************************************/
-/* ClearFlag Instructions ******************************************************************** */
+/* ClearFlag Instructions */
/* CLC */
static void Op18(void)
{
@@ -2643,7 +2604,6 @@ static void Op58(void)
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
- /* CHECK_FOR_IRQ(); */
}
/* CLV */
@@ -2654,9 +2614,8 @@ static void OpB8(void)
CPU.Cycles += ONE_CYCLE;
#endif
}
-/**********************************************************************************************/
-/* DEX/DEY *********************************************************************************** */
+/* DEX/DEY */
static void OpCAX1(void)
{
#ifndef SA1_OPCODES
@@ -2696,9 +2655,8 @@ static void Op88X0(void)
ICPU.Registers.Y.W--;
SetZN16(ICPU.Registers.Y.W);
}
-/**********************************************************************************************/
-/* INX/INY *********************************************************************************** */
+/* INX/INY */
static void OpE8X1(void)
{
#ifndef SA1_OPCODES
@@ -2739,35 +2697,32 @@ static void OpC8X0(void)
SetZN16(ICPU.Registers.Y.W);
}
-/**********************************************************************************************/
-
-/* NOP *************************************************************************************** */
+/* NOP */
static void OpEA(void)
{
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
}
-/**********************************************************************************************/
-/* PUSH Instructions ************************************************************************* */
+/* PUSH Instructions */
#define PushB(b)\
S9xSetByte(b, ICPU.Registers.S.W--);
#define PushBE(b)\
PushB(b);\
- ICPU.Registers.SH = 0x01;
+ ICPU.Registers.SH = 0x01
#define PushW(w)\
S9xSetByte((w) >> 8, ICPU.Registers.S.W);\
S9xSetByte((w) & 0xff, (ICPU.Registers.S.W - 1) & 0xffff);\
- ICPU.Registers.S.W -= 2;
+ ICPU.Registers.S.W -= 2
#define PushWE(w)\
PushW(w); \
- ICPU.Registers.SH = 0x01;
+ ICPU.Registers.SH = 0x01
-//PEA NL
+/* PEA NL */
static void OpF4E1(void)
{
Absolute(false);
@@ -2780,7 +2735,7 @@ static void OpF4(void)
PushW((uint16_t)OpAddress);
}
-//PEI NL
+/* PEI NL */
static void OpD4E1(void)
{
DirectIndirect(false);
@@ -2793,20 +2748,20 @@ static void OpD4(void)
PushW((uint16_t)OpAddress);
}
-//PER NL
+/* PER NL */
static void Op62E1(void)
{
- RelativeLong(false);
+ RelativeLong();
PushWE((uint16_t)OpAddress);
}
static void Op62(void)
{
- RelativeLong(false);
+ RelativeLong();
PushW((uint16_t)OpAddress);
}
-//PHA
+/* PHA */
static void Op48E1(void)
{
PushBE(ICPU.Registers.AL);
@@ -2831,7 +2786,7 @@ static void Op48M0(void)
#endif
}
-//PHB
+/* PHB */
static void Op8BE1(void)
{
PushBE(ICPU.Registers.DB);
@@ -2848,7 +2803,7 @@ static void Op8B(void)
#endif
}
-//PHD NL
+/* PHD NL */
static void Op0BE1(void)
{
PushWE(ICPU.Registers.D.W);
@@ -2865,7 +2820,7 @@ static void Op0B(void)
#endif
}
-//PHK
+/* PHK */
static void Op4BE1(void)
{
PushBE(ICPU.Registers.PB);
@@ -2882,7 +2837,7 @@ static void Op4B(void)
#endif
}
-//PHP
+/* PHP */
static void Op08E1(void)
{
S9xPackStatus();
@@ -2901,7 +2856,7 @@ static void Op08(void)
#endif
}
-//PHX
+/* PHX */
static void OpDAE1(void)
{
PushBE(ICPU.Registers.XL);
@@ -2926,7 +2881,7 @@ static void OpDAX0(void)
#endif
}
-//PHY
+/* PHY */
static void Op5AE1(void)
{
PushBE(ICPU.Registers.YL);
@@ -2950,25 +2905,24 @@ static void Op5AX0(void)
CPU.Cycles += ONE_CYCLE;
#endif
}
-/**********************************************************************************************/
-/* PULL Instructions ************************************************************************* */
+/* PULL Instructions */
#define PullB(b)\
- b = S9xGetByte (++ICPU.Registers.S.W);
+ b = S9xGetByte (++ICPU.Registers.S.W)
#define PullBE(b)\
PullB(b);\
- ICPU.Registers.SH = 0x01;
+ ICPU.Registers.SH = 0x01
#define PullW(w)\
w = S9xGetByte(++ICPU.Registers.S.W);\
- w |= (S9xGetByte(++ICPU.Registers.S.W) << 8);
+ w |= (S9xGetByte(++ICPU.Registers.S.W) << 8)
#define PullWE(w)\
PullW(w);\
- ICPU.Registers.SH = 0x01;
+ ICPU.Registers.SH = 0x01
-//PLA
+/* PLA */
static void Op68E1(void)
{
#ifndef SA1_OPCODES
@@ -2996,7 +2950,7 @@ static void Op68M0(void)
SetZN16(ICPU.Registers.A.W);
}
-//PLB
+/* PLB */
static void OpABE1(void)
{
#ifndef SA1_OPCODES
@@ -3017,8 +2971,7 @@ static void OpAB(void)
ICPU.ShiftedDB = ICPU.Registers.DB << 16;
}
-/* PHP */
-//PLD NL
+/* PLD NL */
static void Op2BE1(void)
{
#ifndef SA1_OPCODES
@@ -3070,7 +3023,7 @@ static void Op28(void)
S9xFixCycles();
}
-//PLX
+/* PLX */
static void OpFAE1(void)
{
#ifndef SA1_OPCODES
@@ -3098,7 +3051,7 @@ static void OpFAX0(void)
SetZN16(ICPU.Registers.X.W);
}
-//PLY
+/* PLY */
static void Op7AE1(void)
{
#ifndef SA1_OPCODES
@@ -3126,9 +3079,6 @@ static void Op7AX0(void)
SetZN16(ICPU.Registers.Y.W);
}
-/**********************************************************************************************/
-
-/* SetFlag Instructions ********************************************************************** */
/* SEC */
static void Op38(void)
{
@@ -3155,9 +3105,7 @@ static void Op78(void)
CPU.Cycles += ONE_CYCLE;
#endif
}
-/**********************************************************************************************/
-/* Transfer Instructions ********************************************************************* */
/* TAX8 */
static void OpAAX1(void)
{
@@ -3335,9 +3283,7 @@ static void OpBBX0(void)
SetZN16(ICPU.Registers.X.W);
}
-/**********************************************************************************************/
-
-/* XCE *************************************************************************************** */
+/* XCE */
static void OpFB(void)
{
uint8_t A1, A2;
@@ -3361,9 +3307,8 @@ static void OpFB(void)
}
S9xFixCycles();
}
-/**********************************************************************************************/
-/* BRK *************************************************************************************** */
+/* BRK */
static void Op00(void)
{
if (!CheckEmulation())
@@ -3400,17 +3345,15 @@ static void Op00(void)
#endif
}
}
-/**********************************************************************************************/
-/* BRL ************************************************************************************** */
+/* BRL */
static void Op82(void)
{
RelativeLong();
S9xSetPCBase(ICPU.ShiftedPB + OpAddress);
}
-/**********************************************************************************************/
-/* IRQ *************************************************************************************** */
+/* IRQ */
void S9xOpcode_IRQ(void)
{
if (!CheckEmulation())
@@ -3426,16 +3369,12 @@ void S9xOpcode_IRQ(void)
ICPU.Registers.PB = 0;
ICPU.ShiftedPB = 0;
#ifdef SA1_OPCODES
- S9xSA1SetPCBase(Memory.FillRAM [0x2207] |
- (Memory.FillRAM [0x2208] << 8));
+ S9xSA1SetPCBase(Memory.FillRAM [0x2207] | (Memory.FillRAM [0x2208] << 8));
#else
if (Settings.SA1 && (Memory.FillRAM [0x2209] & 0x40))
- S9xSetPCBase(Memory.FillRAM [0x220e] |
- (Memory.FillRAM [0x220f] << 8));
+ S9xSetPCBase(Memory.FillRAM [0x220e] | (Memory.FillRAM [0x220f] << 8));
else
S9xSetPCBase(S9xGetWord(0xFFEE));
-#endif
-#ifndef SA1_OPCODES
CPU.Cycles += TWO_CYCLES;
#endif
}
@@ -3451,24 +3390,18 @@ void S9xOpcode_IRQ(void)
ICPU.Registers.PB = 0;
ICPU.ShiftedPB = 0;
#ifdef SA1_OPCODES
- S9xSA1SetPCBase(Memory.FillRAM [0x2207] |
- (Memory.FillRAM [0x2208] << 8));
+ S9xSA1SetPCBase(Memory.FillRAM [0x2207] | (Memory.FillRAM [0x2208] << 8));
#else
if (Settings.SA1 && (Memory.FillRAM [0x2209] & 0x40))
- S9xSetPCBase(Memory.FillRAM [0x220e] |
- (Memory.FillRAM [0x220f] << 8));
+ S9xSetPCBase(Memory.FillRAM [0x220e] | (Memory.FillRAM [0x220f] << 8));
else
S9xSetPCBase(S9xGetWord(0xFFFE));
-#endif
-#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
}
}
-/**********************************************************************************************/
-
-/* NMI *************************************************************************************** */
+/* NMI */
void S9xOpcode_NMI(void)
{
if (!CheckEmulation())
@@ -3484,16 +3417,12 @@ void S9xOpcode_NMI(void)
ICPU.Registers.PB = 0;
ICPU.ShiftedPB = 0;
#ifdef SA1_OPCODES
- S9xSA1SetPCBase(Memory.FillRAM [0x2205] |
- (Memory.FillRAM [0x2206] << 8));
+ S9xSA1SetPCBase(Memory.FillRAM [0x2205] | (Memory.FillRAM [0x2206] << 8));
#else
if (Settings.SA1 && (Memory.FillRAM [0x2209] & 0x20))
- S9xSetPCBase(Memory.FillRAM [0x220c] |
- (Memory.FillRAM [0x220d] << 8));
+ S9xSetPCBase(Memory.FillRAM [0x220c] | (Memory.FillRAM [0x220d] << 8));
else
S9xSetPCBase(S9xGetWord(0xFFEA));
-#endif
-#ifndef SA1_OPCODES
CPU.Cycles += TWO_CYCLES;
#endif
}
@@ -3509,23 +3438,18 @@ void S9xOpcode_NMI(void)
ICPU.Registers.PB = 0;
ICPU.ShiftedPB = 0;
#ifdef SA1_OPCODES
- S9xSA1SetPCBase(Memory.FillRAM [0x2205] |
- (Memory.FillRAM [0x2206] << 8));
+ S9xSA1SetPCBase(Memory.FillRAM [0x2205] | (Memory.FillRAM [0x2206] << 8));
#else
if (Settings.SA1 && (Memory.FillRAM [0x2209] & 0x20))
- S9xSetPCBase(Memory.FillRAM [0x220c] |
- (Memory.FillRAM [0x220d] << 8));
+ S9xSetPCBase(Memory.FillRAM [0x220c] | (Memory.FillRAM [0x220d] << 8));
else
S9xSetPCBase(S9xGetWord(0xFFFA));
-#endif
-#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
#endif
}
}
-/**********************************************************************************************/
-/* COP *************************************************************************************** */
+/* COP */
static void Op02(void)
{
if (!CheckEmulation())
@@ -3562,9 +3486,8 @@ static void Op02(void)
#endif
}
}
-/**********************************************************************************************/
-/* JML *************************************************************************************** */
+/* JML */
static void OpDC(void)
{
AbsoluteIndirectLong(false);
@@ -3583,9 +3506,8 @@ static void Op5C(void)
ICPU.ShiftedPB = OpAddress & 0xff0000;
S9xSetPCBase(OpAddress);
}
-/**********************************************************************************************/
-/* JMP *************************************************************************************** */
+/* JMP */
static void Op4C(void)
{
Absolute(false);
@@ -3609,9 +3531,8 @@ static void Op7C(void)
CPU.Cycles += ONE_CYCLE;
#endif
}
-/**********************************************************************************************/
-/* JSL/RTL *********************************************************************************** */
+/* JSL/RTL */
static void Op22E1(void)
{
AbsoluteLong(false);
@@ -3653,9 +3574,8 @@ static void Op6B(void)
CPU.Cycles += TWO_CYCLES;
#endif
}
-/**********************************************************************************************/
-/* JSR/RTS *********************************************************************************** */
+/* JSR/RTS */
static void Op20(void)
{
Absolute(false);
@@ -3696,9 +3616,7 @@ static void Op60(void)
#endif
}
-/**********************************************************************************************/
-
-/* MVN/MVP *********************************************************************************** */
+/* MVN/MVP */
static void Op54X1(void)
{
#ifndef SA1_OPCODES
@@ -3771,9 +3689,7 @@ static void Op44X0(void)
CPU.PC -= 3;
}
-/**********************************************************************************************/
-
-/* REP/SEP *********************************************************************************** */
+/* REP/SEP */
static void OpC2(void)
{
uint8_t Work8 = ~*CPU.PC++;
@@ -3817,9 +3733,8 @@ static void OpE2(void)
}
S9xFixCycles();
}
-/**********************************************************************************************/
-/* XBA *************************************************************************************** */
+/* XBA */
static void OpEB(void)
{
uint8_t Work8 = ICPU.Registers.AL;
@@ -3830,9 +3745,8 @@ static void OpEB(void)
CPU.Cycles += TWO_CYCLES;
#endif
}
-/**********************************************************************************************/
-/* RTI *************************************************************************************** */
+/* RTI */
static void Op40(void)
{
PullB(ICPU.Registers.PL);
@@ -3857,10 +3771,7 @@ static void Op40(void)
S9xFixCycles();
}
-/**********************************************************************************************/
-
-/* STP/WAI/DB ******************************************************************************** */
-// WAI
+/* WAI */
static void OpCB(void)
{
#ifdef SA1_OPCODES
@@ -3888,8 +3799,7 @@ static void OpCB(void)
#endif
}
-// Usually an STP opcode
-// SNESAdvance speed hack, not implemented in Snes9xTYL | Snes9x-Euphoria (from the speed-hacks branch of CatSFC)
+// Usually an STP opcode; SNESAdvance speed hack, not implemented in Snes9xTYL | Snes9x-Euphoria (from the speed-hacks branch of CatSFC)
static void OpDB(void)
{
#ifndef NO_SPEEDHACKS
@@ -3911,10 +3821,6 @@ static void OpDB(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -3926,10 +3832,6 @@ static void OpDB(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -3962,10 +3864,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -3977,10 +3875,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -3992,10 +3886,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -4007,10 +3897,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -4019,10 +3905,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
return;
@@ -4033,10 +3915,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -4048,10 +3926,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -4063,10 +3937,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -4078,10 +3948,6 @@ static void Op42(void)
CPU.PC = CPU.PCBase + OpAddress;
#ifndef SA1_OPCODES
CPU.Cycles += ONE_CYCLE;
-#else
-#ifndef SA1_OPCODES
- CPU.Cycles++;
-#endif
#endif
CPUShutdown ();
}
@@ -4090,11 +3956,7 @@ static void Op42(void)
#endif
}
-/*****************************************************************************/
-
-/*****************************************************************************/
-/* CPU-S9xOpcodes Definitions */
-/*****************************************************************************/
+/* CPU-S9xOpcodes Definitions */
SOpcodes S9xOpcodesM1X1[256] =
{
{Op00}, {Op01M1}, {Op02}, {Op03M1}, {Op04M1},
diff --git a/source/display.h b/source/display.h
index e7732bc..e43b80c 100644
--- a/source/display.h
+++ b/source/display.h
@@ -14,5 +14,4 @@ void S9xToggleSoundChannel(int32_t channel);
void S9xNextController(void);
const char* S9xGetFilename(const char* extension);
-
#endif
diff --git a/source/dma.c b/source/dma.c
index eefe135..6bc7922 100644
--- a/source/dma.c
+++ b/source/dma.c
@@ -7,8 +7,9 @@
#include "dma.h"
#include "apu.h"
#include "sa1.h"
-#include "spc7110.h"
#include "sdd1emu.h"
+#include "spc7110.h"
+#include "spc7110dec.h"
static uint8_t sdd1_decode_buffer[0x10000];
@@ -17,7 +18,7 @@ extern uint8_t* HDMAMemPointers [8];
extern uint8_t* HDMABasePointers [8];
/**********************************************************************************************/
-/* S9xDoDMA() */
+/* S9xDoDMA() */
/* This function preforms the general dma transfer */
/**********************************************************************************************/
void S9xDoDMA(uint8_t Channel)
@@ -36,7 +37,6 @@ void S9xDoDMA(uint8_t Channel)
CPU.InDMA = true;
d = &DMA[Channel];
-
count = d->TransferBytes;
// Prepare for custom chip DMA
@@ -48,7 +48,7 @@ void S9xDoDMA(uint8_t Channel)
if ((d->ABank == 0x7E || d->ABank == 0x7F) && d->BAddress == 0x80 && !d->TransferDirection)
{
d->AAddress += d->TransferBytes;
- //does an invalid DMA actually take time?
+ // Does an invalid DMA actually take time?
// I'd say yes, since 'invalid' is probably just the WRAM chip
// not being able to read and write itself at the same time
CPU.Cycles += (d->TransferBytes + 1) * SLOW_ONE_CYCLE;
@@ -66,8 +66,7 @@ void S9xDoDMA(uint8_t Channel)
{
if (d->AAddressFixed && Memory.FillRAM [0x4801] > 0)
{
- uint8_t *in_ptr;
-
+ uint8_t* in_ptr;
// XXX: Should probably verify that we're DMAing from ROM?
// And somewhere we should make sure we're not running across a mapping boundary too.
inc = !d->AAddressDecrement ? 1 : -1;
@@ -85,44 +84,23 @@ void S9xDoDMA(uint8_t Channel)
}
if (Settings.SPC7110 && (d->AAddress == 0x4800 || d->ABank == 0x50))
{
- uint32_t i;
- int32_t icount;
- i = (s7r.reg4805 | (s7r.reg4806 << 8));
- i *= s7r.AlignBy;
- i += s7r.bank50Internal;
- i %= DECOMP_BUFFER_SIZE;
- if ((i + d->TransferBytes) < DECOMP_BUFFER_SIZE)
- spc7110_dma = &s7r.bank50[i];
- else
- {
- uint32_t j;
-
- spc7110_dma = (uint8_t*)malloc(d->TransferBytes);
- j = DECOMP_BUFFER_SIZE - i;
- memcpy(spc7110_dma, &s7r.bank50[i], j);
- memcpy(&spc7110_dma[j], s7r.bank50, d->TransferBytes - j);
- s7_wrap = true;
- }
-
- icount = s7r.reg4809 | (s7r.reg480A << 8);
- icount -= d->TransferBytes;
+ int32_t c;
+ spc7110_dma = &s7r.bank50[0];
+ for(c = 0; c < count; c++)
+ s7r.bank50[c] = spc7110dec_read();
+ int32_t icount = (s7r.reg4809 | (s7r.reg480A << 8)) - count;
s7r.reg4809 = 0x00ff & icount;
s7r.reg480A = (0xff00 & icount) >> 8;
-
- s7r.bank50Internal += d->TransferBytes;
- s7r.bank50Internal %= DECOMP_BUFFER_SIZE;
inc = 1;
d->AAddress -= count;
}
if (d->BAddress == 0x18 && SA1.in_char_dma && (d->ABank & 0xf0) == 0x40)
{
- int32_t i;
// Perform packed bitmap to PPU character format conversion on the
// data before transmitting it to V-RAM via-DMA.
+ int32_t i;
int32_t num_chars = 1 << ((Memory.FillRAM [0x2231] >> 2) & 7);
- int32_t depth = (Memory.FillRAM [0x2231] & 3) == 0 ? 8 :
- (Memory.FillRAM [0x2231] & 3) == 1 ? 4 : 2;
-
+ int32_t depth = (Memory.FillRAM [0x2231] & 3) == 0 ? 8 : (Memory.FillRAM [0x2231] & 3) == 1 ? 4 : 2;
int32_t bytes_per_char = 8 * depth;
int32_t bytes_per_line = depth * num_chars;
int32_t char_line_bytes = bytes_per_char * num_chars;
@@ -132,19 +110,16 @@ void S9xDoDMA(uint8_t Channel)
uint8_t* p = buffer;
uint32_t inc = char_line_bytes - (d->AAddress % char_line_bytes);
uint32_t char_count = inc / bytes_per_char;
-
in_sa1_dma = true;
switch (depth)
{
case 2:
- for (i = 0; i < count; i += inc, base += char_line_bytes,
- inc = char_line_bytes, char_count = num_chars)
+ for (i = 0 ; i < count ; i += inc, base += char_line_bytes, inc = char_line_bytes, char_count = num_chars)
{
uint32_t j;
uint8_t* line = base + (num_chars - char_count) * 2;
- for (j = 0; j < char_count && p - buffer < count;
- j++, line += 2)
+ for (j = 0 ; j < char_count && p - buffer < count ; j++, line += 2)
{
int32_t b, l;
uint8_t* q = line;
@@ -168,13 +143,11 @@ void S9xDoDMA(uint8_t Channel)
}
break;
case 4:
- for (i = 0; i < count; i += inc, base += char_line_bytes,
- inc = char_line_bytes, char_count = num_chars)
+ for (i = 0 ; i < count ; i += inc, base += char_line_bytes, inc = char_line_bytes, char_count = num_chars)
{
uint32_t j;
uint8_t* line = base + (num_chars - char_count) * 4;
- for (j = 0; j < char_count && p - buffer < count;
- j++, line += 4)
+ for (j = 0 ; j < char_count && p - buffer < count ; j++, line += 4)
{
uint8_t* q = line;
int32_t b, l;
@@ -199,13 +172,11 @@ void S9xDoDMA(uint8_t Channel)
}
break;
case 8:
- for (i = 0; i < count; i += inc, base += char_line_bytes,
- inc = char_line_bytes, char_count = num_chars)
+ for(i = 0 ; i < count ; i += inc, base += char_line_bytes, inc = char_line_bytes, char_count = num_chars)
{
uint8_t* line = base + (num_chars - char_count) * 8;
uint32_t j;
- for (j = 0; j < char_count && p - buffer < count;
- j++, line += 8)
+ for(j = 0 ; j < char_count && p - buffer < count ; j++, line += 8)
{
uint8_t* q = line;
int32_t b, l;
@@ -236,13 +207,12 @@ void S9xDoDMA(uint8_t Channel)
{
uint8_t* base;
uint16_t p;
-
/* XXX: DMA is potentially broken here for cases where we DMA across
* XXX: memmap boundries. A possible solution would be to re-call
* XXX: GetBasePointer whenever we cross a boundry, and when
* XXX: GetBasePointer returns (0) to take the 'slow path' and use
* XXX: S9xGetByte instead of *base. GetBasePointer() would want to
- * XXX: return (0) for MAP_PPU and whatever else is a register range
+ * XXX: return 0 for MAP_PPU and whatever else is a register range
* XXX: rather than a RAM/ROM block, and we'd want to detect MAP_PPU
* XXX: (or specifically, Address Bus B addresses $2100-$21FF in
* XXX: banks $00-$3F) specially and treat it as MAP_NONE (since
@@ -263,7 +233,6 @@ void S9xDoDMA(uint8_t Channel)
base = &Memory.ROM [MAX_ROM_SIZE - 0x10000];
p = 0;
}
-
if (in_sdd1_dma)
{
base = in_sdd1_dma;
@@ -289,8 +258,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2104(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
break;
case 0x18:
IPPU.FirstVRAMRead = true;
@@ -301,8 +269,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2118_linear(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
}
else
{
@@ -311,8 +278,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2118_tile(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
}
break;
case 0x19:
@@ -324,8 +290,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2119_linear(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
}
else
{
@@ -334,8 +299,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2119_tile(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
}
break;
case 0x22:
@@ -344,8 +308,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2122(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
break;
case 0x80:
do
@@ -353,8 +316,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
REGISTER_2180(Work);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
break;
default:
do
@@ -362,8 +324,7 @@ void S9xDoDMA(uint8_t Channel)
Work = *(base + p);
S9xSetPPU(Work, 0x2100 + d->BAddress);
p += inc;
- }
- while (--count > 0);
+ } while (--count > 0);
break;
}
}
@@ -459,8 +420,7 @@ void S9xDoDMA(uint8_t Channel)
S9xSetPPU(Work, 0x2101 + d->BAddress);
p += inc;
count -= 4;
- }
- while (count > 0);
+ } while (count > 0);
}
else if (d->TransferMode == 4)
{
@@ -488,8 +448,7 @@ void S9xDoDMA(uint8_t Channel)
S9xSetPPU(Work, 0x2103 + d->BAddress);
p += inc;
count -= 4;
- }
- while (count > 0);
+ } while (count > 0);
}
}
else
@@ -549,7 +508,6 @@ void S9xDoDMA(uint8_t Channel)
d->AAddress += inc;
count--;
break;
-
case 4:
Work = S9xGetPPU(0x2100 + d->BAddress);
S9xSetByte(Work, (d->ABank << 16) + d->AAddress);
@@ -574,13 +532,11 @@ void S9xDoDMA(uint8_t Channel)
d->AAddress += inc;
count--;
break;
-
default:
count = 0;
break;
}
- }
- while (count);
+ } while (count);
}
#ifndef USE_BLARGG_APU
IAPU.APUExecuting = Settings.APUEnabled;
@@ -593,20 +549,15 @@ void S9xDoDMA(uint8_t Channel)
while (CPU.Cycles > CPU.NextEvent)
S9xDoHBlankProcessing_NoSFX();
- if (Settings.SPC7110 && spc7110_dma)
- {
- if (spc7110_dma && s7_wrap)
- free(spc7110_dma);
- }
+ if (Settings.SPC7110 && spc7110_dma && s7_wrap)
+ free(spc7110_dma);
update_address:
- // Super Punch-Out requires that the A-BUS address be updated after the
- // DMA transfer.
+ // Super Punch-Out requires that the A-BUS address be updated after the DMA transfer.
Memory.FillRAM[0x4302 + (Channel << 4)] = (uint8_t) d->AAddress;
Memory.FillRAM[0x4303 + (Channel << 4)] = d->AAddress >> 8;
- // Secret of the Mana requires that the DMA bytes transfer count be set to
- // zero when DMA has completed.
+ // Secret of the Mana requires that the DMA bytes transfer count be set to zero when DMA has completed.
Memory.FillRAM [0x4305 + (Channel << 4)] = 0;
Memory.FillRAM [0x4306 + (Channel << 4)] = 0;
@@ -616,16 +567,11 @@ update_address:
CPU.InDMA = false;
}
-void S9xStartHDMA()
+void S9xStartHDMA(void)
{
uint8_t i;
+ IPPU.HDMA = Memory.FillRAM [0x420c];
- if (Settings.DisableHDMA)
- IPPU.HDMA = 0;
- else
- IPPU.HDMA = Memory.FillRAM [0x420c];
-
- //per anomie timing post
if (IPPU.HDMA != 0)
CPU.Cycles += ONE_CYCLE * 3;
@@ -649,7 +595,6 @@ uint8_t S9xDoHDMA(uint8_t byte)
uint8_t mask;
SDMA* p = &DMA [0];
int32_t d = 0;
-
CPU.InDMA = true;
CPU.Cycles += ONE_CYCLE * 3;
@@ -660,7 +605,6 @@ uint8_t S9xDoHDMA(uint8_t byte)
if (!p->LineCount)
{
uint8_t line;
-
//remember, InDMA is set.
//Get/Set incur no charges!
CPU.Cycles += SLOW_ONE_CYCLE;
@@ -718,8 +662,7 @@ uint8_t S9xDoHDMA(uint8_t byte)
p->IndirectAddress = p->Address;
}
- if (!(HDMABasePointers [d] = HDMAMemPointers [d] =
- S9xGetMemPointer((p->IndirectBank << 16) + p->IndirectAddress)))
+ if (!(HDMABasePointers [d] = HDMAMemPointers [d] = S9xGetMemPointer((p->IndirectBank << 16) + p->IndirectAddress)))
{
/* XXX: Instead of this, goto a slow path that first
* XXX: verifies src!=Address Bus B, then uses
@@ -793,7 +736,7 @@ uint8_t S9xDoHDMA(uint8_t byte)
return byte;
}
-void S9xResetDMA()
+void S9xResetDMA(void)
{
int32_t c, d;
for (d = 0; d < 8; d++)
diff --git a/source/dma.h b/source/dma.h
index c2ab8d2..f9ad36f 100644
--- a/source/dma.h
+++ b/source/dma.h
@@ -7,5 +7,4 @@ void S9xResetDMA(void);
uint8_t S9xDoHDMA(uint8_t);
void S9xStartHDMA(void);
void S9xDoDMA(uint8_t);
-
#endif
diff --git a/source/dsp1.c b/source/dsp1.c
index 98c0077..e0ffa99 100644
--- a/source/dsp1.c
+++ b/source/dsp1.c
@@ -9,7 +9,7 @@
void (*SetDSP)(uint8_t, uint16_t) = &DSP1SetByte;
uint8_t(*GetDSP)(uint16_t) = &DSP1GetByte;
-void S9xResetDSP1()
+void S9xResetDSP1(void)
{
DSP1.waiting4command = true;
DSP1.in_count = 0;
@@ -21,10 +21,7 @@ void S9xResetDSP1()
uint8_t S9xGetDSP(uint16_t address)
{
- uint8_t t;
-
- t = (*GetDSP)(address);
- return (t);
+ return (*GetDSP)(address);
}
void S9xSetDSP(uint8_t byte, uint16_t address)
@@ -48,8 +45,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.in_index = 0;
DSP1.waiting4command = false;
DSP1.first_parameter = true;
- // Mario Kart uses 0x00, 0x02, 0x06, 0x0c, 0x28, 0x0a
- switch (byte)
+ switch (byte) // Mario Kart uses 0x00, 0x02, 0x06, 0x0c, 0x28, 0x0a
{
case 0x07:
case 0x0a:
@@ -159,7 +155,6 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
{
if (--DSP1.in_count == 0)
{
- // Actually execute the command
DSP1.waiting4command = true;
DSP1.out_index = 0;
switch (DSP1.command)
@@ -167,33 +162,27 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
case 0x1f:
DSP1.out_count = 2048;
break;
- case 0x00: // Multiple
+ case 0x00: // Multiple
Op00Multiplicand = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op00Multiplier = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
-
DSPOp00();
-
DSP1.out_count = 2;
DSP1.output [0] = Op00Result & 0xFF;
DSP1.output [1] = (Op00Result >> 8) & 0xFF;
break;
- case 0x20: // Multiple
+ case 0x20: // Multiple
Op20Multiplicand = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op20Multiplier = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
-
DSPOp20();
-
DSP1.out_count = 2;
DSP1.output [0] = Op20Result & 0xFF;
DSP1.output [1] = (Op20Result >> 8) & 0xFF;
break;
case 0x30:
- case 0x10: // Inverse
+ case 0x10: // Inverse
Op10Coefficient = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op10Exponent = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
-
DSPOp10();
-
DSP1.out_count = 4;
DSP1.output [0] = (uint8_t)(((int16_t) Op10CoefficientR) & 0xFF);
DSP1.output [1] = (uint8_t)((((int16_t) Op10CoefficientR) >> 8) & 0xFF);
@@ -201,74 +190,62 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [3] = (uint8_t)((((int16_t) Op10ExponentR) >> 8) & 0xff);
break;
case 0x24:
- case 0x04: // Sin and Cos of angle
+ case 0x04: // Sin and Cos of angle
Op04Angle = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op04Radius = (uint16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
-
DSPOp04();
-
DSP1.out_count = 4;
DSP1.output [0] = (uint8_t)(Op04Sin & 0xFF);
DSP1.output [1] = (uint8_t)((Op04Sin >> 8) & 0xFF);
DSP1.output [2] = (uint8_t)(Op04Cos & 0xFF);
DSP1.output [3] = (uint8_t)((Op04Cos >> 8) & 0xFF);
break;
- case 0x08: // Radius
+ case 0x08: // Radius
Op08X = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op08Y = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op08Z = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp08();
-
DSP1.out_count = 4;
DSP1.output [0] = (uint8_t)(((int16_t) Op08Ll) & 0xFF);
DSP1.output [1] = (uint8_t)((((int16_t) Op08Ll) >> 8) & 0xFF);
DSP1.output [2] = (uint8_t)(((int16_t) Op08Lh) & 0xFF);
DSP1.output [3] = (uint8_t)((((int16_t) Op08Lh) >> 8) & 0xFF);
break;
- case 0x18: // Range
+ case 0x18: // Range
Op18X = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op18Y = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op18Z = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
Op18R = (int16_t)(DSP1.parameters [6] | (DSP1.parameters[7] << 8));
-
DSPOp18();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op18D & 0xFF);
DSP1.output [1] = (uint8_t)((Op18D >> 8) & 0xFF);
break;
- case 0x38: // Range
+ case 0x38: // Range
Op38X = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op38Y = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op38Z = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
Op38R = (int16_t)(DSP1.parameters [6] | (DSP1.parameters[7] << 8));
-
DSPOp38();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op38D & 0xFF);
DSP1.output [1] = (uint8_t)((Op38D >> 8) & 0xFF);
break;
- case 0x28: // Distance (vector length)
+ case 0x28: // Distance (vector length)
Op28X = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op28Y = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op28Z = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp28();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op28R & 0xFF);
DSP1.output [1] = (uint8_t)((Op28R >> 8) & 0xFF);
break;
case 0x2c:
- case 0x0c: // Rotate (2D rotate)
+ case 0x0c: // Rotate (2D rotate)
Op0CA = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op0CX1 = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op0CY1 = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp0C();
-
DSP1.out_count = 4;
DSP1.output [0] = (uint8_t)(Op0CX2 & 0xFF);
DSP1.output [1] = (uint8_t)((Op0CX2 >> 8) & 0xFF);
@@ -276,17 +253,14 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [3] = (uint8_t)((Op0CY2 >> 8) & 0xFF);
break;
case 0x3c:
- case 0x1c: // Polar (3D rotate)
+ case 0x1c: // Polar (3D rotate)
Op1CZ = (DSP1.parameters [0] | (DSP1.parameters[1] << 8));
- //MK: reversed X and Y on neviksti and John's advice.
Op1CY = (DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op1CX = (DSP1.parameters [4] | (DSP1.parameters[5] << 8));
Op1CXBR = (DSP1.parameters [6] | (DSP1.parameters[7] << 8));
Op1CYBR = (DSP1.parameters [8] | (DSP1.parameters[9] << 8));
Op1CZBR = (DSP1.parameters [10] | (DSP1.parameters[11] << 8));
-
DSPOp1C();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op1CXAR & 0xFF);
DSP1.output [1] = (uint8_t)((Op1CXAR >> 8) & 0xFF);
@@ -298,7 +272,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
case 0x32:
case 0x22:
case 0x12:
- case 0x02: // Parameter (Projection)
+ case 0x02: // Parameter (Projection)
Op02FX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op02FY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op02FZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
@@ -306,9 +280,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
Op02LES = (int16_t)(DSP1.parameters [8] | (DSP1.parameters[9] << 8));
Op02AAS = (uint16_t)(DSP1.parameters [10] | (DSP1.parameters[11] << 8));
Op02AZS = (uint16_t)(DSP1.parameters [12] | (DSP1.parameters[13] << 8));
-
DSPOp02();
-
DSP1.out_count = 8;
DSP1.output [0] = (uint8_t)(Op02VOF & 0xFF);
DSP1.output [1] = (uint8_t)((Op02VOF >> 8) & 0xFF);
@@ -319,14 +291,12 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [6] = (uint8_t)(Op02CY & 0xFF);
DSP1.output [7] = (uint8_t)((Op02CY >> 8) & 0xFF);
break;
- case 0x3a: //1a Mirror
- case 0x2a: //1a Mirror
- case 0x1a: // Raster mode 7 matrix data
+ case 0x3a: //1a Mirror
+ case 0x2a: //1a Mirror
+ case 0x1a: // Raster mode 7 matrix data
case 0x0a:
Op0AVS = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
-
DSPOp0A();
-
DSP1.out_count = 8;
DSP1.output [0] = (uint8_t)(Op0AA & 0xFF);
DSP1.output [2] = (uint8_t)(Op0AB & 0xFF);
@@ -341,13 +311,11 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
case 0x16:
case 0x26:
case 0x36:
- case 0x06: // Project object
+ case 0x06: // Project object
Op06X = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op06Y = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op06Z = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp06();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op06H & 0xff);
DSP1.output [1] = (uint8_t)((Op06H >> 8) & 0xFF);
@@ -359,20 +327,17 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
case 0x1e:
case 0x2e:
case 0x3e:
- case 0x0e: // Target
+ case 0x0e: // Target
Op0EH = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op0EV = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
-
DSPOp0E();
-
DSP1.out_count = 4;
DSP1.output [0] = (uint8_t)(Op0EX & 0xFF);
DSP1.output [1] = (uint8_t)((Op0EX >> 8) & 0xFF);
DSP1.output [2] = (uint8_t)(Op0EY & 0xFF);
DSP1.output [3] = (uint8_t)((Op0EY >> 8) & 0xFF);
break;
- // Extra commands used by Pilot Wings
- case 0x05:
+ case 0x05: // Extra commands used by Pilot Wings
case 0x35:
case 0x31:
case 0x01: // Set attitude matrix A
@@ -380,37 +345,32 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
Op01Zr = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op01Yr = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
Op01Xr = (int16_t)(DSP1.parameters [6] | (DSP1.parameters[7] << 8));
-
DSPOp01();
break;
case 0x15:
- case 0x11: // Set attitude matrix B
+ case 0x11: // Set attitude matrix B
Op11m = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op11Zr = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op11Yr = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
Op11Xr = (int16_t)(DSP1.parameters [7] | (DSP1.parameters[7] << 8));
-
DSPOp11();
break;
case 0x25:
- case 0x21: // Set attitude matrix C
+ case 0x21: // Set attitude matrix C
Op21m = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op21Zr = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op21Yr = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
Op21Xr = (int16_t)(DSP1.parameters [6] | (DSP1.parameters[7] << 8));
-
DSPOp21();
break;
case 0x09:
case 0x39:
case 0x3d:
- case 0x0d: // Objective matrix A
+ case 0x0d: // Objective matrix A
Op0DX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op0DY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op0DZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp0D();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op0DF & 0xFF);
DSP1.output [1] = (uint8_t)((Op0DF >> 8) & 0xFF);
@@ -420,13 +380,11 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [5] = (uint8_t)((Op0DU >> 8) & 0xFF);
break;
case 0x19:
- case 0x1d: // Objective matrix B
+ case 0x1d: // Objective matrix B
Op1DX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op1DY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op1DZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp1D();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op1DF & 0xFF);
DSP1.output [1] = (uint8_t)((Op1DF >> 8) & 0xFF);
@@ -436,13 +394,11 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [5] = (uint8_t)((Op1DU >> 8) & 0xFF);
break;
case 0x29:
- case 0x2d: // Objective matrix C
+ case 0x2d: // Objective matrix C
Op2DX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op2DY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op2DZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp2D();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op2DF & 0xFF);
DSP1.output [1] = (uint8_t)((Op2DF >> 8) & 0xFF);
@@ -452,13 +408,11 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [5] = (uint8_t)((Op2DU >> 8) & 0xFF);
break;
case 0x33:
- case 0x03: // Subjective matrix A
+ case 0x03: // Subjective matrix A
Op03F = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op03L = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op03U = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp03();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op03X & 0xFF);
DSP1.output [1] = (uint8_t)((Op03X >> 8) & 0xFF);
@@ -467,13 +421,11 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [4] = (uint8_t)(Op03Z & 0xFF);
DSP1.output [5] = (uint8_t)((Op03Z >> 8) & 0xFF);
break;
- case 0x13: // Subjective matrix B
+ case 0x13: // Subjective matrix B
Op13F = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op13L = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op13U = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp13();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op13X & 0xFF);
DSP1.output [1] = (uint8_t)((Op13X >> 8) & 0xFF);
@@ -482,13 +434,11 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
DSP1.output [4] = (uint8_t)(Op13Z & 0xFF);
DSP1.output [5] = (uint8_t)((Op13Z >> 8) & 0xFF);
break;
- case 0x23: // Subjective matrix C
+ case 0x23: // Subjective matrix C
Op23F = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op23L = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op23U = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp23();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op23X & 0xFF);
DSP1.output [1] = (uint8_t)((Op23X >> 8) & 0xFF);
@@ -502,9 +452,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
Op0BX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op0BY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op0BZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp0B();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op0BS & 0xFF);
DSP1.output [1] = (uint8_t)((Op0BS >> 8) & 0xFF);
@@ -513,9 +461,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
Op1BX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op1BY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op1BZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp1B();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op1BS & 0xFF);
DSP1.output [1] = (uint8_t)((Op1BS >> 8) & 0xFF);
@@ -524,9 +470,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
Op2BX = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
Op2BY = (int16_t)(DSP1.parameters [2] | (DSP1.parameters[3] << 8));
Op2BZ = (int16_t)(DSP1.parameters [4] | (DSP1.parameters[5] << 8));
-
DSPOp2B();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op2BS & 0xFF);
DSP1.output [1] = (uint8_t)((Op2BS >> 8) & 0xFF);
@@ -539,9 +483,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
Op14U = (int16_t)(DSP1.parameters [6] | (DSP1.parameters[7] << 8));
Op14F = (int16_t)(DSP1.parameters [8] | (DSP1.parameters[9] << 8));
Op14L = (int16_t)(DSP1.parameters [10] | (DSP1.parameters[11] << 8));
-
DSPOp14();
-
DSP1.out_count = 6;
DSP1.output [0] = (uint8_t)(Op14Zrr & 0xFF);
DSP1.output [1] = (uint8_t)((Op14Zrr >> 8) & 0xFF);
@@ -553,9 +495,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
case 0x27:
case 0x2F:
Op2FUnknown = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
-
DSPOp2F();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op2FSize & 0xFF);
DSP1.output [1] = (uint8_t)((Op2FSize >> 8) & 0xFF);
@@ -563,9 +503,7 @@ void DSP1SetByte(uint8_t byte, uint16_t address)
case 0x07:
case 0x0F:
Op0FRamsize = (int16_t)(DSP1.parameters [0] | (DSP1.parameters[1] << 8));
-
DSPOp0F();
-
DSP1.out_count = 2;
DSP1.output [0] = (uint8_t)(Op0FPass & 0xFF);
DSP1.output [1] = (uint8_t)((Op0FPass >> 8) & 0xFF);
@@ -628,8 +566,7 @@ void DSP2SetByte(uint8_t byte, uint16_t address)
uint32_t temp;
#endif
- if ((address & 0xf000) == 0x6000 ||
- (address >= 0x8000 && address < 0xc000))
+ if ((address & 0xf000) == 0x6000 || (address >= 0x8000 && address < 0xc000))
{
if (DSP1.waiting4command)
{
@@ -665,7 +602,6 @@ void DSP2SetByte(uint8_t byte, uint16_t address)
if (DSP1.in_count == DSP1.in_index)
{
- // Actually execute the command
DSP1.waiting4command = true;
DSP1.out_index = 0;
switch (DSP1.command)
@@ -675,7 +611,6 @@ void DSP2SetByte(uint8_t byte, uint16_t address)
{
DSP2Op0DHasLen = false;
DSP1.out_count = DSP2Op0DOutLen;
- //execute Op5
DSP2_Op0D();
}
else
@@ -694,7 +629,6 @@ void DSP2SetByte(uint8_t byte, uint16_t address)
{
DSP2Op06HasLen = false;
DSP1.out_count = DSP2Op06Len;
- //execute Op5
DSP2_Op06();
}
else
@@ -711,8 +645,7 @@ void DSP2SetByte(uint8_t byte, uint16_t address)
DSP1.out_count = 32;
DSP2_Op01();
break;
- case 0x09:
- // Multiply - don't yet know if this is signed or unsigned
+ case 0x09: // Multiply - don't yet know if this is signed or unsigned
DSP2Op09Word1 = DSP1.parameters[0] | (DSP1.parameters[1] << 8);
DSP2Op09Word2 = DSP1.parameters[2] | (DSP1.parameters[3] << 8);
DSP1.out_count = 4;
@@ -731,7 +664,6 @@ void DSP2SetByte(uint8_t byte, uint16_t address)
{
DSP2Op05HasLen = false;
DSP1.out_count = DSP2Op05Len;
- //execute Op5
DSP2_Op05();
}
else
@@ -797,9 +729,8 @@ bool DSP4_init = false;
void DSP4SetByte(uint8_t byte, uint16_t address)
{
- if (!DSP4_init)
+ if (!DSP4_init) // bootup
{
- // bootup
DSP4.waiting4command = 1;
DSP4_init = true;
}
@@ -876,76 +807,59 @@ void DSP4SetByte(uint8_t byte, uint16_t address)
if (!DSP4.waiting4command && DSP4.in_count == DSP4.in_index)
{
- // Actually execute the command
DSP4.waiting4command = true;
DSP4.out_index = 0;
DSP4.in_index = 0;
switch (DSP4.command)
{
- // 16-bit multiplication
- case 0x0000:
+ case 0x0000: // 16-bit multiplication
{
int16_t multiplier, multiplicand;
int32_t product;
-
multiplier = DSP4_READ_WORD(0);
multiplicand = DSP4_READ_WORD(2);
-
product = DSP4_Multiply(multiplicand, multiplier);
-
DSP4.out_count = 4;
DSP4_WRITE_WORD(0, product);
DSP4_WRITE_WORD(2, product >> 16);
break;
}
- // unknown: horizontal mapping command
- case 0x0011:
+ case 0x0011: // unknown: horizontal mapping command
{
int16_t a, b, c, d, m;
-
a = DSP4_READ_WORD(6);
b = DSP4_READ_WORD(4);
c = DSP4_READ_WORD(2);
d = DSP4_READ_WORD(0);
-
m = DSP4_UnknownOP11(a, b, c, d);
-
DSP4.out_count = 2;
DSP4_WRITE_WORD(0, m);
break;
}
- // track projection
- case 0x0001:
+ case 0x0001: // track projection
DSP4_Op01();
break;
- // track projection (pass 2)
- case 0x0007:
+ case 0x0007: // track projection (pass 2)
DSP4_Op07();
break;
- // zone projections (fuel/repair/lap/teleport/...)
- case 0x0008:
+ case 0x0008: // zone projections (fuel/repair/lap/teleport/...)
DSP4_Op08();
break;
- // sprite transformation
- case 0x0009:
+ case 0x0009: // sprite transformation
DSP4_Op09();
break;
- // fast track projection
- case 0x000D:
+ case 0x000D: // fast track projection
DSP4_Op0D();
break;
- // internal memory management (01)
- case 0x0003:
+ case 0x0003: // internal memory management (01)
{
// reset op09 data
op09_mode = false;
break;
}
- // internal memory management (06)
- case 0x0005:
+ case 0x0005: // internal memory management (06)
{
int32_t lcv;
-
// clear OAM tables
op06_index = 0;
op06_offset = 0;
@@ -953,49 +867,39 @@ void DSP4SetByte(uint8_t byte, uint16_t address)
op06_OAM[lcv] = 0;
break;
}
- // internal memory management (0D)
- case 0x000E:
+ case 0x000E: // internal memory management (0D)
{
// reset op09 data
op09_mode = true;
break;
}
- // sprite OAM post-table data
- case 0x0006:
+ case 0x0006: // sprite OAM post-table data
{
int32_t lcv;
-
DSP4.out_count = 32;
for (lcv = 0; lcv < 32; lcv++)
DSP4.output[lcv] = op06_OAM[lcv];
+ break;
}
- break;
- // unknown
- case 0x000A:
+ case 0x000A: // unknown
{
int16_t out1a, out2a;
-
out1a = (int16_t)0xff40;
out2a = (int16_t)0x00c0;
-
DSP4.out_count = 8;
-
DSP4_WRITE_WORD(0, out1a);
DSP4_WRITE_WORD(2, out2a);
DSP4_WRITE_WORD(4, out1a);
DSP4_WRITE_WORD(6, out2a);
+ break;
}
- break;
-
- // render player positions around track
- case 0x000B:
+ case 0x000B: // render player positions around track
{
int16_t sp_x = DSP4_READ_WORD(0);
int16_t sp_y = DSP4_READ_WORD(2);
int16_t oam = DSP4_READ_WORD(4);
- // Only allow 1p/1p-split to yield output (???)
- if (!op09_mode)
+ if (!op09_mode) // Only allow 1p/1p-split to yield output (???)
{
// yield OAM output
DSP4.out_count = 6;
@@ -1009,8 +913,7 @@ void DSP4SetByte(uint8_t byte, uint16_t address)
// OAM: size,msb data
DSP4_Op06(false, false);
}
- // 4p mode
- else
+ else // 4p mode
{
// no OAM available
DSP4.out_count = 0;
@@ -1028,8 +931,7 @@ void DSP4SetByte(uint8_t byte, uint16_t address)
uint8_t DSP4GetByte(uint16_t address)
{
uint8_t t;
- if ((address & 0xf000) == 0x6000 ||
- (address >= 0x8000 && address < 0xc000))
+ if ((address & 0xf000) == 0x6000 || (address >= 0x8000 && address < 0xc000))
{
if (DSP4.out_count)
{
diff --git a/source/dsp1.h b/source/dsp1.h
index 4cc2eca..63306c8 100644
--- a/source/dsp1.h
+++ b/source/dsp1.h
@@ -35,5 +35,4 @@ void S9xResetDSP1(void);
uint8_t S9xGetDSP(uint16_t Address);
void S9xSetDSP(uint8_t Byte, uint16_t Address);
extern SDSP1 DSP1;
-
#endif
diff --git a/source/dsp1emu.c b/source/dsp1emu.c
index 6ed470c..1a78e5f 100644
--- a/source/dsp1emu.c
+++ b/source/dsp1emu.c
@@ -146,7 +146,7 @@ int16_t Op00Multiplicand;
int16_t Op00Multiplier;
int16_t Op00Result;
-void DSPOp00()
+void DSPOp00(void)
{
Op00Result = Op00Multiplicand * Op00Multiplier >> 15;
}
@@ -155,7 +155,7 @@ int16_t Op20Multiplicand;
int16_t Op20Multiplier;
int16_t Op20Result;
-void DSPOp20()
+void DSPOp20(void)
{
Op20Result = Op20Multiplicand * Op20Multiplier >> 15;
Op20Result++;
@@ -181,7 +181,8 @@ void DSP1_Inverse(int16_t Coefficient, int16_t Exponent, int16_t* iCoefficient,
// Step Two: Remove Sign
if (Coefficient < 0)
{
- if (Coefficient < -32767) Coefficient = -32767;
+ if (Coefficient < -32767)
+ Coefficient = -32767;
Coefficient = -Coefficient;
Sign = -1;
}
@@ -195,12 +196,15 @@ void DSP1_Inverse(int16_t Coefficient, int16_t Exponent, int16_t* iCoefficient,
// Step Four: Special Case
if (Coefficient == 0x4000)
- if (Sign == 1) *iCoefficient = 0x7fff;
+ {
+ if (Sign == 1)
+ *iCoefficient = 0x7fff;
else
{
*iCoefficient = -0x4000;
Exponent--;
}
+ }
else
{
// Step Five: Initial Guess
@@ -217,7 +221,7 @@ void DSP1_Inverse(int16_t Coefficient, int16_t Exponent, int16_t* iCoefficient,
}
}
-void DSPOp10()
+void DSPOp10(void)
{
DSP1_Inverse(Op10Coefficient, Op10Exponent, &Op10CoefficientR, &Op10ExponentR);
}
@@ -305,11 +309,13 @@ int16_t DSP1_Sin(int16_t Angle)
if (Angle < 0)
{
- if (Angle == -32768) return 0;
+ if (Angle == -32768)
+ return 0;
return -DSP1_Sin(-Angle);
}
S = DSP1_SinTable[Angle >> 8] + (DSP1_MulTable[Angle & 0xff] * DSP1_SinTable[0x40 + (Angle >> 8)] >> 15);
- if (S > 32767) S = 32767;
+ if (S > 32767)
+ S = 32767;
return (int16_t) S;
}
@@ -319,11 +325,13 @@ int16_t DSP1_Cos(int16_t Angle)
if (Angle < 0)
{
- if (Angle == -32768) return -32768;
+ if (Angle == -32768)
+ return -32768;
Angle = -Angle;
}
S = DSP1_SinTable[0x40 + (Angle >> 8)] - (DSP1_MulTable[Angle & 0xff] * DSP1_SinTable[Angle >> 8] >> 15);
- if (S < -32768) S = -32767;
+ if (S < -32768)
+ S = -32767;
return (int16_t) S;
}
@@ -333,17 +341,21 @@ void DSP1_Normalize(int16_t m, int16_t* Coefficient, int16_t* Exponent)
int16_t e = 0;
if (m < 0)
+ {
while ((m & i) && i)
{
i >>= 1;
e++;
}
+ }
else
+ {
while (!(m & i) && i)
{
i >>= 1;
e++;
}
+ }
if (e > 0)
*Coefficient = m * DSP1ROM[0x21 + e] << 1;
@@ -361,17 +373,21 @@ void DSP1_NormalizeDouble(int32_t Product, int16_t* Coefficient, int16_t* Expone
int16_t e = 0;
if (m < 0)
+ {
while ((m & i) && i)
{
i >>= 1;
e++;
}
+ }
else
+ {
while (!(m & i) && i)
{
i >>= 1;
e++;
}
+ }
if (e > 0)
{
@@ -384,17 +400,21 @@ void DSP1_NormalizeDouble(int32_t Product, int16_t* Coefficient, int16_t* Expone
i = 0x4000;
if (m < 0)
+ {
while ((n & i) && i)
{
i >>= 1;
e++;
}
+ }
else
+ {
while (!(n & i) && i)
{
i >>= 1;
e++;
}
+ }
if (e > 15)
*Coefficient = n * DSP1ROM[0x0012 + e] << 1;
@@ -412,17 +432,17 @@ int16_t DSP1_Truncate(int16_t C, int16_t E)
{
if (E > 0)
{
- if (C > 0) return 32767;
- else if (C < 0) return -32767;
- }
- else
- {
- if (E < 0) return C * DSP1ROM[0x0031 + E] >> 15;
+ if (C > 0)
+ return 32767;
+ else if (C < 0)
+ return -32767;
}
+ else if (E < 0)
+ return C * DSP1ROM[0x0031 + E] >> 15;
return C;
}
-void DSPOp04()
+void DSPOp04(void)
{
Op04Sin = DSP1_Sin(Op04Angle) * Op04Radius >> 15;
Op04Cos = DSP1_Cos(Op04Angle) * Op04Radius >> 15;
@@ -434,7 +454,7 @@ int16_t Op0CY1;
int16_t Op0CX2;
int16_t Op0CY2;
-void DSPOp0C()
+void DSPOp0C(void)
{
Op0CX2 = (Op0CY1 * DSP1_Sin(Op0CA) >> 15) + (Op0CX1 * DSP1_Cos(Op0CA) >> 15);
Op0CY2 = (Op0CY1 * DSP1_Cos(Op0CA) >> 15) - (Op0CX1 * DSP1_Sin(Op0CA) >> 15);
@@ -524,7 +544,8 @@ void DSP1_Parameter(int16_t Fx, int16_t Fy, int16_t Fz, int16_t Lfe, int16_t Les
if (AZS < 0)
{
MaxAZS = -MaxAZS;
- if (AZS < MaxAZS + 1) AZS = MaxAZS + 1;
+ if (AZS < MaxAZS + 1)
+ AZS = MaxAZS + 1;
}
else if (AZS > MaxAZS)
AZS = MaxAZS;
@@ -550,10 +571,11 @@ void DSP1_Parameter(int16_t Fx, int16_t Fy, int16_t Fz, int16_t Lfe, int16_t Les
if ((Azs != AZS) || (Azs == MaxAZS))
{
- if (Azs == -32768) Azs = -32767;
-
+ if (Azs == -32768)
+ Azs = -32767;
C = Azs - MaxAZS;
- if (C >= 0) C--;
+ if (C >= 0)
+ C--;
Aux = ~(C << 2);
C = Aux * DSP1ROM[0x0328] >> 15;
@@ -620,7 +642,7 @@ int16_t Op02VVA;
int16_t Op02CX;
int16_t Op02CY;
-void DSPOp02()
+void DSPOp02(void)
{
DSP1_Parameter(Op02FX, Op02FY, Op02FZ, Op02LFE, Op02LES, Op02AAS, Op02AZS, &Op02VOF, &Op02VVA, &Op02CX, &Op02CY);
}
@@ -631,7 +653,7 @@ int16_t Op0AB;
int16_t Op0AC;
int16_t Op0AD;
-void DSPOp0A()
+void DSPOp0A(void)
{
DSP1_Raster(Op0AVS, &Op0AA, &Op0AB, &Op0AC, &Op0AD);
Op0AVS++;
@@ -639,7 +661,7 @@ void DSPOp0A()
int16_t DSP1_ShiftR(int16_t C, int16_t E)
{
- return (C * DSP1ROM[0x0031 + E] >> 15);
+ return C * DSP1ROM[0x0031 + E] >> 15;
}
void DSP1_Project(int16_t X, int16_t Y, int16_t Z, int16_t *H, int16_t *V, int16_t *M)
@@ -661,17 +683,17 @@ void DSP1_Project(int16_t X, int16_t Y, int16_t Z, int16_t *H, int16_t *V, int16
Pz>>=1;
E3--;
- refE = MIN(E, E3);
- refE = MIN(refE, E4);
+ refE = MIN(E, E3);
+ refE = MIN(refE, E4);
- Px = DSP1_ShiftR(Px, E4 - refE); // normalize them to the same exponent
- Py = DSP1_ShiftR(Py, E - refE);
- Pz = DSP1_ShiftR(Pz, E3 - refE);
+ Px = DSP1_ShiftR(Px, E4 - refE); // normalize them to the same exponent
+ Py = DSP1_ShiftR(Py, E - refE);
+ Pz = DSP1_ShiftR(Pz, E3 - refE);
- C11 = -(Px * Nx >> 15);
- C8 = -(Py * Ny >> 15);
- C9 = -(Pz * Nz >> 15);
- C12 = C11 + C8 + C9; // this cannot overflow!
+ C11 = -(Px * Nx >> 15);
+ C8 = -(Py * Ny >> 15);
+ C9 = -(Pz * Nz >> 15);
+ C12 = C11 + C8 + C9; // this cannot overflow!
aux4 = C12; // de-normalization with 32-bit arithmetic
refE = 16 - refE; // refE can be up to 3
@@ -723,7 +745,7 @@ int16_t Op06H;
int16_t Op06V;
int16_t Op06M;
-void DSPOp06()
+void DSPOp06(void)
{
DSP1_Project(Op06X, Op06Y, Op06Z, &Op06H, &Op06V, &Op06M);
}
@@ -745,7 +767,7 @@ int16_t Op21Zr;
int16_t Op21Xr;
int16_t Op21Yr;
-void DSPOp01()
+void DSPOp01(void)
{
int16_t SinAz = DSP1_Sin(Op01Zr);
int16_t CosAz = DSP1_Cos(Op01Zr);
@@ -769,7 +791,7 @@ void DSPOp01()
matrixA[2][2] = (Op01m * CosAx >> 15) * CosAy >> 15;
}
-void DSPOp11()
+void DSPOp11(void)
{
int16_t SinAz = DSP1_Sin(Op11Zr);
int16_t CosAz = DSP1_Cos(Op11Zr);
@@ -793,7 +815,7 @@ void DSPOp11()
matrixB[2][2] = (Op11m * CosAx >> 15) * CosAy >> 15;
}
-void DSPOp21()
+void DSPOp21(void)
{
int16_t SinAz = DSP1_Sin(Op21Zr);
int16_t CosAz = DSP1_Cos(Op21Zr);
@@ -836,21 +858,21 @@ int16_t Op2DF;
int16_t Op2DL;
int16_t Op2DU;
-void DSPOp0D()
+void DSPOp0D(void)
{
Op0DF = (Op0DX * matrixA[0][0] >> 15) + (Op0DY * matrixA[0][1] >> 15) + (Op0DZ * matrixA[0][2] >> 15);
Op0DL = (Op0DX * matrixA[1][0] >> 15) + (Op0DY * matrixA[1][1] >> 15) + (Op0DZ * matrixA[1][2] >> 15);
Op0DU = (Op0DX * matrixA[2][0] >> 15) + (Op0DY * matrixA[2][1] >> 15) + (Op0DZ * matrixA[2][2] >> 15);
}
-void DSPOp1D()
+void DSPOp1D(void)
{
Op1DF = (Op1DX * matrixB[0][0] >> 15) + (Op1DY * matrixB[0][1] >> 15) + (Op1DZ * matrixB[0][2] >> 15);
Op1DL = (Op1DX * matrixB[1][0] >> 15) + (Op1DY * matrixB[1][1] >> 15) + (Op1DZ * matrixB[1][2] >> 15);
Op1DU = (Op1DX * matrixB[2][0] >> 15) + (Op1DY * matrixB[2][1] >> 15) + (Op1DZ * matrixB[2][2] >> 15);
}
-void DSPOp2D()
+void DSPOp2D(void)
{
Op2DF = (Op2DX * matrixC[0][0] >> 15) + (Op2DY * matrixC[0][1] >> 15) + (Op2DZ * matrixC[0][2] >> 15);
Op2DL = (Op2DX * matrixC[1][0] >> 15) + (Op2DY * matrixC[1][1] >> 15) + (Op2DZ * matrixC[1][2] >> 15);
@@ -876,21 +898,21 @@ int16_t Op23X;
int16_t Op23Y;
int16_t Op23Z;
-void DSPOp03()
+void DSPOp03(void)
{
Op03X = (Op03F * matrixA[0][0] >> 15) + (Op03L * matrixA[1][0] >> 15) + (Op03U * matrixA[2][0] >> 15);
Op03Y = (Op03F * matrixA[0][1] >> 15) + (Op03L * matrixA[1][1] >> 15) + (Op03U * matrixA[2][1] >> 15);
Op03Z = (Op03F * matrixA[0][2] >> 15) + (Op03L * matrixA[1][2] >> 15) + (Op03U * matrixA[2][2] >> 15);
}
-void DSPOp13()
+void DSPOp13(void)
{
Op13X = (Op13F * matrixB[0][0] >> 15) + (Op13L * matrixB[1][0] >> 15) + (Op13U * matrixB[2][0] >> 15);
Op13Y = (Op13F * matrixB[0][1] >> 15) + (Op13L * matrixB[1][1] >> 15) + (Op13U * matrixB[2][1] >> 15);
Op13Z = (Op13F * matrixB[0][2] >> 15) + (Op13L * matrixB[1][2] >> 15) + (Op13U * matrixB[2][2] >> 15);
}
-void DSPOp23()
+void DSPOp23(void)
{
Op23X = (Op23F * matrixC[0][0] >> 15) + (Op23L * matrixC[1][0] >> 15) + (Op23U * matrixC[2][0] >> 15);
Op23Y = (Op23F * matrixC[0][1] >> 15) + (Op23L * matrixC[1][1] >> 15) + (Op23U * matrixC[2][1] >> 15);
@@ -907,7 +929,7 @@ int16_t Op14Zrr;
int16_t Op14Xrr;
int16_t Op14Yrr;
-void DSPOp14()
+void DSPOp14(void)
{
int16_t CSec, ESec, CTan, CSin, C, E;
@@ -973,7 +995,7 @@ int16_t Op0EV;
int16_t Op0EX;
int16_t Op0EY;
-void DSPOp0E()
+void DSPOp0E(void)
{
DSP1_Target(Op0EH, Op0EV, &Op0EX, &Op0EY);
}
@@ -991,24 +1013,24 @@ int16_t Op2BY;
int16_t Op2BZ;
int16_t Op2BS;
-void DSPOp0B()
+void DSPOp0B(void)
{
Op0BS = (Op0BX * matrixA[0][0] + Op0BY * matrixA[0][1] + Op0BZ * matrixA[0][2]) >> 15;
}
-void DSPOp1B()
+void DSPOp1B(void)
{
Op1BS = (Op1BX * matrixB[0][0] + Op1BY * matrixB[0][1] + Op1BZ * matrixB[0][2]) >> 15;
}
-void DSPOp2B()
+void DSPOp2B(void)
{
Op2BS = (Op2BX * matrixC[0][0] + Op2BY * matrixC[0][1] + Op2BZ * matrixC[0][2]) >> 15;
}
int16_t Op08X, Op08Y, Op08Z, Op08Ll, Op08Lh;
-void DSPOp08()
+void DSPOp08(void)
{
int32_t Op08Size = (Op08X * Op08X + Op08Y * Op08Y + Op08Z * Op08Z) << 1;
Op08Ll = Op08Size & 0xffff;
@@ -1017,14 +1039,14 @@ void DSPOp08()
int16_t Op18X, Op18Y, Op18Z, Op18R, Op18D;
-void DSPOp18()
+void DSPOp18(void)
{
Op18D = (Op18X * Op18X + Op18Y * Op18Y + Op18Z * Op18Z - Op18R * Op18R) >> 15;
}
int16_t Op38X, Op38Y, Op38Z, Op38R, Op38D;
-void DSPOp38()
+void DSPOp38(void)
{
Op38D = (Op38X * Op38X + Op38Y * Op38Y + Op38Z * Op38Z - Op38R * Op38R) >> 15;
Op38D++;
@@ -1035,16 +1057,18 @@ int16_t Op28Y;
int16_t Op28Z;
int16_t Op28R;
-void DSPOp28()
+void DSPOp28(void)
{
int32_t Radius = Op28X * Op28X + Op28Y * Op28Y + Op28Z * Op28Z;
- if (Radius == 0) Op28R = 0;
+ if (Radius == 0)
+ Op28R = 0;
else
{
int16_t C, E, Pos, Node1, Node2;
DSP1_NormalizeDouble(Radius, &C, &E);
- if (E & 1) C = C * 0x4000 >> 15;
+ if (E & 1)
+ C = C * 0x4000 >> 15;
Pos = C * 0x0040 >> 15;
@@ -1065,7 +1089,7 @@ int16_t Op1CX2;
int16_t Op1CY2;
int16_t Op1CZ2;
-void DSPOp1C()
+void DSPOp1C(void)
{
// Rotate Around Op1CZ1
Op1CX1 = (Op1CYBR * DSP1_Sin(Op1CZ) >> 15) + (Op1CXBR * DSP1_Cos(Op1CZ) >> 15);
@@ -1089,7 +1113,7 @@ void DSPOp1C()
uint16_t Op0FRamsize;
uint16_t Op0FPass;
-void DSPOp0F()
+void DSPOp0F(void)
{
Op0FPass = 0x0000;
}
@@ -1097,7 +1121,7 @@ void DSPOp0F()
int16_t Op2FUnknown;
int16_t Op2FSize;
-void DSPOp2F()
+void DSPOp2F(void)
{
Op2FSize = 0x100;
}
diff --git a/source/dsp2emu.c b/source/dsp2emu.c
index 9e3e137..4e22d54 100644
--- a/source/dsp2emu.c
+++ b/source/dsp2emu.c
@@ -8,7 +8,7 @@ bool DSP2Op06HasLen = false;
int32_t DSP2Op06Len = 0;
uint8_t DSP2Op05Transparent = 0;
-void DSP2_Op05()
+void DSP2_Op05(void)
{
uint8_t color;
// Overlay bitmap with transparency.
@@ -49,16 +49,14 @@ void DSP2_Op05()
{
c1 = *p1++;
c2 = *p2++;
- *p3++ = (((c2 >> 4) == color) ? c1 & 0xf0 : c2 & 0xf0) |
- (((c2 & 0x0f) == color) ? c1 & 0x0f : c2 & 0x0f);
+ *p3++ = (((c2 >> 4) == color) ? c1 & 0xf0 : c2 & 0xf0) | (((c2 & 0x0f) == color) ? c1 & 0x0f : c2 & 0x0f);
}
}
-void DSP2_Op01()
+void DSP2_Op01(void)
{
// Op01 size is always 32 bytes input and output.
// The hardware does strange things if you vary the size.
-
int32_t j;
uint8_t c0, c1, c2, c3;
uint8_t* p1 = DSP1.parameters;
@@ -66,55 +64,20 @@ void DSP2_Op01()
uint8_t* p2b = &DSP1.output[16]; // halfway
// Process 8 blocks of 4 bytes each
-
for (j = 0; j < 8; j++)
{
c0 = *p1++;
c1 = *p1++;
c2 = *p1++;
c3 = *p1++;
-
- *p2a++ = (c0 & 0x10) << 3 |
- (c0 & 0x01) << 6 |
- (c1 & 0x10) << 1 |
- (c1 & 0x01) << 4 |
- (c2 & 0x10) >> 1 |
- (c2 & 0x01) << 2 |
- (c3 & 0x10) >> 3 |
- (c3 & 0x01);
-
- *p2a++ = (c0 & 0x20) << 2 |
- (c0 & 0x02) << 5 |
- (c1 & 0x20) |
- (c1 & 0x02) << 3 |
- (c2 & 0x20) >> 2 |
- (c2 & 0x02) << 1 |
- (c3 & 0x20) >> 4 |
- (c3 & 0x02) >> 1;
-
- *p2b++ = (c0 & 0x40) << 1 |
- (c0 & 0x04) << 4 |
- (c1 & 0x40) >> 1 |
- (c1 & 0x04) << 2 |
- (c2 & 0x40) >> 3 |
- (c2 & 0x04) |
- (c3 & 0x40) >> 5 |
- (c3 & 0x04) >> 2;
-
-
- *p2b++ = (c0 & 0x80) |
- (c0 & 0x08) << 3 |
- (c1 & 0x80) >> 2 |
- (c1 & 0x08) << 1 |
- (c2 & 0x80) >> 4 |
- (c2 & 0x08) >> 1 |
- (c3 & 0x80) >> 6 |
- (c3 & 0x08) >> 3;
+ *p2a++ = (c0 & 0x10) << 3 | (c0 & 0x01) << 6 | (c1 & 0x10) << 1 | (c1 & 0x01) << 4 | (c2 & 0x10) >> 1 | (c2 & 0x01) << 2 | (c3 & 0x10) >> 3 | (c3 & 0x01);
+ *p2a++ = (c0 & 0x20) << 2 | (c0 & 0x02) << 5 | (c1 & 0x20) | (c1 & 0x02) << 3 | (c2 & 0x20) >> 2 | (c2 & 0x02) << 1 | (c3 & 0x20) >> 4 | (c3 & 0x02) >> 1;
+ *p2b++ = (c0 & 0x40) << 1 | (c0 & 0x04) << 4 | (c1 & 0x40) >> 1 | (c1 & 0x04) << 2 | (c2 & 0x40) >> 3 | (c2 & 0x04) | (c3 & 0x40) >> 5 | (c3 & 0x04) >> 2;
+ *p2b++ = (c0 & 0x80) | (c0 & 0x08) << 3 | (c1 & 0x80) >> 2 | (c1 & 0x08) << 1 | (c2 & 0x80) >> 4 | (c2 & 0x08) >> 1 | (c3 & 0x80) >> 6 | (c3 & 0x08) >> 3;
}
- return;
}
-void DSP2_Op06()
+void DSP2_Op06(void)
{
// Input:
// size
@@ -131,13 +94,10 @@ int32_t DSP2Op0DOutLen = 0;
int32_t DSP2Op0DInLen = 0;
// Scale bitmap based on input length out output length
-
-void DSP2_Op0D()
+void DSP2_Op0D(void)
{
// (Modified) Overload's algorithm
-
int32_t i;
-
for(i = 0 ; i < DSP2Op0DOutLen ; i++)
{
int32_t j = i << 1;
diff --git a/source/dsp4.h b/source/dsp4.h
index 760891c..53ff365 100644
--- a/source/dsp4.h
+++ b/source/dsp4.h
@@ -3,85 +3,80 @@
#ifndef _DSP4_H_
#define _DSP4_H_
-// debug
-int32_t block; // current block number
-extern int32_t c;
-
// op control
-int8_t DSP4_Logic; // controls op flow
+int8_t DSP4_Logic; // controls op flow
// projection format
-const int16_t PLANE_START = 0x7fff; // starting distance
+const int16_t PLANE_START = 0x7fff; // starting distance
-int16_t view_plane; // viewer location
-int16_t far_plane; // next milestone into screen
-int16_t segments; // # raster segments to draw
-int16_t raster; // current raster line
+int16_t view_plane; // viewer location
+int16_t far_plane; // next milestone into screen
+int16_t segments; // # raster segments to draw
+int16_t raster; // current raster line
-int16_t project_x; // current x-position
-int16_t project_y; // current y-position
+int16_t project_x; // current x-position
+int16_t project_y; // current y-position
-int16_t project_centerx; // x-target of projection
-int16_t project_centery; // y-target of projection
+int16_t project_centerx; // x-target of projection
+int16_t project_centery; // y-target of projection
-int16_t project_x1; // current x-distance
-int16_t project_x1low; // lower 16-bits
-int16_t project_y1; // current y-distance
-int16_t project_y1low; // lower 16-bits
+int16_t project_x1; // current x-distance
+int16_t project_x1low; // lower 16-bits
+int16_t project_y1; // current y-distance
+int16_t project_y1low; // lower 16-bits
-int16_t project_x2; // next projected x-distance
-int16_t project_y2; // next projected y-distance
+int16_t project_x2; // next projected x-distance
+int16_t project_y2; // next projected y-distance
-int16_t project_pitchx; // delta center
-int16_t project_pitchxlow; // lower 16-bits
-int16_t project_pitchy; // delta center
-int16_t project_pitchylow; // lower 16-bits
+int16_t project_pitchx; // delta center
+int16_t project_pitchxlow; // lower 16-bits
+int16_t project_pitchy; // delta center
+int16_t project_pitchylow; // lower 16-bits
-int16_t project_focalx; // x-point of projection at viewer plane
-int16_t project_focaly; // y-point of projection at viewer plane
+int16_t project_focalx; // x-point of projection at viewer plane
+int16_t project_focaly; // y-point of projection at viewer plane
-int16_t project_ptr; // data structure pointer
+int16_t project_ptr; // data structure pointer
// render window
-int16_t center_x; // x-center of viewport
-int16_t center_y; // y-center of viewport
-int16_t viewport_left; // x-left of viewport
-int16_t viewport_right; // x-right of viewport
-int16_t viewport_top; // y-top of viewport
-int16_t viewport_bottom; // y-bottom of viewport
+int16_t center_x; // x-center of viewport
+int16_t center_y; // y-center of viewport
+int16_t viewport_left; // x-left of viewport
+int16_t viewport_right; // x-right of viewport
+int16_t viewport_top; // y-top of viewport
+int16_t viewport_bottom; // y-bottom of viewport
// sprite structure
-int16_t sprite_x; // projected x-pos of sprite
-int16_t sprite_y; // projected y-pos of sprite
-int16_t sprite_offset; // data pointer offset
-int8_t sprite_type; // vehicle, terrain
-bool sprite_size; // sprite size: 8x8 or 16x16
+int16_t sprite_x; // projected x-pos of sprite
+int16_t sprite_y; // projected y-pos of sprite
+int16_t sprite_offset; // data pointer offset
+int8_t sprite_type; // vehicle, terrain
+bool sprite_size; // sprite size: 8x8 or 16x16
// path strips
-int16_t path_clipRight[4]; // value to clip to for x>b
-int16_t path_clipLeft[4]; // value to clip to for x<a
-int16_t path_pos[4]; // x-positions of lanes
-int16_t path_ptr[4]; // data structure pointers
-int16_t path_raster[4]; // current raster
-int16_t path_top[4]; // viewport_top
+int16_t path_clipRight[4]; // value to clip to for x>b
+int16_t path_clipLeft[4]; // value to clip to for x<a
+int16_t path_pos[4]; // x-positions of lanes
+int16_t path_ptr[4]; // data structure pointers
+int16_t path_raster[4]; // current raster
+int16_t path_top[4]; // viewport_top
-int16_t path_y[2]; // current y-position
-int16_t path_x[2]; // current focals
-int16_t path_plane[2]; // previous plane
+int16_t path_y[2]; // current y-position
+int16_t path_x[2]; // current focals
+int16_t path_plane[2]; // previous plane
// op09 window sorting
-int16_t multi_index1; // index counter
-int16_t multi_index2; // index counter
-bool op09_mode; // window mode
+int16_t multi_index1; // index counter
+int16_t multi_index2; // index counter
+bool op09_mode; // window mode
// multi-op storage
-int16_t multi_focaly[64]; // focal_y values
-int16_t multi_farplane[4]; // farthest drawn distance
-int16_t multi_raster[4]; // line where track stops
+int16_t multi_focaly[64]; // focal_y values
+int16_t multi_farplane[4]; // farthest drawn distance
+int16_t multi_raster[4]; // line where track stops
// OAM
-int8_t op06_OAM[32]; // OAM (size,MSB) data
-int8_t op06_index; // index into OAM table
-int8_t op06_offset; // offset into OAM table
-
+int8_t op06_OAM[32]; // OAM (size,MSB) data
+int8_t op06_index; // index into OAM table
+int8_t op06_offset; // offset into OAM table
#endif
diff --git a/source/dsp4emu.c b/source/dsp4emu.c
index b2fcebf..921f896 100644
--- a/source/dsp4emu.c
+++ b/source/dsp4emu.c
@@ -11,10 +11,8 @@
// used to wait for dsp i/o
#define DSP4_WAIT(x) \
- DSP4_Logic = x; return;
-
-//////////////////////////////////////////////////////////////
-//////////////////////////////////////////////////////////////
+ DSP4_Logic = x; \
+ return
int32_t DSP4_Multiply(int16_t Multiplicand, int16_t Multiplier)
{
@@ -23,13 +21,9 @@ int32_t DSP4_Multiply(int16_t Multiplicand, int16_t Multiplier)
int16_t DSP4_UnknownOP11(int16_t A, int16_t B, int16_t C, int16_t D)
{
- return ((A * 0x0155 >> 2) & 0xf000) | ((B * 0x0155 >> 6) & 0x0f00) |
- ((C * 0x0155 >> 10) & 0x00f0) | ((D * 0x0155 >> 14) & 0x000f);
+ return ((A * 0x0155 >> 2) & 0xf000) | ((B * 0x0155 >> 6) & 0x0f00) | ((C * 0x0155 >> 10) & 0x00f0) | ((D * 0x0155 >> 14) & 0x000f);
}
-//////////////////////////////////////////////////////////////
-//////////////////////////////////////////////////////////////
-
void DSP4_Op06(bool size, bool msb)
{
// save post-oam table data for future retrieval
@@ -45,19 +39,16 @@ void DSP4_Op06(bool size, bool msb)
}
}
-void DSP4_Op01()
+void DSP4_Op01(void)
{
int16_t plane;
-
int16_t index, lcv;
int16_t py_dy, px_dx;
int16_t y_out, x_out;
uint16_t command;
-
DSP4.waiting4command = false;
- // op flow control
- switch (DSP4_Logic)
+ switch (DSP4_Logic) // op flow control
{
case 1:
goto resume1;
@@ -99,9 +90,6 @@ void DSP4_Op01()
multi_index1 = 0;
multi_index2 = 0;
- // debug
- block = 0;
-
////////////////////////////////////////////////////
// command check
@@ -109,20 +97,20 @@ void DSP4_Op01()
{
// scan next command
DSP4.in_count = 2;
+ DSP4_WAIT(1);
-DSP4_WAIT(1) resume1:
-
+resume1:
// inspect input
command = DSP4_READ_WORD(0);
// check for termination
- if (command == 0x8000) break;
+ if(command == 0x8000)
+ break;
// already have 2 bytes in queue
DSP4.in_index = 2;
DSP4.in_count = 8;
-
- DSP4_WAIT(2)
+ DSP4_WAIT(2);
////////////////////////////////////////////////////
// process one iteration of projection
@@ -134,7 +122,8 @@ resume2:
px_dx = 0;
// ignore invalid data
- if ((uint16_t) plane == 0x8001) continue;
+ if((uint16_t) plane == 0x8001)
+ continue;
// one-time init
if (far_plane)
@@ -153,31 +142,26 @@ resume2:
// quadratic regression (rough)
if (project_focaly >= -0x0f)
- py_dy = (int16_t)(project_focaly * project_focaly * -0.20533553
- - 1.08330005 * project_focaly - 69.61094639);
+ py_dy = (int16_t)(project_focaly * project_focaly * -0.20533553 - 1.08330005 * project_focaly - 69.61094639);
else
- py_dy = (int16_t)(project_focaly * project_focaly * -0.000657035759
- - 1.07629051 * project_focaly - 65.69315963);
+ py_dy = (int16_t)(project_focaly * project_focaly * -0.000657035759 - 1.07629051 * project_focaly - 65.69315963);
// approximate # of raster lines
segments = ABS(project_y2 - project_y1);
// prevent overdraw
- if (project_y2 >= raster) segments = 0;
- else raster = project_y2;
+ if(project_y2 >= raster)
+ segments = 0;
+ else
+ raster = project_y2;
// don't draw outside the window
- if (project_y2 < viewport_top) segments = 0;
+ if(project_y2 < viewport_top)
+ segments = 0;
// project new positions
if (segments > 0)
- {
- // interpolate between projected points
- px_dx = ((project_x2 - project_x1) << 8) / segments;
- }
-
- // debug
- ++block;
+ px_dx = ((project_x2 - project_x1) << 8) / segments; // interpolate between projected points
// prepare output
DSP4.out_count = 8 + 2 + 6 * segments;
@@ -191,8 +175,7 @@ resume2:
index = 10;
- // iterate through each point
- for (lcv = 0; lcv < segments; lcv++)
+ for (lcv = 0; lcv < segments; lcv++) // iterate through each point
{
// step through the projected line
y_out = project_y + ((py_dy * lcv) >> 8);
@@ -212,8 +195,7 @@ resume2:
project_y += ((py_dy * lcv) >> 8);
project_x += ((px_dx * lcv) >> 8);
- // new positions
- if (segments > 0)
+ if (segments > 0) // new positions
{
project_x1 = project_x2;
project_y1 = project_y2;
@@ -230,15 +212,14 @@ resume2:
project_focaly += project_pitchy;
project_focalx += project_pitchx;
- }
- while (1);
+ } while (1);
// terminate op
DSP4.waiting4command = true;
DSP4.out_count = 0;
}
-void DSP4_Op07()
+void DSP4_Op07(void)
{
uint16_t command;
int16_t plane;
@@ -262,7 +243,6 @@ void DSP4_Op07()
////////////////////////////////////////////////////
// sort inputs
- // 0x00 = DSP4_READ_WORD(0x00);
project_focaly = DSP4_READ_WORD(0x02);
raster = DSP4_READ_WORD(0x04);
viewport_top = DSP4_READ_WORD(0x06);
@@ -272,7 +252,6 @@ void DSP4_Op07()
project_x1 = DSP4_READ_WORD(0x0e);
project_centerx = DSP4_READ_WORD(0x10);
project_ptr = DSP4_READ_WORD(0x12);
- // (envelope?) 0xc0 = DSP4_READ_WORD(0x14);
// pre-compute
view_plane = PLANE_START;
@@ -285,9 +264,6 @@ void DSP4_Op07()
// multi-op storage
multi_index2 = 0;
- // debug
- block = 0;
-
////////////////////////////////////////////////////
// command check
@@ -295,20 +271,20 @@ void DSP4_Op07()
{
// scan next command
DSP4.in_count = 2;
+ DSP4_WAIT(1);
-DSP4_WAIT(1) resume1:
-
+resume1:
// inspect input
command = DSP4_READ_WORD(0);
// check for opcode termination
- if (command == 0x8000) break;
+ if(command == 0x8000)
+ break;
// already have 2 bytes in queue
DSP4.in_index = 2;
DSP4.in_count = 12;
-
- DSP4_WAIT(2)
+ DSP4_WAIT(2);
////////////////////////////////////////////////////
// process one loop of projection
@@ -316,38 +292,36 @@ DSP4_WAIT(1) resume1:
resume2:
px_dx = 0;
- // debug
- ++block;
-
// inspect inputs
plane = DSP4_READ_WORD(0);
project_y2 = DSP4_READ_WORD(2);
- // ? = DSP4_READ_WORD(4);
project_x2 = DSP4_READ_WORD(6);
// ignore invalid data
- if ((uint16_t) plane == 0x8001) continue;
+ if((uint16_t) plane == 0x8001)
+ continue;
// multi-op storage
project_focaly = multi_focaly[multi_index2];
// quadratic regression (rough)
if (project_focaly >= -0x0f)
- py_dy = (int16_t)(project_focaly * project_focaly * -0.20533553
- - 1.08330005 * project_focaly - 69.61094639);
+ py_dy = (int16_t)(project_focaly * project_focaly * -0.20533553 - 1.08330005 * project_focaly - 69.61094639);
else
- py_dy = (int16_t)(project_focaly * project_focaly * -0.000657035759
- - 1.07629051 * project_focaly - 65.69315963);
+ py_dy = (int16_t)(project_focaly * project_focaly * -0.000657035759 - 1.07629051 * project_focaly - 65.69315963);
// approximate # of raster lines
segments = ABS(project_y2 - project_y1);
// prevent overdraw
- if (project_y2 >= raster) segments = 0;
- else raster = project_y2;
+ if(project_y2 >= raster)
+ segments = 0;
+ else
+ raster = project_y2;
// don't draw outside the window
- if (project_y2 < viewport_top) segments = 0;
+ if(project_y2 < viewport_top)
+ segments = 0;
// project new positions
if (segments > 0)
@@ -393,14 +367,13 @@ resume2:
// multi-op storage
multi_index2++;
}
- }
- while (1);
+ } while (1);
DSP4.waiting4command = true;
DSP4.out_count = 0;
}
-void DSP4_Op08()
+void DSP4_Op08(void)
{
uint16_t command;
// used in envelope shaping
@@ -408,7 +381,6 @@ void DSP4_Op08()
int16_t x2_final;
int16_t plane, x_left, y_left, x_right, y_right;
int16_t envelope1, envelope2;
-
DSP4.waiting4command = false;
// op flow control
@@ -436,9 +408,6 @@ void DSP4_Op08()
path_clipLeft[2] = DSP4_READ_WORD(0x0c);
path_clipLeft[3] = DSP4_READ_WORD(0x0e);
- // unknown (constant)
- // unknown (constant)
-
// path positions
path_pos[0] = DSP4_READ_WORD(0x20);
path_pos[1] = DSP4_READ_WORD(0x22);
@@ -467,9 +436,6 @@ void DSP4_Op08()
view_plane = PLANE_START;
- // debug
- block = 0;
-
////////////////////////////////////////////////////
// command check
@@ -477,27 +443,25 @@ void DSP4_Op08()
{
// scan next command
DSP4.in_count = 2;
+ DSP4_WAIT(1);
-DSP4_WAIT(1) resume1:
-
+resume1:
// inspect input
command = DSP4_READ_WORD(0);
// terminate op
- if (command == 0x8000) break;
+ if(command == 0x8000)
+ break;
// already have 2 bytes in queue
DSP4.in_index = 2;
DSP4.in_count = 18;
+ DSP4_WAIT(2);
-DSP4_WAIT(2) resume2:
-
+resume2:
////////////////////////////////////////////////////
// projection begins
- // debug
- ++block;
-
// look at guidelines
plane = DSP4_READ_WORD(0x00);
x_left = DSP4_READ_WORD(0x02);
@@ -510,7 +474,8 @@ DSP4_WAIT(2) resume2:
envelope2 = DSP4_READ_WORD(0x0c);
// ignore invalid data
- if ((uint16_t) plane == 0x8001) continue;
+ if((uint16_t) plane == 0x8001)
+ continue;
// first init
if (plane == 0x7fff)
@@ -534,10 +499,14 @@ DSP4_WAIT(2) resume2:
pos2 = path_pos[1] + envelope2;
// clip offscreen data
- if (pos1 < path_clipLeft[0]) pos1 = path_clipLeft[0];
- if (pos1 > path_clipRight[0]) pos1 = path_clipRight[0];
- if (pos2 < path_clipLeft[1]) pos2 = path_clipLeft[1];
- if (pos2 > path_clipRight[1]) pos2 = path_clipRight[1];
+ if(pos1 < path_clipLeft[0])
+ pos1 = path_clipLeft[0];
+ if(pos1 > path_clipRight[0])
+ pos1 = path_clipRight[0];
+ if(pos2 < path_clipLeft[1])
+ pos2 = path_clipLeft[1];
+ if(pos2 > path_clipRight[1])
+ pos2 = path_clipRight[1];
path_plane[0] = plane;
path_plane[1] = plane;
@@ -558,11 +527,14 @@ DSP4_WAIT(2) resume2:
segments = ABS(y_left - path_y[0]);
// prevent overdraw
- if (y_left >= path_raster[0]) segments = 0;
- else path_raster[0] = y_left;
+ if(y_left >= path_raster[0])
+ segments = 0;
+ else
+ path_raster[0] = y_left;
// don't draw outside the window
- if (path_raster[0] < path_top[0]) segments = 0;
+ if(path_raster[0] < path_top[0])
+ segments = 0;
// proceed if visibility rules apply
if (segments > 0)
@@ -588,7 +560,6 @@ DSP4_WAIT(2) resume2:
// interpolate between projected points with shaping
right_inc = ((x2_final - x1_final) << 8) / segments;
-
path_plane[0] = plane;
}
@@ -606,10 +577,14 @@ DSP4_WAIT(2) resume2:
pos2 = path_pos[1] + ((right_inc * lcv) >> 8) + dx2;
// clip offscreen data
- if (pos1 < path_clipLeft[0]) pos1 = path_clipLeft[0];
- if (pos1 > path_clipRight[0]) pos1 = path_clipRight[0];
- if (pos2 < path_clipLeft[1]) pos2 = path_clipLeft[1];
- if (pos2 > path_clipRight[1]) pos2 = path_clipRight[1];
+ if(pos1 < path_clipLeft[0])
+ pos1 = path_clipLeft[0];
+ if(pos1 > path_clipRight[0])
+ pos1 = path_clipRight[0];
+ if(pos2 < path_clipLeft[1])
+ pos2 = path_clipLeft[1];
+ if(pos2 > path_clipRight[1])
+ pos2 = path_clipRight[1];
// data
DSP4_WRITE_WORD(index, path_ptr[0]);
@@ -641,11 +616,13 @@ DSP4_WAIT(2) resume2:
segments = ABS(y_right - path_y[1]);
// prevent overdraw
- if (y_right >= path_raster[2]) segments = 0;
+ if(y_right >= path_raster[2])
+ segments = 0;
else path_raster[2] = y_right;
// don't draw outside the window
- if (path_raster[2] < path_top[2]) segments = 0;
+ if(path_raster[2] < path_top[2])
+ segments = 0;
// proceed if visibility rules apply
if (segments > 0)
@@ -689,10 +666,14 @@ DSP4_WAIT(2) resume2:
pos2 = path_pos[3] + ((right_inc * lcv) >> 8) + dx2;
// clip offscreen data
- if (pos1 < path_clipLeft[2]) pos1 = path_clipLeft[2];
- if (pos1 > path_clipRight[2]) pos1 = path_clipRight[2];
- if (pos2 < path_clipLeft[3]) pos2 = path_clipLeft[3];
- if (pos2 > path_clipRight[3]) pos2 = path_clipRight[3];
+ if(pos1 < path_clipLeft[2])
+ pos1 = path_clipLeft[2];
+ if(pos1 > path_clipRight[2])
+ pos1 = path_clipRight[2];
+ if(pos2 < path_clipLeft[3])
+ pos2 = path_clipLeft[3];
+ if(pos2 > path_clipRight[3])
+ pos2 = path_clipRight[3];
// data
DSP4_WRITE_WORD(index, path_ptr[2]);
@@ -719,15 +700,14 @@ DSP4_WAIT(2) resume2:
path_y[1] = y_right;
}
}
- }
- while (1);
+ } while (1);
DSP4.waiting4command = true;
DSP4.out_count = 2;
DSP4_WRITE_WORD(0, 0);
}
-void DSP4_Op0D()
+void DSP4_Op0D(void)
{
uint16_t command;
// inspect inputs
@@ -801,9 +781,6 @@ void DSP4_Op0D()
project_y -= viewport_bottom;
project_x = project_centerx + project_x1;
- // debug
- block = 0;
-
////////////////////////////////////////////////////
// command check
@@ -811,39 +788,37 @@ void DSP4_Op0D()
{
// scan next command
DSP4.in_count = 2;
+ DSP4_WAIT(1);
-DSP4_WAIT(1) resume1:
-
+resume1:
// inspect input
command = DSP4_READ_WORD(0);
// terminate op
- if (command == 0x8000) break;
+ if(command == 0x8000)
+ break;
// already have 2 bytes in queue
DSP4.in_index = 2;
DSP4.in_count = 8;
-
- DSP4_WAIT(2)
+ DSP4_WAIT(2);
////////////////////////////////////////////////////
// project section of the track
-
resume2:
-
plane = DSP4_READ_WORD(0);
px_dx = 0;
// ignore invalid data
- if ((uint16_t) plane == 0x8001) continue;
+ if((uint16_t) plane == 0x8001)
+ continue;
// one-time init
if (far_plane)
{
// setup final data
- // low16=plane
project_x1 = project_focalx;
project_y1 = project_focaly;
plane = far_plane;
@@ -856,21 +831,22 @@ resume2:
// quadratic regression (rough)
if (project_focaly >= -0x0f)
- py_dy = (int16_t)(project_focaly * project_focaly * -0.20533553
- - 1.08330005 * project_focaly - 69.61094639);
+ py_dy = (int16_t)(project_focaly * project_focaly * -0.20533553 - 1.08330005 * project_focaly - 69.61094639);
else
- py_dy = (int16_t)(project_focaly * project_focaly * -0.000657035759
- - 1.07629051 * project_focaly - 65.69315963);
+ py_dy = (int16_t)(project_focaly * project_focaly * -0.000657035759 - 1.07629051 * project_focaly - 65.69315963);
// approximate # of raster lines
segments = ABS(project_y2 - project_y1);
// prevent overdraw
- if (project_y2 >= raster) segments = 0;
- else raster = project_y2;
+ if(project_y2 >= raster)
+ segments = 0;
+ else
+ raster = project_y2;
// don't draw outside the window
- if (project_y2 < viewport_top) segments = 0;
+ if(project_y2 < viewport_top)
+ segments = 0;
// project new positions
if (segments > 0)
@@ -879,22 +855,16 @@ resume2:
px_dx = ((project_x2 - project_x1) << 8) / segments;
}
- // debug
- ++block;
-
// prepare output
DSP4.out_count = 8 + 2 + 6 * segments;
-
DSP4_WRITE_WORD(0, project_focalx);
DSP4_WRITE_WORD(2, project_x2);
DSP4_WRITE_WORD(4, project_focaly);
DSP4_WRITE_WORD(6, project_y2);
DSP4_WRITE_WORD(8, segments);
-
index = 10;
- // iterate through each point
- for (lcv = 0; lcv < segments; lcv++)
+ for (lcv = 0; lcv < segments; lcv++) // iterate through each point
{
// step through the projected line
y_out = project_y + ((py_dy * lcv) >> 8);
@@ -930,14 +900,13 @@ resume2:
project_focaly += project_pitchy;
project_focalx += project_pitchx;
- }
- while (1);
+ } while (1);
DSP4.waiting4command = true;
DSP4.out_count = 0;
}
-void DSP4_Op09()
+void DSP4_Op09(void)
{
uint16_t command;
bool clip;
@@ -975,9 +944,6 @@ void DSP4_Op09()
////////////////////////////////////////////////////
// process initial inputs
- // debug
- block = 0;
-
// grab screen information
view_plane = PLANE_START;
center_x = DSP4_READ_WORD(0x00);
@@ -995,8 +961,7 @@ void DSP4_Op09()
multi_index1 %= 4;
// convert track line to the window region
- project_y2 = center_y + multi_raster[multi_index1] *
- (viewport_bottom - center_y) / (0x33 - 0);
+ project_y2 = center_y + multi_raster[multi_index1] * (viewport_bottom - center_y) / (0x33 - 0);
if (!op09_mode)
project_y2 -= 2;
@@ -1006,21 +971,21 @@ void DSP4_Op09()
{
////////////////////////////////////////////////////
// check for new sprites
-
do
{
uint16_t second;
DSP4.in_count = 4;
DSP4.in_index = 2;
+ DSP4_WAIT(1);
-DSP4_WAIT(1) resume1:
-
+resume1:
// try to classify sprite
second = DSP4_READ_WORD(2);
// op termination
- if (second == 0x8000) goto terminate;
+ if(second == 0x8000)
+ goto terminate;
second >>= 8;
sprite_type = 0;
@@ -1040,13 +1005,11 @@ DSP4_WAIT(1) resume1:
no_sprite:
// no sprite. try again
-
DSP4.in_count = 2;
+ DSP4_WAIT(2);
-DSP4_WAIT(2) resume2:
- ;
- }
- while (1);
+resume2:;
+ } while (1);
////////////////////////////////////////////////////
// process projection information
@@ -1063,9 +1026,9 @@ sprite_found:
// we already have 4 bytes we want
DSP4.in_count = 6 + 12;
DSP4.in_index = 4;
+ DSP4_WAIT(3);
-DSP4_WAIT(3) resume3:
-
+resume3:
// filter inputs
project_y1 = DSP4_READ_WORD(0x00);
focal_back = DSP4_READ_WORD(0x06);
@@ -1087,9 +1050,6 @@ DSP4_WAIT(3) resume3:
sprite_y = viewport_bottom - segments;
far_plane = plane;
- // debug
- ++block;
-
// make the car's x-center available
DSP4.out_count = 2;
DSP4_WRITE_WORD(0, project_focalx);
@@ -1097,28 +1057,25 @@ DSP4_WAIT(3) resume3:
// grab a few remaining vehicle values
DSP4.in_count = 4;
- DSP4_WAIT(4)
+ DSP4_WAIT(4);
- // store final values
-
-resume4:
+resume4: // store final values
height = DSP4_READ_WORD(0);
sprite_offset = DSP4_READ_WORD(2);
// vertical lift factor
sprite_y += height;
}
- // terrain sprite
- else if (sprite_type == 2)
+ else if (sprite_type == 2) // terrain sprite
{
int16_t plane;
// we already have 4 bytes we want
DSP4.in_count = 6 + 6 + 2;
DSP4.in_index = 4;
+ DSP4_WAIT(5);
-DSP4_WAIT(5) resume5:
-
+resume5:
// sort loop inputs
project_y1 = DSP4_READ_WORD(0x00);
plane = DSP4_READ_WORD(0x02);
@@ -1136,9 +1093,6 @@ DSP4_WAIT(5) resume5:
sprite_x = center_x + project_x - project_centerx;
sprite_y = viewport_bottom - segments + project_y;
far_plane = plane;
-
- // debug
- ++block;
}
// default sprite size: 16x16
@@ -1149,13 +1103,14 @@ DSP4_WAIT(5) resume5:
do
{
DSP4.in_count = 2;
+ DSP4_WAIT(6);
-DSP4_WAIT(6) resume6:
-
+resume6:
command = DSP4_READ_WORD(0);
// opcode termination
- if (command == 0x8000) goto terminate;
+ if(command == 0x8000)
+ goto terminate;
// toggle sprite size
if (command == 0x0000)
@@ -1166,21 +1121,17 @@ DSP4_WAIT(6) resume6:
// new sprite information
command >>= 8;
- if (command != 0x20 && command != 0x40 &&
- command != 0x60 && command != 0xa0 &&
- command != 0xc0 && command != 0xe0)
+ if (command != 0x20 && command != 0x40 && command != 0x60 && command != 0xa0 && command != 0xc0 && command != 0xe0)
break;
DSP4.in_count = 6;
DSP4.in_index = 2;
-
- DSP4_WAIT(7)
+ DSP4_WAIT(7);
/////////////////////////////////////
// process tile data
resume7:
-
// sprite deltas
sp_dy = DSP4_READ_WORD(2);
sp_dx = DSP4_READ_WORD(4);
@@ -1191,12 +1142,14 @@ resume7:
// reject points outside the clipping window
clip = false;
- if (sp_x < viewport_left || sp_x > viewport_right) clip = true;
- if (sp_y < viewport_top || sp_y > viewport_bottom) clip = true;
+ if(sp_x < viewport_left || sp_x > viewport_right)
+ clip = true;
+ if(sp_y < viewport_top || sp_y > viewport_bottom)
+ clip = true;
// track depth sorting
- if (far_plane <= multi_farplane[multi_index1] &&
- sp_y >= project_y2) clip = true;
+ if(far_plane <= multi_farplane[multi_index1] && sp_y >= project_y2)
+ clip = true;
// don't draw offscreen coordinates
DSP4.out_count = 0;
@@ -1210,11 +1163,7 @@ resume7:
sp_msb = (sp_x < 0 || sp_x > 255);
// emit transparency information
- if (
- (sprite_offset & 0x08) &&
- ((sprite_type == 1 && sp_y >= 0xcc) ||
- (sprite_type == 2 && sp_y >= 0xbb))
- )
+ if((sprite_offset & 0x08) && ((sprite_type == 1 && sp_y >= 0xcc) || (sprite_type == 2 && sp_y >= 0xbb)))
{
DSP4.out_count = 6;
@@ -1225,7 +1174,6 @@ resume7:
DSP4.output[2] = sp_x & 0xFF;
DSP4.output[3] = (sp_y + 6) & 0xFF;
DSP4_WRITE_WORD(4, 0xEE);
-
out_index = 6;
// OAM: size,msb data
@@ -1256,8 +1204,7 @@ resume7:
DSP4.out_count = 2;
DSP4_WRITE_WORD(0, 0);
}
- }
- while (1);
+ } while (1);
/////////////////////////////////////
// special cases: plane == 0x0000
@@ -1275,8 +1222,7 @@ resume7:
goto sprite_found;
}
- // special terrain case
- else if (command != 0x00 && command != 0xff)
+ else if (command != 0x00 && command != 0xff) // special terrain case
{
sprite_type = 2;
@@ -1288,8 +1234,7 @@ resume7:
goto sprite_found;
}
- }
- while (1);
+ } while (1);
terminate:
DSP4.waiting4command = true;
diff --git a/source/fxemu.c b/source/fxemu.c
index 4bdc0b0..1a6ca7b 100644
--- a/source/fxemu.c
+++ b/source/fxemu.c
@@ -9,26 +9,17 @@
/* The FxChip Emulator's internal variables */
FxRegs_s GSU; /* This will be initialized when loading a ROM */
-void FxCacheWriteAccess(uint16_t vAddress)
-{
- if ((vAddress & 0x00f) == 0x00f)
- GSU.vCacheFlags |= 1 << ((vAddress & 0x1f0) >> 4);
-}
-
void FxFlushCache(void)
{
- GSU.vCacheFlags = 0;
GSU.vCacheBaseReg = 0;
GSU.bCacheActive = false;
}
void fx_flushCache(void)
{
- GSU.vCacheFlags = 0;
GSU.bCacheActive = false;
}
-
void fx_updateRamBank(uint8_t Byte)
{
// Update BankReg and Bank pointer
@@ -52,8 +43,6 @@ static void fx_readRegisterSpaceForUse(void)
int32_t i;
uint8_t* p = GSU.pvRegisters;
- GSU.vErrorCode = 0;
-
/* Update R0 - R14 */
for (i = 0; i < 15; i++)
{
@@ -90,10 +79,7 @@ static void fx_readRegisterSpaceForUse(void)
else
GSU.vScreenSize = (GSU.vScreenHeight / 8) * (256 / 8) * avMult[GSU.vMode];
if (GSU.vPlotOptionReg & 0x10)
- {
- /* OBJ Mode (for drawing into sprites) */
- GSU.vScreenHeight = 256;
- }
+ GSU.vScreenHeight = 256; /* OBJ Mode (for drawing into sprites) */
if (GSU.pvScreenBase + GSU.vScreenSize > GSU.pvRam + (GSU.nRamBanks * 65536))
GSU.pvScreenBase = GSU.pvRam + (GSU.nRamBanks * 65536) - GSU.vScreenSize;
GSU.pfPlot = fx_apfPlotTable[GSU.vMode];
@@ -117,7 +103,6 @@ void fx_computeScreenPointers(void)
if (GSU.vMode != GSU.vPrevMode || GSU.vPrevScreenHeight != GSU.vScreenHeight || GSU.vSCBRDirty)
{
int32_t i;
-
GSU.vSCBRDirty = false;
/* Make a list of pointers to the start of each screen column */
@@ -285,7 +270,6 @@ static void fx_writeRegisterSpaceAfterUse(void)
void FxReset(FxInit_s* psFxInfo)
{
int32_t i;
-
/* Clear all internal variables */
memset(&GSU, 0, sizeof(FxRegs_s));
@@ -341,9 +325,6 @@ void FxReset(FxInit_s* psFxInfo)
/* Start with a nop in the pipe */
GSU.vPipe = 0x01;
- /* Set pointer to GSU cache */
- GSU.pvCache = &GSU.pvRegisters[0x100];
-
fx_readRegisterSpaceForCheck();
fx_readRegisterSpaceForUse();
}
@@ -399,45 +380,5 @@ int32_t FxEmulate(uint32_t nInstructions)
fx_writeRegisterSpaceAfterUse();
/* Check for error code */
- if (GSU.vErrorCode)
- return GSU.vErrorCode;
- else
- return vCount;
-}
-
-/* Errors */
-int32_t FxGetErrorCode(void)
-{
- return GSU.vErrorCode;
-}
-
-int32_t FxGetIllegalAddress(void)
-{
- return GSU.vIllegalAddress;
-}
-
-/* Access to internal registers */
-uint32_t FxGetColorRegister(void)
-{
- return GSU.vColorReg & 0xff;
-}
-
-uint32_t FxGetPlotOptionRegister(void)
-{
- return GSU.vPlotOptionReg & 0x1f;
-}
-
-uint32_t FxGetSourceRegisterIndex(void)
-{
- return GSU.pvSreg - GSU.avReg;
-}
-
-uint32_t FxGetDestinationRegisterIndex(void)
-{
- return GSU.pvDreg - GSU.avReg;
-}
-
-uint8_t FxPipe(void)
-{
- return GSU.vPipe;
+ return vCount;
}
diff --git a/source/fxemu.h b/source/fxemu.h
index 3161ebd..66e64e4 100644
--- a/source/fxemu.h
+++ b/source/fxemu.h
@@ -22,39 +22,13 @@ extern void FxReset(FxInit_s* psFxInfo);
extern int32_t FxEmulate(uint32_t nInstructions);
/* Write access to the cache */
-extern void FxCacheWriteAccess(uint16_t vAddress);
extern void FxFlushCache(void); /* Called when the G flag in SFR is set to zero */
-/* Errors */
-extern int32_t FxGetErrorCode(void);
-extern int32_t FxGetIllegalAddress(void);
-
-/* Access to internal registers */
-extern uint32_t FxGetColorRegister(void);
-extern uint32_t FxGetPlotOptionRegister(void);
-extern uint32_t FxGetSourceRegisterIndex(void);
-extern uint32_t FxGetDestinationRegisterIndex(void);
-
-/* Get the byte currently in the pipe */
-extern uint8_t FxPipe(void);
-
/* SCBR write seen. We need to update our cached screen pointers */
extern void fx_dirtySCBR(void);
/* Update RamBankReg and RAM Bank pointer */
extern void fx_updateRamBank(uint8_t Byte);
-/* Option flags */
-#define FX_FLAG_ADDRESS_CHECKING 0x01
-#define FX_FLAG_ROM_BUFFER 0x02
-
-/* Return codes from FxEmulate(), FxStepInto() or FxStepOver() */
-#define FX_BREAKPOINT -1
-#define FX_ERROR_ILLEGAL_ADDRESS -2
-
-/* Return the number of bytes in an opcode */
-#define OPCODE_BYTES(op) ((((op) >= 0x05 && (op) <= 0xf) || ((op) >= 0xa0 && (op) <= 0xaf)) ? 2 : (((op) >= 0xf0) ? 3 : 1))
-
extern void fx_computeScreenPointers(void);
-
#endif
diff --git a/source/fxinst.c b/source/fxinst.c
index 1023e82..d48631d 100644
--- a/source/fxinst.c
+++ b/source/fxinst.c
@@ -1,7 +1,5 @@
#include "../copyright"
-#define FX_DO_ROMBUFFER
-
#include "fxemu.h"
#include "fxinst.h"
#include <string.h>
@@ -10,7 +8,6 @@
#include <retro_inline.h>
extern FxRegs_s GSU;
-int32_t gsu_bank [512] = {0};
/* Codes used:
*
@@ -20,7 +17,6 @@ int32_t gsu_bank [512] = {0};
* (yy) = 8 bit word address (0x0000 - 0x01fe)
* #xx = 16 bit immediate value
* (xx) = 16 bit address (0x0000 - 0xffff)
- *
*/
/* 00 - stop - stop GSU execution (and maybe generate an IRQ) */
@@ -2967,12 +2963,7 @@ static INLINE void fx_inc_r14(void)
/* df - getc - transfer ROM buffer to color register */
static INLINE void fx_getc(void)
{
-#ifndef FX_DO_ROMBUFFER
- uint8_t c;
- c = ROM(R14);
-#else
uint8_t c = GSU.vRomBuffer;
-#endif
if (GSU.vPlotOptionReg & 0x04)
c = (c & 0xf0) | (c >> 4);
if (GSU.vPlotOptionReg & 0x08)
@@ -3077,12 +3068,7 @@ static INLINE void fx_dec_r14(void)
/* ef - getb - get byte from ROM at address R14 */
static INLINE void fx_getb(void)
{
- uint32_t v;
-#ifndef FX_DO_ROMBUFFER
- v = (uint32_t)ROM(R14);
-#else
- v = (uint32_t)GSU.vRomBuffer;
-#endif
+ uint32_t v = (uint32_t)GSU.vRomBuffer;
R15++;
DREG = v;
TESTR14;
@@ -3092,14 +3078,8 @@ static INLINE void fx_getb(void)
/* ef(ALT1) - getbh - get high-byte from ROM at address R14 */
static INLINE void fx_getbh(void)
{
- uint32_t v;
-#ifndef FX_DO_ROMBUFFER
- uint32_t c;
- c = (uint32_t)ROM(R14);
-#else
uint32_t c = USEX8(GSU.vRomBuffer);
-#endif
- v = USEX8(SREG) | (c << 8);
+ uint32_t v = USEX8(SREG) | (c << 8);
R15++;
DREG = v;
TESTR14;
@@ -3109,13 +3089,8 @@ static INLINE void fx_getbh(void)
/* ef(ALT2) - getbl - get low-byte from ROM at address R14 */
static INLINE void fx_getbl(void)
{
- uint32_t v;
-#ifndef FX_DO_ROMBUFFER
- uint32_t c = (uint32_t)ROM(R14);
-#else
uint32_t c = USEX8(GSU.vRomBuffer);
-#endif
- v = (SREG & 0xff00) | c;
+ uint32_t v = (SREG & 0xff00) | c;
R15++;
DREG = v;
TESTR14;
@@ -3125,14 +3100,7 @@ static INLINE void fx_getbl(void)
/* ef(ALT3) - getbs - get sign extended byte from ROM at address R14 */
static INLINE void fx_getbs(void)
{
- uint32_t v;
-#ifndef FX_DO_ROMBUFFER
- int8_t c;
- c = ROM(R14);
- v = SEX8(c);
-#else
- v = SEX8(GSU.vRomBuffer);
-#endif
+ uint32_t v = SEX8(GSU.vRomBuffer);
R15++;
DREG = v;
TESTR14;
@@ -3384,7 +3352,7 @@ uint32_t fx_run(uint32_t nInstructions)
READR14;
while (TF(G) && (GSU.vCounter-- > 0))
FX_STEP;
- return (nInstructions - GSU.vInstCount);
+ return nInstructions - GSU.vInstCount;
}
/*** Special table for the different plot configurations ***/
@@ -3448,11 +3416,9 @@ void (*fx_apfOpcodeTable[])(void) =
/* f0 - ff */
&fx_iwt_r0, &fx_iwt_r1, &fx_iwt_r2, &fx_iwt_r3, &fx_iwt_r4, &fx_iwt_r5, &fx_iwt_r6, &fx_iwt_r7,
&fx_iwt_r8, &fx_iwt_r9, &fx_iwt_r10, &fx_iwt_r11, &fx_iwt_r12, &fx_iwt_r13, &fx_iwt_r14, &fx_iwt_r15,
-
/*
* ALT1 Table
*/
-
/* 00 - 0f */
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
@@ -3501,11 +3467,9 @@ void (*fx_apfOpcodeTable[])(void) =
/* f0 - ff */
&fx_lm_r0, &fx_lm_r1, &fx_lm_r2, &fx_lm_r3, &fx_lm_r4, &fx_lm_r5, &fx_lm_r6, &fx_lm_r7,
&fx_lm_r8, &fx_lm_r9, &fx_lm_r10, &fx_lm_r11, &fx_lm_r12, &fx_lm_r13, &fx_lm_r14, &fx_lm_r15,
-
/*
* ALT2 Table
*/
-
/* 00 - 0f */
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
@@ -3554,11 +3518,9 @@ void (*fx_apfOpcodeTable[])(void) =
/* f0 - ff */
&fx_sm_r0, &fx_sm_r1, &fx_sm_r2, &fx_sm_r3, &fx_sm_r4, &fx_sm_r5, &fx_sm_r6, &fx_sm_r7,
&fx_sm_r8, &fx_sm_r9, &fx_sm_r10, &fx_sm_r11, &fx_sm_r12, &fx_sm_r13, &fx_sm_r14, &fx_sm_r15,
-
/*
* ALT3 Table
*/
-
/* 00 - 0f */
&fx_stop, &fx_nop, &fx_cache, &fx_lsr, &fx_rol, &fx_bra, &fx_bge, &fx_blt,
&fx_bne, &fx_beq, &fx_bpl, &fx_bmi, &fx_bcc, &fx_bcs, &fx_bvc, &fx_bvs,
diff --git a/source/fxinst.h b/source/fxinst.h
index 68555b2..e1acd0d 100644
--- a/source/fxinst.h
+++ b/source/fxinst.h
@@ -135,13 +135,11 @@ typedef struct
uint32_t vRomBankReg; /* Rom bank index register */
uint32_t vRamBankReg; /* Ram bank index register */
uint32_t vCacheBaseReg; /* Cache base address register */
- uint32_t vCacheFlags; /* Saying what parts of the cache was written to */
uint32_t vLastRamAdr; /* Last RAM address accessed */
uint32_t* pvDreg; /* Pointer to current destination register */
uint32_t* pvSreg; /* Pointer to current source register */
uint8_t vRomBuffer; /* Current byte read by R14 */
uint8_t vPipe; /* Instructionset pipe */
- uint32_t vPipeAdr; /* The address of where the pipe was read from */
/* status register optimization stuff */
uint32_t vSign; /* v & 0x8000 */
@@ -150,11 +148,6 @@ typedef struct
int32_t vOverflow; /* (v >= 0x8000 || v < -0x8000) */
/* Other emulator variables */
- int32_t vErrorCode;
- uint32_t vIllegalAddress;
- uint8_t bBreakPoint;
- uint32_t vBreakPoint;
- uint32_t vStepPoint;
uint8_t* pvRegisters; /* 768 bytes located in the memory at address 0x3000 */
uint32_t nRamBanks; /* Number of 64kb-banks in FxRam (Don't confuse it with SNES-Ram!!!) */
uint8_t* pvRam; /* Pointer to FxRam */
@@ -177,30 +170,12 @@ typedef struct
uint8_t* apvRamBank[FX_RAM_BANKS]; /* Ram bank table (max 256kb) */
uint8_t* apvRomBank[256]; /* Rom bank table */
uint8_t bCacheActive;
- uint8_t* pvCache; /* Pointer to the GSU cache */
- uint8_t avCacheBackup[512]; /* Backup of ROM when the cache has replaced it */
uint32_t vCounter;
uint32_t vInstCount;
uint32_t vSCBRDirty; /* if SCBR is written, our cached screen pointers need updating */
} FxRegs_s;
/* GSU registers */
-#define GSU_R0 0x000
-#define GSU_R1 0x002
-#define GSU_R2 0x004
-#define GSU_R3 0x006
-#define GSU_R4 0x008
-#define GSU_R5 0x00a
-#define GSU_R6 0x00c
-#define GSU_R7 0x00e
-#define GSU_R8 0x010
-#define GSU_R9 0x012
-#define GSU_R10 0x014
-#define GSU_R11 0x016
-#define GSU_R12 0x018
-#define GSU_R13 0x01a
-#define GSU_R14 0x01c
-#define GSU_R15 0x01e
#define GSU_SFR 0x030
#define GSU_BRAMR 0x033
#define GSU_PBR 0x034
@@ -212,7 +187,6 @@ typedef struct
#define GSU_VCR 0x03b
#define GSU_RAMBR 0x03c
#define GSU_CBR 0x03e
-#define GSU_CACHERAM 0x100
/* SFR flags */
#define FLG_Z (1 << 1)
@@ -234,7 +208,7 @@ typedef struct
#define SF(a) (GSU.vStatusReg |= FLG_##a )
/* Test and set flag if condition, clear if not */
-#define TS(a,b) GSU.vStatusReg = ((GSU.vStatusReg & (~FLG_##a)) | ((!!(##b)) * FLG_##a ))
+#define TS(a, b) (GSU.vStatusReg = ((GSU.vStatusReg & (~FLG_##a)) | ((!!(##b)) * FLG_##a)))
/* Testing ALT1 & ALT2 bits */
#define ALT0 (!TF(ALT1) && !TF(ALT2))
@@ -263,7 +237,7 @@ typedef struct
GSU.pvDreg = GSU.pvSreg = &R0;
/* Read current RAM-Bank */
-#define RAM(adr) GSU.pvRamBank[USEX16(adr)]
+#define RAM(adr) (GSU.pvRamBank[USEX16(adr)])
/* Read current ROM-Bank */
#define ROM(idx) (GSU.pvRomBank[USEX16(idx)])
@@ -283,23 +257,14 @@ typedef struct
/* Access destination register */
#define DREG (*GSU.pvDreg)
-#ifndef FX_DO_ROMBUFFER
-
-/* Don't read R14 */
-#define READR14
-
-/* Don't test and/or read R14 */
-#define TESTR14
-
-#else
-
/* Read R14 */
-#define READR14 GSU.vRomBuffer = ROM(R14)
+#define READR14 \
+ GSU.vRomBuffer = ROM(R14)
/* Test and/or read R14 */
-#define TESTR14 if(GSU.pvDreg == &R14) READR14
-
-#endif
+#define TESTR14 \
+ if(GSU.pvDreg == &R14) \
+ READR14
/* Access to registers */
#define R0 GSU.avReg[0]
@@ -337,16 +302,11 @@ typedef struct
{ \
uint32_t vOpcode = (uint32_t) PIPE; \
FETCHPIPE; \
- (*fx_apfOpcodeTable[ (GSU.vStatusReg & 0x300) | vOpcode ])(); \
+ (*fx_apfOpcodeTable[(GSU.vStatusReg & 0x300) | vOpcode])(); \
}
extern void (*fx_apfOpcodeTable[])(void);
extern void (*fx_apfPlotTable[])(void);
uint32_t fx_run(uint32_t nInstructions);
-
-/* Set this define if branches are relative to the instruction in the delay slot */
-/* (I think they are) */
-#define BRANCH_DELAY_RELATIVE
-
#endif
diff --git a/source/getset.h b/source/getset.h
index ebf996f..c7f4950 100644
--- a/source/getset.h
+++ b/source/getset.h
@@ -25,29 +25,29 @@ inline uint8_t S9xGetByte(uint32_t Address)
{
if (Memory.BlockIsRAM [block])
CPU.WaitAddress = CPU.PCAtOpcodeStart;
- return (*(GetAddress + (Address & 0xffff)));
+ return GetAddress[Address & 0xffff];
}
switch ((intptr_t) GetAddress)
{
case MAP_PPU:
- return (S9xGetPPU(Address & 0xffff));
+ return S9xGetPPU(Address & 0xffff);
case MAP_CPU:
- return (S9xGetCPU(Address & 0xffff));
+ return S9xGetCPU(Address & 0xffff);
case MAP_DSP:
- return (S9xGetDSP(Address & 0xffff));
+ return S9xGetDSP(Address & 0xffff);
case MAP_SA1RAM:
case MAP_LOROM_SRAM:
- //Address &0x7FFF -offset into bank
- //Address&0xFF0000 -bank
- //bank>>1 | offset = s-ram address, unbound
+ //Address & 0x7FFF - offset into bank
+ //Address & 0xFF0000 - bank
+ //bank >> 1 | offset = s-ram address, unbound
//unbound & SRAMMask = Sram offset
- return (*(Memory.SRAM + ((((Address & 0xFF0000) >> 1) | (Address & 0x7FFF)) &Memory.SRAMMask)));
+ return Memory.SRAM[(((Address & 0xFF0000) >> 1) | (Address & 0x7FFF)) &Memory.SRAMMask];
case MAP_RONLY_SRAM:
case MAP_HIROM_SRAM:
- return (*(Memory.SRAM + (((Address & 0x7fff) - 0x6000 + ((Address & 0xf0000) >> 3)) & Memory.SRAMMask)));
+ return Memory.SRAM[((Address & 0x7fff) - 0x6000 + ((Address & 0xf0000) >> 3)) & Memory.SRAMMask];
case MAP_BWRAM:
- return (*(Memory.BWRAM + ((Address & 0x7fff) - 0x6000)));
+ return Memory.BWRAM[(Address & 0x7fff) - 0x6000];
case MAP_C4:
return S9xGetC4(Address & 0xffff);
case MAP_SPC7110_ROM:
@@ -70,7 +70,7 @@ inline uint16_t S9xGetWord(uint32_t Address)
if ((Address & 0x0fff) == 0x0fff)
{
OpenBus = S9xGetByte(Address);
- return (OpenBus | (S9xGetByte(Address + 1) << 8));
+ return OpenBus | (S9xGetByte(Address + 1) << 8);
}
int32_t block;
uint8_t* GetAddress = Memory.Map [block = (Address >> MEMMAP_SHIFT) & MEMMAP_MASK];
@@ -83,53 +83,46 @@ inline uint16_t S9xGetWord(uint32_t Address)
if (Memory.BlockIsRAM [block])
CPU.WaitAddress = CPU.PCAtOpcodeStart;
#ifdef FAST_LSB_WORD_ACCESS
- return (*(uint16_t*)(GetAddress + (Address & 0xffff)));
+ return *(uint16_t*) (GetAddress + (Address & 0xffff));
#else
- return (*(GetAddress + (Address & 0xffff)) | (*(GetAddress + (Address & 0xffff) + 1) << 8));
+ return *(GetAddress + (Address & 0xffff)) | (*(GetAddress + (Address & 0xffff) + 1) << 8);
#endif
}
switch ((intptr_t) GetAddress)
{
case MAP_PPU:
- return (S9xGetPPU(Address & 0xffff) | (S9xGetPPU((Address + 1) & 0xffff) << 8));
+ return S9xGetPPU(Address & 0xffff) | (S9xGetPPU((Address + 1) & 0xffff) << 8);
case MAP_CPU:
- return (S9xGetCPU(Address & 0xffff) | (S9xGetCPU((Address + 1) & 0xffff) << 8));
+ return S9xGetCPU(Address & 0xffff) | (S9xGetCPU((Address + 1) & 0xffff) << 8);
case MAP_DSP:
- return (S9xGetDSP(Address & 0xffff) | (S9xGetDSP((Address + 1) & 0xffff) << 8));
+ return S9xGetDSP(Address & 0xffff) | (S9xGetDSP((Address + 1) & 0xffff) << 8);
case MAP_SA1RAM:
case MAP_LOROM_SRAM:
- //Address &0x7FFF -offset into bank
- //Address&0xFF0000 -bank
- //bank>>1 | offset = s-ram address, unbound
+ //Address & 0x7FFF - offset into bank
+ //Address & 0xFF0000 - bank
+ //bank >> 1 | offset = s-ram address, unbound
//unbound & SRAMMask = Sram offset
/* BJ: no FAST_LSB_WORD_ACCESS here, since if Memory.SRAMMask=0x7ff
* then the high byte doesn't follow the low byte. */
- return
- (*(Memory.SRAM + ((((Address & 0xFF0000) >> 1) | (Address & 0x7FFF)) & Memory.SRAMMask))) |
- ((*(Memory.SRAM + (((((Address + 1) & 0xFF0000) >> 1) | ((Address + 1) & 0x7FFF)) & Memory.SRAMMask))) << 8);
+ return *(Memory.SRAM + ((((Address & 0xFF0000) >> 1) | (Address & 0x7FFF)) & Memory.SRAMMask)) | ((*(Memory.SRAM + (((((Address + 1) & 0xFF0000) >> 1) | ((Address + 1) & 0x7FFF)) & Memory.SRAMMask))) << 8);
case MAP_RONLY_SRAM:
case MAP_HIROM_SRAM:
/* BJ: no FAST_LSB_WORD_ACCESS here, since if Memory.SRAMMask=0x7ff
* then the high byte doesn't follow the low byte. */
- return (*(Memory.SRAM +
- (((Address & 0x7fff) - 0x6000 +
- ((Address & 0xf0000) >> 3)) & Memory.SRAMMask)) |
- (*(Memory.SRAM +
- ((((Address + 1) & 0x7fff) - 0x6000 +
- (((Address + 1) & 0xf0000) >> 3)) & Memory.SRAMMask)) << 8));
+ return *(Memory.SRAM + (((Address & 0x7fff) - 0x6000 + ((Address & 0xf0000) >> 3)) & Memory.SRAMMask)) | (*(Memory.SRAM + ((((Address + 1) & 0x7fff) - 0x6000 + (((Address + 1) & 0xf0000) >> 3)) & Memory.SRAMMask)) << 8);
case MAP_BWRAM:
#ifdef FAST_LSB_WORD_ACCESS
- return (*(uint16_t*)(Memory.BWRAM + ((Address & 0x7fff) - 0x6000)));
+ return *(uint16_t*) (Memory.BWRAM + ((Address & 0x7fff) - 0x6000));
#else
- return (*(Memory.BWRAM + ((Address & 0x7fff) - 0x6000)) | (*(Memory.BWRAM + (((Address + 1) & 0x7fff) - 0x6000)) << 8));
+ return *(Memory.BWRAM + ((Address & 0x7fff) - 0x6000)) | (*(Memory.BWRAM + (((Address + 1) & 0x7fff) - 0x6000)) << 8);
#endif
case MAP_C4:
- return (S9xGetC4(Address & 0xffff) | (S9xGetC4((Address + 1) & 0xffff) << 8));
+ return S9xGetC4(Address & 0xffff) | (S9xGetC4((Address + 1) & 0xffff) << 8);
case MAP_SPC7110_ROM:
- return (S9xGetSPC7110Byte(Address) | (S9xGetSPC7110Byte(Address + 1)) << 8);
+ return S9xGetSPC7110Byte(Address) | (S9xGetSPC7110Byte(Address + 1)) << 8;
case MAP_SPC7110_DRAM:
- return (S9xGetSPC7110(0x4800) | (S9xGetSPC7110(0x4800) << 8));
+ return S9xGetSPC7110(0x4800) | (S9xGetSPC7110(0x4800) << 8);
case MAP_OBC_RAM:
return GetOBC1(Address & 0xFFFF) | (GetOBC1((Address + 1) & 0xFFFF) << 8);
case MAP_SETA_DSP:
@@ -137,7 +130,7 @@ inline uint16_t S9xGetWord(uint32_t Address)
case MAP_SETA_RISC:
return S9xGetST018(Address) | (S9xGetST018((Address + 1)) << 8);
default:
- return (OpenBus | (OpenBus << 8));
+ return OpenBus | (OpenBus << 8);
}
}
@@ -153,8 +146,7 @@ inline void S9xSetByte(uint8_t Byte, uint32_t Address)
if (SetAddress >= (uint8_t*) MAP_LAST)
{
SetAddress += Address & 0xffff;
- if (SetAddress == SA1.WaitByteAddress1 ||
- SetAddress == SA1.WaitByteAddress2)
+ if (SetAddress == SA1.WaitByteAddress1 || SetAddress == SA1.WaitByteAddress2)
{
SA1.Executing = SA1.S9xOpcodes != NULL;
SA1.WaitCounter = 0;
@@ -199,9 +191,6 @@ inline void S9xSetByte(uint8_t Byte, uint32_t Address)
case MAP_C4:
S9xSetC4(Byte, Address & 0xffff);
return;
- case MAP_SPC7110_DRAM:
- s7r.bank50[(Address & 0xffff)] = (uint8_t) Byte;
- break;
case MAP_OBC_RAM:
SetOBC1(Byte, Address & 0xFFFF);
return;
@@ -235,8 +224,7 @@ inline void S9xSetWord(uint16_t Word, uint32_t Address)
if (SetAddress >= (uint8_t*) MAP_LAST)
{
SetAddress += Address & 0xffff;
- if (SetAddress == SA1.WaitByteAddress1 ||
- SetAddress == SA1.WaitByteAddress2)
+ if (SetAddress == SA1.WaitByteAddress1 || SetAddress == SA1.WaitByteAddress2)
{
SA1.Executing = SA1.S9xOpcodes != NULL;
SA1.WaitCounter = 0;
@@ -279,12 +267,8 @@ inline void S9xSetWord(uint16_t Word, uint32_t Address)
{
/* BJ: no FAST_LSB_WORD_ACCESS here, since if Memory.SRAMMask=0x7ff
* then the high byte doesn't follow the low byte. */
- *(Memory.SRAM +
- (((((Address & 0x7fff) - 0x6000) +
- ((Address & 0xf0000) >> 3)) & Memory.SRAMMask))) = (uint8_t) Word;
- *(Memory.SRAM +
- ((((((Address + 1) & 0x7fff) - 0x6000) +
- (((Address + 1) & 0xf0000) >> 3)) & Memory.SRAMMask))) = (uint8_t)(Word >> 8);
+ *(Memory.SRAM + (((((Address & 0x7fff) - 0x6000) + ((Address & 0xf0000) >> 3)) & Memory.SRAMMask))) = (uint8_t) Word;
+ *(Memory.SRAM + ((((((Address + 1) & 0x7fff) - 0x6000) + (((Address + 1) & 0xf0000) >> 3)) & Memory.SRAMMask))) = (uint8_t)(Word >> 8);
CPU.SRAMModified = true;
}
return;
@@ -297,10 +281,6 @@ inline void S9xSetWord(uint16_t Word, uint32_t Address)
#endif
CPU.SRAMModified = true;
return;
- case MAP_SPC7110_DRAM:
- s7r.bank50[(Address & 0xffff)] = (uint8_t) Word;
- s7r.bank50[((Address + 1) & 0xffff)] = (uint8_t) Word;
- break;
case MAP_SA1RAM:
*(Memory.SRAM + (Address & 0xffff)) = (uint8_t) Word;
*(Memory.SRAM + ((Address + 1) & 0xffff)) = (uint8_t)(Word >> 8);
@@ -331,7 +311,7 @@ inline uint8_t* GetBasePointer(uint32_t Address)
{
uint8_t* GetAddress = Memory.Map [(Address >> MEMMAP_SHIFT) & MEMMAP_MASK];
if (GetAddress >= (uint8_t*) MAP_LAST)
- return (GetAddress);
+ return GetAddress;
if (Settings.SPC7110 && ((Address & 0x7FFFFF) == 0x4800))
return s7r.bank50;
switch ((intptr_t) GetAddress)
@@ -345,17 +325,17 @@ inline uint8_t* GetBasePointer(uint32_t Address)
case MAP_OBC_RAM:
return Memory.FillRAM;
case MAP_DSP:
- return (Memory.FillRAM - 0x6000);
+ return Memory.FillRAM - 0x6000;
case MAP_SA1RAM:
case MAP_LOROM_SRAM:
case MAP_SETA_DSP:
return Memory.SRAM;
case MAP_BWRAM:
- return (Memory.BWRAM - 0x6000);
+ return Memory.BWRAM - 0x6000;
case MAP_HIROM_SRAM:
- return (Memory.SRAM - 0x6000);
+ return Memory.SRAM - 0x6000;
case MAP_C4:
- return (Memory.C4RAM - 0x6000);
+ return Memory.C4RAM - 0x6000;
default:
return NULL;
}
@@ -365,7 +345,7 @@ inline uint8_t* S9xGetMemPointer(uint32_t Address)
{
uint8_t* GetAddress = Memory.Map [(Address >> MEMMAP_SHIFT) & MEMMAP_MASK];
if (GetAddress >= (uint8_t*) MAP_LAST)
- return (GetAddress + (Address & 0xffff));
+ return GetAddress + (Address & 0xffff);
if (Settings.SPC7110 && ((Address & 0x7FFFFF) == 0x4800))
return s7r.bank50;
@@ -375,20 +355,20 @@ inline uint8_t* S9xGetMemPointer(uint32_t Address)
case MAP_SPC7110_DRAM:
return &s7r.bank50[Address & 0xffff];
case MAP_PPU:
- return (Memory.FillRAM + (Address & 0xffff));
+ return Memory.FillRAM + (Address & 0xffff);
case MAP_CPU:
- return (Memory.FillRAM + (Address & 0xffff));
+ return Memory.FillRAM + (Address & 0xffff);
case MAP_DSP:
- return (Memory.FillRAM - 0x6000 + (Address & 0xffff));
+ return Memory.FillRAM - 0x6000 + (Address & 0xffff);
case MAP_SA1RAM:
case MAP_LOROM_SRAM:
- return (Memory.SRAM + (Address & 0xffff));
+ return Memory.SRAM + (Address & 0xffff);
case MAP_BWRAM:
- return (Memory.BWRAM - 0x6000 + (Address & 0xffff));
+ return Memory.BWRAM - 0x6000 + (Address & 0xffff);
case MAP_HIROM_SRAM:
- return (Memory.SRAM - 0x6000 + (Address & 0xffff));
+ return Memory.SRAM - 0x6000 + (Address & 0xffff);
case MAP_C4:
- return (Memory.C4RAM - 0x6000 + (Address & 0xffff));
+ return Memory.C4RAM - 0x6000 + (Address & 0xffff);
case MAP_OBC_RAM:
return GetMemPointerOBC1(Address);
case MAP_SETA_DSP:
diff --git a/source/gfx.c b/source/gfx.c
index e28c4f3..cc946aa 100644
--- a/source/gfx.c
+++ b/source/gfx.c
@@ -14,20 +14,20 @@
#define M7 19
-void ComputeClipWindows();
+void ComputeClipWindows(void);
-extern uint8_t BitShifts[8][4];
-extern uint8_t TileShifts[8][4];
+extern uint8_t BitShifts [8][4];
+extern uint8_t TileShifts [8][4];
extern uint8_t PaletteShifts[8][4];
-extern uint8_t PaletteMasks[8][4];
-extern uint8_t Depths[8][4];
-extern uint8_t BGSizes [2];
+extern uint8_t PaletteMasks [8][4];
+extern uint8_t Depths [8][4];
+extern uint8_t BGSizes [2];
-extern NormalTileRenderer DrawTilePtr;
+extern NormalTileRenderer DrawTilePtr;
extern ClippedTileRenderer DrawClippedTilePtr;
-extern NormalTileRenderer DrawHiResTilePtr;
+extern NormalTileRenderer DrawHiResTilePtr;
extern ClippedTileRenderer DrawHiResClippedTilePtr;
-extern LargePixelRenderer DrawLargePixelPtr;
+extern LargePixelRenderer DrawLargePixelPtr;
extern SBG BG;
@@ -77,100 +77,40 @@ extern uint8_t Mode7Depths [2];
} \
}
-
#define BLACK BUILD_PIXEL(0,0,0)
-void DrawTile16(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-void DrawClippedTile16(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-void DrawTile16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-void DrawClippedTile16HalfWidth(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-void DrawTile16x2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-void DrawClippedTile16x2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-void DrawTile16x2x2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-void DrawClippedTile16x2x2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-void DrawLargePixel16(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-void DrawLargePixel16HalfWidth(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-
-void DrawClippedTile16Add(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawTile16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-
-void DrawClippedTile16Add1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawTile16FixedAdd1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-
-void DrawClippedTile16FixedAdd1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawTile16Sub(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-
-void DrawClippedTile16Sub(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawTile16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-
-void DrawClippedTile16Sub1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawTile16FixedSub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount);
-
-void DrawClippedTile16FixedSub1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawLargePixel16Add(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawLargePixel16Add1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawLargePixel16Sub(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-
-void DrawLargePixel16Sub1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-
-bool S9xInitGFX()
+void DrawTile16(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16x2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16x2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16x2x2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16x2x2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawLargePixel16(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
+void DrawLargePixel16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16FixedAdd1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16FixedAdd1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16Sub(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16Sub(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawTile16FixedSub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+void DrawClippedTile16FixedSub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+void DrawLargePixel16Add(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
+void DrawLargePixel16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
+void DrawLargePixel16Sub(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
+void DrawLargePixel16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
+
+bool S9xInitGFX(void)
{
uint32_t r, g, b;
uint32_t PixelOdd = 1;
uint32_t PixelEven = 2;
-
uint8_t bitshift;
for (bitshift = 0; bitshift < 4; bitshift++)
{
@@ -283,10 +223,9 @@ bool S9xInitGFX()
S9xFixColourBrightness();
if (!(GFX.X2 = (uint16_t*) malloc(sizeof(uint16_t) * 0x10000)))
- return (false);
+ return false;
- if (!(GFX.ZERO_OR_X2 = (uint16_t*) malloc(sizeof(uint16_t) * 0x10000)) ||
- !(GFX.ZERO = (uint16_t*) malloc(sizeof(uint16_t) * 0x10000)))
+ if (!(GFX.ZERO_OR_X2 = (uint16_t*) malloc(sizeof(uint16_t) * 0x10000)) || !(GFX.ZERO = (uint16_t*) malloc(sizeof(uint16_t) * 0x10000)))
{
if (GFX.ZERO_OR_X2)
{
@@ -298,7 +237,7 @@ bool S9xInitGFX()
free(GFX.X2);
GFX.X2 = NULL;
}
- return (false);
+ return false;
}
// Build a lookup table that multiplies a packed RGB value by 2 with
@@ -328,7 +267,6 @@ bool S9xInitGFX()
// Build a lookup table that if the top bit of the color value is zero
// then the value is zero, otherwise multiply the value by 2. Used by
// the color subtraction code.
-
for (r = 0; r <= MAX_RED; r++)
{
uint32_t r2 = r;
@@ -389,10 +327,10 @@ bool S9xInitGFX()
}
}
}
- return (true);
+ return true;
}
-void S9xDeinitGFX()
+void S9xDeinitGFX(void)
{
// Free any memory allocated in S9xInitGFX
if (GFX.X2)
@@ -412,32 +350,19 @@ void S9xDeinitGFX()
}
}
-void S9xBuildDirectColourMaps()
+void S9xBuildDirectColourMaps(void)
{
uint32_t p, c;
for (p = 0; p < 8; p++)
- {
for (c = 0; c < 256; c++)
- {
- // XXX: Brightness
- DirectColourMaps [p][c] = BUILD_PIXEL(((c & 7) << 2) | ((p & 1) << 1),
- ((c & 0x38) >> 1) | (p & 2),
- ((c & 0xc0) >> 3) | (p & 4));
- }
- }
+ DirectColourMaps [p][c] = BUILD_PIXEL(((c & 7) << 2) | ((p & 1) << 1), ((c & 0x38) >> 1) | (p & 2), ((c & 0xc0) >> 3) | (p & 4)); // XXX: Brightness
IPPU.DirectColourMapsNeedRebuild = false;
}
-void S9xStartScreenRefresh()
+void S9xStartScreenRefresh(void)
{
if (IPPU.RenderThisFrame)
{
- if (!S9xInitUpdate())
- {
- IPPU.RenderThisFrame = false;
- return;
- }
-
IPPU.PreviousLine = IPPU.CurrentLine = 0;
if (PPU.BGMode == 5 || PPU.BGMode == 6)
@@ -510,8 +435,7 @@ void RenderLine(uint8_t C)
}
else
{
- if (Settings.StarfoxHack && PPU.BG[2].VOffset == 0 &&
- PPU.BG[2].HOffset == 0xe000)
+ if (Settings.StarfoxHack && PPU.BG[2].VOffset == 0 && PPU.BG[2].HOffset == 0xe000)
{
LineData[C].BG[2].VOffset = 0xe1;
LineData[C].BG[2].HOffset = 0;
@@ -536,7 +460,7 @@ void RenderLine(uint8_t C)
}
}
-void S9xEndScreenRefresh()
+void S9xEndScreenRefresh(void)
{
if (IPPU.RenderThisFrame)
{
@@ -621,11 +545,10 @@ static INLINE void SelectTileRenderer(bool normal)
}
}
-void S9xSetupOBJ()
+void S9xSetupOBJ(void)
{
int32_t Height;
uint8_t S;
-
int32_t SmallWidth, SmallHeight;
int32_t LargeWidth, LargeHeight;
@@ -679,7 +602,6 @@ void S9xSetupOBJ()
uint8_t FirstSprite;
/* normal case */
uint8_t LineOBJ[SNES_HEIGHT_EXTENDED];
-
memset(LineOBJ, 0, sizeof(LineOBJ));
for (i = 0; i < SNES_HEIGHT_EXTENDED; i++)
{
@@ -702,7 +624,8 @@ void S9xSetupOBJ()
Height = SmallHeight;
}
HPos = PPU.OBJ[S].HPos;
- if (HPos == -256) HPos = 256;
+ if (HPos == -256)
+ HPos = 256;
if (HPos > -GFX.OBJWidths[S] && HPos <= 256)
{
uint8_t line, Y;
@@ -714,19 +637,21 @@ void S9xSetupOBJ()
GFX.OBJVisibleTiles[S] = GFX.OBJWidths[S] >> 3;
for (line = 0, Y = (uint8_t)(PPU.OBJ[S].VPos & 0xff); line < Height; Y++, line++)
{
- if (Y >= SNES_HEIGHT_EXTENDED) continue;
+ if (Y >= SNES_HEIGHT_EXTENDED)
+ continue;
if (LineOBJ[Y] >= 32)
{
GFX.OBJLines[Y].RTOFlags |= 0x40;
continue;
}
GFX.OBJLines[Y].Tiles -= GFX.OBJVisibleTiles[S];
- if (GFX.OBJLines[Y].Tiles < 0) GFX.OBJLines[Y].RTOFlags |= 0x80;
+ if (GFX.OBJLines[Y].Tiles < 0)
+ GFX.OBJLines[Y].RTOFlags |= 0x80;
GFX.OBJLines[Y].OBJ[LineOBJ[Y]].Sprite = S;
if (PPU.OBJ[S].VFlip)
{
// Yes, Width not Height. It so happens that the
- // sprites with H=2*W flip as two WxW sprites.
+ // sprites with H = 2 * W flip as two W * W sprites.
GFX.OBJLines[Y].OBJ[LineOBJ[Y]].Line = line ^ (GFX.OBJWidths[S] - 1);
}
else
@@ -735,14 +660,11 @@ void S9xSetupOBJ()
}
}
S = (S + 1) & 0x7F;
- }
- while (S != FirstSprite);
+ } while (S != FirstSprite);
for (Y = 0; Y < SNES_HEIGHT_EXTENDED; Y++)
- {
if (LineOBJ[Y] < 32) // Add the sentinel
GFX.OBJLines[Y].OBJ[LineOBJ[Y]].Sprite = -1;
- }
for (Y = 1; Y < SNES_HEIGHT_EXTENDED; Y++)
GFX.OBJLines[Y].RTOFlags |= GFX.OBJLines[Y - 1].RTOFlags;
}
@@ -750,12 +672,10 @@ void S9xSetupOBJ()
{
int32_t j, Y;
/* evil FirstSprite+Y case */
-
/* First, find out which sprites are on which lines */
uint8_t OBJOnLine[SNES_HEIGHT_EXTENDED][128];
/* We only initialise this per line, as needed. [Neb]
- * Bonus: We can quickly avoid looping if a line has no OBJs.
- */
+ * Bonus: We can quickly avoid looping if a line has no OBJs. */
bool AnyOBJOnLine[SNES_HEIGHT_EXTENDED];
memset(AnyOBJOnLine, false, sizeof(AnyOBJOnLine));
@@ -773,7 +693,8 @@ void S9xSetupOBJ()
Height = SmallHeight;
}
HPos = PPU.OBJ[S].HPos;
- if (HPos == -256) HPos = 256;
+ if (HPos == -256)
+ HPos = 256;
if (HPos > -GFX.OBJWidths[S] && HPos <= 256)
{
uint8_t line, Y;
@@ -785,7 +706,8 @@ void S9xSetupOBJ()
GFX.OBJVisibleTiles[S] = GFX.OBJWidths[S] >> 3;
for (line = 0, Y = (uint8_t)(PPU.OBJ[S].VPos & 0xff); line < Height; Y++, line++)
{
- if (Y >= SNES_HEIGHT_EXTENDED) continue;
+ if (Y >= SNES_HEIGHT_EXTENDED)
+ continue;
if (!AnyOBJOnLine[Y])
{
memset(OBJOnLine[Y], 0, 128);
@@ -807,7 +729,6 @@ void S9xSetupOBJ()
for (Y = 0; Y < SNES_HEIGHT_EXTENDED; Y++)
{
GFX.OBJLines[Y].RTOFlags = Y ? GFX.OBJLines[Y - 1].RTOFlags : 0;
-
GFX.OBJLines[Y].Tiles = 34;
j = 0;
if (AnyOBJOnLine[Y])
@@ -824,15 +745,16 @@ void S9xSetupOBJ()
break;
}
GFX.OBJLines[Y].Tiles -= GFX.OBJVisibleTiles[S];
- if (GFX.OBJLines[Y].Tiles < 0) GFX.OBJLines[Y].RTOFlags |= 0x80;
+ if (GFX.OBJLines[Y].Tiles < 0)
+ GFX.OBJLines[Y].RTOFlags |= 0x80;
GFX.OBJLines[Y].OBJ[j].Sprite = S;
GFX.OBJLines[Y].OBJ[j++].Line = OBJOnLine[Y][S] & ~0x80;
}
S = (S + 1) & 0x7F;
- }
- while (S != FirstSprite);
+ } while (S != FirstSprite);
}
- if (j < 32) GFX.OBJLines[Y].OBJ[j].Sprite = -1;
+ if (j < 32)
+ GFX.OBJLines[Y].OBJ[j].Sprite = -1;
}
}
@@ -841,14 +763,14 @@ void S9xSetupOBJ()
static void DrawOBJS(bool OnMain, uint8_t D)
{
- int32_t clipcount;
struct
{
uint16_t Pos;
bool Value;
} Windows[7];
- uint32_t Y, Offset;
+ int32_t clipcount;
+ uint32_t Y, Offset;
BG.BitShift = 4;
BG.TileShift = 5;
BG.TileAddress = PPU.OBJNameBase;
@@ -859,10 +781,9 @@ static void DrawOBJS(bool OnMain, uint8_t D)
BG.Buffered = IPPU.TileCached [TILE_4BIT];
BG.NameSelect = PPU.OBJNameSelect;
BG.DirectColourMode = false;
-
GFX.PixSize = 1;
-
clipcount = GFX.pCurrentClip->Count [4];
+
if (!clipcount)
{
Windows[0].Pos = 0;
@@ -888,7 +809,8 @@ static void DrawOBJS(bool OnMain, uint8_t D)
else
{
// memmove required: Overlapping addresses [Neb]
- if (j < i) memmove(&Windows[j + 1], &Windows[j], sizeof(Windows[0]) * (i - j));
+ if (j < i)
+ memmove(&Windows[j + 1], &Windows[j], sizeof(Windows[0]) * (i - j));
Windows[j].Pos = GFX.pCurrentClip->Left[clip][4];
Windows[j].Value = true;
i++;
@@ -897,7 +819,8 @@ static void DrawOBJS(bool OnMain, uint8_t D)
if (j >= i || Windows[j].Pos != GFX.pCurrentClip->Right[clip][4])
{
// memmove required: Overlapping addresses [Neb]
- if (j < i) memmove(&Windows[j + 1], &Windows[j], sizeof(Windows[0]) * (i - j));
+ if (j < i)
+ memmove(&Windows[j + 1], &Windows[j], sizeof(Windows[0]) * (i - j));
Windows[j].Pos = GFX.pCurrentClip->Right[clip][4];
Windows[j].Value = false;
i++;
@@ -932,14 +855,12 @@ static void DrawOBJS(bool OnMain, uint8_t D)
GFX.Z1 = D + 2;
- for (Y = GFX.StartY, Offset = Y * GFX.PPL; Y <= GFX.EndY;
- Y++, Offset += GFX.PPL)
+ for (Y = GFX.StartY, Offset = Y * GFX.PPL; Y <= GFX.EndY; Y++, Offset += GFX.PPL)
{
int32_t I = 0;
int32_t tiles = GFX.OBJLines[Y].Tiles;
int32_t S;
- for (S = GFX.OBJLines[Y].OBJ[I].Sprite; S >= 0
- && I < 32; S = GFX.OBJLines[Y].OBJ[++I].Sprite)
+ for (S = GFX.OBJLines[Y].OBJ[I].Sprite; S >= 0 && I < 32; S = GFX.OBJLines[Y].OBJ[++I].Sprite)
{
int32_t TileInc = 1;
int32_t TileLine;
@@ -957,8 +878,7 @@ static void DrawOBJS(bool OnMain, uint8_t D)
if (OnMain && SUB_OR_ADD(4))
SelectTileRenderer(!GFX.Pseudo && PPU.OBJ [S].Palette < 4);
- BaseTile = (((GFX.OBJLines[Y].OBJ[I].Line << 1) + (PPU.OBJ[S].Name & 0xf0))
- & 0xf0) | (PPU.OBJ[S].Name & 0x100) | (PPU.OBJ[S].Palette << 10);
+ BaseTile = (((GFX.OBJLines[Y].OBJ[I].Line << 1) + (PPU.OBJ[S].Name & 0xf0)) & 0xf0) | (PPU.OBJ[S].Name & 0x100) | (PPU.OBJ[S].Palette << 10);
TileX = PPU.OBJ[S].Name & 0x0f;
TileLine = (GFX.OBJLines[Y].OBJ[I].Line & 7) * 8;
@@ -972,37 +892,43 @@ static void DrawOBJS(bool OnMain, uint8_t D)
GFX.Z2 = (PPU.OBJ[S].Priority + 1) * 4 + D;
X = PPU.OBJ[S].HPos;
- if (X == -256) X = 256;
- for (t = tiles, O = Offset + X * GFX.PixSize; X <= 256
- && X < PPU.OBJ[S].HPos + GFX.OBJWidths[S];
- TileX = (TileX + TileInc) & 0x0f, X += 8, O += 8 * GFX.PixSize)
+ if (X == -256)
+ X = 256;
+ for (t = tiles, O = Offset + X * GFX.PixSize; X <= 256 && X < PPU.OBJ[S].HPos + GFX.OBJWidths[S]; TileX = (TileX + TileInc) & 0x0f, X += 8, O += 8 * GFX.PixSize)
{
- if (X < -7 || --t < 0 || X == 256) continue;
+ if (X < -7 || --t < 0 || X == 256)
+ continue;
if (X >= NextPos)
{
for (; WinIdx < 7 && Windows[WinIdx].Pos <= X; WinIdx++);
- if (WinIdx == 0) WinStat = false;
- else WinStat = Windows[WinIdx - 1].Value;
+ if (WinIdx == 0)
+ WinStat = false;
+ else
+ WinStat = Windows[WinIdx - 1].Value;
NextPos = (WinIdx < 7) ? Windows[WinIdx].Pos : 1000;
}
if (X + 8 < NextPos)
{
- if (WinStat)(*DrawTilePtr)(BaseTile | TileX, O, TileLine, 1);
+ if (WinStat)
+ (*DrawTilePtr)(BaseTile | TileX, O, TileLine, 1);
}
else
{
int32_t x = X;
while (x < X + 8)
{
- if (WinStat)(*DrawClippedTilePtr)(BaseTile | TileX, O, x - X, NextPos - x,
- TileLine, 1);
+ if (WinStat)
+ (*DrawClippedTilePtr)(BaseTile | TileX, O, x - X, NextPos - x, TileLine, 1);
x = NextPos;
for (; WinIdx < 7 && Windows[WinIdx].Pos <= x; WinIdx++);
- if (WinIdx == 0) WinStat = false;
- else WinStat = Windows[WinIdx - 1].Value;
+ if (WinIdx == 0)
+ WinStat = false;
+ else
+ WinStat = Windows[WinIdx - 1].Value;
NextPos = (WinIdx < 7) ? Windows[WinIdx].Pos : 1000;
- if (NextPos > X + 8) NextPos = X + 8;
+ if (NextPos > X + 8)
+ NextPos = X + 8;
}
}
}
@@ -1017,14 +943,12 @@ static void DrawBackgroundMosaic(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
uint32_t OffsetShift;
uint32_t Y;
int32_t m5;
-
uint32_t Tile;
uint16_t* SC0;
uint16_t* SC1;
uint16_t* SC2;
uint16_t* SC3;
uint8_t depths [2];
-
depths[0] = Z1;
depths[1] = Z2;
@@ -1091,16 +1015,15 @@ static void DrawBackgroundMosaic(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
uint32_t MosaicOffset = Y % PPU.Mosaic;
for (Lines = 1; Lines < PPU.Mosaic - MosaicOffset; Lines++)
- if ((VOffset != LineData [Y + Lines].BG[bg].VOffset) ||
- (HOffset != LineData [Y + Lines].BG[bg].HOffset))
+ if ((VOffset != LineData [Y + Lines].BG[bg].VOffset) || (HOffset != LineData [Y + Lines].BG[bg].HOffset))
break;
MosaicLine = VOffset + Y - MosaicOffset;
if (Y + Lines > GFX.EndY)
Lines = GFX.EndY + 1 - Y;
- VirtAlign = (MosaicLine & 7) << 3;
+ VirtAlign = (MosaicLine & 7) << 3;
ScreenLine = MosaicLine >> OffsetShift;
Rem16 = MosaicLine & 15;
@@ -1136,9 +1059,7 @@ static void DrawBackgroundMosaic(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
PixWidth = (PPU.Mosaic << m5) - r;
}
s = Y * GFX.PPL + Left * GFX.PixSize;
- for (x = Left; x < Right; x += PixWidth,
- s += (IPPU.HalfWidthPixels ? PixWidth >> 1 : PixWidth) * GFX.PixSize,
- HPos += PixWidth, PixWidth = (PPU.Mosaic << m5))
+ for (x = Left; x < Right; x += PixWidth, s += (IPPU.HalfWidthPixels ? PixWidth >> 1 : PixWidth) * GFX.PixSize, HPos += PixWidth, PixWidth = (PPU.Mosaic << m5))
{
uint32_t Quot = (HPos & OffsetMask) >> 3;
@@ -1173,33 +1094,17 @@ static void DrawBackgroundMosaic(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
{
// Both horzontal & vertical flip
if (Rem16 < 8)
- {
- (*DrawLargePixelPtr)(Tile + 17 - (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + 17 - (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
else
- {
- (*DrawLargePixelPtr)(Tile + 1 - (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + 1 - (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
}
else
{
// Horizontal flip only
if (Rem16 > 7)
- {
- (*DrawLargePixelPtr)(Tile + 17 - (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + 17 - (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
else
- {
- (*DrawLargePixelPtr)(Tile + 1 - (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + 1 - (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
}
}
else
@@ -1209,39 +1114,22 @@ static void DrawBackgroundMosaic(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
{
// Vertical flip only
if (Rem16 < 8)
- {
- (*DrawLargePixelPtr)(Tile + 16 + (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + 16 + (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
else
- {
- (*DrawLargePixelPtr)(Tile + (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
}
else
{
// Normal unflipped
if (Rem16 > 7)
- {
- (*DrawLargePixelPtr)(Tile + 16 + (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + 16 + (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
else
- {
- (*DrawLargePixelPtr)(Tile + (Quot & 1), s,
- HPos & 7, PixWidth,
- VirtAlign, Lines);
- }
+ (*DrawLargePixelPtr)(Tile + (Quot & 1), s, HPos & 7, PixWidth, VirtAlign, Lines);
}
}
}
else
- (*DrawLargePixelPtr)(Tile + (Quot & 1) * m5, s, HPos & 7, PixWidth,
- VirtAlign, Lines);
+ (*DrawLargePixelPtr)(Tile + (Quot & 1) * m5, s, HPos & 7, PixWidth, VirtAlign, Lines);
}
}
}
@@ -1268,7 +1156,6 @@ static void DrawBackgroundOffset(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
uint8_t depths [2] = {Z1, Z2};
BG.StartPalette = 0;
-
BPS0 = (uint16_t*) &Memory.VRAM[PPU.BG[2].SCBase << 1];
if (PPU.BG[2].SCSize & 1)
@@ -1331,7 +1218,6 @@ static void DrawBackgroundOffset(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
{
uint32_t VOff = LineData [Y].BG[2].VOffset - 1;
uint32_t HOff = LineData [Y].BG[2].HOffset;
-
int32_t VirtAlign;
int32_t ScreenLine = VOff >> 3;
int32_t t1;
@@ -1521,33 +1407,17 @@ static void DrawBackgroundOffset(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8
(*DrawClippedTilePtr)(Tile, s, Offset, Count, VirtAlign, Lines);
else
{
- if (!(Tile & (V_FLIP | H_FLIP)))
- {
- // Normal, unflipped
- (*DrawClippedTilePtr)(Tile + t1 + (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
+ if (!(Tile & (V_FLIP | H_FLIP))) // Normal, unflipped
+ (*DrawClippedTilePtr)(Tile + t1 + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
else if (Tile & H_FLIP)
{
- if (Tile & V_FLIP)
- {
- // H & V flip
- (*DrawClippedTilePtr)(Tile + t2 + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- else
- {
- // H flip only
- (*DrawClippedTilePtr)(Tile + t1 + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- }
- else
- {
- // V flip only
- (*DrawClippedTilePtr)(Tile + t2 + (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
+ if (Tile & V_FLIP) // H & V flip
+ (*DrawClippedTilePtr)(Tile + t2 + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
+ else // H flip only
+ (*DrawClippedTilePtr)(Tile + t1 + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
}
+ else // V flip only
+ (*DrawClippedTilePtr)(Tile + t2 + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
}
Left += Count;
@@ -1578,9 +1448,8 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
GFX.Pitch = GFX.RealPitch;
GFX.PPL = GFX.PPLx2 >> 1;
}
- GFX.PixSize = 1;
-
+ GFX.PixSize = 1;
depths[0] = Z1;
depths[1] = Z2;
BG.StartPalette = 0;
@@ -1597,7 +1466,8 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
if ((PPU.BG[bg].SCSize & 2))
SC2 = SC1 + 1024;
- else SC2 = SC0;
+ else
+ SC2 = SC0;
if (((uint8_t*)SC2 - Memory.VRAM) >= 0x10000)
SC2 -= 0x08000;
@@ -1624,25 +1494,21 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
int32_t t2;
uint16_t* b1;
uint16_t* b2;
-
int32_t clipcount;
int32_t clip;
-
int32_t y = IPPU.Interlace ? (Y >> 1) : Y;
uint32_t VOffset = LineData [y].BG[bg].VOffset;
uint32_t HOffset = LineData [y].BG[bg].HOffset;
int32_t VirtAlign = (Y + VOffset) & 7;
for (Lines = 1; Lines < 8 - VirtAlign; Lines++)
- if ((VOffset != LineData [y + Lines].BG[bg].VOffset) ||
- (HOffset != LineData [y + Lines].BG[bg].HOffset))
+ if ((VOffset != LineData [y + Lines].BG[bg].VOffset) || (HOffset != LineData [y + Lines].BG[bg].HOffset))
break;
HOffset <<= 1;
if (Y + Lines > endy)
Lines = endy + 1 - Y;
VirtAlign <<= 3;
-
ScreenLine = (VOffset + Y) >> VOffsetShift;
if (((VOffset + Y) & 15) > 7)
@@ -1695,7 +1561,6 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
s = (IPPU.HalfWidthPixels ? Left >> 1 : Left) * GFX.PixSize + Y * GFX.PPL;
HPos = (HOffset + Left * GFX.PixSize) & 0x3ff;
-
Quot = HPos >> 3;
if (Quot > 63)
@@ -1717,48 +1582,24 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
if (BG.TileSize == 8)
{
- if (!(Tile & H_FLIP))
- {
- // Normal, unflipped
- (*DrawHiResClippedTilePtr)(Tile + (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- else
- {
- // H flip
- (*DrawHiResClippedTilePtr)(Tile + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
+ if (!(Tile & H_FLIP)) // Normal, unflipped
+ (*DrawHiResClippedTilePtr)(Tile + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
+ else // H flip
+ (*DrawHiResClippedTilePtr)(Tile + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
}
else
{
- if (!(Tile & (V_FLIP | H_FLIP)))
- {
- // Normal, unflipped
- (*DrawHiResClippedTilePtr)(Tile + t1 + (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
+ if (!(Tile & (V_FLIP | H_FLIP))) // Normal, unflipped
+ (*DrawHiResClippedTilePtr)(Tile + t1 + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
else if (Tile & H_FLIP)
{
- if (Tile & V_FLIP)
- {
- // H & V flip
- (*DrawHiResClippedTilePtr)(Tile + t2 + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- else
- {
- // H flip only
- (*DrawHiResClippedTilePtr)(Tile + t1 + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- }
- else
- {
- // V flip only
- (*DrawHiResClippedTilePtr)(Tile + t2 + (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
+ if (Tile & V_FLIP) // H & V flip
+ (*DrawHiResClippedTilePtr)(Tile + t2 + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
+ else // H flip only
+ (*DrawHiResClippedTilePtr)(Tile + t1 + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
}
+ else // V flip only
+ (*DrawHiResClippedTilePtr)(Tile + t2 + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
}
t += Quot & 1;
@@ -1781,48 +1622,24 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
GFX.Z1 = GFX.Z2 = depths [(Tile & 0x2000) >> 13];
if (BG.TileSize == 8)
{
- if (!(Tile & H_FLIP))
- {
- // Normal, unflipped
- (*DrawHiResTilePtr)(Tile + (Quot & 1),
- s, VirtAlign, Lines);
- }
- else
- {
- // H flip
- (*DrawHiResTilePtr)(Tile + 1 - (Quot & 1),
- s, VirtAlign, Lines);
- }
+ if (!(Tile & H_FLIP)) // Normal, unflipped
+ (*DrawHiResTilePtr)(Tile + (Quot & 1), s, VirtAlign, Lines);
+ else // H flip
+ (*DrawHiResTilePtr)(Tile + 1 - (Quot & 1), s, VirtAlign, Lines);
}
else
{
- if (!(Tile & (V_FLIP | H_FLIP)))
- {
- // Normal, unflipped
- (*DrawHiResTilePtr)(Tile + t1 + (Quot & 1),
- s, VirtAlign, Lines);
- }
+ if (!(Tile & (V_FLIP | H_FLIP))) // Normal, unflipped
+ (*DrawHiResTilePtr)(Tile + t1 + (Quot & 1), s, VirtAlign, Lines);
else if (Tile & H_FLIP)
{
- if (Tile & V_FLIP)
- {
- // H & V flip
- (*DrawHiResTilePtr)(Tile + t2 + 1 - (Quot & 1),
- s, VirtAlign, Lines);
- }
- else
- {
- // H flip only
- (*DrawHiResTilePtr)(Tile + t1 + 1 - (Quot & 1),
- s, VirtAlign, Lines);
- }
- }
- else
- {
- // V flip only
- (*DrawHiResTilePtr)(Tile + t2 + (Quot & 1),
- s, VirtAlign, Lines);
+ if (Tile & V_FLIP) // H & V flip
+ (*DrawHiResTilePtr)(Tile + t2 + 1 - (Quot & 1), s, VirtAlign, Lines);
+ else // H flip only
+ (*DrawHiResTilePtr)(Tile + t1 + 1 - (Quot & 1), s, VirtAlign, Lines);
}
+ else // V flip only
+ (*DrawHiResTilePtr)(Tile + t2 + (Quot & 1), s, VirtAlign, Lines);
}
t += Quot & 1;
@@ -1839,48 +1656,24 @@ static void DrawBackgroundMode5(uint32_t bg, uint8_t Z1, uint8_t Z2)
GFX.Z1 = GFX.Z2 = depths [(Tile & 0x2000) >> 13];
if (BG.TileSize == 8)
{
- if (!(Tile & H_FLIP))
- {
- // Normal, unflipped
- (*DrawHiResClippedTilePtr)(Tile + (Quot & 1),
- s, 0, Count, VirtAlign, Lines);
- }
- else
- {
- // H flip
- (*DrawHiResClippedTilePtr)(Tile + 1 - (Quot & 1),
- s, 0, Count, VirtAlign, Lines);
- }
+ if (!(Tile & H_FLIP)) // Normal, unflipped
+ (*DrawHiResClippedTilePtr)(Tile + (Quot & 1), s, 0, Count, VirtAlign, Lines);
+ else // H flip
+ (*DrawHiResClippedTilePtr)(Tile + 1 - (Quot & 1), s, 0, Count, VirtAlign, Lines);
}
else
{
- if (!(Tile & (V_FLIP | H_FLIP)))
- {
- // Normal, unflipped
- (*DrawHiResClippedTilePtr)(Tile + t1 + (Quot & 1),
- s, 0, Count, VirtAlign, Lines);
- }
+ if (!(Tile & (V_FLIP | H_FLIP))) // Normal, unflipped
+ (*DrawHiResClippedTilePtr)(Tile + t1 + (Quot & 1), s, 0, Count, VirtAlign, Lines);
else if (Tile & H_FLIP)
{
- if (Tile & V_FLIP)
- {
- // H & V flip
- (*DrawHiResClippedTilePtr)(Tile + t2 + 1 - (Quot & 1),
- s, 0, Count, VirtAlign, Lines);
- }
- else
- {
- // H flip only
- (*DrawHiResClippedTilePtr)(Tile + t1 + 1 - (Quot & 1),
- s, 0, Count, VirtAlign, Lines);
- }
- }
- else
- {
- // V flip only
- (*DrawHiResClippedTilePtr)(Tile + t2 + (Quot & 1),
- s, 0, Count, VirtAlign, Lines);
+ if (Tile & V_FLIP) // H & V flip
+ (*DrawHiResClippedTilePtr)(Tile + t2 + 1 - (Quot & 1), s, 0, Count, VirtAlign, Lines);
+ else // H flip only
+ (*DrawHiResClippedTilePtr)(Tile + t1 + 1 - (Quot & 1), s, 0, Count, VirtAlign, Lines);
}
+ else // V flip only
+ (*DrawHiResClippedTilePtr)(Tile + t2 + (Quot & 1), s, 0, Count, VirtAlign, Lines);
}
}
}
@@ -1902,8 +1695,6 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
int32_t Lines;
int32_t OffsetMask;
int32_t OffsetShift;
-
-
GFX.PixSize = 1;
BG.TileSize = BGSizes [PPU.BG[bg].BGSize];
@@ -1915,8 +1706,7 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
BG.Buffered = IPPU.TileCached [Depths [BGMode][bg]];
BG.PaletteShift = PaletteShifts[BGMode][bg];
BG.PaletteMask = PaletteMasks[BGMode][bg];
- BG.DirectColourMode = (BGMode == 3 || BGMode == 4) && bg == 0 &&
- (GFX.r2130 & 1);
+ BG.DirectColourMode = (BGMode == 3 || BGMode == 4) && bg == 0 && (GFX.r2130 & 1);
if (PPU.BGMosaic [bg] && PPU.Mosaic > 1)
{
@@ -1941,7 +1731,8 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
if (BGMode == 0)
BG.StartPalette = bg << 5;
- else BG.StartPalette = 0;
+ else
+ BG.StartPalette = 0;
SC0 = (uint16_t*) &Memory.VRAM[PPU.BG[bg].SCBase << 1];
@@ -1994,15 +1785,13 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
int32_t VirtAlign = (Y + VOffset) & 7;
for (Lines = 1; Lines < 8 - VirtAlign; Lines++)
- if ((VOffset != LineData [Y + Lines].BG[bg].VOffset) ||
- (HOffset != LineData [Y + Lines].BG[bg].HOffset))
+ if ((VOffset != LineData [Y + Lines].BG[bg].VOffset) || (HOffset != LineData [Y + Lines].BG[bg].HOffset))
break;
if (Y + Lines > GFX.EndY)
Lines = GFX.EndY + 1 - Y;
VirtAlign <<= 3;
-
ScreenLine = (VOffset + Y) >> OffsetShift;
if (((VOffset + Y) & 15) > 7)
@@ -2051,10 +1840,8 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
continue;
}
-
s = Left * GFX.PixSize + Y * GFX.PPL;
HPos = (HOffset + Left) & OffsetMask;
-
Quot = HPos >> 3;
if (BG.TileSize == 8)
@@ -2085,40 +1872,18 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
GFX.Z1 = GFX.Z2 = depths [(Tile & 0x2000) >> 13];
if (BG.TileSize == 8)
+ (*DrawClippedTilePtr)(Tile, s, Offset, Count, VirtAlign, Lines);
+ else if (!(Tile & (V_FLIP | H_FLIP))) // Normal, unflipped
+ (*DrawClippedTilePtr)(Tile + t1 + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
+ else if (Tile & H_FLIP)
{
- (*DrawClippedTilePtr)(Tile, s, Offset, Count, VirtAlign,
- Lines);
- }
- else
- {
- if (!(Tile & (V_FLIP | H_FLIP)))
- {
- // Normal, unflipped
- (*DrawClippedTilePtr)(Tile + t1 + (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- else if (Tile & H_FLIP)
- {
- if (Tile & V_FLIP)
- {
- // H & V flip
- (*DrawClippedTilePtr)(Tile + t2 + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- else
- {
- // H flip only
- (*DrawClippedTilePtr)(Tile + t1 + 1 - (Quot & 1),
- s, Offset, Count, VirtAlign, Lines);
- }
- }
- else
- {
- // V flip only
- (*DrawClippedTilePtr)(Tile + t2 + (Quot & 1), s,
- Offset, Count, VirtAlign, Lines);
- }
+ if (Tile & V_FLIP) // H & V flip
+ (*DrawClippedTilePtr)(Tile + t2 + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
+ else // H flip only
+ (*DrawClippedTilePtr)(Tile + t1 + 1 - (Quot & 1), s, Offset, Count, VirtAlign, Lines);
}
+ else // V flip only
+ (*DrawClippedTilePtr)(Tile + t2 + (Quot & 1), s, Offset, Count, VirtAlign, Lines);
if (BG.TileSize == 8)
{
@@ -2144,46 +1909,25 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
Count = Width - Count;
Middle = Count >> 3;
Count &= 7;
- for (C = Middle; C > 0;
- s += (IPPU.HalfWidthPixels ? 4 : 8) * GFX.PixSize, Quot++, C--)
+
+ for (C = Middle; C > 0; s += (IPPU.HalfWidthPixels ? 4 : 8) * GFX.PixSize, Quot++, C--)
{
Tile = READ_2BYTES(t);
GFX.Z1 = GFX.Z2 = depths [(Tile & 0x2000) >> 13];
if (BG.TileSize != 8)
{
- if (Tile & H_FLIP)
+ if (Tile & H_FLIP) // Horizontal flip, but what about vertical flip?
{
- // Horizontal flip, but what about vertical flip ?
- if (Tile & V_FLIP)
- {
- // Both horzontal & vertical flip
- (*DrawTilePtr)(Tile + t2 + 1 - (Quot & 1), s,
- VirtAlign, Lines);
- }
- else
- {
- // Horizontal flip only
- (*DrawTilePtr)(Tile + t1 + 1 - (Quot & 1), s,
- VirtAlign, Lines);
- }
- }
- else
- {
- // No horizontal flip, but is there a vertical flip ?
- if (Tile & V_FLIP)
- {
- // Vertical flip only
- (*DrawTilePtr)(Tile + t2 + (Quot & 1), s,
- VirtAlign, Lines);
- }
- else
- {
- // Normal unflipped
- (*DrawTilePtr)(Tile + t1 + (Quot & 1), s,
- VirtAlign, Lines);
- }
+ if (Tile & V_FLIP) // Both horzontal & vertical flip
+ (*DrawTilePtr)(Tile + t2 + 1 - (Quot & 1), s, VirtAlign, Lines);
+ else // Horizontal flip only
+ (*DrawTilePtr)(Tile + t1 + 1 - (Quot & 1), s, VirtAlign, Lines);
}
+ else if (Tile & V_FLIP) // Vertical flip only
+ (*DrawTilePtr)(Tile + t2 + (Quot & 1), s, VirtAlign, Lines);
+ else // Normal unflipped
+ (*DrawTilePtr)(Tile + t1 + (Quot & 1), s, VirtAlign, Lines);
}
else
(*DrawTilePtr)(Tile, s, VirtAlign, Lines);
@@ -2212,40 +1956,20 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
GFX.Z1 = GFX.Z2 = depths [(Tile & 0x2000) >> 13];
if (BG.TileSize == 8)
- (*DrawClippedTilePtr)(Tile, s, 0, Count, VirtAlign,
- Lines);
+ (*DrawClippedTilePtr)(Tile, s, 0, Count, VirtAlign, Lines);
else
{
- if (!(Tile & (V_FLIP | H_FLIP)))
- {
- // Normal, unflipped
- (*DrawClippedTilePtr)(Tile + t1 + (Quot & 1), s, 0,
- Count, VirtAlign, Lines);
- }
+ if (!(Tile & (V_FLIP | H_FLIP))) // Normal, unflipped
+ (*DrawClippedTilePtr)(Tile + t1 + (Quot & 1), s, 0, Count, VirtAlign, Lines);
else if (Tile & H_FLIP)
{
- if (Tile & V_FLIP)
- {
- // H & V flip
- (*DrawClippedTilePtr)(Tile + t2 + 1 - (Quot & 1),
- s, 0, Count, VirtAlign,
- Lines);
- }
- else
- {
- // H flip only
- (*DrawClippedTilePtr)(Tile + t1 + 1 - (Quot & 1),
- s, 0, Count, VirtAlign,
- Lines);
- }
- }
- else
- {
- // V flip only
- (*DrawClippedTilePtr)(Tile + t2 + (Quot & 1),
- s, 0, Count, VirtAlign,
- Lines);
+ if (Tile & V_FLIP) // H & V flip
+ (*DrawClippedTilePtr)(Tile + t2 + 1 - (Quot & 1), s, 0, Count, VirtAlign, Lines);
+ else // H flip only
+ (*DrawClippedTilePtr)(Tile + t1 + 1 - (Quot & 1), s, 0, Count, VirtAlign, Lines);
}
+ else // V flip only
+ (*DrawClippedTilePtr)(Tile + t2 + (Quot & 1), s, 0, Count, VirtAlign, Lines);
}
}
}
@@ -2253,26 +1977,26 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
}
#define RENDER_BACKGROUND_MODE7(TYPE,FUNC) \
- uint32_t clip; \
+ uint32_t clip; \
int32_t aa, cc; \
int32_t dir; \
int32_t startx, endx; \
uint32_t Left = 0; \
uint32_t Right = 256; \
uint32_t ClipCount; \
- uint16_t *ScreenColors = IPPU.ScreenColors; \
- uint8_t *VRAM1; \
+ uint16_t* ScreenColors = IPPU.ScreenColors; \
+ uint8_t* VRAM1; \
uint32_t Line; \
- uint8_t *Depth; \
- SLineMatrixData *l; \
+ uint8_t* Depth; \
+ SLineMatrixData* l; \
(void)ScreenColors; \
\
VRAM1 = Memory.VRAM + 1; \
if (GFX.r2130 & 1) \
{ \
- if (IPPU.DirectColourMapsNeedRebuild) \
+ if (IPPU.DirectColourMapsNeedRebuild) \
S9xBuildDirectColourMaps (); \
- ScreenColors = DirectColourMaps [0]; \
+ ScreenColors = DirectColourMaps [0]; \
} \
\
ClipCount = GFX.pCurrentClip->Count [bg]; \
@@ -2286,30 +2010,30 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
\
for (Line = GFX.StartY; Line <= GFX.EndY; Line++, Screen += GFX.Pitch, Depth += GFX.PPL, l++) \
{ \
- int32_t yy; \
- int32_t BB,DD; \
+ int32_t yy; \
+ int32_t BB,DD; \
\
- int32_t HOffset = ((int32_t) LineData [Line].BG[0].HOffset << M7) >> M7; \
- int32_t VOffset = ((int32_t) LineData [Line].BG[0].VOffset << M7) >> M7; \
+ int32_t HOffset = ((int32_t) LineData [Line].BG[0].HOffset << M7) >> M7; \
+ int32_t VOffset = ((int32_t) LineData [Line].BG[0].VOffset << M7) >> M7; \
\
- int32_t CentreX = ((int32_t) l->CentreX << M7) >> M7; \
- int32_t CentreY = ((int32_t) l->CentreY << M7) >> M7; \
+ int32_t CentreX = ((int32_t) l->CentreX << M7) >> M7; \
+ int32_t CentreY = ((int32_t) l->CentreY << M7) >> M7; \
\
- if (PPU.Mode7VFlip) \
+ if (PPU.Mode7VFlip) \
yy = 255 - (int32_t) Line; \
- else \
+ else \
yy = Line; \
\
yy += CLIP_10_BIT_SIGNED(VOffset - CentreY); \
\
- BB = l->MatrixB * yy + (CentreX << 8); \
- DD = l->MatrixD * yy + (CentreY << 8); \
+ BB = l->MatrixB * yy + (CentreX << 8); \
+ DD = l->MatrixD * yy + (CentreY << 8); \
\
- for (clip = 0; clip < ClipCount; clip++) \
- { \
- TYPE *p; \
- uint8_t *d; \
- int32_t xx, AA, CC; \
+ for (clip = 0; clip < ClipCount; clip++) \
+ { \
+ TYPE *p; \
+ uint8_t *d; \
+ int32_t xx, AA, CC; \
if (GFX.pCurrentClip->Count [bg]) \
{ \
Left = GFX.pCurrentClip->Left [clip][bg]; \
@@ -2400,7 +2124,7 @@ static void DrawBackground(uint32_t BGMode, uint32_t bg, uint8_t Z1, uint8_t Z2)
static void DrawBGMode7Background(uint8_t* Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7(uint8_t, (uint8_t)(b & GFX.Mode7Mask))
+ RENDER_BACKGROUND_MODE7(uint8_t, (uint8_t) (b & GFX.Mode7Mask))
}
static void DrawBGMode7Background16(uint8_t* Screen, int32_t bg)
@@ -2408,48 +2132,24 @@ static void DrawBGMode7Background16(uint8_t* Screen, int32_t bg)
RENDER_BACKGROUND_MODE7(uint16_t, ScreenColors [b & GFX.Mode7Mask]);
}
-static void DrawBGMode7Background16Add(uint8_t* Screen, int32_t bg)
+static void DrawBGMode7Background16Add(uint8_t * Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_ADD(ScreenColors [b & GFX.Mode7Mask],
- p [GFX.Delta]) :
- COLOR_ADD(ScreenColors [b & GFX.Mode7Mask],
- GFX.FixedColour)) :
- ScreenColors [b & GFX.Mode7Mask]);
+ RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_ADD(ScreenColors[b & GFX.Mode7Mask], p[GFX.Delta]) : COLOR_ADD(ScreenColors[b & GFX.Mode7Mask], GFX.FixedColour)) : ScreenColors[b & GFX.Mode7Mask]);
}
-static void DrawBGMode7Background16Add1_2(uint8_t* Screen, int32_t bg)
+static void DrawBGMode7Background16Add1_2(uint8_t * Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_ADD1_2(ScreenColors [b & GFX.Mode7Mask],
- p [GFX.Delta]) :
- COLOR_ADD(ScreenColors [b & GFX.Mode7Mask],
- GFX.FixedColour)) :
- ScreenColors [b & GFX.Mode7Mask]);
+ RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_ADD1_2(ScreenColors[b & GFX.Mode7Mask], p[GFX.Delta]) : COLOR_ADD(ScreenColors[b & GFX.Mode7Mask], GFX.FixedColour)) : ScreenColors[b & GFX.Mode7Mask]);
}
-static void DrawBGMode7Background16Sub(uint8_t* Screen, int32_t bg)
+static void DrawBGMode7Background16Sub(uint8_t * Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_SUB(ScreenColors [b & GFX.Mode7Mask],
- p [GFX.Delta]) :
- COLOR_SUB(ScreenColors [b & GFX.Mode7Mask],
- GFX.FixedColour)) :
- ScreenColors [b & GFX.Mode7Mask]);
+ RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_SUB(ScreenColors[b & GFX.Mode7Mask], p[GFX.Delta]) : COLOR_SUB(ScreenColors[b & GFX.Mode7Mask], GFX.FixedColour)) : ScreenColors[b & GFX.Mode7Mask]);
}
-static void DrawBGMode7Background16Sub1_2(uint8_t* Screen, int32_t bg)
+static void DrawBGMode7Background16Sub1_2(uint8_t * Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_SUB1_2(ScreenColors [b & GFX.Mode7Mask],
- p [GFX.Delta]) :
- COLOR_SUB(ScreenColors [b & GFX.Mode7Mask],
- GFX.FixedColour)) :
- ScreenColors [b & GFX.Mode7Mask]);
+ RENDER_BACKGROUND_MODE7(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_SUB1_2(ScreenColors[b & GFX.Mode7Mask], p[GFX.Delta]) : COLOR_SUB(ScreenColors[b & GFX.Mode7Mask], GFX.FixedColour)) : ScreenColors[b & GFX.Mode7Mask]);
}
#define RENDER_BACKGROUND_MODE7_i(TYPE,FUNC,COLORFUNC) \
@@ -2842,14 +2542,8 @@ static void DrawBGMode7Background16Sub1_2(uint8_t* Screen, int32_t bg)
static uint32_t Q_INTERPOLATE(uint32_t A, uint32_t B, uint32_t C, uint32_t D)
{
- uint32_t x = ((A >> 2) & HIGH_BITS_SHIFTED_TWO_MASK) +
- ((B >> 2) & HIGH_BITS_SHIFTED_TWO_MASK) +
- ((C >> 2) & HIGH_BITS_SHIFTED_TWO_MASK) +
- ((D >> 2) & HIGH_BITS_SHIFTED_TWO_MASK);
- uint32_t y = (A & TWO_LOW_BITS_MASK) +
- (B & TWO_LOW_BITS_MASK) +
- (C & TWO_LOW_BITS_MASK) +
- (D & TWO_LOW_BITS_MASK);
+ uint32_t x = ((A >> 2) & HIGH_BITS_SHIFTED_TWO_MASK) + ((B >> 2) & HIGH_BITS_SHIFTED_TWO_MASK) + ((C >> 2) & HIGH_BITS_SHIFTED_TWO_MASK) + ((D >> 2) & HIGH_BITS_SHIFTED_TWO_MASK);
+ uint32_t y = (A & TWO_LOW_BITS_MASK) + (B & TWO_LOW_BITS_MASK) + (C & TWO_LOW_BITS_MASK) + (D & TWO_LOW_BITS_MASK);
y = (y >> 2) & TWO_LOW_BITS_MASK;
return x + y;
}
@@ -2861,46 +2555,22 @@ static void DrawBGMode7Background16_i(uint8_t* Screen, int32_t bg)
static void DrawBGMode7Background16Add_i(uint8_t* Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- (COLOR_ADD(theColor,
- p [GFX.Delta])) :
- (COLOR_ADD(theColor,
- GFX.FixedColour))) :
- theColor, (ScreenColors[b & GFX.Mode7Mask]));
+ RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? (COLOR_ADD(theColor, p[GFX.Delta])) : (COLOR_ADD(theColor, GFX.FixedColour))) : theColor, (ScreenColors[b & GFX.Mode7Mask]));
}
static void DrawBGMode7Background16Add1_2_i(uint8_t* Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_ADD1_2(theColor,
- p [GFX.Delta]) :
- COLOR_ADD(theColor,
- GFX.FixedColour)) :
- theColor, (ScreenColors[b & GFX.Mode7Mask]));
+ RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_ADD1_2(theColor, p[GFX.Delta]) : COLOR_ADD(theColor, GFX.FixedColour)) : theColor, (ScreenColors[b & GFX.Mode7Mask]));
}
static void DrawBGMode7Background16Sub_i(uint8_t* Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_SUB(theColor,
- p [GFX.Delta]) :
- COLOR_SUB(theColor,
- GFX.FixedColour)) :
- theColor, (ScreenColors[b & GFX.Mode7Mask]));
+ RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_SUB(theColor, p[GFX.Delta]) : COLOR_SUB(theColor, GFX.FixedColour)) : theColor, (ScreenColors[b & GFX.Mode7Mask]));
}
static void DrawBGMode7Background16Sub1_2_i(uint8_t* Screen, int32_t bg)
{
- RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ?
- (*(d + GFX.DepthDelta) != 1 ?
- COLOR_SUB1_2(theColor,
- p [GFX.Delta]) :
- COLOR_SUB(theColor,
- GFX.FixedColour)) :
- theColor, (ScreenColors[b & GFX.Mode7Mask]));
+ RENDER_BACKGROUND_MODE7_i(uint16_t, *(d + GFX.DepthDelta) ? (*(d + GFX.DepthDelta) != 1 ? COLOR_SUB1_2(theColor, p[GFX.Delta]) : COLOR_SUB(theColor, GFX.FixedColour)) : theColor, (ScreenColors[b & GFX.Mode7Mask]));
}
static void RenderScreen(uint8_t* Screen, bool sub, bool force_no_add, uint8_t D)
@@ -2956,8 +2626,7 @@ static void RenderScreen(uint8_t* Screen, bool sub, bool force_no_add, uint8_t D
if (BG2)
{
SelectTileRenderer(sub || !SUB_OR_ADD(2));
- DrawBackground(PPU.BGMode, 2, D + 3,
- PPU.BG3Priority ? D + 17 : D + 6);
+ DrawBackground(PPU.BGMode, 2, D + 3, PPU.BG3Priority ? D + 17 : D + 6);
}
if (BG3 && PPU.BGMode == 0)
{
@@ -3013,69 +2682,40 @@ static void RenderScreen(uint8_t* Screen, bool sub, bool force_no_add, uint8_t D
bg = 0;
}
if (sub || !SUB_OR_ADD(0))
- {
- if (!Settings.Mode7Interpolate)
- DrawBGMode7Background16(Screen, bg);
- else
- DrawBGMode7Background16_i(Screen, bg);
- }
+ DrawBGMode7Background16(Screen, bg);
else
{
if (GFX.r2131 & 0x80)
{
if (GFX.r2131 & 0x40)
- {
- if (!Settings.Mode7Interpolate)
- DrawBGMode7Background16Sub1_2(Screen, bg);
- else
- DrawBGMode7Background16Sub1_2_i(Screen, bg);
- }
+ DrawBGMode7Background16Sub1_2(Screen, bg);
else
- {
- if (!Settings.Mode7Interpolate)
- DrawBGMode7Background16Sub(Screen, bg);
- else
- DrawBGMode7Background16Sub_i(Screen, bg);
- }
+ DrawBGMode7Background16Sub(Screen, bg);
}
else
{
if (GFX.r2131 & 0x40)
- {
- if (!Settings.Mode7Interpolate)
- DrawBGMode7Background16Add1_2(Screen, bg);
- else
- DrawBGMode7Background16Add1_2_i(Screen, bg);
- }
+ DrawBGMode7Background16Add1_2(Screen, bg);
else
- {
- if (!Settings.Mode7Interpolate)
- DrawBGMode7Background16Add(Screen, bg);
- else
- DrawBGMode7Background16Add_i(Screen, bg);
+ DrawBGMode7Background16Add(Screen, bg);
}
}
}
- }
- break;
- default:
- break;
+ break;
+ default:
+ break;
}
}
-void S9xUpdateScreen()
+void S9xUpdateScreen(void)
{
int32_t x2 = 1;
-
GFX.S = GFX.Screen;
GFX.r2131 = Memory.FillRAM [0x2131];
GFX.r212c = Memory.FillRAM [0x212c];
GFX.r212d = Memory.FillRAM [0x212d];
GFX.r2130 = Memory.FillRAM [0x2130];
-
- GFX.Pseudo = (Memory.FillRAM [0x2133] & 8) != 0 &&
- (GFX.r212c & 15) != (GFX.r212d & 15) &&
- (GFX.r2131 & 0x3f) == 0;
+ GFX.Pseudo = (Memory.FillRAM [0x2133] & 8) != 0 && (GFX.r212c & 15) != (GFX.r212d & 15) && (GFX.r2131 & 0x3f) == 0;
if (IPPU.OBJChanged)
S9xSetupOBJ();
@@ -3146,238 +2786,170 @@ void S9xUpdateScreen()
{
// memmove converted: Same malloc, different addresses, and identical addresses at line 0 [Neb]
// DS2 DMA notes: This code path is unused [Neb]
- memcpy(GFX.Screen + y * 2 * GFX.Pitch2,
- GFX.Screen + y * GFX.Pitch2,
- GFX.Pitch2);
+ memcpy(GFX.Screen + y * 2 * GFX.Pitch2, GFX.Screen + y * GFX.Pitch2, GFX.Pitch2);
// memmove converted: Same malloc, different addresses [Neb]
- memcpy(GFX.Screen + (y * 2 + 1) * GFX.Pitch2,
- GFX.Screen + y * GFX.Pitch2,
- GFX.Pitch2);
+ memcpy(GFX.Screen + (y * 2 + 1) * GFX.Pitch2, GFX.Screen + y * GFX.Pitch2, GFX.Pitch2);
}
}
}
uint32_t black = BLACK | (BLACK << 16);
- if (Settings.Transparency)
+ if (GFX.Pseudo)
{
- if (GFX.Pseudo)
- {
- GFX.r2131 = 0x5f;
- GFX.r212c &= (Memory.FillRAM [0x212d] | 0xf0);
- GFX.r212d |= (Memory.FillRAM [0x212c] & 0x0f);
- GFX.r2130 |= 2;
- }
+ GFX.r2131 = 0x5f;
+ GFX.r212c &= (Memory.FillRAM [0x212d] | 0xf0);
+ GFX.r212d |= (Memory.FillRAM [0x212c] & 0x0f);
+ GFX.r2130 |= 2;
+ }
- if (!PPU.ForcedBlanking && ADD_OR_SUB_ON_ANYTHING &&
- (GFX.r2130 & 0x30) != 0x30 &&
- !((GFX.r2130 & 0x30) == 0x10 && IPPU.Clip[1].Count[5] == 0))
- {
- ClipData* pClip;
+ if (!PPU.ForcedBlanking && ADD_OR_SUB_ON_ANYTHING && (GFX.r2130 & 0x30) != 0x30 && !((GFX.r2130 & 0x30) == 0x10 && IPPU.Clip[1].Count[5] == 0))
+ {
+ ClipData* pClip;
- GFX.FixedColour = BUILD_PIXEL(IPPU.XB [PPU.FixedColourRed],
- IPPU.XB [PPU.FixedColourGreen],
- IPPU.XB [PPU.FixedColourBlue]);
+ GFX.FixedColour = BUILD_PIXEL(IPPU.XB [PPU.FixedColourRed], IPPU.XB [PPU.FixedColourGreen], IPPU.XB [PPU.FixedColourBlue]);
- // Clear the z-buffer, marking areas 'covered' by the fixed
- // colour as depth 1.
- pClip = &IPPU.Clip [1];
+ // Clear the z-buffer, marking areas 'covered' by the fixed
+ // colour as depth 1.
+ pClip = &IPPU.Clip [1];
- // Clear the z-buffer
- if (pClip->Count [5])
+ // Clear the z-buffer
+ if (pClip->Count [5])
+ {
+ // Colour window enabled.
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
{
- // Colour window enabled.
- uint32_t y;
- for (y = starty; y <= endy; y++)
+ memset(GFX.SubZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
+ memset(GFX.ZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
+
+ if (IPPU.Clip [0].Count [5])
{
- memset(GFX.SubZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
- memset(GFX.ZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
+ uint32_t* p = (uint32_t*)(GFX.SubScreen + y * GFX.Pitch2);
+ uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
+ while (p < q)
+ *p++ = black;
+ }
- if (IPPU.Clip [0].Count [5])
+ uint32_t c;
+ for (c = 0; c < pClip->Count [5]; c++)
+ {
+ if (pClip->Right [c][5] > pClip->Left [c][5])
{
- uint32_t* p = (uint32_t*)(GFX.SubScreen + y * GFX.Pitch2);
- uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
- while (p < q)
- *p++ = black;
- }
+ memset(GFX.SubZBuffer + y * GFX.ZPitch + pClip->Left [c][5] * x2, 1, (pClip->Right [c][5] - pClip->Left [c][5]) * x2);
- uint32_t c;
- for (c = 0; c < pClip->Count [5]; c++)
- {
- if (pClip->Right [c][5] > pClip->Left [c][5])
+ if (IPPU.Clip [0].Count [5])
{
- memset(GFX.SubZBuffer + y * GFX.ZPitch + pClip->Left [c][5] * x2,
- 1, (pClip->Right [c][5] - pClip->Left [c][5]) * x2);
-
- if (IPPU.Clip [0].Count [5])
- {
- // Blast, have to clear the sub-screen to the fixed-colour
- // because there is a colour window in effect clipping
- // the main screen that will allow the sub-screen
- // 'underneath' to show through.
+ // Blast, have to clear the sub-screen to the fixed-colour
+ // because there is a colour window in effect clipping
+ // the main screen that will allow the sub-screen
+ // 'underneath' to show through.
- uint16_t* p = (uint16_t*)(GFX.SubScreen + y * GFX.Pitch2);
- uint16_t* q = p + pClip->Right [c][5] * x2;
- p += pClip->Left [c][5] * x2;
+ uint16_t* p = (uint16_t*)(GFX.SubScreen + y * GFX.Pitch2);
+ uint16_t* q = p + pClip->Right [c][5] * x2;
+ p += pClip->Left [c][5] * x2;
- while (p < q)
- *p++ = (uint16_t) GFX.FixedColour;
- }
+ while (p < q)
+ *p++ = (uint16_t) GFX.FixedColour;
}
}
}
}
- else
+ }
+ else
+ {
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
{
- uint32_t y;
- for (y = starty; y <= endy; y++)
- {
- memset(GFX.ZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
- memset(GFX.SubZBuffer + y * GFX.ZPitch, 1, IPPU.RenderedScreenWidth);
+ memset(GFX.ZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
+ memset(GFX.SubZBuffer + y * GFX.ZPitch, 1, IPPU.RenderedScreenWidth);
- if (IPPU.Clip [0].Count [5])
- {
- // Blast, have to clear the sub-screen to the fixed-colour
- // because there is a colour window in effect clipping
- // the main screen that will allow the sub-screen
- // 'underneath' to show through.
- uint32_t b = GFX.FixedColour | (GFX.FixedColour << 16);
- uint32_t* p = (uint32_t*)(GFX.SubScreen + y * GFX.Pitch2);
- uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
+ if (IPPU.Clip [0].Count [5])
+ {
+ // Blast, have to clear the sub-screen to the fixed-colour
+ // because there is a colour window in effect clipping
+ // the main screen that will allow the sub-screen
+ // 'underneath' to show through.
+ uint32_t b = GFX.FixedColour | (GFX.FixedColour << 16);
+ uint32_t* p = (uint32_t*)(GFX.SubScreen + y * GFX.Pitch2);
+ uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
- while (p < q)
- *p++ = b;
- }
+ while (p < q)
+ *p++ = b;
}
}
+ }
- if (ANYTHING_ON_SUB)
- {
- GFX.DB = GFX.SubZBuffer;
- RenderScreen(GFX.SubScreen, true, true, SUB_SCREEN_DEPTH);
- }
+ if (ANYTHING_ON_SUB)
+ {
+ GFX.DB = GFX.SubZBuffer;
+ RenderScreen(GFX.SubScreen, true, true, SUB_SCREEN_DEPTH);
+ }
- if (IPPU.Clip [0].Count [5])
+ if (IPPU.Clip [0].Count [5])
+ {
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
{
- uint32_t y;
- for (y = starty; y <= endy; y++)
- {
- uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2);
- uint8_t* d = GFX.SubZBuffer + y * GFX.ZPitch;
- uint8_t* e = d + IPPU.RenderedScreenWidth;
+ uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2);
+ uint8_t* d = GFX.SubZBuffer + y * GFX.ZPitch;
+ uint8_t* e = d + IPPU.RenderedScreenWidth;
- while (d < e)
- {
- if (*d > 1)
- *p = *(p + GFX.Delta);
- else
- *p = BLACK;
- d++;
- p++;
- }
+ while (d < e)
+ {
+ if (*d > 1)
+ *p = *(p + GFX.Delta);
+ else
+ *p = BLACK;
+ d++;
+ p++;
}
}
+ }
- GFX.DB = GFX.ZBuffer;
- RenderScreen(GFX.Screen, false, false, MAIN_SCREEN_DEPTH);
+ GFX.DB = GFX.ZBuffer;
+ RenderScreen(GFX.Screen, false, false, MAIN_SCREEN_DEPTH);
+
+ if (SUB_OR_ADD(5))
+ {
+ uint32_t back = IPPU.ScreenColors [0];
+ uint32_t Left = 0;
+ uint32_t Right = 256;
+ uint32_t Count;
- if (SUB_OR_ADD(5))
+ pClip = &IPPU.Clip [0];
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
{
- uint32_t back = IPPU.ScreenColors [0];
- uint32_t Left = 0;
- uint32_t Right = 256;
- uint32_t Count;
+ if (!(Count = pClip->Count [5]))
+ {
+ Left = 0;
+ Right = 256 * x2;
+ Count = 1;
+ }
- pClip = &IPPU.Clip [0];
- uint32_t y;
- for (y = starty; y <= endy; y++)
+ uint32_t b;
+ for (b = 0; b < Count; b++)
{
- if (!(Count = pClip->Count [5]))
+ if (pClip->Count [5])
{
- Left = 0;
- Right = 256 * x2;
- Count = 1;
+ Left = pClip->Left [b][5] * x2;
+ Right = pClip->Right [b][5] * x2;
+ if (Right <= Left)
+ continue;
}
- uint32_t b;
- for (b = 0; b < Count; b++)
+ if (GFX.r2131 & 0x80)
{
- if (pClip->Count [5])
- {
- Left = pClip->Left [b][5] * x2;
- Right = pClip->Right [b][5] * x2;
- if (Right <= Left)
- continue;
- }
-
- if (GFX.r2131 & 0x80)
- {
- if (GFX.r2131 & 0x40)
- {
- /* Subtract, halving the result. */
- uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
- uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
- uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
- uint8_t* e = d + Right;
- uint16_t back_fixed = COLOR_SUB(back, GFX.FixedColour);
-
- d += Left;
- while (d < e)
- {
- if (*d == 0)
- {
- if (*s)
- {
- if (*s != 1)
- *p = COLOR_SUB1_2(back, *(p + GFX.Delta));
- else
- *p = back_fixed;
- }
- else
- *p = (uint16_t) back;
- }
- d++;
- p++;
- s++;
- }
- }
- else
- {
- // Subtract
- uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
- uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
- uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
- uint8_t* e = d + Right;
- uint16_t back_fixed = COLOR_SUB(back, GFX.FixedColour);
-
- d += Left;
- while (d < e)
- {
- if (*d == 0)
- {
- if (*s)
- {
- if (*s != 1)
- *p = COLOR_SUB(back, *(p + GFX.Delta));
- else
- *p = back_fixed;
- }
- else
- *p = (uint16_t) back;
- }
- d++;
- p++;
- s++;
- }
- }
- }
- else if (GFX.r2131 & 0x40)
+ if (GFX.r2131 & 0x40)
{
+ /* Subtract, halving the result. */
uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
uint8_t* e = d + Right;
- uint16_t back_fixed = COLOR_ADD(back, GFX.FixedColour);
+ uint16_t back_fixed = COLOR_SUB(back, GFX.FixedColour);
+
d += Left;
while (d < e)
{
@@ -3386,7 +2958,7 @@ void S9xUpdateScreen()
if (*s)
{
if (*s != 1)
- *p = COLOR_ADD1_2(back, *(p + GFX.Delta));
+ *p = COLOR_SUB1_2(back, *(p + GFX.Delta));
else
*p = back_fixed;
}
@@ -3398,13 +2970,15 @@ void S9xUpdateScreen()
s++;
}
}
- else if (back != 0)
+ else
{
+ // Subtract
uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
- uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
+ uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
uint8_t* e = d + Right;
- uint16_t back_fixed = COLOR_ADD(back, GFX.FixedColour);
+ uint16_t back_fixed = COLOR_SUB(back, GFX.FixedColour);
+
d += Left;
while (d < e)
{
@@ -3413,7 +2987,7 @@ void S9xUpdateScreen()
if (*s)
{
if (*s != 1)
- *p = COLOR_ADD(back, *(p + GFX.Delta));
+ *p = COLOR_SUB(back, *(p + GFX.Delta));
else
*p = back_fixed;
}
@@ -3425,80 +2999,117 @@ void S9xUpdateScreen()
s++;
}
}
- else
+ }
+ else if (GFX.r2131 & 0x40)
+ {
+ uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
+ uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
+ uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
+ uint8_t* e = d + Right;
+ uint16_t back_fixed = COLOR_ADD(back, GFX.FixedColour);
+ d += Left;
+ while (d < e)
+ {
+ if (*d == 0)
+ {
+ if (*s)
+ {
+ if (*s != 1)
+ *p = COLOR_ADD1_2(back, *(p + GFX.Delta));
+ else
+ *p = back_fixed;
+ }
+ else
+ *p = (uint16_t) back;
+ }
+ d++;
+ p++;
+ s++;
+ }
+ }
+ else if (back != 0)
+ {
+ uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
+ uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
+ uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
+ uint8_t* e = d + Right;
+ uint16_t back_fixed = COLOR_ADD(back, GFX.FixedColour);
+ d += Left;
+ while (d < e)
{
- if (!pClip->Count [5])
+ if (*d == 0)
{
- // The backdrop has not been cleared yet - so
- // copy the sub-screen to the main screen
- // or fill it with the back-drop colour if the
- // sub-screen is clear.
- uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
- uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
- uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
- uint8_t* e = d + Right;
- d += Left;
- while (d < e)
+ if (*s)
{
- if (*d == 0)
- {
- if (*s)
- {
- if (*s != 1)
- *p = *(p + GFX.Delta);
- else
- *p = GFX.FixedColour;
- }
- else
- *p = (uint16_t) back;
- }
- d++;
- p++;
- s++;
+ if (*s != 1)
+ *p = COLOR_ADD(back, *(p + GFX.Delta));
+ else
+ *p = back_fixed;
}
+ else
+ *p = (uint16_t) back;
}
+ d++;
+ p++;
+ s++;
}
}
- }
- } // --if (SUB_OR_ADD(5))
- else
- {
- uint32_t y;
- // Subscreen not being added to back
- uint32_t back = IPPU.ScreenColors [0] | (IPPU.ScreenColors [0] << 16);
- pClip = &IPPU.Clip [0];
-
- if (pClip->Count [5])
- {
- for (y = starty; y <= endy; y++)
+ else
{
- uint32_t b;
- for (b = 0; b < pClip->Count [5]; b++)
+ if (!pClip->Count [5])
{
- uint32_t Left = pClip->Left [b][5] * x2;
- uint32_t Right = pClip->Right [b][5] * x2;
+ // The backdrop has not been cleared yet - so
+ // copy the sub-screen to the main screen
+ // or fill it with the back-drop colour if the
+ // sub-screen is clear.
uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
+ uint8_t* s = GFX.SubZBuffer + y * GFX.ZPitch + Left;
uint8_t* e = d + Right;
d += Left;
-
while (d < e)
{
if (*d == 0)
- *p = (int16_t) back;
+ {
+ if (*s)
+ {
+ if (*s != 1)
+ *p = *(p + GFX.Delta);
+ else
+ *p = GFX.FixedColour;
+ }
+ else
+ *p = (uint16_t) back;
+ }
d++;
p++;
+ s++;
}
}
}
}
- else
+ }
+ } // --if (SUB_OR_ADD(5))
+ else
+ {
+ uint32_t y;
+ // Subscreen not being added to back
+ uint32_t back = IPPU.ScreenColors [0] | (IPPU.ScreenColors [0] << 16);
+ pClip = &IPPU.Clip [0];
+
+ if (pClip->Count [5])
+ {
+ for (y = starty; y <= endy; y++)
{
- for (y = starty; y <= endy; y++)
+ uint32_t b;
+ for (b = 0; b < pClip->Count [5]; b++)
{
- uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2);
+ uint32_t Left = pClip->Left [b][5] * x2;
+ uint32_t Right = pClip->Right [b][5] * x2;
+ uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2) + Left;
uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
- uint8_t* e = d + 256 * x2;
+ uint8_t* e = d + Right;
+ d += Left;
while (d < e)
{
@@ -3510,64 +3121,81 @@ void S9xUpdateScreen()
}
}
}
- } //force blanking
- else
- {
- // 16bit and transparency but currently no transparency effects in
- // operation.
-
- uint32_t back = IPPU.ScreenColors [0] | (IPPU.ScreenColors [0] << 16);
-
- if (PPU.ForcedBlanking)
- back = black;
-
- if (IPPU.Clip [0].Count[5])
+ else
{
- uint32_t y;
for (y = starty; y <= endy; y++)
{
- uint32_t* p = (uint32_t*)(GFX.Screen + y * GFX.Pitch2);
- uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
-
- while (p < q)
- *p++ = black;
+ uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2);
+ uint8_t* d = GFX.ZBuffer + y * GFX.ZPitch;
+ uint8_t* e = d + 256 * x2;
- uint32_t c;
- for (c = 0; c < IPPU.Clip [0].Count [5]; c++)
+ while (d < e)
{
- if (IPPU.Clip [0].Right [c][5] > IPPU.Clip [0].Left [c][5])
- {
- uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2);
- uint16_t* q = p + IPPU.Clip [0].Right [c][5] * x2;
- p += IPPU.Clip [0].Left [c][5] * x2;
-
- while (p < q)
- *p++ = (uint16_t) back;
- }
+ if (*d == 0)
+ *p = (int16_t) back;
+ d++;
+ p++;
}
}
}
- else
+ }
+ } //force blanking
+ else
+ {
+ // 16bit and transparency but currently no transparency effects in
+ // operation.
+
+ uint32_t back = IPPU.ScreenColors [0] | (IPPU.ScreenColors [0] << 16);
+
+ if (PPU.ForcedBlanking)
+ back = black;
+
+ if (IPPU.Clip [0].Count[5])
+ {
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
{
- uint32_t y;
- for (y = starty; y <= endy; y++)
+ uint32_t* p = (uint32_t*)(GFX.Screen + y * GFX.Pitch2);
+ uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
+
+ while (p < q)
+ *p++ = black;
+
+ uint32_t c;
+ for (c = 0; c < IPPU.Clip [0].Count [5]; c++)
{
- uint32_t* p = (uint32_t*)(GFX.Screen + y * GFX.Pitch2);
- uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
- while (p < q)
- *p++ = back;
+ if (IPPU.Clip [0].Right [c][5] > IPPU.Clip [0].Left [c][5])
+ {
+ uint16_t* p = (uint16_t*)(GFX.Screen + y * GFX.Pitch2);
+ uint16_t* q = p + IPPU.Clip [0].Right [c][5] * x2;
+ p += IPPU.Clip [0].Left [c][5] * x2;
+
+ while (p < q)
+ *p++ = (uint16_t) back;
+ }
}
}
-
- if (!PPU.ForcedBlanking)
+ }
+ else
+ {
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
{
- uint32_t y;
- for (y = starty; y <= endy; y++)
- memset(GFX.ZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
- GFX.DB = GFX.ZBuffer;
- RenderScreen(GFX.Screen, false, true, SUB_SCREEN_DEPTH);
+ uint32_t* p = (uint32_t*)(GFX.Screen + y * GFX.Pitch2);
+ uint32_t* q = (uint32_t*)((uint16_t*) p + IPPU.RenderedScreenWidth);
+ while (p < q)
+ *p++ = back;
}
}
+
+ if (!PPU.ForcedBlanking)
+ {
+ uint32_t y;
+ for (y = starty; y <= endy; y++)
+ memset(GFX.ZBuffer + y * GFX.ZPitch, 0, IPPU.RenderedScreenWidth);
+ GFX.DB = GFX.ZBuffer;
+ RenderScreen(GFX.Screen, false, true, SUB_SCREEN_DEPTH);
+ }
}
if (PPU.BGMode != 5 && PPU.BGMode != 6 && IPPU.DoubleWidthPixels)
diff --git a/source/gfx.h b/source/gfx.h
index 1b4cc6a..5599527 100644
--- a/source/gfx.h
+++ b/source/gfx.h
@@ -19,7 +19,6 @@ void S9xBuildDirectColourMaps(void);
bool S9xInitGFX(void);
void S9xDeinitGFX(void);
-bool S9xInitUpdate(void);
typedef struct
{
@@ -154,10 +153,7 @@ static INLINE uint16_t COLOR_ADD(uint16_t C1, uint16_t C2)
else if (C2 == 0)
return C1;
else
- return GFX.X2 [(((C1 & RGB_REMOVE_LOW_BITS_MASK) +
- (C2 & RGB_REMOVE_LOW_BITS_MASK)) >> 1) +
- (C1 & C2 & RGB_LOW_BITS_MASK)] |
- ((C1 ^ C2) & RGB_LOW_BITS_MASK);
+ return GFX.X2[(((C1 & RGB_REMOVE_LOW_BITS_MASK) + (C2 & RGB_REMOVE_LOW_BITS_MASK)) >> 1) + (C1 & C2 & RGB_LOW_BITS_MASK)] | ((C1 ^ C2) & RGB_LOW_BITS_MASK);
}
#define COLOR_ADD1_2(C1, C2) \
@@ -175,13 +171,7 @@ static INLINE uint16_t COLOR_ADD(uint16_t C1, uint16_t C2)
GFX.ZERO [(((C1) | RGB_HI_BITS_MASKx2) - \
((C2) & RGB_REMOVE_LOW_BITS_MASK)) >> 1]
-typedef void (*NormalTileRenderer)(uint32_t Tile, int32_t Offset,
- uint32_t StartLine, uint32_t LineCount);
-typedef void (*ClippedTileRenderer)(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount);
-typedef void (*LargePixelRenderer)(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount);
-
+typedef void (*NormalTileRenderer)(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount);
+typedef void (*ClippedTileRenderer)(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount);
+typedef void (*LargePixelRenderer)(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount);
#endif
diff --git a/source/globals.c b/source/globals.c
index 830c6ab..36d0e67 100644
--- a/source/globals.c
+++ b/source/globals.c
@@ -17,7 +17,6 @@
char String[513];
SICPU ICPU;
-
SCPUState CPU;
#ifndef USE_BLARGG_APU
@@ -27,9 +26,7 @@ SSoundData SoundData;
#endif
SSettings Settings;
-
SDSP1 DSP1;
-
SSA1 SA1;
SnesModel M1SNES = {1, 3, 2};
@@ -40,8 +37,6 @@ int32_t OpAddress = 0;
CMemory Memory;
-SSNESGameFixes SNESGameFixes;
-
uint8_t OpenBus = 0;
FxInit_s SuperFX;
@@ -191,15 +186,14 @@ uint32_t TailMask [5] =
uint8_t APUROM [64] =
{
- 0xCD, 0xEF, 0xBD, 0xE8, 0x00, 0xC6, 0x1D, 0xD0, 0xFC, 0x8F, 0xAA, 0xF4, 0x8F,
- 0xBB, 0xF5, 0x78, 0xCC, 0xF4, 0xD0, 0xFB, 0x2F, 0x19, 0xEB, 0xF4, 0xD0, 0xFC,
- 0x7E, 0xF4, 0xD0, 0x0B, 0xE4, 0xF5, 0xCB, 0xF4, 0xD7, 0x00, 0xFC, 0xD0, 0xF3,
- 0xAB, 0x01, 0x10, 0xEF, 0x7E, 0xF4, 0x10, 0xEB, 0xBA, 0xF6, 0xDA, 0x00, 0xBA,
- 0xF4, 0xC4, 0xF4, 0xDD, 0x5D, 0xD0, 0xDB, 0x1F, 0x00, 0x00, 0xC0, 0xFF
+ 0xCD, 0xEF, 0xBD, 0xE8, 0x00, 0xC6, 0x1D, 0xD0, 0xFC, 0x8F, 0xAA, 0xF4, 0x8F, 0xBB, 0xF5, 0x78,
+ 0xCC, 0xF4, 0xD0, 0xFB, 0x2F, 0x19, 0xEB, 0xF4, 0xD0, 0xFC, 0x7E, 0xF4, 0xD0, 0x0B, 0xE4, 0xF5,
+ 0xCB, 0xF4, 0xD7, 0x00, 0xFC, 0xD0, 0xF3, 0xAB, 0x01, 0x10, 0xEF, 0x7E, 0xF4, 0x10, 0xEB, 0xBA,
+ 0xF6, 0xDA, 0x00, 0xBA, 0xF4, 0xC4, 0xF4, 0xDD, 0x5D, 0xD0, 0xDB, 0x1F, 0x00, 0x00, 0xC0, 0xFF
};
// Raw SPC700 instruction cycle lengths
-uint16_t S9xAPUCycleLengths [256] =
+uint8_t S9xAPUCycleLengths [256] =
{
/* 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, a, b, c, d, e, f, */
/* 00 */ 2, 8, 4, 5, 3, 4, 3, 6, 2, 6, 5, 4, 5, 4, 6, 8,
@@ -222,7 +216,7 @@ uint16_t S9xAPUCycleLengths [256] =
// Actual data used by CPU emulation, will be scaled by APUReset routine
// to be relative to the 65c816 instruction lengths.
-uint16_t S9xAPUCycles [256] =
+uint8_t S9xAPUCycles [256] =
{
/* 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, a, b, c, d, e, f, */
/* 00 */ 2, 8, 4, 5, 3, 4, 3, 6, 2, 6, 5, 4, 5, 4, 6, 8,
diff --git a/source/hardware.txt b/source/hardware.txt
deleted file mode 100644
index ec6f2e5..0000000
--- a/source/hardware.txt
+++ /dev/null
@@ -1,502 +0,0 @@
-This document gives a brief description of the known hardware features of the
-SNES giving you some idea what SNES emulation authors are up against.
-
-Quick Overview
---------------
-
-o 65c816 CPU running at up to 3.58MHz.
-
-o SPC700 CPU core running at 2.48MHz with built-in custom sound digital
- signal processor.
-
-o Two custom graphics processors used to produce displays of up to 512x478
- pixels with up to 32768 colours, 128 sprites, scaling, rotation, and mosaic
- effects, scrolling over a virtual screen, transparency, coloured lens and
- window effects and raster effects.
-
-o 128k work RAM, 64k sound CPU RAM and 64k video RAM.
-
-Game packs can include:
-
-o Up to 6MBytes (or more) of ROM containing the game code, graphics and sound
- data.
-
-o Additional, battery-backed RAM (S-RAM) used to save game positions.
-
-o 10.5/21MHz RISC CPU (Super FX) used to implement some 3D games and add
- special effects to games.
-
-o A maths co-processor (DSP1) used by some games with lots of physics
- calculations involved (Pilot Wings) or mainly as a protection device
- (Mario Kart).
-
-o Other custom chips produced by some software companies to help speed up
- games, fit more graphics into a given sized ROM or act as a protection
- device.
-
-Users could buy:
-
-o A five player adapter, allowing up to five people to play at once on games
- that supported it.
-
-o A 2-button mouse, originally supplied with a paint program.
-
-o A light-gun that looked like riffle; it used infra-red to provide wire-less
- communication between the gun and the console unit. About 10 games
- supported it.
-
-o A GameBoy adapter that allowed the owners to play GameBoy games on their
- SNES, some in colour.
-
-o Copier units that plugged into the cartridge slot on the SNES and allowed
- game pak ROMs to be downloaded into RAM on the copier and saved onto
- disk. The game could then be played without the original ROM being present
- - unless the game pack contained additional hardware, or the game ROM code
- tried to detect the copier being present and deliberately crashed the game.
-
-More Detail
------------
-
-65c816
-------
-
-The 65c816 is an 8/16-bit CPU that is basically an enhanced 6502: it even
-has an emulation mode to make it behave almost exactly like a real 6502. No
-doubt Nintendo were hoping to provide a compatibility mode for old NES
-games, but failed.
-
-The CPU features a 24-bit address bus and 8-bit data bus allowing a 16Mb
-address space. It has an accumulator and two index registers, all can be
-switch to either 8 or 16-bit mode.
-
-The address space is broken up into 256 banks, each bank being 64k in size,
-although there are addressing modes to treat the entire address space as one
-continuous block. Bank 0 is special in that the stack, a few addressing
-modes and the interrupt and reset vectors all reside there. The stack
-pointer is 16-bits wide.
-
-The 6502 has an addressing mode called zero-page, where the 1-byte address
-specified in the instruction refers to a location in the first 256 bytes of
-memory, allowing for 2-byte instructions and increased execution speed. The
-65c816 extends this idea by allowing the 'zero-page' to be moved anywhere
-inside bank 0 by use of the 16-bit direct page register.
-
-There are other addressing modes available that use the bank specified in
-the data bank register, again to help reduce code size and speed up
-execution.
-
-Code normally executes in single bank at a time, the current bank number
-being specified by the 8-bit program bank register. There are instructions
-available to call subroutines in other banks or just jump to code in other
-banks.
-
-The 65c816 internally runs at 3.58MHz, but other SNES hardware can temporarily
-slow it down to 2.58MHz or even 1.56MHz when the CPU attempts read from them
-or write to them. In particular, there are a mixture of fast and slow ROMs
-inside game packs, slow ROMs can only be accessed at 2.58MHz.
-
-The 65c816 has direct access to 128k of work RAM plus any additional RAM that
-might be in the game pack. Video RAM and sound RAM cannot be accessed
-directly.
-
-SPC700
-------
-
-The SPC700 is an 8-bit CPU core, similar to a 6502, but with a different
-instruction set, some new addressing modes and multiple and divide
-instructions, together with a custom sound digital signal processor, all
-contained inside one module.
-
-The SPC700 and the 65c816 communicate via 4 bi-directional, 8-bit I/O ports.
-The SPC700 has its own 64k RAM used to store a program downloaded from 65c816
-and sound sample data.
-
-The CPU has a built-in, small, 64 byte ROM used as boot-strap code to
-download a more complex program and sample data from the game ROM via the
-65c816. The ROM can be switched off and replaced with 64 bytes of RAM once
-the boot-strap code has done its work.
-
-The sound DSP can only play compresses sound samples, compressed using a
-custom fixed-ratio compression algorithm that compresses 16 16-bit samples
-into 8 bytes plus a one byte header. The minimum unit of a sample is one
-block. The block header byte contains a shift and filter value (algorithm
-decompression information) plus a last block flag and a loop flag; the loop
-flag is only used if the last block flag is also set.
-
-There are 8 separate sound channels allowing up to 8 samples to be played
-simultaneously. Each sound channel has a left and right volume setting and
-frequency setting. A hardware volume envelope can be defined for each
-channel, and echo effects can be turned on and off individually for each
-channel. The combined echo waveform can be subjected to an 8-tap FIR
-digital filter. The wave output of a channel can be used to modulate the
-frequency of the next sound channel, in numerical order.
-
-The DSP also has a white noise source that can played on a sound channel
-instead of sample data. All 8 channels, and any echo sound data, are mixed
-together and subjected to a left and right master volume control.
-
-The DSP also provides 3 interval timers, the first two running at 8kHz and
-the last at 64kHz; games normally only use one of them to provide a constant
-music playback rate.
-
-Interrupts
-----------
-
-The 65c816 provides two external interrupt sources: IRQ, which can be masked,
-and NMI, which cannot.
-
-The IRQ line is connected to an output on one of the graphics chips, that,
-it turn can be programmed to generate an IRQ at the start of a scanline, at
-a particular position on a scanline or at a particular position of every
-scanline. The IRQ line also is connected to one of the pins on the ROM
-connector, so additional hardware inside the ROM game pak, such as the Super FX
-and SA-1 chips, can also generate interrupts.
-
-The NMI line is connected to another output of one of the graphics chips and
-it can be programmed to generate an interrupt when the vertical blank period
-starts.
-
-The SPC700 chip can also generate interrupts, but they are not used on the
-SNES and probably physically not connected.
-
-Joypad Reading
---------------
-
-Data from the SNES joy-pads is sent serially between the pad and the console.
-Games can choose to read each bit in, one at a time or allow hardware inside
-one of the custom chips to automatically read the joy-pad values once every
-frame. The game can then read the values from registers.
-
-SNES joy-pads themselves have direction controls and 8 other buttons named
-A, B, X, Y, Left, Right, Start and Select.
-
-Colour Palette
---------------
-
-The SNES has a 256 entry 15-bit colour palette, allowing for 256 colours on
-screen out of a total palette of 32768. However, games can and do change
-colour entries during a frame, this combined with hardware colour value
-addition and subtraction and an overall brightness setting, can easily boost
-the number of colours on screen to several thousand!
-
-Tiles
------
-
-All SNES graphics data is made up of tiles, a tile being an 8x8 block of
-pixels, with each pixel made up of 2, 4 or 8 bits, allowing for 4, 16 or 256
-colours per pixel.
-
-To complicate matters, the SNES hardware stores tile data in planar format,
-that is all the bit 1's of all the pixels of a tile are stored together,
-then all the bit 2's and so on. Its like a sequence of 1-bit deep 8x8 pixel
-blocks.
-
-When a tile is used as background data, a 3-bit palette start address is
-associated with each tile, allowing the programmer to choose a different
-block of colours for each tile from the larger SNES colour palette. Sprites
-can only use tiles of depth 4 (16 colours), but each sprite has a palette
-start address.
-
-Background Graphic Modes
-------------------------
-
-The SNES has eight background graphics modes, each mode varies the number of
-individual background layers available, the depth of each layer and what
-other features are available. Programmers can change the background mode
-during a frame.
-
-The two most commonly used modes are mode 1, which allows two 16-colour
-background layers and one 4-colour layer and mode 7, which allows one 256-
-colour layer, but the layer can be rotated, stretched, squashed, sheared and
-generally messed around with.
-
-Each background is made up 32x32 8x8 tiles. However, the number of tiles in
-each direction can be double as can the individual tile size. This allows for
-a virtual background size of 256x256 up to 1024x1024. Switching to 16x16 tile
-size actually just groups four 8x8 tiles together, there is no true 16x16 tile
-on the SNES.
-
-Backgrounds have a pre-defined priority order - when pixels from one
-background layer overlap on the screen pixels from another background layer,
-the pixels from the lower numbered background are displayed.
-
-Each tile can be flipped horizontally or vertically, has a 3-bit "palette
-number" and a priority bit. The priority bit is used to make all the pixels
-in the tile appear in front of pixels from another background layer or
-sprite that would otherwise normally appear in front of them. Colour 0 from
-each tile is special and means "transparent", allowing non-transparent
-pixels from background layers or sprites "underneath" to be visible.
-
-Normally only 256x224 or 256x239 pixels are visible on screen and backgrounds
-have a scroll setting that allows the screen to act as a window onto any
-portion of their virtual size.
-
-Two background modes are available that can display up to 512x478 pixels,
-but they're not used by many games because the flicker, caused by the
-interlace used display the image on a standard television, would give game
-players headaches.
-
-Sprites
--------
-
-The SNES has 128 hardware sprites, each sprite can be made up of one or
-several 16-colour, 8x8 tiles. Each sprite is assigned a number which defines
-its pixel priority when two sprites overlap on screen, it also has a separate
-sprite-to-background priority value which defines whether the sprite should
-appear in front or behind of the various background layers. Each sprite also
-has a 3-bit palette number, horizontal and vertical flip flags, a start tile
-number and, of course, an X and Y position.
-
-There's no way to turn off a sprite - if you don't want it to be visible you
-have to place the sprite at off-screen position.
-
-The SNES hardware seems to impose limits on the number of sprites that can
-appear on each scanline; there are one or two games out there that rely on
-this 'feature' to hide sprites they don't want visible.
-
-Mosaic
-------
-
-The SNES has a hardware mosaic effect. The upper left-hand pixel from a block
-of pixels up to 16 pixels wide can be made to cover the area of the other
-15; the pixel appears up to 16 times its original size. The effect can only
-be used on the background layers, not sprites. All backgrounds share the
-same size setting but the effect can be turned on and off per background
-layer.
-
-Many ROMs combine the mosaic effect with a brightness fade to zoom out of one
-game screen then zoom on to the next.
-
-Offset Per Tile
----------------
-
-Three of the background screen modes reduce the number of visible background
-layers by one, and use its screen data as per-tile background scroll data
-for the remaining visible layers.
-
-The background modes vary as to whether the vertical and horizontal scroll
-values can both be altered, or just one of them. Tetris Attack uses the
-effect to allow different parts of the screen to scroll vertically at
-different rates.
-
-The horizontal per-tile offset feature is very limited and only allows
-adjustment in steps of 8 pixels.
-
-Mode 7
-------
-
-Nintendo use this background screen mode to really show off the SNES compared
-to the Sega's Mega Drive.
-
-By specifying a centre of rotation and a 2 by 2 transformation matrix, the
-mode 7 screen can be rotated, scaled, stretched, squashed, etc. just by
-writing to a few PPU registers. By varying the values on each scanline, some
-very interesting effects can be produced, these include a perspective
-effect, shears, split-screen zooms, etc.
-
-Each pixel is 8-bit (256 colours per pixel) and the screen itself has a
-virtual screen size of 1024x1024.
-
-Mode 7 has another feature where the number of colours are reduced to 128
-and the spare bit is used to swap a pixel between background layer 0 and 1,
-thus altering the sprite to background priority. This allows some pixels to
-be appear in front of sprites and others appear behind.
-
-Colour Addition / Subtraction
------------------------------
-
-The pixels of background layers and sprites can be directed to one of two
-places, the main-screen or the sub-screen. The sub-screen is like a virtual
-screen that cannot normally be seen, but the SNES has hardware that can add
-or subtract the RGB colour palette values of each pixel on the sub-screen to
-or from RGB values of pixels on the main-screen. The effect is that
-background layers on the on the main-screen appear translucent, allowing the
-sub-screen partly to show through. Examples are cloud, mist and water effects.
-
-The effect can be turned on and off for each background layer on the
-main-screen. There's a master switch for sprites as well, but when turned
-on, only sprites with certain colour palette numbers actually have their
-pixel values added to or subtracted from.
-
-The SNES also has a separate fixed colour value; if colour addition or
-subtraction is enabled and there's nothing on the sub-screen, the fixed
-colour is added or subtracted instead. I've seen it used by games to darken
-an area of the screen then overlay a menu on top, to implement a
-fade-to-white effect and to tint an area of the screen a particular colour.
-
-Windows
--------
-
-The SNES provides two "clip windows". Each window is just an area defined by
-a left and right position. A background layer or all sprites can be selected
-to appear only inside or outside the window.
-
-If both windows are enabled on the same background layer or for all sprites,
-the areas they define are combined using one of four logical combination
-modes: OR, AND, XOR and N-XOR.
-
-If the left and/or right values are altered on each scanline (normally using
-H-DMA), many different shaped windows can be created; I've seen circles,
-pentagons, wavy lines, doughnuts, G's, etc.
-
-There's also the colour window. Each window or both windows can be used to
-define the area of the colour window. When the colour window is enabled for
-the sub-screen, transparency effects occur only inside or outside the colour
-window. When the colour window is enabled for the main-screen, it acts like
-a master clip window, clipping all background layers and sprites and even
-the back-drop colour to the area either inside or outside the colour window;
-in the clipped areas, the sub-screen is displayed or just black.
-
-Direct Colour Mode
-------------------
-
-On the 256 colour background modes, the otherwise unused 3-bit per-tile
-colour palette number can be used in combination with the 8-bit tile pixel
-data to form an 11-bit colour value (2048 colours) without using the SNES
-colour palette registers.
-
-Mode 7 has the same feature, but since mode 7 uses a different tile layout
-with no 3-bit colour palette number, a fixed 256 colours are available
-instead, again without using the SNES colour palette registers.
-
-Interlace
----------
-
-The SNES normally generates a non-interlaced picture. Interlace can turned
-on and the only thing that happens is that the screen appears to flicker
-slightly due to the way a television works. However, in the two hi-res.
-background screen modes, if interlace is turned on the vertical resolution
-doubles from 512x224 to 512x448 (or 512x478 if the expand vertical flag is
-also set).
-
-Not many games use the feature due to the flicker introduced by the
-interlace, so its use is normally limited to title screens. However, one
-game I know of, RPM Racing, uses the effect during the game.
-
-DMA
----
-
-The SNES provides 8 DMA (direct-memory-access) channels, although only one can
-be active at once. Without any intervention of the 65c816 CPU, up to 64K of
-data can be transferred from RAM or ROM to any PPU (picture processing unit)
-register. Since V-RAM, colour palette and sprite position and display data can
-only be written to via PPU registers, DMA provides a very convenient method of
-transferring data faster than the CPU alone could provide. There are PPU
-registers to read or write to the 128k work RAM, so DMA could be used to copy
-data from ROM to RAM as well.
-
-There is also a DMA read mode, where data is transferred from PPU registers
-to RAM.
-
-There are various limitations on DMA - if multiple DMA channels are started
-at once, they execute in order, the numerically lowest one first, then the
-next highest and so on. The 65c816 is stopped while DMA takes place. Each
-DMA operation can only access one 64k bank at once. However, the biggest
-limitation is that DMA can only take place when the graphics chips aren't
-also performing graphics data DMA, i.e. DMA can only be used during the
-v-blank period or when the screen is forcible blanked.
-
-H-DMA
------
-
-H-DMA is like DMA in that data is transferred from ROM or RAM to PPU
-registers without the intervention of the CPU. However, instead of all the
-data being transferred in one block, a few bytes are transferred at a time,
-just before the start of the each scanline.
-
-H-DMA shares the same channels as normal DMA, so each channel can be set up
-for DMA or H-DMA, but since normal DMA only occurs during v-blank and H-DMA
-is disabled during this time, its actually easy to reuse a channel for both
-types of DMA.
-
-There are various H-DMA modes that define how many bytes should be
-transferred each scanline, whether the destination PPU register is 8-bit or
-16-bit, should new data be transferred each scanline, or can the same data
-be reused if a count value hasn't reached zero, etc.
-
-H-DMA gives a very powerful weapon to programmers, it allows PPU register
-values to be easily changed each scanline, so many games can and do use it
-to change screen colours, background scroll values, window shape values,
-mode 7 matrix values, transparency effects, etc. during the frame.
-
-Extra Chips Used by the SNES Inside Some Game Paks
-==================================================
-
-Super FX
---------
-
-The Super FX is just a fast integer RISC-type processor but with a built-in
-plot instruction that can draw a single pixel in the SNES' planar format into
-a virtual screen very quickly, very handy for 3d polygon rendering. Its a
-strange chip though - no stack, a 512 byte cache and a one stage pipe-line
-that causes the instruction following a branch instruction to be executed.
-Instructions fetched from the cache often execute in a single cycle.
-
-Super FX games came with additional RAM inside the game pak that is used as
-work RAM for the 'FX chip and as save-game positions, if the ROM supports it.
-
-The 'FX chip has 16 16-bit registers and built-in fast integer multiply.
-Although the Super FX and the 65c816 can run in parallel, the 'FX chip can't
-access the game pack ROM or RAM at the same time as the main SNES CPU, so most
-games just get the SNES CPU to execute a wait loop in the SNES work RAM.
-
-The 'FX can't access the SNES custom hardware chips, so if the 'FX has
-rendered a screen image in its work RAM, it has to go to sleep while the SNES
-CPU copies the screen to video RAM, usually using DMA. The SNES CPU can pass
-parameters to 'FX routines either by writing them into the 'FX work RAM or
-writing directly into the 'FX registers, which it can be accessed by the CPU
-only when the 'FX chip is sleeping.
-
-There are two versions of the 'FX chip, the original 10MHz chip used in Star
-Fox and limited to 1Mb of ROM access and 64K RAM and a newer version used in
-Yoshi's Island, Doom, Vortex, Winter Gold, Star Fox 2, etc. which can be
-clocked at 21MHz and can access twice as much ROM and RAM.
-
-DSP1
-----
-
-The DSP1 is an early digital signal processor with an on-board ROM,
-manufactured by NEC. The on-board ROM was loaded with a program developed
-by Nintendo to turn the chip into a 3d maths co-processor, able to perform
-most primitive, but time-consuming, calculations required when manipulating
-objects in a 3d coordinate system relatively quickly, compared to the
-speed of the 65c816 CPU alone, that is.
-
-Most of the calculations supported seemed to be those required by a simple
-flight simulator, i.e. the calculations available were choosen with Pilot
-Wings in mind.
-
-The DSP1 has been used in several other games, may be as many as 20, though
-most ignore a lot of the available features. The games include Mario Kart, Top
-Gear 3000, Battle Racers, Super Air Diver and Bases Loaded 2.
-
-SA-1
-----
-
-The SA-1 is a fast, custom 65c816 8/16-bit processor, the same as inside the
-SNES itself, but clocked at 10MHz compared to a maximum of 3.58MHz for the CPU
-inside the SNES.
-
-The SA-1 isn't just a CPU, it also contains some extra circuits developed by
-Nintendo which includes some very fast RAM, a memory mapper, DMA, several
-real-time timers, and the region lock-out chip.
-
-The SNES (or ROM copiers) can only access the ROM inside the game pak via the
-SA-1; and the SA-1 only enables access to the ROM once its internal region
-lock-out chip has verified it has successfully communicated with a lock-out
-chip inside the SNES. This very effectively prevents SNES ROM copiers from
-being able to copy the ROM.
-
-The SA-1 is used in Mario RPG and seems to be used in several other games
-that Nintendo released in 1996 and beyond.
-
-S-DD1
------
-
-Very little is known about this chip. It seems to be another digital signal
-processor, possibly made by Texas Instruments, dedicated to decompressing
-graphics data. Only two games I know of use the chip, Street Fighter Alpha 2
-and Star Ocean.
-
-Like the SA-1, the SNES and ROM copiers can only access the ROM via the S-DD1,
-again preventing ROM copiers from dumping the ROM image.
diff --git a/source/memmap.c b/source/memmap.c
index 5efeacb..d116330 100644
--- a/source/memmap.c
+++ b/source/memmap.c
@@ -129,8 +129,7 @@ static int32_t ScoreHiROM(bool skip_header, int32_t romoff)
if (Memory.ROM [o + 0xd4] == 0x20)
score += 2;
- if ((Memory.ROM [o + 0xdc] + (Memory.ROM [o + 0xdd] << 8) +
- Memory.ROM [o + 0xde] + (Memory.ROM [o + 0xdf] << 8)) == 0xffff)
+ if ((Memory.ROM [o + 0xdc] + (Memory.ROM [o + 0xdd] << 8) + Memory.ROM [o + 0xde] + (Memory.ROM [o + 0xdf] << 8)) == 0xffff)
{
score += 2;
if (0 != (Memory.ROM [o + 0xde] + (Memory.ROM [o + 0xdf] << 8)))
@@ -171,8 +170,7 @@ static int32_t ScoreLoROM(bool skip_header, int32_t romoff)
if (Memory.ROM [o + 0xd5] == 0x23)
score += 2;
- if ((Memory.ROM [o + 0xdc] + (Memory.ROM [o + 0xdd] << 8) +
- Memory.ROM [o + 0xde] + (Memory.ROM [o + 0xdf] << 8)) == 0xffff)
+ if ((Memory.ROM [o + 0xdc] + (Memory.ROM [o + 0xdd] << 8) + Memory.ROM [o + 0xde] + (Memory.ROM [o + 0xdf] << 8)) == 0xffff)
{
score += 2;
if (0 != (Memory.ROM [o + 0xde] + (Memory.ROM [o + 0xdf] << 8)))
@@ -188,7 +186,7 @@ static int32_t ScoreLoROM(bool skip_header, int32_t romoff)
if (!(Memory.ROM [o + 0xfd] & 0x80))
score -= 6;
if ((Memory.ROM [o + 0xfc] | (Memory.ROM [o + 0xfd] << 8)) > 0xFFB0)
- score -= 2;//reduced per Cowering suggestion
+ score -= 2; //reduced per Cowering suggestion
if ((1 << (Memory.ROM [o + 0xd7] - 7)) > 48)
score -= 1;
if (!AllASCII(&Memory.ROM [o + 0xb0], 6))
@@ -261,10 +259,7 @@ bool S9xInitMemory(void)
IPPU.TileCached [TILE_4BIT] = (uint8_t*) calloc(MAX_4BIT_TILES, 1);
IPPU.TileCached [TILE_8BIT] = (uint8_t*) calloc(MAX_8BIT_TILES, 1);
- if (!Memory.RAM || !Memory.SRAM || !Memory.VRAM || !Memory.ROM || !Memory.BSRAM ||
- !IPPU.TileCache [TILE_2BIT] || !IPPU.TileCache [TILE_4BIT] ||
- !IPPU.TileCache [TILE_8BIT] || !IPPU.TileCached [TILE_2BIT] ||
- !IPPU.TileCached [TILE_4BIT] || !IPPU.TileCached [TILE_8BIT])
+ if (!Memory.RAM || !Memory.SRAM || !Memory.VRAM || !Memory.ROM || !Memory.BSRAM || !IPPU.TileCache [TILE_2BIT] || !IPPU.TileCache [TILE_4BIT] || !IPPU.TileCache [TILE_8BIT] || !IPPU.TileCached [TILE_2BIT] || !IPPU.TileCached [TILE_4BIT] || !IPPU.TileCached [TILE_8BIT])
{
S9xDeinitMemory();
return false;
@@ -272,15 +267,14 @@ bool S9xInitMemory(void)
// FillRAM uses first 32K of ROM image area, otherwise space just
// wasted. Might be read by the SuperFX code.
-
Memory.FillRAM = Memory.ROM;
// Add 0x8000 to ROM image pointer to stop SuperFX code accessing
// unallocated memory (can cause crash on some ports).
- Memory.ROM += 0x8000; // still 32-byte aligned
+ Memory.ROM += 0x8000; // still 32-byte aligned
SuperFX.pvRegisters = &Memory.FillRAM [0x3000];
- SuperFX.nRamBanks = 2; // Most only use 1. 1=64KB, 2=128KB=1024Mb
+ SuperFX.nRamBanks = 2; // Most only use 1. 1 = 64KB, 2 = 128KB = 1024Mb
SuperFX.pvRam = Memory.SRAM;
SuperFX.nRomBanks = (2 * 1024 * 1024) / (32 * 1024);
SuperFX.pvRom = (uint8_t*) Memory.ROM;
@@ -440,10 +434,7 @@ static void CheckForIPSPatch(const char* rom_filename, bool header, int32_t* rom
// Check if ROM image needs to be truncated
ofs = ReadInt(patch_file, 3);
if (ofs != -1 && ofs - offset < *rom_size)
- {
- // Need to truncate ROM image
- *rom_size = ofs - offset;
- }
+ *rom_size = ofs - offset; // Need to truncate ROM image
fclose(patch_file);
return;
@@ -485,13 +476,12 @@ static uint32_t FileLoader(uint8_t* buffer, const char* filename, int32_t maxsiz
do
{
+ int32_t calc_size;
FileSize = fread(ptr, 1, maxsize + 0x200 - (ptr - Memory.ROM), ROMFile);
fclose(ROMFile);
+ calc_size = FileSize & ~0x1FFF; // round to the lower 0x2000
- int32_t calc_size = FileSize & ~0x1FFF; // round to the lower 0x2000
-
- if ((FileSize - calc_size == 512 && !Settings.ForceNoHeader) ||
- Settings.ForceHeader)
+ if ((FileSize - calc_size == 512 && !Settings.ForceNoHeader) || Settings.ForceHeader)
{
// memmove required: Overlapping addresses [Neb]
// DS2 DMA notes: Can be split into 512-byte DMA blocks [Neb]
@@ -519,8 +509,7 @@ static uint32_t FileLoader(uint8_t* buffer, const char* filename, int32_t maxsiz
// check for multi file roms
- if ((ptr - Memory.ROM) < (maxsize + 0x200) &&
- (isdigit(ext [0]) && ext [1] == 0 && ext [0] < '9'))
+ if ((ptr - Memory.ROM) < (maxsize + 0x200) && (isdigit(ext [0]) && ext [1] == 0 && ext [0] < '9'))
{
more = true;
ext [0]++;
@@ -531,11 +520,7 @@ static uint32_t FileLoader(uint8_t* buffer, const char* filename, int32_t maxsiz
#endif
_makepath(fname, drive, dir, name, ext);
}
- else if (ptr - Memory.ROM < maxsize + 0x200 &&
- (((len = strlen(name)) == 7 || len == 8) &&
- strncasecmp(name, "sf", 2) == 0 &&
- isdigit(name [2]) && isdigit(name [3]) && isdigit(name [4]) &&
- isdigit(name [5]) && isalpha(name [len - 1])))
+ else if (ptr - Memory.ROM < maxsize + 0x200 && (((len = strlen(name)) == 7 || len == 8) && strncasecmp(name, "sf", 2) == 0 && isdigit(name [2]) && isdigit(name [3]) && isdigit(name [4]) && isdigit(name [5]) && isalpha(name [len - 1])))
{
more = true;
name [len - 1]++;
@@ -578,11 +563,7 @@ bool LoadROM(
uint8_t* RomHeader = Memory.ROM;
Memory.ExtendedFormat = NOPE;
- if (CleanUp7110 != NULL)
- (*CleanUp7110)();
-
- memset(&SNESGameFixes, 0, sizeof(SNESGameFixes));
- SNESGameFixes.SRAMInitialValue = 0x60;
+ Del7110Gfx();
memset(bytes0x2000, 0, 0x2000);
CPU.TriedInterleavedMode2 = false;
@@ -631,7 +612,8 @@ again:
// SNESAdvance speed hacks (from the speed-hacks branch of CatSFC)
if (strncmp("YOSHI'S ISLAND", (char *) &Memory.ROM[0x7FC0], 14) == 0)
{
- Memory.ROM[0x0000F4] = 0x42; Memory.ROM[0x0000F5] = 0x3B;
+ Memory.ROM[0x0000F4] = 0x42;
+ Memory.ROM[0x0000F5] = 0x3B;
}
else if (strncmp("SUPER MARIOWORLD", (char *) &Memory.ROM[0x7FC0], 16) == 0)
{
@@ -639,33 +621,47 @@ again:
}
else if (strncmp("ALL_STARS + WORLD", (char *) &Memory.ROM[0x7FC0], 17) == 0)
{
- Memory.ROM[0x0003D0] = 0x42; Memory.ROM[0x0003D1] = 0x5B;
- Memory.ROM[0x018522] = 0x42; Memory.ROM[0x018523] = 0x5B;
- Memory.ROM[0x02C804] = 0x42; Memory.ROM[0x02C805] = 0xBA;
- Memory.ROM[0x0683B5] = 0x42; Memory.ROM[0x0683B6] = 0x5B;
- Memory.ROM[0x0696AC] = 0x42; Memory.ROM[0x0696AD] = 0xBA;
- Memory.ROM[0x089233] = 0xDB; Memory.ROM[0x089234] = 0x61;
- Memory.ROM[0x0895DF] = 0x42; Memory.ROM[0x0895E0] = 0x5B;
- Memory.ROM[0x0A7A9D] = 0x42; Memory.ROM[0x0A7A9E] = 0xBA;
- Memory.ROM[0x1072E7] = 0x42; Memory.ROM[0x1072E8] = 0xD9;
- Memory.ROM[0x107355] = 0x42; Memory.ROM[0x107356] = 0x5B;
- Memory.ROM[0x1073CF] = 0x42; Memory.ROM[0x1073D0] = 0x5B;
- Memory.ROM[0x107443] = 0x42; Memory.ROM[0x107444] = 0x5B;
- Memory.ROM[0x107498] = 0x42; Memory.ROM[0x107499] = 0x5B;
- Memory.ROM[0x107505] = 0x42; Memory.ROM[0x107506] = 0x5B;
- Memory.ROM[0x107539] = 0x42; Memory.ROM[0x10753A] = 0x5B;
- Memory.ROM[0x107563] = 0x42; Memory.ROM[0x107564] = 0x5B;
- Memory.ROM[0x18041D] = 0x42; Memory.ROM[0x18041E] = 0x79;
+ Memory.ROM[0x0003D0] = 0x42;
+ Memory.ROM[0x0003D1] = 0x5B;
+ Memory.ROM[0x018522] = 0x42;
+ Memory.ROM[0x018523] = 0x5B;
+ Memory.ROM[0x02C804] = 0x42;
+ Memory.ROM[0x02C805] = 0xBA;
+ Memory.ROM[0x0683B5] = 0x42;
+ Memory.ROM[0x0683B6] = 0x5B;
+ Memory.ROM[0x0696AC] = 0x42;
+ Memory.ROM[0x0696AD] = 0xBA;
+ Memory.ROM[0x089233] = 0xDB;
+ Memory.ROM[0x089234] = 0x61;
+ Memory.ROM[0x0895DF] = 0x42;
+ Memory.ROM[0x0895E0] = 0x5B;
+ Memory.ROM[0x0A7A9D] = 0x42;
+ Memory.ROM[0x0A7A9E] = 0xBA;
+ Memory.ROM[0x1072E7] = 0x42;
+ Memory.ROM[0x1072E8] = 0xD9;
+ Memory.ROM[0x107355] = 0x42;
+ Memory.ROM[0x107356] = 0x5B;
+ Memory.ROM[0x1073CF] = 0x42;
+ Memory.ROM[0x1073D0] = 0x5B;
+ Memory.ROM[0x107443] = 0x42;
+ Memory.ROM[0x107444] = 0x5B;
+ Memory.ROM[0x107498] = 0x42;
+ Memory.ROM[0x107499] = 0x5B;
+ Memory.ROM[0x107505] = 0x42;
+ Memory.ROM[0x107506] = 0x5B;
+ Memory.ROM[0x107539] = 0x42;
+ Memory.ROM[0x10753A] = 0x5B;
+ Memory.ROM[0x107563] = 0x42;
+ Memory.ROM[0x107564] = 0x5B;
+ Memory.ROM[0x18041D] = 0x42;
+ Memory.ROM[0x18041E] = 0x79;
}
#endif
hi_score = ScoreHiROM(true, 0);
lo_score = ScoreLoROM(true, 0);
- if (Memory.HeaderCount == 0 && !Settings.ForceNoHeader &&
- strncmp((char *) &Memory.ROM [0], "BANDAI SFC-ADX", 14) &&
- ((hi_score > lo_score && ScoreHiROM(true, 0) > hi_score) ||
- (hi_score <= lo_score && ScoreLoROM(true, 0) > lo_score)))
+ if (Memory.HeaderCount == 0 && !Settings.ForceNoHeader && strncmp((char *) &Memory.ROM [0], "BANDAI SFC-ADX", 14) && ((hi_score > lo_score && ScoreHiROM(true, 0) > hi_score) || (hi_score <= lo_score && ScoreLoROM(true, 0) > lo_score)))
{
#ifdef DS2_DMA
__dcache_writeback_all();
@@ -695,11 +691,7 @@ again:
//If both vectors are invalid, it's type 1 LoROM
- if (Memory.ExtendedFormat == NOPE &&
- strncmp ((char *) &Memory.ROM [0], "BANDAI SFC-ADX", 14) &&
- ((Memory.ROM[0x7FFC] | (Memory.ROM[0x7FFD] << 8)) < 0x8000) &&
- ((Memory.ROM[0xFFFC] | (Memory.ROM[0xFFFD] << 8)) < 0x8000) &&
- !Settings.ForceInterleaved)
+ if(Memory.ExtendedFormat == NOPE && strncmp((char *) &Memory.ROM[0], "BANDAI SFC-ADX", 14) && ((Memory.ROM[0x7ffc] | (Memory.ROM[0x7ffd] << 8)) < 0x8000) && ((Memory.ROM[0xfffc] | (Memory.ROM[0xFffd] << 8)) < 0x8000) && !Settings.ForceInterleaved)
S9xDeinterleaveType1(TotalFileSize, Memory.ROM);
//CalculatedSize is now set, so rescore
@@ -715,7 +707,6 @@ again:
swappedhirom = ScoreHiROM(false, 0x400000);
//set swapped here.
-
if (MAX(swappedlorom, swappedhirom) >= MAX(loromscore, hiromscore))
{
Memory.ExtendedFormat = BIGFIRST;
@@ -784,29 +775,24 @@ again:
!Settings.ForceC4 &&
!Settings.ForceNoC4 &&
!Settings.ForceSDD1 &&
- !Settings.ForceNoSDD1 &&
- !Settings.ForceInterleaveGD24)
+ !Settings.ForceNoSDD1)
{
/* スーファミターボ BIOS読み込み */
- if ((strncmp((char*) &Memory.ROM [0], "BANDAI SFC-ADX", 14) == 0) &&
- !(strncmp((char*) &Memory.ROM [0x10], "SFC-ADX BACKUP", 14) == 0))
+ if ((strncmp((char*) &Memory.ROM [0], "BANDAI SFC-ADX", 14) == 0) && !(strncmp((char*) &Memory.ROM [0x10], "SFC-ADX BACKUP", 14) == 0))
{
Memory.LoROM = true;
Memory.HiROM = false;
Interleaved = false;
Tales = false;
}
- else if (strncmp ((char *) &Memory.ROM [0x7fc0], "YUYU NO QUIZ DE GO!GO!", 22) == 0 ||
- strncmp ((char *) &Memory.ROM [0x7fc0], "SP MOMOTAROU DENTETSU2", 22) == 0 ||
- strncmp ((char *) &Memory.ROM [0x7fc0], "SUPER FORMATION SOCCE", 21) == 0)
+ else if (strncmp ((char *) &Memory.ROM [0x7fc0], "YUYU NO QUIZ DE GO!GO!", 22) == 0 || strncmp ((char *) &Memory.ROM [0x7fc0], "SP MOMOTAROU DENTETSU2", 22) == 0 || strncmp ((char *) &Memory.ROM [0x7fc0], "SUPER FORMATION SOCCE", 21) == 0)
{
Memory.LoROM = true;
Memory.HiROM = false;
Interleaved = false;
}
/* BS Zooっと麻雀 */
- else if ((strncmp ((char *) &Memory.ROM [0xffc0], "Zooっと麻雀!", 16) == 0)||
- (strncmp ((char *) &Memory.ROM [0xffc0], "Zooっと麻雀!IVT", 15) == 0))
+ else if ((strncmp ((char *) &Memory.ROM [0xffc0], "Zooっと麻雀!", 16) == 0)|| (strncmp ((char *) &Memory.ROM [0xffc0], "Zooっと麻雀!IVT", 15) == 0))
{
Memory.LoROM = false;
Memory.HiROM = true;
@@ -889,7 +875,7 @@ again:
}
else if (Settings.ForceInterleaved2)
S9xDeinterleaveType2(false);
- else if (Settings.ForceInterleaveGD24 && Memory.CalculatedSize == 0x300000)
+ else if (Memory.CalculatedSize == 0x300000)
{
bool t = Memory.LoROM;
@@ -910,8 +896,7 @@ again:
hi_score = ScoreHiROM(false, 0);
lo_score = ScoreLoROM(false, 0);
- if ((Memory.HiROM && (lo_score >= hi_score || hi_score < 0)) ||
- (Memory.LoROM && (hi_score > lo_score || lo_score < 0)))
+ if ((Memory.HiROM && (lo_score >= hi_score || hi_score < 0)) || (Memory.LoROM && (hi_score > lo_score || lo_score < 0)))
{
if (retry_count == 0)
{
@@ -956,8 +941,7 @@ void S9xDeinterleaveType2(bool reset)
for (i = 0; i < nblocks * 2; i++)
{
- blocks [i] = (i & ~0xF) | ((i & 3) << 2) |
- ((i & 12) >> 2);
+ blocks [i] = (i & ~0xF) | ((i & 3) << 2) | ((i & 12) >> 2);
}
#ifdef DS2_DMA
@@ -980,28 +964,20 @@ void S9xDeinterleaveType2(bool reset)
{
uint8_t b;
#ifdef DS2_DMA
- ds2_DMAcopy_32Byte(2 /* channel: emu internal */, tmp,
- &Memory.ROM [blocks [j] * 0x10000], 0x10000);
+ ds2_DMAcopy_32Byte(2 /* channel: emu internal */, tmp, &Memory.ROM [blocks [j] * 0x10000], 0x10000);
ds2_DMA_wait(2);
ds2_DMA_stop(2);
-
- ds2_DMAcopy_32Byte(2 /* channel: emu internal */,
- &Memory.ROM [blocks [j] * 0x10000],
- &Memory.ROM [blocks [i] * 0x10000], 0x10000);
+ ds2_DMAcopy_32Byte(2 /* channel: emu internal */, &Memory.ROM [blocks [j] * 0x10000], &Memory.ROM [blocks [i] * 0x10000], 0x10000);
ds2_DMA_wait(2);
ds2_DMA_stop(2);
-
- ds2_DMAcopy_32Byte(2 /* channel: emu internal */,
- &Memory.ROM [blocks [i] * 0x10000], tmp, 0x10000);
+ ds2_DMAcopy_32Byte(2 /* channel: emu internal */, &Memory.ROM [blocks [i] * 0x10000], tmp, 0x10000);
ds2_DMA_wait(2);
ds2_DMA_stop(2);
#else
// memmove converted: Different mallocs [Neb]
memcpy(tmp, &Memory.ROM [blocks [j] * 0x10000], 0x10000);
-
// memmove converted: Different addresses, or identical if blocks[i] == blocks[j] [Neb]
- memcpy(&Memory.ROM [blocks [j] * 0x10000],
- &Memory.ROM [blocks [i] * 0x10000], 0x10000);
+ memcpy(&Memory.ROM [blocks [j] * 0x10000], &Memory.ROM [blocks [i] * 0x10000], 0x10000);
// memmove converted: Different mallocs [Neb]
memcpy(&Memory.ROM [blocks [i] * 0x10000], tmp, 0x10000);
#endif
@@ -1080,8 +1056,7 @@ void InitROM(bool Interleaved)
ParseSNESHeader(RomHeader);
- //// Detect and initialize chips
- //// detection codes are compatible with NSRT
+ //// Detect and initialize chips - detection codes are compatible with NSRT
// DSP1/2/3/4
if (Memory.ROMType == 0x03)
@@ -1162,8 +1137,7 @@ void InitROM(bool Interleaved)
Settings.SuperFX = !Settings.ForceNoSuperFX;
//OBC1 hack ROM
- if (strncmp(Memory.ROMName, "METAL COMBAT", 12) == 0 &&
- Memory.ROMType == 0x13 && Memory.ROMSpeed == 0x42)
+ if (strncmp(Memory.ROMName, "METAL COMBAT", 12) == 0 && Memory.ROMType == 0x13 && Memory.ROMSpeed == 0x42)
{
Settings.OBC1 = true;
Settings.SuperFX = Settings.ForceSuperFX;
@@ -1177,7 +1151,6 @@ void InitROM(bool Interleaved)
if (((Memory.ROMType & 0xF0) == 0xF0) & ((Memory.ROMSpeed & 0x0F) != 5))
{
Memory.SRAMSize = 2;
- SNESGameFixes.SRAMInitialValue = 0x00;
if ((Memory.ROMType & 0x0F) == 6)
{
if (Memory.ROM[0x7FD7] == 0x09)
@@ -1197,9 +1170,7 @@ void InitROM(bool Interleaved)
Settings.SETA = ST_018;
}
Settings.C4 = Settings.ForceC4;
- if ((Memory.ROMType & 0xf0) == 0xf0 &&
- (strncmp(Memory.ROMName, "MEGAMAN X", 9) == 0 ||
- strncmp(Memory.ROMName, "ROCKMAN X", 9) == 0))
+ if ((Memory.ROMType & 0xf0) == 0xf0 && (strncmp(Memory.ROMName, "MEGAMAN X", 9) == 0 || strncmp(Memory.ROMName, "ROCKMAN X", 9) == 0))
Settings.C4 = !Settings.ForceNoC4;
if (Settings.SETA && Settings.SETA != ST_018)
@@ -1213,9 +1184,7 @@ void InitROM(bool Interleaved)
Settings.C4 = false;
Settings.SDD1 = false;
}
- else if (Settings.ForceSA1 ||
- (!Settings.ForceNoSA1 && (Memory.ROMSpeed & ~0x10) == 0x23 &&
- (Memory.ROMType & 0xf) > 3 && (Memory.ROMType & 0xf0) == 0x30))
+ else if (Settings.ForceSA1 || (!Settings.ForceNoSA1 && (Memory.ROMSpeed & ~0x10) == 0x23 && (Memory.ROMType & 0xf) > 3 && (Memory.ROMType & 0xf0) == 0x30))
{
Settings.SA1 = true;
Settings.DSP1Master = false;
@@ -1227,14 +1196,12 @@ void InitROM(bool Interleaved)
TalesROMMap(Interleaved);
else if (Memory.ExtendedFormat != NOPE)
JumboLoROMMap(Interleaved);
- else if (strncmp((char*) &Memory.ROM [0x7fc0], "SOUND NOVEL-TCOOL", 17) == 0 ||
- strncmp((char*) &Memory.ROM [0x7fc0], "DERBY STALLION 96", 17) == 0)
+ else if (strncmp((char*) &Memory.ROM [0x7fc0], "SOUND NOVEL-TCOOL", 17) == 0 || strncmp((char*) &Memory.ROM [0x7fc0], "DERBY STALLION 96", 17) == 0)
{
LoROM24MBSMap();
Settings.DSP1Master = false;
}
- else if (strncmp((char*) &Memory.ROM [0x7fc0], "THOROUGHBRED BREEDER3", 21) == 0 ||
- strncmp((char*) &Memory.ROM [0x7fc0], "RPG-TCOOL 2", 11) == 0)
+ else if (strncmp((char*) &Memory.ROM [0x7fc0], "THOROUGHBRED BREEDER3", 21) == 0 || strncmp((char*) &Memory.ROM [0x7fc0], "RPG-TCOOL 2", 11) == 0)
{
SRAM512KLoROMMap();
Settings.DSP1Master = false;
@@ -1254,13 +1221,9 @@ void InitROM(bool Interleaved)
Memory.SRAMSize = 5;
SufamiTurboLoROMMap();
}
- else if ((strncmp((char *) &Memory.ROM [0x7fc0], "ROCKMAN X ", 11) == 0)||
- (strncmp((char *) &Memory.ROM [0x7fc0], "MEGAMAN X ", 11) == 0)||
- (strncmp((char *) &Memory.ROM [0x7fc0], "demon's blazon", 14) == 0)||
- (strncmp((char *) &Memory.ROM [0x7fc0], "demon's crest", 13) == 0))
+ else if ((strncmp((char *) &Memory.ROM [0x7fc0], "ROCKMAN X ", 11) == 0)|| (strncmp((char *) &Memory.ROM [0x7fc0], "MEGAMAN X ", 11) == 0)|| (strncmp((char *) &Memory.ROM [0x7fc0], "demon's blazon", 14) == 0)|| (strncmp((char *) &Memory.ROM [0x7fc0], "demon's crest", 13) == 0))
CapcomProtectLoROMMap();
- else if ((Memory.ROMSpeed & ~0x10) == 0x22 &&
- strncmp(Memory.ROMName, "Super Street Fighter", 20) != 0)
+ else if ((Memory.ROMSpeed & ~0x10) == 0x22 && strncmp(Memory.ROMName, "Super Street Fighter", 20) != 0)
AlphaROMMap();
else if (strncmp ((char *) &Memory.ROM [0x7fc0], "HITOMI3", 7) == 0)
{
@@ -1276,7 +1239,7 @@ void InitROM(bool Interleaved)
uint32_t sum1 = 0;
uint32_t sum2 = 0;
- if (0 == Memory.CalculatedChecksum)
+ if (!Memory.CalculatedChecksum)
{
int32_t power2 = 0;
int32_t size = Memory.CalculatedSize;
@@ -1359,7 +1322,7 @@ void InitROM(bool Interleaved)
#ifndef USE_BLARGG_APU
IAPU.OneCycle = ONE_APU_CYCLE;
#endif
- Settings.Shutdown = Settings.ShutdownMaster;
+ Settings.Shutdown = true;
ResetSpeedMap();
ApplyROMFixes();
sprintf(Memory.ROMName, "%s", Safe(Memory.ROMName));
@@ -1368,22 +1331,9 @@ void InitROM(bool Interleaved)
fprintf(stderr,
"\"%s\" [%s] %s, %s, Type: %s, Mode: %s, TV: %s, S-RAM: %s, ROMId: %s Company: %2.2s\n",
- Memory.ROMName,
- (Memory.ROMChecksum + Memory.ROMComplementChecksum != 0xffff ||
- Memory.ROMChecksum != Memory.CalculatedChecksum) ? "bad checksum" : "checksum ok",
- MapType(),
- Size(),
- KartContents(),
- MapMode(),
- TVStandard(),
- StaticRAMSize(),
- Memory.ROMId,
- Memory.CompanyId);
-
- Settings.ForceHeader = Settings.ForceHiROM = Settings.ForceLoROM =
- Settings.ForceInterleaved = Settings.ForceNoHeader =
- Settings.ForceNotInterleaved =
- Settings.ForceInterleaved2 = false;
+ Memory.ROMName, (Memory.ROMChecksum + Memory.ROMComplementChecksum != 0xffff || Memory.ROMChecksum != Memory.CalculatedChecksum) ? "bad checksum" : "checksum ok", MapType(), Size(), KartContents(), MapMode(), TVStandard(), StaticRAMSize(), Memory.ROMId, Memory.CompanyId);
+
+ Settings.ForceHeader = Settings.ForceHiROM = Settings.ForceLoROM = Settings.ForceInterleaved = Settings.ForceNoHeader = Settings.ForceNotInterleaved = Settings.ForceInterleaved2 = false;
}
void FixROMSpeed(void)
@@ -1694,8 +1644,7 @@ void SetaDSPMap(void)
{
for (i = 0; i < 0x08; i++)
{
- //where does the SETA chip access, anyway?
- //please confirm this?
+ // Where does the SETA chip access, anyway? Please confirm this.
Memory.Map[c + 0x80 + i] = (uint8_t*)MAP_SETA_DSP;
Memory.BlockIsROM [c + 0x80 + i] = false;
Memory.BlockIsRAM [c + 0x80 + i] = true;
@@ -1805,14 +1754,12 @@ void TalesROMMap(bool Interleaved)
{
Memory.Map [c + 6] = Memory.Map [c + 0x806] = MAP_HIROM_SRAM_OR_NONE;
Memory.Map [c + 7] = Memory.Map [c + 0x807] = MAP_HIROM_SRAM_OR_NONE;
- Memory.BlockIsRAM [6 + c] = Memory.BlockIsRAM [7 + c] =
- Memory.BlockIsRAM [0x806 + c] = Memory.BlockIsRAM [0x807 + c] = true;
+ Memory.BlockIsRAM [6 + c] = Memory.BlockIsRAM [7 + c] = Memory.BlockIsRAM [0x806 + c] = Memory.BlockIsRAM [0x807 + c] = true;
}
else
{
Memory.Map [c + 6] = Memory.Map [c + 0x806] = (uint8_t*) MAP_NONE;
Memory.Map [c + 7] = Memory.Map [c + 0x807] = (uint8_t*) MAP_NONE;
-
}
for (i = c + 8; i < c + 16; i++)
{
@@ -1855,7 +1802,6 @@ void TalesROMMap(bool Interleaved)
}
Memory.CalculatedChecksum = sum & 0xFFFF;
-
MapRAM();
WriteProtectROM();
}
@@ -1980,10 +1926,8 @@ void SuperFXROMMap(void)
for (c = 0; c < 64; c++)
{
#ifdef DS2_DMA
- ds2_DMAcopy_32Byte(2 /* channel: emu internal */, &ROM [0x200000 + c * 0x10000],
- &ROM [c * 0x8000], 0x8000);
- ds2_DMAcopy_32Byte(3 /* channel: emu internal 2 */,
- &ROM [0x208000 + c * 0x10000], &ROM [c * 0x8000], 0x8000);
+ ds2_DMAcopy_32Byte(2 /* channel: emu internal */, &ROM [0x200000 + c * 0x10000], &ROM [c * 0x8000], 0x8000);
+ ds2_DMAcopy_32Byte(3 /* channel: emu internal 2 */, &ROM [0x208000 + c * 0x10000], &ROM [c * 0x8000], 0x8000);
ds2_DMA_wait(2);
ds2_DMA_wait(3);
ds2_DMA_stop(2);
@@ -2408,7 +2352,6 @@ void JumboLoROMMap(bool Interleaved)
sum += bank[l];
}
Memory.CalculatedChecksum = sum & 0xFFFF;
-
MapRAM();
WriteProtectROM();
}
@@ -2727,38 +2670,15 @@ void ApplyROMFixes(void)
#endif
//Specific game fixes
-
- Settings.StarfoxHack = match_na("STAR FOX") ||
- match_na("STAR WING");
- Settings.WinterGold = match_na("FX SKIING NINTENDO 96") ||
- match_na("DIRT RACER") ||
- Settings.StarfoxHack;
-
- if((match_na("LEGEND") && !Settings.PAL)||
- match_na("King Arthurs World"))
- SNESGameFixes.EchoOnlyOutput = true;
-
+ Settings.StarfoxHack = match_na("STAR FOX") || match_na("STAR WING");
+ Settings.WinterGold = match_na("FX SKIING NINTENDO 96") || match_na("DIRT RACER") || Settings.StarfoxHack;
Settings.HBlankStart = (256 * Settings.H_Max) / SNES_HCOUNTER_MAX;
- //OAM hacks because we don't fully understand the
- //behavior of the SNES.
-
- if (match_na("\xBD\xB0\xCA\xDF\xB0\xCC\xA7\xD0\xBD\xC0") || //Super Famista
- match_na("\xBD\xB0\xCA\xDF\xB0\xCC\xA7\xD0\xBD\xC0 2") || //Super Famista 2
- match_na("ZENKI TENCHIMEIDOU") ||
- match_na("GANBA LEAGUE"))
- SNESGameFixes.APU_OutPorts_ReturnValueFix = true;
- else if (match_na("FURAI NO SIREN"))
- SNESGameFixes.SoundEnvelopeHeightReading2 = true;
-
//CPU timing hacks
Settings.H_Max = (SNES_CYCLES_PER_SCANLINE * Settings.CyclesPercentage) / 100;
// A Couple of HDMA related hacks - Lantus
- if ((match_na("SFX SUPERBUTOUDEN2")) ||
- (match_na("ALIEN vs. PREDATOR")) ||
- (match_na("STONE PROTECTORS")) ||
- (match_na("SUPER BATTLETANK 2")))
+ if ((match_na("SFX SUPERBUTOUDEN2")) || (match_na("ALIEN vs. PREDATOR")) || (match_na("STONE PROTECTORS")) || (match_na("SUPER BATTLETANK 2")))
Settings.H_Max = (SNES_CYCLES_PER_SCANLINE * 130) / 100;
else if (match_na("HOME IMPROVEMENT"))
Settings.H_Max = (SNES_CYCLES_PER_SCANLINE * 200) / 100;
@@ -2776,10 +2696,7 @@ void ApplyROMFixes(void)
else if (match_na("\x0bd\x0da\x0b2\x0d4\x0b0\x0bd\x0de") &&
Settings.CyclesPercentage == 100)
Settings.H_Max = (SNES_CYCLES_PER_SCANLINE * 101) / 100;
- else if (match_na("WILD TRAX") ||
- match_na("STAR FOX 2") ||
- match_na("YOSSY'S ISLAND") ||
- match_na("YOSHI'S ISLAND"))
+ else if (match_na("WILD TRAX") || match_na("STAR FOX 2") || match_na("YOSSY'S ISLAND") || match_na("YOSHI'S ISLAND"))
CPU.TriedInterleavedMode2 = true;
// Start Trek: Deep Sleep 9
else if (strncmp(Memory.ROMId, "A9D", 3) == 0 && Settings.CyclesPercentage == 100)
@@ -2969,15 +2886,8 @@ void ApplyROMFixes(void)
bytes0x2000 [0xb18] = 0x4c;
bytes0x2000 [0xb19] = 0x4b;
bytes0x2000 [0xb1a] = 0xea;
- SNESGameFixes.SRAMInitialValue = 0x6b;
}
- //sram value fixes
- if (match_na("SUPER DRIFT OUT") ||
- match_na("SATAN IS OUR FATHER!") ||
- match_na("goemon 4"))
- SNESGameFixes.SRAMInitialValue = 0x00;
-
if(Settings.BS && Memory.LoROM &&
match_na("F-ZERO") &&
Memory.ROMChecksum == 0xb10d &&
@@ -3010,7 +2920,7 @@ static bool is_bsx(uint8_t *p) // p == "0xFFC0" or "0x7FC0" ROM offset pointer
// Maker ID
c = p[0x1a];
- if ((c != 0x33) && (c != 0xff)) // 0x33 = Manufacturer: Nintendo
+ if ((c != 0x33) && (c != 0xff)) // 0x33 = Manufacturer: Nintendo
return false;
// Month, Day
@@ -3069,16 +2979,12 @@ static bool bs_name(uint8_t* p)
return false;
}
//SJIS single byte char
- else if((*p >= 0x20 && *p <= 0x7f) ||
- (*p >= 0xa0 && *p <= 0xdf))
+ else if((*p >= 0x20 && *p <= 0x7f) || (*p >= 0xa0 && *p <= 0xdf))
p++;
//SJIS multi byte char
else if(lcount >= 2)
{
- if(((*p >= 0x81 && *p <= 0x9f) ||
- (*p >= 0xe0 && *p <= 0xfc)) &&
- ((*(p + 1) >= 0x40 && *(p + 1) <= 0x7e) ||
- (*(p + 1) >= 0x80 && *(p + 1) <= 0xfc)))
+ if(((*p >= 0x81 && *p <= 0x9f) || (*p >= 0xe0 && *p <= 0xfc)) && ((*(p + 1) >= 0x40 && *(p + 1) <= 0x7e) || (*(p + 1) >= 0x80 && *(p + 1) <= 0xfc)))
{
p += 2;
lcount--;
diff --git a/source/memmap.h b/source/memmap.h
index e60cb52..157f4f3 100644
--- a/source/memmap.h
+++ b/source/memmap.h
@@ -88,7 +88,8 @@ enum
MAP_RONLY_SRAM, MAP_OBC_RAM, MAP_SETA_DSP, MAP_SETA_RISC, MAP_LAST
};
-enum {
+enum
+{
MAX_ROM_SIZE = 0x800000
};
@@ -149,5 +150,4 @@ uint8_t* S9xGetMemPointer(uint32_t Address);
uint8_t* GetBasePointer(uint32_t Address);
extern uint8_t OpenBus;
-
#endif // _memmap_h_
diff --git a/source/obc1.c b/source/obc1.c
index 4263284..3ff2f24 100644
--- a/source/obc1.c
+++ b/source/obc1.c
@@ -86,6 +86,5 @@ void ResetOBC1()
OBC1_BasePtr = 0x1c00;
OBC1_Shift = 0;
OBC1_RAM = &Memory.FillRAM[0x6000];
-
memset(OBC1_RAM, 0x00, 0x2000);
}
diff --git a/source/obc1.h b/source/obc1.h
index e781f5b..3d388f8 100644
--- a/source/obc1.h
+++ b/source/obc1.h
@@ -6,6 +6,5 @@
uint8_t GetOBC1(uint16_t Address);
void SetOBC1(uint8_t Byte, uint16_t Address);
uint8_t* GetMemPointerOBC1(uint32_t Address);
-void ResetOBC1();
-
+void ResetOBC1(void);
#endif
diff --git a/source/port.h b/source/port.h
index 3d96509..dfd08f1 100644
--- a/source/port.h
+++ b/source/port.h
@@ -54,8 +54,7 @@ void _splitpath(const char* path, char* drive, char* dir, char* fname, char* ext
#define SLASH_STR "/"
#define SLASH_CHAR '/'
-#if defined(__i386__) || defined(__i486__) || defined(__i586__) || \
- defined(__WIN32__) || defined(__alpha__)
+#if defined(__i386__) || defined(__i486__) || defined(__i586__) || defined(__WIN32__) || defined(__alpha__)
#define FAST_LSB_WORD_ACCESS
#elif defined(__MIPSEL__)
// On little-endian MIPS, a 16-bit word can be read directly from an address
@@ -88,12 +87,12 @@ static INLINE int32_t _isqrt(int32_t val)
remainder -= (squaredbit | root);
root >>= 1;
root |= squaredbit;
- } else
+ }
+ else
root >>= 1;
squaredbit >>= 2;
}
return root;
}
-
#endif
diff --git a/source/ppu.c b/source/ppu.c
index 167610d..4ef83ad 100644
--- a/source/ppu.c
+++ b/source/ppu.c
@@ -40,8 +40,7 @@ void S9xUpdateHTimer()
if (PPU.HTimerEnabled)
{
PPU.HTimerPosition = PPU.IRQHBeamPos * Settings.H_Max / SNES_HCOUNTER_MAX;
- if (PPU.HTimerPosition == Settings.H_Max ||
- PPU.HTimerPosition == Settings.HBlankStart)
+ if (PPU.HTimerPosition == Settings.H_Max || PPU.HTimerPosition == Settings.HBlankStart)
PPU.HTimerPosition--;
if (!PPU.VTimerEnabled || CPU.V_Counter == PPU.IRQVBeamPos)
@@ -49,8 +48,7 @@ void S9xUpdateHTimer()
if (PPU.HTimerPosition < CPU.Cycles)
{
// Missed the IRQ on this line already
- if (CPU.WhichEvent == HBLANK_END_EVENT ||
- CPU.WhichEvent == HTIMER_AFTER_EVENT)
+ if (CPU.WhichEvent == HBLANK_END_EVENT || CPU.WhichEvent == HTIMER_AFTER_EVENT)
{
CPU.WhichEvent = HBLANK_END_EVENT;
CPU.NextEvent = Settings.H_Max;
@@ -63,13 +61,11 @@ void S9xUpdateHTimer()
}
else
{
- if (CPU.WhichEvent == HTIMER_BEFORE_EVENT ||
- CPU.WhichEvent == HBLANK_START_EVENT)
+ if (CPU.WhichEvent == HTIMER_BEFORE_EVENT || CPU.WhichEvent == HBLANK_START_EVENT)
{
if (PPU.HTimerPosition > Settings.HBlankStart)
{
- // HTimer was to trigger before h-blank start,
- // now triggers after start of h-blank
+ // HTimer was to trigger before h-blank start, now triggers after start of h-blank
CPU.NextEvent = Settings.HBlankStart;
CPU.WhichEvent = HBLANK_START_EVENT;
}
@@ -117,21 +113,12 @@ static void S9xSetSuperFX(uint8_t Byte, uint16_t Address)
if ((old_fill_ram ^ Byte) & FLG_G)
{
Memory.FillRAM [Address] = Byte;
- // Go flag has been changed
- if (Byte & FLG_G)
+ if (Byte & FLG_G) // Go flag has been changed
S9xSuperFXExec();
else
FxFlushCache();
}
break;
- case 0x3031:
- case 0x3033:
- case 0x3037:
- case 0x3039:
- case 0x303a:
- case 0x303b:
- case 0x303f:
- break;
case 0x3034:
case 0x3036:
Memory.FillRAM [Address] &= 0x7f;
@@ -147,8 +134,6 @@ static void S9xSetSuperFX(uint8_t Byte, uint16_t Address)
S9xSuperFXExec();
break;
default:
- if (Address >= 0x3100)
- FxCacheWriteAccess(Address);
break;
}
}
@@ -163,8 +148,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
{
switch (Address)
{
- case 0x2100:
- // Brightness and screen blank bit
+ case 0x2100: // Brightness and screen blank bit
if (Byte != Memory.FillRAM [0x2100])
{
FLUSH_REDRAW();
@@ -182,8 +166,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
}
}
break;
- case 0x2101:
- // Sprite (OBJ) tile address
+ case 0x2101: // Sprite (OBJ) tile address
if (Byte != Memory.FillRAM [0x2101])
{
FLUSH_REDRAW();
@@ -193,8 +176,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
IPPU.OBJChanged = true;
}
break;
- case 0x2102:
- // Sprite write address (low)
+ case 0x2102: // Sprite write address (low)
PPU.OAMAddr = ((Memory.FillRAM[0x2103] & 1) << 8) | Byte;
PPU.OAMFlip = 2;
PPU.SavedOAMAddr = PPU.OAMAddr;
@@ -204,9 +186,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
IPPU.OBJChanged = true;
}
break;
- case 0x2103:
- // Sprite register write address (high), sprite priority rotation
- // bit.
+ case 0x2103: // Sprite register write address (high), sprite priority rotation bit.
PPU.OAMAddr = ((Byte & 1) << 8) | Memory.FillRAM[0x2102];
PPU.OAMPriorityRotation = (Byte & 0x80) ? 1 : 0;
@@ -229,13 +209,10 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.OAMFlip = 0;
PPU.SavedOAMAddr = PPU.OAMAddr;
break;
- case 0x2104:
- // Sprite register write
+ case 0x2104: // Sprite register write
REGISTER_2104(Byte);
break;
- case 0x2105:
- // Screen mode (0 - 7), background tile sizes and background 3
- // priority
+ case 0x2105: // Screen mode (0 - 7), background tile sizes and background 3 priority
if (Byte != Memory.FillRAM [0x2105])
{
FLUSH_REDRAW();
@@ -250,8 +227,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
IPPU.Interlace = (bool) (Memory.FillRAM[0x2133] & 1);
}
break;
- case 0x2106:
- // Mosaic pixel size and enable
+ case 0x2106: // Mosaic pixel size and enable
if (Byte != Memory.FillRAM [0x2106])
{
FLUSH_REDRAW();
@@ -262,10 +238,10 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.BGMosaic [3] = (Byte & 8) && PPU.Mosaic > 1;
}
break;
- case 0x2107: // [BG0SC]
- case 0x2108: // [BG1SC]
- case 0x2109: // [BG2SC]
- case 0x210A: // [BG3SC]
+ case 0x2107: // [BG0SC]
+ case 0x2108: // [BG1SC]
+ case 0x2109: // [BG2SC]
+ case 0x210A: // [BG3SC]
if (Byte != Memory.FillRAM [Address])
{
FLUSH_REDRAW();
@@ -273,7 +249,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.BG[Address - 0x2107].SCBase = (Byte & 0x7c) << 8;
}
break;
- case 0x210B: // [BG01NBA]
+ case 0x210B: // [BG01NBA]
if (Byte != Memory.FillRAM [0x210b])
{
FLUSH_REDRAW();
@@ -281,7 +257,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.BG[1].NameBase = ((Byte >> 4) & 7) << 12;
}
break;
- case 0x210C: // [BG23NBA]
+ case 0x210C: // [BG23NBA]
if (Byte != Memory.FillRAM [0x210c])
{
FLUSH_REDRAW();
@@ -321,8 +297,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.BG[3].VOffset = (Byte << 8) | PPU.BGnxOFSbyte;
PPU.BGnxOFSbyte = Byte;
break;
- case 0x2115:
- // VRAM byte/word access flag and increment
+ case 0x2115: // VRAM byte/word access flag and increment
PPU.VMA.High = (bool) (Byte & 0x80);
switch (Byte & 3)
{
@@ -349,14 +324,12 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
else
PPU.VMA.FullGraphicCount = 0;
break;
- case 0x2116:
- // VRAM read/write address (low)
+ case 0x2116: // VRAM read/write address (low)
PPU.VMA.Address &= 0xFF00;
PPU.VMA.Address |= Byte;
IPPU.FirstVRAMRead = true;
break;
- case 0x2117:
- // VRAM read/write address (high)
+ case 0x2117: // VRAM read/write address (high)
PPU.VMA.Address &= 0x00FF;
PPU.VMA.Address |= Byte << 8;
IPPU.FirstVRAMRead = true;
@@ -366,13 +339,11 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
IPPU.FirstVRAMRead = true;
REGISTER_2118(Byte);
break;
- case 0x2119:
- // VRAM write data (high)
+ case 0x2119: // VRAM write data (high)
IPPU.FirstVRAMRead = true;
REGISTER_2119(Byte);
break;
- case 0x211a:
- // Mode 7 outside rotation area display mode and flipping
+ case 0x211a: // Mode 7 outside rotation area display mode and flipping
if (Byte != Memory.FillRAM [0x211a])
{
FLUSH_REDRAW();
@@ -383,34 +354,27 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.Mode7HFlip = (bool) (Byte & 1);
}
break;
- case 0x211b:
- // Mode 7 matrix A (low & high)
+ case 0x211b: // Mode 7 matrix A (low & high)
PPU.MatrixA = ((PPU.MatrixA >> 8) & 0xff) | (Byte << 8);
PPU.Need16x8Multiply = true;
break;
- case 0x211c:
- // Mode 7 matrix B (low & high)
+ case 0x211c: // Mode 7 matrix B (low & high)
PPU.MatrixB = ((PPU.MatrixB >> 8) & 0xff) | (Byte << 8);
PPU.Need16x8Multiply = true;
break;
- case 0x211d:
- // Mode 7 matrix C (low & high)
+ case 0x211d: // Mode 7 matrix C (low & high)
PPU.MatrixC = ((PPU.MatrixC >> 8) & 0xff) | (Byte << 8);
break;
- case 0x211e:
- // Mode 7 matrix D (low & high)
+ case 0x211e: // Mode 7 matrix D (low & high)
PPU.MatrixD = ((PPU.MatrixD >> 8) & 0xff) | (Byte << 8);
break;
- case 0x211f:
- // Mode 7 centre of rotation X (low & high)
+ case 0x211f: // Mode 7 centre of rotation X (low & high)
PPU.CentreX = ((PPU.CentreX >> 8) & 0xff) | (Byte << 8);
break;
- case 0x2120:
- // Mode 7 centre of rotation Y (low & high)
+ case 0x2120: // Mode 7 centre of rotation Y (low & high)
PPU.CentreY = ((PPU.CentreY >> 8) & 0xff) | (Byte << 8);
break;
- case 0x2121:
- // CG-RAM address
+ case 0x2121: // CG-RAM address
PPU.CGFLIP = false;
PPU.CGFLIPRead = false;
PPU.CGADD = Byte;
@@ -418,8 +382,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
case 0x2122:
REGISTER_2122(Byte);
break;
- case 0x2123:
- // Window 1 and 2 enable for backgrounds 1 and 2
+ case 0x2123: // Window 1 and 2 enable for backgrounds 1 and 2
if (Byte != Memory.FillRAM [0x2123])
{
FLUSH_REDRAW();
@@ -434,8 +397,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x2124:
- // Window 1 and 2 enable for backgrounds 3 and 4
+ case 0x2124: // Window 1 and 2 enable for backgrounds 3 and 4
if (Byte != Memory.FillRAM [0x2124])
{
FLUSH_REDRAW();
@@ -450,8 +412,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x2125:
- // Window 1 and 2 enable for objects and colour window
+ case 0x2125: // Window 1 and 2 enable for objects and colour window
if (Byte != Memory.FillRAM [0x2125])
{
FLUSH_REDRAW();
@@ -466,8 +427,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x2126:
- // Window 1 left position
+ case 0x2126: // Window 1 left position
if (Byte != Memory.FillRAM [0x2126])
{
FLUSH_REDRAW();
@@ -475,8 +435,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x2127:
- // Window 1 right position
+ case 0x2127: // Window 1 right position
if (Byte != Memory.FillRAM [0x2127])
{
FLUSH_REDRAW();
@@ -484,8 +443,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x2128:
- // Window 2 left position
+ case 0x2128: // Window 2 left position
if (Byte != Memory.FillRAM [0x2128])
{
FLUSH_REDRAW();
@@ -493,8 +451,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x2129:
- // Window 2 right position
+ case 0x2129: // Window 2 right position
if (Byte != Memory.FillRAM [0x2129])
{
FLUSH_REDRAW();
@@ -502,8 +459,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x212a:
- // Windows 1 & 2 overlap logic for backgrounds 1 - 4
+ case 0x212a: // Windows 1 & 2 overlap logic for backgrounds 1 - 4
if (Byte != Memory.FillRAM [0x212a])
{
FLUSH_REDRAW();
@@ -514,8 +470,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x212b:
- // Windows 1 & 2 overlap logic for objects and colour window
+ case 0x212b: // Windows 1 & 2 overlap logic for objects and colour window
if (Byte != Memory.FillRAM [0x212b])
{
FLUSH_REDRAW();
@@ -524,8 +479,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.RecomputeClipWindows = true;
}
break;
- case 0x212c:
- // Main screen designation (backgrounds 1 - 4 and objects)
+ case 0x212c: // Main screen designation (backgrounds 1 - 4 and objects)
if (Byte != Memory.FillRAM [0x212c])
{
FLUSH_REDRAW();
@@ -534,8 +488,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
return;
}
break;
- case 0x212d:
- // Sub-screen designation (backgrounds 1 - 4 and objects)
+ case 0x212d: // Sub-screen designation (backgrounds 1 - 4 and objects)
if (Byte != Memory.FillRAM [0x212d])
{
FLUSH_REDRAW();
@@ -574,8 +527,7 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
PPU.FixedColourRed = Byte & 0x1f;
}
break;
- case 0x2133:
- // Screen settings
+ case 0x2133: // Screen settings
if (Byte != Memory.FillRAM [0x2133])
{
if (Byte & 0x04)
@@ -601,31 +553,16 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
break;
case 0x2134:
case 0x2135:
- case 0x2136:
- // Matrix 16bit x 8bit multiply result (read-only)
- /* fall through */
- case 0x2137:
- // Software latch for horizontal and vertical timers (read-only)
- /* fall through */
- case 0x2138:
- // OAM read data (read-only)
- /* fall through */
+ case 0x2136: // Matrix 16bit x 8bit multiply result (read-only)
+ case 0x2137: // Software latch for horizontal and vertical timers (read-only)
+ case 0x2138: // OAM read data (read-only)
case 0x2139:
- case 0x213a:
- // VRAM read data (read-only)
- /* fall through */
- case 0x213b:
- // CG-RAM read data (read-only)
- /* fall through */
+ case 0x213a: // VRAM read data (read-only)
+ case 0x213b: // CG-RAM read data (read-only)
case 0x213c:
- case 0x213d:
- // Horizontal and vertical (low/high) read counter (read-only)
- /* fall through */
- case 0x213e:
- // PPU status (time over and range over)
- /* fall through */
- case 0x213f:
- // NTSC/PAL select and field (read-only)
+ case 0x213d: // Horizontal and vertical (low/high) read counter (read-only)
+ case 0x213e: // PPU status (time over and range over)
+ case 0x213f: // NTSC/PAL select and field (read-only)
return;
case 0x2140:
case 0x2141:
@@ -728,15 +665,13 @@ void S9xSetPPU(uint8_t Byte, uint16_t Address)
Memory.FillRAM [Address] = Byte;
return;
}
- else
- // Dai Kaijyu Monogatari II
- if (Address == 0x2801 && Settings.SRTC)
- S9xSetSRTC(Byte, Address);
- else if (Address >= 0x3000 && Address < 0x3300)
- {
- S9xSetSuperFX(Byte, Address);
- return;
- }
+ else if (Address == 0x2801 && Settings.SRTC) // Dai Kaijyu Monogatari II
+ S9xSetSRTC(Byte, Address);
+ else if (Address >= 0x3000 && Address < 0x3300)
+ {
+ S9xSetSuperFX(Byte, Address);
+ return;
+ }
}
Memory.FillRAM[Address] = Byte;
}
@@ -775,8 +710,7 @@ uint8_t S9xGetPPU(uint16_t Address)
return PPU.OpenBus1;
case 0x2134:
case 0x2135:
- case 0x2136:
- // 16bit x 8bit multiply read result.
+ case 0x2136: // 16bit x 8bit multiply read result.
if (PPU.Need16x8Multiply)
{
int32_t r = (int32_t) PPU.MatrixA * (int32_t)(PPU.MatrixB >> 8);
@@ -785,12 +719,11 @@ uint8_t S9xGetPPU(uint16_t Address)
Memory.FillRAM[0x2136] = (uint8_t)(r >> 16);
PPU.Need16x8Multiply = false;
}
- return (PPU.OpenBus1 = Memory.FillRAM[Address]);
+ return PPU.OpenBus1 = Memory.FillRAM[Address];
case 0x2137:
S9xLatchCounters(0);
return OpenBus;
- case 0x2138:
- // Read OAM (sprite) control data
+ case 0x2138: // Read OAM (sprite) control data
if (PPU.OAMAddr & 0x100)
{
if (!(PPU.OAMFlip & 1))
@@ -823,17 +756,14 @@ uint8_t S9xGetPPU(uint16_t Address)
}
PPU.OAMFlip ^= 1;
return (PPU.OpenBus1 = byte);
- case 0x2139:
- // Read vram low byte
+ case 0x2139: // Read vram low byte
if (IPPU.FirstVRAMRead)
byte = Memory.VRAM[(PPU.VMA.Address << 1) & 0xffff];
else if (PPU.VMA.FullGraphicCount)
{
uint32_t addr = PPU.VMA.Address - 1;
uint32_t rem = addr & PPU.VMA.Mask1;
- uint32_t address = (addr & ~PPU.VMA.Mask1) +
- (rem >> PPU.VMA.Shift) +
- ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3);
+ uint32_t address = (addr & ~PPU.VMA.Mask1) + (rem >> PPU.VMA.Shift) + ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3);
byte = Memory.VRAM [((address << 1) - 2) & 0xffff];
}
else
@@ -845,17 +775,14 @@ uint8_t S9xGetPPU(uint16_t Address)
IPPU.FirstVRAMRead = false;
}
return (PPU.OpenBus1 = byte);
- case 0x213A:
- // Read vram high byte
+ case 0x213A: // Read vram high byte
if (IPPU.FirstVRAMRead)
byte = Memory.VRAM[((PPU.VMA.Address << 1) + 1) & 0xffff];
else if (PPU.VMA.FullGraphicCount)
{
uint32_t addr = PPU.VMA.Address - 1;
uint32_t rem = addr & PPU.VMA.Mask1;
- uint32_t address = (addr & ~PPU.VMA.Mask1) +
- (rem >> PPU.VMA.Shift) +
- ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3);
+ uint32_t address = (addr & ~PPU.VMA.Mask1) + (rem >> PPU.VMA.Shift) + ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3);
byte = Memory.VRAM [((address << 1) - 1) & 0xffff];
}
else
@@ -866,8 +793,7 @@ uint8_t S9xGetPPU(uint16_t Address)
IPPU.FirstVRAMRead = false;
}
return (PPU.OpenBus1 = byte);
- case 0x213B:
- // Read palette data
+ case 0x213B: // Read palette data
if (PPU.CGFLIPRead)
byte = (PPU.OpenBus2 & 0x80) | ((PPU.CGDATA[PPU.CGADD++] >> 8) & 0x7f);
else
@@ -875,29 +801,25 @@ uint8_t S9xGetPPU(uint16_t Address)
PPU.CGFLIPRead = !PPU.CGFLIPRead;
return (PPU.OpenBus2 = byte);
- case 0x213C:
- // Horizontal counter value 0-339
+ case 0x213C: // Horizontal counter value 0-339
if (PPU.HBeamFlip)
byte = (PPU.OpenBus2 & 0xfe) | ((PPU.HBeamPosLatched >> 8) & 0x01);
else
byte = (uint8_t)PPU.HBeamPosLatched;
PPU.HBeamFlip ^= 1;
return (PPU.OpenBus2 = byte);
- case 0x213D:
- // Vertical counter value 0-262
+ case 0x213D: // Vertical counter value 0-262
if (PPU.VBeamFlip)
byte = (PPU.OpenBus2 & 0xfe) | ((PPU.VBeamPosLatched >> 8) & 0x01);
else
byte = (uint8_t)PPU.VBeamPosLatched;
PPU.VBeamFlip ^= 1;
return (PPU.OpenBus2 = byte);
- case 0x213E:
- // PPU time and range over flags
+ case 0x213E: // PPU time and range over flags
FLUSH_REDRAW();
byte = (PPU.OpenBus1 & 0x10) | PPU.RangeTimeOver | Model->_5C77;
return (PPU.OpenBus1 = byte);
- case 0x213F:
- // NTSC/PAL and which field flags
+ case 0x213F: // NTSC/PAL and which field flags
PPU.VBeamFlip = PPU.HBeamFlip = 0;
byte = (PPU.OpenBus2 & 0x20) | (Memory.FillRAM[0x213f] & 0xc0) | (Settings.PAL ? 0x10 : 0) | Model->_5C78;
Memory.FillRAM[0x213f] &= ~0x40;
@@ -971,13 +893,7 @@ uint8_t S9xGetPPU(uint16_t Address)
IAPU.WaitCounter++;
if (Settings.APUEnabled)
- {
- if (SNESGameFixes.APU_OutPorts_ReturnValueFix &&
- Address >= 0x2140 && Address <= 0x2143 && !CPU.V_Counter)
- return (uint8_t)((Address & 1) ? ((rand() & 0xff00) >> 8) : (rand() & 0xff));
-
- return (APU.OutPorts [Address & 3]);
- }
+ return APU.OutPorts [Address & 3];
CPU.BranchSkip = true;
@@ -987,23 +903,22 @@ uint8_t S9xGetPPU(uint16_t Address)
if (r & 2)
{
if (r & 4)
- return ((Address & 3) == 1 ? 0xaa : 0xbb);
+ return (Address & 3) == 1 ? 0xaa : 0xbb;
else
- return ((r >> 3) & 0xff);
+ return (r >> 3) & 0xff;
}
}
else
{
int32_t r = rand();
if (r & 2)
- return ((r >> 3) & 0xff);
+ return (r >> 3) & 0xff;
}
- return (Memory.FillRAM[Address]);
+ return Memory.FillRAM[Address];
#else
- return (S9xAPUReadPort(Address & 3));
+ return S9xAPUReadPort(Address & 3);
#endif //#ifndef USE_BLARGG_APU
- case 0x2180:
- // Read WRAM
+ case 0x2180: // Read WRAM
byte = Memory.RAM [PPU.WRAM++];
PPU.WRAM &= 0x1FFFF;
return byte;
@@ -1024,11 +939,11 @@ uint8_t S9xGetPPU(uint16_t Address)
{
case 0x21c2:
if (Model->_5C77 == 2)
- return (0x20);
+ return 0x20;
return OpenBus;
case 0x21c3:
if (Model->_5C77 == 2)
- return (0);
+ return 0;
return OpenBus;
default:
return OpenBus;
@@ -1065,8 +980,7 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
CPU.Cycles += ONE_CYCLE;
switch (Address)
{
- case 0x4016:
- // S9xReset reading of old-style joypads
+ case 0x4016: // S9xReset reading of old-style joypads
if ((byte & 1) && !(Memory.FillRAM [Address] & 1))
{
PPU.Joypad1ButtonReadPos = 0;
@@ -1083,8 +997,7 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
else
switch (Address)
{
- case 0x4200:
- // NMI, V & H IRQ and joypad reading enable flags
+ case 0x4200: // NMI, V & H IRQ and joypad reading enable flags
if (byte & 0x20)
{
if (!PPU.VTimerEnabled)
@@ -1097,9 +1010,7 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
}
}
else
- {
PPU.VTimerEnabled = false;
- }
if (byte & 0x10)
{
@@ -1139,12 +1050,10 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
S9xLatchCounters(1);
Memory.FillRAM[0x4201] = Memory.FillRAM[0x4213] = byte;
break;
- case 0x4202:
- // Multiplier (for multiply)
+ case 0x4202: // Multiplier (for multiply)
break;
- case 0x4203:
+ case 0x4203: // Multiplicand
{
- // Multiplicand
uint32_t res = Memory.FillRAM[0x4202] * byte;
#if defined FAST_LSB_WORD_ACCESS || defined FAST_ALIGNED_LSB_WORD_ACCESS
@@ -1157,8 +1066,7 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
break;
}
case 0x4204:
- case 0x4205:
- // Low and high muliplier (for divide)
+ case 0x4205: // Low and high muliplier (for divide)
break;
case 0x4206:
{
@@ -1205,11 +1113,8 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
{
if (PPU.HTimerEnabled)
S9xUpdateHTimer();
- else
- {
- if (PPU.IRQVBeamPos == CPU.V_Counter)
- S9xSetIRQ(PPU_V_BEAM_IRQ_SOURCE);
- }
+ else if (PPU.IRQVBeamPos == CPU.V_Counter)
+ S9xSetIRQ(PPU_V_BEAM_IRQ_SOURCE);
}
break;
case 0x420A:
@@ -1219,11 +1124,8 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
{
if (PPU.HTimerEnabled)
S9xUpdateHTimer();
- else
- {
- if (PPU.IRQVBeamPos == CPU.V_Counter)
- S9xSetIRQ(PPU_V_BEAM_IRQ_SOURCE);
- }
+ else if (PPU.IRQVBeamPos == CPU.V_Counter)
+ S9xSetIRQ(PPU_V_BEAM_IRQ_SOURCE);
}
break;
case 0x420B:
@@ -1245,13 +1147,10 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
S9xDoDMA(7);
break;
case 0x420C:
- if (Settings.DisableHDMA)
- byte = 0;
Memory.FillRAM[0x420c] = byte;
IPPU.HDMA = byte;
break;
- case 0x420d:
- // Cycle speed 0 - 2.68Mhz, 1 - 3.58Mhz (banks 0x80 +)
+ case 0x420d: // Cycle speed 0 - 2.68Mhz, 1 - 3.58Mhz (banks 0x80 +)
if ((byte & 1) != (Memory.FillRAM [0x420d] & 1))
{
if (byte & 1)
@@ -1263,15 +1162,12 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
}
break;
case 0x420e:
- case 0x420f:
- // --->>> Unknown
+ case 0x420f: // --->>> Unknown
break;
- case 0x4210:
- // NMI ocurred flag (reset on read or write)
+ case 0x4210: // NMI ocurred flag (reset on read or write)
Memory.FillRAM[0x4210] = Model->_5A22;
return;
- case 0x4211:
- // IRQ ocurred flag (reset on read or write)
+ case 0x4211: // IRQ ocurred flag (reset on read or write)
CLEAR_IRQ_SOURCE(PPU_V_BEAM_IRQ_SOURCE | PPU_H_BEAM_IRQ_SOURCE);
break;
case 0x4212: // v-blank, h-blank and joypad being scanned flags (read-only)
@@ -1447,8 +1343,6 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
case 0x437F:
Memory.FillRAM [Address | 0xf] = byte;
break;
- /* fall through */
- //These registers are used by both the S-DD1 and the SPC7110
case 0x4800:
case 0x4801:
case 0x4802:
@@ -1459,13 +1353,12 @@ void S9xSetCPU(uint8_t byte, uint16_t Address)
case 0x4804:
case 0x4805:
case 0x4806:
- case 0x4807:
+ case 0x4807: //These registers are used by both the S-DD1 and the SPC7110
if (Settings.SPC7110)
S9xSetSPC7110(byte, Address);
else
S9xSetSDD1MemoryMap(Address - 0x4804, byte & 7);
break;
- //these are used by the SPC7110
case 0x4808:
case 0x4809:
case 0x480A:
@@ -1540,10 +1433,8 @@ uint8_t S9xGetCPU(uint16_t Address)
{
if (Memory.FillRAM [0x4016] & 1)
{
- // MultiPlayer5 adaptor is only allowed to be plugged into port 2
- if (IPPU.Controller == SNES_MULTIPLAYER5)
+ if (IPPU.Controller == SNES_MULTIPLAYER5) // MultiPlayer5 adaptor is only allowed to be plugged into port 2
return 2;
-
return 0;
}
@@ -1551,17 +1442,15 @@ uint8_t S9xGetCPU(uint16_t Address)
{
if (Memory.FillRAM [0x4201] & 0x80)
{
- byte = ((IPPU.Joypads[1] >> (PPU.Joypad2ButtonReadPos ^ 15)) & 1) |
- (((IPPU.Joypads[2] >> (PPU.Joypad2ButtonReadPos ^ 15)) & 1) << 1);
+ byte = ((IPPU.Joypads[1] >> (PPU.Joypad2ButtonReadPos ^ 15)) & 1) | (((IPPU.Joypads[2] >> (PPU.Joypad2ButtonReadPos ^ 15)) & 1) << 1);
PPU.Joypad2ButtonReadPos++;
- return (byte);
+ return byte;
}
else
{
- byte = ((IPPU.Joypads[3] >> (PPU.Joypad3ButtonReadPos ^ 15)) & 1) |
- (((IPPU.Joypads[4] >> (PPU.Joypad3ButtonReadPos ^ 15)) & 1) << 1);
+ byte = ((IPPU.Joypads[3] >> (PPU.Joypad3ButtonReadPos ^ 15)) & 1) | (((IPPU.Joypads[4] >> (PPU.Joypad3ButtonReadPos ^ 15)) & 1) << 1);
PPU.Joypad3ButtonReadPos++;
- return (byte);
+ return byte;
}
}
else if (IPPU.Controller == SNES_JUSTIFIER || IPPU.Controller == SNES_JUSTIFIER_2)
@@ -1606,28 +1495,20 @@ uint8_t S9xGetCPU(uint16_t Address)
CPU.WaitAddress = CPU.PCAtOpcodeStart;
byte = Memory.FillRAM[0x4210];
Memory.FillRAM[0x4210] = Model->_5A22; //SNEeSe returns 2 for 5A22 version.
- return ((byte & 0x80) | (OpenBus & 0x70) | Model->_5A22);
+ return (byte & 0x80) | (OpenBus & 0x70) | Model->_5A22;
case 0x4211:
byte = (CPU.IRQActive & (PPU_V_BEAM_IRQ_SOURCE | PPU_H_BEAM_IRQ_SOURCE)) ? 0x80 : 0;
CLEAR_IRQ_SOURCE(PPU_V_BEAM_IRQ_SOURCE | PPU_H_BEAM_IRQ_SOURCE);
byte |= OpenBus & 0x7f;
- return (byte);
- case 0x4212:
- // V-blank, h-blank and joypads being read flags (read-only)
+ return byte;
+ case 0x4212: // V-blank, h-blank and joypads being read flags (read-only)
CPU.WaitAddress = CPU.PCAtOpcodeStart;
- return (REGISTER_4212() | (OpenBus & 0x3E));
- case 0x4213:
- // I/O port input - returns 0 wherever $4201 is 0, and 1 elsewhere
- // unless something else pulls it down (i.e. a gun)
- /* fall through */
+ return REGISTER_4212() | (OpenBus & 0x3E);
+ case 0x4213: // I/O port input - returns 0 wherever $4201 is 0, and 1 elsewhere unless something else pulls it down (i.e. a gun)
case 0x4214:
- case 0x4215:
- // Quotient of divide result
+ case 0x4215: // Quotient of divide result
case 0x4216:
- case 0x4217:
- // Multiplcation result (for multiply) or remainder of
- // divison.
- /* fall through */
+ case 0x4217: // Multiplcation result (for multiply) or remainder of divison.
case 0x4218:
case 0x4219:
case 0x421a:
@@ -1635,9 +1516,8 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x421c:
case 0x421d:
case 0x421e:
- case 0x421f:
- // Joypads 1-4 button and direction state.
- return (Memory.FillRAM [Address]);
+ case 0x421f: // Joypads 1-4 button and direction state.
+ return Memory.FillRAM [Address];
case 0x4300:
case 0x4310:
case 0x4320:
@@ -1647,12 +1527,9 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4360:
case 0x4370:
d = (Address >> 4) & 0x7;
- return ((DMA[d].TransferDirection ? 0x80 : 0x00) |
- (DMA[d].HDMAIndirectAddressing ? 0x40 : 0x00) |
- ((uint8_t) Memory.FillRAM [Address]) |
- (DMA[d].AAddressDecrement ? 0x10 : 0x00) |
- (DMA[d].AAddressFixed ? 0x08 : 0x00) |
- (DMA[d].TransferMode & 7));
+ return (DMA[d].TransferDirection ? 0x80 : 0x00) | (DMA[d].HDMAIndirectAddressing ? 0x40 : 0x00) |
+ ((uint8_t) Memory.FillRAM [Address]) | (DMA[d].AAddressDecrement ? 0x10 : 0x00) |
+ (DMA[d].AAddressFixed ? 0x08 : 0x00) | (DMA[d].TransferMode & 7);
case 0x4301:
case 0x4311:
case 0x4321:
@@ -1670,7 +1547,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4352:
case 0x4362:
case 0x4372:
- return (DMA[((Address >> 4) & 0x7)].AAddress & 0xFF);
+ return DMA[((Address >> 4) & 0x7)].AAddress & 0xFF;
case 0x4303:
case 0x4313:
case 0x4323:
@@ -1679,7 +1556,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4353:
case 0x4363:
case 0x4373:
- return (DMA[((Address >> 4) & 0x7)].AAddress >> 8);
+ return DMA[((Address >> 4) & 0x7)].AAddress >> 8;
case 0x4304:
case 0x4314:
case 0x4324:
@@ -1697,7 +1574,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4355:
case 0x4365:
case 0x4375:
- return (DMA[((Address >> 4) & 0x7)].IndirectAddress & 0xff);
+ return DMA[((Address >> 4) & 0x7)].IndirectAddress & 0xff;
case 0x4306:
case 0x4316:
case 0x4326:
@@ -1706,7 +1583,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4356:
case 0x4366:
case 0x4376:
- return (DMA[((Address >> 4) & 0x7)].IndirectAddress >> 8);
+ return DMA[((Address >> 4) & 0x7)].IndirectAddress >> 8;
case 0x4307:
case 0x4317:
case 0x4327:
@@ -1724,7 +1601,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4358:
case 0x4368:
case 0x4378:
- return (DMA[((Address >> 4) & 0x7)].Address & 0xFF);
+ return DMA[((Address >> 4) & 0x7)].Address & 0xFF;
case 0x4309:
case 0x4319:
case 0x4329:
@@ -1733,7 +1610,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x4359:
case 0x4369:
case 0x4379:
- return (DMA[((Address >> 4) & 0x7)].Address >> 8);
+ return DMA[((Address >> 4) & 0x7)].Address >> 8;
case 0x430A:
case 0x431A:
case 0x432A:
@@ -1743,7 +1620,7 @@ uint8_t S9xGetCPU(uint16_t Address)
case 0x436A:
case 0x437A:
d = (Address >> 4) & 0x7;
- return (DMA[d].LineCount ^ (DMA[d].Repeat ? 0x00 : 0x80));
+ return DMA[d].LineCount ^ (DMA[d].Repeat ? 0x00 : 0x80);
case 0x430B:
case 0x431B:
case 0x432B:
@@ -1764,10 +1641,8 @@ uint8_t S9xGetCPU(uint16_t Address)
default:
if (Address >= 0x4800 && Settings.SPC7110)
return S9xGetSPC7110(Address);
-
if (Address >= 0x4800 && Address <= 0x4807 && Settings.SDD1)
return Memory.FillRAM[Address];
-
return OpenBus;
}
}
@@ -1933,13 +1808,8 @@ void S9xResetPPU()
{
if (!Settings.SuperFX)
memset(&Memory.FillRAM [c], c >> 8, 0x100);
- else if ((uint32_t)c < 0x3000 || (uint32_t)c >= 0x3300)
- {
- /* Don't overwrite SFX pvRegisters at 0x3000-0x32FF,
- * they were set in FxReset.
- */
+ else if ((uint32_t) c < 0x3000 || (uint32_t) c >= 0x3300) // Don't overwrite SFX pvRegisters at 0x3000-0x32FF, they were set in FxReset.
memset(&Memory.FillRAM [c], c >> 8, 0x100);
- }
}
memset(&Memory.FillRAM [0x2100], 0, 0x100);
@@ -2033,15 +1903,10 @@ void ProcessSuperScope()
int32_t x, y;
uint32_t buttons;
- if (IPPU.Controller == SNES_SUPERSCOPE &&
- S9xReadSuperScopePosition(&x, &y, &buttons))
+ if (IPPU.Controller == SNES_SUPERSCOPE && S9xReadSuperScopePosition(&x, &y, &buttons))
{
#define SUPERSCOPE_SIGNATURE 0x00ff
- uint32_t scope;
-
- scope = SUPERSCOPE_SIGNATURE | ((buttons & 1) << (7 + 8)) |
- ((buttons & 2) << (5 + 8)) | ((buttons & 4) << (3 + 8)) |
- ((buttons & 8) << (1 + 8));
+ uint32_t scope = SUPERSCOPE_SIGNATURE | ((buttons & 1) << (7 + 8)) | ((buttons & 2) << (5 + 8)) | ((buttons & 4) << (3 + 8)) | ((buttons & 8) << (1 + 8));
if (Memory.FillRAM[0x4201] & 0x80)
{
x += 40;
@@ -2141,8 +2006,7 @@ void S9xUpdateJustifiers()
{
Memory.FillRAM [0x213F] = Model->_5C78;
- //process latch as Justifier 2
- if (Settings.SecondJustifier)
+ if (Settings.SecondJustifier) //process latch as Justifier 2
{
if (IPPU.Controller == SNES_JUSTIFIER_2)
{
@@ -2159,8 +2023,7 @@ void S9xUpdateJustifiers()
{
Memory.FillRAM [0x213F] = Model->_5C78;
- //emulate player 1.
- if (IPPU.Controller == SNES_JUSTIFIER)
+ if (IPPU.Controller == SNES_JUSTIFIER) //emulate player 1.
{
if (!offscreen)
{
@@ -2171,11 +2034,9 @@ void S9xUpdateJustifiers()
}
}
- //needs restructure
- if (!offscreen)
+ if (!offscreen) //needs restructure
{
- if ((!last_p1 && IPPU.Controller == SNES_JUSTIFIER) || (last_p1
- && IPPU.Controller == SNES_JUSTIFIER_2))
+ if ((!last_p1 && IPPU.Controller == SNES_JUSTIFIER) || (last_p1 && IPPU.Controller == SNES_JUSTIFIER_2))
{
PPU.VBeamPosLatched = (uint16_t)(y + 1);
PPU.HBeamPosLatched = (uint16_t) x;
@@ -2203,23 +2064,16 @@ void S9xUpdateJoypads()
IPPU.Joypads [i] &= ~SNES_DOWN_MASK;
}
- // BJ: This is correct behavior AFAICT (used to be Touhaiden hack)
if (IPPU.Controller == SNES_JOYPAD || IPPU.Controller == SNES_MULTIPLAYER5)
- {
for (i = 0; i < 5; i++)
if (IPPU.Joypads [i])
IPPU.Joypads [i] |= 0xffff0000;
- }
- // Read mouse position if enabled
- if (Settings.MouseMaster)
- {
+ if (Settings.MouseMaster) // Read mouse position if enabled
for (i = 0; i < 2; i++)
S9xProcessMouse(i);
- }
- // Read SuperScope if enabled
- if (Settings.SuperScopeMaster)
+ if (Settings.SuperScopeMaster) // Read SuperScope if enabled
ProcessSuperScope();
if (Memory.FillRAM [0x4200] & 1)
diff --git a/source/ppu.h b/source/ppu.h
index ba1a857..fef59d3 100644
--- a/source/ppu.h
+++ b/source/ppu.h
@@ -257,8 +257,7 @@ static INLINE void REGISTER_2104(uint8_t byte)
int32_t addr = ((PPU.OAMAddr & 0x10f) << 1) + (PPU.OAMFlip & 1);
if (byte != PPU.OAMData [addr])
{
- SOBJ* pObj = NULL;
-
+ SOBJ* pObj;
FLUSH_REDRAW();
PPU.OAMData [addr] = byte;
IPPU.OBJChanged = true;
@@ -301,16 +300,13 @@ static INLINE void REGISTER_2104(uint8_t byte)
{
int32_t addr;
uint8_t lowbyte, highbyte;
-
PPU.OAMWriteRegister &= 0x00ff;
lowbyte = (uint8_t)(PPU.OAMWriteRegister);
highbyte = byte;
PPU.OAMWriteRegister |= byte << 8;
-
addr = (PPU.OAMAddr << 1);
- if (lowbyte != PPU.OAMData [addr] ||
- highbyte != PPU.OAMData [addr + 1])
+ if (lowbyte != PPU.OAMData [addr] || highbyte != PPU.OAMData [addr + 1])
{
FLUSH_REDRAW();
PPU.OAMData [addr] = lowbyte;
@@ -355,9 +351,7 @@ static INLINE void REGISTER_2118(uint8_t Byte)
if (PPU.VMA.FullGraphicCount)
{
uint32_t rem = PPU.VMA.Address & PPU.VMA.Mask1;
- address = (((PPU.VMA.Address & ~PPU.VMA.Mask1) +
- (rem >> PPU.VMA.Shift) +
- ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) & 0xffff;
+ address = (((PPU.VMA.Address & ~PPU.VMA.Mask1) + (rem >> PPU.VMA.Shift) + ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) & 0xffff;
Memory.VRAM [address] = Byte;
}
else
@@ -373,9 +367,7 @@ static INLINE void REGISTER_2118_tile(uint8_t Byte)
{
uint32_t address;
uint32_t rem = PPU.VMA.Address & PPU.VMA.Mask1;
- address = (((PPU.VMA.Address & ~PPU.VMA.Mask1) +
- (rem >> PPU.VMA.Shift) +
- ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) & 0xffff;
+ address = (((PPU.VMA.Address & ~PPU.VMA.Mask1) + (rem >> PPU.VMA.Shift) + ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) & 0xffff;
Memory.VRAM [address] = Byte;
IPPU.TileCached [TILE_2BIT][address >> 4] = false;
IPPU.TileCached [TILE_4BIT][address >> 5] = false;
@@ -386,8 +378,8 @@ static INLINE void REGISTER_2118_tile(uint8_t Byte)
static INLINE void REGISTER_2118_linear(uint8_t Byte)
{
- uint32_t address;
- Memory.VRAM[address = (PPU.VMA.Address << 1) & 0xFFFF] = Byte;
+ uint32_t address = (PPU.VMA.Address << 1) & 0xFFFF;
+ Memory.VRAM[address] = Byte;
IPPU.TileCached [TILE_2BIT][address >> 4] = false;
IPPU.TileCached [TILE_4BIT][address >> 5] = false;
IPPU.TileCached [TILE_8BIT][address >> 6] = false;
@@ -401,9 +393,7 @@ static INLINE void REGISTER_2119(uint8_t Byte)
if (PPU.VMA.FullGraphicCount)
{
uint32_t rem = PPU.VMA.Address & PPU.VMA.Mask1;
- address = ((((PPU.VMA.Address & ~PPU.VMA.Mask1) +
- (rem >> PPU.VMA.Shift) +
- ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) + 1) & 0xFFFF;
+ address = ((((PPU.VMA.Address & ~PPU.VMA.Mask1) + (rem >> PPU.VMA.Shift) + ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) + 1) & 0xFFFF;
Memory.VRAM [address] = Byte;
}
else
@@ -418,9 +408,7 @@ static INLINE void REGISTER_2119(uint8_t Byte)
static INLINE void REGISTER_2119_tile(uint8_t Byte)
{
uint32_t rem = PPU.VMA.Address & PPU.VMA.Mask1;
- uint32_t address = ((((PPU.VMA.Address & ~PPU.VMA.Mask1) +
- (rem >> PPU.VMA.Shift) +
- ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) + 1) & 0xFFFF;
+ uint32_t address = ((((PPU.VMA.Address & ~PPU.VMA.Mask1) + (rem >> PPU.VMA.Shift) + ((rem & (PPU.VMA.FullGraphicCount - 1)) << 3)) << 1) + 1) & 0xFFFF;
Memory.VRAM [address] = Byte;
IPPU.TileCached [TILE_2BIT][address >> 4] = false;
IPPU.TileCached [TILE_4BIT][address >> 5] = false;
@@ -452,9 +440,7 @@ static INLINE void REGISTER_2122(uint8_t Byte)
IPPU.ColorsChanged = true;
IPPU.Blue [PPU.CGADD] = IPPU.XB [(Byte >> 2) & 0x1f];
IPPU.Green [PPU.CGADD] = IPPU.XB [(PPU.CGDATA[PPU.CGADD] >> 5) & 0x1f];
- IPPU.ScreenColors [PPU.CGADD] = (uint16_t) BUILD_PIXEL(IPPU.Red [PPU.CGADD],
- IPPU.Green [PPU.CGADD],
- IPPU.Blue [PPU.CGADD]);
+ IPPU.ScreenColors [PPU.CGADD] = (uint16_t) BUILD_PIXEL(IPPU.Red [PPU.CGADD], IPPU.Green [PPU.CGADD], IPPU.Blue [PPU.CGADD]);
}
PPU.CGADD++;
}
@@ -481,8 +467,7 @@ static INLINE void REGISTER_2180(uint8_t Byte)
static INLINE uint8_t REGISTER_4212(void)
{
uint8_t GetBank = 0;
- if (CPU.V_Counter >= PPU.ScreenHeight + FIRST_VISIBLE_LINE &&
- CPU.V_Counter < PPU.ScreenHeight + FIRST_VISIBLE_LINE + 3)
+ if (CPU.V_Counter >= PPU.ScreenHeight + FIRST_VISIBLE_LINE && CPU.V_Counter < PPU.ScreenHeight + FIRST_VISIBLE_LINE + 3)
GetBank = 1;
GetBank |= CPU.Cycles >= Settings.HBlankStart ? 0x40 : 0;
@@ -491,5 +476,4 @@ static INLINE uint8_t REGISTER_4212(void)
return GetBank;
}
-
#endif
diff --git a/source/problems.txt b/source/problems.txt
deleted file mode 100644
index 293df59..0000000
--- a/source/problems.txt
+++ /dev/null
@@ -1,459 +0,0 @@
-o FIXED: Aladdin: Mode 7 tile 0 corruption. bug in my delay-by-one-word read
- V-RAM code.
-o FIXED: Missing sprites on Captain Commando (all negative x coords) - only
- on asm version, but now...
-o FIXED: Flashing screen on Captain Commando.
-o FIXED: Super Street Fighter 2 won't respond to joy-pad controls.
-o FIXED: Final Fantasy V - text windows are partially hidden - background
- priority-per tile problem. - needed dual-windowing AND/OR logic mode.
-o FIXED: Tinytoons uses Mode 7 priority per pixel.
-o FIXED: Formation Soccer seems to need transparent mode 7 graphics and sprites
- behind background.
-o FIXED: Super Aleste has corrupted sprites on title screen.
-o FIXED: Junk on backgrounds of Battle Clash. sub-screen subtract.
-o FIXED: Total Carnage scrolling text is off by a couple of lines, h-dma problem.
-o FIXED: Puzzle Bobble says "this ROM is not designed for you snes".
-o FIXED: Sensible Soccer Mode 0 problems (colours).
-o FIXED: Cannon Fodder shows signs of H-DMA problems when showing mission
- title screen.
-o FIXED: Alien vs Predator: sprites messed up and screen flashing (screen
- flashing done on purpose).
-o FIXED: ffmq might require MVN/MVP to always use a 16-bit accumulator.
-o FIXED: Battle Toads crashes after player selection.
-o Mode 5 graphics on snestest.smc Controller Test are corrupt - offset per
- tile not implemented. NO - H-512 mode not implemented!
-o FIXED: snestest.smc locked up on Electronics test - IRQ never cleared.
-o FIXED: Formation Soccer is doing DMA from a bad area of RAM to V-RAM -
- needed SPC700 emulation.
-o FIXED: Pacman crashes and problems with sprites on title screen - v-ram
- reading problem.
-o FIXED: New FF5 graphics problems on title screen.
-o FIXED: MS-DOS machine with single joystick doesn't work.
-o FIXED: Seiken 3 locks up - first in a IRQ loop (its never cleared) then waiting
- for sound CPU.
-o FIXED: Metriod 3 has a corrupted screen and locks up.
-o FIXED: Mechawarrior 3050 has major problems - corrupt screen, extreamly slow frame
- time, etc., etc. - needs SetByte and GetByte CPU push code. - So does
- Weapon Lord!
-o FIXED: **Weapon Lord uses PCALL and TCALL in SPC-700 code **
-o FIXED: Another World is transfering a lot of data using DAM to VRAM and $2180.
-o FIXED: Soulblazer has minor h-dma glitches on vertical background scrolling.
-o FIXED: Secret of the Stars no go - Sound cpu wait problem.
-o FIXED: With reseting IRQ in place in v-line != h-irq-line Battle Toads: Double
- Dragon works, but Spawn doesn't.
-o FIXED: Memory map problem with Donald Duck (corrupt ROM image).
-o FIXED: Comsmo Puzzle and Yuu Yuu all show "V-RAM increment" type
- problems (problem was with reading V-RAM).
-o FIXED: Some games rewrite the sprite registers during a frame.
-o FIXED: ILLUSION has horrible sound... (sample decode routine had a bug).
-o FIXED: Find out why graphics are all squashed up on sensi soccer/ bman with old
- old tile drawing code...
-o FIXED: IRQ wobble on Aladdin at top of screen - causes occational palette flash.
-o FIXED: 7th Saga, Actraiser2 and Addams Family causes a core dump on exit.
-o FIXED: aleste.smc has sprite display problems on title screen on tile-based redraw
- code.
-o INFO: DKC3 crashes itself if V-RAM isn't filled with zeros!
-o FIXED:Tazmania has stopped working - missing IRQ. Adding a field to
- Settings moved a variable being used in asm code.
-o FIXED: Castleviana 5 and Sparkster use DMA address decrement to update sprites.
-o FIXED: X-Men has some sprite glitches - but only on tile based redraw code -
- it blanks the screen early and then DMAs new spite tile data before the
- usual end of frame.
-o FIXED: Super Punchout shows corrupt knockout timer when H-DMA is enabled.
- (was triggering H-DMA during a frame) - only trigger H-DMA during v-blank or
- h-blank, otherwise write to register is ignored.
-o SOUND: Clayfighter, Tazmania, Madden 94-97, NHL 96-97, and WeaponLord
- all spool sound samples into APU RAM using H-DMA!
-o FIXED: ASM version bug: SOUND: Super Punchout has lots of sound repeat problems.
-o FIXED: DMA? problems on background 3 in SMW.
-o FIXED: Weapon Lord hangs on player selection screen (waiting for SPC700?).
-o FIXED: Killer Instinct shows BG3 on title screen when it shouldn't.
-o FIXED: Blank screen on Rock 'n' Roll Racing - DMA problem.
-o FIXED: SFC Bastard has offset problems on Mode 7 graphics on title screen.
-o FIXED: SOUND: SFC Bastard (bastardy.smc) locks with sound enabled.
-o FIXED: GANBARE GOEMON 2 (sf16232a.078) requires SavedOAMAddr to be set in
- OAMAddr after end of frame (at max v_counter?).
-o FIXED: SuperOffroad: The Baj has screen flash/timing problems during game -
- skipping IRQs ? - It was reseting the H-IRQ position so another IRQ would
- occur on the scanline it was already on.
-o FIXED: R-TYPE 3 has missing scrolling "space" background during game.
- (location $54 contains $10, needs to contain $11) - not a bug, when run on a
- real SNES the background is missing as well.
-o FIXED: Toy Story plays sample that walks off end of memory - was causing
- emulator crashes.
-o FIXED: Bubsy has corrupt graphics - it required register $4210 bit 7 to
- trigger at the end of h-blank of line 224, not at the start.
-o FIXED: Jurrassic Park locks with sound enabled. (Corrupt ROM image).
-o FIXED: Q*Bert 3 has major graphics problems after 3rd level or locks. With -h 120
- and VAR_CYCLES game has corrupt graphics, otherwise it crashes. - ROM image
- is corrupt as same thing happens on a real SNES.
-o FIXED: Evo - Chapter 1 title page shown on two backgrounds at same time -
- not bug just sub-screen addition being used.
-o FIXED: Ghosts and Ghouls needs H-DMA to be enabled if register is written after
- start of frame - Super Punchout needs the opposite... -
- only start H-DMA if enable register written to during v-blank or h-blank.
-o Exhaust Heat might require scanline drawing just before H-DMA rather than
- at end of h-blank.
-o FIXED: Exhaust Heat has mode 7 offset problems. v0.24 accidentally fixed the problem
- but now its back again.
-o FIXED: Goal locks at start with sound enabled. - The SPC700 seems to need to
- start executing instructions before the 65c816 so it has initialised $2140
- to $BBAA before the '816 checks for that value.
-o FIXED: Final Fight 3 crashes with BRK instruction. - corrupt ROM.
-o FIXED: T2 - has problems with clip windows.
-o Robocop v Terminator: screen flashes like Alien vs Predator.
-o FIXED:Nhl97 crashes - sound APU problem ?
-o FIXED: Nhl97 now doesn't crash, instead the game won't start after the teams have
- been chosen.
-o FIXED: Shadow Run shows sprite-sprite priority problems.
-o Alien vs Predator uses colour subtraction on sprites but only on palettes 6-7?
- (Collect cloaking device, second object).
-o NMI might need to be delayed to end of line.
-o JanjYu Gakuen 2 looks at bit 6 of $4211 - might need to show source of
- IRQ - $40 for V-IRQ and $20 for H-IRQ?
-o FIXED: Sunset riders has repeated background problems(?) - could be sub-screen
- addition/subtraction. Colour palette changes during game.
-o Priority on snestest.smc Character Test are incorrect.
-o Clay Fighter 2 writes to $3007-8 and reads from $3211-4.
-o FIXED: Shien The Blade Chase tests bit 0 of $4200 until it goes 0 - reading
- $4200 must reflect joypad reading status. No, actually $4200 is open bus.
-o FIXED: NBAJam (not Tournament Edition) has an odd memory map - accesses
- code at $3Dxxxx. (corrupt ROM image).
-o Super Bases Loaded uses Mode 5 and the background offset mode.
-o FIXED: Super Bases Loaded 2 has an odd memory map: writes to $E00000 and reads from
- $E04000! (DSP1 chip)
-o PilotWings uses a DSP.
-o FIXED: Return Of Double Dragon needs -FL -ss 1 -o.
-o FIXED: Tazmania is rewriting the colour palette just before the end of the frame.
- Shows up a a flashing screen. Uses software not DMA.
-o FIXED: Actraiser2 has screen flash/timing problems. Was rewriting H-DMA
- start addresses during a frame - real SNES seems to ignore these.
-o Actraiser2 uses mode 7 fixed colour palette mode - "32K" mode.
-o FIXED: Agmawo locks at start. corrupt ROM - looks to be only first part of a
- multi-part ROM on the CD.
-o FIXED: Alfred Chicken needs -FL.
-o Mighty Max and Addams Family 2 are very slow with sound enabled.
-o Williams Arcade Classics plays sound samples by varying the volume level
- using the gain control...
-o FIXED:Zoop: Sound works initially but then stops during game.
-o FIXED:Lion King locks up with sound enabled...
-o Sparkster has lots of sound repeat problems.
-o FIXED: Killer Instinct shows corruption at top of screen during the actual fight.
- Seems to need sub-screen emulation turned on.
-o Look into passing mask into StartHDMA so the routine only resets DMA
- channels with the bit set in the bit mask.
-o FIXED: RPM racing uses mode 5 512x512.
-o FIXED: Biker Mice From Mars (sfbiker) needs -N and flipped layering '8'.
-o FIXED: Mario and Wario (sfmarwar.smc) needs a mouse.
-o Battle Toads: Double Dragon needs the -h parameter set to 100.
-o FIXED: SF8752.smc has sprites problems when using tile-redraw code.
-o FIXED: NHL '97 does a JMP $4320 in bank $87 - $4320 is a register area in that
- bank.
-o FIXED: Chuck Rock has corrupted sprites on the title screen.
-o FIXED: War of the Gems crashes with a STP instruction - bug in asm version of
- CPU core - appears when status register spliting was added. Snap6.
- If IRQ pending after returning from an NMI then incorrect status pushed
- onto stack.
-o Adventures of Batman & Robin (bat.smc) has really slow music on title
- screen with SPC700 shutdown enabled.
-o Return of the Jedi has lots of sound problems.
-o WORMS has vertical offset problems on BG1 when displaying life bar at top
- of screen.
-o FIXED: Street Racer has corrupt sprites on the player select screen.
-o FIXED: Super Bases Loaded 1 uses Mode 5 (512x224) on intro.
-o INFO: Batman Forever uses sub-screen addition with bg's being on both the main
- screen and sub-screen.
-o INFO: Intro of Beavis and Butthead uses sub-screen addition with bg's being on
- both the main screen and sub-screen.
-o INFO: ebreaker.smc bg's on both main and sub-screen.
-o Big Sky Trooper's music is very slow.
-o FIXED: Bomberman 4 needs -FH.
-o FIXED: Captain Commando is interleaved.
-o FIXED: Crystal Bean locks at start and is interleaved. - corrupt ROM image.
-o Jim Power (jimp.smc) has scrolling/H-DMA problems on the intro.
-o FIXED: Ogre Battle has fixed colour subtract problems on the main screen.
-o pacman and Jim Power intro seem to rely on maximum number of sprites on a
- scanline at once to hide unwanted sprites.
-o FIXED: Pacman 2 has scrolling/H-DMA problems during the game.
-o FIXED: Power Drive (snk_powd.smc) uses 512x448 on intro.
-o FIXED: SKI PARADISE (skiparad.fig) uses mode 7 priority per pixel
-o The Magical Quest (smmouse.smc) has an intro with bad raster glitches - uses
- exact CPU timing!
-o FIXED: Madden 96 needs -FH and has new Antherox intro.
-o FIXED: College Slam (collslam.1 on CD) thinks non-standard controllers are
- connected. It writes $12 and $02 to $4016.
-o The sound on Speedy Gonzales sounds out of tune - seems to vary the
- frequency a lot - first low byte then the high byte.
-o Stargate has sound clicks on channels 5 and 6 on title page.
-o FIXED: Ranman1/2 Part 1 (ranma121.smc) crashes SPC700 at start up - looks like a
- 65c816 addressing mode/instruction bug! - corrupt ROM image.
-o Speedy Gonzales starts a long DMA to $2180 at v-line 1 with the screen not
- blanked.
-o FIXED: acc-elf.smc works on v0.1 but crashes with a STP instruction in v0.24.
-o FIXED: FLYING HERO BUGYURU flyhero.smc does not work and image is odd size so code
- doesn't detect header and remove it.
-o FIXED: Yaiba (lmk_yaib.smc) locks waiting for an interrupt to occur to no interrupts
- are enabled.
-o CORRUPT ROM: Rise of the Robots does a long DMA to V-RAM during in the NMI handler -
- it might require the DMA to automatically stop when H-DMA starts if the
- screen is not blanked.
-o FIXED: Andrew Agassi Tennis has missing sprites - only writes to low byte
- of sprite write address register. Writing to low byte must clear hi-byte as
- well.
-o FIXED: Bonkers locks sound enabled. SPC700 code expected KOFF DSP register
- to return value just written.
-o Mario Kart has odd graphic windows settings when first started.
-o FIXED: Street Racer has odd size sprites and has missing tiles when displayed.
-o FIXED: ASM 65c816, Aladdin keeps warping to other side of screen on first level.
-o The Great Circus Mystery (circusmystery.smc) doesn't work.
-o James Pond 2 gets stuck waiting for SPC700 to respond after first title
- screen. works on v0.24.
-o Mario All stars - Super Mario 2 uses mode 2 (offset per tile) but with
- height 64 on bg #2. Black screen.
-o FIXED: Chrono Trigger uses mode 2 with width 64 for wavey 'Trigger' text. Text
- should scroll onto screen, it doesn't. It does if width 64 is disabled,
- but it starts on the screen first. - bug with not wrapping Quot variable
- with mode 2 width 64.
-o Tetris Attack uses screen screen height of 64 on offset-per-tile mode 2.
-o Batman forever might show that if OBJs are on both the main and sub-screens
- and colour addition is enabled, then only OBjs with palette numbers
- 4 or greater take part in colour addition and are not displayed on the
- sub-screen.
-o FIXED: CPUShutdown causes problems for ROMs that wait for h-blank using $4212
- during the v-blank period - Reschedule doesn't enable HBLANK_START_EVENT
- during the v-blank period so the cycle skipping code doesn't wake up the
- CPU until h-blank end.
-o FIXED: Putty Squad seems to use wrong palette for each tile when mosaic effect
- is being used - could be only the ROM so far with 16x16 pixels tiles and
- mosaic effect.
-o FIXED: Mortal Kombat 1 restarts level when once the 'fight' message has disappeared.
- - 0.31 has problems.
- - 13-03-98 problems.
- - 1-4-98 problems.
- Problem with asm code only and intermittent.
-o FIXED: Clay fighter has problems with colour window when the game starts and the
- game locks up anyway. ... but still sound repeat problems.
-o FIXED: VAMPIRES KISS has corrupt, flashing OBJs and some v-ram problems,
- no problems in 0.41. Problem not dma, ppu, or cpuexec.
- - Problem was H-DMA was being started inside v-blank period.
-o FIXED: VAMPIRES KISS has tile glitches on bg#2 during game - not present in
- 0.41. H-DMA?
-o FIXED: Contra 3 has clip window problems resulting in black screen, only
- OBJs can be seen when game first starts.
-o FIXED: YUUYUU TOKUBETUHEN (yuyut.smc) has missing OBJs when screen is split -
- it enables both clip windows on OBJ and sets the combination mode to
- AND but both clip windows only overlap a few pixels so the OBJs are not
- displayed. - bug in window code, needed to flip the logic window
- combination mode because the window area add already been flipped.
-o Multiple colour add/sub and clip window bugs with Killer Instinct.
-o T2: Arcade Game displays multi-coloured blocks of tiles when a mode 3
- screen is being displayed. Screen shown when game-over.
-o Bomberman 5 flickers on title screen when H-DMA is enabled.
-o FIXED: XOR window clip code is producing bands that overlap.
-o TOKIMEKI MEMORIAL (tokmemor.smc) uses hi-res., mosasic effect and fixed
- colour addition.
-o FIXED: TOKIMEKI MEMORIAL (tokmemor.smc) displays corrupted sprites - every other
- group of four pixels is missing.
-o RPM racing (hi-res. interlace) displays blank screen with 8-bit renderer.
-o FIXED: Lufia I locks is shutdown is enabled. Asm only.
-o FIXED: The SPC700 in Universal Solider locks if shutdown is enabled. Asm only.
-o FIXED: Disabling the Multi Player 5 switches controller 1 to a mouse !
-o FIXED: Top Gear 3000 - DSP1? game, though ROM header doesn't mention it.
- Bug in ROM header detection code.
-o FIXED: Lost Vikings II locks after title screen - works in 0.24.
-o Lost Vikings locks or resets on all versions.
-o Uniracers enables sub-screen addition on bg # 2 but with nothing on the
- the sub-screen. Should it be either the fixed colour or back-drop
- colour get added instead?
-o Wile E's Revenge has missing music, and very quite sound. Also displays
- rotated mode 7 character picture during titles.
-o Pilot Wings tries to set up a V-IRQ to happen on the line its already on,
- uses IRQ to switch to mode 7. -h 120 fixes problem. Cycle timing must be
- off.
-o Top Gear 3000 sits in a loop waiting for the H-DMA line count register
- to reach a particular value - it never does because its not emulated yet!
-o Theme Park starts a rather crummy tune then crashes. Hacker intro?
-o Killer Instinct has a one-pixel wide bright line part way down screen
- on orcid practice level - colour window invert bug?
-o Killer Instinct bg #1 is hidding some transparency effects and the
- fighter's sprite shadows on some levels.
-o Batman - revenge of the joker locks at start - waiting for SPC700 which
- has hit a stop instruction.
-o Eye of the Beholder has strange mouse pointer movement problems when SNES
- mouse emulation is enabled.
-o FIXED: Daffy duck has lots of scrolling glitches on background parallax effects -
- timing problems? Missing NMI - ROM kept turning NMI enable flag on and off.
-o Aero the AcroBat 2 might wait until a bit in $420B (h-dma enable) clears -
- does reading that register indicate H-DMA channels in progress?
-o Ardy Lightfoot and Oblix both flash the sprite of the main character
- on and off every other frame, should they do this or is it a bug?
-o FIXED: Firemen locks at start waiting for SPC700. Human game, needs -ratio 3.
-o Jim Power has a one pixel wide bright line down one side of screen on the
- level map screen.
-o Lots of missing sound effects in Home Alone 2.
-o Missing sound effects in Earth Worm Jim 2.
-o Captain America has single pixel high line corruption through some of its
- tiles.
-o NHL STANLEY CUP locks at start waiting for the SPC700 - works with sound
- disabled.
-o Sailor Moon has colour window problems during intro of game.
-o Sailor Moon R screen flashes black during game. -h 102 fixes.
-o Tile corruption on Super Pang on 3rd level+ - only on DOS port.
-o Mighty Max uses colour window on main screen to clip background colour
- palette changes - except its not working on Snes9x.
-o FIXED: Battle Toads: Battlemanics crashes during intro - ROM's NMI handler does not
- switch index registers to 16-bit before pushing them onto the stack, but
- always switches them to 16-bit mode when restoring them.
- NMI timing problem.
-o FIXED: Contra 3 has missing fire effects when bomber plane drops bombs - use
- freeze-game to see. - Colour window is fully clipping the sub-screen and
- the fire effects are only on the sub-screen.
- - colour window invert bug.
-o FIXED: Illusion of Gaia uses sub-screen subtraction with half flag during game
- select. Also, uses colour window to cut a hole in the main-screen,
- should the sub-screen be visible at this point? Only two backgrounds are
- being displayed, one on the main-screen and one on the sub-screen and
- the sub-screen is not being added to the background only the background on
- the main-screen.
-o FIXED: Gun force uses background #2 to display horizontal bullets fired from
- player, but they are displayed offset from the main firing them. H-DMA transfer
- size array was not set for the mirrored channels.
-o FIXED: Empire Strikes Back needs -ratio 5 to work.
-o Empire Strikes Back: The Hoth battle stage uses mode 7, priority per pixel
- and part of the graphics are missing.
-o FIXED: llusion of Gaia menu on the first screen should be on a dark background.
- ZSNES gets it correct.
-o DOS port can't load some ROMs from CD. NLKE has same problem. Allegro?
- ZSNES loads most of them fine.
-o -frametime option is broken.
-o Jap version of Tetris Attack might have a scrolling bug on the title screen.
-o FIXED: Commodore 64 emulator doesn't work with asm CPU core - works fine with
- C code.
-o FIXED: Chrono Trigger: crash bug in clipping code - use snapshot F6 and press 's'.
- Corrupts stack causing crash.
-o FIXED: Toy Story only updates screen every-other frame and woody flashes
- continuously. (Woody flashing is due to hacked ROM)
- Toy Story requires NMI to happen immediately after a WAI instruction.
-o Toy Story has garbled sound output just before game starts. DOS port only.
-o FIXED: Tales of Phantasia executes across a 0x8000 boundary and either side of
- the boundary are at different offsets into the ROM. Causes a crash.
-o RPM Racing doesn't display correctly when interpolation and 16-bit screen
- mode. X11 port. Works when full-screen X.
-o FIXED: Itchy & Scratchy has sprite corruption on the one of the title screens.
- Problem appeared between v0.4 and v0.41.
-o Start screen on Chase HQ is corrupt. v0.24 has same problem.
-o FIXED: King of the Rally has missing music on the car-feature selection screen.
- Works in v1.00. Requires DP+X addresses to wrap in zero page.
-o FIXED: (again) Lufia I locks is shutdown is enabled waiting for SPC700.
- made the SPC700 wake up if the 65c816 reads from the one of the 4 comm ports.
- The SPC700 used to only wake up when the 65c816 wrote to the ports.
-o WWF-Arcade crashes - calls a subroutine at address $EE758D which only
- contains zeros at that address. Memory map bug?
-o WWF-Super Wrestlemania and WWF-Raw both seem to display junk background
- layers - the ROM enables the layers but does not set up any background data.
-o WWF-Super Wrestlemania crashes just before the game starts. Memory map bug?
-o FIXED: Ninja Warriors has missing graphics on the title screen - V-IRQ problem?
- Problem appears after v1.16. Didn't like H-IRQ triggering on the same line
- if the H-IRQ position register was updated.
-o Yoshi's Island might show there's an 'off-by-one' bug in the clip window code
- when it pops up a message box.
-o Tazmania has ticking on the sound during the initial title music.
-o SuperScope 6 has a repeating gun-fire sound on the initial aim screen.
- Channel volume is set to gain mode.
-o Stargate sometimes has a slight click sound at the end of each rapid soft
- beep when the start button is pressed on the title page. Channel 6 is the
- problem, has attack rate of 10ms, decay 1200ms, infinate sustain and a
- sustain level of 5.
-o Kirby Superstar - SA-1 game? YEP!
-o The Lion King: under sound test, continue long song contains some odd
- sounding notes. Is a particular type of sample not being decoded correctly?
-o Clicks on the title music of Madden 98.
-o Bugs Bunny has sound repeat problems when Bugs kicks a dog. - not a problem,
- just needs -envx.
-o Bugs Bunny has junk snow characters on bg #1.
-o Airwolf hacker intro is very corrupt - corrupt ROM?
-o FIXED:Pac-in-time title music sounds a lot worse with new envelope sound code in
- 1.17. The game requires samples to be keyed on without being keyed off first.
-o DOREMI Fantasy milon uses decay exponential volume envelope on the notes at
- the end of the title music - they seem to take too long to decay.
-o Winter Gold causes ZSNES Super FX code to overwrite the static data after its
- allocation - was causing the X library to crash since that was the .o after
- linked after zsnes.o.
-o FIXED: Micromachines 2 has continuing engine noise problem when race is over -
- sound channel 7 is in sustain mode with time period set to infinite.
- - must have updated ADSR parameters as the it was in progress.
-o DONE: Check Micromachines on real SNES - does the Ocean logo shear at the start?
- If it doesn't it could be a bug in the mode 7 code - the same problem that
- affects battle racers. - Does the same thing on a real SNES.
-o FIXED: LAMBORGHINI AMERICAN has lots of missing music notes -
- uses bent-line inc with attack rate of infinite?
- - S9xSetEnvRate thought that the channel was silent and hence the number of
- volume steps was 0 so erate was always being set to 0.
-o Slight clicking on the sound of Wild Snake - not sound sync or interpolation
- bug.
-o FIXED: Zoop locks up at the end of each level playing random sound data on one
- channel and the main SNES CPU seems to be waiting for the sound channel to
- finish. - ENVX should return zero when channel is in gain mode.
-o Battle Racers might have missing music during the race.
-o Tactics Ogre has missing music and ROM locks up if you visit the sound menu.
-o FIXED: Dragon Quest 5: the monsters disappear during fight sequences when the
- fireball is used (freeze file 004).
- - H-DMA needs to continue for one extra line, e.g. for screen height of
- 239 H-DMA needs to run from line 0 to line 239, inclusive.
-o Stargate sound click problem during some music - makes large changes to
- channel volume levels while the sound is still playing, but other problems
- seem to be causing the click.
-o Super Air Diver 2, Pilotwings and SD Racing DSP all suffer from the same
- shear DSP1 emulation bug. Bug in op 0x02/0x1a/0x0a to do with viewing angle
- and rotation.
-o Super Air Diver 1 locks at start waiting for sound CPU.
-o FIXED: Ballz generates unknown DSP1 command 2f and 0f then locks up on title
- screen. Implemented new Ops
-o FIXED: Highway Battle 2 generates unknown DSP1 command 2f and lots of different
- unknown DSP1 commands during play. The mode 7 screen is all messed up. Hacked game,
- but the DSP routines were inaccurate.
-o FIXED: Dungenon Master generates lots of unknown DSP1 commands errors. It's DSP-2.
-o F1 Roc 2 isn't a DSP1 - uses some form of custom chip.
-o F1 Roc 2 has graphic window clip problems with the clip code.
-o Vertical mouse movement on Eye of the Beholder is very erratic.
-o BT IN BATTLEMANIACS has slight sound click problems during the opening music.
-o Bomberman 5 - sound test, music 22 doesn't sound correct. Sounds different on
- three different versions 1.00, 1.16 and 1.18.
-o FIXED: Sprite priority bug on title screen of ILLUSION OF GAIA.
-o FIXED: James Pond 2 - random horizontal sprite movement between two places, seems
- OK in 1.11 (only slight bug with battery level) but broken in 1.16. Sprite code relies
- on funky behavior.
-o Jigsaw Party objects to multi-player 5 emulation, game won't start with it
- enabled.
-o FIXED: The ROM load code thinks Eek The Cat is in interleave format, it isn't.
-o FIXED: Primal Rage v1.20 - major graphics problem problems, not in v1.19.
-o FIXED: Mario Kart - single player mode, pressing select to use rear view mirror
- results in corrupt graphics. - inverse of an empty colour window should be
- whole screen.
-o FIXED:Missing logo from FF5 start up screen. - mode 7 was always using
- main-screen z-buffer.
-o FIXED: Something is zeroing the ZSNES SfxnRamBanks variable in Yoshi's Island,
- allowing the code to set junk values in the ROMBR register which in turn
- can cause an illegal memory access when the Super FX tries to access its RAM.
- - ZSNES code bug.
-o FIXED: Background scrolling glitches on Stargate.
- - skipping NMIs. Stopped NMI retriggering in same screen.
-o Metal Combat - in the title screen speech the '93 of the 1993 words get cut
- off earily
-o POWER RANGERS FIGHT during the game the score area breaks up and scrolls when
- it shouldn't and there is corruption in the character graphics.
-o FIXED: Maui Mallard doesn't show the water background layer correctly.
-
-o Seiken 3 3dfx type screen breakup bug in windows port with 'sal mode
- enabled.
-o FIXED: RexRonanExperimentalSurgeonUSA layering problem in title screens.
- - wasn't wrapping V-RAM addresses for screen (bg) locations.
-o PANIC BOMBER WORLD locks up on title screen
-o Doom segfaults after splash screen if i386 core is used on Linux.
-o Doom hangs when shooting a barrel if C++ core is used on Linux.
-o Cu On Pa locks up at the player select screen - input not handled. Timing?
-o Madara 2 menus should have a blue window clipped out - hi-res clipping behavior?
-o Shin Megami Tensei fusion screen - corrupt graphics - $2105 seems to be getting set
- to 0 instead of 9.
-o Dragon Ball Z - Super Butoden 2 (J) 1.1 - flashing graphics at the start of
- a fight and then later if the screen gets splitted.
diff --git a/source/sa1.c b/source/sa1.c
index b19ba11..005a9c7 100644
--- a/source/sa1.c
+++ b/source/sa1.c
@@ -111,29 +111,29 @@ uint8_t S9xSA1GetByte(uint32_t address)
{
uint8_t* GetAddress = SA1.Map [(address >> MEMMAP_SHIFT) & MEMMAP_MASK];
if (GetAddress >= (uint8_t*) MAP_LAST)
- return (*(GetAddress + (address & 0xffff)));
+ return GetAddress[address & 0xffff];
switch ((intptr_t) GetAddress)
{
case MAP_PPU:
- return (S9xGetSA1(address & 0xffff));
+ return S9xGetSA1(address & 0xffff);
case MAP_LOROM_SRAM:
case MAP_SA1RAM:
- return (*(Memory.SRAM + (address & 0xffff)));
+ return Memory.SRAM[address & 0xffff];
case MAP_BWRAM:
- return (*(SA1.BWRAM + ((address & 0x7fff) - 0x6000)));
+ return SA1.BWRAM[(address & 0x7fff) - 0x6000];
case MAP_BWRAM_BITMAP:
address -= 0x600000;
if (SA1.VirtualBitmapFormat == 2)
- return ((Memory.SRAM [(address >> 2) & 0xffff] >> ((address & 3) << 1)) & 3);
+ return (Memory.SRAM [(address >> 2) & 0xffff] >> ((address & 3) << 1)) & 3;
else
- return ((Memory.SRAM [(address >> 1) & 0xffff] >> ((address & 1) << 2)) & 15);
+ return (Memory.SRAM [(address >> 1) & 0xffff] >> ((address & 1) << 2)) & 15;
case MAP_BWRAM_BITMAP2:
address = (address & 0xffff) - 0x6000;
if (SA1.VirtualBitmapFormat == 2)
- return ((SA1.BWRAM [(address >> 2) & 0xffff] >> ((address & 3) << 1)) & 3);
+ return (SA1.BWRAM [(address >> 2) & 0xffff] >> ((address & 3) << 1)) & 3;
else
- return ((SA1.BWRAM [(address >> 1) & 0xffff] >> ((address & 1) << 2)) & 15);
+ return (SA1.BWRAM [(address >> 1) & 0xffff] >> ((address & 1) << 2)) & 15;
default:
return OpenBus;
}
@@ -142,7 +142,7 @@ uint8_t S9xSA1GetByte(uint32_t address)
uint16_t S9xSA1GetWord(uint32_t address)
{
OpenBus = S9xSA1GetByte(address);
- return (OpenBus | (S9xSA1GetByte(address + 1) << 8));
+ return OpenBus | (S9xSA1GetByte(address + 1) << 8);
}
void S9xSA1SetByte(uint8_t byte, uint32_t address)
@@ -264,9 +264,8 @@ void S9xSetSA1MemMap(uint32_t which1, uint8_t map)
for (c = 0; c < 0x200; c += 16)
{
- /*Code from Snes9x 1.54.1 - *
- * This allows Super Mario World VLDC 9 hack to load *
- * conversion to int is needed here - map is promoted but which1 is not */
+ /*Code from Snes9x 1.54.1 - This allows Super Mario World VLDC 9 hack to load.
+ Conversion to int is needed here - map is promoted but which1 is not */
int32_t offset = (((map & 0x80) ? map : which1) & 7) * 0x100000 + (c << 11) - 0x8000;
block = &Memory.ROM [offset];
for (i = c + 8; i < c + 16; i++)
@@ -279,32 +278,30 @@ uint8_t S9xGetSA1(uint32_t address)
switch (address)
{
case 0x2300:
- return ((uint8_t)((Memory.FillRAM [0x2209] & 0x5f) |
- (CPU.IRQActive & (SA1_IRQ_SOURCE | SA1_DMA_IRQ_SOURCE))));
+ return (uint8_t)((Memory.FillRAM [0x2209] & 0x5f) | (CPU.IRQActive & (SA1_IRQ_SOURCE | SA1_DMA_IRQ_SOURCE)));
case 0x2301:
- return ((Memory.FillRAM [0x2200] & 0xf) |
- (Memory.FillRAM [0x2301] & 0xf0));
+ return (Memory.FillRAM [0x2200] & 0xf) | (Memory.FillRAM [0x2301] & 0xf0);
case 0x2306:
- return ((uint8_t) SA1.sum);
+ return (uint8_t) SA1.sum;
case 0x2307:
- return ((uint8_t)(SA1.sum >> 8));
+ return (uint8_t) (SA1.sum >> 8);
case 0x2308:
- return ((uint8_t)(SA1.sum >> 16));
+ return (uint8_t) (SA1.sum >> 16);
case 0x2309:
- return ((uint8_t)(SA1.sum >> 24));
+ return (uint8_t) (SA1.sum >> 24);
case 0x230a:
- return ((uint8_t)(SA1.sum >> 32));
+ return (uint8_t) (SA1.sum >> 32);
case 0x230d:
{
uint8_t byte = Memory.FillRAM [0x230d];
if (Memory.FillRAM [0x2258] & 0x80)
S9xSA1ReadVariableLengthData(true, false);
- return (byte);
+ return byte;
}
}
- return (Memory.FillRAM [address]);
+ return Memory.FillRAM [address];
}
void S9xSetSA1(uint8_t byte, uint32_t address)
@@ -333,11 +330,9 @@ void S9xSetSA1(uint8_t byte, uint32_t address)
Memory.FillRAM [0x2301] |= 0x10;
break;
case 0x2201:
- if (((byte ^ Memory.FillRAM [0x2201]) & 0x80) &&
- (Memory.FillRAM [0x2300] & byte & 0x80))
+ if (((byte ^ Memory.FillRAM [0x2201]) & 0x80) && (Memory.FillRAM [0x2300] & byte & 0x80))
S9xSetIRQ(SA1_IRQ_SOURCE);
- if (((byte ^ Memory.FillRAM [0x2201]) & 0x20) &&
- (Memory.FillRAM [0x2300] & byte & 0x20))
+ if (((byte ^ Memory.FillRAM [0x2201]) & 0x20) && (Memory.FillRAM [0x2300] & byte & 0x20))
S9xSetIRQ(SA1_DMA_IRQ_SOURCE);
break;
case 0x2202:
@@ -361,20 +356,17 @@ void S9xSetSA1(uint8_t byte, uint32_t address)
S9xSetIRQ(SA1_IRQ_SOURCE);
return;
case 0x220a:
- if (((byte ^ Memory.FillRAM [0x220a]) & 0x80) &&
- (Memory.FillRAM [0x2301] & byte & 0x80))
+ if (((byte ^ Memory.FillRAM [0x220a]) & 0x80) && (Memory.FillRAM [0x2301] & byte & 0x80))
{
SA1.Flags |= IRQ_PENDING_FLAG;
SA1.IRQActive |= SNES_IRQ_SOURCE;
}
- if (((byte ^ Memory.FillRAM [0x220a]) & 0x40) &&
- (Memory.FillRAM [0x2301] & byte & 0x40))
+ if (((byte ^ Memory.FillRAM [0x220a]) & 0x40) && (Memory.FillRAM [0x2301] & byte & 0x40))
{
SA1.Flags |= IRQ_PENDING_FLAG;
SA1.IRQActive |= TIMER_IRQ_SOURCE;
}
- if (((byte ^ Memory.FillRAM [0x220a]) & 0x20) &&
- (Memory.FillRAM [0x2301] & byte & 0x20))
+ if (((byte ^ Memory.FillRAM [0x220a]) & 0x20) && (Memory.FillRAM [0x2301] & byte & 0x20))
{
SA1.Flags |= IRQ_PENDING_FLAG;
SA1.IRQActive |= DMA_IRQ_SOURCE;
@@ -444,8 +436,7 @@ void S9xSetSA1(uint8_t byte, uint32_t address)
{
// Char conversion 2 DMA enabled
// memmove converted: Same malloc but constant non-overlapping addresses [Neb]
- memcpy(&Memory.ROM [MAX_ROM_SIZE - 0x10000] + (SA1.in_char_dma << 4),
- &Memory.FillRAM [0x2240], 16);
+ memcpy(&Memory.ROM [MAX_ROM_SIZE - 0x10000] + (SA1.in_char_dma << 4), &Memory.FillRAM [0x2240], 16);
SA1.in_char_dma = (SA1.in_char_dma + 1) & 7;
if ((SA1.in_char_dma & 3) == 0)
S9xSA1CharConv2();
@@ -488,15 +479,14 @@ void S9xSetSA1(uint8_t byte, uint32_t address)
break;
}
break;
- case 0x2258: // Variable bit-field length/auto inc/start.
+ case 0x2258: // Variable bit-field length/auto inc/start.
Memory.FillRAM [0x2258] = byte;
S9xSA1ReadVariableLengthData(true, false);
return;
case 0x2259:
case 0x225a:
- case 0x225b: // Variable bit-field start address
+ case 0x225b: // Variable bit-field start address
Memory.FillRAM [address] = byte;
- // XXX: ???
SA1.variable_bit_pos = 0;
S9xSA1ReadVariableLengthData(false, true);
return;
@@ -509,8 +499,7 @@ static void S9xSA1CharConv2()
{
uint32_t dest = Memory.FillRAM [0x2235] | (Memory.FillRAM [0x2236] << 8);
uint32_t offset = (SA1.in_char_dma & 7) ? 0 : 1;
- int32_t depth = (Memory.FillRAM [0x2231] & 3) == 0 ? 8 :
- (Memory.FillRAM [0x2231] & 3) == 1 ? 4 : 2;
+ int32_t depth = (Memory.FillRAM [0x2231] & 3) == 0 ? 8 : (Memory.FillRAM [0x2231] & 3) == 1 ? 4 : 2;
int32_t bytes_per_char = 8 * depth;
uint8_t* p = &Memory.FillRAM [0x3000] + dest + offset * bytes_per_char;
uint8_t* q = &Memory.ROM [MAX_ROM_SIZE - 0x10000] + offset * 64;
@@ -540,15 +529,9 @@ static void S9xSA1CharConv2()
static void S9xSA1DMA()
{
- uint32_t src = Memory.FillRAM [0x2232] |
- (Memory.FillRAM [0x2233] << 8) |
- (Memory.FillRAM [0x2234] << 16);
- uint32_t dst = Memory.FillRAM [0x2235] |
- (Memory.FillRAM [0x2236] << 8) |
- (Memory.FillRAM [0x2237] << 16);
- uint32_t len = Memory.FillRAM [0x2238] |
- (Memory.FillRAM [0x2239] << 8);
-
+ uint32_t src = Memory.FillRAM[0x2232] | (Memory.FillRAM[0x2233] << 8) | (Memory.FillRAM[0x2234] << 16);
+ uint32_t dst = Memory.FillRAM[0x2235] | (Memory.FillRAM[0x2236] << 8) | (Memory.FillRAM[0x2237] << 16);
+ uint32_t len = Memory.FillRAM[0x2238] | (Memory.FillRAM[0x2239] << 8);
uint8_t* s;
uint8_t* d;
@@ -598,9 +581,7 @@ static void S9xSA1DMA()
void S9xSA1ReadVariableLengthData(bool inc, bool no_shift)
{
- uint32_t addr = Memory.FillRAM [0x2259] |
- (Memory.FillRAM [0x225a] << 8) |
- (Memory.FillRAM [0x225b] << 16);
+ uint32_t addr = Memory.FillRAM[0x2259] | (Memory.FillRAM[0x225a] << 8) | (Memory.FillRAM[0x225b] << 16);
uint8_t shift = Memory.FillRAM [0x2258] & 15;
if (no_shift)
diff --git a/source/sar.h b/source/sar.h
index 487cdaa..2fc14d2 100644
--- a/source/sar.h
+++ b/source/sar.h
@@ -50,7 +50,5 @@ static INLINE int64_t SAR64(const int64_t b, const int32_t n)
#endif
return b >> n;
}
-
#endif
-
#endif
diff --git a/source/sdd1.c b/source/sdd1.c
index 58b934e..3a46ec9 100644
--- a/source/sdd1.c
+++ b/source/sdd1.c
@@ -9,9 +9,7 @@ void S9xSetSDD1MemoryMap(uint32_t bank, uint32_t value)
{
bank = 0xc00 + bank * 0x100;
value = value * 1024 * 1024;
-
int32_t c;
-
for (c = 0; c < 0x100; c += 16)
{
uint8_t* block = &Memory.ROM [value + (c << 12)];
diff --git a/source/sdd1.h b/source/sdd1.h
index 7f1df83..4ade993 100644
--- a/source/sdd1.h
+++ b/source/sdd1.h
@@ -4,7 +4,6 @@
#define _SDD1_H_
void S9xSetSDD1MemoryMap(uint32_t bank, uint32_t value);
-void S9xResetSDD1();
-void S9xSDD1PostLoadState();
-
+void S9xResetSDD1(void);
+void S9xSDD1PostLoadState(void);
#endif
diff --git a/source/sdd1emu.c b/source/sdd1emu.c
index 25da823..3aa1591 100644
--- a/source/sdd1emu.c
+++ b/source/sdd1emu.c
@@ -98,7 +98,8 @@ static INLINE uint8_t GetCodeword(int32_t bits)
in_stream <<= 1;
valid_bits--;
in_stream ^= 0x8000;
- if (in_stream & 0x8000) return 0x80 + (1 << bits);
+ if (in_stream & 0x8000)
+ return 0x80 + (1 << bits);
tmp = (in_stream >> 8) | (0x7f >> bits);
in_stream <<= bits;
valid_bits -= bits;
@@ -112,7 +113,8 @@ static INLINE uint8_t GetCodeword(int32_t bits)
static INLINE uint8_t GolombGetBit(int32_t code_size)
{
- if (!bit_ctr[code_size]) bit_ctr[code_size] = GetCodeword(code_size);
+ if (!bit_ctr[code_size])
+ bit_ctr[code_size] = GetCodeword(code_size);
bit_ctr[code_size]--;
if (bit_ctr[code_size] == 0x80)
{
@@ -150,12 +152,7 @@ static INLINE uint8_t ProbGetBit(uint8_t context)
static INLINE uint8_t GetBit(uint8_t cur_bitplane)
{
- uint8_t bit;
-
- bit = ProbGetBit(((cur_bitplane & 1) << 4)
- | ((prev_bits[cur_bitplane] & high_context_bits) >> 5)
- | (prev_bits[cur_bitplane] & low_context_bits));
-
+ uint8_t bit = ProbGetBit(((cur_bitplane & 1) << 4) | ((prev_bits[cur_bitplane] & high_context_bits) >> 5) | (prev_bits[cur_bitplane] & low_context_bits));
prev_bits[cur_bitplane] <<= 1;
prev_bits[cur_bitplane] |= bit;
return bit;
@@ -165,9 +162,8 @@ void SDD1_decompress(uint8_t* out, uint8_t* in, int32_t len)
{
uint8_t bit, i, plane;
uint8_t byte1, byte2;
-
- if (len == 0) len = 0x10000;
-
+ if (len == 0)
+ len = 0x10000;
bitplane_type = in[0] >> 6;
switch (in[0] & 0x30)
@@ -205,13 +201,17 @@ void SDD1_decompress(uint8_t* out, uint8_t* in, int32_t len)
{
for (byte1 = byte2 = 0, bit = 0x80; bit; bit >>= 1)
{
- if (GetBit(0)) byte1 |= bit;
- if (GetBit(1)) byte2 |= bit;
+ if(GetBit(0))
+ byte1 |= bit;
+ if(GetBit(1))
+ byte2 |= bit;
}
*(out++) = byte1;
- if (!--len) return;
+ if(!--len)
+ return;
*(out++) = byte2;
- if (!--len) return;
+ if(!--len)
+ return;
}
break;
case 1:
@@ -220,14 +220,19 @@ void SDD1_decompress(uint8_t* out, uint8_t* in, int32_t len)
{
for (byte1 = byte2 = 0, bit = 0x80; bit; bit >>= 1)
{
- if (GetBit(plane)) byte1 |= bit;
- if (GetBit(plane + 1)) byte2 |= bit;
+ if(GetBit(plane))
+ byte1 |= bit;
+ if(GetBit(plane + 1))
+ byte2 |= bit;
}
*(out++) = byte1;
- if (!--len) return;
+ if(!--len)
+ return;
*(out++) = byte2;
- if (!--len) return;
- if (!(i += 32)) plane = (plane + 2) & 7;
+ if(!--len)
+ return;
+ if(!(i += 32))
+ plane = (plane + 2) & 7;
}
break;
case 2:
@@ -236,26 +241,29 @@ void SDD1_decompress(uint8_t* out, uint8_t* in, int32_t len)
{
for (byte1 = byte2 = 0, bit = 0x80; bit; bit >>= 1)
{
- if (GetBit(plane)) byte1 |= bit;
- if (GetBit(plane + 1)) byte2 |= bit;
+ if(GetBit(plane))
+ byte1 |= bit;
+ if(GetBit(plane + 1))
+ byte2 |= bit;
}
*(out++) = byte1;
- if (!--len) return;
+ if(!--len)
+ return;
*(out++) = byte2;
- if (!--len) return;
- if (!(i += 32)) plane ^= 2;
+ if(!--len)
+ return;
+ if(!(i += 32))
+ plane ^= 2;
}
break;
case 3:
do
{
- for (byte1 = plane = 0, bit = 1; bit; bit <<= 1, plane++)
- {
- if (GetBit(plane)) byte1 |= bit;
- }
+ for(byte1 = plane = 0, bit = 1; bit; bit <<= 1, plane++)
+ if(GetBit(plane))
+ byte1 |= bit;
*(out++) = byte1;
- }
- while (--len);
+ } while(--len);
break;
}
}
diff --git a/source/sdd1emu.h b/source/sdd1emu.h
index 07bdeb6..07162f5 100644
--- a/source/sdd1emu.h
+++ b/source/sdd1emu.h
@@ -4,5 +4,4 @@
#define SDD1EMU_H
void SDD1_decompress(uint8_t* out, uint8_t* in, int32_t output_length);
-
#endif
diff --git a/source/seta.h b/source/seta.h
index 339ba05..6244f98 100644
--- a/source/seta.h
+++ b/source/seta.h
@@ -1,6 +1,5 @@
#include "../copyright"
-#ifndef NO_SETA
#ifndef _seta_h
#define _seta_h
@@ -59,6 +58,4 @@ typedef struct
uint8_t parameters [512];
uint8_t output [512];
} ST018_Regs;
-
-#endif
#endif
diff --git a/source/seta010.c b/source/seta010.c
index 4e1a293..ca6f121 100644
--- a/source/seta010.c
+++ b/source/seta010.c
@@ -36,7 +36,6 @@ uint8_t S9xGetST010(uint32_t Address)
{
if (!(Address & 0x80000))
return 0x80;
-
if ((Address & 0xFFF) == 0x20)
return ST010.op_reg;
if ((Address & 0xFFF) == 0x21)
@@ -257,7 +256,8 @@ void ST010_OP01(int16_t x0, int16_t y0, int16_t* x1, int16_t* y1, int16_t* Quadr
*y1 >>= 1;
}
- if (*y1 == 0) *Quadrant += 0x4000;
+ if(*y1 == 0)
+ *Quadrant += 0x4000;
*Theta = (ST010_ArcTan[*y1][*x1] << 8) ^ *Quadrant;
}
@@ -284,13 +284,15 @@ void ST010_SortDrivers(uint16_t Positions, uint16_t Places[32], uint16_t Drivers
bool Sorted;
uint16_t Temp;
- if (Positions > 1)
+ if(Positions > 1)
+ {
do
{
Sorted = true;
int32_t i;
- for (i = 0; i < Positions - 1; i++)
- if (Places[i] < Places[i + 1])
+ for(i = 0; i < Positions - 1; i++)
+ {
+ if(Places[i] < Places[i + 1])
{
Temp = Places[i + 1];
Places[i + 1] = Places[i];
@@ -302,9 +304,10 @@ void ST010_SortDrivers(uint16_t Positions, uint16_t Places[32], uint16_t Drivers
Sorted = false;
}
+ }
Positions--;
- }
- while (!Sorted);
+ } while(!Sorted);
+ }
}
#define ST010_WORD(offset) (Memory.SRAM[offset + 1] << 8) | Memory.SRAM[offset]
@@ -321,7 +324,8 @@ void S9xSetST010(uint32_t Address, uint8_t Byte)
ST010.op_reg = Byte;
if ((Address & 0xFFF) == 0x21 && ST010.control_enable)
ST010.execute = Byte;
- else Memory.SRAM[Address & Memory.SRAMMask] = Byte;
+ else
+ Memory.SRAM[Address & Memory.SRAMMask] = Byte;
if (ST010.execute & 0x80)
{
@@ -462,7 +466,8 @@ void S9xSetST010(uint32_t Address, uint8_t Byte)
Memory.SRAM[0x0250 + offset] = (uint8_t)(data);
Memory.SRAM[0x0251 + offset] = (uint8_t)(data >> 8);
- if (data) data = ~data;
+ if(data)
+ data = ~data;
Memory.SRAM[0x03b0 + offset] = (uint8_t)(data);
Memory.SRAM[0x03b1 + offset] = (uint8_t)(data >> 8);
@@ -618,49 +623,45 @@ void S9xSetST010(uint32_t Address, uint8_t Byte)
wrap = true;
}
- uint16_t old_speed;
-
- old_speed = speed;
+ uint16_t old_speed = speed;
- // special case
- if (ABS(o1 - rot) == 0x8000)
+ if (ABS(o1 - rot) == 0x8000) // special case
speed = 0x100;
- // slow down for sharp curves
- else if (ABS(o1 - rot) >= 0x1000)
+ else if (ABS(o1 - rot) >= 0x1000) // slow down for sharp curves
{
uint32_t slow = ABS(o1 - rot);
slow >>= 4; // scaling
speed -= slow;
}
- // otherwise accelerate
- else
+ else // otherwise accelerate
{
speed += accel;
if (speed > speed_max)
- {
- // clip speed
- speed = speed_max;
- }
+ speed = speed_max; // clip speed
}
// prevent negative/positive overflow
if (ABS(old_speed - speed) > 0x8000)
{
- if (old_speed < speed) speed = 0;
- else speed = 0xff00;
+ if(old_speed < speed)
+ speed = 0;
+ else
+ speed = 0xff00;
}
// adjust direction by so many degrees
// be careful of negative adjustments
- if ((o1 > rot && (o1 - rot) > 0x80) ||
- (o1 < rot && (rot - o1) >= 0x80))
+ if((o1 > rot && (o1 - rot) > 0x80) || (o1 < rot && (rot - o1) >= 0x80))
{
- if (o1 < rot) rot -= 0x280;
- else if (o1 > rot) rot += 0x280;
+ if(o1 < rot)
+ rot -= 0x280;
+ else if(o1 > rot)
+ rot += 0x280;
}
- // turn off wrapping
- if (wrap) rot -= 0x8000;
+ /* turn off wrapping */
+ if(wrap)
+ rot -= 0x8000;
// now check the distances (store for later)
dx = (xpos_max << 16) - xpos;
@@ -669,8 +670,7 @@ void S9xSetST010(uint32_t Address, uint8_t Byte)
dy >>= 16;
// if we're in so many units of the target, signal it
- if ((system && (dy <= 6 && dy >= -8) && (dx <= 126 && dx >= -128)) ||
- (!system && (dx <= 6 && dx >= -8) && (dy <= 126 && dy >= -128)))
+ if ((system && (dy <= 6 && dy >= -8) && (dx <= 126 && dx >= -128)) || (!system && (dx <= 6 && dx >= -8) && (dy <= 126 && dy >= -128)))
{
// announce our new destination and flag it
xpos_max = xpos_new & 0x7FFF;
diff --git a/source/seta011.c b/source/seta011.c
index a005f98..10cd40b 100644
--- a/source/seta011.c
+++ b/source/seta011.c
@@ -1,30 +1,19 @@
#include "../copyright"
-#include <stdio.h>
#include "seta.h"
#include "memmap.h"
ST011_Regs ST011;
-
-// shougi playboard
-uint8_t board[9][9];
-
-// debug
-static int32_t line = 0;
+uint8_t board[9][9]; // shougi playboard
uint8_t S9xGetST011(uint32_t Address)
{
uint8_t t;
uint16_t address = (uint16_t) Address & 0xFFFF;
- // line counter
- line++;
-
- // status check
- if (address == 0x01)
+ if (address == 0x01) // status check
t = 0xFF;
- // read directly from s-ram
- else
+ else // read directly from s-ram
t = Memory.SRAM[address];
return t;
@@ -35,23 +24,17 @@ void S9xSetST011(uint32_t Address, uint8_t Byte)
uint16_t address = (uint16_t) Address & 0xFFFF;
static bool reset = false;
- // debug
- line++;
-
- if (!reset)
+ if (!reset) // bootup values
{
- // bootup values
ST011.waiting4command = true;
reset = true;
}
Memory.SRAM[address] = Byte;
- // op commands/data goes through this address
- if (address == 0x00)
+ if (address == 0x00) // op commands/data goes through this address
{
- // check for new commands
- if (ST011.waiting4command)
+ if (ST011.waiting4command) // check for new commands
{
ST011.waiting4command = false;
ST011.command = Byte;
@@ -84,15 +67,13 @@ void S9xSetST011(uint32_t Address, uint8_t Byte)
}
}
- if (ST011.in_count == ST011.in_index)
+ if (ST011.in_count == ST011.in_index) // Actually execute the command
{
- // Actually execute the command
ST011.waiting4command = true;
ST011.out_index = 0;
switch (ST011.command)
{
- // unknown: download playboard
- case 0x01:
+ case 0x01: // unknown: download playboard
{
// 9x9 board data: top to bottom, left to right
// Values represent piece types and ownership
@@ -101,11 +82,7 @@ void S9xSetST011(uint32_t Address, uint8_t Byte)
memcpy(board[lcv], ST011.parameters + lcv * 10, 9 * 1);
break;
}
- // unknown
- case 0x02:
- break;
- // unknown
- case 0x04:
+ case 0x04: // unknown
case 0x05:
{
// outputs
@@ -113,18 +90,13 @@ void S9xSetST011(uint32_t Address, uint8_t Byte)
Memory.SRAM[0x12E] = 0x00;
break;
}
- // unknown
- case 0x06:
- case 0x07:
- break;
- // unknown
- case 0x0E:
+ case 0x0E: // unknown
{
// outputs
Memory.SRAM[0x12C] = 0x00;
Memory.SRAM[0x12D] = 0x00;
+ break;
}
- break;
}
}
}
diff --git a/source/seta018.c b/source/seta018.c
index 840e479..a608960 100644
--- a/source/seta018.c
+++ b/source/seta018.c
@@ -5,15 +5,11 @@
ST018_Regs ST018;
-static int32_t line; // line counter
-
uint8_t S9xGetST018(uint32_t Address)
{
uint8_t t = 0; // Initialise to some value for the compiler
uint16_t address = (uint16_t) Address & 0xFFFF;
- line++;
-
// these roles may be flipped
// op output
if (address == 0x3804)
@@ -28,8 +24,7 @@ uint8_t S9xGetST018(uint32_t Address)
else
t = 0x81;
}
- // status register
- else if (address == 0x3800)
+ else if (address == 0x3800) // status register
t = ST018.status;
return t;
@@ -39,26 +34,20 @@ void S9xSetST018(uint8_t Byte, uint32_t Address)
{
uint16_t address = (uint16_t) Address & 0xFFFF;
static bool reset = false;
- line++;
- if (!reset)
+ if (!reset) // bootup values
{
- // bootup values
ST018.waiting4command = true;
ST018.part_command = 0;
reset = true;
}
Memory.SRAM[address] = Byte;
+ ST018.status = 0x00; // default status for now
- // default status for now
- ST018.status = 0x00;
-
- // op data goes through this address
- if (address == 0x3804)
+ if (address == 0x3804) // op data goes through this address
{
- // check for new commands: 3 bytes length
- if (ST018.waiting4command && ST018.part_command == 2)
+ if (ST018.waiting4command && ST018.part_command == 2) // check for new commands: 3 bytes length
{
ST018.waiting4command = false;
ST018.command <<= 8;
@@ -78,32 +67,28 @@ void S9xSetST018(uint8_t Byte, uint32_t Address)
break;
}
}
- else if (ST018.waiting4command)
+ else if (ST018.waiting4command) // 3-byte commands
{
- // 3-byte commands
ST018.part_command++;
ST018.command <<= 8;
ST018.command |= Byte;
}
}
- // extra parameters
- else if (address == 0x3802)
+ else if (address == 0x3802) // extra parameters
{
ST018.parameters[ST018.in_index] = Byte;
ST018.in_index++;
}
- if (ST018.in_count == ST018.in_index)
+ if (ST018.in_count == ST018.in_index) // Actually execute the command
{
- // Actually execute the command
ST018.waiting4command = true;
ST018.in_index = 0;
ST018.out_index = 0;
switch (ST018.command)
{
- // hardware check?
case 0x0100:
- case 0xFF00:
+ case 0xFF00: // hardware check?
ST018.waiting4command = false;
ST018.pass++;
if (ST018.pass == 1)
diff --git a/source/snes9x.h b/source/snes9x.h
index 3107494..536642b 100644
--- a/source/snes9x.h
+++ b/source/snes9x.h
@@ -131,16 +131,6 @@ typedef struct
int32_t HBlankStart;
int32_t CyclesPercentage;
bool DisableIRQ;
- bool Paused;
- bool ForcedPause;
- bool StopEmulation;
-
- /* Tracing options */
- bool TraceDMA;
- bool TraceHDMA;
- bool TraceVRAM;
- bool TraceUnknownRegisters;
- bool TraceDSP;
/* Joystick options */
bool JoystickEnabled;
@@ -152,7 +142,6 @@ typedef struct
uint32_t FrameTimePAL;
uint32_t FrameTimeNTSC;
uint32_t FrameTime;
- uint32_t SkipFrames;
/* ROM image options */
bool ForceLoROM;
@@ -179,7 +168,6 @@ typedef struct
bool SuperScope;
bool SRTC;
uint32_t ControllerOption;
- bool ShutdownMaster;
bool MultiPlayer5Master;
bool SuperScopeMaster;
bool MouseMaster;
@@ -212,18 +200,6 @@ typedef struct
bool Mute;
bool NextAPUEnabled;
- /* Graphics options */
- bool Transparency;
- bool Mode7Interpolate;
-
- /* SNES graphics options */
- bool BGLayering;
- bool DisableGraphicWindows;
- bool ForceTransparency;
- bool ForceNoTransparency;
- bool DisableHDMA;
- bool DisplayFrameRate;
-
/* Others */
bool ApplyCheats;
@@ -231,35 +207,16 @@ typedef struct
bool StarfoxHack;
bool WinterGold;
bool BS; /* Japanese Satellite System games. */
- uint8_t APURAMInitialValue;
- bool SampleCatchup;
bool JustifierMaster;
bool Justifier;
bool SecondJustifier;
int8_t SETA;
- bool TakeScreenshot;
- int8_t StretchScreenshots;
- uint16_t DisplayColor;
- int32_t SoundDriver;
- int32_t AIDOShmId;
- bool NoPatch;
- bool ForceInterleaveGD24;
} SSettings;
-typedef struct
-{
- uint8_t APU_OutPorts_ReturnValueFix;
- uint8_t SoundEnvelopeHeightReading2;
- uint8_t SRAMInitialValue;
- bool EchoOnlyOutput;
-} SSNESGameFixes;
-
extern SSettings Settings;
extern SCPUState CPU;
-extern SSNESGameFixes SNESGameFixes;
extern char String [513];
void S9xSetPause(uint32_t mask);
void S9xClearPause(uint32_t mask);
-
#endif
diff --git a/source/soundux.c b/source/soundux.c
index 86db199..f053290 100644
--- a/source/soundux.c
+++ b/source/soundux.c
@@ -81,8 +81,6 @@ uint32_t KeyOffERate [10];
#define VOL_DIV16 0x0080
#define ENVX_SHIFT 24
-void DecodeBlockAsm(int8_t*, int16_t*, int32_t*, int32_t*);
-
// F is channel's current frequency and M is the 16-bit modulation waveform
// from the previous channel multiplied by the current envelope volume level.
#define PITCH_MOD(F,M) ((F) * ((((uint32_t) (M)) + 0x800000) >> 16) >> 7)
@@ -167,7 +165,7 @@ void S9xSetSoundVolume(int32_t channel, int16_t volume_left, int16_t volume_righ
void S9xSetMasterVolume(int16_t volume_left, int16_t volume_right)
{
- if (Settings.DisableMasterVolume || SNESGameFixes.EchoOnlyOutput)
+ if (Settings.DisableMasterVolume)
SoundData.master_volume [0] = SoundData.master_volume [1] = 127;
else
{
@@ -211,7 +209,8 @@ void S9xSetEchoFeedback(int32_t feedback)
void S9xSetEchoDelay(int32_t delay)
{
- SoundData.echo_buffer_size = (delay * so.playback_rate) >> 5;
+ SoundData.echo_buffer_size = (512 * delay * so.playback_rate) / 32000;
+ SoundData.echo_buffer_size <<= 1;
if (SoundData.echo_buffer_size)
SoundData.echo_ptr %= SoundData.echo_buffer_size;
else
@@ -323,20 +322,6 @@ void S9xSetEnvelopeHeight(int32_t channel, int32_t level)
S9xAPUSetEndOfSample(channel, ch);
}
-int32_t S9xGetEnvelopeHeight(int32_t channel)
-{
- if ((Settings.SoundEnvelopeHeightReading ||
- SNESGameFixes.SoundEnvelopeHeightReading2) &&
- SoundData.channels[channel].state != SOUND_SILENT &&
- SoundData.channels[channel].state != SOUND_GAIN)
- return SoundData.channels[channel].envx;
-
- if (SNESGameFixes.SoundEnvelopeHeightReading2 && SoundData.channels[channel].state != SOUND_SILENT)
- return SoundData.channels[channel].envx;
-
- return 0;
-}
-
void S9xSetSoundFrequency(int32_t channel, int32_t hertz) // hertz [0~64K<<1]
{
if (SoundData.channels[channel].type == SOUND_NOISE)
@@ -362,7 +347,7 @@ void DecodeBlock(Channel* ch)
uint8_t shift;
int8_t sample1, sample2;
- if (ch->block_pointer >= 0x10000 - 9)
+ if (ch->block_pointer > 0x10000 - 9)
{
ch->last_block = true;
ch->loop = false;
diff --git a/source/spc700.c b/source/spc700.c
index 53bbb1b..fed54ff 100644
--- a/source/spc700.c
+++ b/source/spc700.c
@@ -19,8 +19,8 @@ uint8_t Work8 = 0;
uint16_t Work16 = 0;
uint32_t Work32 = 0;
-#define OP1 (*(IAPU.PC + 1))
-#define OP2 (*(IAPU.PC + 2))
+#define OP1 IAPU.PC[1]
+#define OP2 IAPU.PC[2]
#define APUShutdown() \
if (Settings.Shutdown && (IAPU.PC == IAPU.WaitAddress1 || IAPU.PC == IAPU.WaitAddress2)) \
@@ -28,9 +28,9 @@ uint32_t Work32 = 0;
if (IAPU.WaitCounter == 0) \
{ \
if (!ICPU.CPUExecuting) \
- APU.Cycles = CPU.Cycles = CPU.NextEvent; \
+ APU.Cycles = CPU.Cycles = CPU.NextEvent; \
else \
- IAPU.APUExecuting = false; \
+ IAPU.APUExecuting = false; \
} \
else if (IAPU.WaitCounter >= 2) \
IAPU.WaitCounter = 1; \
@@ -39,19 +39,19 @@ uint32_t Work32 = 0;
}
#define APUSetZN8(b) \
- IAPU._Zero = (b);
+ IAPU._Zero = (b)
#define APUSetZN16(w) \
- IAPU._Zero = ((w) != 0) | ((w) >> 8);
+ IAPU._Zero = ((w) != 0) | ((w) >> 8)
#define TCALL(n) \
{\
- SPC700_PushW (IAPU.PC - IAPU.RAM + 1); \
+ PushW (IAPU.PC - IAPU.RAM + 1); \
IAPU.PC = IAPU.RAM + S9xAPUGetByte(0xffc0 + ((15 - n) << 1)) + \
(S9xAPUGetByte(0xffc1 + ((15 - n) << 1)) << 8); \
}
-#define SBC(a,b) \
+#define SBC(a, b) \
Int16 = (int16_t) (a) - (int16_t) (b) + (int16_t) (APUCheckCarry ()) - 1; \
IAPU._Carry = Int16 >= 0; \
if ((((a) ^ (b)) & 0x80) && (((a) ^ (uint8_t) Int16) & 0x80)) \
@@ -80,62 +80,68 @@ uint32_t Work32 = 0;
#define CMP(a,b) \
Int16 = (int16_t) (a) - (int16_t) (b); \
IAPU._Carry = Int16 >= 0; \
- APUSetZN8((uint8_t) Int16);
+ APUSetZN8((uint8_t) Int16)
#define ASL(b) \
IAPU._Carry = ((b) & 0x80) != 0; \
(b) <<= 1; \
- APUSetZN8 (b);
+ APUSetZN8 (b)
+
#define LSR(b) \
IAPU._Carry = (b) & 1; \
(b) >>= 1; \
- APUSetZN8 (b);
+ APUSetZN8 (b)
+
#define ROL(b) \
Work16 = ((b) << 1) | APUCheckCarry (); \
IAPU._Carry = Work16 >= 0x100; \
(b) = (uint8_t) Work16; \
- APUSetZN8 (b);
+ APUSetZN8 (b)
+
#define ROR(b) \
Work16 = (b) | ((uint16_t) APUCheckCarry () << 8); \
IAPU._Carry = (uint8_t) Work16 & 1; \
Work16 >>= 1; \
(b) = (uint8_t) Work16; \
- APUSetZN8 (b);
+ APUSetZN8 (b)
#define Push(b) \
- *(IAPU.RAM + 0x100 + IAPU.Registers.S) = b; \
- IAPU.Registers.S--;
+ IAPU.RAM[0x100 + IAPU.Registers.S] = b; \
+ IAPU.Registers.S--
#define Pop(b) \
IAPU.Registers.S++; \
- (b) = *(IAPU.RAM + 0x100 + IAPU.Registers.S);
+ (b) = *(IAPU.RAM + 0x100 + IAPU.Registers.S)
#ifdef FAST_LSB_WORD_ACCESS
-#define SPC700_PushW(w) \
- if (IAPU.Registers.S == 0) {\
+#define PushW(w) \
+ if (IAPU.Registers.S == 0) \
+ {\
*(IAPU.RAM + 0x1ff) = (w); \
*(IAPU.RAM + 0x100) = ((w) >> 8); \
- } else { \
+ } \
+ else \
*(uint16_t *) (IAPU.RAM + 0xff + IAPU.Registers.S) = w; \
- }\
- IAPU.Registers.S -= 2;
+ IAPU.Registers.S -= 2
+
#define PopW(w) \
IAPU.Registers.S += 2; \
if (IAPU.Registers.S == 0) \
(w) = *(IAPU.RAM + 0x1ff) | (*(IAPU.RAM + 0x100) << 8); \
else \
- (w) = *(uint16_t *) (IAPU.RAM + 0xff + IAPU.Registers.S);
+ (w) = *(uint16_t *) (IAPU.RAM + 0xff + IAPU.Registers.S)
#else
-#define SPC700_PushW(w) \
+#define PushW(w) \
*(IAPU.RAM + 0xff + IAPU.Registers.S) = w; \
*(IAPU.RAM + 0x100 + IAPU.Registers.S) = ((w) >> 8); \
- IAPU.Registers.S -= 2;
+ IAPU.Registers.S -= 2
+
#define PopW(w) \
IAPU.Registers.S += 2; \
if(IAPU.Registers.S == 0) \
(w) = *(IAPU.RAM + 0x1ff) | (*(IAPU.RAM + 0x100) << 8); \
else \
- (w) = *(IAPU.RAM + 0xff + IAPU.Registers.S) + (*(IAPU.RAM + 0x100 + IAPU.Registers.S) << 8);
+ (w) = *(IAPU.RAM + 0xff + IAPU.Registers.S) + (*(IAPU.RAM + 0x100 + IAPU.Registers.S) << 8)
#endif
#define Relative() \
@@ -170,6 +176,7 @@ uint32_t Work32 = 0;
#define IndexedXIndirect() \
IAPU.Address = *(IAPU.DirectPage + ((OP1 + IAPU.Registers.X) & 0xff)) + \
(*(IAPU.DirectPage + ((OP1 + IAPU.Registers.X + 1) & 0xff)) << 8);
+
#define Absolute() \
IAPU.Address = OP1 + (OP2 << 8);
@@ -190,104 +197,103 @@ uint32_t Work32 = 0;
IAPU.Registers.YA.B.Y;
#endif
-void Apu00()
+void Apu00(void) // NOP
{
- // NOP
IAPU.PC++;
}
-void Apu01()
+void Apu01(void)
{
- TCALL(0)
+ TCALL(0);
}
-void Apu11()
+void Apu11(void)
{
- TCALL(1)
+ TCALL(1);
}
-void Apu21()
+void Apu21(void)
{
- TCALL(2)
+ TCALL(2);
}
-void Apu31()
+void Apu31(void)
{
- TCALL(3)
+ TCALL(3);
}
-void Apu41()
+void Apu41(void)
{
- TCALL(4)
+ TCALL(4);
}
-void Apu51()
+void Apu51(void)
{
- TCALL(5)
+ TCALL(5);
}
-void Apu61()
+void Apu61(void)
{
- TCALL(6)
+ TCALL(6);
}
-void Apu71()
+void Apu71(void)
{
- TCALL(7)
+ TCALL(7);
}
-void Apu81()
+void Apu81(void)
{
- TCALL(8)
+ TCALL(8);
}
-void Apu91()
+void Apu91(void)
{
- TCALL(9)
+ TCALL(9);
}
-void ApuA1()
+void ApuA1(void)
{
- TCALL(10)
+ TCALL(10);
}
-void ApuB1()
+void ApuB1(void)
{
- TCALL(11)
+ TCALL(11);
}
-void ApuC1()
+void ApuC1(void)
{
- TCALL(12)
+ TCALL(12);
}
-void ApuD1()
+void ApuD1(void)
{
- TCALL(13)
+ TCALL(13);
}
-void ApuE1()
+void ApuE1(void)
{
- TCALL(14)
+ TCALL(14);
}
-void ApuF1()
+void ApuF1(void)
{
- TCALL(15)
+ TCALL(15);
}
-void Apu3F() // CALL absolute
+void Apu3F(void) // CALL absolute
{
Absolute();
// 0xB6f for Star Fox 2
- SPC700_PushW(IAPU.PC + 3 - IAPU.RAM);
+ PushW(IAPU.PC + 3 - IAPU.RAM);
IAPU.PC = IAPU.RAM + IAPU.Address;
}
-void Apu4F() // PCALL $XX
+void Apu4F(void) // PCALL $XX
{
Work8 = OP1;
- SPC700_PushW(IAPU.PC + 2 - IAPU.RAM);
+ PushW(IAPU.PC + 2 - IAPU.RAM);
IAPU.PC = IAPU.RAM + 0xff00 + Work8;
}
@@ -295,42 +301,42 @@ void Apu4F() // PCALL $XX
S9xAPUSetByteZ ((uint8_t) (S9xAPUGetByteZ (OP1 ) | (1 << (b))), OP1); \
IAPU.PC += 2
-void Apu02()
+void Apu02(void)
{
SET(0);
}
-void Apu22()
+void Apu22(void)
{
SET(1);
}
-void Apu42()
+void Apu42(void)
{
SET(2);
}
-void Apu62()
+void Apu62(void)
{
SET(3);
}
-void Apu82()
+void Apu82(void)
{
SET(4);
}
-void ApuA2()
+void ApuA2(void)
{
SET(5);
}
-void ApuC2()
+void ApuC2(void)
{
SET(6);
}
-void ApuE2()
+void ApuE2(void)
{
SET(7);
}
@@ -339,42 +345,42 @@ void ApuE2()
S9xAPUSetByteZ ((uint8_t) (S9xAPUGetByteZ (OP1) & ~(1 << (b))), OP1); \
IAPU.PC += 2;
-void Apu12()
+void Apu12(void)
{
CLR(0);
}
-void Apu32()
+void Apu32(void)
{
CLR(1);
}
-void Apu52()
+void Apu52(void)
{
CLR(2);
}
-void Apu72()
+void Apu72(void)
{
CLR(3);
}
-void Apu92()
+void Apu92(void)
{
CLR(4);
}
-void ApuB2()
+void ApuB2(void)
{
CLR(5);
}
-void ApuD2()
+void ApuD2(void)
{
CLR(6);
}
-void ApuF2()
+void ApuF2(void)
{
CLR(7);
}
@@ -390,42 +396,42 @@ if (S9xAPUGetByteZ (Work8) & (1 << (b))) \
else \
IAPU.PC += 3
-void Apu03()
+void Apu03(void)
{
BBS(0);
}
-void Apu23()
+void Apu23(void)
{
BBS(1);
}
-void Apu43()
+void Apu43(void)
{
BBS(2);
}
-void Apu63()
+void Apu63(void)
{
BBS(3);
}
-void Apu83()
+void Apu83(void)
{
BBS(4);
}
-void ApuA3()
+void ApuA3(void)
{
BBS(5);
}
-void ApuC3()
+void ApuC3(void)
{
BBS(6);
}
-void ApuE3()
+void ApuE3(void)
{
BBS(7);
}
@@ -441,47 +447,47 @@ if (!(S9xAPUGetByteZ (Work8) & (1 << (b)))) \
else \
IAPU.PC += 3
-void Apu13()
+void Apu13(void)
{
BBC(0);
}
-void Apu33()
+void Apu33(void)
{
BBC(1);
}
-void Apu53()
+void Apu53(void)
{
BBC(2);
}
-void Apu73()
+void Apu73(void)
{
BBC(3);
}
-void Apu93()
+void Apu93(void)
{
BBC(4);
}
-void ApuB3()
+void ApuB3(void)
{
BBC(5);
}
-void ApuD3()
+void ApuD3(void)
{
BBC(6);
}
-void ApuF3()
+void ApuF3(void)
{
BBC(7);
}
-void Apu04()
+void Apu04(void)
{
// OR A,dp
IAPU.Registers.YA.B.A |= S9xAPUGetByteZ(OP1);
@@ -489,7 +495,7 @@ void Apu04()
IAPU.PC += 2;
}
-void Apu05()
+void Apu05(void)
{
// OR A,abs
Absolute();
@@ -498,7 +504,7 @@ void Apu05()
IAPU.PC += 3;
}
-void Apu06()
+void Apu06(void)
{
// OR A,(X)
IAPU.Registers.YA.B.A |= S9xAPUGetByteZ(IAPU.Registers.X);
@@ -506,7 +512,7 @@ void Apu06()
IAPU.PC++;
}
-void Apu07()
+void Apu07(void)
{
// OR A,(dp+X)
IndexedXIndirect();
@@ -515,7 +521,7 @@ void Apu07()
IAPU.PC += 2;
}
-void Apu08()
+void Apu08(void)
{
// OR A,#00
IAPU.Registers.YA.B.A |= OP1;
@@ -523,7 +529,7 @@ void Apu08()
IAPU.PC += 2;
}
-void Apu09()
+void Apu09(void)
{
// OR dp(dest),dp(src)
Work8 = S9xAPUGetByteZ(OP1);
@@ -533,7 +539,7 @@ void Apu09()
IAPU.PC += 3;
}
-void Apu14()
+void Apu14(void)
{
// OR A,dp+X
IAPU.Registers.YA.B.A |= S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -541,7 +547,7 @@ void Apu14()
IAPU.PC += 2;
}
-void Apu15()
+void Apu15(void)
{
// OR A,abs+X
AbsoluteX();
@@ -550,7 +556,7 @@ void Apu15()
IAPU.PC += 3;
}
-void Apu16()
+void Apu16(void)
{
// OR A,abs+Y
AbsoluteY();
@@ -559,7 +565,7 @@ void Apu16()
IAPU.PC += 3;
}
-void Apu17()
+void Apu17(void)
{
// OR A,(dp)+Y
IndirectIndexedY();
@@ -568,7 +574,7 @@ void Apu17()
IAPU.PC += 2;
}
-void Apu18()
+void Apu18(void)
{
// OR dp,#00
Work8 = OP1;
@@ -578,7 +584,7 @@ void Apu18()
IAPU.PC += 3;
}
-void Apu19()
+void Apu19(void)
{
// OR (X),(Y)
Work8 = S9xAPUGetByteZ(IAPU.Registers.X) | S9xAPUGetByteZ(IAPU.Registers.YA.B.Y);
@@ -587,55 +593,47 @@ void Apu19()
IAPU.PC++;
}
-void Apu0A()
+void Apu0A(void)
{
// OR1 C,membit
MemBit();
if (!APUCheckCarry())
- {
if (S9xAPUGetByte(IAPU.Address) & (1 << IAPU.Bit))
APUSetCarry();
- }
IAPU.PC += 3;
}
-void Apu2A()
+void Apu2A(void)
{
// OR1 C,not membit
MemBit();
if (!APUCheckCarry())
- {
if (!(S9xAPUGetByte(IAPU.Address) & (1 << IAPU.Bit)))
APUSetCarry();
- }
IAPU.PC += 3;
}
-void Apu4A()
+void Apu4A(void)
{
// AND1 C,membit
MemBit();
if (APUCheckCarry())
- {
if (!(S9xAPUGetByte(IAPU.Address) & (1 << IAPU.Bit)))
APUClearCarry();
- }
IAPU.PC += 3;
}
-void Apu6A()
+void Apu6A(void)
{
// AND1 C, not membit
MemBit();
if (APUCheckCarry())
- {
if ((S9xAPUGetByte(IAPU.Address) & (1 << IAPU.Bit)))
APUClearCarry();
- }
IAPU.PC += 3;
}
-void Apu8A()
+void Apu8A(void)
{
// EOR1 C, membit
MemBit();
@@ -649,7 +647,7 @@ void Apu8A()
IAPU.PC += 3;
}
-void ApuAA()
+void ApuAA(void)
{
// MOV1 C,membit
MemBit();
@@ -660,7 +658,7 @@ void ApuAA()
IAPU.PC += 3;
}
-void ApuCA()
+void ApuCA(void)
{
// MOV1 membit,C
MemBit();
@@ -671,7 +669,7 @@ void ApuCA()
IAPU.PC += 3;
}
-void ApuEA()
+void ApuEA(void)
{
// NOT1 membit
MemBit();
@@ -679,7 +677,7 @@ void ApuEA()
IAPU.PC += 3;
}
-void Apu0B()
+void Apu0B(void)
{
// ASL dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -688,7 +686,7 @@ void Apu0B()
IAPU.PC += 2;
}
-void Apu0C()
+void Apu0C(void)
{
// ASL abs
Absolute();
@@ -698,7 +696,7 @@ void Apu0C()
IAPU.PC += 3;
}
-void Apu1B()
+void Apu1B(void)
{
// ASL dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -707,14 +705,14 @@ void Apu1B()
IAPU.PC += 2;
}
-void Apu1C()
+void Apu1C(void)
{
// ASL A
ASL(IAPU.Registers.YA.B.A);
IAPU.PC++;
}
-void Apu0D()
+void Apu0D(void)
{
// PUSH PSW
S9xAPUPackStatus();
@@ -722,28 +720,28 @@ void Apu0D()
IAPU.PC++;
}
-void Apu2D()
+void Apu2D(void)
{
// PUSH A
Push(IAPU.Registers.YA.B.A);
IAPU.PC++;
}
-void Apu4D()
+void Apu4D(void)
{
// PUSH X
Push(IAPU.Registers.X);
IAPU.PC++;
}
-void Apu6D()
+void Apu6D(void)
{
// PUSH Y
Push(IAPU.Registers.YA.B.Y);
IAPU.PC++;
}
-void Apu8E()
+void Apu8E(void)
{
// POP PSW
Pop(IAPU.Registers.P);
@@ -755,28 +753,28 @@ void Apu8E()
IAPU.PC++;
}
-void ApuAE()
+void ApuAE(void)
{
// POP A
Pop(IAPU.Registers.YA.B.A);
IAPU.PC++;
}
-void ApuCE()
+void ApuCE(void)
{
// POP X
Pop(IAPU.Registers.X);
IAPU.PC++;
}
-void ApuEE()
+void ApuEE(void)
{
// POP Y
Pop(IAPU.Registers.YA.B.Y);
IAPU.PC++;
}
-void Apu0E()
+void Apu0E(void)
{
// TSET1 abs
Absolute();
@@ -787,7 +785,7 @@ void Apu0E()
IAPU.PC += 3;
}
-void Apu4E()
+void Apu4E(void)
{
// TCLR1 abs
Absolute();
@@ -798,10 +796,10 @@ void Apu4E()
IAPU.PC += 3;
}
-void Apu0F()
+void Apu0F(void)
{
// BRK
- SPC700_PushW(IAPU.PC + 1 - IAPU.RAM);
+ PushW(IAPU.PC + 1 - IAPU.RAM);
S9xAPUPackStatus();
Push(IAPU.Registers.P);
APUSetBreak();
@@ -809,14 +807,14 @@ void Apu0F()
IAPU.PC = IAPU.RAM + S9xAPUGetByte(0xffde) + (S9xAPUGetByte(0xffdf) << 8);
}
-void ApuEF()
+void ApuEF(void)
{
// SLEEP
APU.TimerEnabled[0] = APU.TimerEnabled[1] = APU.TimerEnabled[2] = false;
IAPU.APUExecuting = false;
}
-void ApuFF()
+void ApuFF(void)
{
// STOP
APU.TimerEnabled[0] = APU.TimerEnabled[1] = APU.TimerEnabled[2] = false;
@@ -824,7 +822,7 @@ void ApuFF()
Settings.APUEnabled = false; // re-enabled on next APU reset
}
-void Apu10()
+void Apu10(void)
{
// BPL
Relative();
@@ -838,7 +836,7 @@ void Apu10()
IAPU.PC += 2;
}
-void Apu30()
+void Apu30(void)
{
// BMI
Relative();
@@ -852,7 +850,7 @@ void Apu30()
IAPU.PC += 2;
}
-void Apu90()
+void Apu90(void)
{
// BCC
Relative();
@@ -866,7 +864,7 @@ void Apu90()
IAPU.PC += 2;
}
-void ApuB0()
+void ApuB0(void)
{
// BCS
Relative();
@@ -880,7 +878,7 @@ void ApuB0()
IAPU.PC += 2;
}
-void ApuD0()
+void ApuD0(void)
{
// BNE
Relative();
@@ -894,7 +892,7 @@ void ApuD0()
IAPU.PC += 2;
}
-void ApuF0()
+void ApuF0(void)
{
// BEQ
Relative();
@@ -908,7 +906,7 @@ void ApuF0()
IAPU.PC += 2;
}
-void Apu50()
+void Apu50(void)
{
// BVC
Relative();
@@ -921,7 +919,7 @@ void Apu50()
IAPU.PC += 2;
}
-void Apu70()
+void Apu70(void)
{
// BVS
Relative();
@@ -934,28 +932,28 @@ void Apu70()
IAPU.PC += 2;
}
-void Apu2F()
+void Apu2F(void)
{
// BRA
Relative();
IAPU.PC = IAPU.RAM + (uint16_t) Int16;
}
-void Apu80()
+void Apu80(void)
{
// SETC
APUSetCarry();
IAPU.PC++;
}
-void ApuED()
+void ApuED(void)
{
// NOTC
IAPU._Carry ^= 1;
IAPU.PC++;
}
-void Apu40()
+void Apu40(void)
{
// SETP
APUSetDirectPage();
@@ -963,7 +961,7 @@ void Apu40()
IAPU.PC++;
}
-void Apu1A()
+void Apu1A(void)
{
// DECW dp
Work16 = S9xAPUGetByteZ(OP1) + (S9xAPUGetByteZ(OP1 + 1) << 8) - 1;
@@ -973,7 +971,7 @@ void Apu1A()
IAPU.PC += 2;
}
-void Apu5A()
+void Apu5A(void)
{
// CMPW YA,dp
Work16 = S9xAPUGetByteZ(OP1) + (S9xAPUGetByteZ(OP1 + 1) << 8);
@@ -983,7 +981,7 @@ void Apu5A()
IAPU.PC += 2;
}
-void Apu3A()
+void Apu3A(void)
{
// INCW dp
Work16 = S9xAPUGetByteZ(OP1) + (S9xAPUGetByteZ(OP1 + 1) << 8) + 1;
@@ -993,7 +991,7 @@ void Apu3A()
IAPU.PC += 2;
}
-void Apu7A()
+void Apu7A(void)
{
// ADDW YA,dp
Work16 = S9xAPUGetByteZ(OP1) + (S9xAPUGetByteZ(OP1 + 1) << 8);
@@ -1011,15 +1009,14 @@ void Apu7A()
IAPU.PC += 2;
}
-void Apu9A()
+void Apu9A(void)
{
// SUBW YA,dp
Work16 = S9xAPUGetByteZ(OP1) + (S9xAPUGetByteZ(OP1 + 1) << 8);
Int32 = (int32_t) IAPU.Registers.YA.W - (int32_t) Work16;
APUClearHalfCarry();
IAPU._Carry = Int32 >= 0;
- if (((IAPU.Registers.YA.W ^ Work16) & 0x8000) &&
- ((IAPU.Registers.YA.W ^ (uint16_t) Int32) & 0x8000))
+ if (((IAPU.Registers.YA.W ^ Work16) & 0x8000) && ((IAPU.Registers.YA.W ^ (uint16_t) Int32) & 0x8000))
APUSetOverflow();
else
APUClearOverflow();
@@ -1031,7 +1028,7 @@ void Apu9A()
IAPU.PC += 2;
}
-void ApuBA()
+void ApuBA(void)
{
// MOVW YA,dp
IAPU.Registers.YA.B.A = S9xAPUGetByteZ(OP1);
@@ -1040,7 +1037,7 @@ void ApuBA()
IAPU.PC += 2;
}
-void ApuDA()
+void ApuDA(void)
{
// MOVW dp,YA
S9xAPUSetByteZ(IAPU.Registers.YA.B.A, OP1);
@@ -1048,7 +1045,7 @@ void ApuDA()
IAPU.PC += 2;
}
-void Apu64()
+void Apu64(void)
{
// CMP A,dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1056,7 +1053,7 @@ void Apu64()
IAPU.PC += 2;
}
-void Apu65()
+void Apu65(void)
{
// CMP A,abs
Absolute();
@@ -1065,7 +1062,7 @@ void Apu65()
IAPU.PC += 3;
}
-void Apu66()
+void Apu66(void)
{
// CMP A,(X)
Work8 = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1073,7 +1070,7 @@ void Apu66()
IAPU.PC++;
}
-void Apu67()
+void Apu67(void)
{
// CMP A,(dp+X)
IndexedXIndirect();
@@ -1082,7 +1079,7 @@ void Apu67()
IAPU.PC += 2;
}
-void Apu68()
+void Apu68(void)
{
// CMP A,#00
Work8 = OP1;
@@ -1090,7 +1087,7 @@ void Apu68()
IAPU.PC += 2;
}
-void Apu69()
+void Apu69(void)
{
// CMP dp(dest), dp(src)
W1 = S9xAPUGetByteZ(OP1);
@@ -1099,7 +1096,7 @@ void Apu69()
IAPU.PC += 3;
}
-void Apu74()
+void Apu74(void)
{
// CMP A, dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1107,7 +1104,7 @@ void Apu74()
IAPU.PC += 2;
}
-void Apu75()
+void Apu75(void)
{
// CMP A,abs+X
AbsoluteX();
@@ -1116,7 +1113,7 @@ void Apu75()
IAPU.PC += 3;
}
-void Apu76()
+void Apu76(void)
{
// CMP A, abs+Y
AbsoluteY();
@@ -1125,7 +1122,7 @@ void Apu76()
IAPU.PC += 3;
}
-void Apu77()
+void Apu77(void)
{
// CMP A,(dp)+Y
IndirectIndexedY();
@@ -1134,7 +1131,7 @@ void Apu77()
IAPU.PC += 2;
}
-void Apu78()
+void Apu78(void)
{
// CMP dp,#00
Work8 = OP1;
@@ -1143,7 +1140,7 @@ void Apu78()
IAPU.PC += 3;
}
-void Apu79()
+void Apu79(void)
{
// CMP (X),(Y)
W1 = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1152,7 +1149,7 @@ void Apu79()
IAPU.PC++;
}
-void Apu1E()
+void Apu1E(void)
{
// CMP X,abs
Absolute();
@@ -1161,7 +1158,7 @@ void Apu1E()
IAPU.PC += 3;
}
-void Apu3E()
+void Apu3E(void)
{
// CMP X,dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1169,14 +1166,14 @@ void Apu3E()
IAPU.PC += 2;
}
-void ApuC8()
+void ApuC8(void)
{
// CMP X,#00
CMP(IAPU.Registers.X, OP1);
IAPU.PC += 2;
}
-void Apu5E()
+void Apu5E(void)
{
// CMP Y,abs
Absolute();
@@ -1185,7 +1182,7 @@ void Apu5E()
IAPU.PC += 3;
}
-void Apu7E()
+void Apu7E(void)
{
// CMP Y,dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1193,7 +1190,7 @@ void Apu7E()
IAPU.PC += 2;
}
-void ApuAD()
+void ApuAD(void)
{
// CMP Y,#00
Work8 = OP1;
@@ -1201,22 +1198,21 @@ void ApuAD()
IAPU.PC += 2;
}
-void Apu1F()
+void Apu1F(void)
{
// JMP (abs+X)
Absolute();
- IAPU.PC = IAPU.RAM + S9xAPUGetByte(IAPU.Address + IAPU.Registers.X) +
- (S9xAPUGetByte(IAPU.Address + IAPU.Registers.X + 1) << 8);
+ IAPU.PC = IAPU.RAM + S9xAPUGetByte(IAPU.Address + IAPU.Registers.X) + (S9xAPUGetByte(IAPU.Address + IAPU.Registers.X + 1) << 8);
}
-void Apu5F()
+void Apu5F(void)
{
// JMP abs
Absolute();
IAPU.PC = IAPU.RAM + IAPU.Address;
}
-void Apu20()
+void Apu20(void)
{
// CLRP
APUClearDirectPage();
@@ -1224,14 +1220,14 @@ void Apu20()
IAPU.PC++;
}
-void Apu60()
+void Apu60(void)
{
// CLRC
APUClearCarry();
IAPU.PC++;
}
-void ApuE0()
+void ApuE0(void)
{
// CLRV
APUClearHalfCarry();
@@ -1239,7 +1235,7 @@ void ApuE0()
IAPU.PC++;
}
-void Apu24()
+void Apu24(void)
{
// AND A,dp
IAPU.Registers.YA.B.A &= S9xAPUGetByteZ(OP1);
@@ -1247,7 +1243,7 @@ void Apu24()
IAPU.PC += 2;
}
-void Apu25()
+void Apu25(void)
{
// AND A,abs
Absolute();
@@ -1256,7 +1252,7 @@ void Apu25()
IAPU.PC += 3;
}
-void Apu26()
+void Apu26(void)
{
// AND A,(X)
IAPU.Registers.YA.B.A &= S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1264,7 +1260,7 @@ void Apu26()
IAPU.PC++;
}
-void Apu27()
+void Apu27(void)
{
// AND A,(dp+X)
IndexedXIndirect();
@@ -1273,7 +1269,7 @@ void Apu27()
IAPU.PC += 2;
}
-void Apu28()
+void Apu28(void)
{
// AND A,#00
IAPU.Registers.YA.B.A &= OP1;
@@ -1281,7 +1277,7 @@ void Apu28()
IAPU.PC += 2;
}
-void Apu29()
+void Apu29(void)
{
// AND dp(dest),dp(src)
Work8 = S9xAPUGetByteZ(OP1);
@@ -1291,7 +1287,7 @@ void Apu29()
IAPU.PC += 3;
}
-void Apu34()
+void Apu34(void)
{
// AND A,dp+X
IAPU.Registers.YA.B.A &= S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1299,7 +1295,7 @@ void Apu34()
IAPU.PC += 2;
}
-void Apu35()
+void Apu35(void)
{
// AND A,abs+X
AbsoluteX();
@@ -1308,7 +1304,7 @@ void Apu35()
IAPU.PC += 3;
}
-void Apu36()
+void Apu36(void)
{
// AND A,abs+Y
AbsoluteY();
@@ -1317,7 +1313,7 @@ void Apu36()
IAPU.PC += 3;
}
-void Apu37()
+void Apu37(void)
{
// AND A,(dp)+Y
IndirectIndexedY();
@@ -1326,7 +1322,7 @@ void Apu37()
IAPU.PC += 2;
}
-void Apu38()
+void Apu38(void)
{
// AND dp,#00
Work8 = OP1;
@@ -1336,7 +1332,7 @@ void Apu38()
IAPU.PC += 3;
}
-void Apu39()
+void Apu39(void)
{
// AND (X),(Y)
Work8 = S9xAPUGetByteZ(IAPU.Registers.X) & S9xAPUGetByteZ(IAPU.Registers.YA.B.Y);
@@ -1345,7 +1341,7 @@ void Apu39()
IAPU.PC++;
}
-void Apu2B()
+void Apu2B(void)
{
// ROL dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1354,7 +1350,7 @@ void Apu2B()
IAPU.PC += 2;
}
-void Apu2C()
+void Apu2C(void)
{
// ROL abs
Absolute();
@@ -1364,7 +1360,7 @@ void Apu2C()
IAPU.PC += 3;
}
-void Apu3B()
+void Apu3B(void)
{
// ROL dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1373,14 +1369,14 @@ void Apu3B()
IAPU.PC += 2;
}
-void Apu3C()
+void Apu3C(void)
{
// ROL A
ROL(IAPU.Registers.YA.B.A);
IAPU.PC++;
}
-void Apu2E()
+void Apu2E(void)
{
// CBNE dp,rel
Work8 = OP1;
@@ -1396,7 +1392,7 @@ void Apu2E()
IAPU.PC += 3;
}
-void ApuDE()
+void ApuDE(void)
{
// CBNE dp+X,rel
Work8 = OP1 + IAPU.Registers.X;
@@ -1412,7 +1408,7 @@ void ApuDE()
IAPU.PC += 3;
}
-void Apu3D()
+void Apu3D(void)
{
// INC X
IAPU.Registers.X++;
@@ -1421,7 +1417,7 @@ void Apu3D()
IAPU.PC++;
}
-void ApuFC()
+void ApuFC(void)
{
// INC Y
IAPU.Registers.YA.B.Y++;
@@ -1430,7 +1426,7 @@ void ApuFC()
IAPU.PC++;
}
-void Apu1D()
+void Apu1D(void)
{
// DEC X
IAPU.Registers.X--;
@@ -1439,7 +1435,7 @@ void Apu1D()
IAPU.PC++;
}
-void ApuDC()
+void ApuDC(void)
{
// DEC Y
IAPU.Registers.YA.B.Y--;
@@ -1448,7 +1444,7 @@ void ApuDC()
IAPU.PC++;
}
-void ApuAB()
+void ApuAB(void)
{
// INC dp
Work8 = S9xAPUGetByteZ(OP1) + 1;
@@ -1458,7 +1454,7 @@ void ApuAB()
IAPU.PC += 2;
}
-void ApuAC()
+void ApuAC(void)
{
// INC abs
Absolute();
@@ -1469,7 +1465,7 @@ void ApuAC()
IAPU.PC += 3;
}
-void ApuBB()
+void ApuBB(void)
{
// INC dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X) + 1;
@@ -1479,7 +1475,7 @@ void ApuBB()
IAPU.PC += 2;
}
-void ApuBC()
+void ApuBC(void)
{
// INC A
IAPU.Registers.YA.B.A++;
@@ -1488,7 +1484,7 @@ void ApuBC()
IAPU.PC++;
}
-void Apu8B()
+void Apu8B(void)
{
// DEC dp
Work8 = S9xAPUGetByteZ(OP1) - 1;
@@ -1498,7 +1494,7 @@ void Apu8B()
IAPU.PC += 2;
}
-void Apu8C()
+void Apu8C(void)
{
// DEC abs
Absolute();
@@ -1509,7 +1505,7 @@ void Apu8C()
IAPU.PC += 3;
}
-void Apu9B()
+void Apu9B(void)
{
// DEC dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X) - 1;
@@ -1519,7 +1515,7 @@ void Apu9B()
IAPU.PC += 2;
}
-void Apu9C()
+void Apu9C(void)
{
// DEC A
IAPU.Registers.YA.B.A--;
@@ -1528,7 +1524,7 @@ void Apu9C()
IAPU.PC++;
}
-void Apu44()
+void Apu44(void)
{
// EOR A,dp
IAPU.Registers.YA.B.A ^= S9xAPUGetByteZ(OP1);
@@ -1536,7 +1532,7 @@ void Apu44()
IAPU.PC += 2;
}
-void Apu45()
+void Apu45(void)
{
// EOR A,abs
Absolute();
@@ -1545,7 +1541,7 @@ void Apu45()
IAPU.PC += 3;
}
-void Apu46()
+void Apu46(void)
{
// EOR A,(X)
IAPU.Registers.YA.B.A ^= S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1553,7 +1549,7 @@ void Apu46()
IAPU.PC++;
}
-void Apu47()
+void Apu47(void)
{
// EOR A,(dp+X)
IndexedXIndirect();
@@ -1562,7 +1558,7 @@ void Apu47()
IAPU.PC += 2;
}
-void Apu48()
+void Apu48(void)
{
// EOR A,#00
IAPU.Registers.YA.B.A ^= OP1;
@@ -1570,7 +1566,7 @@ void Apu48()
IAPU.PC += 2;
}
-void Apu49()
+void Apu49(void)
{
// EOR dp(dest),dp(src)
Work8 = S9xAPUGetByteZ(OP1);
@@ -1580,7 +1576,7 @@ void Apu49()
IAPU.PC += 3;
}
-void Apu54()
+void Apu54(void)
{
// EOR A,dp+X
IAPU.Registers.YA.B.A ^= S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1588,7 +1584,7 @@ void Apu54()
IAPU.PC += 2;
}
-void Apu55()
+void Apu55(void)
{
// EOR A,abs+X
AbsoluteX();
@@ -1597,7 +1593,7 @@ void Apu55()
IAPU.PC += 3;
}
-void Apu56()
+void Apu56(void)
{
// EOR A,abs+Y
AbsoluteY();
@@ -1606,7 +1602,7 @@ void Apu56()
IAPU.PC += 3;
}
-void Apu57()
+void Apu57(void)
{
// EOR A,(dp)+Y
IndirectIndexedY();
@@ -1615,7 +1611,7 @@ void Apu57()
IAPU.PC += 2;
}
-void Apu58()
+void Apu58(void)
{
// EOR dp,#00
Work8 = OP1;
@@ -1625,7 +1621,7 @@ void Apu58()
IAPU.PC += 3;
}
-void Apu59()
+void Apu59(void)
{
// EOR (X),(Y)
Work8 = S9xAPUGetByteZ(IAPU.Registers.X) ^ S9xAPUGetByteZ(IAPU.Registers.YA.B.Y);
@@ -1634,7 +1630,7 @@ void Apu59()
IAPU.PC++;
}
-void Apu4B()
+void Apu4B(void)
{
// LSR dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1643,7 +1639,7 @@ void Apu4B()
IAPU.PC += 2;
}
-void Apu4C()
+void Apu4C(void)
{
// LSR abs
Absolute();
@@ -1653,7 +1649,7 @@ void Apu4C()
IAPU.PC += 3;
}
-void Apu5B()
+void Apu5B(void)
{
// LSR dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1662,14 +1658,14 @@ void Apu5B()
IAPU.PC += 2;
}
-void Apu5C()
+void Apu5C(void)
{
// LSR A
LSR(IAPU.Registers.YA.B.A);
IAPU.PC++;
}
-void Apu7D()
+void Apu7D(void)
{
// MOV A,X
IAPU.Registers.YA.B.A = IAPU.Registers.X;
@@ -1677,7 +1673,7 @@ void Apu7D()
IAPU.PC++;
}
-void ApuDD()
+void ApuDD(void)
{
// MOV A,Y
IAPU.Registers.YA.B.A = IAPU.Registers.YA.B.Y;
@@ -1685,7 +1681,7 @@ void ApuDD()
IAPU.PC++;
}
-void Apu5D()
+void Apu5D(void)
{
// MOV X,A
IAPU.Registers.X = IAPU.Registers.YA.B.A;
@@ -1693,7 +1689,7 @@ void Apu5D()
IAPU.PC++;
}
-void ApuFD()
+void ApuFD(void)
{
// MOV Y,A
IAPU.Registers.YA.B.Y = IAPU.Registers.YA.B.A;
@@ -1701,7 +1697,7 @@ void ApuFD()
IAPU.PC++;
}
-void Apu9D()
+void Apu9D(void)
{
//MOV X,SP
IAPU.Registers.X = IAPU.Registers.S;
@@ -1709,14 +1705,14 @@ void Apu9D()
IAPU.PC++;
}
-void ApuBD()
+void ApuBD(void)
{
// MOV SP,X
IAPU.Registers.S = IAPU.Registers.X;
IAPU.PC++;
}
-void Apu6B()
+void Apu6B(void)
{
// ROR dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1725,7 +1721,7 @@ void Apu6B()
IAPU.PC += 2;
}
-void Apu6C()
+void Apu6C(void)
{
// ROR abs
Absolute();
@@ -1735,7 +1731,7 @@ void Apu6C()
IAPU.PC += 3;
}
-void Apu7B()
+void Apu7B(void)
{
// ROR dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1744,14 +1740,14 @@ void Apu7B()
IAPU.PC += 2;
}
-void Apu7C()
+void Apu7C(void)
{
// ROR A
ROR(IAPU.Registers.YA.B.A);
IAPU.PC++;
}
-void Apu6E()
+void Apu6E(void)
{
// DBNZ dp,rel
Work8 = OP1;
@@ -1767,7 +1763,7 @@ void Apu6E()
IAPU.PC += 3;
}
-void ApuFE()
+void ApuFE(void)
{
// DBNZ Y,rel
Relative();
@@ -1781,14 +1777,14 @@ void ApuFE()
IAPU.PC += 2;
}
-void Apu6F()
+void Apu6F(void)
{
// RET
PopW(IAPU.Registers.PC);
IAPU.PC = IAPU.RAM + IAPU.Registers.PC;
}
-void Apu7F()
+void Apu7F(void)
{
// RETI
Pop(IAPU.Registers.P);
@@ -1797,7 +1793,7 @@ void Apu7F()
IAPU.PC = IAPU.RAM + IAPU.Registers.PC;
}
-void Apu84()
+void Apu84(void)
{
// ADC A,dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1805,7 +1801,7 @@ void Apu84()
IAPU.PC += 2;
}
-void Apu85()
+void Apu85(void)
{
// ADC A, abs
Absolute();
@@ -1814,7 +1810,7 @@ void Apu85()
IAPU.PC += 3;
}
-void Apu86()
+void Apu86(void)
{
// ADC A,(X)
Work8 = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1822,7 +1818,7 @@ void Apu86()
IAPU.PC++;
}
-void Apu87()
+void Apu87(void)
{
// ADC A,(dp+X)
IndexedXIndirect();
@@ -1831,7 +1827,7 @@ void Apu87()
IAPU.PC += 2;
}
-void Apu88()
+void Apu88(void)
{
// ADC A,#00
Work8 = OP1;
@@ -1839,7 +1835,7 @@ void Apu88()
IAPU.PC += 2;
}
-void Apu89()
+void Apu89(void)
{
// ADC dp(dest),dp(src)
Work8 = S9xAPUGetByteZ(OP1);
@@ -1849,7 +1845,7 @@ void Apu89()
IAPU.PC += 3;
}
-void Apu94()
+void Apu94(void)
{
// ADC A,dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -1857,7 +1853,7 @@ void Apu94()
IAPU.PC += 2;
}
-void Apu95()
+void Apu95(void)
{
// ADC A, abs+X
AbsoluteX();
@@ -1866,7 +1862,7 @@ void Apu95()
IAPU.PC += 3;
}
-void Apu96()
+void Apu96(void)
{
// ADC A, abs+Y
AbsoluteY();
@@ -1875,7 +1871,7 @@ void Apu96()
IAPU.PC += 3;
}
-void Apu97()
+void Apu97(void)
{
// ADC A, (dp)+Y
IndirectIndexedY();
@@ -1884,7 +1880,7 @@ void Apu97()
IAPU.PC += 2;
}
-void Apu98()
+void Apu98(void)
{
// ADC dp,#00
Work8 = OP1;
@@ -1894,7 +1890,7 @@ void Apu98()
IAPU.PC += 3;
}
-void Apu99()
+void Apu99(void)
{
// ADC (X),(Y)
W1 = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1904,7 +1900,7 @@ void Apu99()
IAPU.PC++;
}
-void Apu8D()
+void Apu8D(void)
{
// MOV Y,#00
IAPU.Registers.YA.B.Y = OP1;
@@ -1912,7 +1908,7 @@ void Apu8D()
IAPU.PC += 2;
}
-void Apu8F()
+void Apu8F(void)
{
// MOV dp,#00
Work8 = OP1;
@@ -1920,7 +1916,7 @@ void Apu8F()
IAPU.PC += 3;
}
-void Apu9E()
+void Apu9E(void)
{
// DIV YA,X
if ((IAPU.Registers.X & 0x0f) <= (IAPU.Registers.YA.B.Y & 0x0f))
@@ -1935,13 +1931,10 @@ void Apu9E()
for (i = 0 ; i < 9 ; ++i)
{
yva <<= 1;
-
if (yva & 0x20000)
yva = (yva & 0x1ffff) | 1;
-
if (yva >= x)
yva ^= 1;
-
if (yva & 1)
yva = (yva - x) & 0x1ffff;
}
@@ -1957,7 +1950,7 @@ void Apu9E()
IAPU.PC++;
}
-void Apu9F()
+void Apu9F(void)
{
// XCN A
IAPU.Registers.YA.B.A = (IAPU.Registers.YA.B.A >> 4) | (IAPU.Registers.YA.B.A << 4);
@@ -1965,7 +1958,7 @@ void Apu9F()
IAPU.PC++;
}
-void ApuA4()
+void ApuA4(void)
{
// SBC A, dp
Work8 = S9xAPUGetByteZ(OP1);
@@ -1973,7 +1966,7 @@ void ApuA4()
IAPU.PC += 2;
}
-void ApuA5()
+void ApuA5(void)
{
// SBC A, abs
Absolute();
@@ -1982,7 +1975,7 @@ void ApuA5()
IAPU.PC += 3;
}
-void ApuA6()
+void ApuA6(void)
{
// SBC A, (X)
Work8 = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -1990,7 +1983,7 @@ void ApuA6()
IAPU.PC++;
}
-void ApuA7()
+void ApuA7(void)
{
// SBC A,(dp+X)
IndexedXIndirect();
@@ -1999,7 +1992,7 @@ void ApuA7()
IAPU.PC += 2;
}
-void ApuA8()
+void ApuA8(void)
{
// SBC A,#00
Work8 = OP1;
@@ -2007,7 +2000,7 @@ void ApuA8()
IAPU.PC += 2;
}
-void ApuA9()
+void ApuA9(void)
{
// SBC dp(dest), dp(src)
Work8 = S9xAPUGetByteZ(OP1);
@@ -2017,7 +2010,7 @@ void ApuA9()
IAPU.PC += 3;
}
-void ApuB4()
+void ApuB4(void)
{
// SBC A, dp+X
Work8 = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -2025,7 +2018,7 @@ void ApuB4()
IAPU.PC += 2;
}
-void ApuB5()
+void ApuB5(void)
{
// SBC A,abs+X
AbsoluteX();
@@ -2034,7 +2027,7 @@ void ApuB5()
IAPU.PC += 3;
}
-void ApuB6()
+void ApuB6(void)
{
// SBC A,abs+Y
AbsoluteY();
@@ -2043,7 +2036,7 @@ void ApuB6()
IAPU.PC += 3;
}
-void ApuB7()
+void ApuB7(void)
{
// SBC A,(dp)+Y
IndirectIndexedY();
@@ -2052,7 +2045,7 @@ void ApuB7()
IAPU.PC += 2;
}
-void ApuB8()
+void ApuB8(void)
{
// SBC dp,#00
Work8 = OP1;
@@ -2062,7 +2055,7 @@ void ApuB8()
IAPU.PC += 3;
}
-void ApuB9()
+void ApuB9(void)
{
// SBC (X),(Y)
W1 = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -2072,14 +2065,14 @@ void ApuB9()
IAPU.PC++;
}
-void ApuAF()
+void ApuAF(void)
{
// MOV (X)+, A
S9xAPUSetByteZ(IAPU.Registers.YA.B.A, IAPU.Registers.X++);
IAPU.PC++;
}
-void ApuBE()
+void ApuBE(void)
{
// DAS
if (IAPU.Registers.YA.B.A > 0x99 || !IAPU._Carry)
@@ -2097,7 +2090,7 @@ void ApuBE()
IAPU.PC++;
}
-void ApuBF()
+void ApuBF(void)
{
// MOV A,(X)+
IAPU.Registers.YA.B.A = S9xAPUGetByteZ(IAPU.Registers.X++);
@@ -2105,28 +2098,28 @@ void ApuBF()
IAPU.PC++;
}
-void ApuC0()
+void ApuC0(void)
{
// DI
APUClearInterrupt();
IAPU.PC++;
}
-void ApuA0()
+void ApuA0(void)
{
// EI
APUSetInterrupt();
IAPU.PC++;
}
-void ApuC4()
+void ApuC4(void)
{
// MOV dp,A
S9xAPUSetByteZ(IAPU.Registers.YA.B.A, OP1);
IAPU.PC += 2;
}
-void ApuC5()
+void ApuC5(void)
{
// MOV abs,A
Absolute();
@@ -2134,14 +2127,14 @@ void ApuC5()
IAPU.PC += 3;
}
-void ApuC6()
+void ApuC6(void)
{
// MOV (X), A
S9xAPUSetByteZ(IAPU.Registers.YA.B.A, IAPU.Registers.X);
IAPU.PC++;
}
-void ApuC7()
+void ApuC7(void)
{
// MOV (dp+X),A
IndexedXIndirect();
@@ -2149,7 +2142,7 @@ void ApuC7()
IAPU.PC += 2;
}
-void ApuC9()
+void ApuC9(void)
{
// MOV abs,X
Absolute();
@@ -2157,14 +2150,14 @@ void ApuC9()
IAPU.PC += 3;
}
-void ApuCB()
+void ApuCB(void)
{
// MOV dp,Y
S9xAPUSetByteZ(IAPU.Registers.YA.B.Y, OP1);
IAPU.PC += 2;
}
-void ApuCC()
+void ApuCC(void)
{
// MOV abs,Y
Absolute();
@@ -2172,7 +2165,7 @@ void ApuCC()
IAPU.PC += 3;
}
-void ApuCD()
+void ApuCD(void)
{
// MOV X,#00
IAPU.Registers.X = OP1;
@@ -2180,7 +2173,7 @@ void ApuCD()
IAPU.PC += 2;
}
-void ApuCF()
+void ApuCF(void)
{
// MUL YA
IAPU.Registers.YA.W = (uint16_t) IAPU.Registers.YA.B.A * IAPU.Registers.YA.B.Y;
@@ -2188,14 +2181,14 @@ void ApuCF()
IAPU.PC++;
}
-void ApuD4()
+void ApuD4(void)
{
// MOV dp+X, A
S9xAPUSetByteZ(IAPU.Registers.YA.B.A, OP1 + IAPU.Registers.X);
IAPU.PC += 2;
}
-void ApuD5()
+void ApuD5(void)
{
// MOV abs+X,A
AbsoluteX();
@@ -2203,7 +2196,7 @@ void ApuD5()
IAPU.PC += 3;
}
-void ApuD6()
+void ApuD6(void)
{
// MOV abs+Y,A
AbsoluteY();
@@ -2211,7 +2204,7 @@ void ApuD6()
IAPU.PC += 3;
}
-void ApuD7()
+void ApuD7(void)
{
// MOV (dp)+Y,A
IndirectIndexedY();
@@ -2219,28 +2212,28 @@ void ApuD7()
IAPU.PC += 2;
}
-void ApuD8()
+void ApuD8(void)
{
// MOV dp,X
S9xAPUSetByteZ(IAPU.Registers.X, OP1);
IAPU.PC += 2;
}
-void ApuD9()
+void ApuD9(void)
{
// MOV dp+Y,X
S9xAPUSetByteZ(IAPU.Registers.X, OP1 + IAPU.Registers.YA.B.Y);
IAPU.PC += 2;
}
-void ApuDB()
+void ApuDB(void)
{
// MOV dp+X,Y
S9xAPUSetByteZ(IAPU.Registers.YA.B.Y, OP1 + IAPU.Registers.X);
IAPU.PC += 2;
}
-void ApuDF()
+void ApuDF(void)
{
// DAA
if (IAPU.Registers.YA.B.A > 0x99 || IAPU._Carry)
@@ -2258,7 +2251,7 @@ void ApuDF()
IAPU.PC++;
}
-void ApuE4()
+void ApuE4(void)
{
// MOV A, dp
IAPU.Registers.YA.B.A = S9xAPUGetByteZ(OP1);
@@ -2266,7 +2259,7 @@ void ApuE4()
IAPU.PC += 2;
}
-void ApuE5()
+void ApuE5(void)
{
// MOV A,abs
Absolute();
@@ -2275,7 +2268,7 @@ void ApuE5()
IAPU.PC += 3;
}
-void ApuE6()
+void ApuE6(void)
{
// MOV A,(X)
IAPU.Registers.YA.B.A = S9xAPUGetByteZ(IAPU.Registers.X);
@@ -2283,7 +2276,7 @@ void ApuE6()
IAPU.PC++;
}
-void ApuE7()
+void ApuE7(void)
{
// MOV A,(dp+X)
IndexedXIndirect();
@@ -2292,7 +2285,7 @@ void ApuE7()
IAPU.PC += 2;
}
-void ApuE8()
+void ApuE8(void)
{
// MOV A,#00
IAPU.Registers.YA.B.A = OP1;
@@ -2300,7 +2293,7 @@ void ApuE8()
IAPU.PC += 2;
}
-void ApuE9()
+void ApuE9(void)
{
// MOV X, abs
Absolute();
@@ -2309,7 +2302,7 @@ void ApuE9()
IAPU.PC += 3;
}
-void ApuEB()
+void ApuEB(void)
{
// MOV Y,dp
IAPU.Registers.YA.B.Y = S9xAPUGetByteZ(OP1);
@@ -2317,7 +2310,7 @@ void ApuEB()
IAPU.PC += 2;
}
-void ApuEC()
+void ApuEC(void)
{
// MOV Y,abs
Absolute();
@@ -2326,7 +2319,7 @@ void ApuEC()
IAPU.PC += 3;
}
-void ApuF4()
+void ApuF4(void)
{
// MOV A, dp+X
IAPU.Registers.YA.B.A = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -2334,7 +2327,7 @@ void ApuF4()
IAPU.PC += 2;
}
-void ApuF5()
+void ApuF5(void)
{
// MOV A, abs+X
AbsoluteX();
@@ -2343,7 +2336,7 @@ void ApuF5()
IAPU.PC += 3;
}
-void ApuF6()
+void ApuF6(void)
{
// MOV A, abs+Y
AbsoluteY();
@@ -2352,7 +2345,7 @@ void ApuF6()
IAPU.PC += 3;
}
-void ApuF7()
+void ApuF7(void)
{
// MOV A, (dp)+Y
IndirectIndexedY();
@@ -2361,7 +2354,7 @@ void ApuF7()
IAPU.PC += 2;
}
-void ApuF8()
+void ApuF8(void)
{
// MOV X,dp
IAPU.Registers.X = S9xAPUGetByteZ(OP1);
@@ -2369,7 +2362,7 @@ void ApuF8()
IAPU.PC += 2;
}
-void ApuF9()
+void ApuF9(void)
{
// MOV X,dp+Y
IAPU.Registers.X = S9xAPUGetByteZ(OP1 + IAPU.Registers.YA.B.Y);
@@ -2377,14 +2370,14 @@ void ApuF9()
IAPU.PC += 2;
}
-void ApuFA()
+void ApuFA(void)
{
// MOV dp(dest),dp(src)
S9xAPUSetByteZ(S9xAPUGetByteZ(OP1), OP2);
IAPU.PC += 3;
}
-void ApuFB()
+void ApuFB(void)
{
// MOV Y,dp+X
IAPU.Registers.YA.B.Y = S9xAPUGetByteZ(OP1 + IAPU.Registers.X);
@@ -2392,7 +2385,7 @@ void ApuFB()
IAPU.PC += 2;
}
-void (*S9xApuOpcodes[256])() =
+void (*S9xApuOpcodes[256])(void) =
{
Apu00, Apu01, Apu02, Apu03, Apu04, Apu05, Apu06, Apu07,
Apu08, Apu09, Apu0A, Apu0B, Apu0C, Apu0D, Apu0E, Apu0F,
diff --git a/source/spc700.h b/source/spc700.h
index 53bc1b6..3de94ae 100644
--- a/source/spc700.h
+++ b/source/spc700.h
@@ -70,10 +70,8 @@ typedef struct
#define APU_EXECUTE() \
if (IAPU.APUExecuting) \
-{\
- while (APU.Cycles <= CPU.Cycles) \
- APU_EXECUTE1(); \
-}
+ while (APU.Cycles <= CPU.Cycles) \
+ APU_EXECUTE1();
#endif
#endif
diff --git a/source/spc7110.c b/source/spc7110.c
index bdeb7be..7ed339d 100644
--- a/source/spc7110.c
+++ b/source/spc7110.c
@@ -1,87 +1,19 @@
#include "../copyright"
#include "spc7110.h"
+#include "spc7110dec.h"
#include "memmap.h"
#include <time.h>
-#include <sys/stat.h>
-
-//Windows includes
-#ifdef __WIN32__
-#ifndef _XBOX // chdir and getcwd not supported on Xbox hardware
-#include <direct.h>
-#define chdir _chdir
-#define getcwd _getcwd
-#endif
-
-#ifndef PATH_MAX
-#ifdef MAX_PATH
-#define PATH_MAX MAX_PATH
-#else
-#define PATH_MAX 1024
-#endif
-#endif
-
-#else // Unix
-#include "display.h"
-#include <limits.h>
-#include <unistd.h>
-#endif
const char* S9xGetFilename(const char*);
-char* osd_GetPackDir(void);
-//really not needed, but usually MS adds the _ to POSIX functions,
-//while *nix doesn't, so this was to "un-M$" the function.
-#define splitpath _splitpath
-
-//not much headroom, but FEOEZ has 41 tables, I think, and SPL4 has 38.
-#define MAX_TABLES 48
-
-//default to using 5 megs of RAM for method 3 caching.
-uint16_t cacheMegs = 5;
-
-//using function pointers to initialize cache management
-void (*CleanUp7110)(void) = NULL;
-void (*LoadUp7110)(char*) = &SPC7110Load;
-void (*Copy7110)(void) = NULL;
-
-//size and offset of the pack data
-//offset and size of reads from pack
-typedef struct
-{
- uint32_t offset;
- uint32_t size;
- uint16_t used_offset;
- uint16_t used_len;
-} Data7110;
-
-//this maps an index.bin table to the decompression pack
-typedef struct
-{
- int32_t table;
- bool is_file;
- Data7110 location[256];
-} Index7110;
-
-//this contains all the data for the decompression pack.
-typedef struct
-{
- uint8_t* binfiles[MAX_TABLES];
- Index7110 tableEnts[MAX_TABLES];
- int32_t last_table;
- int32_t idx;
- uint8_t last_idx;
- uint16_t last_offset;
-} Pack7110;
-char pfold[9]; // Hack variable for log naming (each game makes a different log)
-Pack7110* decompack = NULL; // Decompression pack uses a fair chunk of RAM, so dynalloc it.
-SPC7110Regs s7r; // SPC7110 registers, about 33KB
-S7RTC rtc_f9; // FEOEZ (and Shounen Jump no SHou) RTC
-void S9xUpdateRTC(void); // S-RTC function hacked to work with the RTC
+SPC7110Regs s7r; // SPC7110 registers, about 33KB
+S7RTC rtc_f9; // FEOEZ (and Shounen Jump no SHou) RTC
+void S9xUpdateRTC(void); // S-RTC function hacked to work with the RTC
-//Emulate power on state
-void S9xSpc7110Init(void)
+void S9xSpc7110Init(void) // Emulate power on state
{
+ spc7110dec_init();
s7r.DataRomOffset = 0x00100000; //handy constant!
s7r.DataRomSize = Memory.CalculatedSize - s7r.DataRomOffset;
s7r.reg4800 = 0;
@@ -132,308 +64,10 @@ void S9xSpc7110Init(void)
s7r.written = 0;
s7r.offset_add = 0;
s7r.AlignBy = 1;
-
- (*LoadUp7110)(osd_GetPackDir());
-
s7r.bank50Internal = 0;
memset(s7r.bank50, 0x00, DECOMP_BUFFER_SIZE);
}
-//full cache decompression routine (memcpy) Method 1
-void MovePackData(void)
-{
- //log the last entry
- Data7110* log = &(decompack->tableEnts[decompack->idx].location[decompack->last_idx]);
- if ((log->used_len + log->used_offset) < (decompack->last_offset +
- (uint16_t)s7r.bank50Internal))
- {
- log->used_len = s7r.bank50Internal;
- log->used_offset = decompack->last_offset;
- }
-
- //set up for next logging
- decompack->last_offset = (s7r.reg4805) | (s7r.reg4806 << 8);
-
- decompack->last_idx = s7r.reg4804;
-
- //start decompression
- int32_t table = (s7r.reg4803 << 16) | (s7r.reg4802 << 8) | s7r.reg4801;
-
- //the table is a offset multiplier byte and a big-endian pointer
- int32_t j = 4 * s7r.reg4804;
- j += s7r.DataRomOffset;
- j += table;
-
- //set proper offsetting.
- if (s7r.reg480B == 0)
- s7r.AlignBy = 0;
- else
- {
- switch (Memory.ROM[j])
- {
- case 0x03:
- s7r.AlignBy = 8;
- break;
- case 0x01:
- s7r.AlignBy = 2;
- break;
- case 0x02:
- s7r.AlignBy = 4;
- break;
- case 0x00:
- default:
- s7r.AlignBy = 1;
- break;
- }
- }
- //note that we are still setting up for the next log.
- decompack->last_offset *= s7r.AlignBy;
- decompack->last_offset %= DECOMP_BUFFER_SIZE;
-
- //find the table
- if (table != decompack->last_table)
- {
- int32_t i = 0;
- while (i < MAX_TABLES && decompack->tableEnts[i].table != table)
- i++;
- if (i == MAX_TABLES)
- {
-#ifdef _XBOX
- FILE* fp = fopen("T:\\sp7err.out", "a");
-#else
- FILE* fp = fopen("sp7err.out", "a");
-#endif
-
- fclose(fp);
- return;
- }
- decompack->idx = i;
- decompack->last_table = table;
- }
-
- //copy data
- if (decompack->binfiles[decompack->idx])
- {
- memcpy(s7r.bank50,
- &(decompack->binfiles[decompack->idx][decompack->tableEnts[decompack->idx].location[s7r.reg4804].offset]),
- decompack->tableEnts[decompack->idx].location[s7r.reg4804].size);
- }
-}
-
-// This is similar to the last function, but it keeps the last 5 accessed files open,
-// and reads the data directly. Method 2
-void ReadPackData(void)
-{
- static int32_t table_age_2;
- static int32_t table_age_3;
- static int32_t table_age_4;
- static int32_t table_age_5;
-
- int32_t table = (s7r.reg4803 << 16) | (s7r.reg4802 << 8) | s7r.reg4801;
-
- if (table == 0)
- {
- table_age_2 = table_age_3 = table_age_4 = table_age_5 = MAX_TABLES;
- return;
- }
-
- if (table_age_2 == 0 && table_age_3 == 0 && table_age_4 == 0 && table_age_5 == 0)
- table_age_2 = table_age_3 = table_age_4 = table_age_5 = MAX_TABLES;
- Data7110* log = &(decompack->tableEnts[decompack->idx].location[decompack->last_idx]);
- if ((log->used_len + log->used_offset) < (decompack->last_offset +
- (uint16_t)s7r.bank50Internal))
- {
- log->used_len = s7r.bank50Internal;
- log->used_offset = decompack->last_offset;
- }
-
- decompack->last_offset = (s7r.reg4805) | (s7r.reg4806 << 8);
-
- decompack->last_idx = s7r.reg4804;
-
- int32_t j = 4 * s7r.reg4804;
- j += s7r.DataRomOffset;
- j += table;
-
- if (s7r.reg480B == 0)
- s7r.AlignBy = 0;
- else
- {
- switch (Memory.ROM[j])
- {
- case 0x03:
- s7r.AlignBy = 8;
- break;
- case 0x01:
- s7r.AlignBy = 2;
- break;
- case 0x02:
- s7r.AlignBy = 4;
- break;
- case 0x00:
- default:
- s7r.AlignBy = 1;
- break;
- }
- }
- decompack->last_offset *= s7r.AlignBy;
- decompack->last_offset %= DECOMP_BUFFER_SIZE;
- if (table != decompack->last_table)
- {
- int32_t i = 0;
- while (i < MAX_TABLES && decompack->tableEnts[i].table != table)
- i++;
- if (i == MAX_TABLES)
- {
- FILE* fp = fopen("sp7err.out", "a");
- fclose(fp);
- return;
- }
- if (i != table_age_2 && i != table_age_3 && i != table_age_4 && i != table_age_5)
- {
- if (table_age_5 != MAX_TABLES && decompack->binfiles[table_age_5])
- {
- fclose((FILE*)(decompack->binfiles[table_age_5]));
- (decompack->binfiles[table_age_5]) = NULL;
- }
- table_age_5 = table_age_4;
- table_age_4 = table_age_3;
- table_age_3 = table_age_2;
- table_age_2 = decompack->idx;
- char name[PATH_MAX];
- //open file
- char drive [_MAX_DRIVE + 1];
- char dir [_MAX_DIR + 1];
- char fname [_MAX_FNAME + 1];
- char ext [_MAX_EXT + 1];
-
- splitpath(Memory.ROMFilename, drive, dir, fname, ext);
- strcpy(name, drive);
- strcat(name, dir);
-
- strcat(name, pfold);
- char bfname[11];
- sprintf(bfname, "%06X.bin", table);
- strcat(name, "/");
- strcat(name, bfname);
- decompack->binfiles[i] = (uint8_t*)fopen(name, "rb");
- }
- else
- {
- //fix tables in this case
- if (table_age_5 == i)
- {
- table_age_5 = table_age_4;
- table_age_4 = table_age_3;
- table_age_3 = table_age_2;
- table_age_2 = decompack->idx;
- }
- if (table_age_4 == i)
- {
- table_age_4 = table_age_3;
- table_age_3 = table_age_2;
- table_age_2 = decompack->idx;
- }
- if (table_age_3 == i)
- {
- table_age_3 = table_age_2;
- table_age_2 = decompack->idx;
- }
- if (table_age_2 == i)
- table_age_2 = decompack->idx;
- }
- decompack->idx = i;
- decompack->last_table = table;
- }
- //do read here.
- if (decompack->binfiles[decompack->idx])
- {
- fseek((FILE*)(decompack->binfiles[decompack->idx]),
- decompack->tableEnts[decompack->idx].location[s7r.reg4804].offset, 0);
- fread(s7r.bank50, 1, (
- decompack->tableEnts[decompack->idx].location[s7r.reg4804].size),
- (FILE*)(decompack->binfiles[decompack->idx]));
- }
-}
-
-//Cache Method 3: some entries are cached, others are file handles.
-//use is_file to distinguish.
-void GetPackData(void)
-{
- Data7110* log = &
- (decompack->tableEnts[decompack->idx].location[decompack->last_idx]);
- if ((log->used_len + log->used_offset) < (decompack->last_offset +
- (uint16_t)s7r.bank50Internal))
- {
- log->used_len = s7r.bank50Internal;
- log->used_offset = decompack->last_offset;
- }
-
- decompack->last_offset = (s7r.reg4805) | (s7r.reg4806 << 8);
-
- decompack->last_idx = s7r.reg4804;
- int32_t table = (s7r.reg4803 << 16) | (s7r.reg4802 << 8) | s7r.reg4801;
-
- int32_t j = 4 * s7r.reg4804;
- j += s7r.DataRomOffset;
- j += table;
-
- if (s7r.reg480B == 0)
- s7r.AlignBy = 0;
- else
- {
- switch (Memory.ROM[j])
- {
- case 0x03:
- s7r.AlignBy = 8;
- break;
- case 0x01:
- s7r.AlignBy = 2;
- break;
- case 0x02:
- s7r.AlignBy = 4;
- break;
- case 0x00:
- default:
- s7r.AlignBy = 1;
- break;
- }
- }
- decompack->last_offset *= s7r.AlignBy;
- decompack->last_offset %= DECOMP_BUFFER_SIZE;
- if (table != decompack->last_table)
- {
- int32_t i = 0;
- while (i < MAX_TABLES && decompack->tableEnts[i].table != table)
- i++;
- if (i == MAX_TABLES)
- {
- FILE* fp = fopen("sp7err.out", "a");
- fclose(fp);
- return;
- }
- decompack->idx = i;
- decompack->last_table = table;
- }
- if (decompack->binfiles[decompack->idx])
- {
- if (decompack->tableEnts[decompack->idx].is_file)
- {
- fseek((FILE*)decompack->binfiles[decompack->idx],
- decompack->tableEnts[decompack->idx].location[s7r.reg4804].offset, 0);
- fread(s7r.bank50, 1, (
- decompack->tableEnts[decompack->idx].location[s7r.reg4804].size),
- (FILE*)(decompack->binfiles[decompack->idx]));
- }
- else
- {
- memcpy(s7r.bank50,
- &(decompack->binfiles[decompack->idx][decompack->tableEnts[decompack->idx].location[s7r.reg4804].offset]),
- decompack->tableEnts[decompack->idx].location[s7r.reg4804].size);
- }
- }
-}
-
//reads SPC7110 and RTC registers.
uint8_t S9xGetSPC7110(uint16_t Address)
{
@@ -448,40 +82,26 @@ uint8_t S9xGetSPC7110(uint16_t Address)
case 0x4800:
{
uint16_t count = s7r.reg4809 | (s7r.reg480A << 8);
- uint32_t i, j;
- j = (s7r.reg4805 | (s7r.reg4806 << 8));
- j *= s7r.AlignBy;
- i = j;
if (count > 0)
count--;
- else count = 0xFFFF;
+ else
+ count = 0xFFFF;
s7r.reg4809 = 0x00ff & count;
s7r.reg480A = (0xff00 & count) >> 8;
- i += s7r.bank50Internal;
- i %= DECOMP_BUFFER_SIZE;
- s7r.reg4800 = s7r.bank50[i];
-
- s7r.bank50Internal++;
- s7r.bank50Internal %= DECOMP_BUFFER_SIZE;
+ s7r.reg4800 = spc7110dec_read();
+ return s7r.reg4800;
}
- return s7r.reg4800;
- //table register low
- case 0x4801:
+ case 0x4801: //table register low
return s7r.reg4801;
- //table register middle
- case 0x4802:
+ case 0x4802: //table register middle
return s7r.reg4802;
- //table register high
- case 0x4803:
+ case 0x4803: //table register high
return s7r.reg4803;
- //index of pointer in table (each entry is 4 bytes)
- case 0x4804:
+ case 0x4804: //index of pointer in table (each entry is 4 bytes)
return s7r.reg4804;
- //offset register low
- case 0x4805:
+ case 0x4805: //offset register low
return s7r.reg4805;
- //offset register high
- case 0x4806:
+ case 0x4806: //offset register high
return s7r.reg4806;
//DMA channel (not that I see this usually set,
//regardless of what channel DMA is on)
@@ -496,15 +116,13 @@ uint8_t S9xGetSPC7110(uint16_t Address)
//this is set by the ROM, and wraps on bounds.
case 0x4809:
return s7r.reg4809;
- //C Length high
- case 0x480A:
+ case 0x480A: //C Length high
return s7r.reg480A;
//Offset enable.
//if this is zero, 4805-6 are useless. Emulated by setting AlignBy to 0
case 0x480B:
return s7r.reg480B;
- //decompression finished: just emulated by switching each read.
- case 0x480C:
+ case 0x480C: //decompression finished: just emulated by switching each read.
s7r.reg480C ^= 0x80;
return s7r.reg480C ^ 0x80;
//Data access port
@@ -536,7 +154,8 @@ uint8_t S9xGetSPC7110(uint16_t Address)
i += r4814;
if (r4814 != 0xFFFF)
r4814++;
- else r4814 = 0;
+ else
+ r4814 = 0;
s7r.reg4815 = (uint8_t)(r4814 >> 8);
s7r.reg4814 = (uint8_t)(r4814 & 0x00FF);
@@ -553,8 +172,7 @@ uint8_t S9xGetSPC7110(uint16_t Address)
{
if (s7r.reg4818 & 0x04)
{
- int16_t inc;
- inc = (s7r.reg4817 << 8) | s7r.reg4816;
+ int16_t inc = (s7r.reg4817 << 8) | s7r.reg4816;
if (!(s7r.reg4818 & 0x10))
i += inc;
@@ -640,31 +258,23 @@ uint8_t S9xGetSPC7110(uint16_t Address)
s7r.reg4813 = ((i & 0xFF0000) >> 16);
return tmp;
}
- else return 0;
- //direct read address low
- case 0x4811:
+ else
+ return 0;
+ case 0x4811: //direct read address low
return s7r.reg4811;
- //direct read address middle
- case 0x4812:
+ case 0x4812: //direct read address middle
return s7r.reg4812;
- //direct read access high
- case 0x4813:
+ case 0x4813: //direct read access high
return s7r.reg4813;
- //read adjust low
- case 0x4814:
+ case 0x4814: //read adjust low
return s7r.reg4814;
- //read adjust high
- case 0x4815:
+ case 0x4815: //read adjust high
return s7r.reg4815;
- //read increment low
- case 0x4816:
+ case 0x4816: //read increment low
return s7r.reg4816;
- //read increment high
- case 0x4817:
+ case 0x4817: //read increment high
return s7r.reg4817;
- //Data ROM command mode
- //essentially, this controls the insane code of $4810 and $481A
- case 0x4818:
+ case 0x4818: //Data ROM command mode; essentially, this controls the insane code of $4810 and $481A
return s7r.reg4818;
//read after adjust port
//what this does, besides more nasty stuff like 4810,
@@ -675,22 +285,14 @@ uint8_t S9xGetSPC7110(uint16_t Address)
{
uint32_t i = ((s7r.reg4813 << 16) | (s7r.reg4812 << 8) | s7r.reg4811);
if (s7r.reg4818 & 0x08)
- {
- int16_t adj;
- adj = ((int16_t)(s7r.reg4815 << 8)) | s7r.reg4814;
- i += adj;
- }
+ i += ((int16_t)(s7r.reg4815 << 8)) | s7r.reg4814;
else
- {
- uint16_t adj;
- adj = (s7r.reg4815 << 8) | s7r.reg4814;
- i += adj;
- }
+ i += (s7r.reg4815 << 8) | s7r.reg4814;
i %= s7r.DataRomSize;
i += s7r.DataRomOffset;
uint8_t tmp = Memory.ROM[i];
- if (0x60 == (s7r.reg4818 & 0x60))
+ if ((s7r.reg4818 & 0x60) == 0x60)
{
i = ((s7r.reg4813 << 16) | (s7r.reg4812 << 8) | s7r.reg4811);
@@ -733,85 +335,60 @@ uint8_t S9xGetSPC7110(uint16_t Address)
}
return tmp;
}
- else return 0;
-
- //multiplicand low or dividend lowest
- case 0x4820:
+ else
+ return 0;
+ case 0x4820: //multiplicand low or dividend lowest
return s7r.reg4820;
- //multiplicand high or divdend lower
- case 0x4821:
+ case 0x4821: //multiplicand high or divdend lower
return s7r.reg4821;
- //dividend higher
- case 0x4822:
+ case 0x4822: //dividend higher
return s7r.reg4822;
- //dividend highest
- case 0x4823:
+ case 0x4823: //dividend highest
return s7r.reg4823;
- //multiplier low
- case 0x4824:
+ case 0x4824: //multiplier low
return s7r.reg4824;
- //multiplier high
- case 0x4825:
+ case 0x4825: //multiplier high
return s7r.reg4825;
- //divisor low
- case 0x4826:
+ case 0x4826: //divisor low
return s7r.reg4826;
- //divisor high
- case 0x4827:
+ case 0x4827: //divisor high
return s7r.reg4827;
- //result lowest
- case 0x4828:
+ case 0x4828: //result lowest
return s7r.reg4828;
- //result lower
- case 0x4829:
+ case 0x4829: //result lower
return s7r.reg4829;
- //result higher
- case 0x482A:
+ case 0x482A: //result higher
return s7r.reg482A;
- //result highest
- case 0x482B:
+ case 0x482B: //result highest
return s7r.reg482B;
- //remainder (division) low
- case 0x482C:
+ case 0x482C: //remainder (division) low
return s7r.reg482C;
- //remainder (division) high
- case 0x482D:
+ case 0x482D: //remainder (division) high
return s7r.reg482D;
- //signed/unsigned
- case 0x482E:
+ case 0x482E: //signed/unsigned
return s7r.reg482E;
- //finished flag, emulated as an on-read toggle.
- case 0x482F:
+ case 0x482F: //finished flag, emulated as an on-read toggle.
if (s7r.reg482F)
{
s7r.reg482F = 0;
return 0x80;
}
return 0;
- break;
-
- //SRAM toggle
- case 0x4830:
+ case 0x4830: //SRAM toggle
return s7r.reg4830;
- //DX bank mapping
- case 0x4831:
+ case 0x4831: //DX bank mapping
return s7r.reg4831;
- //EX bank mapping
- case 0x4832:
+ case 0x4832: //EX bank mapping
return s7r.reg4832;
- //FX bank mapping
- case 0x4833:
+ case 0x4833: //FX bank mapping
return s7r.reg4833;
- //SRAM mapping? We have no clue!
- case 0x4834:
+ case 0x4834: //SRAM mapping? We have no clue!
return s7r.reg4834;
- //RTC enable
- case 0x4840:
+ case 0x4840: //RTC enable
if (!Settings.SPC7110RTC)
return Address >> 8;
return s7r.reg4840;
- //command/index/value of RTC (essentially, zero unless we're in read mode
- case 0x4841:
+ case 0x4841: //command/index/value of RTC (essentially, zero unless we're in read mode
if (!Settings.SPC7110RTC)
return Address >> 8;
if (rtc_f9.init)
@@ -822,9 +399,9 @@ uint8_t S9xGetSPC7110(uint16_t Address)
rtc_f9.index %= 0x10;
return tmp;
}
- else return 0;
- //RTC done flag
- case 0x4842:
+ else
+ return 0;
+ case 0x4842: //RTC done flag
if (!Settings.SPC7110RTC)
return Address >> 8;
s7r.reg4842 ^= 0x80;
@@ -834,14 +411,21 @@ uint8_t S9xGetSPC7110(uint16_t Address)
}
}
+static uint32_t datarom_addr(uint32_t addr)
+{
+ uint32_t size = Memory.CalculatedSize - 0x100000;
+
+ while(addr >= size)
+ addr -= size;
+ return addr + 0x100000;
+}
+
void S9xSetSPC7110(uint8_t data, uint16_t Address)
{
switch (Address)
{
//Writes to $4800 are undefined.
-
- //table low, middle, and high.
- case 0x4801:
+ case 0x4801: //table low, middle, and high.
s7r.reg4801 = data;
break;
case 0x4802:
@@ -850,54 +434,44 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
case 0x4803:
s7r.reg4803 = data;
break;
-
- //table index (4 byte entries, bigendian with a multiplier byte)
- case 0x4804:
+ case 0x4804: //table index (4 byte entries, bigendian with a multiplier byte)
s7r.reg4804 = data;
break;
-
- //offset low
- case 0x4805:
+ case 0x4805: //offset low
s7r.reg4805 = data;
break;
-
- //offset high, starts decompression
- case 0x4806:
+ case 0x4806: //offset high, starts decompression
+ {
+ uint32_t table = (s7r.reg4801 + (s7r.reg4802 << 8) + (s7r.reg4803 << 16));
+ uint32_t index = (s7r.reg4804 << 2);
+ uint32_t addr = datarom_addr(table + index);
+ uint32_t mode = (Memory.ROM[addr + 0]);
+ uint32_t offset = (Memory.ROM[addr + 1] << 16) + (Memory.ROM[addr + 2] << 8) + (Memory.ROM[addr + 3]);
s7r.reg4806 = data;
- (*Copy7110)();
+ spc7110dec_clear(mode, offset, (s7r.reg4805 + (s7r.reg4806 << 8)) << mode);
s7r.bank50Internal = 0;
s7r.reg480C &= 0x7F;
break;
-
- //DMA channel register (Is it used??)
- case 0x4807:
+ }
+ case 0x4807: //DMA channel register (Is it used??)
s7r.reg4807 = data;
break;
-
//C r/w? I have no idea. If you get weird values written here before a bug,
//The Dumper should probably be contacted about running a test.
case 0x4808:
s7r.reg4808 = data;
break;
-
- //C-Length low
- case 0x4809:
+ case 0x4809: //C-Length low
s7r.reg4809 = data;
break;
- //C-Length high
- case 0x480A:
+ case 0x480A: //C-Length high
s7r.reg480A = data;
break;
-
- //Offset enable
- case 0x480B:
+ case 0x480B: //Offset enable
{
s7r.reg480B = data;
int32_t table = (s7r.reg4803 << 16) | (s7r.reg4802 << 8) | s7r.reg4801;
-
- int32_t j = 4 * s7r.reg4804;
- j += s7r.DataRomOffset;
- j += table;
+ int32_t j = 4 * s7r.reg4804 + s7r.DataRomOffset + table;
if (s7r.reg480B == 0)
s7r.AlignBy = 0;
@@ -920,30 +494,21 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
break;
}
}
+ break;
}
- break;
- //$4810 is probably read only.
-
- //Data port address low
- case 0x4811:
+ case 0x4811: //Data port address low
s7r.reg4811 = data;
s7r.written |= 0x01;
break;
-
- //data port address middle
- case 0x4812:
+ case 0x4812: //data port address middle
s7r.reg4812 = data;
s7r.written |= 0x02;
break;
-
- //data port address high
- case 0x4813:
+ case 0x4813: //data port address high
s7r.reg4813 = data;
s7r.written |= 0x04;
break;
-
- //data port adjust low (has a funky immediate increment mode)
- case 0x4814:
+ case 0x4814: //data port adjust low (has a funky immediate increment mode)
s7r.reg4814 = data;
if (s7r.reg4818 & 0x02)
{
@@ -952,10 +517,7 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
s7r.offset_add |= 0x01;
if (s7r.offset_add == 3)
{
- if (s7r.reg4818 & 0x10)
- {
- }
- else
+ if(!(s7r.reg4818 & 0x10))
{
uint32_t i = (s7r.reg4813 << 16) | (s7r.reg4812 << 8) | s7r.reg4811;
if (s7r.reg4818 & 0x08)
@@ -974,39 +536,25 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
s7r.offset_add |= 0x01;
if (s7r.offset_add == 3)
{
- if (s7r.reg4818 & 0x10)
- {
- }
- else
+ if(!(s7r.reg4818 & 0x10))
{
uint32_t i = (s7r.reg4813 << 16) | (s7r.reg4812 << 8) | s7r.reg4811;
if (s7r.reg4818 & 0x08)
- {
- int16_t adj;
- adj = ((int16_t)(s7r.reg4815 << 8)) | s7r.reg4814;
- i += adj;
- }
+ i += ((int16_t)(s7r.reg4815 << 8)) | s7r.reg4814;
else
- {
- uint16_t adj;
- adj = (s7r.reg4815 << 8) | s7r.reg4814;
- i += adj;
- }
+ i += (s7r.reg4815 << 8) | s7r.reg4814;
i %= s7r.DataRomSize;
s7r.reg4811 = i & 0x00FF;
s7r.reg4812 = (i & 0x00FF00) >> 8;
s7r.reg4813 = ((i & 0xFF0000) >> 16);
}
}
-
}
}
s7r.written |= 0x08;
break;
-
- //data port adjust high (has a funky immediate increment mode)
- case 0x4815:
+ case 0x4815: //data port adjust high (has a funky immediate increment mode)
s7r.reg4815 = data;
if (s7r.reg4818 & 0x02)
{
@@ -1015,10 +563,7 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
s7r.offset_add |= 0x02;
if (s7r.offset_add == 3)
{
- if (s7r.reg4818 & 0x10)
- {
- }
- else
+ if(!(s7r.reg4818 & 0x10))
{
uint32_t i = (s7r.reg4813 << 16) | (s7r.reg4812 << 8) | s7r.reg4811;
@@ -1038,24 +583,13 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
s7r.offset_add |= 0x02;
if (s7r.offset_add == 3)
{
- if (s7r.reg4818 & 0x10)
- {
- }
- else
+ if(!(s7r.reg4818 & 0x10))
{
uint32_t i = (s7r.reg4813 << 16) | (s7r.reg4812 << 8) | s7r.reg4811;
if (s7r.reg4818 & 0x08)
- {
- int16_t adj;
- adj = ((int16_t)(s7r.reg4815 << 8)) | s7r.reg4814;
- i += adj;
- }
+ i += ((int16_t)(s7r.reg4815 << 8)) | s7r.reg4814;
else
- {
- uint16_t adj;
- adj = (s7r.reg4815 << 8) | s7r.reg4814;
- i += adj;
- }
+ i += (s7r.reg4815 << 8) | s7r.reg4814;
i %= s7r.DataRomSize;
s7r.reg4811 = i & 0x00FF;
s7r.reg4812 = (i & 0x00FF00) >> 8;
@@ -1066,15 +600,12 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
}
s7r.written |= 0x10;
break;
- //data port increment low
- case 0x4816:
+ case 0x4816: //data port increment low
s7r.reg4816 = data;
break;
- //data port increment high
- case 0x4817:
+ case 0x4817: //data port increment high
s7r.reg4817 = data;
break;
-
//data port mode switches
//note that it starts inactive.
case 0x4818:
@@ -1083,29 +614,22 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
s7r.offset_add = 0;
s7r.reg4818 = data;
break;
-
- //multiplicand low or dividend lowest
- case 0x4820:
+ case 0x4820: //multiplicand low or dividend lowest
s7r.reg4820 = data;
break;
- //multiplicand high or dividend lower
- case 0x4821:
+ case 0x4821: //multiplicand high or dividend lower
s7r.reg4821 = data;
break;
- //dividend higher
- case 0x4822:
+ case 0x4822: //dividend higher
s7r.reg4822 = data;
break;
- //dividend highest
- case 0x4823:
+ case 0x4823: //dividend highest
s7r.reg4823 = data;
break;
- //multiplier low
- case 0x4824:
+ case 0x4824: //multiplier low
s7r.reg4824 = data;
break;
- //multiplier high (triggers operation)
- case 0x4825:
+ case 0x4825: //multiplier high (triggers operation)
s7r.reg4825 = data;
if (s7r.reg482E & 0x01)
{
@@ -1129,19 +653,16 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
}
s7r.reg482F = 0x80;
break;
- //divisor low
- case 0x4826:
+ case 0x4826: //divisor low
s7r.reg4826 = data;
break;
- //divisor high (triggers operation)
- case 0x4827:
+ case 0x4827: //divisor high (triggers operation)
s7r.reg4827 = data;
if (s7r.reg482E & 0x01)
{
int32_t quotient;
int16_t remainder;
- int32_t dividend = (int32_t)(s7r.reg4820 | (s7r.reg4821 << 8) | (s7r.reg4822 << 16) |
- (s7r.reg4823 << 24));
+ int32_t dividend = (int32_t)(s7r.reg4820 | (s7r.reg4821 << 8) | (s7r.reg4822 << 16) | (s7r.reg4823 << 24));
int16_t divisor = (int16_t)(s7r.reg4826 | (s7r.reg4827 << 8));
if (divisor != 0)
{
@@ -1164,8 +685,7 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
{
uint32_t quotient;
uint16_t remainder;
- uint32_t dividend = (uint32_t)(s7r.reg4820 | (s7r.reg4821 << 8) |
- (s7r.reg4822 << 16) | (s7r.reg4823 << 24));
+ uint32_t dividend = (uint32_t)(s7r.reg4820 | (s7r.reg4821 << 8) | (s7r.reg4822 << 16) | (s7r.reg4823 << 24));
uint16_t divisor = (uint16_t)(s7r.reg4826 | (s7r.reg4827 << 8));
if (divisor != 0)
{
@@ -1191,42 +711,30 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
//reset: writes here nuke the whole math unit
//Zero indicates unsigned math, resets with non-zero values turn on signed math
case 0x482E:
- s7r.reg4820 = s7r.reg4821 = s7r.reg4822 = s7r.reg4823 = s7r.reg4824 =
- s7r.reg4825 = s7r.reg4826 = s7r.reg4827 = s7r.reg4828 = s7r.reg4829 =
- s7r.reg482A = s7r.reg482B = s7r.reg482C = s7r.reg482D = 0;
+ s7r.reg4820 = s7r.reg4821 = s7r.reg4822 = s7r.reg4823 = s7r.reg4824 = s7r.reg4825 = s7r.reg4826 = s7r.reg4827 = s7r.reg4828 = s7r.reg4829 = s7r.reg482A = s7r.reg482B = s7r.reg482C = s7r.reg482D = 0;
s7r.reg482E = data;
break;
-
- //math status register possibly read only
-
- //SRAM toggle
- case 0x4830:
+ //math status register possibly read only
+ case 0x4830: //SRAM toggle
SPC7110Sram(data);
s7r.reg4830 = data;
break;
- //Bank DX mapping
- case 0x4831:
+ case 0x4831: //Bank DX mapping
s7r.reg4831 = data;
break;
- //Bank EX mapping
- case 0x4832:
+ case 0x4832: //Bank EX mapping
s7r.reg4832 = data;
break;
- //Bank FX mapping
- case 0x4833:
+ case 0x4833: //Bank FX mapping
s7r.reg4833 = data;
break;
- //S-RAM mapping? who knows?
- case 0x4834:
+ case 0x4834: //S-RAM mapping? who knows?
s7r.reg4834 = data;
break;
- //RTC Toggle
- case 0x4840:
- if (0 == data)
- {
+ case 0x4840: //RTC Toggle
+ if(!data)
S9xUpdateRTC();
- }
- if (data & 0x01)
+ else if(data & 0x01)
{
s7r.reg4842 = 0x80;
rtc_f9.init = false;
@@ -1234,11 +742,10 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
}
s7r.reg4840 = data;
break;
- //RTC init/command/index register
- case 0x4841:
+ case 0x4841: //RTC init/command/index register
if (rtc_f9.init)
{
- if (-1 == rtc_f9.index)
+ if (rtc_f9.index == -1)
{
rtc_f9.index = data & 0x0F;
break;
@@ -1251,8 +758,7 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
}
else
{
-
- if (0x0D == rtc_f9.index)
+ if (rtc_f9.index == 0x0D)
{
if (data & 0x08)
{
@@ -1285,7 +791,7 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
}
}
}
- if (0x0F == rtc_f9.index)
+ if (rtc_f9.index == 0x0F)
{
if (data & 0x01 && !(rtc_f9.reg[0x0F] & 0x01))
{
@@ -1315,10 +821,6 @@ void S9xSetSPC7110(uint8_t data, uint16_t Address)
}
}
break;
- //writes to RTC status register aren't expected to be meaningful
- default:
- break;
- //16 BIT MULTIPLIER: ($FF00) high byte, defval:00
}
}
@@ -1352,35 +854,25 @@ uint8_t S9xGetSPC7110Byte(uint32_t Address)
/**********************************************************************************************/
int32_t S9xRTCDaysInMonth(int32_t month, int32_t year)
{
- int32_t mdays;
-
- switch (month)
+ switch(month)
{
case 2:
- if ((year % 4 == 0)) // DKJM2 only uses 199x - 22xx
- mdays = 29;
- else
- mdays = 28;
- break;
-
+ if(year % 4 == 0) // DKJM2 only uses 199x - 22xx
+ return 29;
+ return 28;
case 4:
case 6:
case 9:
case 11:
- mdays = 30;
- break;
-
+ return 30;
default: // months 1,3,5,7,8,10,12
- mdays = 31;
- break;
+ return 31;
}
-
- return mdays;
}
-#define DAYTICKS (60*60*24)
-#define HOURTICKS (60*60)
-#define MINUTETICKS 60
+#define MINUTETICKS 60
+#define HOURTICKS (60 * MINUTETICKS)
+#define DAYTICKS (24 * HOURTICKS)
/**********************************************************************************************/
/* S9xUpdateRTC() */
@@ -1416,11 +908,9 @@ void S9xUpdateRTC(void)
int32_t month;
int32_t year;
int32_t temp_days;
-
int32_t year_tens;
int32_t year_ones;
-
if (time_diff > DAYTICKS)
{
days = time_diff / DAYTICKS;
@@ -1535,296 +1025,12 @@ uint8_t* Get7110BasePtr(uint32_t Address)
return &Memory.ROM[i];
}
-//loads the index into memory.
-//index.bin is little-endian
-//format index (1)-table(3)-file offset(4)-length(4)
-bool Load7110Index(char* filename)
-{
- FILE* fp;
- uint8_t buffer[12];
- int32_t table = 0;
- uint8_t index = 0;
- uint32_t offset = 0;
- uint32_t size = 0;
- int32_t i = 0;
- fp = fopen(filename, "rb");
- if (NULL == fp)
- return false;
-
- int32_t f_len;
- while (1)
- {
- i = 0;
- f_len = fread(buffer, 1, 12, fp);
- if (f_len < 12) break;
-
- table = (buffer[3] << 16) | (buffer[2] << 8) | buffer[1];
- index = buffer[0];
- offset = (buffer[7] << 24) | (buffer[6] << 16) | (buffer[5] << 8) | buffer[4];
- size = (buffer[11] << 24) | (buffer[10] << 16) | (buffer[9] << 8) | buffer[8];
- while (i < MAX_TABLES && decompack->tableEnts[i].table != table
- && decompack->tableEnts[i].table != 0)
- i++;
- if (i == MAX_TABLES)
- return false;
- //added
- decompack->tableEnts[i].table = table;
- //-----
- decompack->tableEnts[i].location[index].offset = offset;
- decompack->tableEnts[i].location[index].size = size;
- decompack->tableEnts[i].location[index].used_len = 0;
- decompack->tableEnts[i].location[index].used_offset = 0;
- }
- fclose(fp);
- return true;
-}
-
-//Cache 1 load function
-void SPC7110Load(char* dirname)
-{
- char temp_path[PATH_MAX];
- int32_t i = 0;
-
- decompack = (Pack7110*)malloc(sizeof(Pack7110));
-
-#if !defined(_XBOX) && !defined(VITA)
- getcwd(temp_path, PATH_MAX);
-#endif
-
- memset(decompack, 0, sizeof(Pack7110));
-
-#if !defined(_XBOX) && !defined(VITA)
- chdir(dirname);
-#endif
-
-#ifndef _XBOX
- Load7110Index("index.bin");
-#else
- // D:\\ is always app.path in Xbox
- Load7110Index("d:\\index.bin");
-#endif
-
- for (i = 0; i < MAX_TABLES; i++)
- {
- if (decompack->tableEnts[i].table != 0)
- {
- char binname[PATH_MAX];
-#ifndef _XBOX
- sprintf(binname, "%06X.bin", decompack->tableEnts[i].table);
-#else
- sprintf(binname, "%s%06X.bin", filename, decompack->tableEnts[i].table);
-#endif
- struct stat buf;
- if (-1 != stat(binname, &buf))
- decompack->binfiles[i] = (uint8_t*)malloc(buf.st_size);
- FILE* fp = fopen(binname, "rb");
- if (fp)
- {
- fread(decompack->binfiles[i], buf.st_size, 1, fp);
- fclose(fp);
- }
- }
- }
-
-#if !defined(_XBOX) && !defined(VITA)
- chdir(temp_path);
-#endif
-
- Copy7110 = &MovePackData;
- CleanUp7110 = &Del7110Gfx;
-
-}
-
-//Cache 2 load function
-void SPC7110Open(char* dirname)
-{
- char temp_path[PATH_MAX];
- int32_t i = 0;
-
- decompack = (Pack7110*)malloc(sizeof(Pack7110));
-
-#if !defined(_XBOX) && !defined(VITA)
- getcwd(temp_path, PATH_MAX);
-#endif
-
- memset(decompack, 0, sizeof(Pack7110));
-
-#if !defined(_XBOX) && !defined(VITA)
- chdir(dirname);
-#endif
-
-#ifndef _XBOX
- Load7110Index("index.bin");
-#else
- // D:\\ is always app.path in Xbox
- Load7110Index("d:\\index.bin");
-#endif
-
- for (i = 0; i < MAX_TABLES; i++)
- decompack->binfiles[i] = NULL;
-
- ReadPackData();
-
-#if !defined(_XBOX) && !defined(VITA)
- chdir(temp_path);
-#endif
-
- Copy7110 = &ReadPackData;
- CleanUp7110 = &Close7110Gfx;
-}
-
-//Cache 3's load function
-void SPC7110Grab(char* dirname)
-{
- char temp_path[PATH_MAX];
- int32_t i = 0;
-
- decompack = (Pack7110*)malloc(sizeof(Pack7110));
-
-#if !defined(_XBOX) && !defined(VITA)
- getcwd(temp_path, PATH_MAX);
-#endif
-
- int32_t buffer_size = 1024 * 1024 * cacheMegs; //*some setting
-
- memset(decompack, 0, sizeof(Pack7110));
-#if !defined(_XBOX) && !defined(VITA)
- chdir(dirname);
-#endif
-
-#ifndef _XBOX
- Load7110Index("index.bin");
-#else
- // D:\\ is always app.path in Xbox
- Load7110Index("d:\\index.bin");
-#endif
-
- for (i = 0; i < MAX_TABLES; i++)
- {
- if (decompack->tableEnts[i].table != 0)
- {
- char binname[PATH_MAX];
-#ifndef _XBOX
- sprintf(binname, "%06X.bin", decompack->tableEnts[i].table);
-#else
- sprintf(binname, "%s%06X.bin", filename, decompack->tableEnts[i].table);
-#endif
- struct stat buf;
- //add load/no load calculations here
- if (-1 != stat(binname, &buf))
- {
- if (buf.st_size < buffer_size)
- decompack->binfiles[i] = (uint8_t*)malloc(buf.st_size);
- FILE* fp = fopen(binname, "rb");
- //use them here
- if (fp)
- {
- if (buf.st_size < buffer_size)
- {
- fread(decompack->binfiles[i], buf.st_size, 1, fp);
- fclose(fp);
- buffer_size -= buf.st_size;
- decompack->tableEnts[i].is_file = false;
- }
- else
- {
- decompack->binfiles[i] = (uint8_t*)fp;
- decompack->tableEnts[i].is_file = true;
- }
- }
- }
- }
- }
-
-#if !defined(_XBOX) && !defined(VITA)
- chdir(temp_path);
-#endif
-
- Copy7110 = &GetPackData;
- CleanUp7110 = &Drop7110Gfx;
-}
-
//Cache 1 clean up function
void Del7110Gfx(void)
{
- int32_t i;
- if (Settings.SPC7110)
- {
- Do7110Logging();
- }
- for (i = 0; i < MAX_TABLES; i++)
- {
- if (decompack->binfiles[i] != NULL)
- {
- free(decompack->binfiles[i]);
- decompack->binfiles[i] = NULL;
- }
- }
+ spc7110dec_deinit();
Settings.SPC7110 = false;
Settings.SPC7110RTC = false;
- if (NULL != decompack)
- free(decompack);
- decompack = NULL;
- CleanUp7110 = NULL;
- Copy7110 = NULL;
-}
-
-//Cache2 cleanup function
-void Close7110Gfx(void)
-{
- int32_t i;
- if (Settings.SPC7110)
- {
- Do7110Logging();
- }
- for (i = 0; i < MAX_TABLES; i++)
- {
- if (decompack->binfiles[i] != NULL)
- {
- fclose((FILE*)decompack->binfiles[i]);
- decompack->binfiles[i] = NULL;
- }
- }
- Settings.SPC7110 = false;
- Settings.SPC7110RTC = false;
- if (NULL != decompack)
- free(decompack);
- decompack = NULL;
- CleanUp7110 = NULL;
- Copy7110 = NULL;
-}
-
-//cache 3's clean-up code
-void Drop7110Gfx(void)
-{
- int32_t i;
- if (Settings.SPC7110)
- {
- Do7110Logging();
- }
- for (i = 0; i < MAX_TABLES; i++)
- {
- if (decompack->binfiles[i] != NULL)
- {
- if (decompack->tableEnts[i].is_file)
- {
- fclose((FILE*)decompack->binfiles[i]);
- decompack->binfiles[i] = NULL;
- }
- else
- {
- free(decompack->binfiles[i]);
- decompack->binfiles[i] = NULL;
- }
- }
- }
- Settings.SPC7110 = false;
- Settings.SPC7110RTC = false;
- if (NULL != decompack)
- free(decompack);
- decompack = NULL;
- CleanUp7110 = NULL;
- Copy7110 = NULL;
}
//emulate a reset.
@@ -1880,266 +1086,5 @@ void S9xSpc7110Reset(void)
s7r.AlignBy = 1;
s7r.bank50Internal = 0;
memset(s7r.bank50, 0x00, DECOMP_BUFFER_SIZE);
-}
-
-//outputs a cumulative log for the game.
-//there's nothing really weird here, just
-//reading the old log, and writing a new one.
-//note the logs are explicitly little-endian, not host byte order.
-void Do7110Logging(void)
-{
- uint8_t ent_temp;
- FILE* flog;
- int32_t entries = 0;
-
- if (Settings.SPC7110)
- {
- //flush last read into logging
- (*Copy7110)();
-
- if (!strncmp((char*)&Memory.ROM [0xffc0], "SUPER POWER LEAG 4 ", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\spl4-sp7.dat", "rb");
-#else
- flog = fopen("spl4-sp7.dat", "rb");
-#endif
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "MOMOTETSU HAPPY ", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\smht-sp7.dat", "rb");
-#else
- flog = fopen("smht-sp7.dat", "rb");
-#endif
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "HU TENGAI MAKYO ZERO ", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\feoezsp7.dat", "rb");
-#else
- flog = fopen("feoezsp7.dat", "rb");
-#endif
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "JUMP TENGAIMAKYO ZERO", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\sjumpsp7.dat", "rb");
-#else
- flog = fopen("sjumpsp7.dat", "rb");
-#endif
- }
- else
- {
-#ifdef _XBOX
- flog = fopen("T:\\misc-sp7.dat", "rb");
-#else
- flog = fopen("misc-sp7.dat", "rb");
-#endif
- }
-
- if (flog)
- {
- uint8_t buffer[8];
- int32_t table = 0;
- uint16_t offset = 0;
- uint16_t length = 0;
- fseek(flog, 35, 0);
-
- int32_t f_len;
- while (1)
- {
- int32_t i = 0;
- Data7110* log = NULL;
- f_len = fread(buffer, 1, 8, flog);
- if (f_len < 8) break;
-
- table = buffer[0] | (buffer[1] << 8) | (buffer[2] << 16);
- offset = buffer[6] | (buffer[7] << 8);
- length = buffer[4] | (buffer[5] << 8);
- while (i < MAX_TABLES && log == NULL)
- {
- if (decompack->tableEnts[i].table == table)
- {
- log = &(decompack->tableEnts[i].location[(buffer[3])]);
- if ((log->used_offset + log->used_len) < (offset + length))
- {
- log->used_offset = offset;
- log->used_len = length;
- }
- }
- i++;
- }
- }
- fclose(flog);
- }
-
-
- if (!strncmp((char*)&Memory.ROM [0xffc0], "SUPER POWER LEAG 4 ", 21))
- {
-#ifdef _XBOX // cwd could be the dvd-rom, so write to T:\\ which is storage region for each title
- flog = fopen("T:\\spl4-sp7.dat", "wb");
-#else
- flog = fopen("spl4-sp7.dat", "wb");
-#endif
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "MOMOTETSU HAPPY ", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\smht-sp7.dat", "wb");
-#else
- flog = fopen("smht-sp7.dat", "wb");
-#endif
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "HU TENGAI MAKYO ZERO ", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\feoezsp7.dat", "wb");
-#else
- flog = fopen("feoezsp7.dat", "wb");
-#endif
- }
- else if (!strncmp((char*)&Memory.ROM [0xffc0], "JUMP TENGAIMAKYO ZERO", 21))
- {
-#ifdef _XBOX
- flog = fopen("T:\\sjumpsp7.dat", "wb");
-#else
- flog = fopen("sjumpsp7.dat", "wb");
-#endif
- }
- else
- {
-#ifdef _XBOX
- flog = fopen("T:\\misc-sp7.dat", "wb");
-#else
- flog = fopen("misc-sp7.dat", "wb");
-#endif
- }
- //count entries
- if (flog)
- {
- int32_t j = 0;
- int32_t temp = 0;
- for (j = 0; j < MAX_TABLES; j++)
- {
- int32_t k;
- for (k = 0; k < 256; k++)
- {
- if (decompack->tableEnts[j].location[k].used_len != 0)
- entries++;
- }
- }
- ent_temp = entries & 0xFF;
- fwrite(&ent_temp, 1, 1, flog);
- ent_temp = (entries >> 8) & 0xFF;
- fwrite(&ent_temp, 1, 1, flog);
- ent_temp = (entries >> 16) & 0xFF;
- fwrite(&ent_temp, 1, 1, flog);
- ent_temp = (entries >> 24) & 0xFF;
- fwrite(&ent_temp, 1, 1, flog);
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
-
- ent_temp = 0;
- fwrite(&ent_temp, 1, 1, flog);
- ent_temp = 0;
- fwrite(&ent_temp, 1, 1, flog);
- ent_temp = 0;
- fwrite(&ent_temp, 1, 1, flog);
-
- for (j = 0; j < MAX_TABLES; j++)
- {
- int32_t k;
- for (k = 0; k < 256; k++)
- {
- if (decompack->tableEnts[j].location[k].used_len != 0)
- {
- ent_temp = decompack->tableEnts[j].table & 0xFF;
- fwrite(&ent_temp, 1, 1, flog); //801
- ent_temp = (decompack->tableEnts[j].table >> 8) & 0xFF;;
- fwrite(&ent_temp, 1, 1, flog); //802
- ent_temp = (decompack->tableEnts[j].table >> 16) & 0xFF;;
- fwrite(&ent_temp, 1, 1, flog); //803
- ent_temp = k & 0xFF;
- fwrite(&ent_temp, 1, 1, flog); //804
- ent_temp = decompack->tableEnts[j].location[k].used_len & 0xFF;
- fwrite(&ent_temp, 1, 1, flog); //lsb of
- ent_temp = (decompack->tableEnts[j].location[k].used_len >> 8) & 0xFF;
- fwrite(&ent_temp, 1, 1, flog); //msb of
- ent_temp = (decompack->tableEnts[j].location[k].used_offset) & 0xFF;
- fwrite(&ent_temp, 1, 1, flog); //lsb of
- ent_temp = (decompack->tableEnts[j].location[k].used_offset >> 8) & 0xFF;
- fwrite(&ent_temp, 1, 1, flog); //msb of
- }
- }
- }
- fwrite(&temp, 1, 4, flog);
- fwrite(&temp, 1, 4, flog);
- fclose(flog);
- }
- }
-}
-
-bool S9xSaveSPC7110RTC(S7RTC* rtc_f9)
-{
- FILE* fp;
-
- if ((fp = fopen(S9xGetFilename("rtc"), "wb")) == NULL)
- return (false);
- int32_t i = 0;
- uint8_t temp = 0;
- for (i = 0; i < 16; i++)
- fwrite(&rtc_f9->reg[i], 1, 1, fp);
- temp = rtc_f9->index & 0x00FF;
- fwrite(&temp, 1, 1, fp);
- temp = (rtc_f9->index) >> 8;
- fwrite(&temp, 1, 1, fp);
- temp = (uint8_t)rtc_f9->control;
- fwrite(&temp, 1, 1, fp);
- temp = (uint8_t)rtc_f9->init;
- fwrite(&temp, 1, 1, fp);
- temp = rtc_f9->last_used & 0x00FF;
- fwrite(&temp, 1, 1, fp);
- temp = (rtc_f9->last_used >> 8) & 0x00FF;
- fwrite(&temp, 1, 1, fp);
- temp = (rtc_f9->last_used >> 16) & 0x00FF;
- fwrite(&temp, 1, 1, fp);
- temp = (rtc_f9->last_used >> 24) & 0x00FF;;
- fwrite(&temp, 1, 1, fp);
- fclose(fp);
- return (true);
-}
-
-bool S9xLoadSPC7110RTC(S7RTC* rtc_f9)
-{
- FILE* fp;
-
- if ((fp = fopen(S9xGetFilename("rtc"), "rb")) == NULL)
- return (false);
- int32_t i;
- for (i = 0; i < 16; i++)
- fread(&(rtc_f9->reg[i]), 1, 1, fp);
- uint8_t temp = 0;
- fread(&temp, 1, 1, fp);
- rtc_f9->index = temp;
- fread(&temp, 1, 1, fp);
- rtc_f9->index |= (temp << 8);
- fread(&rtc_f9->control, 1, 1, fp);
- fread(&rtc_f9->init, 1, 1, fp);
-
- fread(&temp, 1, 1, fp);
- rtc_f9->last_used = temp;
- fread(&temp, 1, 1, fp);
- rtc_f9->last_used |= (temp << 8);
- fread(&temp, 1, 1, fp);
- rtc_f9->last_used |= (temp << 16);
- fread(&temp, 1, 1, fp);
- rtc_f9->last_used |= (temp << 24);
- fclose(fp);
- return (true);
+ spc7110dec_reset();
}
diff --git a/source/spc7110.h b/source/spc7110.h
index dd93d50..a5edea6 100644
--- a/source/spc7110.h
+++ b/source/spc7110.h
@@ -6,15 +6,7 @@
#define DECOMP_BUFFER_SIZE 0x10000
-extern void (*LoadUp7110)(char*);
-extern void (*CleanUp7110)(void);
-extern void (*Copy7110)(void);
-
-extern uint16_t cacheMegs;
-
void Del7110Gfx(void);
-void Close7110Gfx(void);
-void Drop7110Gfx(void);
uint8_t S9xGetSPC7110(uint16_t Address);
uint8_t S9xGetSPC7110Byte(uint32_t Address);
uint8_t* Get7110BasePtr(uint32_t);
@@ -26,12 +18,6 @@ void S9xUpdateRTC(void);
void Do7110Logging(void);
int32_t S9xRTCDaysInMonth(int32_t month, int32_t year);
-//These are platform-dependant functions, but should work on
-//most systems that use GNU compilers, and on Win32.
-void SPC7110Load(char*);
-void SPC7110Open(char*);
-void SPC7110Grab(char*);
-
typedef struct
{
uint8_t reg[16];
@@ -99,9 +85,4 @@ typedef struct
extern SPC7110Regs s7r;
extern S7RTC rtc_f9;
-
-// These are defined in spc7110.cpp
-bool S9xSaveSPC7110RTC(S7RTC* rtc_f9);
-bool S9xLoadSPC7110RTC(S7RTC* rtc_f9);
-
#endif
diff --git a/source/spc7110dec.c b/source/spc7110dec.c
new file mode 100644
index 0000000..f5dc963
--- /dev/null
+++ b/source/spc7110dec.c
@@ -0,0 +1,611 @@
+#include "../copyright"
+
+#include "memmap.h"
+#include "spc7110dec.h"
+
+#define SPC7110_DECOMP_BUFFER_SIZE 64 /* must be >= 64, and must be a power of two */
+
+static const uint8_t evolution_table[53][4] = /* { prob, nextlps, nextmps, toggle invert } */
+{
+ {0x5a, 1, 1, 1}, {0x25, 6, 2, 0}, {0x11, 8, 3, 0},
+ {0x08, 10, 4, 0}, {0x03, 12, 5, 0}, {0x01, 15, 5, 0},
+ {0x5a, 7, 7, 1}, {0x3f, 19, 8, 0}, {0x2c, 21, 9, 0},
+ {0x20, 22, 10, 0}, {0x17, 23, 11, 0}, {0x11, 25, 12, 0},
+ {0x0c, 26, 13, 0}, {0x09, 28, 14, 0}, {0x07, 29, 15, 0},
+ {0x05, 31, 16, 0}, {0x04, 32, 17, 0}, {0x03, 34, 18, 0},
+ {0x02, 35, 5, 0}, {0x5a, 20, 20, 1}, {0x48, 39, 21, 0},
+ {0x3a, 40, 22, 0}, {0x2e, 42, 23, 0}, {0x26, 44, 24, 0},
+ {0x1f, 45, 25, 0}, {0x19, 46, 26, 0}, {0x15, 25, 27, 0},
+ {0x11, 26, 28, 0}, {0x0e, 26, 29, 0}, {0x0b, 27, 30, 0},
+ {0x09, 28, 31, 0}, {0x08, 29, 32, 0}, {0x07, 30, 33, 0},
+ {0x05, 31, 34, 0}, {0x04, 33, 35, 0}, {0x04, 33, 36, 0},
+ {0x03, 34, 37, 0}, {0x02, 35, 38, 0}, {0x02, 36, 5, 0},
+ {0x58, 39, 40, 1}, {0x4d, 47, 41, 0}, {0x43, 48, 42, 0},
+ {0x3b, 49, 43, 0}, {0x34, 50, 44, 0}, {0x2e, 51, 45, 0},
+ {0x29, 44, 46, 0}, {0x25, 45, 24, 0}, {0x56, 47, 48, 1},
+ {0x4f, 47, 49, 0}, {0x47, 48, 50, 0}, {0x41, 49, 51, 0},
+ {0x3c, 50, 52, 0}, {0x37, 51, 43, 0},
+};
+
+static const uint8_t mode2_context_table[32][2] = /* { next 0, next 1 } */
+{
+ {1, 2}, {3, 8}, {13, 14}, {15, 16},
+ {17, 18}, {19, 20}, {21, 22}, {23, 24},
+ {25, 26}, {25, 26}, {25, 26}, {25, 26},
+ {25, 26}, {27, 28}, {29, 30}, {31, 31},
+ {31, 31}, {31, 31}, {31, 31}, {31, 31},
+ {31, 31}, {31, 31}, {31, 31}, {31, 31},
+ {31, 31}, {31, 31}, {31, 31}, {31, 31},
+ {31, 31}, {31, 31}, {31, 31}, {31, 31},
+};
+
+typedef struct
+{
+ uint32_t mode;
+ uint32_t offset;
+ uint32_t original_mode;
+ uint32_t original_offset;
+ uint32_t original_index;
+ uint32_t read_counter;
+ uint8_t *buffer;
+ uint32_t buffer_rdoffset;
+ uint32_t buffer_wroffset;
+ uint32_t buffer_length;
+
+ struct ContextState
+ {
+ uint8_t index;
+ uint8_t invert;
+ } context[32];
+
+ uint32_t morton16[2][256];
+ uint32_t morton32[4][256];
+} SPC7110Decomp;
+
+SPC7110Decomp decomp;
+
+uint8_t spc7110dec_read(void)
+{
+ decomp.read_counter++;
+
+ if(decomp.buffer_length == 0)
+ {
+ switch(decomp.mode)
+ {
+ case 0:
+ spc7110dec_mode0(false);
+ break;
+ case 1:
+ spc7110dec_mode1(false);
+ break;
+ case 2:
+ spc7110dec_mode2(false);
+ break;
+ default:
+ return 0x00;
+ }
+ }
+
+ uint8_t data = decomp.buffer[decomp.buffer_rdoffset++];
+ decomp.buffer_rdoffset &= SPC7110_DECOMP_BUFFER_SIZE - 1;
+ decomp.buffer_length--;
+ return data;
+}
+
+void spc7110dec_write(uint8_t data)
+{
+ decomp.buffer[decomp.buffer_wroffset++] = data;
+ decomp.buffer_wroffset &= SPC7110_DECOMP_BUFFER_SIZE - 1;
+ decomp.buffer_length++;
+}
+
+uint8_t spc7110dec_dataread(void)
+{
+ uint32_t size = Memory.CalculatedSize - 0x100000;
+ while(decomp.offset >= size)
+ decomp.offset -= size;
+ return Memory.ROM[0x100000 + decomp.offset++];
+}
+
+void spc7110dec_clear(uint32_t mode, uint32_t offset, uint32_t index)
+{
+ decomp.original_mode = mode;
+ decomp.original_offset = offset;
+ decomp.original_index = index;
+ decomp.mode = mode;
+ decomp.offset = offset;
+ decomp.buffer_rdoffset = 0;
+ decomp.buffer_wroffset = 0;
+ decomp.buffer_length = 0;
+ uint32_t i;
+
+ for(i = 0; i < 32; i++) // reset decomp.context states
+ {
+ decomp.context[i].index = 0;
+ decomp.context[i].invert = 0;
+ }
+
+ switch(decomp.mode)
+ {
+ case 0:
+ spc7110dec_mode0(true);
+ break;
+ case 1:
+ spc7110dec_mode1(true);
+ break;
+ case 2:
+ spc7110dec_mode2(true);
+ break;
+ }
+
+ while(index--) // decompress up to requested output data index
+ spc7110dec_read();
+
+ decomp.read_counter = 0;
+}
+
+void spc7110dec_mode0(bool init)
+{
+ static uint8_t val, in, span;
+ static int32_t out, inverts, lps, in_count;
+
+ if(init)
+ {
+ out = inverts = lps = 0;
+ span = 0xff;
+ val = spc7110dec_dataread();
+ in = spc7110dec_dataread();
+ in_count = 8;
+ return;
+ }
+
+ while(decomp.buffer_length < (SPC7110_DECOMP_BUFFER_SIZE >> 1))
+ {
+ uint32_t bit;
+ for(bit = 0; bit < 8; bit++)
+ {
+ /* Get decomp.context */
+ uint8_t mask = (1 << (bit & 3)) - 1;
+ uint8_t con = mask + ((inverts & mask) ^ (lps & mask));
+
+ if(bit > 3)
+ con += 15;
+
+ /* Get prob and mps */
+ uint32_t prob = spc7110dec_probability(con);
+ uint32_t mps = (((out >> 15) & 1) ^ decomp.context[con].invert);
+
+ /* Get bit */
+ uint32_t flag_lps;
+
+ if(val <= span - prob) // mps
+ {
+ span = span - prob;
+ out = (out << 1) + mps;
+ flag_lps = 0;
+ }
+ else // lps
+ {
+ val = val - (span - (prob - 1));
+ span = prob - 1;
+ out = (out << 1) + 1 - mps;
+ flag_lps = 1;
+ }
+
+ /* Renormalize */
+ uint32_t shift = 0;
+
+ while(span < 0x7f)
+ {
+ shift++;
+ span = (span << 1) + 1;
+ val = (val << 1) + (in >> 7);
+ in <<= 1;
+
+ if(--in_count == 0)
+ {
+ in = spc7110dec_dataread();
+ in_count = 8;
+ }
+ }
+
+ /* Update processing info */
+ lps = (lps << 1) + flag_lps;
+ inverts = (inverts << 1) + decomp.context[con].invert;
+
+ /* Update context state */
+ if(flag_lps & spc7110dec_toggle_invert(con))
+ decomp.context[con].invert ^= 1;
+
+ if(flag_lps)
+ decomp.context[con].index = spc7110dec_next_lps(con);
+ else if(shift)
+ decomp.context[con].index = spc7110dec_next_mps(con);
+ }
+
+ /* Save byte */
+ spc7110dec_write(out);
+ }
+}
+
+void spc7110dec_mode1(bool init)
+{
+ static uint32_t pixelorder[4], realorder[4];
+ static uint8_t in, val, span;
+ static int32_t out, inverts, lps, in_count;
+
+ if(init)
+ {
+ uint32_t i;
+ for(i = 0; i < 4; i++)
+ pixelorder[i] = i;
+ out = inverts = lps = 0;
+ span = 0xff;
+ val = spc7110dec_dataread();
+ in = spc7110dec_dataread();
+ in_count = 8;
+ return;
+ }
+
+ while(decomp.buffer_length < (SPC7110_DECOMP_BUFFER_SIZE >> 1))
+ {
+ uint32_t pixel;
+
+ for(pixel = 0; pixel < 8; pixel++)
+ {
+ /* Get first symbol decomp.context */
+ uint32_t a = ((out >> (1 * 2)) & 3);
+ uint32_t b = ((out >> (7 * 2)) & 3);
+ uint32_t c = ((out >> (8 * 2)) & 3);
+ uint32_t con = (a == b) ? (b != c) : (b == c) ? 2 : 4 - (a == c);
+
+ /* Update pixel order */
+ uint32_t m, n;
+
+ for(m = 0; m < 4; m++)
+ if(pixelorder[m] == a)
+ break;
+
+ for(n = m; n > 0; n--)
+ pixelorder[n] = pixelorder[n - 1];
+
+ pixelorder[0] = a;
+
+ /* Calculate the real pixel order */
+ for(m = 0; m < 4; m++)
+ realorder[m] = pixelorder[m];
+
+ /* Rotate reference pixel c value to top */
+ for(m = 0; m < 4; m++)
+ if(realorder[m] == c)
+ break;
+
+ for(n = m; n > 0; n--)
+ realorder[n] = realorder[n - 1];
+
+ realorder[0] = c;
+
+ /* Rotate reference pixel b value to top */
+ for(m = 0; m < 4; m++)
+ if(realorder[m] == b)
+ break;
+
+ for(n = m; n > 0; n--)
+ realorder[n] = realorder[n - 1];
+
+ realorder[0] = b;
+
+ /* Rotate reference pixel a value to top */
+ for(m = 0; m < 4; m++)
+ if(realorder[m] == a)
+ break;
+
+ for(n = m; n > 0; n--)
+ realorder[n] = realorder[n - 1];
+
+ realorder[0] = a;
+
+ /* Get 2 symbols */
+ uint32_t bit;
+
+ for(bit = 0; bit < 2; bit++)
+ {
+ /* Get prob */
+ uint32_t prob = spc7110dec_probability(con);
+
+ /* Get symbol */
+ uint32_t flag_lps;
+
+ if(val <= span - prob) // mps
+ {
+ span = span - prob;
+ flag_lps = 0;
+ }
+ else // lps
+ {
+ val = val - (span - (prob - 1));
+ span = prob - 1;
+ flag_lps = 1;
+ }
+
+ /* Renormalize */
+ uint32_t shift = 0;
+
+ while(span < 0x7f)
+ {
+ shift++;
+ span = (span << 1) + 1;
+ val = (val << 1) + (in >> 7);
+ in <<= 1;
+ if(--in_count == 0)
+ {
+ in = spc7110dec_dataread();
+ in_count = 8;
+ }
+ }
+
+ /* Update processing info */
+ lps = (lps << 1) + flag_lps;
+ inverts = (inverts << 1) + decomp.context[con].invert;
+
+ /* Update context state */
+ if(flag_lps & spc7110dec_toggle_invert(con))
+ decomp.context[con].invert ^= 1;
+
+ if(flag_lps)
+ decomp.context[con].index = spc7110dec_next_lps(con);
+ else if(shift)
+ decomp.context[con].index = spc7110dec_next_mps(con);
+
+ /* Get next decomp.context */
+ con = 5 + (con << 1) + ((lps ^ inverts) & 1);
+ }
+
+ /* Get pixel */
+ b = realorder[(lps ^ inverts) & 3];
+ out = (out << 2) + b;
+ }
+
+ /* Turn pixel data into bitplanes */
+ uint32_t data = spc7110dec_morton_2x8(out);
+
+ spc7110dec_write(data >> 8);
+ spc7110dec_write(data >> 0);
+ }
+}
+
+void spc7110dec_mode2(bool init)
+{
+ static uint32_t pixelorder[16], realorder[16];
+ static uint8_t bitplanebuffer[16], buffer_index;
+ static uint8_t in, val, span;
+ static int32_t out0, out1, inverts, lps, in_count;
+
+ if(init)
+ {
+ uint32_t i;
+
+ for(i = 0; i < 16; i++)
+ pixelorder[i] = i;
+ buffer_index = 0;
+ out0 = out1 = inverts = lps = 0;
+ span = 0xff;
+ val = spc7110dec_dataread();
+ in = spc7110dec_dataread();
+ in_count = 8;
+ return;
+ }
+
+ while(decomp.buffer_length < (SPC7110_DECOMP_BUFFER_SIZE >> 1))
+ {
+ uint32_t pixel;
+
+ for(pixel = 0; pixel < 8; pixel++)
+ {
+ /* Get first symbol context */
+ uint32_t a = ((out0 >> (0 * 4)) & 15);
+ uint32_t b = ((out0 >> (7 * 4)) & 15);
+ uint32_t c = ((out1 >> (0 * 4)) & 15);
+ uint32_t con = 0;
+ uint32_t refcon = (a == b) ? (b != c) : (b == c) ? 2 : 4 - (a == c);
+
+ /* Update pixel order */
+ uint32_t m, n;
+
+ for(m = 0; m < 16; m++)
+ if(pixelorder[m] == a)
+ break;
+
+ for(n = m; n > 0; n--)
+ pixelorder[n] = pixelorder[n - 1];
+
+ pixelorder[0] = a;
+
+ /* Calculate the real pixel order */
+ for(m = 0; m < 16; m++)
+ realorder[m] = pixelorder[m];
+
+ /* Rotate reference pixel c value to top */
+ for(m = 0; m < 16; m++)
+ if(realorder[m] == c)
+ break;
+
+ for(n = m; n > 0; n--)
+ realorder[n] = realorder[n - 1];
+
+ realorder[0] = c;
+
+ /* Rotate reference pixel b value to top */
+ for(m = 0; m < 16; m++)
+ if(realorder[m] == b)
+ break;
+
+ for(n = m; n > 0; n--)
+ realorder[n] = realorder[n - 1];
+
+ realorder[0] = b;
+
+ /* Rotate reference pixel a value to top */
+ for(m = 0; m < 16; m++)
+ if(realorder[m] == a)
+ break;
+
+ for(n = m; n > 0; n--)
+ realorder[n] = realorder[n - 1];
+
+ realorder[0] = a;
+
+ /* Get 4 symbols */
+ uint32_t bit;
+
+ for(bit = 0; bit < 4; bit++)
+ {
+ /* Get prob */
+ uint32_t prob = spc7110dec_probability(con);
+
+ /* Get symbol */
+ uint32_t flag_lps;
+
+ if(val <= span - prob) // mps
+ {
+ span = span - prob;
+ flag_lps = 0;
+ }
+ else // lps
+ {
+ val = val - (span - (prob - 1));
+ span = prob - 1;
+ flag_lps = 1;
+ }
+
+ /* Renormalize */
+ uint32_t shift = 0;
+
+ while(span < 0x7f)
+ {
+ shift++;
+ span = (span << 1) + 1;
+ val = (val << 1) + (in >> 7);
+ in <<= 1;
+ if(--in_count == 0)
+ {
+ in = spc7110dec_dataread();
+ in_count = 8;
+ }
+ }
+
+ /* Update processing info */
+ lps = (lps << 1) + flag_lps;
+ uint32_t invertbit = decomp.context[con].invert;
+ inverts = (inverts << 1) + invertbit;
+
+ /* Update decomp.context state */
+ if(flag_lps & spc7110dec_toggle_invert(con))
+ decomp.context[con].invert ^= 1;
+
+ if(flag_lps)
+ decomp.context[con].index = spc7110dec_next_lps(con);
+ else if(shift)
+ decomp.context[con].index = spc7110dec_next_mps(con);
+
+ /* Get next decomp.context */
+ con = mode2_context_table[con][flag_lps ^ invertbit] + (con == 1 ? refcon : 0);
+ }
+
+ /* Get pixel */
+ b = realorder[(lps ^ inverts) & 0x0f];
+ out1 = (out1 << 4) + ((out0 >> 28) & 0x0f);
+ out0 = (out0 << 4) + b;
+ }
+
+ /* Convert pixel data into bitplanes */
+ uint32_t data = spc7110dec_morton_4x8(out0);
+ spc7110dec_write(data >> 24);
+ spc7110dec_write(data >> 16);
+ bitplanebuffer[buffer_index++] = data >> 8;
+ bitplanebuffer[buffer_index++] = data >> 0;
+
+ if(buffer_index == 16)
+ {
+ uint32_t i;
+ for(i = 0; i < 16; i++)
+ spc7110dec_write(bitplanebuffer[i]);
+ buffer_index = 0;
+ }
+ }
+}
+
+uint8_t spc7110dec_probability(uint32_t n)
+{
+ return evolution_table[decomp.context[n].index][0];
+}
+
+uint8_t spc7110dec_next_lps(uint32_t n)
+{
+ return evolution_table[decomp.context[n].index][1];
+}
+
+uint8_t spc7110dec_next_mps(uint32_t n)
+{
+ return evolution_table[decomp.context[n].index][2];
+}
+
+bool spc7110dec_toggle_invert(uint32_t n)
+{
+ return evolution_table[decomp.context[n].index][3];
+}
+
+uint32_t spc7110dec_morton_2x8(uint32_t data)
+{
+ /* Reverse morton lookup: de-interleave two 8-bit values
+ * 15, 13, 11, 9, 7, 5, 3, 1 -> 15-8
+ * 14, 12, 10, 8, 6, 4, 2, 0 -> 7 -0 */
+ return decomp.morton16[0][(data >> 0) & 255] + decomp.morton16[1][(data >> 8) & 255];
+}
+
+uint32_t spc7110dec_morton_4x8(uint32_t data)
+{
+ /* Reverse morton lookup: de-interleave four 8-bit values
+ * 31, 27, 23, 19, 15, 11, 7, 3 -> 31-24
+ * 30, 26, 22, 18, 14, 10, 6, 2 -> 23-16
+ * 29, 25, 21, 17, 13, 9, 5, 1 -> 15-8
+ * 28, 24, 20, 16, 12, 8, 4, 0 -> 7 -0 */
+ return decomp.morton32[0][(data >> 0) & 255] + decomp.morton32[1][(data >> 8) & 255] + decomp.morton32[2][(data >> 16) & 255] + decomp.morton32[3][(data >> 24) & 255];
+}
+
+void spc7110dec_reset(void)
+{
+ /* Mode 3 is invalid; this is treated as a special case to always return 0x00
+ * set to mode 3 so that reading decomp port before starting first decomp will return 0x00 */
+ decomp.mode = 3;
+ decomp.buffer_rdoffset = 0;
+ decomp.buffer_wroffset = 0;
+ decomp.buffer_length = 0;
+}
+
+void spc7110dec_init(void)
+{
+ decomp.buffer = malloc(SPC7110_DECOMP_BUFFER_SIZE);
+ spc7110dec_reset();
+
+ /* Initialize reverse morton lookup tables */
+ uint32_t i;
+ for(i = 0; i < 256; i++)
+ {
+ #define map(x, y) (((i >> x) & 1) << y)
+ /* 2x8-bit */
+ decomp.morton16[1][i] = map(7, 15) + map(6, 7) + map(5, 14) + map(4, 6) + map(3, 13) + map(2, 5) + map(1, 12) + map(0, 4);
+ decomp.morton16[0][i] = map(7, 11) + map(6, 3) + map(5, 10) + map(4, 2) + map(3, 9) + map(2, 1) + map(1, 8) + map(0, 0);
+ /* 4x8-bit */
+ decomp.morton32[3][i] = map(7, 31) + map(6, 23) + map(5, 15) + map(4, 7) + map(3, 30) + map(2, 22) + map(1, 14) + map(0, 6);
+ decomp.morton32[2][i] = map(7, 29) + map(6, 21) + map(5, 13) + map(4, 5) + map(3, 28) + map(2, 20) + map(1, 12) + map(0, 4);
+ decomp.morton32[1][i] = map(7, 27) + map(6, 19) + map(5, 11) + map(4, 3) + map(3, 26) + map(2, 18) + map(1, 10) + map(0, 2);
+ decomp.morton32[0][i] = map(7, 25) + map(6, 17) + map(5, 9) + map(4, 1) + map(3, 24) + map(2, 16) + map(1, 8) + map(0, 0);
+ #undef map
+ }
+}
+
+void spc7110dec_deinit(void)
+{
+ free(decomp.buffer);
+}
diff --git a/source/spc7110dec.h b/source/spc7110dec.h
new file mode 100644
index 0000000..e03a855
--- /dev/null
+++ b/source/spc7110dec.h
@@ -0,0 +1,28 @@
+#include "../copyright"
+
+#ifndef _SPC7110DEC_H_
+#define _SPC7110DEC_H_
+#include "port.h"
+
+uint8_t spc7110dec_read(void);
+void spc7110dec_clear(uint32_t mode, uint32_t offset, uint32_t index);
+void spc7110dec_reset(void);
+
+void spc7110dec_init(void);
+void spc7110dec_deinit(void);
+
+void spc7110dec_write(uint8_t data);
+uint8_t spc7110dec_dataread(void);
+
+void spc7110dec_mode0(bool init);
+void spc7110dec_mode1(bool init);
+void spc7110dec_mode2(bool init);
+
+uint8_t spc7110dec_probability(uint32_t n);
+uint8_t spc7110dec_next_lps(uint32_t n);
+uint8_t spc7110dec_next_mps(uint32_t n);
+bool spc7110dec_toggle_invert(uint32_t n);
+
+uint32_t spc7110dec_morton_2x8(uint32_t data);
+uint32_t spc7110dec_morton_4x8(uint32_t data);
+#endif
diff --git a/source/srtc.c b/source/srtc.c
index 9e7c2e9..6427f0b 100644
--- a/source/srtc.c
+++ b/source/srtc.c
@@ -34,12 +34,10 @@ Index Description Range (nibble)
SRTC_DATA rtc;
-
static int32_t month_keys[12] = { 1, 4, 4, 0, 2, 5, 0, 3, 6, 1, 4, 6 };
/*********************************************************************************************
- *
* Note, if you are doing a save state for this game:
*
* On save:
@@ -50,8 +48,6 @@ static int32_t month_keys[12] = { 1, 4, 4, 0, 2, 5, 0, 3, 6, 1, 4, 6 };
*
* restore the rtc data structure
* rtc.system_timestamp = time (NULL);
- *
- *
*********************************************************************************************/
@@ -68,13 +64,11 @@ void S9xHardResetSRTC(void)
rtc.mode = MODE_READ;
rtc.count_enable = false;
rtc.needs_init = true;
-
- // Get system timestamp
- rtc.system_timestamp = time(NULL);
+ rtc.system_timestamp = time(NULL); // Get system timestamp
}
/**********************************************************************************************/
-/* S9xSRTCComputeDayOfWeek(void) */
+/* S9xSRTCComputeDayOfWeek(void) */
/* Return 0-6 for Sunday-Saturday */
/**********************************************************************************************/
uint32_t S9xSRTCComputeDayOfWeek(void)
@@ -83,10 +77,9 @@ uint32_t S9xSRTCComputeDayOfWeek(void)
uint32_t month = rtc.data[8];
uint32_t day = rtc.data[7] * 10 + rtc.data[6];
uint32_t day_of_week;
-
year += (rtc.data[11] - 9) * 100;
- // Range check the month for valid array indicies
+ // Range check the month for valid array indices
if (month > 12)
month = 1;
@@ -102,70 +95,48 @@ uint32_t S9xSRTCComputeDayOfWeek(void)
/**********************************************************************************************/
-/* S9xSRTCDaysInMonth(void) */
+/* S9xSRTCDaysInMonth() */
/* Return the number of days in a specific month for a certain year */
/**********************************************************************************************/
int32_t S9xSRTCDaysInMmonth(int32_t month, int32_t year)
{
- int32_t mdays;
-
- switch (month)
+ switch(month)
{
- case 2:
- if ((year % 4 == 0)) // DKJM2 only uses 199x - 22xx
- mdays = 29;
- else
- mdays = 28;
- break;
-
- case 4:
- case 6:
- case 9:
- case 11:
- mdays = 30;
- break;
-
- default: // months 1,3,5,7,8,10,12
- mdays = 31;
- break;
+ case 2:
+ if((year % 4 == 0)) /* DKJM2 only uses 199x - 22xx */
+ return 29;
+ return 28;
+ case 4:
+ case 6:
+ case 9:
+ case 11:
+ return 30;
+ default:
+ return 31;
}
-
- return mdays;
}
-
-#define DAYTICKS (60*60*24)
-#define HOURTICKS (60*60)
-#define MINUTETICKS 60
+#define MINUTETICKS 60
+#define HOURTICKS (60 * MINUTETICKS)
+#define DAYTICKS (24 * HOURTICKS)
/**********************************************************************************************/
-/* S9xUpdateSrtcTime(void) */
+/* S9xUpdateSrtcTime() */
/* Advance the S-RTC time if counting is enabled */
/**********************************************************************************************/
-void S9xUpdateSrtcTime(void)
+void S9xUpdateSrtcTime(void)
{
time_t cur_systime;
int32_t time_diff;
// Keep track of game time by computing the number of seconds that pass on the system
// clock and adding the same number of seconds to the S-RTC clock structure.
- // I originally tried using mktime and localtime library functions to keep track
- // of time but some of the GNU time functions fail when the year goes to 2099
- // (and maybe less) and this would have caused a bug with DKJM2 so I'm doing
- // it this way to get around that problem.
-
// Note: Dai Kaijyu Monogatari II only allows dates in the range 1996-21xx.
if (rtc.count_enable && !rtc.needs_init)
{
cur_systime = time(NULL);
-
- // This method assumes one time_t clock tick is one second
- // which should work on PCs and GNU systems.
- // If your tick interval is different adjust the
- // DAYTICK, HOURTICK, and MINUTETICK defines
-
time_diff = (int32_t)(cur_systime - rtc.system_timestamp);
rtc.system_timestamp = cur_systime;
@@ -178,7 +149,6 @@ void S9xUpdateSrtcTime(void)
int32_t month;
int32_t year;
int32_t temp_days;
-
int32_t year_hundreds;
int32_t year_tens;
int32_t year_ones;
@@ -238,7 +208,6 @@ void S9xUpdateSrtcTime(void)
{
year = rtc.data[10] * 10 + rtc.data[9];
year += (1000 + rtc.data[11] * 100);
-
month = rtc.data[8];
days += (rtc.data[7] * 10 + rtc.data[6]);
while (days > (temp_days = S9xSRTCDaysInMmonth(month, year)))
@@ -256,7 +225,6 @@ void S9xUpdateSrtcTime(void)
year_ones = year_tens % 10;
year_tens /= 10;
year_hundreds = (year - 1000) / 100;
-
rtc.data[6] = days % 10;
rtc.data[7] = days / 10;
rtc.data[8] = month;
@@ -272,40 +240,31 @@ void S9xUpdateSrtcTime(void)
rtc.data[3] = minutes / 10;
rtc.data[4] = hours % 10;
rtc.data[5] = hours / 10;
-
return;
}
}
}
-
/**********************************************************************************************/
-/* S9xSetSRTC(void) */
+/* S9xSetSRTC(void) */
/* This function sends data to the S-RTC used in Dai Kaijyu Monogatari II */
/**********************************************************************************************/
void S9xSetSRTC(uint8_t data, uint16_t Address)
{
data &= 0x0F; // Data is only 4-bits, mask out unused bits.
- if (data >= 0xD)
+ if (data >= 0xD) // It's an RTC command
{
- // It's an RTC command
-
switch (data)
{
case 0xD:
rtc.mode = MODE_READ;
rtc.index = -1;
break;
-
case 0xE:
rtc.mode = MODE_COMMAND;
break;
-
default:
- // Ignore the write if it's an 0xF ???
- // Probably should switch back to read mode -- but this
- // sequence never occurs in DKJM2
break;
}
@@ -318,14 +277,10 @@ void S9xSetSRTC(uint8_t data, uint16_t Address)
{
rtc.data[rtc.index++] = data;
- if (rtc.index == MAX_RTC_INDEX)
+ if (rtc.index == MAX_RTC_INDEX) // We have all the data for the RTC load
{
- // We have all the data for the RTC load
-
- rtc.system_timestamp = time(NULL); // Get local system time
-
- // Get the day of the week
- rtc.data[rtc.index++] = S9xSRTCComputeDayOfWeek();
+ rtc.system_timestamp = time(NULL); // Get local system time
+ rtc.data[rtc.index++] = S9xSRTCComputeDayOfWeek(); // Get the day of the week
// Start RTC counting again
rtc.count_enable = true;
@@ -334,55 +289,32 @@ void S9xSetSRTC(uint8_t data, uint16_t Address)
return;
}
- else
- {
- // Attempting to write too much data
- // error(); // ignore??
- }
}
else if (rtc.mode == MODE_COMMAND)
{
switch (data)
{
case COMMAND_CLEAR_RTC:
- // Disable RTC counter
- rtc.count_enable = false;
-
+ rtc.count_enable = false; // Disable RTC counter
memset(rtc.data, 0, MAX_RTC_INDEX + 1);
rtc.index = -1;
rtc.mode = MODE_COMMAND_DONE;
break;
case COMMAND_LOAD_RTC:
- // Disable RTC counter
- rtc.count_enable = false;
-
+ rtc.count_enable = false; // Disable RTC counter
rtc.index = 0; // Setup for writing
rtc.mode = MODE_LOAD_RTC;
break;
default:
- rtc.mode = MODE_COMMAND_DONE;
- // unrecognized command - need to implement.
+ rtc.mode = MODE_COMMAND_DONE; // unrecognized command - need to implement.
}
return;
}
- else
- {
- if (rtc.mode == MODE_READ)
- {
- // Attempting to write while in read mode. Ignore.
- }
-
- if (rtc.mode == MODE_COMMAND_DONE)
- {
- // Maybe this isn't an error. Maybe we should kick off
- // a new E command. But is this valid?
- }
- }
}
/**********************************************************************************************/
-/* S9xGetSRTC(void) */
+/* S9xGetSRTC() */
/* This function retrieves data from the S-RTC */
/**********************************************************************************************/
uint8_t S9xGetSRTC(uint16_t Address)
@@ -391,20 +323,17 @@ uint8_t S9xGetSRTC(uint16_t Address)
{
if (rtc.index < 0)
{
- S9xUpdateSrtcTime(); // Only update it if the game reads it
+ S9xUpdateSrtcTime(); // Only update it if the game reads it
rtc.index++;
- return (0x0f); // Send start marker.
+ return 0x0f; // Send start marker.
}
else if (rtc.index > MAX_RTC_INDEX)
{
- rtc.index = -1; // Setup for next set of reads
- return (0x0f); // Data done marker.
+ rtc.index = -1; // Setup for next set of reads
+ return 0x0f; // Data done marker.
}
else
- {
- // Feed out the data
- return rtc.data[rtc.index++];
- }
+ return rtc.data[rtc.index++]; // Feed out the data
}
else
return 0x0;
@@ -415,8 +344,8 @@ void S9xSRTCPreSaveState()
if (Settings.SRTC)
{
S9xUpdateSrtcTime();
-
int32_t s = Memory.SRAMSize ? (1 << (Memory.SRAMSize + 3)) * 128 : 0;
+
if (s > 0x20000)
s = 0x20000;
diff --git a/source/srtc.h b/source/srtc.h
index 98fe369..2bd3490 100644
--- a/source/srtc.h
+++ b/source/srtc.h
@@ -36,11 +36,11 @@ Index Description Range (nibble)
typedef struct
{
bool needs_init;
- bool count_enable; // Does RTC mark time or is it frozen
+ bool count_enable; // Does RTC mark time or is it frozen
uint8_t data [MAX_RTC_INDEX + 1];
int8_t index;
uint8_t mode;
- time_t system_timestamp; // Of latest RTC load time
+ time_t system_timestamp; // Of latest RTC load time
} SRTC_DATA;
extern SRTC_DATA rtc;
@@ -52,7 +52,4 @@ void S9xSRTCPreSaveState(void);
void S9xSRTCPostLoadState(void);
void S9xResetSRTC(void);
void S9xHardResetSRTC(void);
-
-#define SRTC_SRAM_PAD (4 + 8 + 1 + MAX_RTC_INDEX)
-
#endif // _srtc_h
diff --git a/source/tile.c b/source/tile.c
index 7b78c27..e203ee1 100644
--- a/source/tile.c
+++ b/source/tile.c
@@ -27,42 +27,42 @@ static uint8_t ConvertTile(uint8_t* pCache, uint32_t TileAddr)
for (line = 8; line != 0; line--, tp += 2)
{
p1 = p2 = 0;
- if ((pix = *(tp + 0)))
+ if((pix = tp[0]))
{
p1 |= odd_high[0][pix >> 4];
p2 |= odd_low[0][pix & 0xf];
}
- if ((pix = *(tp + 1)))
+ if((pix = tp[1]))
{
p1 |= even_high[0][pix >> 4];
p2 |= even_low[0][pix & 0xf];
}
- if ((pix = *(tp + 16)))
+ if((pix = tp[16]))
{
p1 |= odd_high[1][pix >> 4];
p2 |= odd_low[1][pix & 0xf];
}
- if ((pix = *(tp + 17)))
+ if((pix = tp[17]))
{
p1 |= even_high[1][pix >> 4];
p2 |= even_low[1][pix & 0xf];
}
- if ((pix = *(tp + 32)))
+ if((pix = tp[32]))
{
p1 |= odd_high[2][pix >> 4];
p2 |= odd_low[2][pix & 0xf];
}
- if ((pix = *(tp + 33)))
+ if((pix = tp[33]))
{
p1 |= even_high[2][pix >> 4];
p2 |= even_low[2][pix & 0xf];
}
- if ((pix = *(tp + 48)))
+ if((pix = tp[48]))
{
p1 |= odd_high[3][pix >> 4];
p2 |= odd_low[3][pix & 0xf];
}
- if ((pix = *(tp + 49)))
+ if((pix = tp[49]))
{
p1 |= even_high[3][pix >> 4];
p2 |= even_low[3][pix & 0xf];
@@ -72,27 +72,26 @@ static uint8_t ConvertTile(uint8_t* pCache, uint32_t TileAddr)
non_zero |= p1 | p2;
}
break;
-
case 4:
for (line = 8; line != 0; line--, tp += 2)
{
p1 = p2 = 0;
- if ((pix = *(tp + 0)))
+ if((pix = tp[0]))
{
p1 |= odd_high[0][pix >> 4];
p2 |= odd_low[0][pix & 0xf];
}
- if ((pix = *(tp + 1)))
+ if((pix = tp[1]))
{
p1 |= even_high[0][pix >> 4];
p2 |= even_low[0][pix & 0xf];
}
- if ((pix = *(tp + 16)))
+ if((pix = tp[16]))
{
p1 |= odd_high[1][pix >> 4];
p2 |= odd_low[1][pix & 0xf];
}
- if ((pix = *(tp + 17)))
+ if((pix = tp[17]))
{
p1 |= even_high[1][pix >> 4];
p2 |= even_low[1][pix & 0xf];
@@ -102,17 +101,16 @@ static uint8_t ConvertTile(uint8_t* pCache, uint32_t TileAddr)
non_zero |= p1 | p2;
}
break;
-
case 2:
for (line = 8; line != 0; line--, tp += 2)
{
p1 = p2 = 0;
- if ((pix = *(tp + 0)))
+ if((pix = tp[0]))
{
p1 |= odd_high[0][pix >> 4];
p2 |= odd_low[0][pix & 0xf];
}
- if ((pix = *(tp + 1)))
+ if((pix = tp[1]))
{
p1 |= even_high[0][pix >> 4];
p2 |= even_low[0][pix & 0xf];
@@ -123,7 +121,7 @@ static uint8_t ConvertTile(uint8_t* pCache, uint32_t TileAddr)
}
break;
}
- return (non_zero ? true : BLANK_TILE);
+ return non_zero ? 1 : BLANK_TILE;
}
#define PLOT_PIXEL(screen, pixel) (pixel)
@@ -234,10 +232,8 @@ static void WRITE_4PIXELS16x2x2(int32_t Offset, uint8_t* Pixels, uint16_t* Scree
{
if (GFX.Z1 > Depth [N * 2] && (Pixel = Pixels[N]))
{
- Screen [N * 2] = Screen [N * 2 + 1] = Screen [(GFX.RealPitch >> 1) + N * 2] =
- Screen [(GFX.RealPitch >> 1) + N * 2 + 1] = ScreenColors [Pixel];
- Depth [N * 2] = Depth [N * 2 + 1] = Depth [(GFX.RealPitch >> 1) + N * 2] =
- Depth [(GFX.RealPitch >> 1) + N * 2 + 1] = GFX.Z2;
+ Screen [N * 2] = Screen [N * 2 + 1] = Screen [(GFX.RealPitch >> 1) + N * 2] = Screen [(GFX.RealPitch >> 1) + N * 2 + 1] = ScreenColors [Pixel];
+ Depth [N * 2] = Depth [N * 2 + 1] = Depth [(GFX.RealPitch >> 1) + N * 2] = Depth [(GFX.RealPitch >> 1) + N * 2 + 1] = GFX.Z2;
}
}
}
@@ -252,113 +248,93 @@ static void WRITE_4PIXELS16_FLIPPEDx2x2(int32_t Offset, uint8_t* Pixels, uint16_
{
if (GFX.Z1 > Depth [N * 2] && (Pixel = Pixels[3 - N]))
{
- Screen [N * 2] = Screen [N * 2 + 1] = Screen [(GFX.RealPitch >> 1) + N * 2] =
- Screen [(GFX.RealPitch >> 1) + N * 2 + 1] = ScreenColors [Pixel];
- Depth [N * 2] = Depth [N * 2 + 1] = Depth [(GFX.RealPitch >> 1) + N * 2] =
- Depth [(GFX.RealPitch >> 1) + N * 2 + 1] = GFX.Z2;
+ Screen [N * 2] = Screen [N * 2 + 1] = Screen [(GFX.RealPitch >> 1) + N * 2] = Screen [(GFX.RealPitch >> 1) + N * 2 + 1] = ScreenColors [Pixel];
+ Depth [N * 2] = Depth [N * 2 + 1] = Depth [(GFX.RealPitch >> 1) + N * 2] = Depth [(GFX.RealPitch >> 1) + N * 2 + 1] = GFX.Z2;
}
}
}
void DrawTile16(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16, WRITE_4PIXELS16_FLIPPED, 4)
+ RENDER_TILE(WRITE_4PIXELS16, WRITE_4PIXELS16_FLIPPED, 4);
}
void DrawClippedTile16(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16, WRITE_4PIXELS16_FLIPPED, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16, WRITE_4PIXELS16_FLIPPED, 4);
}
void DrawTile16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16_HALFWIDTH, WRITE_4PIXELS16_FLIPPED_HALFWIDTH, 2)
+ RENDER_TILE(WRITE_4PIXELS16_HALFWIDTH, WRITE_4PIXELS16_FLIPPED_HALFWIDTH, 2);
}
void DrawClippedTile16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_HALFWIDTH, WRITE_4PIXELS16_FLIPPED_HALFWIDTH, 2)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_HALFWIDTH, WRITE_4PIXELS16_FLIPPED_HALFWIDTH, 2);
}
void DrawTile16x2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16x2, WRITE_4PIXELS16_FLIPPEDx2, 8)
+ RENDER_TILE(WRITE_4PIXELS16x2, WRITE_4PIXELS16_FLIPPEDx2, 8);
}
void DrawClippedTile16x2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16x2, WRITE_4PIXELS16_FLIPPEDx2, 8)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16x2, WRITE_4PIXELS16_FLIPPEDx2, 8);
}
-void DrawTile16x2x2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount)
+void DrawTile16x2x2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16x2x2, WRITE_4PIXELS16_FLIPPEDx2x2, 8)
+ RENDER_TILE(WRITE_4PIXELS16x2x2, WRITE_4PIXELS16_FLIPPEDx2x2, 8);
}
-void DrawClippedTile16x2x2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount)
+void DrawClippedTile16x2x2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16x2x2, WRITE_4PIXELS16_FLIPPEDx2x2, 8)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16x2x2, WRITE_4PIXELS16_FLIPPEDx2x2, 8);
}
-void DrawLargePixel16(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Pixels,
- uint32_t StartLine, uint32_t LineCount)
+void DrawLargePixel16(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
-
+ TILE_PREAMBLE();
uint16_t* sp = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.DB + Offset;
uint16_t pixel;
-
- RENDER_TILE_LARGE(ScreenColors [pixel], PLOT_PIXEL)
+ RENDER_TILE_LARGE(ScreenColors [pixel], PLOT_PIXEL);
}
void DrawLargePixel16HalfWidth(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
-
+ TILE_PREAMBLE();
uint16_t* sp = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.DB + Offset;
uint16_t pixel;
-
- RENDER_TILE_LARGE_HALFWIDTH(ScreenColors [pixel], PLOT_PIXEL)
+ RENDER_TILE_LARGE_HALFWIDTH(ScreenColors [pixel], PLOT_PIXEL);
}
static void WRITE_4PIXELS16_ADD(int32_t Offset, uint8_t* Pixels, uint16_t* ScreenColors)
{
uint8_t Pixel, N;
-
uint16_t* Screen = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.ZBuffer + Offset;
uint8_t* SubDepth = GFX.SubZBuffer + Offset;
@@ -387,7 +363,6 @@ static void WRITE_4PIXELS16_ADD(int32_t Offset, uint8_t* Pixels, uint16_t* Scree
static void WRITE_4PIXELS16_FLIPPED_ADD(int32_t Offset, uint8_t* Pixels, uint16_t* ScreenColors)
{
uint8_t Pixel, N;
-
uint16_t* Screen = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.ZBuffer + Offset;
uint8_t* SubDepth = GFX.SubZBuffer + Offset;
@@ -433,8 +408,7 @@ static void WRITE_4PIXELS16_ADD1_2(int32_t Offset, uint8_t* Pixels, uint16_t* Sc
Screen [N] = COLOR_ADD(ScreenColors [Pixel], GFX.FixedColour);
break;
default:
- Screen [N] = (uint16_t)(COLOR_ADD1_2(ScreenColors [Pixel],
- Screen [GFX.Delta + N]));
+ Screen [N] = (uint16_t)(COLOR_ADD1_2(ScreenColors [Pixel], Screen [GFX.Delta + N]));
break;
}
Depth [N] = GFX.Z2;
@@ -462,8 +436,7 @@ static void WRITE_4PIXELS16_FLIPPED_ADD1_2(int32_t Offset, uint8_t* Pixels, uint
Screen [N] = COLOR_ADD(ScreenColors [Pixel], GFX.FixedColour);
break;
default:
- Screen [N] = (uint16_t)(COLOR_ADD1_2(ScreenColors [Pixel],
- Screen [GFX.Delta + N]));
+ Screen [N] = (uint16_t)(COLOR_ADD1_2(ScreenColors [Pixel], Screen [GFX.Delta + N]));
break;
}
Depth [N] = GFX.Z2;
@@ -575,8 +548,7 @@ static void WRITE_4PIXELS16_FLIPPED_SUB1_2(int32_t Offset, uint8_t* Pixels, uint
Screen [N] = (uint16_t) COLOR_SUB(ScreenColors [Pixel], GFX.FixedColour);
break;
default:
- Screen [N] = (uint16_t) COLOR_SUB1_2(ScreenColors [Pixel],
- Screen [GFX.Delta + N]);
+ Screen [N] = (uint16_t) COLOR_SUB1_2(ScreenColors [Pixel], Screen [GFX.Delta + N]);
break;
}
Depth [N] = GFX.Z2;
@@ -587,7 +559,7 @@ static void WRITE_4PIXELS16_FLIPPED_SUB1_2(int32_t Offset, uint8_t* Pixels, uint
void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
uint8_t Pixel;
uint16_t* Screen = (uint16_t*) GFX.S + Offset;
@@ -598,8 +570,7 @@ void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t L
{
case 0:
bp = pCache + StartLine;
- for (l = LineCount; l != 0;
- l--, bp += 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
+ for (l = LineCount; l != 0; l--, bp += 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
{
uint8_t N;
for (N = 0; N < 8; N++)
@@ -625,8 +596,7 @@ void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t L
break;
case H_FLIP:
bp = pCache + StartLine;
- for (l = LineCount; l != 0;
- l--, bp += 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
+ for (l = LineCount; l != 0; l--, bp += 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
{
uint8_t N;
for (N = 0; N < 8; N++)
@@ -652,8 +622,7 @@ void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t L
break;
case H_FLIP | V_FLIP:
bp = pCache + 56 - StartLine;
- for (l = LineCount; l != 0;
- l--, bp -= 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
+ for (l = LineCount; l != 0; l--, bp -= 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
{
uint8_t N;
for (N = 0; N < 8; N++)
@@ -679,8 +648,7 @@ void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t L
break;
case V_FLIP:
bp = pCache + 56 - StartLine;
- for (l = LineCount; l != 0;
- l--, bp -= 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
+ for (l = LineCount; l != 0; l--, bp -= 8, Screen += GFX.PPL, Depth += GFX.PPL, SubDepth += GFX.PPL)
{
uint8_t N;
for (N = 0; N < 8; N++)
@@ -711,67 +679,55 @@ void DrawTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t L
void DrawClippedTile16Add(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_ADD, WRITE_4PIXELS16_FLIPPED_ADD, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_ADD, WRITE_4PIXELS16_FLIPPED_ADD, 4);
}
void DrawTile16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16_ADD1_2, WRITE_4PIXELS16_FLIPPED_ADD1_2, 4)
+ RENDER_TILE(WRITE_4PIXELS16_ADD1_2, WRITE_4PIXELS16_FLIPPED_ADD1_2, 4);
}
void DrawClippedTile16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_ADD1_2, WRITE_4PIXELS16_FLIPPED_ADD1_2, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_ADD1_2, WRITE_4PIXELS16_FLIPPED_ADD1_2, 4);
}
void DrawTile16Sub(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16_SUB, WRITE_4PIXELS16_FLIPPED_SUB, 4)
+ RENDER_TILE(WRITE_4PIXELS16_SUB, WRITE_4PIXELS16_FLIPPED_SUB, 4);
}
-void DrawClippedTile16Sub(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount)
+void DrawClippedTile16Sub(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_SUB, WRITE_4PIXELS16_FLIPPED_SUB, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_SUB, WRITE_4PIXELS16_FLIPPED_SUB, 4);
}
-void DrawTile16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine,
- uint32_t LineCount)
+void DrawTile16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16_SUB1_2, WRITE_4PIXELS16_FLIPPED_SUB1_2, 4)
+ RENDER_TILE(WRITE_4PIXELS16_SUB1_2, WRITE_4PIXELS16_FLIPPED_SUB1_2, 4);
}
-void DrawClippedTile16Sub1_2(uint32_t Tile, int32_t Offset,
- uint32_t StartPixel, uint32_t Width,
- uint32_t StartLine, uint32_t LineCount)
+void DrawClippedTile16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_SUB1_2, WRITE_4PIXELS16_FLIPPED_SUB1_2, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_SUB1_2, WRITE_4PIXELS16_FLIPPED_SUB1_2, 4);
}
static void WRITE_4PIXELS16_ADDF1_2(int32_t Offset, uint8_t* Pixels, uint16_t* ScreenColors)
@@ -852,78 +808,67 @@ static void WRITE_4PIXELS16_FLIPPED_SUBF1_2(int32_t Offset, uint8_t* Pixels, uin
void DrawTile16FixedAdd1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16_ADDF1_2, WRITE_4PIXELS16_FLIPPED_ADDF1_2, 4)
+ RENDER_TILE(WRITE_4PIXELS16_ADDF1_2, WRITE_4PIXELS16_FLIPPED_ADDF1_2, 4);
}
void DrawClippedTile16FixedAdd1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_ADDF1_2,
- WRITE_4PIXELS16_FLIPPED_ADDF1_2, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_ADDF1_2, WRITE_4PIXELS16_FLIPPED_ADDF1_2, 4);
}
void DrawTile16FixedSub1_2(uint32_t Tile, int32_t Offset, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- RENDER_TILE(WRITE_4PIXELS16_SUBF1_2, WRITE_4PIXELS16_FLIPPED_SUBF1_2, 4)
+ RENDER_TILE(WRITE_4PIXELS16_SUBF1_2, WRITE_4PIXELS16_FLIPPED_SUBF1_2, 4);
}
void DrawClippedTile16FixedSub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Width, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
+ TILE_PREAMBLE();
uint8_t* bp;
-
- TILE_CLIP_PREAMBLE
- RENDER_CLIPPED_TILE(WRITE_4PIXELS16_SUBF1_2,
- WRITE_4PIXELS16_FLIPPED_SUBF1_2, 4)
+ TILE_CLIP_PREAMBLE();
+ RENDER_CLIPPED_TILE(WRITE_4PIXELS16_SUBF1_2, WRITE_4PIXELS16_FLIPPED_SUBF1_2, 4);
}
void DrawLargePixel16Add(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
-
+ TILE_PREAMBLE();
uint16_t* sp = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.ZBuffer + Offset;
uint16_t pixel;
#define LARGE_ADD_PIXEL(s, p) \
(Depth [z + GFX.DepthDelta] ? (Depth [z + GFX.DepthDelta] != 1 ? \
- COLOR_ADD (p, *(s + GFX.Delta)) : \
- COLOR_ADD (p, GFX.FixedColour)) \
- : p)
+ COLOR_ADD (p, *(s + GFX.Delta)) : \
+ COLOR_ADD (p, GFX.FixedColour)) : p)
- RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_ADD_PIXEL)
+ RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_ADD_PIXEL);
}
void DrawLargePixel16Add1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
-
+ TILE_PREAMBLE();
uint16_t* sp = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.ZBuffer + Offset;
uint16_t pixel;
#define LARGE_ADD_PIXEL1_2(s, p) \
((uint16_t) (Depth [z + GFX.DepthDelta] ? (Depth [z + GFX.DepthDelta] != 1 ? \
- COLOR_ADD1_2 (p, *(s + GFX.Delta)) : \
- COLOR_ADD (p, GFX.FixedColour)) \
- : p))
+ COLOR_ADD1_2 (p, *(s + GFX.Delta)) : \
+ COLOR_ADD (p, GFX.FixedColour)) : p))
- RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_ADD_PIXEL1_2)
+ RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_ADD_PIXEL1_2);
}
void DrawLargePixel16Sub(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
-
+ TILE_PREAMBLE();
uint16_t* sp = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.ZBuffer + Offset;
uint16_t pixel;
@@ -931,16 +876,14 @@ void DrawLargePixel16Sub(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uin
#define LARGE_SUB_PIXEL(s, p) \
(Depth [z + GFX.DepthDelta] ? (Depth [z + GFX.DepthDelta] != 1 ? \
COLOR_SUB (p, *(s + GFX.Delta)) : \
- COLOR_SUB (p, GFX.FixedColour)) \
- : p)
+ COLOR_SUB (p, GFX.FixedColour)) : p)
- RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_SUB_PIXEL)
+ RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_SUB_PIXEL);
}
void DrawLargePixel16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel, uint32_t Pixels, uint32_t StartLine, uint32_t LineCount)
{
- TILE_PREAMBLE
-
+ TILE_PREAMBLE();
uint16_t* sp = (uint16_t*) GFX.S + Offset;
uint8_t* Depth = GFX.ZBuffer + Offset;
uint16_t pixel;
@@ -948,8 +891,7 @@ void DrawLargePixel16Sub1_2(uint32_t Tile, int32_t Offset, uint32_t StartPixel,
#define LARGE_SUB_PIXEL1_2(s, p) \
(Depth [z + GFX.DepthDelta] ? (Depth [z + GFX.DepthDelta] != 1 ? \
COLOR_SUB1_2 (p, *(s + GFX.Delta)) : \
- COLOR_SUB (p, GFX.FixedColour)) \
- : p)
+ COLOR_SUB (p, GFX.FixedColour)) : p)
- RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_SUB_PIXEL1_2)
+ RENDER_TILE_LARGE(ScreenColors [pixel], LARGE_SUB_PIXEL1_2);
}
diff --git a/source/tile.h b/source/tile.h
index 2a273f6..12b294d 100644
--- a/source/tile.h
+++ b/source/tile.h
@@ -3,24 +3,18 @@
#ifndef _TILE_H_
#define _TILE_H_
-#define TILE_PREAMBLE \
+#define TILE_PREAMBLE() \
uint8_t *pCache; \
-\
uint32_t TileAddr = BG.TileAddress + ((Tile & 0x3ff) << BG.TileShift); \
if ((Tile & 0x1ff) >= 256) \
TileAddr += BG.NameSelect; \
-\
TileAddr &= 0xffff; \
-\
uint32_t TileNumber; \
pCache = &BG.Buffer[(TileNumber = (TileAddr >> BG.TileShift)) << 6]; \
-\
if (!BG.Buffered [TileNumber]) \
BG.Buffered[TileNumber] = ConvertTile (pCache, TileAddr); \
-\
if (BG.Buffered [TileNumber] == BLANK_TILE) \
return; \
-\
uint32_t l; \
uint16_t *ScreenColors; \
if (BG.DirectColourMode) \
@@ -30,7 +24,7 @@
ScreenColors = DirectColourMaps [(Tile >> 10) & BG.PaletteMask]; \
} \
else \
- ScreenColors = &IPPU.ScreenColors [(((Tile >> 10) & BG.PaletteMask) << BG.PaletteShift) + BG.StartPalette];
+ ScreenColors = &IPPU.ScreenColors [(((Tile >> 10) & BG.PaletteMask) << BG.PaletteShift) + BG.StartPalette]
#define RENDER_TILE(NORMAL, FLIPPED, N) \
switch (Tile & (V_FLIP | H_FLIP)) \
@@ -79,10 +73,9 @@
break; \
}
-#define TILE_CLIP_PREAMBLE \
+#define TILE_CLIP_PREAMBLE() \
uint32_t d1; \
uint32_t d2; \
-\
if (StartPixel < 4) \
{ \
d1 = HeadMask [StartPixel]; \
@@ -91,18 +84,16 @@
} \
else \
d1 = 0; \
-\
if (StartPixel + Width > 4) \
{ \
if (StartPixel > 4) \
d2 = HeadMask [StartPixel - 4]; \
else \
d2 = 0xffffffff; \
-\
d2 &= TailMask [(StartPixel + Width - 4)]; \
} \
else \
- d2 = 0;
+ d2 = 0
#define RENDER_CLIPPED_TILE(NORMAL, FLIPPED, N) \
uint32_t dd; \
@@ -114,8 +105,7 @@
{ \
/* This is perfectly OK, regardless of endianness. The tiles are \
* cached in leftmost-endian order (when not horiz flipped) by \
- * the ConvertTile function. \
- */ \
+ * the ConvertTile function. */ \
if ((dd = (*(uint32_t *) bp) & d1)) \
NORMAL (Offset, (uint8_t *) &dd, ScreenColors); \
if ((dd = (*(uint32_t *) (bp + 4)) & d2)) \
@@ -167,7 +157,7 @@
StartPixel = 7 - StartPixel; \
/* fallthrough for no-flip case - above was a horizontal flip */ \
case 0: \
- if ((pixel = *(pCache + StartLine + StartPixel))) \
+ if((pixel = pCache[StartLine + StartPixel])) \
{ \
pixel = PIXEL; \
for (l = LineCount; l != 0; l--, sp += GFX.PPL, Depth += GFX.PPL) \
@@ -188,7 +178,7 @@
StartPixel = 7 - StartPixel; \
/* fallthrough for V_FLIP-only case - above was a horizontal flip */ \
case V_FLIP: \
- if ((pixel = *(pCache + 56 - StartLine + StartPixel))) \
+ if((pixel = pCache[56 - StartLine + StartPixel])) \
{ \
pixel = PIXEL; \
for (l = LineCount; l != 0; l--, sp += GFX.PPL, Depth += GFX.PPL) \
@@ -216,7 +206,7 @@
StartPixel = 7 - StartPixel; \
/* fallthrough for no-flip case - above was a horizontal flip */ \
case 0: \
- if ((pixel = *(pCache + StartLine + StartPixel))) \
+ if((pixel = pCache[StartLine + StartPixel])) \
{ \
pixel = PIXEL; \
for (l = LineCount; l != 0; l--, sp += GFX.PPL, Depth += GFX.PPL) \
@@ -237,7 +227,7 @@
StartPixel = 7 - StartPixel; \
/* fallthrough for V_FLIP-only case - above was a horizontal flip */ \
case V_FLIP: \
- if ((pixel = *(pCache + 56 - StartLine + StartPixel))) \
+ if((pixel = pCache[56 - StartLine + StartPixel])) \
{ \
pixel = PIXEL; \
for (l = LineCount; l != 0; l--, sp += GFX.PPL, Depth += GFX.PPL) \