Files
ComSquare/sources/PPU/Background.cpp
Clément Le Bihan feabd56991 style \n
2020-05-26 01:54:43 +02:00

128 lines
3.4 KiB
C++

//
// Created by cbihan on 5/14/20.
//
#include "PPUUtils.hpp"
#include "PPU.hpp"
#include "Background.hpp"
namespace ComSquare::PPU
{
Background::Background(ComSquare::PPU::PPU &_ppu, int backGroundNumber, bool hasPriority):
priority(hasPriority),
bgNumber(backGroundNumber)
{
_cgram = _ppu.cgram;
_vram = _ppu.vram;
_bpp = _ppu.getBPP(backGroundNumber);
_characterSize = _ppu.getCharacterSize(backGroundNumber);
_TileMapStartAddress = _ppu.getTileMapStartAddress(backGroundNumber);
_tileSetAddress = _ppu.getTileSetAddress(backGroundNumber);
_tileMaps = _ppu.getBackgroundSize(backGroundNumber);
_directColor = false;
_highRes = false;
}
void Background::renderBackground(void)
{
uint16_t vramAddress = this->_TileMapStartAddress;
Vector2<int> offset(0, 0);
this->backgroundSize.x = this->_tileMaps.x * this->_characterSize.x * 32;
this->backgroundSize.y = this->_tileMaps.y * this->_characterSize.y * 32;
for (int i = 0; i < 4; i++) {
if (!(i == 1 && this->_tileMaps.x == 1) && !(i > 1 && this->_tileMaps.y == 1)) {
drawBasicTileMap(vramAddress, offset);
}
vramAddress+= 0x800;
offset.x += 32 * this->_characterSize.x;
if (i == 2) {
offset.x = 0;
offset.y += 32 * this->_characterSize.y;
}
}
}
void Background::drawBgTile(uint16_t data, Vector2<int> pos)
{
uint16_t graphicAddress;
union TileMapData tileData;
std::vector<uint16_t> palette;
int index = 0;
uint8_t reference = 0;
uint32_t color = 0;
tileData.raw = data;
graphicAddress = this->_tileSetAddress + (tileData.posX * 16 * this->_bpp * 8) + (tileData.posY * this->_bpp * 8);
for (int i = 0; i < this->_characterSize.y; i++) {
for (int j = 0; j < this->_characterSize.x; j++) {
palette = getPalette(tileData.palette);
reference = getTilePixelReference(graphicAddress, index);
color = getRealColor(palette[reference]);
if (tileData.tilePriority == this->priority)
this->buffer[pos.x][pos.y] = color;
index++;
pos.x++;
if (index == (8 / this->_bpp) - 1) {
index = 0;
graphicAddress++;
}
}
index = 0;
pos.x -= this->_characterSize.x;
pos.y++;
}
}
std::vector<uint16_t> Background::getPalette(int nbPalette)
{
std::vector<uint16_t> palette(0xF);
uint16_t addr = nbPalette * 0x10;
for (int i = 0; i < 0xF; i++) {
palette[i] = this->_cgram->read_internal(addr);
palette[i] += this->_cgram->read_internal(addr + 1) << 8U;
}
return palette;
}
uint8_t Background::getTilePixelReference(uint16_t addr, int nb)
{
uint8_t reference = this->_vram->read_internal(addr);
switch (this->_bpp) {
case 8:
return reference;
case 4:
return (reference & (0xFU << ((1 - nb) * 4U))) >> (1 - nb) * 4U;
case 2:
return (reference & (0x3U << ((3 - nb) * 2U))) >> (3 - nb) * 2U;
default:
break;
}
return 0;
}
void Background::drawBasicTileMap(uint16_t baseAddress, Vector2<int> offset)
{
uint16_t tileMapValue = 0;
Vector2<int> pos(0,0);
uint16_t vramAddress = baseAddress;
while (vramAddress < 0x800 + baseAddress) {
tileMapValue = this->_vram->read_internal(vramAddress);
tileMapValue += this->_vram->read_internal(vramAddress + 1) << 8U;
vramAddress += 2;
drawBgTile(tileMapValue, {(pos.x * this->_characterSize.x) + offset.x, (pos.y * this->_characterSize.y) + offset.y});
if (pos.x % 31 == 0 && pos.x) {
pos.y++;
pos.x = 0;
}
else
pos.x++;
}
}
}