GLS  1.0.0
GL Stuff - A library aimed at reducing the boilerplate OpenGL code you always have to write.
buffer.hpp
1 /* This Source Code Form is subject to the terms of the Mozilla Public
2  * License, v. 2.0. If a copy of the MPL was not distributed with this
3  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
4 
5 #pragma once
6 
7 #include <gls/headercheck.hpp>
8 #include <gls/errorcheck.hpp>
9 #include <gls/objects/object.hpp>
10 #include <cassert>
11 
12 namespace gls {
13 
14 namespace priv {
15  void gen_buffers( GLsizei size, GLuint* name ) { glGenBuffers( size, name ); }
16  void delete_buffers( GLsizei size, const GLuint* name ) { glDeleteBuffers( size, name ); }
17 }
18 
26 template<GLenum Target = GL_ARRAY_BUFFER, GLenum Usage = GL_DYNAMIC_DRAW>
27 class buffer {
28 public:
35  GLuint name() const {
36  return m_object.name();
37  }
38 
49  GLsizeiptr size() const {
50  return m_size;
51  }
52 
60  void bind() {
61  check_gl_error( glBindBuffer( Target, m_object.name() ) );
62  }
63 
71  static void unbind() {
72  check_gl_error( glBindBuffer( Target, 0 ) );
73  }
74 
87  template<typename T>
88  void bind_range( GLuint index, GLintptr offset, T range_size ) {
89  check_gl_error( glBindBufferRange( Target, index, m_object.name(), offset, static_cast<GLsizeiptr>( range_size ) ) );
90  }
91 
111  template<typename T>
112  void data( T data_size, const GLvoid* data_ptr ) {
113  assert( data_size >= 0 );
114  m_size = static_cast<GLsizeiptr>( data_size );
115 
116  bind();
117  check_gl_error( glBufferData( Target, m_size, data_ptr, Usage ) );
118 
119  if( data_ptr ) {
120  sub_data( 0, m_size, data_ptr );
121  }
122  }
123 
138  template<typename T>
139  void sub_data( GLintptr offset, T data_size, const GLvoid* data_ptr ) {
140  if( m_size < offset + static_cast<GLsizeiptr>( data_size ) ) {
141  buffer new_buffer;
142 
143  new_buffer.data( offset + static_cast<GLsizeiptr>( data_size ), nullptr );
144  new_buffer.copy_sub_data( *this, 0, 0, m_size );
145 
146  std::swap( *this, new_buffer );
147  }
148 
149  bind();
150  check_gl_error( glBufferSubData( Target, offset, static_cast<GLsizeiptr>( data_size ), data_ptr ) );
151  unbind();
152  }
153 
175  template<GLenum SourceTarget, GLenum SourceUsage, typename V>
176  void copy_sub_data( const buffer<SourceTarget, SourceUsage>& source, GLintptr readoffset, GLintptr writeoffset, V data_size ) {
177  assert( data_size > 0 );
178  assert( source.size() >= readoffset + static_cast<GLsizeiptr>( data_size ) );
179 
180  if( m_size < writeoffset + static_cast<GLsizeiptr>( data_size ) ) {
181  buffer new_buffer;
182 
183  new_buffer.data( writeoffset + static_cast<GLsizeiptr>( data_size ), nullptr );
184 
185  if( m_size ) {
186  new_buffer.copy_sub_data( *this, 0, 0, m_size );
187  }
188 
189  std::swap( *this, new_buffer );
190  }
191 
192  check_gl_error( glBindBuffer( GL_COPY_READ_BUFFER, source.name() ) );
193  check_gl_error( glBindBuffer( GL_COPY_WRITE_BUFFER, name() ) );
194 
195  check_gl_error( glCopyBufferSubData( GL_COPY_READ_BUFFER, GL_COPY_WRITE_BUFFER, readoffset, writeoffset, static_cast<GLsizeiptr>( data_size ) ) );
196 
197  check_gl_error( glBindBuffer( GL_COPY_READ_BUFFER, 0 ) );
198  check_gl_error( glBindBuffer( GL_COPY_WRITE_BUFFER, 0 ) );
199  }
200 
211  template<typename T>
212  void get_sub_data( GLintptr offset, T data_size, GLvoid* data_ptr ) {
213  assert( m_size >= offset + static_cast<GLsizeiptr>( data_size ) );
214 
215  bind();
216  check_gl_error( glGetBufferSubData( Target, offset, static_cast<GLsizeiptr>( data_size ), data_ptr ) );
217  unbind();
218  }
219 
220 private:
222  GLsizeiptr m_size = 0;
223 };
224 
225 }
226 
void sub_data(GLintptr offset, T data_size, const GLvoid *data_ptr)
Upload a data range.
Definition: buffer.hpp:139
static void unbind()
Unbind the current buffer from the target.
Definition: buffer.hpp:71
void data(T data_size, const GLvoid *data_ptr)
Allocate storage and upload data.
Definition: buffer.hpp:112
Class encapsulating an OpenGL buffer object.
Definition: buffer.hpp:27
GLuint name() const
Retrieve the OpenGL name of this buffer.
Definition: buffer.hpp:35
void bind()
Bind this buffer to its target.
Definition: buffer.hpp:60
void get_sub_data(GLintptr offset, T data_size, GLvoid *data_ptr)
Read back data from the buffer.
Definition: buffer.hpp:212
void bind_range(GLuint index, GLintptr offset, T range_size)
Bind a range to an indexed target.
Definition: buffer.hpp:88
GLsizeiptr size() const
Retrieve the size of buffer.
Definition: buffer.hpp:49
Definition: buffer.hpp:12
void copy_sub_data(const buffer< SourceTarget, SourceUsage > &source, GLintptr readoffset, GLintptr writeoffset, V data_size)
Copy a data range from another buffer into this one.
Definition: buffer.hpp:176