GPU: Remove some assumptions with how OBJ layers will read custom VRAM, instead assigning the read location on a per-line basis based on the isLineCaptureNative flag. Fixes screen flickering in Kingdom Hearts: 358/2 Days.

This commit is contained in:
rogerman 2017-08-14 16:37:56 -07:00
parent 9a5e52a7fa
commit 7487bf2295
2 changed files with 28 additions and 57 deletions

View File

@ -762,8 +762,7 @@ void GPUEngineBase::_Reset_Base()
this->_needUpdateWINH[0] = true; this->_needUpdateWINH[0] = true;
this->_needUpdateWINH[1] = true; this->_needUpdateWINH[1] = true;
this->vramBlockOBJIndex = VRAM_NO_3D_USAGE; this->vramBlockOBJAddress = 0;
this->vramBlockOBJAddress = this->_sprMem;
this->nativeLineRenderCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT; this->nativeLineRenderCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT;
this->nativeLineOutputCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT; this->nativeLineOutputCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT;
@ -4115,6 +4114,7 @@ void GPUEngineBase::_SpriteRenderPerform(GPUEngineCompositorInfo &compInfo, u16
dst_alpha[sprX] = 0xFF; dst_alpha[sprX] = 0xFF;
typeTab[sprX] = objMode; typeTab[sprX] = objMode;
prioTab[sprX] = prio; prioTab[sprX] = prio;
this->_sprNum[sprX] = i;
} }
} }
} }
@ -4146,13 +4146,13 @@ void GPUEngineBase::_SpriteRenderPerform(GPUEngineCompositorInfo &compInfo, u16
{ {
if (DISPCNT.OBJ_BMP_2D_dim) if (DISPCNT.OBJ_BMP_2D_dim)
//tested by knights in the nightmare //tested by knights in the nightmare
offset = (this->_SpriteAddressBMP(compInfo, spriteInfo, sprSize, auxY)-srcadr)/2+auxX; offset = ((this->_SpriteAddressBMP(compInfo, spriteInfo, sprSize, auxY) - srcadr) / 2) + auxX;
else //tested by lego indiana jones (somehow?) else //tested by lego indiana jones (somehow?)
//tested by buffy sacrifice damage blood splatters in corner //tested by buffy sacrifice damage blood splatters in corner
offset = auxX + (auxY*sprSize.width); offset = auxX + (auxY * sprSize.width);
const u32 finalAddr = srcadr + (offset << 1);
u16* mem = (u16*)MMU_gpu_map(srcadr + (offset<<1)); u16 *mem = (u16 *)MMU_gpu_map(finalAddr);
colour = LE_TO_LOCAL_16(*mem); colour = LE_TO_LOCAL_16(*mem);
if (ISDEBUGRENDER) if (ISDEBUGRENDER)
@ -4170,6 +4170,7 @@ void GPUEngineBase::_SpriteRenderPerform(GPUEngineCompositorInfo &compInfo, u16
dst_alpha[sprX] = spriteInfo.PaletteIndex; dst_alpha[sprX] = spriteInfo.PaletteIndex;
typeTab[sprX] = objMode; typeTab[sprX] = objMode;
prioTab[sprX] = prio; prioTab[sprX] = prio;
this->_sprNum[sprX] = i;
} }
} }
} }
@ -4233,6 +4234,7 @@ void GPUEngineBase::_SpriteRenderPerform(GPUEngineCompositorInfo &compInfo, u16
dst_alpha[sprX] = 0xFF; dst_alpha[sprX] = 0xFF;
typeTab[sprX] = objMode; typeTab[sprX] = objMode;
prioTab[sprX] = prio; prioTab[sprX] = prio;
this->_sprNum[sprX] = i;
} }
} }
} }
@ -4278,6 +4280,20 @@ void GPUEngineBase::_SpriteRenderPerform(GPUEngineCompositorInfo &compInfo, u16
srcadr = this->_SpriteAddressBMP(compInfo, spriteInfo, sprSize, y); srcadr = this->_SpriteAddressBMP(compInfo, spriteInfo, sprSize, y);
this->_RenderSpriteBMP<ISDEBUGRENDER>(compInfo, i, dst, srcadr, dst_alpha, typeTab, prioTab, prio, lg, sprX, x, xdir, spriteInfo.PaletteIndex); this->_RenderSpriteBMP<ISDEBUGRENDER>(compInfo, i, dst, srcadr, dst_alpha, typeTab, prioTab, prio, lg, sprX, x, xdir, spriteInfo.PaletteIndex);
const size_t vramPixel = (size_t)((u8 *)MMU_gpu_map(srcadr) - MMU.ARM9_LCD) / sizeof(u16);
if (vramPixel < (GPU_FRAMEBUFFER_NATIVE_WIDTH * GPU_VRAM_BLOCK_LINES * 4))
{
const size_t blockID = vramPixel / (GPU_FRAMEBUFFER_NATIVE_WIDTH * GPU_VRAM_BLOCK_LINES);
const size_t blockPixel = vramPixel % (GPU_FRAMEBUFFER_NATIVE_WIDTH * GPU_VRAM_BLOCK_LINES);
const size_t blockLine = blockPixel / GPU_FRAMEBUFFER_NATIVE_WIDTH;
const size_t linePixel = blockPixel % GPU_FRAMEBUFFER_NATIVE_WIDTH;
if (!GPU->GetEngineMain()->isLineCaptureNative[blockID][blockLine] && (linePixel == 0))
{
this->vramBlockOBJAddress = srcadr;
}
}
} }
else if (spriteInfo.PaletteMode == PaletteMode_1x256) //256 colors else if (spriteInfo.PaletteMode == PaletteMode_1x256) //256 colors
{ {
@ -4336,6 +4352,7 @@ void GPUEngineBase::_RenderLine_Layers(const size_t l)
// for all the pixels in the line // for all the pixels in the line
if (this->_enableLayer[GPULayerID_OBJ]) if (this->_enableLayer[GPULayerID_OBJ])
{ {
this->vramBlockOBJAddress = 0;
this->_RenderLine_SetupSprites(compInfo); this->_RenderLine_SetupSprites(compInfo);
} }
@ -4477,9 +4494,8 @@ template <GPUCompositorMode COMPOSITORMODE, NDSColorFormat OUTPUTFORMAT, bool WI
void GPUEngineBase::_RenderLine_LayerOBJ(GPUEngineCompositorInfo &compInfo, itemsForPriority_t *__restrict item) void GPUEngineBase::_RenderLine_LayerOBJ(GPUEngineCompositorInfo &compInfo, itemsForPriority_t *__restrict item)
{ {
bool useCustomVRAM = false; bool useCustomVRAM = false;
size_t vramLine = compInfo.line.indexNative;
if (this->vramBlockOBJIndex != VRAM_NO_3D_USAGE) if (this->vramBlockOBJAddress != 0)
{ {
const size_t vramPixel = (size_t)((u8 *)MMU_gpu_map(this->vramBlockOBJAddress) - MMU.ARM9_LCD) / sizeof(u16); const size_t vramPixel = (size_t)((u8 *)MMU_gpu_map(this->vramBlockOBJAddress) - MMU.ARM9_LCD) / sizeof(u16);
@ -4489,9 +4505,8 @@ void GPUEngineBase::_RenderLine_LayerOBJ(GPUEngineCompositorInfo &compInfo, item
const size_t blockPixel = vramPixel % (GPU_FRAMEBUFFER_NATIVE_WIDTH * GPU_VRAM_BLOCK_LINES); const size_t blockPixel = vramPixel % (GPU_FRAMEBUFFER_NATIVE_WIDTH * GPU_VRAM_BLOCK_LINES);
const size_t blockLine = blockPixel / GPU_FRAMEBUFFER_NATIVE_WIDTH; const size_t blockLine = blockPixel / GPU_FRAMEBUFFER_NATIVE_WIDTH;
vramLine += blockLine; GPU->GetEngineMain()->VerifyVRAMLineDidChange(blockID, blockLine);
GPU->GetEngineMain()->VerifyVRAMLineDidChange(blockID, vramLine); useCustomVRAM = !GPU->GetEngineMain()->isLineCaptureNative[blockID][blockLine];
useCustomVRAM = !GPU->GetEngineMain()->isLineCaptureNative[blockID][vramLine];
} }
} }
@ -4504,7 +4519,7 @@ void GPUEngineBase::_RenderLine_LayerOBJ(GPUEngineCompositorInfo &compInfo, item
{ {
if (useCustomVRAM && (OUTPUTFORMAT == NDSColorFormat_BGR888_Rev)) if (useCustomVRAM && (OUTPUTFORMAT == NDSColorFormat_BGR888_Rev))
{ {
const FragmentColor *__restrict vramColorPtr = (FragmentColor *)GPU->GetCustomVRAMAddressUsingMappedAddress<OUTPUTFORMAT>(this->vramBlockOBJAddress, compInfo.line.blockOffsetCustom); const FragmentColor *__restrict vramColorPtr = (FragmentColor *)GPU->GetCustomVRAMAddressUsingMappedAddress<OUTPUTFORMAT>(this->vramBlockOBJAddress, 0);
for (size_t i = 0; i < item->nbPixelsX; i++) for (size_t i = 0; i < item->nbPixelsX; i++)
{ {
@ -4554,7 +4569,7 @@ void GPUEngineBase::_RenderLine_LayerOBJ(GPUEngineCompositorInfo &compInfo, item
if (useCustomVRAM) if (useCustomVRAM)
{ {
const void *__restrict vramColorPtr = GPU->GetCustomVRAMAddressUsingMappedAddress<OUTPUTFORMAT>(this->vramBlockOBJAddress, compInfo.line.blockOffsetCustom); const void *__restrict vramColorPtr = GPU->GetCustomVRAMAddressUsingMappedAddress<OUTPUTFORMAT>(this->vramBlockOBJAddress, 0);
for (size_t line = 0; line < compInfo.line.renderCount; line++) for (size_t line = 0; line < compInfo.line.renderCount; line++)
{ {
@ -5099,36 +5114,6 @@ void GPUEngineBase::_PerformWindowTesting(GPUEngineCompositorInfo &compInfo)
} }
} }
void GPUEngineBase::UpdateVRAM3DUsageProperties_OBJLayer(const size_t bankIndex)
{
const IOREG_DISPCNT &DISPCNT = this->_IORegisterMap->DISPCNT;
if (!this->_enableLayer[GPULayerID_OBJ] || (DISPCNT.OBJ_BMP_mapping != 0) || (DISPCNT.OBJ_BMP_2D_dim == 0))
{
return;
}
const GPUEngineA *mainEngine = GPU->GetEngineMain();
const IOREG_DISPCAPCNT &DISPCAPCNT = mainEngine->GetIORegisterMap().DISPCAPCNT;
for (size_t spriteIndex = 0; spriteIndex < 128; spriteIndex++)
{
const OAMAttributes &spriteInfo = this->_oamList[spriteIndex];
if ( ((spriteInfo.RotScale != 0) || (spriteInfo.Disable == 0)) && (spriteInfo.Mode == OBJMode_Bitmap) && (spriteInfo.PaletteIndex != 0) )
{
const u32 vramAddress = ((spriteInfo.TileIndex & 0x1F) << 5) + ((spriteInfo.TileIndex & ~0x1F) << 7);
const SpriteSize sprSize = GPUEngineBase::_sprSizeTab[spriteInfo.Size][spriteInfo.Shape];
if( (vramAddress == (DISPCAPCNT.VRAMWriteOffset * ADDRESS_STEP_32KB)) && (sprSize.width == 64) && (sprSize.height == 64) )
{
this->vramBlockOBJIndex = bankIndex;
this->vramBlockOBJAddress = this->_sprMem + (((spriteInfo.TileIndex & 0x3E0) * 64 + (spriteInfo.TileIndex & 0x1F) * 8) << 1);
return;
}
}
}
}
template <GPUCompositorMode COMPOSITORMODE, NDSColorFormat OUTPUTFORMAT, bool MOSAIC, bool WILLPERFORMWINDOWTEST, bool ISCUSTOMRENDERINGNEEDED> template <GPUCompositorMode COMPOSITORMODE, NDSColorFormat OUTPUTFORMAT, bool MOSAIC, bool WILLPERFORMWINDOWTEST, bool ISCUSTOMRENDERINGNEEDED>
FORCEINLINE void GPUEngineBase::_RenderLine_LayerBG_Final(GPUEngineCompositorInfo &compInfo) FORCEINLINE void GPUEngineBase::_RenderLine_LayerBG_Final(GPUEngineCompositorInfo &compInfo)
{ {
@ -7670,9 +7655,6 @@ void GPUSubsystem::ResetDisplayCaptureEnable()
void GPUSubsystem::UpdateRenderProperties() void GPUSubsystem::UpdateRenderProperties()
{ {
this->_engineMain->vramBlockOBJIndex = VRAM_NO_3D_USAGE;
this->_engineSub->vramBlockOBJIndex = VRAM_NO_3D_USAGE;
this->_engineMain->nativeLineRenderCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT; this->_engineMain->nativeLineRenderCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT;
this->_engineMain->nativeLineOutputCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT; this->_engineMain->nativeLineOutputCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT;
this->_engineSub->nativeLineRenderCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT; this->_engineSub->nativeLineRenderCount = GPU_FRAMEBUFFER_NATIVE_HEIGHT;
@ -7740,14 +7722,8 @@ void GPUSubsystem::UpdateRenderProperties()
case VramConfiguration::ABG: case VramConfiguration::ABG:
case VramConfiguration::BBG: case VramConfiguration::BBG:
case VramConfiguration::LCDC: case VramConfiguration::LCDC:
break;
case VramConfiguration::AOBJ: case VramConfiguration::AOBJ:
this->_engineMain->UpdateVRAM3DUsageProperties_OBJLayer(i);
break;
case VramConfiguration::BOBJ: case VramConfiguration::BOBJ:
this->_engineSub->UpdateVRAM3DUsageProperties_OBJLayer(i);
break; break;
default: default:

View File

@ -1123,8 +1123,6 @@ typedef struct
bool needApplyMasterBrightness[2]; // Reports if the display still needs to apply the master brightness. bool needApplyMasterBrightness[2]; // Reports if the display still needs to apply the master brightness.
} NDSDisplayInfo; } NDSDisplayInfo;
#define VRAM_NO_3D_USAGE 0xFF
typedef struct typedef struct
{ {
u8 begin; u8 begin;
@ -1480,7 +1478,6 @@ public:
void UpdatePropertiesWithoutRender(const u16 l); void UpdatePropertiesWithoutRender(const u16 l);
void LastLineProcess(); void LastLineProcess();
u8 vramBlockOBJIndex;
u32 vramBlockOBJAddress; u32 vramBlockOBJAddress;
size_t nativeLineRenderCount; size_t nativeLineRenderCount;
@ -1518,8 +1515,6 @@ public:
const BGLayerInfo& GetBGLayerInfoByID(const GPULayerID layerID); const BGLayerInfo& GetBGLayerInfoByID(const GPULayerID layerID);
void UpdateVRAM3DUsageProperties_OBJLayer(const size_t bankIndex);
void SpriteRenderDebug(const u16 lineIndex, u16 *dst); void SpriteRenderDebug(const u16 lineIndex, u16 *dst);
void RenderLayerBG(const GPULayerID layerID, u16 *dstLineColor); void RenderLayerBG(const GPULayerID layerID, u16 *dstLineColor);