2017-06-06 21:53:23 +00:00
|
|
|
//
|
|
|
|
// Video.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 06/06/2017.
|
|
|
|
// Copyright © 2017 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "Video.hpp"
|
|
|
|
|
|
|
|
using namespace ZX8081;
|
|
|
|
|
2018-04-08 14:35:07 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
/*!
|
|
|
|
The number of bytes of PCM data to allocate at once; if/when more are required,
|
|
|
|
the class will simply allocate another batch.
|
|
|
|
*/
|
|
|
|
const std::size_t StandardAllocationSize = 40;
|
|
|
|
|
|
|
|
/// The amount of time a byte takes to output.
|
|
|
|
const std::size_t HalfCyclesPerByte = 8;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-06-06 21:53:23 +00:00
|
|
|
Video::Video() :
|
2017-11-10 03:14:22 +00:00
|
|
|
crt_(new Outputs::CRT::CRT(207 * 2, 1, Outputs::CRT::DisplayType::PAL50, 1)) {
|
2017-06-06 21:53:23 +00:00
|
|
|
|
2018-04-08 02:17:47 +00:00
|
|
|
// Set a composite sampling function that assumes 1bpp input.
|
2017-06-06 21:53:23 +00:00
|
|
|
crt_->set_composite_sampling_function(
|
|
|
|
"float composite_sample(usampler2D sampler, vec2 coordinate, vec2 icoordinate, float phase, float amplitude)"
|
|
|
|
"{"
|
2018-04-08 02:17:47 +00:00
|
|
|
"uint texValue = texture(sampler, coordinate).r;"
|
2018-04-18 23:29:03 +00:00
|
|
|
"texValue <<= int(icoordinate.x) & 7;"
|
2018-04-08 02:17:47 +00:00
|
|
|
"return float(texValue & 128u);"
|
2017-06-06 21:53:23 +00:00
|
|
|
"}");
|
2018-04-18 23:29:03 +00:00
|
|
|
crt_->set_integer_coordinate_multiplier(8.0f);
|
2017-06-11 21:24:32 +00:00
|
|
|
|
|
|
|
// Show only the centre 80% of the TV frame.
|
2018-04-04 23:01:18 +00:00
|
|
|
crt_->set_video_signal(Outputs::CRT::VideoSignal::Composite);
|
2017-06-11 21:24:32 +00:00
|
|
|
crt_->set_visible_area(Outputs::CRT::Rect(0.1f, 0.1f, 0.8f, 0.8f));
|
2017-06-06 21:53:23 +00:00
|
|
|
}
|
|
|
|
|
2017-07-28 02:05:29 +00:00
|
|
|
void Video::run_for(const HalfCycles half_cycles) {
|
2017-06-06 22:01:33 +00:00
|
|
|
// Just keep a running total of the amount of time that remains owed to the CRT.
|
2017-10-21 23:49:04 +00:00
|
|
|
cycles_since_update_ += static_cast<unsigned int>(half_cycles.as_int());
|
2017-06-06 21:53:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Video::flush() {
|
2017-06-06 22:01:33 +00:00
|
|
|
flush(sync_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Video::flush(bool next_sync) {
|
2017-06-06 21:53:23 +00:00
|
|
|
if(sync_) {
|
2017-06-06 22:01:33 +00:00
|
|
|
// If in sync, that takes priority. Output the proper amount of sync.
|
2017-06-06 21:53:23 +00:00
|
|
|
crt_->output_sync(cycles_since_update_);
|
|
|
|
} else {
|
2017-06-06 22:01:33 +00:00
|
|
|
// If not presently in sync, then...
|
|
|
|
|
2017-06-06 21:53:23 +00:00
|
|
|
if(line_data_) {
|
2017-06-06 22:01:33 +00:00
|
|
|
// If there is output data queued, output it either if it's being interrupted by
|
|
|
|
// sync, or if we're past its end anyway. Otherwise let it be.
|
2018-04-08 14:35:07 +00:00
|
|
|
unsigned int data_length = static_cast<unsigned int>(line_data_pointer_ - line_data_) * HalfCyclesPerByte;
|
2017-06-06 22:01:33 +00:00
|
|
|
if(data_length < cycles_since_update_ || next_sync) {
|
2017-07-09 23:33:05 +00:00
|
|
|
unsigned int output_length = std::min(data_length, cycles_since_update_);
|
2018-04-17 00:00:56 +00:00
|
|
|
crt_->output_data(output_length, output_length / HalfCyclesPerByte);
|
2017-06-06 21:53:23 +00:00
|
|
|
line_data_pointer_ = line_data_ = nullptr;
|
2017-07-09 23:33:05 +00:00
|
|
|
cycles_since_update_ -= output_length;
|
2017-06-06 21:53:23 +00:00
|
|
|
} else return;
|
|
|
|
}
|
|
|
|
|
2017-06-06 22:01:33 +00:00
|
|
|
// Any pending pixels being dealt with, pad with the white level.
|
2017-10-22 02:30:15 +00:00
|
|
|
uint8_t *colour_pointer = static_cast<uint8_t *>(crt_->allocate_write_area(1));
|
2017-06-06 21:53:23 +00:00
|
|
|
if(colour_pointer) *colour_pointer = 0xff;
|
|
|
|
crt_->output_level(cycles_since_update_);
|
|
|
|
}
|
|
|
|
|
|
|
|
cycles_since_update_ = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Video::set_sync(bool sync) {
|
2017-06-06 22:01:33 +00:00
|
|
|
// Do nothing if sync hasn't changed.
|
2017-06-06 21:53:23 +00:00
|
|
|
if(sync_ == sync) return;
|
2017-06-06 22:01:33 +00:00
|
|
|
|
|
|
|
// Complete whatever was being drawn, and update sync.
|
|
|
|
flush(sync);
|
2017-06-06 21:53:23 +00:00
|
|
|
sync_ = sync;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Video::output_byte(uint8_t byte) {
|
2017-06-06 22:01:33 +00:00
|
|
|
// Complete whatever was going on.
|
2017-07-09 01:01:07 +00:00
|
|
|
if(sync_) return;
|
2017-06-06 21:53:23 +00:00
|
|
|
flush();
|
|
|
|
|
2017-06-06 22:01:33 +00:00
|
|
|
// Grab a buffer if one isn't already available.
|
2017-06-06 21:53:23 +00:00
|
|
|
if(!line_data_) {
|
2018-04-08 14:35:07 +00:00
|
|
|
line_data_pointer_ = line_data_ = crt_->allocate_write_area(StandardAllocationSize);
|
2017-06-06 21:53:23 +00:00
|
|
|
}
|
|
|
|
|
2017-06-06 22:01:33 +00:00
|
|
|
// If a buffer was obtained, serialise the new pixels.
|
2017-06-06 21:53:23 +00:00
|
|
|
if(line_data_) {
|
2017-07-09 23:33:05 +00:00
|
|
|
// If the buffer is full, output it now and obtain a new one
|
2018-04-08 14:35:07 +00:00
|
|
|
if(line_data_pointer_ - line_data_ == StandardAllocationSize) {
|
2018-04-17 00:00:56 +00:00
|
|
|
crt_->output_data(StandardAllocationSize * HalfCyclesPerByte, StandardAllocationSize);
|
2018-04-08 14:35:07 +00:00
|
|
|
cycles_since_update_ -= StandardAllocationSize * HalfCyclesPerByte;
|
|
|
|
line_data_pointer_ = line_data_ = crt_->allocate_write_area(StandardAllocationSize);
|
2017-07-09 23:33:05 +00:00
|
|
|
if(!line_data_) return;
|
|
|
|
}
|
|
|
|
|
2018-04-08 02:17:47 +00:00
|
|
|
line_data_pointer_[0] = byte;
|
|
|
|
line_data_pointer_ ++;
|
2017-06-06 21:53:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-18 02:26:06 +00:00
|
|
|
Outputs::CRT::CRT *Video::get_crt() {
|
|
|
|
return crt_.get();
|
2017-06-06 21:53:23 +00:00
|
|
|
}
|