From 4f870623da57a7340c22b55efb7c7a55c48d9062 Mon Sep 17 00:00:00 2001 From: exmix Date: Sat, 12 Jul 2014 17:15:38 +0200 Subject: [PATCH] [drape] apply attribute mutation to VAO --- drape/attribute_buffer_mutator.cpp | 6 ++ drape/attribute_buffer_mutator.hpp | 37 ++++++++++ drape/drape_common.pri | 2 + drape/glconstants.cpp | 10 +++ drape/glconstants.hpp | 3 + drape/glextensions_list.cpp | 2 + drape/glextensions_list.hpp | 3 +- drape/glfunctions.cpp | 24 ++++++- drape/glfunctions.hpp | 3 + drape/gpu_buffer.cpp | 108 +++++++++++++++++++++++++++++ drape/gpu_buffer.hpp | 28 ++++++++ drape/index_buffer_mutator.cpp | 10 ++- drape/index_buffer_mutator.hpp | 6 +- drape/render_bucket.cpp | 22 ++++-- drape/vertex_array_buffer.cpp | 27 +++++++- drape/vertex_array_buffer.hpp | 6 ++ 16 files changed, 283 insertions(+), 14 deletions(-) create mode 100644 drape/attribute_buffer_mutator.cpp create mode 100644 drape/attribute_buffer_mutator.hpp diff --git a/drape/attribute_buffer_mutator.cpp b/drape/attribute_buffer_mutator.cpp new file mode 100644 index 0000000000..88a2650179 --- /dev/null +++ b/drape/attribute_buffer_mutator.cpp @@ -0,0 +1,6 @@ +#include "attribute_buffer_mutator.hpp" + +void AttributeBufferMutator::AddMutation(BindingInfo const & info, MutateNode const & node) +{ + m_data[info].push_back(node); +} diff --git a/drape/attribute_buffer_mutator.hpp b/drape/attribute_buffer_mutator.hpp new file mode 100644 index 0000000000..a661e7e173 --- /dev/null +++ b/drape/attribute_buffer_mutator.hpp @@ -0,0 +1,37 @@ +#pragma once + +#include "pointers.hpp" +#include "binding_info.hpp" + +#include "../std/stdint.hpp" +#include "../std/map.hpp" + +struct MutateRegion +{ + MutateRegion() : m_offset(0), m_count(0) {} + MutateRegion(uint16_t offset, uint16_t count) : m_offset(offset), m_count(count) {} + + uint16_t m_offset; // Offset from buffer begin in "Elements" not in bytes + uint16_t m_count; // Count of "Elements". +}; + +struct MutateNode +{ + MutateRegion m_region; + RefPointer m_data; +}; + +class AttributeBufferMutator +{ + typedef vector mutate_nodes_t; + typedef map mutate_data_t; +public: + void AddMutation(BindingInfo const & info, MutateNode const & node); + +private: + friend class VertexArrayBuffer; + mutate_data_t const & GetMutateData() const { return m_data; } + +private: + mutate_data_t m_data; +}; diff --git a/drape/drape_common.pri b/drape/drape_common.pri index a973b75f3b..4619528cb9 100644 --- a/drape/drape_common.pri +++ b/drape/drape_common.pri @@ -34,6 +34,7 @@ SOURCES += \ $$DRAPE_DIR/font_texture.cpp \ $$DRAPE_DIR/texture_set_holder.cpp \ $$DRAPE_DIR/utils/stb_image.c \ + $$DRAPE_DIR/attribute_buffer_mutator.cpp \ HEADERS += \ $$DRAPE_DIR/data_buffer.hpp \ @@ -72,3 +73,4 @@ HEADERS += \ $$DRAPE_DIR/overlay_tree.hpp \ $$DRAPE_DIR/font_texture.hpp \ $$DRAPE_DIR/utils/stb_image.h \ + $$DRAPE_DIR/attribute_buffer_mutator.hpp \ diff --git a/drape/glconstants.cpp b/drape/glconstants.cpp index 3d079db4ed..5ea90c7788 100644 --- a/drape/glconstants.cpp +++ b/drape/glconstants.cpp @@ -25,6 +25,14 @@ #define GL_LUMINANCE8_ALPHA4_OES 0x8043 #endif +#if defined(GL_WRITE_ONLY) + #define WRITE_ONLY_DEF GL_WRITE_ONLY +#elif defined(GL_WRITE_ONLY_OES) + #define WRITE_ONLY_DEF GL_WRITE_ONLY_OES +#else + #define WRITE_ONLY_DEF 0x88B9 +#endif + namespace gl_const { @@ -35,6 +43,8 @@ const glConst GLMaxTextureSize = GL_MAX_TEXTURE_SIZE; const glConst GLArrayBuffer = GL_ARRAY_BUFFER; const glConst GLElementArrayBuffer = GL_ELEMENT_ARRAY_BUFFER; +const glConst GLWriteOnly = WRITE_ONLY_DEF; + const glConst GLStaticDraw = GL_STATIC_DRAW; const glConst GLStreamDraw = GL_STREAM_DRAW; const glConst GLDynamicDraw = GL_DYNAMIC_DRAW; diff --git a/drape/glconstants.hpp b/drape/glconstants.hpp index a1de2f3d59..9ddb141e9d 100644 --- a/drape/glconstants.hpp +++ b/drape/glconstants.hpp @@ -16,6 +16,9 @@ extern const glConst GLMaxTextureSize; extern const glConst GLArrayBuffer; extern const glConst GLElementArrayBuffer; +/// VBO Access +extern const glConst GLWriteOnly; + /// BufferUsage extern const glConst GLStaticDraw; extern const glConst GLStreamDraw; diff --git a/drape/glextensions_list.cpp b/drape/glextensions_list.cpp index 0b0b022f06..e18bba571f 100644 --- a/drape/glextensions_list.cpp +++ b/drape/glextensions_list.cpp @@ -61,10 +61,12 @@ GLExtensionsList::GLExtensionsList() m_impl->CheckExtension(VertexArrayObject, "GL_OES_vertex_array_object"); m_impl->CheckExtension(TextureNPOT, "GL_OES_texture_npot"); m_impl->CheckExtension(RequiredInternalFormat, "GL_OES_required_internalformat"); + m_impl->CheckExtension(MapBuffer, "GL_OES_mapbuffer"); #else m_impl->CheckExtension(VertexArrayObject, "GL_APPLE_vertex_array_object"); m_impl->CheckExtension(TextureNPOT, "GL_ARB_texture_non_power_of_two"); m_impl->CheckExtension(RequiredInternalFormat, "GL_OES_required_internalformat"); + m_impl->CheckExtension(MapBuffer, "GL_OES_mapbuffer"); #endif } diff --git a/drape/glextensions_list.hpp b/drape/glextensions_list.hpp index 92fc83c508..6aa9fb6e21 100644 --- a/drape/glextensions_list.hpp +++ b/drape/glextensions_list.hpp @@ -9,7 +9,8 @@ public: { VertexArrayObject, TextureNPOT, - RequiredInternalFormat + RequiredInternalFormat, + MapBuffer }; static GLExtensionsList & Instance(); diff --git a/drape/glfunctions.cpp b/drape/glfunctions.cpp index 9a12e407a5..5265d65a6e 100644 --- a/drape/glfunctions.cpp +++ b/drape/glfunctions.cpp @@ -27,11 +27,13 @@ namespace void (*glDeleteVertexArrayFn)(GLsizei n, GLuint const * ids) = NULL; /// VBO - void (*glGenBuffersFn)(GLsizei n, GLuint * buffers) = NULL; + void (*glGenBuffersFn)(GLsizei n, GLuint * buffers) = NULL; void (*glBindBufferFn)(GLenum target, GLuint buffer) = NULL; - void (*glDeleteBuffersFn)(GLsizei n, GLuint const * buffers) = NULL; + void (*glDeleteBuffersFn)(GLsizei n, GLuint const * buffers) = NULL; void (*glBufferDataFn)(GLenum target, GLsizeiptr size, GLvoid const * data, GLenum usage) = NULL; void (*glBufferSubDataFn)(GLenum target, GLintptr offset, GLsizeiptr size, GLvoid const * data) = NULL; + void * (*glMapBufferFn)(GLenum target, GLenum access) = NULL; + GLboolean (*glUnmapBufferFn)(GLenum target) = NULL; /// Shaders GLuint (*glCreateShaderFn)(GLenum type) = NULL; @@ -119,6 +121,8 @@ void GLFunctions::Init() glDeleteBuffersFn = &::glDeleteBuffers; glBufferDataFn = &::glBufferData; glBufferSubDataFn = &::glBufferSubData; + glMapBufferFn = &::glMapBuffer; + glUnmapBufferFn = &::glUnmapBuffer; /// Shaders glCreateShaderFn = &::glCreateShader; @@ -302,6 +306,22 @@ void GLFunctions::glBufferSubData(glConst target, uint32_t size, void const * da GLCHECK(glBufferSubDataFn(target, offset, size, data)); } +void * GLFunctions::glMapBuffer(glConst target) +{ + ASSERT(glMapBufferFn != NULL, ()); + void * result = glMapBufferFn(target, gl_const::GLWriteOnly); + GLCHECKCALL(); + return result; +} + +void GLFunctions::glUnmapBuffer(glConst target) +{ + ASSERT(glUnmapBufferFn != NULL, ()); + GLboolean result = glUnmapBufferFn(target); + ASSERT(result == GL_TRUE, ()); + GLCHECKCALL(); +} + uint32_t GLFunctions::glCreateShader(glConst type) { ASSERT(glCreateShaderFn != NULL, ()); diff --git a/drape/glfunctions.hpp b/drape/glfunctions.hpp index 5c17f104cd..3046224d59 100644 --- a/drape/glfunctions.hpp +++ b/drape/glfunctions.hpp @@ -40,6 +40,9 @@ public: static void glBufferData(glConst target, uint32_t size, void const * data, glConst usage); static void glBufferSubData(glConst target, uint32_t size, void const * data, uint32_t offset); + static void * glMapBuffer(glConst target); + static void glUnmapBuffer(glConst target); + /// Shaders support static uint32_t glCreateShader(glConst type); static void glShaderSource(uint32_t shaderID, string const & src); diff --git a/drape/gpu_buffer.cpp b/drape/gpu_buffer.cpp index 60bed54833..f7a235cfca 100644 --- a/drape/gpu_buffer.cpp +++ b/drape/gpu_buffer.cpp @@ -1,8 +1,18 @@ #include "gpu_buffer.hpp" #include "glfunctions.hpp" +#include "glextensions_list.hpp" #include "../base/assert.hpp" +namespace +{ + bool IsMapBufferSupported() + { + static bool isSupported = GLExtensionsList::Instance().IsSupported(GLExtensionsList::MapBuffer); + return isSupported; + } +} + glConst glTarget(GPUBuffer::Target t) { if (t == GPUBuffer::ElementBuffer) @@ -14,6 +24,9 @@ glConst glTarget(GPUBuffer::Target t) GPUBuffer::GPUBuffer(Target t, uint8_t elementSize, uint16_t capacity) : base_t(elementSize, capacity) , m_t(t) +#ifdef DEBUG + , m_isMapped(false) +#endif { m_bufferID = GLFunctions::glGenBuffer(); Resize(capacity); @@ -26,6 +39,8 @@ GPUBuffer::~GPUBuffer() void GPUBuffer::UploadData(void const * data, uint16_t elementCount) { + ASSERT(m_isMapped == false, ()); + uint16_t currentSize = GetCurrentSize(); uint8_t elementSize = GetElementSize(); ASSERT(GetCapacity() >= elementCount + currentSize, ("Not enough memory to upload ", elementCount, " elements")); @@ -39,9 +54,102 @@ void GPUBuffer::Bind() GLFunctions::glBindBuffer(m_bufferID, glTarget((m_t))); } +void * GPUBuffer::Map() +{ +#ifdef DEBUG + ASSERT(m_isMapped == false, ()); + m_isMapped = true; +#endif + + if (IsMapBufferSupported()) + return GLFunctions::glMapBuffer(glTarget(m_t)); + + return NULL; +} + +void GPUBuffer::UpdateData(void * gpuPtr, void const * data, uint16_t elementOffset, uint16_t elementCount) +{ + uint16_t elementSize = GetElementSize(); + uint32_t byteOffset = elementOffset * (uint32_t)elementSize; + uint32_t byteCount = elementCount * (uint32_t)elementSize; + ASSERT(m_isMapped == true, ()); + if (IsMapBufferSupported()) + { + ASSERT(gpuPtr != NULL, ()); + memcpy((uint8_t *)gpuPtr + byteOffset, data, byteCount); + } + else + { + ASSERT(gpuPtr == NULL, ()); + if (byteOffset == 0 && byteCount == GetCapacity()) + GLFunctions::glBufferData(glTarget(m_t), byteCount, data, gl_const::GLStaticDraw); + else + GLFunctions::glBufferSubData(glTarget(m_t), byteCount, data, byteOffset); + } +} + +void GPUBuffer::Unmap() +{ +#ifdef DEBUG + ASSERT(m_isMapped == true, ()); + m_isMapped = false; +#endif + if (IsMapBufferSupported()) + GLFunctions::glUnmapBuffer(glTarget(m_t)); +} + void GPUBuffer::Resize(uint16_t elementCount) { base_t::Resize(elementCount); Bind(); GLFunctions::glBufferData(glTarget(m_t), GetCapacity() * GetElementSize(), NULL, gl_const::GLStaticDraw); } + +//////////////////////////////////////////////////////////////////////////// +GPUBufferMapper::GPUBufferMapper(RefPointer buffer) + : m_buffer(buffer) +{ +#ifdef DEBUG + if (m_buffer->m_t == GPUBuffer::ElementBuffer) + { + ASSERT(m_mappedDataBuffer == 0, ()); + m_mappedDataBuffer = m_buffer->m_bufferID; + } + else + { + ASSERT(m_mappedIndexBuffer == 0, ()); + m_mappedIndexBuffer = m_buffer->m_bufferID; + } +#endif + + m_buffer->Bind(); + m_gpuPtr = m_buffer->Map(); +} + +GPUBufferMapper::~GPUBufferMapper() +{ +#ifdef DEBUG + if (m_buffer->m_t == GPUBuffer::ElementBuffer) + { + ASSERT(m_mappedDataBuffer == m_buffer->m_bufferID, ()); + m_mappedDataBuffer = 0; + } + else + { + ASSERT(m_mappedIndexBuffer == m_buffer->m_bufferID, ()); + m_mappedIndexBuffer = 0; + } +#endif + + m_buffer->Unmap(); +} + +void GPUBufferMapper::UpdateData(void const * data, uint16_t elementOffset, uint16_t elementCount) +{ + m_buffer->UpdateData(m_gpuPtr, data, elementOffset, elementCount); +} + +#ifdef DEBUG + uint32_t GPUBufferMapper::m_mappedDataBuffer; + uint32_t GPUBufferMapper::m_mappedIndexBuffer; +#endif diff --git a/drape/gpu_buffer.hpp b/drape/gpu_buffer.hpp index a9b2479d66..4593f1813d 100644 --- a/drape/gpu_buffer.hpp +++ b/drape/gpu_buffer.hpp @@ -1,5 +1,6 @@ #pragma once +#include "pointers.hpp" #include "buffer_base.hpp" class GPUBuffer : public BufferBase @@ -20,10 +21,37 @@ public: void Bind(); protected: + void * Map(); + void UpdateData(void * gpuPtr, void const * data, uint16_t elementOffset, uint16_t elementCount); + void Unmap(); + /// discard old data void Resize(uint16_t elementCount); private: + friend class GPUBufferMapper; Target m_t; uint32_t m_bufferID; + +#ifdef DEBUG + bool m_isMapped; +#endif +}; + +class GPUBufferMapper +{ +public: + GPUBufferMapper(RefPointer buffer); + ~GPUBufferMapper(); + + void UpdateData(void const * data, uint16_t elementOffset, uint16_t elementCount); + +private: +#ifdef DEBUG + static uint32_t m_mappedDataBuffer; + static uint32_t m_mappedIndexBuffer; +#endif + + RefPointer m_buffer; + void * m_gpuPtr; }; diff --git a/drape/index_buffer_mutator.cpp b/drape/index_buffer_mutator.cpp index bda8f22e85..637c79b711 100644 --- a/drape/index_buffer_mutator.cpp +++ b/drape/index_buffer_mutator.cpp @@ -20,7 +20,13 @@ void IndexBufferMutator::AppendIndexes(uint16_t const * indexes, uint16_t count) m_activeSize = dstActiveSize; } -void IndexBufferMutator::Submit(RefPointer vao) +uint16_t const * IndexBufferMutator::GetIndexes() const { - vao->UpdateIndexBuffer(&m_buffer[0], m_activeSize); + return &m_buffer[0]; } + +uint16_t IndexBufferMutator::GetIndexCount() const +{ + return m_activeSize; +} + diff --git a/drape/index_buffer_mutator.hpp b/drape/index_buffer_mutator.hpp index d5143c8820..44884ff539 100644 --- a/drape/index_buffer_mutator.hpp +++ b/drape/index_buffer_mutator.hpp @@ -10,7 +10,11 @@ public: IndexBufferMutator(uint16_t baseSize); void AppendIndexes(uint16_t const * indexes, uint16_t count); - void Submit(RefPointer vao); + +private: + friend class VertexArrayBuffer; + uint16_t const * GetIndexes() const; + uint16_t GetIndexCount() const; private: vector m_buffer; diff --git a/drape/render_bucket.cpp b/drape/render_bucket.cpp index 54d94e1ba7..8f50414aa0 100644 --- a/drape/render_bucket.cpp +++ b/drape/render_bucket.cpp @@ -1,6 +1,7 @@ #include "render_bucket.hpp" #include "overlay_handle.hpp" +#include "attribute_buffer_mutator.hpp" #include "vertex_array_buffer.hpp" #include "overlay_tree.hpp" @@ -37,10 +38,16 @@ void RenderBucket::CollectOverlayHandles(RefPointer tree) namespace { -void AccumulateIndexes(MasterPointer handle, RefPointer mutator) +void AccumulateMutations(MasterPointer const & handle, + RefPointer indexMutator, + RefPointer attributeMutator) { if (handle->IsVisible()) - handle->GetElementIndexes(mutator); + { + handle->GetElementIndexes(indexMutator); + if (handle->HasDynamicAttributes()) + handle->GetAttributeMutation(attributeMutator); + } } } // namespace @@ -50,10 +57,13 @@ void RenderBucket::Render() if (!m_overlay.empty()) { // in simple case when overlay is symbol each element will be contains 6 indexes - IndexBufferMutator mutator(6 * m_overlay.size()); - for_each(m_overlay.begin(), m_overlay.end(), bind(&AccumulateIndexes, _1, - MakeStackRefPointer(&mutator))); - mutator.Submit(m_buffer.GetRefPointer()); + AttributeBufferMutator attributeMutator; + IndexBufferMutator indexMutator(6 * m_overlay.size()); + for_each(m_overlay.begin(), m_overlay.end(), bind(&AccumulateMutations, _1, + MakeStackRefPointer(&indexMutator), + MakeStackRefPointer(&attributeMutator))); + m_buffer->ApplyMutation(MakeStackRefPointer(&indexMutator), + MakeStackRefPointer(&attributeMutator)); } m_buffer->Render(); } diff --git a/drape/vertex_array_buffer.cpp b/drape/vertex_array_buffer.cpp index e4bb22de6a..dbe4afc71d 100644 --- a/drape/vertex_array_buffer.cpp +++ b/drape/vertex_array_buffer.cpp @@ -145,6 +145,11 @@ uint16_t VertexArrayBuffer::GetStartIndexValue() const return m_staticBuffers.begin()->second->GetCurrentSize(); } +uint16_t VertexArrayBuffer::GetDynamicBufferOffset(BindingInfo const & bindingInfo) +{ + return GetOrCreateDynamicBuffer(bindingInfo)->GetCurrentSize(); +} + bool VertexArrayBuffer::IsFilled() const { return GetAvailableIndexCount() < 3 || GetAvailableVertexCount() < 3; @@ -156,9 +161,27 @@ void VertexArrayBuffer::UploadIndexes(uint16_t const * data, uint16_t count) m_indexBuffer->UploadData(data, count); } -void VertexArrayBuffer::UpdateIndexBuffer(uint16_t const * data, uint16_t count) +void VertexArrayBuffer::ApplyMutation(RefPointer indexMutator, + RefPointer attrMutator) { - m_indexBuffer->UpdateData(data, count); + m_indexBuffer->UpdateData(indexMutator->GetIndexes(), indexMutator->GetIndexCount()); + + typedef AttributeBufferMutator::mutate_data_t mutate_data_t; + typedef AttributeBufferMutator::mutate_nodes_t mutate_nodes_t; + mutate_data_t const & data = attrMutator->GetMutateData(); + for (mutate_data_t::const_iterator it = data.begin(); it != data.end(); ++it) + { + RefPointer buffer = GetDynamicBuffer(it->first); + ASSERT(!buffer.IsNull(), ()); + GPUBufferMapper mapper(buffer); + mutate_nodes_t const & nodes = it->second; + + for (size_t i = 0; i < nodes.size(); ++i) + { + MutateNode const & node = nodes[i]; + mapper.UpdateData(node.m_data.GetRaw(), node.m_region.m_offset, node.m_region.m_count); + } + } } void VertexArrayBuffer::Bind() diff --git a/drape/vertex_array_buffer.hpp b/drape/vertex_array_buffer.hpp index a99ae5165b..fe6b88fcf3 100644 --- a/drape/vertex_array_buffer.hpp +++ b/drape/vertex_array_buffer.hpp @@ -1,5 +1,7 @@ #pragma once +#include "index_buffer_mutator.hpp" +#include "attribute_buffer_mutator.hpp" #include "pointers.hpp" #include "index_buffer.hpp" #include "data_buffer.hpp" @@ -26,11 +28,15 @@ public: uint16_t GetAvailableVertexCount() const; uint16_t GetAvailableIndexCount() const; uint16_t GetStartIndexValue() const; + uint16_t GetDynamicBufferOffset(BindingInfo const & bindingInfo); bool IsFilled() const; void UploadData(BindingInfo const & bindingInfo, void const * data, uint16_t count); void UploadIndexes(uint16_t const * data, uint16_t count); + void ApplyMutation(RefPointer indexMutator, + RefPointer attrMutator); + private: RefPointer GetOrCreateStaticBuffer(BindingInfo const & bindingInfo); RefPointer GetOrCreateDynamicBuffer(BindingInfo const & bindingInfo);