2019-07-14 17:46:57 +01:00
|
|
|
|
// <copyright file="Display.cs" company="Adrian Conlon">
|
|
|
|
|
// Copyright (c) Adrian Conlon. All rights reserved.
|
|
|
|
|
// </copyright>
|
2019-07-28 11:50:25 +01:00
|
|
|
|
namespace EightBit.GameBoy
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 23:43:22 +01:00
|
|
|
|
public sealed class Display<T>
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private readonly Bus bus;
|
|
|
|
|
private readonly Ram oam;
|
|
|
|
|
private readonly Ram vram;
|
2019-07-28 23:43:22 +01:00
|
|
|
|
private readonly AbstractColourPalette<T> colours;
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private readonly ObjectAttribute[] objectAttributes = new ObjectAttribute[40];
|
|
|
|
|
private byte control;
|
|
|
|
|
private byte scanLine = 0;
|
|
|
|
|
|
2019-07-28 23:43:22 +01:00
|
|
|
|
public Display(AbstractColourPalette<T> colours, Bus bus, Ram oam, Ram vram)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
this.colours = colours;
|
|
|
|
|
this.bus = bus;
|
|
|
|
|
this.oam = oam;
|
|
|
|
|
this.vram = vram;
|
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 23:43:22 +01:00
|
|
|
|
public T[] Pixels { get; } = new T[DisplayCharacteristics.PixelCount];
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
public void Render()
|
|
|
|
|
{
|
|
|
|
|
this.scanLine = this.bus.IO.Peek(IoRegisters.LY);
|
2019-07-28 23:43:22 +01:00
|
|
|
|
if (this.scanLine < DisplayCharacteristics.RasterHeight)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
this.control = this.bus.IO.Peek(IoRegisters.LCDC);
|
|
|
|
|
if ((this.control & (byte)LcdcControl.LcdEnable) != 0)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
if ((this.control & (byte)LcdcControl.DisplayBackground) != 0)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
this.RenderBackground();
|
2019-07-14 17:46:57 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
if ((this.control & (byte)LcdcControl.ObjectEnable) != 0)
|
|
|
|
|
{
|
|
|
|
|
this.RenderObjects();
|
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-28 11:50:25 +01:00
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
public void LoadObjectAttributes()
|
|
|
|
|
{
|
|
|
|
|
for (var i = 0; i < 40; ++i)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
this.objectAttributes[i] = new ObjectAttribute(this.oam, (ushort)(4 * i));
|
2019-07-14 17:46:57 +01:00
|
|
|
|
}
|
2019-07-28 11:50:25 +01:00
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private int[] CreatePalette(ushort address)
|
|
|
|
|
{
|
|
|
|
|
var raw = this.bus.IO.Peek(address);
|
|
|
|
|
return new int[4]
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
raw & 0b11,
|
|
|
|
|
(raw & 0b1100) >> 2,
|
|
|
|
|
(raw & 0b110000) >> 4,
|
|
|
|
|
(raw & 0b11000000) >> 6,
|
|
|
|
|
};
|
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private void RenderBackground()
|
|
|
|
|
{
|
|
|
|
|
var palette = this.CreatePalette(IoRegisters.BGP);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var window = (this.control & (byte)LcdcControl.WindowEnable) != 0;
|
|
|
|
|
var bgArea = (this.control & (byte)LcdcControl.BackgroundCodeAreaSelection) != 0 ? 0x1c00 : 0x1800;
|
|
|
|
|
var bgCharacters = (this.control & (byte)LcdcControl.BackgroundCharacterDataSelection) != 0 ? 0 : 0x800;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var wx = this.bus.IO.Peek(IoRegisters.WX);
|
|
|
|
|
var wy = this.bus.IO.Peek(IoRegisters.WY);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var offsetX = window ? wx - 7 : 0;
|
|
|
|
|
var offsetY = window ? wy : 0;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var scrollX = this.bus.IO.Peek(IoRegisters.SCX);
|
|
|
|
|
var scrollY = this.bus.IO.Peek(IoRegisters.SCY);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
this.RenderBackground(bgArea, bgCharacters, offsetX - scrollX, offsetY - scrollY, palette);
|
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private void RenderBackground(int bgArea, int bgCharacters, int offsetX, int offsetY, int[] palette)
|
|
|
|
|
{
|
|
|
|
|
var row = (this.scanLine - offsetY) / 8;
|
2019-07-28 23:43:22 +01:00
|
|
|
|
var address = bgArea + (row * DisplayCharacteristics.BufferCharacterWidth);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 23:43:22 +01:00
|
|
|
|
for (var column = 0; column < DisplayCharacteristics.BufferCharacterWidth; ++column)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var character = this.vram.Peek((ushort)address++);
|
|
|
|
|
var definition = new CharacterDefinition(this.vram, (ushort)(bgCharacters + (16 * character)));
|
|
|
|
|
this.RenderTile(8, (column * 8) + offsetX, (row * 8) + offsetY, false, false, false, palette, definition);
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private void RenderObjects()
|
|
|
|
|
{
|
|
|
|
|
var objBlockHeight = (this.control & (byte)LcdcControl.ObjectBlockCompositionSelection) != 0 ? 16 : 8;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var palettes = new int[2][];
|
|
|
|
|
palettes[0] = this.CreatePalette(IoRegisters.OBP0);
|
|
|
|
|
palettes[1] = this.CreatePalette(IoRegisters.OBP1);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var characterAddressMultiplier = objBlockHeight == 8 ? 16 : 8;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
for (var i = 0; i < 40; ++i)
|
|
|
|
|
{
|
|
|
|
|
var current = this.objectAttributes[i];
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var spriteY = current.PositionY;
|
|
|
|
|
var drawY = spriteY - 16;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
if ((this.scanLine >= drawY) && (this.scanLine < (drawY + objBlockHeight)))
|
|
|
|
|
{
|
|
|
|
|
var spriteX = current.PositionX;
|
|
|
|
|
var drawX = spriteX - 8;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var sprite = current.Pattern;
|
|
|
|
|
var definition = new CharacterDefinition(this.vram, (ushort)(characterAddressMultiplier * sprite));
|
|
|
|
|
var palette = palettes[current.Palette];
|
|
|
|
|
var flipX = current.FlipX;
|
|
|
|
|
var flipY = current.FlipY;
|
|
|
|
|
|
|
|
|
|
this.RenderTile(objBlockHeight, drawX, drawY, flipX, flipY, true, palette, definition);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-28 11:50:25 +01:00
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
private void RenderTile(int height, int drawX, int drawY, bool flipX, bool flipY, bool allowTransparencies, int[] palette, CharacterDefinition definition)
|
|
|
|
|
{
|
|
|
|
|
const int width = 8;
|
|
|
|
|
const int flipMaskX = width - 1;
|
|
|
|
|
var flipMaskY = height - 1;
|
|
|
|
|
|
|
|
|
|
var y = this.scanLine;
|
|
|
|
|
|
|
|
|
|
var cy = y - drawY;
|
|
|
|
|
if (flipY)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
cy = ~cy & flipMaskY;
|
|
|
|
|
}
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var rowDefinition = definition.Get(cy);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
|
2019-07-28 23:43:22 +01:00
|
|
|
|
var lineAddress = y * DisplayCharacteristics.RasterWidth;
|
2019-07-28 11:50:25 +01:00
|
|
|
|
for (var cx = 0; cx < width; ++cx)
|
|
|
|
|
{
|
|
|
|
|
var x = drawX + (flipX ? ~cx & flipMaskX : cx);
|
2019-07-28 23:43:22 +01:00
|
|
|
|
if (x >= DisplayCharacteristics.RasterWidth)
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
break;
|
2019-07-14 17:46:57 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var colour = rowDefinition[cx];
|
|
|
|
|
if (!allowTransparencies || (allowTransparencies && (colour > 0)))
|
2019-07-14 17:46:57 +01:00
|
|
|
|
{
|
2019-07-28 11:50:25 +01:00
|
|
|
|
var outputPixel = lineAddress + x;
|
|
|
|
|
this.Pixels[outputPixel] = this.colours.Colour(palette[colour]);
|
2019-07-14 17:46:57 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|