// Copyright (C) 2014 Jérôme Leclercq // This file is part of the "Nazara Engine - Renderer module" // For conditions of distribution and use, see copyright notice in Config.hpp #include #include #include #include #include #include #include #include NzHardwareBuffer::NzHardwareBuffer(NzBuffer* parent, nzBufferType type) : m_type(type), m_parent(parent) { } NzHardwareBuffer::~NzHardwareBuffer() = default; bool NzHardwareBuffer::Create(unsigned int size, nzBufferUsage usage) { NzContext::EnsureContext(); m_buffer = 0; glGenBuffers(1, &m_buffer); NzOpenGL::BindBuffer(m_type, m_buffer); glBufferData(NzOpenGL::BufferTarget[m_type], size, nullptr, NzOpenGL::BufferUsage[usage]); return true; } void NzHardwareBuffer::Destroy() { NzContext::EnsureContext(); NzOpenGL::DeleteBuffer(m_type, m_buffer); } bool NzHardwareBuffer::Fill(const void* data, unsigned int offset, unsigned int size, bool forceDiscard) { NzContext::EnsureContext(); unsigned int totalSize = m_parent->GetSize(); if (!forceDiscard) forceDiscard = (size == totalSize); NzOpenGL::BindBuffer(m_type, m_buffer); // Il semblerait que glBuffer(Sub)Data soit plus performant que glMapBuffer(Range) en dessous d'un certain seuil // http://www.stevestreeting.com/2007/03/17/glmapbuffer-vs-glbuffersubdata-the-return/ if (size < 32*1024) { // http://www.opengl.org/wiki/Buffer_Object_Streaming if (forceDiscard) glBufferData(NzOpenGL::BufferTarget[m_type], totalSize, nullptr, NzOpenGL::BufferUsage[m_parent->GetUsage()]); // Discard glBufferSubData(NzOpenGL::BufferTarget[m_type], offset, size, data); } else { void* ptr = Map((forceDiscard) ? nzBufferAccess_DiscardAndWrite : nzBufferAccess_WriteOnly, offset, size); if (!ptr) { NazaraError("Failed to map buffer"); return false; } std::memcpy(ptr, data, size); Unmap(); } return true; } bool NzHardwareBuffer::IsHardware() const { return true; } void* NzHardwareBuffer::Map(nzBufferAccess access, unsigned int offset, unsigned int size) { NzContext::EnsureContext(); NzOpenGL::BindBuffer(m_type, m_buffer); if (glMapBufferRange) return glMapBufferRange(NzOpenGL::BufferTarget[m_type], offset, size, NzOpenGL::BufferLockRange[access]); else { // http://www.opengl.org/wiki/Buffer_Object_Streaming if (access == nzBufferAccess_DiscardAndWrite) glBufferData(NzOpenGL::BufferTarget[m_type], m_parent->GetSize(), nullptr, NzOpenGL::BufferUsage[m_parent->GetUsage()]); // Discard nzUInt8* ptr = static_cast(glMapBuffer(NzOpenGL::BufferTarget[m_type], NzOpenGL::BufferLock[access])); if (ptr) ptr += offset; return ptr; } } bool NzHardwareBuffer::Unmap() { NzContext::EnsureContext(); NzOpenGL::BindBuffer(m_type, m_buffer); if (glUnmapBuffer(NzOpenGL::BufferTarget[m_type]) != GL_TRUE) { glBufferData(NzOpenGL::BufferTarget[m_type], m_parent->GetSize(), nullptr, NzOpenGL::BufferUsage[m_parent->GetUsage()]); // Une erreur rare est survenue, nous devons réinitialiser le buffer NazaraError("Failed to unmap buffer, reinitialising content... (OpenGL error : 0x" + NzString::Number(glGetError(), 16) + ')'); return false; } return true; } void NzHardwareBuffer::Bind() const { NzOpenGL::BindBuffer(m_type, m_buffer); } unsigned int NzHardwareBuffer::GetOpenGLID() const { return m_buffer; }