Mesen-SX/Core/Ppu.h

230 lines
6.5 KiB
C
Raw Normal View History

#pragma once
#include "stdafx.h"
#include "PpuTypes.h"
2019-02-13 18:44:39 -05:00
class Console;
class InternalRegisters;
2019-02-13 18:44:39 -05:00
2019-02-19 23:37:27 -05:00
struct SpriteInfo
{
int16_t X;
bool HorizontalMirror;
bool VerticalMirror;
uint8_t Priority;
uint8_t TileColumn;
uint8_t TileRow;
uint8_t Palette;
bool UseSecondTable;
uint8_t LargeSprite;
};
class Ppu
{
2019-02-15 21:33:13 -05:00
public:
constexpr static uint32_t SpriteRamSize = 544;
constexpr static uint32_t CgRamSize = 512;
constexpr static uint32_t VideoRamSize = 0x10000;
private:
constexpr static int SpriteLayerIndex = 4;
2019-02-22 22:19:20 -05:00
constexpr static int ColorWindowIndex = 5;
constexpr static const uint8_t _oamSizes[8][2][2] = {
{ { 1, 1 }, { 2, 2 } }, //8x8 + 16x16
{ { 1, 1 }, { 4, 4 } }, //8x8 + 32x32
{ { 1, 1 }, { 8, 8 } }, //8x8 + 64x64
{ { 2, 2 }, { 4, 4 } }, //16x16 + 32x32
{ { 2, 2 }, { 8, 8 } }, //16x16 + 64x64
{ { 4, 4 }, { 8, 8 } }, //32x32 + 64x64
{ { 2, 4 }, { 4, 8 } }, //16x32 + 32x64
{ { 2, 4 }, { 4, 4 } } //16x32 + 32x32
};
2019-02-13 18:44:39 -05:00
shared_ptr<Console> _console;
shared_ptr<InternalRegisters> _regs;
2019-02-13 18:44:39 -05:00
bool _forcedVblank = false;
uint8_t _screenBrightness = 0;
uint16_t _cycle = 0;
uint16_t _scanline = 0;
uint32_t _frameCount = 0;
uint8_t _bgMode = 0;
bool _mode1Bg3Priority = false;
uint8_t _mainScreenLayers = 0;
uint8_t _subScreenLayers = 0;
LayerConfig _layerConfig[4];
WindowConfig _window[2];
WindowMaskLogic _maskLogic[6];
bool _windowMaskMain[5];
bool _windowMaskSub[5];
2019-02-23 15:40:32 -05:00
Mode7Config _mode7;
2019-02-13 18:44:39 -05:00
uint8_t *_vram;
uint16_t _vramAddress;
uint8_t _vramIncrementValue;
uint8_t _vramAddressRemapping;
bool _vramAddrIncrementOnSecondReg;
2019-02-24 09:38:22 -05:00
uint16_t _vramReadBuffer = 0;
2019-02-13 18:44:39 -05:00
uint16_t _cgramAddress;
2019-02-15 21:33:13 -05:00
uint8_t _cgram[Ppu::CgRamSize];
2019-02-13 18:44:39 -05:00
uint16_t *_outputBuffers[2];
uint16_t *_currentBuffer;
2019-02-19 23:35:43 -05:00
2019-02-19 23:37:27 -05:00
SpriteInfo _sprites[32] = {};
uint8_t _spriteCount = 0;
uint8_t _spritePriority[256] = {};
uint8_t _spritePalette[256] = {};
uint16_t _spritePixels[256] = {};
2019-02-21 22:40:08 -05:00
uint16_t _pixelsDrawn = 0;
uint16_t _subPixelsDrawn = 0;
2019-02-19 23:35:43 -05:00
uint8_t _rowPixelFlags[256];
uint16_t _mainScreenBuffer[256];
bool _subScreenFilled[256];
2019-02-21 22:40:08 -05:00
uint16_t _subScreenBuffer[256];
2019-02-20 17:39:14 -05:00
uint16_t _mosaicColor[256] = {};
uint8_t _mosaicSize = 0;
uint8_t _mosaicEnabled = 0;
uint16_t _mosaicStartScanline = 0;
2019-02-17 21:09:33 -05:00
uint8_t _oamMode = 0;
uint16_t _oamBaseAddress = 0;
uint16_t _oamAddressOffset = 0;
2019-02-17 21:09:33 -05:00
uint8_t _oamRam[Ppu::SpriteRamSize];
uint16_t _oamRamAddress = 0;
bool _enableOamPriority = false;
uint16_t _internalOamAddress = 0;
uint8_t _oamWriteBuffer = 0;
bool _timeOver = false;
bool _rangeOver = false;
bool _hiResMode = false;
bool _screenInterlace = false;
bool _objInterlace = false;
bool _overscanMode = false;
2019-02-18 22:27:22 -05:00
bool _directColorMode = false;
2019-02-22 22:19:20 -05:00
ColorWindowMode _colorMathClipMode = ColorWindowMode::Never;
ColorWindowMode _colorMathPreventMode = ColorWindowMode::Never;
bool _colorMathAddSubscreen = false;
uint8_t _colorMathEnabled = 0;
bool _colorMathSubstractMode = false;
bool _colorMathHalveResult = false;
2019-02-19 23:35:43 -05:00
uint16_t _fixedColor = 0;
2019-02-19 17:23:21 -05:00
uint8_t _hvScrollLatchValue = 0;
uint8_t _hScrollLatchValue = 0;
uint16_t _horizontalLocation = 0;
bool _horizontalLocToggle = false;
uint16_t _verticalLocation = 0;
bool _verticalLocationToggle = false;
bool _locationLatched = false;
2019-02-22 22:35:53 -05:00
void EvaluateNextLineSprites();
template<uint8_t priority, bool forMainScreen>
2019-02-22 18:41:43 -05:00
void RenderSprites();
2019-02-24 01:11:26 -05:00
template<bool forMainScreen> void RenderMode0();
template<bool forMainScreen> void RenderMode1();
template<bool forMainScreen> void RenderMode2();
template<bool forMainScreen> void RenderMode3();
template<bool forMainScreen> void RenderMode4();
template<bool forMainScreen> void RenderMode5();
2019-02-24 10:30:19 -05:00
template<bool forMainScreen> void RenderMode6();
2019-02-24 01:11:26 -05:00
template<bool forMainScreen> void RenderMode7();
2019-02-22 18:41:43 -05:00
2019-02-24 01:11:26 -05:00
void RenderScanline();
template<bool forMainScreen>
2019-02-24 01:11:26 -05:00
void RenderBgColor();
2019-02-24 01:11:26 -05:00
template<uint8_t layerIndex, uint8_t bpp, bool processHighPriority, bool forMainScreen, uint16_t basePaletteOffset = 0>
__forceinline void RenderTilemap();
template<uint8_t layerIndex, uint8_t bpp, bool processHighPriority, bool forMainScreen, uint16_t basePaletteOffset, bool hiResMode>
__forceinline void RenderTilemap();
2019-02-23 15:40:32 -05:00
template<uint8_t layerIndex, uint8_t bpp, bool processHighPriority, bool forMainScreen, uint16_t basePaletteOffset, bool hiResMode, bool largeTileWidth, bool largeTileHeight>
2019-02-24 01:11:26 -05:00
__forceinline void RenderTilemap();
2019-02-22 22:35:53 -05:00
template<uint8_t layerIndex, uint8_t bpp, bool processHighPriority, bool forMainScreen, uint16_t basePaletteOffset, bool hiResMode, bool largeTileWidth, bool largeTileHeight, uint8_t activeWindowCount>
2019-02-24 01:11:26 -05:00
__forceinline void RenderTilemap();
2019-02-22 22:35:53 -05:00
template<uint8_t layerIndex, uint8_t bpp, bool processHighPriority, bool forMainScreen, uint16_t basePaletteOffset, bool hiResMode, bool largeTileWidth, bool largeTileHeight, uint8_t activeWindowCount, bool applyMosaic>
2019-02-24 01:11:26 -05:00
__forceinline void RenderTilemap();
template<uint8_t layerIndex, uint8_t bpp, bool processHighPriority, bool forMainScreen, uint16_t basePaletteOffset, bool hiResMode, bool largeTileWidth, bool largeTileHeight, uint8_t activeWindowCount, bool applyMosaic, bool directColorMode>
2019-02-22 22:35:53 -05:00
void RenderTilemap();
template<uint8_t layerIndex, bool largeTileWidth, bool largeTileHeight>
void ProcessOffsetMode(uint8_t x, uint16_t realX, uint16_t realY, uint16_t &hScroll, uint16_t &vScroll, uint16_t &addr);
2019-02-24 01:30:55 -05:00
template<bool forMainScreen>
__forceinline bool IsRenderRequired(uint8_t layerIndex);
template<uint8_t bpp>
__forceinline uint16_t GetTilePixelColor(const uint16_t pixelStart, const uint8_t shift);
2019-02-24 01:11:26 -05:00
template<uint8_t layerIndex, bool forMainScreen, bool processHighPriority>
__forceinline void RenderTilemapMode7();
template<uint8_t layerIndex, bool forMainScreen, bool processHighPriority, bool applyMosaic>
__forceinline void RenderTilemapMode7();
2019-02-23 01:28:41 -05:00
2019-02-24 01:11:26 -05:00
template<uint8_t layerIndex, bool forMainScreen, bool processHighPriority, bool applyMosaic, bool directColorMode>
2019-02-23 15:40:32 -05:00
void RenderTilemapMode7();
template<bool applyMosaic>
2019-02-24 01:11:26 -05:00
__forceinline void DrawMainPixel(uint8_t x, uint16_t color, uint8_t flags);
2019-02-23 15:40:32 -05:00
2019-02-24 01:11:26 -05:00
__forceinline void DrawSubPixel(uint8_t x, uint16_t color);
2019-02-23 15:40:32 -05:00
void ApplyColorMath();
template<bool forMainScreen>
2019-02-22 22:31:20 -05:00
void ApplyBrightness();
void ApplyHiResMode();
template<uint8_t layerIndex>
bool ProcessMaskWindow(uint8_t activeWindowCount, int x);
void ProcessWindowMaskSettings(uint8_t value, uint8_t offset);
2019-02-24 09:38:22 -05:00
void UpdateVramReadBuffer();
void SendFrame();
2019-02-13 18:44:39 -05:00
public:
Ppu(shared_ptr<Console> console);
2019-02-16 01:16:57 -05:00
~Ppu();
2019-02-17 19:54:29 -05:00
uint32_t GetFrameCount();
2019-02-13 18:44:39 -05:00
PpuState GetState();
2019-02-13 18:44:39 -05:00
void Exec();
2019-02-15 21:33:13 -05:00
uint8_t* GetVideoRam();
uint8_t* GetCgRam();
uint8_t* GetSpriteRam();
void LatchLocationValues();
2019-02-13 18:44:39 -05:00
uint8_t Read(uint16_t addr);
void Write(uint32_t addr, uint8_t value);
};