aboutsummaryrefslogtreecommitdiff
path: root/src/GPU2D.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/GPU2D.cpp')
-rw-r--r--src/GPU2D.cpp476
1 files changed, 387 insertions, 89 deletions
diff --git a/src/GPU2D.cpp b/src/GPU2D.cpp
index ff9072d..77b79b4 100644
--- a/src/GPU2D.cpp
+++ b/src/GPU2D.cpp
@@ -32,6 +32,8 @@
// * VRAM/FIFO display modes convert colors the same way
// * 3D engine converts colors differently (18bit = 15bit * 2 + 1, except 0 = 0)
// * 'screen disabled' white is 63,63,63
+// * [Gericom] bit15 is used as bottom green bit for palettes. TODO: check where this applies.
+// tested on the normal BG palette and applies there
//
// oh also, changing DISPCNT bit16-17 midframe doesn't work (ignored? applied for next frame?)
// TODO, eventually: check whether other DISPCNT bits can be changed midframe
@@ -68,6 +70,9 @@
// * if BG0 is selected as 1st target, destination not selected as 2nd target:
// brightness up/down effect is applied if selected. if blending is selected, it doesn't apply.
// * 3D layer pixels with alpha=0 are always transparent.
+//
+// mosaic:
+// * mosaic grid starts at 0,0 regardless of the BG/sprite position
GPU2D::GPU2D(u32 num)
@@ -94,6 +99,10 @@ void GPU2D::Reset()
memset(BGRotC, 0, 2*2);
memset(BGRotD, 0, 2*2);
+ memset(Win0Coords, 0, 4);
+ memset(Win1Coords, 0, 4);
+ memset(WinCnt, 0, 4);
+
BlendCnt = 0;
EVA = 16;
EVB = 0;
@@ -120,7 +129,15 @@ void GPU2D::SetFramebuffer(u32* buf)
u8 GPU2D::Read8(u32 addr)
{
- printf("!! GPU2D READ8 %08X\n", addr);
+ switch (addr & 0x00000FFF)
+ {
+ case 0x048: return WinCnt[0];
+ case 0x049: return WinCnt[1];
+ case 0x04A: return WinCnt[2];
+ case 0x04B: return WinCnt[3];
+ }
+
+ printf("unknown GPU read8 %08X\n", addr);
return 0;
}
@@ -136,10 +153,15 @@ u16 GPU2D::Read16(u32 addr)
case 0x00C: return BGCnt[2];
case 0x00E: return BGCnt[3];
+ case 0x048: return WinCnt[0] | (WinCnt[1] << 8);
+ case 0x04A: return WinCnt[2] | (WinCnt[3] << 8);
+
case 0x050: return BlendCnt;
case 0x064: return CaptureCnt & 0xFFFF;
case 0x066: return CaptureCnt >> 16;
+
+ case 0x06C: return MasterBrightness;
}
printf("unknown GPU read16 %08X\n", addr);
@@ -160,7 +182,40 @@ u32 GPU2D::Read32(u32 addr)
void GPU2D::Write8(u32 addr, u8 val)
{
- printf("!! GPU2D WRITE8 %08X %02X\n", addr, val);
+ switch (addr & 0x00000FFF)
+ {
+ case 0x040: Win0Coords[1] = val; return;
+ case 0x041: Win0Coords[0] = val; return;
+ case 0x042: Win1Coords[1] = val; return;
+ case 0x043: Win1Coords[0] = val; return;
+
+ case 0x044: Win0Coords[3] = val; return;
+ case 0x045: Win0Coords[2] = val; return;
+ case 0x046: Win1Coords[3] = val; return;
+ case 0x047: Win1Coords[2] = val; return;
+
+ case 0x048: WinCnt[0] = val; return;
+ case 0x049: WinCnt[1] = val; return;
+ case 0x04A: WinCnt[2] = val; return;
+ case 0x04B: WinCnt[3] = val; return;
+
+ case 0x050: BlendCnt = (BlendCnt & 0xFF00) | val; return;
+ case 0x051: BlendCnt = (BlendCnt & 0x00FF) | (val << 8); return;
+ case 0x052:
+ EVA = val & 0x1F;
+ if (EVA > 16) EVA = 16;
+ return;
+ case 0x53:
+ EVB = val & 0x1F;
+ if (EVB > 16) EVB = 16;
+ return;
+ case 0x054:
+ EVY = val & 0x1F;
+ if (EVY > 16) EVY = 16;
+ return;
+ }
+
+ printf("unknown GPU write8 %08X %02X\n", addr, val);
}
void GPU2D::Write16(u32 addr, u16 val)
@@ -234,6 +289,33 @@ void GPU2D::Write16(u32 addr, u16 val)
if (GPU::VCount < 192) BGYRefInternal[1] = BGYRef[1];
return;
+ case 0x040:
+ Win0Coords[1] = val & 0xFF;
+ Win0Coords[0] = val >> 8;
+ return;
+ case 0x042:
+ Win1Coords[1] = val & 0xFF;
+ Win1Coords[0] = val >> 8;
+ return;
+
+ case 0x044:
+ Win0Coords[3] = val & 0xFF;
+ Win0Coords[2] = val >> 8;
+ return;
+ case 0x046:
+ Win1Coords[3] = val & 0xFF;
+ Win1Coords[2] = val >> 8;
+ return;
+
+ case 0x048:
+ WinCnt[0] = val & 0xFF;
+ WinCnt[1] = val >> 8;
+ return;
+ case 0x04A:
+ WinCnt[2] = val & 0xFF;
+ WinCnt[3] = val >> 8;
+ return;
+
case 0x050: BlendCnt = val; return;
case 0x052:
EVA = val & 0x1F;
@@ -301,6 +383,25 @@ void GPU2D::DrawScanline(u32 line)
u32 dispmode = DispCnt >> 16;
dispmode &= (Num ? 0x1 : 0x3);
+ // always render regular graphics
+ DrawScanline_Mode1(line, dst);
+
+ // capture
+ if ((Num == 0) && (CaptureCnt & (1<<31)))
+ {
+ u32 capwidth, capheight;
+ switch ((CaptureCnt >> 20) & 0x3)
+ {
+ case 0: capwidth = 128; capheight = 128; break;
+ case 1: capwidth = 256; capheight = 64; break;
+ case 2: capwidth = 256; capheight = 128; break;
+ case 3: capwidth = 256; capheight = 192; break;
+ }
+
+ if (line < capheight)
+ DoCapture(line, capwidth, dst);
+ }
+
switch (dispmode)
{
case 0: // screen off
@@ -310,10 +411,7 @@ void GPU2D::DrawScanline(u32 line)
}
break;
- case 1: // regular display
- {
- DrawScanline_Mode1(line, dst);
- }
+ case 1: // regular display, already taken care of
break;
case 2: // VRAM display
@@ -359,22 +457,6 @@ void GPU2D::DrawScanline(u32 line)
break;
}
- // capture
- if ((!Num) && (CaptureCnt & (1<<31)))
- {
- u32 capwidth, capheight;
- switch ((CaptureCnt >> 20) & 0x3)
- {
- case 0: capwidth = 128; capheight = 128; break;
- case 1: capwidth = 256; capheight = 64; break;
- case 2: capwidth = 256; capheight = 128; break;
- case 3: capwidth = 256; capheight = 192; break;
- }
-
- if (line < capheight)
- DoCapture(line, capwidth, dst);
- }
-
// master brightness
if (dispmode != 0)
{
@@ -480,7 +562,7 @@ void GPU2D::DoCapture(u32 line, u32 width, u32* src)
dstaddr &= 0xFFFF;
srcBaddr &= 0xFFFF;
- switch ((DispCnt >> 29) & 0x3)
+ switch ((CaptureCnt >> 29) & 0x3)
{
case 0: // source A
{
@@ -526,8 +608,8 @@ void GPU2D::DoCapture(u32 line, u32 width, u32* src)
case 2: // sources A+B
case 3:
{
- u32 eva = DispCnt & 0x1F;
- u32 evb = (DispCnt >> 8) & 0x1F;
+ u32 eva = CaptureCnt & 0x1F;
+ u32 evb = (CaptureCnt >> 8) & 0x1F;
// checkme
if (eva > 16) eva = 16;
@@ -558,6 +640,10 @@ void GPU2D::DoCapture(u32 line, u32 width, u32* src)
u32 bD = ((bA * aA * eva) + (bB * aB * evb)) >> 4;
u32 aD = (eva>0 ? aA : 0) | (evb>0 ? aB : 0);
+ if (rD > 0x1F) rD = 0x1F;
+ if (gD > 0x1F) gD = 0x1F;
+ if (bD > 0x1F) bD = 0x1F;
+
dst[dstaddr] = rD | (gD << 5) | (bD << 10) | (aD << 15);
srcBaddr = (srcBaddr + 1) & 0xFFFF;
dstaddr = (dstaddr + 1) & 0xFFFF;
@@ -683,6 +769,59 @@ u16* GPU2D::GetOBJExtPal(u32 pal)
}
+void GPU2D::CheckWindows(u32 line)
+{
+ line &= 0xFF;
+ if (line == Win0Coords[3]) Win0Active = false;
+ else if (line == Win0Coords[2]) Win0Active = true;
+ if (line == Win1Coords[3]) Win1Active = false;
+ else if (line == Win1Coords[2]) Win1Active = true;
+}
+
+void GPU2D::CalculateWindowMask(u32 line, u8* mask)
+{
+ for (u32 i = 0; i < 256; i++)
+ mask[i] = WinCnt[2]; // window outside
+
+ if ((DispCnt & (1<<15)) && (DispCnt & (1<<12)))
+ {
+ // OBJ window
+ u8 objwin[256];
+ memset(objwin, 0, 256);
+ DrawSpritesWindow(line, objwin);
+
+ for (u32 i = 0; i < 256; i++)
+ {
+ if (objwin[i]) mask[i] = WinCnt[3];
+ }
+ }
+
+ if ((DispCnt & (1<<14)) && Win1Active)
+ {
+ // window 1
+ u32 x1 = Win1Coords[0];
+ u32 x2 = Win1Coords[1];
+ if (x2 == 0 && x1 > 0) x2 = 256;
+ if (x1 > x2) x2 = 255; // checkme
+
+ for (u32 i = x1; i < x2; i++)
+ mask[i] = WinCnt[1];
+ }
+
+ if ((DispCnt & (1<<13)) && Win0Active)
+ {
+ // window 0
+ u32 x1 = Win0Coords[0];
+ u32 x2 = Win0Coords[1];
+ if (x2 == 0 && x1 > 0) x2 = 256;
+ if (x1 > x2) x2 = 255; // checkme
+
+ for (u32 i = x1; i < x2; i++)
+ mask[i] = WinCnt[0];
+ }
+}
+
+
template<u32 bgmode>
void GPU2D::DrawScanlineBGMode(u32 line, u32* spritebuf, u32* dst)
{
@@ -736,7 +875,8 @@ void GPU2D::DrawScanlineBGMode(u32 line, u32* spritebuf, u32* dst)
void GPU2D::DrawScanline_Mode1(u32 line, u32* dst)
{
- u32 linebuf[256*2];
+ u32 linebuf[256*2 + 64];
+ u8* windowmask = (u8*)&linebuf[256*2];
u32 backdrop;
if (Num) backdrop = *(u16*)&GPU::Palette[0x400];
@@ -753,6 +893,11 @@ void GPU2D::DrawScanline_Mode1(u32 line, u32* dst)
linebuf[i] = backdrop;
}
+ if (DispCnt & 0xE000)
+ CalculateWindowMask(line, windowmask);
+ else
+ memset(windowmask, 0xFF, 256);
+
// prerender sprites
u32 spritebuf[256];
memset(spritebuf, 0, 256*4);
@@ -781,7 +926,11 @@ void GPU2D::DrawScanline_Mode1(u32 line, u32* dst)
u32 coloreffect, eva, evb;
u32 flag1 = val1 >> 24;
- if ((flag1 & 0x80) && (BlendCnt & ((val2 >> 16) & 0xFF00)))
+ if (!(windowmask[i] & 0x20))
+ {
+ coloreffect = 0;
+ }
+ else if ((flag1 & 0x80) && (BlendCnt & ((val2 >> 16) & 0xFF00)))
{
// sprite blending
@@ -904,10 +1053,10 @@ void GPU2D::DrawPixel(u32* dst, u16 color, u32 flag)
void GPU2D::DrawBG_3D(u32 line, u32* dst)
{
- // TODO: window, as for everything
- // also check if window can prevent blending from happening
+ // TODO: check if window can prevent blending from happening
u32* src = GPU3D::GetLine(line);
+ u8* windowmask = (u8*)&dst[256*2];
u16 xoff = BGXPos[0];
int i = 0;
@@ -929,6 +1078,7 @@ void GPU2D::DrawBG_3D(u32 line, u32* dst)
xoff++;
if ((c >> 24) == 0) continue;
+ if (!(windowmask[i] & 0x01)) continue;
dst[i+256] = dst[i];
dst[i] = c | 0x40000000;
@@ -937,6 +1087,7 @@ void GPU2D::DrawBG_3D(u32 line, u32* dst)
void GPU2D::DrawBG_Text(u32 line, u32* dst, u32 bgnum)
{
+ u8* windowmask = (u8*)&dst[256*2];
u16 bgcnt = BGCnt[bgnum];
u32 tilesetaddr, tilemapaddr;
@@ -1012,12 +1163,15 @@ void GPU2D::DrawBG_Text(u32 line, u32* dst, u32 bgnum)
}
// draw pixel
- u8 color;
- u32 tilexoff = (curtile & 0x0400) ? (7-(xoff&0x7)) : (xoff&0x7);
- color = GPU::ReadVRAM_BG<u8>(pixelsaddr + tilexoff);
+ if (windowmask[i] & (1<<bgnum))
+ {
+ u8 color;
+ u32 tilexoff = (curtile & 0x0400) ? (7-(xoff&0x7)) : (xoff&0x7);
+ color = GPU::ReadVRAM_BG<u8>(pixelsaddr + tilexoff);
- if (color)
- DrawPixel(&dst[i], curpal[color], 0x01000000<<bgnum);
+ if (color)
+ DrawPixel(&dst[i], curpal[color], 0x01000000<<bgnum);
+ }
xoff++;
}
@@ -1049,19 +1203,22 @@ void GPU2D::DrawBG_Text(u32 line, u32* dst, u32 bgnum)
// draw pixel
// TODO: optimize VRAM access
- u8 color;
- u32 tilexoff = (curtile & 0x0400) ? (7-(xoff&0x7)) : (xoff&0x7);
- if (tilexoff & 0x1)
+ if (windowmask[i] & (1<<bgnum))
{
- color = GPU::ReadVRAM_BG<u8>(pixelsaddr + (tilexoff >> 1)) >> 4;
- }
- else
- {
- color = GPU::ReadVRAM_BG<u8>(pixelsaddr + (tilexoff >> 1)) & 0x0F;
- }
+ u8 color;
+ u32 tilexoff = (curtile & 0x0400) ? (7-(xoff&0x7)) : (xoff&0x7);
+ if (tilexoff & 0x1)
+ {
+ color = GPU::ReadVRAM_BG<u8>(pixelsaddr + (tilexoff >> 1)) >> 4;
+ }
+ else
+ {
+ color = GPU::ReadVRAM_BG<u8>(pixelsaddr + (tilexoff >> 1)) & 0x0F;
+ }
- if (color)
- DrawPixel(&dst[i], curpal[color], 0x01000000<<bgnum);
+ if (color)
+ DrawPixel(&dst[i], curpal[color], 0x01000000<<bgnum);
+ }
xoff++;
}
@@ -1070,6 +1227,7 @@ void GPU2D::DrawBG_Text(u32 line, u32* dst, u32 bgnum)
void GPU2D::DrawBG_Affine(u32 line, u32* dst, u32 bgnum)
{
+ u8* windowmask = (u8*)&dst[256*2];
u16 bgcnt = BGCnt[bgnum];
u32 tilesetaddr, tilemapaddr;
@@ -1118,7 +1276,7 @@ void GPU2D::DrawBG_Affine(u32 line, u32* dst, u32 bgnum)
for (int i = 0; i < 256; i++)
{
- if (!((rotX|rotY) & overflowmask))
+ if ((!((rotX|rotY) & overflowmask)) && (windowmask[i] & (1<<bgnum)))
{
curtile = GPU::ReadVRAM_BG<u8>(tilemapaddr + ((((rotY & coordmask) >> 11) << yshift) + ((rotX & coordmask) >> 11)));
@@ -1143,6 +1301,7 @@ void GPU2D::DrawBG_Affine(u32 line, u32* dst, u32 bgnum)
void GPU2D::DrawBG_Extended(u32 line, u32* dst, u32 bgnum)
{
+ u8* windowmask = (u8*)&dst[256*2];
u16 bgcnt = BGCnt[bgnum];
u32 tilesetaddr, tilemapaddr;
@@ -1188,7 +1347,7 @@ void GPU2D::DrawBG_Extended(u32 line, u32* dst, u32 bgnum)
for (int i = 0; i < 256; i++)
{
- if (!((rotX|rotY) & overflowmask))
+ if ((!((rotX|rotY) & overflowmask)) && (windowmask[i] & (1<<bgnum)))
{
u16 color = GPU::ReadVRAM_BG<u16>(tilemapaddr + (((((rotY & coordmask) >> 8) << yshift) + ((rotX & coordmask) >> 8)) << 1));
@@ -1209,7 +1368,7 @@ void GPU2D::DrawBG_Extended(u32 line, u32* dst, u32 bgnum)
for (int i = 0; i < 256; i++)
{
- if (!((rotX|rotY) & overflowmask))
+ if ((!((rotX|rotY) & overflowmask)) && (windowmask[i] & (1<<bgnum)))
{
u8 color = GPU::ReadVRAM_BG<u8>(tilemapaddr + (((rotY & coordmask) >> 8) << yshift) + ((rotX & coordmask) >> 8));
@@ -1248,7 +1407,7 @@ void GPU2D::DrawBG_Extended(u32 line, u32* dst, u32 bgnum)
for (int i = 0; i < 256; i++)
{
- if (!((rotX|rotY) & overflowmask))
+ if ((!((rotX|rotY) & overflowmask)) && (windowmask[i] & (1<<bgnum)))
{
curtile = GPU::ReadVRAM_BG<u16>(tilemapaddr + (((((rotY & coordmask) >> 11) << yshift) + ((rotX & coordmask) >> 11)) << 1));
@@ -1280,9 +1439,11 @@ void GPU2D::DrawBG_Extended(u32 line, u32* dst, u32 bgnum)
void GPU2D::InterleaveSprites(u32* buf, u32 prio, u32* dst)
{
+ u8* windowmask = (u8*)&dst[256*2];
+
for (u32 i = 0; i < 256; i++)
{
- if ((buf[i] & 0xF8000) == prio)
+ if (((buf[i] & 0xF8000) == prio) && (windowmask[i] & 0x10))
{
u32 blendfunc = 0;
DrawPixel(&dst[i], buf[i] & 0x7FFF, buf[i] & 0xFF000000);
@@ -1318,6 +1479,9 @@ void GPU2D::DrawSprites(u32 line, u32* dst)
if ((attrib[2] & 0x0C00) != bgnum)
continue;
+ if (((attrib[0] >> 10) & 0x3) == 2)
+ continue;
+
if (attrib[0] & 0x0100)
{
u32 sizeparam = (attrib[0] >> 14) | ((attrib[1] & 0xC000) >> 12);
@@ -1343,7 +1507,7 @@ void GPU2D::DrawSprites(u32 line, u32* dst)
u32 rotparamgroup = (attrib[1] >> 9) & 0x1F;
- DrawSprite_Rotscale(attrib, &oam[(rotparamgroup*16) + 3], boundwidth, boundheight, width, height, xpos, ypos, dst);
+ DrawSprite_Rotscale<false>(attrib, &oam[(rotparamgroup*16) + 3], boundwidth, boundheight, width, height, xpos, ypos, dst);
}
else
{
@@ -1367,17 +1531,98 @@ void GPU2D::DrawSprites(u32 line, u32* dst)
if (attrib[1] & 0x2000)
ypos = height-1 - ypos;
- DrawSprite_Normal(attrib, width, xpos, ypos, dst);
+ DrawSprite_Normal<false>(attrib, width, xpos, ypos, dst);
+ }
+ }
+ }
+}
+
+void GPU2D::DrawSpritesWindow(u32 line, u8* dst)
+{
+ u16* oam = (u16*)&GPU::OAM[Num ? 0x400 : 0];
+
+ const s32 spritewidth[16] =
+ {
+ 8, 16, 8, 0,
+ 16, 32, 8, 0,
+ 32, 32, 16, 0,
+ 64, 64, 32, 0
+ };
+ const s32 spriteheight[16] =
+ {
+ 8, 8, 16, 0,
+ 16, 8, 32, 0,
+ 32, 16, 32, 0,
+ 64, 32, 64, 0
+ };
+
+ for (int sprnum = 127; sprnum >= 0; sprnum--)
+ {
+ u16* attrib = &oam[sprnum*4];
+
+ if (((attrib[0] >> 10) & 0x3) != 2)
+ continue;
+
+ if (attrib[0] & 0x0100)
+ {
+ u32 sizeparam = (attrib[0] >> 14) | ((attrib[1] & 0xC000) >> 12);
+ s32 width = spritewidth[sizeparam];
+ s32 height = spriteheight[sizeparam];
+ s32 boundwidth = width;
+ s32 boundheight = height;
+
+ if (attrib[0] & 0x0200)
+ {
+ boundwidth <<= 1;
+ boundheight <<= 1;
}
+
+ u32 ypos = attrib[0] & 0xFF;
+ ypos = (line - ypos) & 0xFF;
+ if (ypos >= (u32)boundheight)
+ continue;
+
+ s32 xpos = (s32)(attrib[1] << 23) >> 23;
+ if (xpos <= -boundwidth)
+ continue;
+
+ u32 rotparamgroup = (attrib[1] >> 9) & 0x1F;
+
+ DrawSprite_Rotscale<true>(attrib, &oam[(rotparamgroup*16) + 3], boundwidth, boundheight, width, height, xpos, ypos, (u32*)dst);
+ }
+ else
+ {
+ if (attrib[0] & 0x0200)
+ continue;
+
+ u32 sizeparam = (attrib[0] >> 14) | ((attrib[1] & 0xC000) >> 12);
+ s32 width = spritewidth[sizeparam];
+ s32 height = spriteheight[sizeparam];
+
+ u32 ypos = attrib[0] & 0xFF;
+ ypos = (line - ypos) & 0xFF;
+ if (ypos >= (u32)height)
+ continue;
+
+ s32 xpos = (s32)(attrib[1] << 23) >> 23;
+ if (xpos <= -width)
+ continue;
+
+ // yflip
+ if (attrib[1] & 0x2000)
+ ypos = height-1 - ypos;
+
+ DrawSprite_Normal<true>(attrib, width, xpos, ypos, (u32*)dst);
}
}
}
+template<bool window>
void GPU2D::DrawSprite_Rotscale(u16* attrib, u16* rotparams, u32 boundwidth, u32 boundheight, u32 width, u32 height, s32 xpos, u32 ypos, u32* dst)
{
u32 prio = ((attrib[2] & 0x0C00) << 6) | 0x8000;
u32 tilenum = attrib[2] & 0x03FF;
- u32 spritemode = (attrib[0] >> 10) & 0x3;
+ u32 spritemode = window ? 0 : ((attrib[0] >> 10) & 0x3);
u32 ytilefactor;
@@ -1448,13 +1693,13 @@ void GPU2D::DrawSprite_Rotscale(u16* attrib, u16* rotparams, u32 boundwidth, u32
{
if ((u32)rotX < width && (u32)rotY < height)
{
- u8 color;
-
- // blaaaarg
- color = GPU::ReadVRAM_OBJ<u16>(pixelsaddr + ((rotY >> 8) * ytilefactor) + ((rotX >> 8) << 1));
+ u8 color = GPU::ReadVRAM_OBJ<u16>(pixelsaddr + ((rotY >> 8) * ytilefactor) + ((rotX >> 8) << 1));
if (color & 0x8000)
- dst[xpos] = color | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = color | prio;
+ }
}
rotX += rotA;
@@ -1488,20 +1733,23 @@ void GPU2D::DrawSprite_Rotscale(u16* attrib, u16* rotparams, u32 boundwidth, u32
u32 extpal = (DispCnt & 0x80000000);
u16* pal;
- if (extpal) pal = GetOBJExtPal(attrib[2] >> 12);
- else pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
+ if (!window)
+ {
+ if (extpal) pal = GetOBJExtPal(attrib[2] >> 12);
+ else pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
+ }
for (; xoff < boundwidth;)
{
if ((u32)rotX < width && (u32)rotY < height)
{
- u8 color;
-
- // blaaaarg
- color = GPU::ReadVRAM_OBJ<u8>(pixelsaddr + ((rotY>>11)*ytilefactor) + ((rotY&0x700)>>5) + ((rotX>>11)*64) + ((rotX&0x700)>>8));
+ u8 color = GPU::ReadVRAM_OBJ<u8>(pixelsaddr + ((rotY>>11)*ytilefactor) + ((rotY&0x700)>>5) + ((rotX>>11)*64) + ((rotX&0x700)>>8));
if (color)
- dst[xpos] = pal[color] | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = pal[color] | prio;
+ }
}
rotX += rotA;
@@ -1517,17 +1765,18 @@ void GPU2D::DrawSprite_Rotscale(u16* attrib, u16* rotparams, u32 boundwidth, u32
ytilefactor <<= 5;
u32 pixelsaddr = (Num ? 0x06600000 : 0x06400000) + tilenum;
- u16* pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
- pal += (attrib[2] & 0xF000) >> 8;
+ u16* pal;
+ if (!window)
+ {
+ pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
+ pal += (attrib[2] & 0xF000) >> 8;
+ }
for (; xoff < boundwidth;)
{
if ((u32)rotX < width && (u32)rotY < height)
{
- u8 color;
-
- // blaaaarg
- color = GPU::ReadVRAM_OBJ<u8>(pixelsaddr + ((rotY>>11)*ytilefactor) + ((rotY&0x700)>>6) + ((rotX>>11)*32) + ((rotX&0x700)>>9));
+ u8 color = GPU::ReadVRAM_OBJ<u8>(pixelsaddr + ((rotY>>11)*ytilefactor) + ((rotY&0x700)>>6) + ((rotX>>11)*32) + ((rotX&0x700)>>9));
if (rotX & 0x100)
color >>= 4;
@@ -1535,7 +1784,10 @@ void GPU2D::DrawSprite_Rotscale(u16* attrib, u16* rotparams, u32 boundwidth, u32
color &= 0x0F;
if (color)
- dst[xpos] = pal[color] | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = pal[color] | prio;
+ }
}
rotX += rotA;
@@ -1547,11 +1799,12 @@ void GPU2D::DrawSprite_Rotscale(u16* attrib, u16* rotparams, u32 boundwidth, u32
}
}
+template<bool window>
void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* dst)
{
u32 prio = ((attrib[2] & 0x0C00) << 6) | 0x8000;
u32 tilenum = attrib[2] & 0x03FF;
- u32 spritemode = (attrib[0] >> 10) & 0x3;
+ u32 spritemode = window ? 0 : ((attrib[0] >> 10) & 0x3);
u32 wmask = width - 8; // really ((width - 1) & ~0x7)
@@ -1606,18 +1859,44 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
}
u32 pixelsaddr = (Num ? 0x06600000 : 0x06400000) + tilenum;
- pixelsaddr += (xoff << 1);
- for (; xoff < xend;)
+ if (attrib[1] & 0x1000)
{
- u16 color = GPU::ReadVRAM_OBJ<u16>(pixelsaddr);
- pixelsaddr += 2;
+ pixelsaddr += ((width-1 - xoff) << 1);
- if (color & 0x8000)
- dst[xpos] = color | prio;
+ for (; xoff < xend;)
+ {
+ u16 color = GPU::ReadVRAM_OBJ<u16>(pixelsaddr);
+ pixelsaddr -= 2;
- xoff++;
- xpos++;
+ if (color & 0x8000)
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = color | prio;
+ }
+
+ xoff++;
+ xpos++;
+ }
+ }
+ else
+ {
+ pixelsaddr += (xoff << 1);
+
+ for (; xoff < xend;)
+ {
+ u16 color = GPU::ReadVRAM_OBJ<u16>(pixelsaddr);
+ pixelsaddr += 2;
+
+ if (color & 0x8000)
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = color | prio;
+ }
+
+ xoff++;
+ xpos++;
+ }
}
}
else
@@ -1645,8 +1924,11 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
u32 extpal = (DispCnt & 0x80000000);
u16* pal;
- if (extpal) pal = GetOBJExtPal(attrib[2] >> 12);
- else pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
+ if (!window)
+ {
+ if (extpal) pal = GetOBJExtPal(attrib[2] >> 12);
+ else pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
+ }
if (attrib[1] & 0x1000) // xflip. TODO: do better? oh well for now this works
{
@@ -1659,7 +1941,10 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
pixelsaddr--;
if (color)
- dst[xpos] = pal[color] | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = pal[color] | prio;
+ }
xoff++;
xpos++;
@@ -1677,7 +1962,10 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
pixelsaddr++;
if (color)
- dst[xpos] = pal[color] | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = pal[color] | prio;
+ }
xoff++;
xpos++;
@@ -1692,8 +1980,12 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
u32 pixelsaddr = (Num ? 0x06600000 : 0x06400000) + tilenum;
pixelsaddr += ((ypos & 0x7) << 2);
- u16* pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
- pal += (attrib[2] & 0xF000) >> 8;
+ u16* pal;
+ if (!window)
+ {
+ pal = (u16*)&GPU::Palette[Num ? 0x600 : 0x200];
+ pal += (attrib[2] & 0xF000) >> 8;
+ }
if (attrib[1] & 0x1000) // xflip. TODO: do better? oh well for now this works
{
@@ -1714,7 +2006,10 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
}
if (color)
- dst[xpos] = pal[color] | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = pal[color] | prio;
+ }
xoff++;
xpos++;
@@ -1740,7 +2035,10 @@ void GPU2D::DrawSprite_Normal(u16* attrib, u32 width, s32 xpos, u32 ypos, u32* d
}
if (color)
- dst[xpos] = pal[color] | prio;
+ {
+ if (window) ((u8*)dst)[xpos] = 1;
+ else dst[xpos] = pal[color] | prio;
+ }
xoff++;
xpos++;