mirror of https://github.com/mgba-emu/mgba.git
Remove "finalized" flag in favor of stricter sorting
This commit is contained in:
parent
a2a306bb03
commit
55c6103116
|
@ -355,7 +355,10 @@ static void GBAVideoSoftwareRendererDrawScanline(struct GBAVideoRenderer* render
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(softwareRenderer->spriteLayer, 0, sizeof(softwareRenderer->spriteLayer));
|
int x;
|
||||||
|
for (x = 0; x < VIDEO_HORIZONTAL_PIXELS; ++x) {
|
||||||
|
softwareRenderer->spriteLayer[x] = FLAG_UNWRITTEN;
|
||||||
|
}
|
||||||
|
|
||||||
softwareRenderer->windows[0].endX = VIDEO_HORIZONTAL_PIXELS;
|
softwareRenderer->windows[0].endX = VIDEO_HORIZONTAL_PIXELS;
|
||||||
softwareRenderer->nWindows = 1;
|
softwareRenderer->nWindows = 1;
|
||||||
|
@ -372,7 +375,7 @@ static void GBAVideoSoftwareRendererDrawScanline(struct GBAVideoRenderer* render
|
||||||
}
|
}
|
||||||
|
|
||||||
int w;
|
int w;
|
||||||
int x = 0;
|
x = 0;
|
||||||
for (w = 0; w < softwareRenderer->nWindows; ++w) {
|
for (w = 0; w < softwareRenderer->nWindows; ++w) {
|
||||||
// TOOD: handle objwin on backdrop
|
// TOOD: handle objwin on backdrop
|
||||||
uint32_t backdrop = FLAG_UNWRITTEN | FLAG_PRIORITY | FLAG_IS_BACKGROUND;
|
uint32_t backdrop = FLAG_UNWRITTEN | FLAG_PRIORITY | FLAG_IS_BACKGROUND;
|
||||||
|
@ -392,7 +395,7 @@ static void GBAVideoSoftwareRendererDrawScanline(struct GBAVideoRenderer* render
|
||||||
if (softwareRenderer->target2Bd) {
|
if (softwareRenderer->target2Bd) {
|
||||||
x = 0;
|
x = 0;
|
||||||
for (w = 0; w < softwareRenderer->nWindows; ++w) {
|
for (w = 0; w < softwareRenderer->nWindows; ++w) {
|
||||||
uint32_t backdrop = FLAG_UNWRITTEN | FLAG_PRIORITY | FLAG_IS_BACKGROUND;
|
uint32_t backdrop = FLAG_UNWRITTEN;
|
||||||
if (!softwareRenderer->target1Bd || softwareRenderer->blendEffect == BLEND_NONE || softwareRenderer->blendEffect == BLEND_ALPHA || !softwareRenderer->windows[w].control.blendEnable) {
|
if (!softwareRenderer->target1Bd || softwareRenderer->blendEffect == BLEND_NONE || softwareRenderer->blendEffect == BLEND_ALPHA || !softwareRenderer->windows[w].control.blendEnable) {
|
||||||
backdrop |= softwareRenderer->normalPalette[0];
|
backdrop |= softwareRenderer->normalPalette[0];
|
||||||
} else {
|
} else {
|
||||||
|
@ -401,7 +404,7 @@ static void GBAVideoSoftwareRendererDrawScanline(struct GBAVideoRenderer* render
|
||||||
int end = softwareRenderer->windows[w].endX;
|
int end = softwareRenderer->windows[w].endX;
|
||||||
for (; x < end; ++x) {
|
for (; x < end; ++x) {
|
||||||
uint32_t color = softwareRenderer->row[x];
|
uint32_t color = softwareRenderer->row[x];
|
||||||
if (color & FLAG_TARGET_1 && !(color & FLAG_FINALIZED)) {
|
if (color & FLAG_TARGET_1) {
|
||||||
softwareRenderer->row[x] = _mix(softwareRenderer->bldb, backdrop, softwareRenderer->blda, color);
|
softwareRenderer->row[x] = _mix(softwareRenderer->bldb, backdrop, softwareRenderer->blda, color);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -624,20 +627,14 @@ static void _composite(struct GBAVideoSoftwareRenderer* renderer, uint32_t* pixe
|
||||||
// We stash the priority on the top bits so we can do a one-operator comparison
|
// We stash the priority on the top bits so we can do a one-operator comparison
|
||||||
// The lower the number, the higher the priority, and sprites take precendence over backgrounds
|
// The lower the number, the higher the priority, and sprites take precendence over backgrounds
|
||||||
// We want to do special processing if the color pixel is target 1, however
|
// We want to do special processing if the color pixel is target 1, however
|
||||||
if (current & FLAG_UNWRITTEN) {
|
if (color >= current){
|
||||||
color |= (current & FLAG_OBJWIN);
|
|
||||||
} else if ((color & FLAG_ORDER_MASK) < (current & FLAG_ORDER_MASK)) {
|
|
||||||
if (!(color & FLAG_TARGET_1) || !(current & FLAG_TARGET_2)) {
|
|
||||||
color |= FLAG_FINALIZED;
|
|
||||||
} else {
|
|
||||||
color = _mix(renderer->bldb, current, renderer->blda, color) | FLAG_FINALIZED;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (current & FLAG_TARGET_1 && color & FLAG_TARGET_2) {
|
if (current & FLAG_TARGET_1 && color & FLAG_TARGET_2) {
|
||||||
color = _mix(renderer->blda, current, renderer->bldb, color) | FLAG_FINALIZED;
|
color = _mix(renderer->blda, current, renderer->bldb, color);
|
||||||
} else {
|
} else {
|
||||||
color = current | FLAG_FINALIZED;
|
color = current & 0x00FFFFFF;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
color = (color & ~FLAG_TARGET_2) | (current & FLAG_OBJWIN);
|
||||||
}
|
}
|
||||||
*pixel = color;
|
*pixel = color;
|
||||||
}
|
}
|
||||||
|
@ -645,7 +642,7 @@ static void _composite(struct GBAVideoSoftwareRenderer* renderer, uint32_t* pixe
|
||||||
#define BACKGROUND_DRAW_PIXEL_16 \
|
#define BACKGROUND_DRAW_PIXEL_16 \
|
||||||
pixelData = tileData & 0xF; \
|
pixelData = tileData & 0xF; \
|
||||||
current = *pixel; \
|
current = *pixel; \
|
||||||
if (pixelData && !(current & FLAG_FINALIZED)) { \
|
if (pixelData && IS_WRITABLE(current)) { \
|
||||||
if (!objwinSlowPath) { \
|
if (!objwinSlowPath) { \
|
||||||
_composite(renderer, pixel, palette[pixelData] | flags, current); \
|
_composite(renderer, pixel, palette[pixelData] | flags, current); \
|
||||||
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) { \
|
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) { \
|
||||||
|
@ -658,7 +655,7 @@ static void _composite(struct GBAVideoSoftwareRenderer* renderer, uint32_t* pixe
|
||||||
#define BACKGROUND_DRAW_PIXEL_256 \
|
#define BACKGROUND_DRAW_PIXEL_256 \
|
||||||
pixelData = tileData & 0xFF; \
|
pixelData = tileData & 0xFF; \
|
||||||
current = *pixel; \
|
current = *pixel; \
|
||||||
if (pixelData && !(current & FLAG_FINALIZED)) { \
|
if (pixelData && IS_WRITABLE(current)) { \
|
||||||
if (!objwinSlowPath) { \
|
if (!objwinSlowPath) { \
|
||||||
_composite(renderer, pixel, palette[pixelData] | flags, current); \
|
_composite(renderer, pixel, palette[pixelData] | flags, current); \
|
||||||
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) { \
|
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) { \
|
||||||
|
@ -734,12 +731,9 @@ static void _drawBackgroundMode0(struct GBAVideoSoftwareRenderer* renderer, stru
|
||||||
|
|
||||||
unsigned xBase;
|
unsigned xBase;
|
||||||
|
|
||||||
int flags = (background->priority << OFFSET_PRIORITY) | FLAG_IS_BACKGROUND;
|
int flags = (background->priority << OFFSET_PRIORITY) | (background->index << OFFSET_INDEX) | FLAG_IS_BACKGROUND;
|
||||||
flags |= FLAG_TARGET_1 * (background->target1 && renderer->blendEffect == BLEND_ALPHA);
|
flags |= FLAG_TARGET_1 * (background->target1 && renderer->blendEffect == BLEND_ALPHA);
|
||||||
flags |= FLAG_TARGET_2 * background->target2;
|
flags |= FLAG_TARGET_2 * background->target2;
|
||||||
if (!renderer->anyTarget2) {
|
|
||||||
flags |= FLAG_FINALIZED;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t screenBase;
|
uint32_t screenBase;
|
||||||
uint32_t charBase;
|
uint32_t charBase;
|
||||||
|
@ -1057,12 +1051,9 @@ static void _drawBackgroundMode0(struct GBAVideoSoftwareRenderer* renderer, stru
|
||||||
int32_t localX; \
|
int32_t localX; \
|
||||||
int32_t localY; \
|
int32_t localY; \
|
||||||
\
|
\
|
||||||
int flags = (background->priority << OFFSET_PRIORITY) | FLAG_IS_BACKGROUND; \
|
int flags = (background->priority << OFFSET_PRIORITY) | (background->index << OFFSET_INDEX) | FLAG_IS_BACKGROUND; \
|
||||||
flags |= FLAG_TARGET_1 * (background->target1 && renderer->blendEffect == BLEND_ALPHA); \
|
flags |= FLAG_TARGET_1 * (background->target1 && renderer->blendEffect == BLEND_ALPHA); \
|
||||||
flags |= FLAG_TARGET_2 * background->target2; \
|
flags |= FLAG_TARGET_2 * background->target2; \
|
||||||
if (!renderer->anyTarget2) { \
|
|
||||||
flags |= FLAG_FINALIZED; \
|
|
||||||
} \
|
|
||||||
int variant = background->target1 && renderer->currentWindow.blendEnable && (renderer->blendEffect == BLEND_BRIGHTEN || renderer->blendEffect == BLEND_DARKEN); \
|
int variant = background->target1 && renderer->currentWindow.blendEnable && (renderer->blendEffect == BLEND_BRIGHTEN || renderer->blendEffect == BLEND_DARKEN); \
|
||||||
color_t* palette = renderer->normalPalette; \
|
color_t* palette = renderer->normalPalette; \
|
||||||
if (variant) { \
|
if (variant) { \
|
||||||
|
@ -1110,7 +1101,7 @@ static void _drawBackgroundMode2(struct GBAVideoSoftwareRenderer* renderer, stru
|
||||||
tileData = ((uint8_t*)renderer->d.vram)[charBase + (mapData << 6) + ((localY & 0x700) >> 5) + ((localX & 0x700) >> 8)];
|
tileData = ((uint8_t*)renderer->d.vram)[charBase + (mapData << 6) + ((localY & 0x700) >> 5) + ((localX & 0x700) >> 8)];
|
||||||
|
|
||||||
uint32_t current = *pixel;
|
uint32_t current = *pixel;
|
||||||
if (tileData && !(current & FLAG_FINALIZED)) {
|
if (tileData && IS_WRITABLE(current)) {
|
||||||
if (!objwinSlowPath) {
|
if (!objwinSlowPath) {
|
||||||
_composite(renderer, pixel, palette[tileData] | flags, current);
|
_composite(renderer, pixel, palette[tileData] | flags, current);
|
||||||
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) {
|
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) {
|
||||||
|
@ -1142,7 +1133,7 @@ static void _drawBackgroundMode3(struct GBAVideoSoftwareRenderer* renderer, stru
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
uint32_t current = *pixel;
|
uint32_t current = *pixel;
|
||||||
if (!(current & FLAG_FINALIZED) && (!objwinSlowPath || !(current & FLAG_OBJWIN) != objwinOnly)) {
|
if (!objwinSlowPath || !(current & FLAG_OBJWIN) != objwinOnly) {
|
||||||
if (!variant) {
|
if (!variant) {
|
||||||
_composite(renderer, pixel, color | flags, current);
|
_composite(renderer, pixel, color | flags, current);
|
||||||
} else if (renderer->blendEffect == BLEND_BRIGHTEN) {
|
} else if (renderer->blendEffect == BLEND_BRIGHTEN) {
|
||||||
|
@ -1171,7 +1162,7 @@ static void _drawBackgroundMode4(struct GBAVideoSoftwareRenderer* renderer, stru
|
||||||
color = ((uint8_t*)renderer->d.vram)[offset + (localX >> 8) + (localY >> 8) * VIDEO_HORIZONTAL_PIXELS];
|
color = ((uint8_t*)renderer->d.vram)[offset + (localX >> 8) + (localY >> 8) * VIDEO_HORIZONTAL_PIXELS];
|
||||||
|
|
||||||
uint32_t current = *pixel;
|
uint32_t current = *pixel;
|
||||||
if (color && !(current & FLAG_FINALIZED)) {
|
if (color && IS_WRITABLE(current)) {
|
||||||
if (!objwinSlowPath) {
|
if (!objwinSlowPath) {
|
||||||
_composite(renderer, pixel, palette[color] | flags, current);
|
_composite(renderer, pixel, palette[color] | flags, current);
|
||||||
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) {
|
} else if (objwinForceEnable || !(current & FLAG_OBJWIN) == objwinOnly) {
|
||||||
|
@ -1206,7 +1197,7 @@ static void _drawBackgroundMode5(struct GBAVideoSoftwareRenderer* renderer, stru
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
uint32_t current = *pixel;
|
uint32_t current = *pixel;
|
||||||
if (!(current & FLAG_FINALIZED) && (!objwinSlowPath || !(current & FLAG_OBJWIN) != objwinOnly)) {
|
if (!objwinSlowPath || !(current & FLAG_OBJWIN) != objwinOnly) {
|
||||||
if (!variant) {
|
if (!variant) {
|
||||||
_composite(renderer, pixel, color | flags, current);
|
_composite(renderer, pixel, color | flags, current);
|
||||||
} else if (renderer->blendEffect == BLEND_BRIGHTEN) {
|
} else if (renderer->blendEffect == BLEND_BRIGHTEN) {
|
||||||
|
@ -1287,7 +1278,7 @@ static const int _objSizes[32] = {
|
||||||
#define SPRITE_DRAW_PIXEL_16_NORMAL(localX) \
|
#define SPRITE_DRAW_PIXEL_16_NORMAL(localX) \
|
||||||
unsigned tileData = renderer->d.vram[(yBase + charBase + xBase) >> 1]; \
|
unsigned tileData = renderer->d.vram[(yBase + charBase + xBase) >> 1]; \
|
||||||
tileData = (tileData >> ((localX & 3) << 2)) & 0xF; \
|
tileData = (tileData >> ((localX & 3) << 2)) & 0xF; \
|
||||||
if (tileData && (!(renderer->spriteLayer[outX]) || ((renderer->spriteLayer[outX] & FLAG_ORDER_MASK) > flags))) { \
|
if (tileData && (renderer->spriteLayer[outX] & FLAG_ORDER_MASK) > flags) { \
|
||||||
renderer->spriteLayer[outX] = palette[tileData] | flags; \
|
renderer->spriteLayer[outX] = palette[tileData] | flags; \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1304,7 +1295,7 @@ static const int _objSizes[32] = {
|
||||||
#define SPRITE_DRAW_PIXEL_256_NORMAL(localX) \
|
#define SPRITE_DRAW_PIXEL_256_NORMAL(localX) \
|
||||||
unsigned tileData = renderer->d.vram[(yBase + charBase + xBase) >> 1]; \
|
unsigned tileData = renderer->d.vram[(yBase + charBase + xBase) >> 1]; \
|
||||||
tileData = (tileData >> ((localX & 1) << 3)) & 0xFF; \
|
tileData = (tileData >> ((localX & 1) << 3)) & 0xFF; \
|
||||||
if (tileData && (!(renderer->spriteLayer[outX]) || ((renderer->spriteLayer[outX] & FLAG_ORDER_MASK) > flags))) { \
|
if (tileData && (renderer->spriteLayer[outX] & FLAG_ORDER_MASK) > flags) { \
|
||||||
renderer->spriteLayer[outX] = palette[tileData] | flags; \
|
renderer->spriteLayer[outX] = palette[tileData] | flags; \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1327,9 +1318,8 @@ static int _preprocessSprite(struct GBAVideoSoftwareRenderer* renderer, struct G
|
||||||
int width = _objSizes[sprite->shape * 8 + sprite->size * 2];
|
int width = _objSizes[sprite->shape * 8 + sprite->size * 2];
|
||||||
int start = renderer->start;
|
int start = renderer->start;
|
||||||
int end = renderer->end;
|
int end = renderer->end;
|
||||||
uint32_t flags = (sprite->priority << OFFSET_PRIORITY) | FLAG_FINALIZED;
|
uint32_t flags = sprite->priority << OFFSET_PRIORITY;
|
||||||
flags |= FLAG_TARGET_1 * ((renderer->currentWindow.blendEnable && renderer->target1Obj && renderer->blendEffect == BLEND_ALPHA) || sprite->mode == OBJ_MODE_SEMITRANSPARENT);
|
flags |= FLAG_TARGET_1 * ((renderer->currentWindow.blendEnable && renderer->target1Obj && renderer->blendEffect == BLEND_ALPHA) || sprite->mode == OBJ_MODE_SEMITRANSPARENT);
|
||||||
flags |= FLAG_TARGET_2 *renderer->target2Obj;
|
|
||||||
flags |= FLAG_OBJWIN * (sprite->mode == OBJ_MODE_OBJWIN);
|
flags |= FLAG_OBJWIN * (sprite->mode == OBJ_MODE_OBJWIN);
|
||||||
int x = sprite->x;
|
int x = sprite->x;
|
||||||
unsigned charBase = BASE_TILE + sprite->tile * 0x20;
|
unsigned charBase = BASE_TILE + sprite->tile * 0x20;
|
||||||
|
@ -1399,9 +1389,8 @@ static int _preprocessTransformedSprite(struct GBAVideoSoftwareRenderer* rendere
|
||||||
int totalWidth = width << sprite->doublesize;
|
int totalWidth = width << sprite->doublesize;
|
||||||
int start = renderer->start;
|
int start = renderer->start;
|
||||||
int end = renderer->end;
|
int end = renderer->end;
|
||||||
uint32_t flags = (sprite->priority << OFFSET_PRIORITY) | FLAG_FINALIZED;
|
uint32_t flags = sprite->priority << OFFSET_PRIORITY;
|
||||||
flags |= FLAG_TARGET_1 * ((renderer->currentWindow.blendEnable && renderer->target1Obj && renderer->blendEffect == BLEND_ALPHA) || sprite->mode == OBJ_MODE_SEMITRANSPARENT);
|
flags |= FLAG_TARGET_1 * ((renderer->currentWindow.blendEnable && renderer->target1Obj && renderer->blendEffect == BLEND_ALPHA) || sprite->mode == OBJ_MODE_SEMITRANSPARENT);
|
||||||
flags |= FLAG_TARGET_2 * renderer->target2Obj;
|
|
||||||
flags |= FLAG_OBJWIN * (sprite->mode == OBJ_MODE_OBJWIN);
|
flags |= FLAG_OBJWIN * (sprite->mode == OBJ_MODE_OBJWIN);
|
||||||
int x = sprite->x;
|
int x = sprite->x;
|
||||||
unsigned charBase = BASE_TILE + sprite->tile * 0x20;
|
unsigned charBase = BASE_TILE + sprite->tile * 0x20;
|
||||||
|
@ -1439,11 +1428,12 @@ static int _preprocessTransformedSprite(struct GBAVideoSoftwareRenderer* rendere
|
||||||
static void _postprocessSprite(struct GBAVideoSoftwareRenderer* renderer, unsigned priority) {
|
static void _postprocessSprite(struct GBAVideoSoftwareRenderer* renderer, unsigned priority) {
|
||||||
int x;
|
int x;
|
||||||
uint32_t* pixel = renderer->row;
|
uint32_t* pixel = renderer->row;
|
||||||
|
uint32_t flags = FLAG_TARGET_2 * renderer->target2Obj;
|
||||||
for (x = 0; x < VIDEO_HORIZONTAL_PIXELS; ++x, ++pixel) {
|
for (x = 0; x < VIDEO_HORIZONTAL_PIXELS; ++x, ++pixel) {
|
||||||
uint32_t color = renderer->spriteLayer[x];
|
uint32_t color = renderer->spriteLayer[x] & ~FLAG_OBJWIN;
|
||||||
uint32_t current = *pixel;
|
uint32_t current = *pixel;
|
||||||
if ((color & FLAG_FINALIZED) && (color & FLAG_PRIORITY) >> OFFSET_PRIORITY == priority && !(current & FLAG_FINALIZED)) {
|
if ((color & FLAG_UNWRITTEN) != FLAG_UNWRITTEN && (color & FLAG_PRIORITY) >> OFFSET_PRIORITY == priority) {
|
||||||
_composite(renderer, pixel, color & ~FLAG_FINALIZED, current);
|
_composite(renderer, pixel, color | flags, current);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -52,21 +52,24 @@ enum {
|
||||||
#else
|
#else
|
||||||
GBA_COLOR_WHITE = 0x00F8F8F8,
|
GBA_COLOR_WHITE = 0x00F8F8F8,
|
||||||
#endif
|
#endif
|
||||||
OFFSET_PRIORITY = 29
|
OFFSET_PRIORITY = 30,
|
||||||
|
OFFSET_INDEX = 28,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum PixelFlags {
|
enum PixelFlags {
|
||||||
FLAG_FINALIZED = 0x80000000,
|
FLAG_PRIORITY = 0xC0000000,
|
||||||
FLAG_PRIORITY = 0x60000000,
|
FLAG_INDEX = 0x30000000,
|
||||||
FLAG_IS_BACKGROUND = 0x10000000,
|
FLAG_IS_BACKGROUND = 0x08000000,
|
||||||
FLAG_UNWRITTEN = 0x08000000,
|
FLAG_UNWRITTEN = 0xFC000000,
|
||||||
FLAG_TARGET_1 = 0x04000000,
|
FLAG_TARGET_1 = 0x02000000,
|
||||||
FLAG_TARGET_2 = 0x02000000,
|
FLAG_TARGET_2 = 0x01000000,
|
||||||
FLAG_OBJWIN = 0x01000000,
|
FLAG_OBJWIN = 0x01000000,
|
||||||
|
|
||||||
FLAG_ORDER_MASK = 0xF8000000
|
FLAG_ORDER_MASK = 0xF8000000
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#define IS_WRITABLE(PIXEL) ((PIXEL) & 0xFE000000)
|
||||||
|
|
||||||
union WindowRegion {
|
union WindowRegion {
|
||||||
struct {
|
struct {
|
||||||
uint8_t end;
|
uint8_t end;
|
||||||
|
|
Loading…
Reference in New Issue