2016-02-13 20:52:23 -05:00
|
|
|
//
|
2016-11-16 22:57:17 +08:00
|
|
|
// TextureBuilder.cpp
|
2016-02-13 20:52:23 -05:00
|
|
|
// Clock Signal
|
|
|
|
//
|
2016-03-08 20:59:16 -05:00
|
|
|
// Created by Thomas Harte on 08/03/2016.
|
2016-02-13 20:52:23 -05:00
|
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
2016-11-16 22:57:17 +08:00
|
|
|
#include "TextureBuilder.hpp"
|
2016-02-13 20:52:23 -05:00
|
|
|
#include "CRTOpenGL.hpp"
|
2016-11-16 23:13:06 +08:00
|
|
|
#include "OpenGL.hpp"
|
2016-03-08 20:59:16 -05:00
|
|
|
#include <string.h>
|
2016-02-13 20:52:23 -05:00
|
|
|
|
2016-03-08 20:59:16 -05:00
|
|
|
using namespace Outputs::CRT;
|
2016-02-13 20:52:23 -05:00
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
static const GLint internalFormatForDepth(size_t depth)
|
|
|
|
{
|
|
|
|
switch(depth)
|
|
|
|
{
|
|
|
|
default: return GL_FALSE;
|
|
|
|
case 1: return GL_R8UI;
|
|
|
|
case 2: return GL_RG8UI;
|
|
|
|
case 3: return GL_RGB8UI;
|
|
|
|
case 4: return GL_RGBA8UI;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const GLenum formatForDepth(size_t depth)
|
|
|
|
{
|
|
|
|
switch(depth)
|
|
|
|
{
|
|
|
|
default: return GL_FALSE;
|
|
|
|
case 1: return GL_RED_INTEGER;
|
|
|
|
case 2: return GL_RG_INTEGER;
|
|
|
|
case 3: return GL_RGB_INTEGER;
|
|
|
|
case 4: return GL_RGBA_INTEGER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-17 12:26:04 +08:00
|
|
|
TextureBuilder::TextureBuilder(size_t bytes_per_pixel, GLenum texture_unit) :
|
2016-11-16 23:13:06 +08:00
|
|
|
bytes_per_pixel_(bytes_per_pixel),
|
|
|
|
next_write_x_position_(0),
|
|
|
|
next_write_y_position_(0)
|
2016-11-16 12:31:32 +08:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
image_.resize(bytes_per_pixel * InputBufferBuilderWidth * InputBufferBuilderHeight);
|
|
|
|
glGenTextures(1, &texture_name_);
|
2016-11-17 12:26:04 +08:00
|
|
|
|
|
|
|
glActiveTexture(texture_unit);
|
2016-11-16 23:13:06 +08:00
|
|
|
glBindTexture(GL_TEXTURE_2D, texture_name_);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, internalFormatForDepth(bytes_per_pixel), InputBufferBuilderWidth, InputBufferBuilderHeight, 0, formatForDepth(bytes_per_pixel), GL_UNSIGNED_BYTE, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
TextureBuilder::~TextureBuilder()
|
|
|
|
{
|
|
|
|
glDeleteTextures(1, &texture_name_);
|
2016-11-16 12:31:32 +08:00
|
|
|
}
|
2016-03-19 22:46:17 -04:00
|
|
|
|
2016-11-16 22:57:17 +08:00
|
|
|
uint8_t *TextureBuilder::allocate_write_area(size_t required_length)
|
2016-02-13 20:52:23 -05:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
if(next_write_y_position_ != InputBufferBuilderHeight)
|
2016-02-13 20:52:23 -05:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
last_allocation_amount_ = required_length;
|
2016-02-13 20:52:23 -05:00
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
if(next_write_x_position_ + required_length + 2 > InputBufferBuilderWidth)
|
2016-05-04 07:39:45 -04:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
next_write_x_position_ = 0;
|
|
|
|
next_write_y_position_++;
|
2016-05-04 07:39:45 -04:00
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
if(next_write_y_position_ == InputBufferBuilderHeight)
|
2016-11-16 13:15:50 +08:00
|
|
|
return nullptr;
|
2016-05-04 07:39:45 -04:00
|
|
|
}
|
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
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;
|
2016-05-07 18:37:18 -04:00
|
|
|
}
|
2016-11-16 13:15:50 +08:00
|
|
|
else return nullptr;
|
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
return &image_[write_target_pointer_ * bytes_per_pixel_];
|
2016-05-07 18:37:18 -04:00
|
|
|
}
|
2016-05-05 07:22:49 -04:00
|
|
|
|
2016-11-16 22:57:17 +08:00
|
|
|
bool TextureBuilder::is_full()
|
2016-02-13 20:52:23 -05:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
return (next_write_y_position_ == InputBufferBuilderHeight);
|
2016-05-09 07:02:12 -04:00
|
|
|
}
|
|
|
|
|
2016-11-16 22:57:17 +08:00
|
|
|
void TextureBuilder::reduce_previous_allocation_to(size_t actual_length)
|
2016-05-09 07:02:12 -04:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
if(next_write_y_position_ == InputBufferBuilderHeight) return;
|
2016-05-04 07:39:45 -04:00
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
uint8_t *const image_pointer = image_.data();
|
2016-05-08 21:11:56 -04:00
|
|
|
|
2016-05-08 16:43:11 -04:00
|
|
|
// correct if the writing cursor was reset while a client was writing
|
2016-11-16 23:13:06 +08:00
|
|
|
if(next_write_x_position_ == 0 && next_write_y_position_ == 0)
|
2016-05-08 16:43:11 -04:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
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;
|
2016-05-08 16:43:11 -04:00
|
|
|
}
|
|
|
|
|
2016-03-19 17:07:05 -04:00
|
|
|
// book end the allocation with duplicates of the first and last pixel, to protect
|
|
|
|
// against rounding errors when this run is drawn
|
2016-11-16 23:13:06 +08:00
|
|
|
memcpy( &image_pointer[(write_target_pointer_ - 1) * bytes_per_pixel_],
|
|
|
|
&image_pointer[write_target_pointer_ * bytes_per_pixel_],
|
|
|
|
bytes_per_pixel_);
|
2016-02-13 20:52:23 -05:00
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
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_);
|
2016-02-13 20:52:23 -05:00
|
|
|
|
2016-03-19 17:07:05 -04:00
|
|
|
// return any allocated length that wasn't actually used to the available pool
|
2016-11-16 23:13:06 +08:00
|
|
|
next_write_x_position_ -= (last_allocation_amount_ - actual_length);
|
2016-05-04 21:27:10 -04:00
|
|
|
}
|
|
|
|
|
2016-11-16 22:57:17 +08:00
|
|
|
uint16_t TextureBuilder::get_last_write_x_position()
|
2016-05-04 21:27:10 -04:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
return write_x_position_;
|
2016-05-04 21:27:10 -04:00
|
|
|
}
|
|
|
|
|
2016-11-16 22:57:17 +08:00
|
|
|
uint16_t TextureBuilder::get_last_write_y_position()
|
2016-05-04 21:27:10 -04:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
return write_y_position_;
|
2016-05-04 21:27:10 -04:00
|
|
|
}
|
|
|
|
|
2016-11-16 23:13:06 +08:00
|
|
|
void TextureBuilder::submit()
|
2016-05-04 21:27:10 -04:00
|
|
|
{
|
2016-11-16 23:13:06 +08:00
|
|
|
uint16_t height = write_y_position_ + (next_write_x_position_ ? 1 : 0);
|
|
|
|
next_write_x_position_ = next_write_y_position_ = 0;
|
|
|
|
|
|
|
|
glTexSubImage2D( GL_TEXTURE_2D, 0,
|
|
|
|
0, 0,
|
|
|
|
InputBufferBuilderWidth, height,
|
|
|
|
formatForDepth(bytes_per_pixel_), GL_UNSIGNED_BYTE,
|
|
|
|
image_.data());
|
2016-05-04 21:27:10 -04:00
|
|
|
}
|