mirror of
https://github.com/TomHarte/CLK.git
synced 2024-11-19 23:32:28 +00:00
749 lines
26 KiB
C++
749 lines
26 KiB
C++
//
|
|
// 9918Base.hpp
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 14/12/2017.
|
|
// Copyright 2017 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#ifndef TMS9918Base_hpp
|
|
#define TMS9918Base_hpp
|
|
|
|
#include "ClockConverter.hpp"
|
|
|
|
#include "../../../ClockReceiver/ClockReceiver.hpp"
|
|
#include "../../../Numeric/BitReverse.hpp"
|
|
#include "../../../Outputs/CRT/CRT.hpp"
|
|
|
|
#include "AccessEnums.hpp"
|
|
#include "PersonalityTraits.hpp"
|
|
#include "YamahaCommands.hpp"
|
|
|
|
#include <array>
|
|
#include <cassert>
|
|
#include <cstdint>
|
|
#include <cstring>
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
namespace TI {
|
|
namespace TMS {
|
|
|
|
// Temporary buffers collect a representation of each line prior to pixel serialisation.
|
|
//
|
|
// TODO: either template on personality, to avoid having to be the union of all potential footprints,
|
|
// or just stop keeping so many of these in the 9918.
|
|
struct LineBuffer {
|
|
LineBuffer() {}
|
|
|
|
// The line mode describes the proper timing diagram for this line;
|
|
// screen mode captures proper output mode.
|
|
LineMode line_mode = LineMode::Text;
|
|
ScreenMode screen_mode = ScreenMode::Text;
|
|
|
|
// Holds the horizontal scroll position to apply to this line;
|
|
// of those VDPs currently implemented, affects the Master System only.
|
|
uint8_t latched_horizontal_scroll = 0;
|
|
|
|
// The names array holds pattern names, as an offset into memory, and
|
|
// potentially flags also.
|
|
union {
|
|
// The TMS and Sega VDPs are close enough to always tile-based;
|
|
// this struct captures maximal potential detail there.
|
|
struct {
|
|
struct {
|
|
size_t offset = 0; // TODO: could presumably be much smaller. One byte maybe?
|
|
uint8_t flags = 0;
|
|
} names[40];
|
|
|
|
// The patterns array holds tile patterns, corresponding 1:1 with names.
|
|
// Four bytes per pattern is the maximum required by any
|
|
// currently-implemented VDP.
|
|
uint8_t patterns[40][4];
|
|
};
|
|
|
|
// The Yamaha VDP also has a variety of bitmap modes, the widest of which is
|
|
// 512px @ 4bpp.
|
|
uint8_t bitmap[256];
|
|
};
|
|
|
|
/*
|
|
Horizontal layout (on a 342-cycle clock):
|
|
|
|
15 cycles right border
|
|
58 cycles blanking & sync
|
|
13 cycles left border
|
|
|
|
... i.e. to cycle 86, then:
|
|
|
|
border up to first_pixel_output_column;
|
|
pixels up to next_border_column;
|
|
border up to the end.
|
|
|
|
e.g. standard 256-pixel modes will want to set
|
|
first_pixel_output_column = 86, next_border_column = 342.
|
|
*/
|
|
int first_pixel_output_column = 94;
|
|
int next_border_column = 334;
|
|
size_t pixel_count = 256;
|
|
|
|
// An active sprite is one that has been selected for composition onto
|
|
// this line.
|
|
struct ActiveSprite {
|
|
int index = 0; // The original in-table index of this sprite.
|
|
int row = 0; // The row of the sprite that should be drawn.
|
|
int x = 0; // The sprite's x position on screen.
|
|
|
|
uint8_t image[4]; // Up to four bytes of image information.
|
|
int shift_position = 0; // An offset representing how much of the image information has already been drawn.
|
|
} active_sprites[8];
|
|
|
|
int active_sprite_slot = 0; // A pointer to the slot into which a new active sprite will be deposited, if required.
|
|
bool sprites_stopped = false; // A special TMS feature is that a sentinel value can be used to prevent any further sprites
|
|
// being evaluated for display. This flag determines whether the sentinel has yet been reached.
|
|
|
|
void reset_sprite_collection();
|
|
};
|
|
|
|
struct LineBufferPointer {
|
|
int row, column;
|
|
};
|
|
|
|
constexpr uint8_t StatusInterrupt = 0x80;
|
|
constexpr uint8_t StatusSpriteOverflow = 0x40;
|
|
|
|
constexpr int StatusSpriteCollisionShift = 5;
|
|
constexpr uint8_t StatusSpriteCollision = 0x20;
|
|
|
|
/// A container for personality-specific storage; see specific instances below.
|
|
template <Personality personality, typename Enable = void> struct Storage {
|
|
};
|
|
|
|
template <> struct Storage<Personality::TMS9918A> {
|
|
void begin_line(ScreenMode, bool, bool) {}
|
|
};
|
|
|
|
// Yamaha-specific storage.
|
|
template <Personality personality> struct Storage<personality, std::enable_if_t<is_yamaha_vdp(personality)>> {
|
|
int selected_status_ = 0;
|
|
|
|
int indirect_register_ = 0;
|
|
bool increment_indirect_register_ = false;
|
|
|
|
uint32_t palette_[16]{};
|
|
uint8_t new_colour_ = 0;
|
|
uint8_t palette_entry_ = 0;
|
|
|
|
uint8_t mode_ = 0;
|
|
|
|
uint8_t vertical_offset_ = 0;
|
|
|
|
/// Describes an _observable_ memory access event. i.e. anything that it is safe
|
|
/// (and convenient) to treat as atomic in between external slots.
|
|
struct Event {
|
|
/// Offset of the _beginning_ of the event. Not completely arbitrarily: this is when
|
|
/// external data must be ready by in order to take part in those slots.
|
|
int offset = 1368;
|
|
enum class Type {
|
|
External,
|
|
DataBlock,
|
|
SpriteY,
|
|
SpriteContents,
|
|
} type = Type::External;
|
|
|
|
constexpr Event(int offset, Type type) noexcept :
|
|
offset(grauw_to_internal(offset)),
|
|
type(type) {}
|
|
|
|
constexpr Event(int offset) noexcept :
|
|
offset(grauw_to_internal(offset)) {}
|
|
|
|
constexpr Event() noexcept {}
|
|
};
|
|
|
|
// State that tracks fetching position within a line.
|
|
const Event *next_event_ = nullptr;
|
|
int data_block_ = 0;
|
|
int sprite_block_ = 0;
|
|
|
|
/// Resets line-ephemeral state for a new line.
|
|
void begin_line([[maybe_unused]] ScreenMode mode, bool is_refresh, [[maybe_unused]] bool sprites_enabled) {
|
|
// TODO: reinstate upon completion of the Yamaha pipeline.
|
|
// assert(mode < ScreenMode::YamahaText80 || next_event_ == nullptr || next_event_->offset == 1368);
|
|
|
|
data_block_ = 0;
|
|
sprite_block_ = 0;
|
|
|
|
if(is_refresh) {
|
|
next_event_ = refresh_events;
|
|
return;
|
|
}
|
|
|
|
// TODO: obey sprites_enabled flag, at least.
|
|
next_event_ = no_sprites_events;
|
|
}
|
|
|
|
// Command engine state.
|
|
CommandContext command_context_;
|
|
std::unique_ptr<Command> command_ = nullptr;
|
|
|
|
enum class CommandStep {
|
|
None,
|
|
ReadPixel,
|
|
WritePixel,
|
|
};
|
|
CommandStep next_command_step_ = CommandStep::None;
|
|
int minimum_command_column_ = 0;
|
|
uint8_t command_latch_ = 0;
|
|
|
|
void update_command_step(int current_column) {
|
|
if(!command_) {
|
|
next_command_step_ = CommandStep::None;
|
|
return;
|
|
}
|
|
if(command_->done()) {
|
|
command_ = nullptr;
|
|
next_command_step_ = CommandStep::None;
|
|
return;
|
|
}
|
|
|
|
minimum_command_column_ = current_column + command_->cycles;
|
|
switch(command_->access) {
|
|
case Command::AccessType::PlotPoint:
|
|
next_command_step_ = CommandStep::ReadPixel;
|
|
break;
|
|
case Command::AccessType::WaitForColourReceipt:
|
|
// i.e. nothing to do until a colour is received.
|
|
next_command_step_ = CommandStep::None;
|
|
break;
|
|
}
|
|
}
|
|
|
|
Storage() noexcept {
|
|
// Perform sanity checks on the event lists.
|
|
#ifndef NDEBUG
|
|
const Event *lists[] = { no_sprites_events, refresh_events, nullptr };
|
|
const Event **list = lists;
|
|
while(*list) {
|
|
const Event *cursor = *list;
|
|
++list;
|
|
|
|
while(cursor[1].offset != 1368) {
|
|
assert(cursor[1].offset > cursor[0].offset);
|
|
++cursor;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// Seed to _something_ meaningful.
|
|
//
|
|
// TODO: this is a workaround [/hack], in effect, for the main TMS' habit of starting
|
|
// in a randomised position, which means that start-of-line isn't announced.
|
|
//
|
|
// Do I really want that behaviour?
|
|
next_event_ = refresh_events;
|
|
}
|
|
|
|
private:
|
|
// This emulator treats position 0 as being immediately after the standard pixel area.
|
|
// i.e. offset 1282 on Grauw's http://map.grauw.nl/articles/vdp-vram-timing/vdp-timing.png
|
|
constexpr static int grauw_to_internal(int offset) {
|
|
return (offset + 1368 - 1282) % 1368;
|
|
}
|
|
|
|
static constexpr Event refresh_events[] = {
|
|
Event(1284), Event(1292), Event(1300), Event(1308), Event(1316), Event(1324),
|
|
Event(1334), Event(1344), Event(1352), Event(1360), Event(0), Event(8),
|
|
Event(16), Event(24), Event(32), Event(40), Event(48), Event(56),
|
|
Event(64), Event(72), Event(80), Event(88), Event(96), Event(104),
|
|
Event(112), Event(120),
|
|
|
|
Event(164), Event(172), Event(180), Event(188), Event(196), Event(204),
|
|
Event(212), Event(220), Event(228), Event(236), Event(244), Event(252),
|
|
Event(260), Event(268), Event(276), /* Refresh. */ Event(292), Event(300),
|
|
Event(308), Event(316), Event(324), Event(332), Event(340), Event(348),
|
|
Event(356), Event(364), Event(372), Event(380), Event(388), Event(396),
|
|
Event(404), /* Refresh. */ Event(420), Event(428), Event(436), Event(444),
|
|
Event(452), Event(460), Event(468), Event(476), Event(484), Event(492),
|
|
Event(500), Event(508), Event(516), Event(524), Event(532), /* Refresh. */
|
|
Event(548), Event(556), Event(564), Event(570), Event(580), Event(588),
|
|
Event(596), Event(604), Event(612), Event(620), Event(628), Event(636),
|
|
Event(644), Event(652), Event(660), /* Refresh. */ Event(676), Event(684),
|
|
Event(692), Event(700), Event(708), Event(716), Event(724), Event(732),
|
|
Event(740), Event(748), Event(756), Event(764), Event(772), Event(780),
|
|
Event(788), /* Refresh. */ Event(804), Event(812), Event(820), Event(828),
|
|
Event(836), Event(844), Event(852), Event(860), Event(868), Event(876),
|
|
Event(884), Event(892), Event(900), Event(908), Event(916), /* Refresh. */
|
|
Event(932), Event(940), Event(948), Event(956), Event(964), Event(972),
|
|
Event(980), Event(988), Event(996), Event(1004), Event(1012), Event(1020),
|
|
Event(1028), Event(1036), Event(1044), /* Refresh. */ Event(1060), Event(1068),
|
|
Event(1076), Event(1084), Event(1092), Event(1100), Event(1108), Event(1116),
|
|
Event(1124), Event(1132), Event(1140), Event(1148), Event(1156), Event(1164),
|
|
Event(1172), /* Refresh. */ Event(1188), Event(1196), Event(1204), Event(1212),
|
|
Event(1220), Event(1228),
|
|
|
|
Event(1268), Event(1276),
|
|
|
|
Event()
|
|
};
|
|
|
|
static constexpr Event no_sprites_events[] = {
|
|
Event(1282), Event(1290), Event(1298), Event(1306),
|
|
Event(1314), Event(1322), Event(1332), Event(1342),
|
|
Event(1350), Event(1358), Event(1366),
|
|
|
|
Event(6), Event(14), Event(22), Event(30),
|
|
Event(38), Event(46), Event(54), Event(62),
|
|
Event(70), Event(78), Event(86), Event(94),
|
|
Event(102), Event(110), Event(118),
|
|
|
|
Event(162), Event(170), Event(182), Event(188),
|
|
// Omitted: dummy data block. Is not observable.
|
|
Event(214), Event(220),
|
|
|
|
Event(226, Event::Type::DataBlock), Event(246), Event(252),
|
|
Event(258, Event::Type::DataBlock), Event(278), // Omitted: refresh.
|
|
Event(290, Event::Type::DataBlock), Event(310), Event(316),
|
|
Event(322, Event::Type::DataBlock), Event(342), Event(348),
|
|
Event(354, Event::Type::DataBlock), Event(374), Event(380),
|
|
Event(386, Event::Type::DataBlock), Event(406), // Omitted: refresh.
|
|
Event(418, Event::Type::DataBlock), Event(438), Event(444),
|
|
Event(450, Event::Type::DataBlock), Event(470), Event(476),
|
|
|
|
Event(482, Event::Type::DataBlock), Event(502), Event(508),
|
|
Event(514, Event::Type::DataBlock), Event(534), // Omitted: refresh.
|
|
Event(546, Event::Type::DataBlock), Event(566), Event(572),
|
|
Event(578, Event::Type::DataBlock), Event(598), Event(604),
|
|
Event(610, Event::Type::DataBlock), Event(630), Event(636),
|
|
Event(642, Event::Type::DataBlock), Event(662), // Omitted: refresh.
|
|
Event(674, Event::Type::DataBlock), Event(694), Event(700),
|
|
Event(706, Event::Type::DataBlock), Event(726), Event(732),
|
|
|
|
Event(738, Event::Type::DataBlock), Event(758), Event(764),
|
|
Event(770, Event::Type::DataBlock), Event(790), // Omitted: refresh.
|
|
Event(802, Event::Type::DataBlock), Event(822), Event(828),
|
|
Event(834, Event::Type::DataBlock), Event(854), Event(860),
|
|
Event(866, Event::Type::DataBlock), Event(886), Event(892),
|
|
Event(898, Event::Type::DataBlock), Event(918), // Omitted: refresh.
|
|
Event(930, Event::Type::DataBlock), Event(950), Event(956),
|
|
Event(962, Event::Type::DataBlock), Event(982), Event(988),
|
|
|
|
Event(994, Event::Type::DataBlock), Event(1014), Event(1020),
|
|
Event(1026, Event::Type::DataBlock), Event(1046), // Omitted: refresh.
|
|
Event(1058, Event::Type::DataBlock), Event(1078), Event(1084),
|
|
Event(1090, Event::Type::DataBlock), Event(1110), Event(1116),
|
|
Event(1122, Event::Type::DataBlock), Event(1142), Event(1148),
|
|
Event(1154, Event::Type::DataBlock), Event(1174), // Omitted: refresh.
|
|
Event(1186, Event::Type::DataBlock), Event(1206), Event(1212),
|
|
Event(1218, Event::Type::DataBlock),
|
|
|
|
Event(1266),
|
|
Event(1274),
|
|
|
|
Event()
|
|
};
|
|
};
|
|
|
|
// Master System-specific storage.
|
|
template <Personality personality> struct Storage<personality, std::enable_if_t<is_sega_vdp(personality)>> {
|
|
// The SMS VDP has a programmer-set colour palette, with a dedicated patch of RAM. But the RAM is only exactly
|
|
// fast enough for the pixel clock. So when the programmer writes to it, that causes a one-pixel glitch; there
|
|
// isn't the bandwidth for the read both write to occur simultaneously. The following buffer therefore keeps
|
|
// track of pending collisions, for visual reproduction.
|
|
struct CRAMDot {
|
|
LineBufferPointer location;
|
|
uint32_t value;
|
|
};
|
|
std::vector<CRAMDot> upcoming_cram_dots_;
|
|
|
|
// The Master System's additional colour RAM.
|
|
uint32_t colour_ram_[32];
|
|
bool cram_is_selected_ = false;
|
|
|
|
// Fields below affect only the Master System output mode.
|
|
|
|
// Programmer-set flags.
|
|
bool vertical_scroll_lock_ = false;
|
|
bool horizontal_scroll_lock_ = false;
|
|
bool hide_left_column_ = false;
|
|
bool shift_sprites_8px_left_ = false;
|
|
bool mode4_enable_ = false;
|
|
uint8_t horizontal_scroll_ = 0;
|
|
uint8_t vertical_scroll_ = 0;
|
|
|
|
// Holds the vertical scroll position for this frame; this is latched
|
|
// once and cannot dynamically be changed until the next frame.
|
|
uint8_t latched_vertical_scroll_ = 0;
|
|
|
|
// Various resource addresses with VDP-version-specific modifications
|
|
// built int.
|
|
size_t pattern_name_address_;
|
|
size_t sprite_attribute_table_address_;
|
|
size_t sprite_generator_table_address_;
|
|
|
|
void begin_line(ScreenMode, bool, bool) {}
|
|
};
|
|
|
|
template <Personality personality> struct Base: public Storage<personality> {
|
|
Base();
|
|
|
|
static constexpr int output_lag = 11; // i.e. pixel output will occur 11 cycles
|
|
// after corresponding data read.
|
|
|
|
static constexpr uint32_t palette_pack(uint8_t r, uint8_t g, uint8_t b) {
|
|
#if TARGET_RT_BIG_ENDIAN
|
|
return uint32_t((r << 24) | (g << 16) | (b << 8));
|
|
#else
|
|
return uint32_t((b << 16) | (g << 8) | r);
|
|
#endif
|
|
}
|
|
|
|
// The default TMS palette.
|
|
static constexpr std::array<uint32_t, 16> palette {
|
|
palette_pack(0, 0, 0),
|
|
palette_pack(0, 0, 0),
|
|
palette_pack(33, 200, 66),
|
|
palette_pack(94, 220, 120),
|
|
|
|
palette_pack(84, 85, 237),
|
|
palette_pack(125, 118, 252),
|
|
palette_pack(212, 82, 77),
|
|
palette_pack(66, 235, 245),
|
|
|
|
palette_pack(252, 85, 84),
|
|
palette_pack(255, 121, 120),
|
|
palette_pack(212, 193, 84),
|
|
palette_pack(230, 206, 128),
|
|
|
|
palette_pack(33, 176, 59),
|
|
palette_pack(201, 91, 186),
|
|
palette_pack(204, 204, 204),
|
|
palette_pack(255, 255, 255)
|
|
};
|
|
|
|
Outputs::CRT::CRT crt_;
|
|
TVStandard tv_standard_ = TVStandard::NTSC;
|
|
|
|
// Personality-specific metrics and converters.
|
|
ClockConverter<personality> clock_converter_;
|
|
|
|
// This VDP's DRAM.
|
|
std::array<uint8_t, memory_size(personality)> ram_;
|
|
|
|
// State of the DRAM/CRAM-access mechanism.
|
|
size_t ram_pointer_ = 0;
|
|
uint8_t read_ahead_buffer_ = 0;
|
|
MemoryAccess queued_access_ = MemoryAccess::None;
|
|
int minimum_access_column_ = 0;
|
|
|
|
// The main status register.
|
|
uint8_t status_ = 0;
|
|
|
|
// Current state of programmer input.
|
|
bool write_phase_ = false; // Determines whether the VDP is expecting the low or high byte of a write.
|
|
uint8_t low_write_ = 0; // Buffers the low byte of a write.
|
|
|
|
// Various programmable flags.
|
|
bool mode1_enable_ = false;
|
|
bool mode2_enable_ = false;
|
|
bool mode3_enable_ = false;
|
|
bool blank_display_ = false;
|
|
bool sprites_16x16_ = false;
|
|
bool sprites_magnified_ = false;
|
|
bool generate_interrupts_ = false;
|
|
int sprite_height_ = 8;
|
|
|
|
// Programmer-specified addresses.
|
|
size_t pattern_name_address_ = 0; // i.e. address of the tile map.
|
|
size_t colour_table_address_ = 0; // address of the colour map (if applicable).
|
|
size_t pattern_generator_table_address_ = 0; // address of the tile contents.
|
|
size_t sprite_attribute_table_address_ = 0; // address of the sprite list.
|
|
size_t sprite_generator_table_address_ = 0; // address of the sprite contents.
|
|
|
|
// Default colours.
|
|
uint8_t text_colour_ = 0;
|
|
uint8_t background_colour_ = 0;
|
|
|
|
// Internal mechanisms for position tracking.
|
|
int latched_column_ = 0;
|
|
|
|
// A struct to contain timing information that is a function of the current mode.
|
|
struct {
|
|
/*
|
|
Vertical layout:
|
|
|
|
Lines 0 to [pixel_lines]: standard data fetch and drawing will occur.
|
|
... to [first_vsync_line]: refresh fetches will occur and border will be output.
|
|
.. to [2.5 or 3 lines later]: vertical sync is output.
|
|
... to [total lines - 1]: refresh fetches will occur and border will be output.
|
|
... for one line: standard data fetch will occur, without drawing.
|
|
*/
|
|
int total_lines = 262;
|
|
int pixel_lines = 192;
|
|
int first_vsync_line = 227;
|
|
|
|
// Maximum number of sprite slots to populate;
|
|
// if sprites beyond this number should be visible
|
|
// then the appropriate status information will be set.
|
|
int maximum_visible_sprites = 4;
|
|
|
|
// Set the position, in cycles, of the two interrupts,
|
|
// within a line.
|
|
struct {
|
|
int column = 4;
|
|
int row = 193;
|
|
} end_of_frame_interrupt_position;
|
|
int line_interrupt_position = -1;
|
|
|
|
// Enables or disabled the recognition of the sprite
|
|
// list terminator, and sets the terminator value.
|
|
bool allow_sprite_terminator = true;
|
|
uint8_t sprite_terminator = 0xd0;
|
|
} mode_timing_;
|
|
|
|
uint8_t line_interrupt_target_ = 0xff;
|
|
uint8_t line_interrupt_counter_ = 0;
|
|
bool enable_line_interrupts_ = false;
|
|
bool line_interrupt_pending_ = false;
|
|
|
|
ScreenMode screen_mode_;
|
|
LineBuffer line_buffers_[313];
|
|
void posit_sprite(LineBuffer &buffer, int sprite_number, int sprite_y, int screen_row);
|
|
|
|
// There is a delay between reading into the line buffer and outputting from there to the screen. That delay
|
|
// is observeable because reading time affects availability of memory accesses and therefore time in which
|
|
// to update sprites and tiles, but writing time affects when the palette is used and when the collision flag
|
|
// may end up being set. So the two processes are slightly decoupled. The end of reading one line may overlap
|
|
// with the beginning of writing the next, hence the two separate line buffers.
|
|
LineBufferPointer output_pointer_, fetch_pointer_;
|
|
|
|
int fetch_line() const;
|
|
bool is_vertical_blank() const;
|
|
bool is_horizontal_blank() const;
|
|
|
|
int masked_address(int address) const;
|
|
void write_vram(uint8_t);
|
|
void write_register(uint8_t);
|
|
void write_palette(uint8_t);
|
|
void write_register_indirect(uint8_t);
|
|
uint8_t read_vram();
|
|
uint8_t read_register();
|
|
uint8_t read_palette();
|
|
uint8_t read_register_indirect();
|
|
|
|
void commit_register(int reg, uint8_t value);
|
|
|
|
ScreenMode current_screen_mode() const {
|
|
if(blank_display_) {
|
|
return ScreenMode::Blank;
|
|
}
|
|
|
|
if constexpr (is_sega_vdp(personality)) {
|
|
if(Storage<personality>::mode4_enable_) {
|
|
return ScreenMode::SMSMode4;
|
|
}
|
|
}
|
|
|
|
if constexpr (is_yamaha_vdp(personality)) {
|
|
switch(Storage<personality>::mode_) {
|
|
case 0b00001: return ScreenMode::Text;
|
|
case 0b01001: return ScreenMode::YamahaText80;
|
|
case 0b00010: return ScreenMode::MultiColour;
|
|
case 0b00000: return ScreenMode::YamahaGraphics1;
|
|
case 0b00100: return ScreenMode::YamahaGraphics2;
|
|
case 0b01000: return ScreenMode::YamahaGraphics3;
|
|
case 0b01100: return ScreenMode::YamahaGraphics4;
|
|
case 0b10000: return ScreenMode::YamahaGraphics5;
|
|
case 0b10100: return ScreenMode::YamahaGraphics6;
|
|
case 0b11100: return ScreenMode::YamahaGraphics7;
|
|
}
|
|
}
|
|
|
|
if(!mode1_enable_ && !mode2_enable_ && !mode3_enable_) {
|
|
return ScreenMode::ColouredText;
|
|
}
|
|
|
|
if(mode1_enable_ && !mode2_enable_ && !mode3_enable_) {
|
|
return ScreenMode::Text;
|
|
}
|
|
|
|
if(!mode1_enable_ && mode2_enable_ && !mode3_enable_) {
|
|
return ScreenMode::Graphics;
|
|
}
|
|
|
|
if(!mode1_enable_ && !mode2_enable_ && mode3_enable_) {
|
|
return ScreenMode::MultiColour;
|
|
}
|
|
|
|
// TODO: undocumented TMS modes.
|
|
return ScreenMode::Blank;
|
|
}
|
|
|
|
void do_external_slot(int access_column) {
|
|
// Don't do anything if the required time for the access to become executable
|
|
// has yet to pass.
|
|
if(queued_access_ == MemoryAccess::None || access_column < minimum_access_column_) {
|
|
if constexpr (is_yamaha_vdp(personality)) {
|
|
using CommandStep = typename Storage<personality>::CommandStep;
|
|
|
|
if(
|
|
Storage<personality>::next_command_step_ == CommandStep::None ||
|
|
access_column < Storage<personality>::minimum_command_column_
|
|
) {
|
|
return;
|
|
}
|
|
|
|
// Compute the affected address and pixel
|
|
unsigned address;
|
|
uint8_t mask;
|
|
switch(this->screen_mode_) {
|
|
default:
|
|
case ScreenMode::YamahaGraphics4: // 256 pixels @ 4bpp
|
|
address =
|
|
(Storage<personality>::command_->location.v[0] >> 1) +
|
|
(Storage<personality>::command_->location.v[1] << 7);
|
|
mask = 0xf0 >> ((Storage<personality>::command_->location.v[0] & 1) << 2);
|
|
break;
|
|
case ScreenMode::YamahaGraphics5: // 512 pixels @ 2bpp
|
|
address =
|
|
(Storage<personality>::command_->location.v[0] >> 2) +
|
|
(Storage<personality>::command_->location.v[1] << 7);
|
|
mask = 0xc0 >> ((Storage<personality>::command_->location.v[0] & 3) << 1);
|
|
break;
|
|
case ScreenMode::YamahaGraphics6: // 512 pixels @ 4bpp
|
|
address =
|
|
(Storage<personality>::command_->location.v[0] >> 1) +
|
|
(Storage<personality>::command_->location.v[1] << 8);
|
|
mask = 0xf0 >> ((Storage<personality>::command_->location.v[0] & 1) << 2);
|
|
break;
|
|
case ScreenMode::YamahaGraphics7: // 256 pixels @ 8bpp
|
|
address =
|
|
(Storage<personality>::command_->location.v[0] >> 0) +
|
|
(Storage<personality>::command_->location.v[1] << 8);
|
|
mask = 0xff;
|
|
break;
|
|
}
|
|
|
|
switch(Storage<personality>::next_command_step_) {
|
|
// Duplicative, but keeps the compiler happy.
|
|
case CommandStep::None:
|
|
break;
|
|
|
|
case CommandStep::ReadPixel:
|
|
Storage<personality>::command_latch_ = ram_[address];
|
|
|
|
Storage<personality>::minimum_command_column_ = access_column + 24;
|
|
Storage<personality>::next_command_step_ = CommandStep::WritePixel;
|
|
break;
|
|
|
|
case CommandStep::WritePixel: {
|
|
uint8_t packed_colour = Storage<personality>::command_context_.colour & 3;
|
|
packed_colour |= packed_colour << 2;
|
|
packed_colour |= packed_colour << 4;
|
|
|
|
Storage<personality>::command_latch_ &= ~mask;
|
|
Storage<personality>::command_latch_ |= packed_colour & mask;
|
|
ram_[address] = Storage<personality>::command_latch_;
|
|
|
|
Storage<personality>::command_->advance();
|
|
Storage<personality>::update_command_step(access_column);
|
|
} break;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
size_t address = ram_pointer_;
|
|
++ram_pointer_;
|
|
|
|
if constexpr (is_yamaha_vdp(personality)) {
|
|
const ScreenMode mode = current_screen_mode();
|
|
if(mode == ScreenMode::YamahaGraphics6 || mode == ScreenMode::YamahaGraphics7) {
|
|
// Rotate address one to the right as the hardware accesses
|
|
// the underlying banks of memory alternately but presents
|
|
// them as if linear.
|
|
address = (address >> 1) | (address << 16);
|
|
}
|
|
}
|
|
|
|
switch(queued_access_) {
|
|
default: break;
|
|
|
|
case MemoryAccess::Write:
|
|
if constexpr (is_sega_vdp(personality)) {
|
|
if(Storage<personality>::cram_is_selected_) {
|
|
// Adjust the palette. In a Master System blue has a slightly different
|
|
// scale; cf. https://www.retrorgb.com/sega-master-system-non-linear-blue-channel-findings.html
|
|
constexpr uint8_t rg_scale[] = {0, 85, 170, 255};
|
|
constexpr uint8_t b_scale[] = {0, 104, 170, 255};
|
|
Storage<personality>::colour_ram_[address & 0x1f] = palette_pack(
|
|
rg_scale[(read_ahead_buffer_ >> 0) & 3],
|
|
rg_scale[(read_ahead_buffer_ >> 2) & 3],
|
|
b_scale[(read_ahead_buffer_ >> 4) & 3]
|
|
);
|
|
|
|
// Schedule a CRAM dot; this is scheduled for wherever it should appear
|
|
// on screen. So it's wherever the output stream would be now. Which
|
|
// is output_lag cycles ago from the point of view of the input stream.
|
|
auto &dot = Storage<personality>::upcoming_cram_dots_.emplace_back();
|
|
dot.location.column = fetch_pointer_.column - output_lag;
|
|
dot.location.row = fetch_pointer_.row;
|
|
|
|
// Handle before this row conditionally; then handle after (or, more realistically,
|
|
// exactly at the end of) naturally.
|
|
if(dot.location.column < 0) {
|
|
--dot.location.row;
|
|
dot.location.column += 342;
|
|
}
|
|
dot.location.row += dot.location.column / 342;
|
|
dot.location.column %= 342;
|
|
|
|
dot.value = Storage<personality>::colour_ram_[address & 0x1f];
|
|
break;
|
|
}
|
|
}
|
|
ram_[address & memory_mask(personality)] = read_ahead_buffer_;
|
|
break;
|
|
case MemoryAccess::Read:
|
|
read_ahead_buffer_ = ram_[address & memory_mask(personality)];
|
|
break;
|
|
}
|
|
queued_access_ = MemoryAccess::None;
|
|
}
|
|
|
|
// Various fetchers.
|
|
template<bool use_end> void fetch_tms_refresh(LineBuffer &, int y, int start, int end);
|
|
template<bool use_end> void fetch_tms_text(LineBuffer &, int y, int start, int end);
|
|
template<bool use_end> void fetch_tms_character(LineBuffer &, int y, int start, int end);
|
|
|
|
template<bool use_end> void fetch_yamaha(LineBuffer &, int y, int start, int end);
|
|
template<ScreenMode> void fetch_yamaha(LineBuffer &, int y, int end);
|
|
|
|
template<bool use_end> void fetch_sms(LineBuffer &, int y, int start, int end);
|
|
|
|
// A helper function to output the current border colour for
|
|
// the number of cycles supplied.
|
|
void output_border(int cycles, uint32_t cram_dot);
|
|
|
|
// Output serialisation state.
|
|
uint32_t *pixel_target_ = nullptr, *pixel_origin_ = nullptr;
|
|
bool asked_for_write_area_ = false;
|
|
|
|
// Output serialisers.
|
|
void draw_tms_character(int start, int end);
|
|
void draw_tms_text(int start, int end);
|
|
void draw_sms(int start, int end, uint32_t cram_dot);
|
|
|
|
template<ScreenMode mode> void draw_yamaha(LineBuffer &, int start, int end);
|
|
void draw_yamaha(int start, int end);
|
|
};
|
|
|
|
#include "Fetch.hpp"
|
|
#include "Draw.hpp"
|
|
|
|
}
|
|
}
|
|
|
|
#endif /* TMS9918Base_hpp */
|