mirror of https://github.com/bsnes-emu/bsnes.git
200 lines
3.7 KiB
C++
200 lines
3.7 KiB
C++
//Hudson Soft HuC6270 -- Video Display Controller
|
|
|
|
struct VDC : Thread {
|
|
inline auto bus() const -> uint9 { return data; }
|
|
inline auto irqLine() const -> bool { return irq.line; }
|
|
|
|
static auto Enter() -> void;
|
|
auto main() -> void;
|
|
auto step(uint clocks) -> void;
|
|
auto scanline() -> void;
|
|
auto frame() -> void;
|
|
|
|
auto power() -> void;
|
|
|
|
//io.cpp
|
|
auto read(uint2 addr) -> uint8;
|
|
auto write(uint2 addr, uint8 data) -> void;
|
|
|
|
//serialization.cpp
|
|
auto serialize(serializer&) -> void;
|
|
|
|
private:
|
|
uint9 data;
|
|
|
|
struct VRAM {
|
|
//memory.cpp
|
|
auto read(uint16 addr) -> uint16;
|
|
auto write(uint16 addr, uint16 data) -> void;
|
|
|
|
uint16 data[0x8000];
|
|
|
|
uint16 addressRead;
|
|
uint16 addressWrite;
|
|
uint16 addressIncrement;
|
|
|
|
uint16 dataRead;
|
|
uint16 dataWrite;
|
|
} vram;
|
|
|
|
struct SATB {
|
|
//memory.cpp
|
|
auto read(uint8 addr) -> uint16;
|
|
auto write(uint8 addr, uint16 data) -> void;
|
|
|
|
uint16 data[0x100];
|
|
} satb;
|
|
|
|
struct Timing {
|
|
uint5 horizontalSyncWidth;
|
|
uint7 horizontalDisplayStart;
|
|
uint7 horizontalDisplayLength;
|
|
uint7 horizontalDisplayEnd;
|
|
|
|
uint5 verticalSyncWidth;
|
|
uint8 verticalDisplayStart;
|
|
uint9 verticalDisplayLength;
|
|
uint8 verticalDisplayEnd;
|
|
|
|
bool vpulse = 0;
|
|
bool hpulse = 0;
|
|
|
|
uint hclock = 0;
|
|
uint vclock = 0;
|
|
|
|
uint hoffset = 0;
|
|
uint voffset = 0;
|
|
|
|
uint hstart = 0;
|
|
uint vstart = 0;
|
|
|
|
uint hlength = 0;
|
|
uint vlength = 0;
|
|
} timing;
|
|
|
|
struct IRQ {
|
|
enum class Line : uint {
|
|
Collision,
|
|
Overflow,
|
|
LineCoincidence,
|
|
Vblank,
|
|
TransferVRAM,
|
|
TransferSATB,
|
|
};
|
|
|
|
//irq.cpp
|
|
auto poll() -> void;
|
|
auto raise(Line) -> void;
|
|
auto lower() -> void;
|
|
|
|
bool enableCollision = 0;
|
|
bool enableOverflow = 0;
|
|
bool enableLineCoincidence = 0;
|
|
bool enableVblank = 0;
|
|
bool enableTransferVRAM = 0;
|
|
bool enableTransferSATB = 0;
|
|
|
|
bool pendingCollision = 0;
|
|
bool pendingOverflow = 0;
|
|
bool pendingLineCoincidence = 0;
|
|
bool pendingVblank = 0;
|
|
bool pendingTransferVRAM = 0;
|
|
bool pendingTransferSATB = 0;
|
|
|
|
bool line = 0;
|
|
} irq;
|
|
|
|
struct DMA {
|
|
VDC* vdc = nullptr;
|
|
|
|
//dma.cpp
|
|
auto step(uint clocks) -> void;
|
|
auto vramStart() -> void;
|
|
auto satbStart() -> void;
|
|
auto satbQueue() -> void;
|
|
|
|
bool sourceIncrementMode = 0;
|
|
bool targetIncrementMode = 0;
|
|
bool satbRepeat = 0;
|
|
uint16 source;
|
|
uint16 target;
|
|
uint16 length;
|
|
uint16 satbSource;
|
|
|
|
bool vramActive = 0;
|
|
bool satbActive = 0;
|
|
bool satbPending = 0;
|
|
uint16 satbOffset;
|
|
} dma;
|
|
|
|
struct Background {
|
|
VDC* vdc = nullptr;
|
|
|
|
//background.cpp
|
|
auto scanline(uint y) -> void;
|
|
auto run(uint x, uint y) -> void;
|
|
|
|
bool enable = 0;
|
|
uint10 hscroll;
|
|
uint9 vscroll;
|
|
uint9 vcounter;
|
|
uint8 width;
|
|
uint8 height;
|
|
|
|
uint10 hoffset;
|
|
uint9 voffset;
|
|
|
|
uint4 color;
|
|
uint4 palette;
|
|
} background;
|
|
|
|
struct Sprite {
|
|
VDC* vdc = nullptr;
|
|
|
|
//sprite.cpp
|
|
auto scanline(uint y) -> void;
|
|
auto run(uint x, uint y) -> void;
|
|
|
|
bool enable = 0;
|
|
|
|
struct Object {
|
|
uint10 y;
|
|
uint10 x;
|
|
bool mode = 0;
|
|
uint10 pattern;
|
|
uint4 palette;
|
|
bool priority = 0;
|
|
uint width = 0;
|
|
bool hflip = 0;
|
|
uint height = 0;
|
|
bool vflip = 0;
|
|
bool first = 0;
|
|
};
|
|
array<Object, 64> objects;
|
|
|
|
uint4 color;
|
|
uint4 palette;
|
|
bool priority = 0;
|
|
} sprite;
|
|
|
|
struct IO {
|
|
uint5 address;
|
|
|
|
//$0005 CR (W)
|
|
uint2 externalSync;
|
|
uint2 displayOutput;
|
|
bool dramRefresh = 0;
|
|
|
|
//$0006 RCR
|
|
uint10 lineCoincidence;
|
|
|
|
//$0009 MWR
|
|
uint2 vramAccess;
|
|
uint2 spriteAccess;
|
|
bool cgMode = 0;
|
|
} io;
|
|
};
|
|
|
|
extern VDC vdc0;
|
|
extern VDC vdc1;
|