mirror of
https://github.com/TomHarte/CLK.git
synced 2024-11-03 08:05:40 +00:00
111 lines
3.2 KiB
C++
111 lines
3.2 KiB
C++
//
|
|
// CRTInputBufferBuilder.cpp
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 08/03/2016.
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#include "CRTInputBufferBuilder.hpp"
|
|
#include "CRTOpenGL.hpp"
|
|
#include <string.h>
|
|
|
|
using namespace Outputs::CRT;
|
|
|
|
CRTInputBufferBuilder::CRTInputBufferBuilder(size_t bytes_per_pixel) :
|
|
_bytes_per_pixel(bytes_per_pixel),
|
|
_next_write_x_position(0),
|
|
_next_write_y_position(0),
|
|
_image(new uint8_t[bytes_per_pixel * InputBufferBuilderWidth * InputBufferBuilderHeight])
|
|
{}
|
|
|
|
void CRTInputBufferBuilder::allocate_write_area(size_t required_length)
|
|
{
|
|
if(_next_write_y_position != InputBufferBuilderHeight)
|
|
{
|
|
_last_allocation_amount = required_length;
|
|
|
|
if(_next_write_x_position + required_length + 2 > InputBufferBuilderWidth)
|
|
{
|
|
_next_write_x_position = 0;
|
|
_next_write_y_position++;
|
|
|
|
if(_next_write_y_position == InputBufferBuilderHeight)
|
|
return;
|
|
}
|
|
|
|
_write_x_position = _next_write_x_position + 1;
|
|
_write_y_position = _next_write_y_position;
|
|
_write_target_pointer = (_write_y_position * InputBufferBuilderWidth) + _write_x_position;
|
|
_next_write_x_position += required_length + 2;
|
|
}
|
|
}
|
|
|
|
bool CRTInputBufferBuilder::is_full()
|
|
{
|
|
return (_next_write_y_position == InputBufferBuilderHeight);
|
|
}
|
|
|
|
void CRTInputBufferBuilder::reduce_previous_allocation_to(size_t actual_length)
|
|
{
|
|
if(_next_write_y_position == InputBufferBuilderHeight) return;
|
|
|
|
uint8_t *const image_pointer = _image.get();
|
|
|
|
// correct if the writing cursor was reset while a client was writing
|
|
if(_next_write_x_position == 0 && _next_write_y_position == 0)
|
|
{
|
|
memmove(&image_pointer[_bytes_per_pixel], &image_pointer[_write_target_pointer * _bytes_per_pixel], actual_length * _bytes_per_pixel);
|
|
_write_target_pointer = 1;
|
|
_last_allocation_amount = actual_length;
|
|
_next_write_x_position = (uint16_t)(actual_length + 2);
|
|
_write_x_position = 1;
|
|
_write_y_position = 0;
|
|
}
|
|
|
|
// book end the allocation with duplicates of the first and last pixel, to protect
|
|
// against rounding errors when this run is drawn
|
|
memcpy( &image_pointer[(_write_target_pointer - 1) * _bytes_per_pixel],
|
|
&image_pointer[_write_target_pointer * _bytes_per_pixel],
|
|
_bytes_per_pixel);
|
|
|
|
memcpy( &image_pointer[(_write_target_pointer + actual_length) * _bytes_per_pixel],
|
|
&image_pointer[(_write_target_pointer + actual_length - 1) * _bytes_per_pixel],
|
|
_bytes_per_pixel);
|
|
|
|
// return any allocated length that wasn't actually used to the available pool
|
|
_next_write_x_position -= (_last_allocation_amount - actual_length);
|
|
}
|
|
|
|
uint8_t *CRTInputBufferBuilder::get_image_pointer()
|
|
{
|
|
return _image.get();
|
|
}
|
|
|
|
uint16_t CRTInputBufferBuilder::get_and_finalise_current_line()
|
|
{
|
|
uint16_t result = _write_y_position + (_next_write_x_position ? 1 : 0);
|
|
_next_write_x_position = _next_write_y_position = 0;
|
|
return result;
|
|
}
|
|
|
|
uint8_t *CRTInputBufferBuilder::get_write_target()
|
|
{
|
|
return (_next_write_y_position == InputBufferBuilderHeight) ? nullptr : &_image.get()[_write_target_pointer * _bytes_per_pixel];
|
|
}
|
|
|
|
uint16_t CRTInputBufferBuilder::get_last_write_x_position()
|
|
{
|
|
return _write_x_position;
|
|
}
|
|
|
|
uint16_t CRTInputBufferBuilder::get_last_write_y_position()
|
|
{
|
|
return _write_y_position;
|
|
}
|
|
|
|
size_t CRTInputBufferBuilder::get_bytes_per_pixel()
|
|
{
|
|
return _bytes_per_pixel;
|
|
}
|