X-Git-Url: http://git.tdb.fi/?p=libs%2Fgl.git;a=blobdiff_plain;f=source%2Fcore%2Funiformblock.cpp;h=adadf0561c8051c54d06c2ac43a3a3608e6539dd;hp=4b07241fab3943d14f35932ea498024bdc5ad2d2;hb=d16d28d2ccf7c6255204f02975834f713ff1df08;hpb=1b23728908f5ec9beb08b2b70737c3903745fddc diff --git a/source/core/uniformblock.cpp b/source/core/uniformblock.cpp index 4b07241f..adadf056 100644 --- a/source/core/uniformblock.cpp +++ b/source/core/uniformblock.cpp @@ -1,73 +1,80 @@ -#include -#include -#include -#include "buffer.h" -#include "color.h" -#include "deviceinfo.h" -#include "error.h" -#include "matrix.h" -#include "uniform.h" +#include +#include "device.h" #include "uniformblock.h" -#include "vector.h" using namespace std; namespace Msp { namespace GL { -DefaultUniformBlock::DefaultUniformBlock() -{ - static Require _req(ARB_shader_objects); -} +UniformBlock::UniformBlock(const ReflectData::UniformBlockInfo &info): + UniformBlockBackend(info.bind_point>=0), + data(info.data_size) +{ } -void DefaultUniformBlock::attach(const ReflectData::UniformInfo &info, const Uniform &uni) +size_t UniformBlock::get_alignment() const { - if(info.block->bind_point>=0) - throw invalid_argument("DefaultUniformBlock::attach"); - - attach(info.location, uni); + return Device::get_current().get_info().limits.uniform_buffer_alignment; } -void DefaultUniformBlock::attach(int index, const Uniform &uni) +void UniformBlock::store(const ReflectData::UniformInfo &info, size_t array_size, const void *value) { - if(index<0) - return; - - if(static_cast(index)>=uniforms.size()) - uniforms.resize(index+1, 0); - uniforms[index] = &uni; -} - -void DefaultUniformBlock::apply() const -{ - for(unsigned i=0; iapply(i); -} - + array_size = min(array_size, max(info.array_size, 1U)); -BufferBackedUniformBlock::BufferBackedUniformBlock(unsigned s): - size(s), - data(size) -{ - static Require _req(ARB_shader_objects); - static Require _req2(ARB_uniform_buffer_object); - - if(!size) - throw invalid_argument("BufferBackedUniformBlock::BufferBackedUniformBlock"); -} + char *store_ptr; + bool packed; + if(info.block->bind_point<0) + { + if(info.location<0) + return; -unsigned BufferBackedUniformBlock::get_alignment() const -{ - return Limits::get_global().uniform_buffer_alignment; -} + store_ptr = data.data()+info.location*16; + packed = true; + } + else + { + store_ptr = data.data()+info.offset; + if(array_size!=1 && info.array_stride!=get_type_size(info.type)) + packed = false; + else if(is_matrix(info.type)) + packed = (info.matrix_stride==get_type_size(get_matrix_column_type(info.type))); + else + packed = true; + } -void BufferBackedUniformBlock::attach(const ReflectData::UniformInfo &info, const Uniform &uni) -{ - if(info.block->bind_point<0) - throw invalid_argument("BufferBackedUniformBlock::attach"); + const char *value_ptr = static_cast(value); + if(packed) + { + const char *data_end = value_ptr+array_size*get_type_size(info.type); + copy(value_ptr, data_end, store_ptr); + } + else if(is_matrix(info.type)) + { + unsigned col_size = get_type_size(get_matrix_column_type(info.type)); + unsigned cols = get_type_size(info.type)/col_size; + for(unsigned i=0; i