diff --git a/Assets/dll/gpgx.wbx.zst b/Assets/dll/gpgx.wbx.zst index b3fe93dfe8..949fbe2822 100644 Binary files a/Assets/dll/gpgx.wbx.zst and b/Assets/dll/gpgx.wbx.zst differ diff --git a/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.ISettable.cs b/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.ISettable.cs index f98e14b184..50913cd510 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.ISettable.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.ISettable.cs @@ -26,6 +26,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx bool ret = GPGXSettings.NeedsReboot(_settings, o); _settings = o; Core.gpgx_set_draw_mask(_settings.GetDrawMask()); + Core.gpgx_set_sprite_limit_enabled(!_settings.NoSpriteLimit); return ret ? PutSettingsDirtyBits.RebootCore : PutSettingsDirtyBits.None; } @@ -182,20 +183,32 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx set => _PadScreen320 = value; } - [DeepEqualsIgnore] [JsonIgnore] private bool _Backdrop; [DisplayName("Use custom backdrop color")] [Description("Filler when layers are off")] - [DefaultValue((bool)false)] + [DefaultValue(false)] public bool Backdrop { get => _Backdrop; set => _Backdrop = value; } + [DeepEqualsIgnore] + [JsonIgnore] + private bool _noSpriteLimit; + + [DisplayName("Remove Per-Line Sprite Limit")] + [Description("Removes the original sprite-per-scanline hardware limit")] + [DefaultValue(false)] + public bool NoSpriteLimit + { + get => _noSpriteLimit; + set => _noSpriteLimit = value; + } + public GPGXSettings() { diff --git a/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.IStatable.cs b/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.IStatable.cs index ac16f4ebf2..5adc9f293f 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.IStatable.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/GPGX.IStatable.cs @@ -21,6 +21,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx Core.gpgx_set_cdd_callback(cd_callback_handle); Core.gpgx_invalidate_pattern_cache(); Core.gpgx_set_draw_mask(_settings.GetDrawMask()); + Core.gpgx_set_sprite_limit_enabled(!_settings.NoSpriteLimit); UpdateVideo(); } diff --git a/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/LibGPGX.cs b/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/LibGPGX.cs index 94e0e0b3f2..2e7f64ecf1 100644 --- a/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/LibGPGX.cs +++ b/src/BizHawk.Emulation.Cores/Consoles/Sega/gpgx64/LibGPGX.cs @@ -350,6 +350,8 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx [BizImport(CallingConvention.Cdecl)] public abstract void gpgx_set_draw_mask(DrawMask mask); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_set_sprite_limit_enabled(bool enabled); [BizImport(CallingConvention.Cdecl)] public abstract void gpgx_write_m68k_bus(uint addr, byte data); diff --git a/waterbox/gpgx/Makefile b/waterbox/gpgx/Makefile index 0e87086ba5..c3bdf6d8ac 100644 --- a/waterbox/gpgx/Makefile +++ b/waterbox/gpgx/Makefile @@ -2,7 +2,7 @@ CCFLAGS := -Icore -Iutil -Icore/m68k -Icore/z80 -Icore/input_hw \ -Icore/cart_hw -Icore/cart_hw/svp -Icore/sound -Icore/ntsc -Icore/cd_hw \ -Wall -Werror=pointer-to-int-cast -Werror=int-to-pointer-cast -Werror=implicit-function-declaration \ -std=c99 -fomit-frame-pointer \ - -DLSB_FIRST -DUSE_32BPP_RENDERING -DINLINE=static\ __inline__ + -DLSB_FIRST -DUSE_32BPP_RENDERING -DINLINE=static\ __inline__ -fcommon LDFLAGS := diff --git a/waterbox/gpgx/cinterface/cinterface.c b/waterbox/gpgx/cinterface/cinterface.c index 63b29217a0..378c4792e8 100644 --- a/waterbox/gpgx/cinterface/cinterface.c +++ b/waterbox/gpgx/cinterface/cinterface.c @@ -635,6 +635,11 @@ GPGX_EX void gpgx_set_draw_mask(int mask) color_update_m5(0x00, *(uint16 *)&cram[border << 1]); } +GPGX_EX void gpgx_set_sprite_limit_enabled(int enabled) +{ + config.no_sprite_limit = !enabled; +} + GPGX_EX void gpgx_invalidate_pattern_cache(void) { vdp_invalidate_full_cache(); diff --git a/waterbox/gpgx/core/vdp_render.c b/waterbox/gpgx/core/vdp_render.c index 161503d606..96ff0f517b 100644 --- a/waterbox/gpgx/core/vdp_render.c +++ b/waterbox/gpgx/core/vdp_render.c @@ -595,7 +595,7 @@ typedef struct uint16 size; } object_info_t; -static object_info_t obj_info[2][20]; +static object_info_t obj_info[2][MAX_SPRITES_PER_LINE]; /* Sprite Counter */ static uint8 object_count[2]; @@ -3212,6 +3212,7 @@ void render_obj_m5(int line) int xpos, width; int pixelcount = 0; int masked = 0; + int max_pixels = MODE5_MAX_SPRITE_PIXELS; uint8 *src, *s, *lb; uint32 temp, v_line; @@ -3276,10 +3277,10 @@ void render_obj_m5(int line) lb = &linebuf[0][0x20 + xpos]; /* Max. number of sprite pixels rendered per line */ - if (pixelcount > max_sprite_pixels) + if (pixelcount > max_pixels) { /* Adjust number of pixels to draw */ - width -= (pixelcount - max_sprite_pixels); + width -= (pixelcount - max_pixels); } /* Number of tiles to draw */ @@ -3298,7 +3299,7 @@ void render_obj_m5(int line) } /* Sprite limit */ - if (pixelcount >= max_sprite_pixels) + if (pixelcount >= max_pixels) { /* Sprite masking is effective on next line if max pixel width is reached */ spr_ovr = (pixelcount >= bitmap.viewport.w); @@ -3321,6 +3322,7 @@ void render_obj_m5_ste(int line) int xpos, width; int pixelcount = 0; int masked = 0; + int max_pixels = MODE5_MAX_SPRITE_PIXELS; uint8 *src, *s, *lb; uint32 temp, v_line; @@ -3388,9 +3390,9 @@ void render_obj_m5_ste(int line) lb = &linebuf[1][0x20 + xpos]; /* Adjust number of pixels to draw for sprite limit */ - if (pixelcount > max_sprite_pixels) + if (pixelcount > max_pixels) { - width -= (pixelcount - max_sprite_pixels); + width -= (pixelcount - max_pixels); } /* Number of tiles to draw */ @@ -3409,7 +3411,7 @@ void render_obj_m5_ste(int line) } /* Sprite limit */ - if (pixelcount >= max_sprite_pixels) + if (pixelcount >= max_pixels) { /* Sprite masking is effective on next line if max pixel width is reached */ spr_ovr = (pixelcount >= bitmap.viewport.w); @@ -3439,6 +3441,7 @@ void render_obj_m5_im2(int line) int pixelcount = 0; int masked = 0; int odd = odd_frame; + int max_pixels = MODE5_MAX_SPRITE_PIXELS; uint8 *src, *s, *lb; uint32 temp, v_line; @@ -3503,9 +3506,9 @@ void render_obj_m5_im2(int line) lb = &linebuf[0][0x20 + xpos]; /* Adjust width for sprite limit */ - if (pixelcount > max_sprite_pixels) + if (pixelcount > max_pixels) { - width -= (pixelcount - max_sprite_pixels); + width -= (pixelcount - max_pixels); } /* Number of tiles to draw */ @@ -3524,7 +3527,7 @@ void render_obj_m5_im2(int line) } /* Sprite Limit */ - if (pixelcount >= max_sprite_pixels) + if (pixelcount >= max_pixels) { /* Sprite masking is effective on next line if max pixel width is reached */ spr_ovr = (pixelcount >= bitmap.viewport.w); @@ -3548,6 +3551,7 @@ void render_obj_m5_im2_ste(int line) int pixelcount = 0; int masked = 0; int odd = odd_frame; + int max_pixels = MODE5_MAX_SPRITE_PIXELS; uint8 *src, *s, *lb; uint32 temp, v_line; @@ -3615,9 +3619,9 @@ void render_obj_m5_im2_ste(int line) lb = &linebuf[1][0x20 + xpos]; /* Adjust width for sprite limit */ - if (pixelcount > max_sprite_pixels) + if (pixelcount > max_pixels) { - width -= (pixelcount - max_sprite_pixels); + width -= (pixelcount - max_pixels); } /* Number of tiles to draw */ @@ -3636,7 +3640,7 @@ void render_obj_m5_im2_ste(int line) } /* Sprite Limit */ - if (pixelcount >= max_sprite_pixels) + if (pixelcount >= max_pixels) { /* Sprite masking is effective on next line if max pixel width is reached */ spr_ovr = (pixelcount >= bitmap.viewport.w); @@ -3717,7 +3721,7 @@ void parse_satb_tms(int line) if ((ypos >= 0) && (ypos < height)) { /* Sprite overflow */ - if (count == 4) + if (count == TMS_MAX_SPRITES_PER_LINE) { /* Flag is set only during active area */ if (line < bitmap.viewport.h) @@ -3813,7 +3817,7 @@ void parse_satb_m4(int line) if ((ypos >= 0) && (ypos < height)) { /* Sprite overflow */ - if (count == 8) + if (count == MODE4_MAX_SPRITES_PER_LINE) { /* Flag is set only during active area */ if ((line >= 0) && (line < bitmap.viewport.h)) @@ -3859,7 +3863,7 @@ void parse_satb_m5(int line) int count = 0; /* max. number of rendered sprites (16 or 20 sprites per line by default) */ - int max = bitmap.viewport.w >> 4; + int max = MODE5_MAX_SPRITES_PER_LINE; /* max. number of parsed sprites (64 or 80 sprites per line by default) */ int total = max_sprite_pixels >> 2; diff --git a/waterbox/gpgx/util/osd.h b/waterbox/gpgx/util/osd.h index 2caccbdd5e..f2c8cdff8f 100644 --- a/waterbox/gpgx/util/osd.h +++ b/waterbox/gpgx/util/osd.h @@ -27,14 +27,20 @@ typedef unsigned char bool; #define M_PI 3.1415926535897932385 #endif -typedef struct +#define MAX_SPRITES_PER_LINE 80 +#define TMS_MAX_SPRITES_PER_LINE (config.no_sprite_limit ? MAX_SPRITES_PER_LINE : 4) +#define MODE4_MAX_SPRITES_PER_LINE (config.no_sprite_limit ? MAX_SPRITES_PER_LINE : 8) +#define MODE5_MAX_SPRITES_PER_LINE (config.no_sprite_limit ? MAX_SPRITES_PER_LINE : (bitmap.viewport.w >> 4)) +#define MODE5_MAX_SPRITE_PIXELS (config.no_sprite_limit ? MAX_SPRITES_PER_LINE * 32 : max_sprite_pixels) + +typedef struct { int8 device; uint8 port; uint8 padtype; } t_input_config; -struct +struct { char version[16]; uint8 hq_fm; @@ -64,6 +70,7 @@ struct uint8 gg_extra; uint8 render; t_input_config input[MAX_INPUTS]; + uint8 no_sprite_limit; } config; extern char GG_ROM[256];