Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
GLArrayDataProxy.hpp
Go to the documentation of this file.
1/*
2 * Author: Sven Gothel <sgothel@jausoft.com>
3 * Copyright Gothel Software e.K.
4 *
5 * SPDX-License-Identifier: MIT
6 *
7 * This Source Code Form is subject to the terms of the MIT License
8 * If a copy of the MIT was not distributed with this file,
9 * you can obtain one at https://opensource.org/license/mit/.
10 */
11
12#ifndef GAMP_GLARRAYDATAPROXY_HPP_
13#define GAMP_GLARRAYDATAPROXY_HPP_
14
15#include <memory>
16
17#include <jau/darray.hpp>
18
21
22namespace gamp::render::gl::data {
23 using namespace gamp::render::gl;
24
25 /** \addtogroup Gamp_GLData
26 *
27 * @{
28 */
29
30 template<typename Value_type>
31 class GLArrayDataProxy;
32
33 template<typename Value_type>
34 using GLArrayDataProxyRef = std::shared_ptr<GLArrayDataProxy<Value_type>>;
35
36 /**
37 * Proxying a data buffer for GLArrayData usage of given template-type Value_type.
38 *
39 * The GL data type is determined via template type glType<Value_type>().
40 */
41 template<typename Value_type>
43 public:
46 typedef std::shared_ptr<proxy_t> proxy_ref;
47
49 typedef std::unique_ptr<buffer_t> buffer_ref;
50
51 ~GLArrayDataProxy() noexcept override = default;
52
53 /**
54 * Create a VBO, using a custom GLSL array attribute name, proxying the given data.
55 *
56 * The GL data type is determined via template type glType<Value_type>().
57 *
58 * This buffer is always {@link #sealed()}.
59 *
60 * @param name The custom name for the GL attribute, maybe null if gpuBufferTarget is {@link GL#GL_ELEMENT_ARRAY_BUFFER}
61 * @param componentsPerElement The array component number
62 * @param normalized Whether the data shall be normalized
63 * @param stride
64 * @param buffer the user define data, passed through by reference
65 * @param vboName
66 * @param vboOffset
67 * @param vboUsage GL_STREAM_DRAW, GL_STATIC_DRAW or GL_DYNAMIC_DRAW
68 * @param vboTarget GL_ARRAY_BUFFER or GL_ELEMENT_ARRAY_BUFFER
69 * @return the new create instance
70 * @throws GLException
71 */
72 static proxy_ref createGLSL(const std::string& name, GLsizei componentsPerElement, bool normalized, GLsizei stride,
73 buffer_t& buffer, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget) {
74 return std::make_shared<GLArrayDataProxy>(Private(),
75 name, componentsPerElement, normalized, stride, buffer,
76 /*isVertexAttribute=*/true, vboName, vboOffset, vboUsage, vboTarget);
77 }
78
79 /**
80 * Create a VBO, using a custom GLSL array attribute name, proxying the mapped data characteristics.
81 *
82 * The GL data type is determined via template type glType<Value_type>().
83 *
84 * This buffer is always {@link #sealed()}.
85 *
86 * @param name The custom name for the GL attribute, maybe null if gpuBufferTarget is {@link GL#GL_ELEMENT_ARRAY_BUFFER}
87 * @param componentsPerElement The array component number
88 * @param normalized Whether the data shall be normalized
89 * @param stride
90 * @param mappedElementCount
91 * @param vboName
92 * @param vboOffset
93 * @param vboUsage GL_STREAM_DRAW, GL_STATIC_DRAW or GL_DYNAMIC_DRAW
94 * @param vboTarget GL_ARRAY_BUFFER or GL_ELEMENT_ARRAY_BUFFER
95 * @return the new create instance
96 * @throws GLException
97 */
98 static proxy_ref createGLSL(const std::string& name, GLsizei componentsPerElement, bool normalized, GLsizei stride,
99 GLsizei mappedElementCount, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget) {
100 return std::make_shared<GLArrayDataProxy>(Private(),
101 name, componentsPerElement, normalized, stride, mappedElementCount,
102 /*isVertexAttribute=*/true, vboName, vboOffset, vboUsage, vboTarget);
103 }
104
105 /** Returns true if using mapped memory, otherwise false for client-data */
106 constexpr bool usesMappedMem() const noexcept { return !m_bufferptr; }
107 /** Returns true if using client-data, otherwise false for mapped memory */
108 constexpr bool usesClientMem() const noexcept { return m_bufferptr; }
109
110 std::string_view className() const noexcept override { return "GLArrayDataProxy"; }
111
112 const jau::type_info& classSignature() const noexcept override {
114 }
115
117
118 /** Returns client-data pointer at current position if usesClientMem(), otherwise nullptr */
119 const void* data() const noexcept override { return usesClientMem() ? reinterpret_cast<const void*>(m_bufferptr->position_ptr()) : nullptr; }
120
121 GLsizei elemCount() const noexcept override {
122 if( usesClientMem() ) {
123 if( m_sealed ) {
124 return static_cast<GLsizei>( (m_bufferptr->limit() * m_bytesPerComp) / m_strideB );
125 } else {
126 return static_cast<GLsizei>( (m_bufferptr->position() * m_bytesPerComp) / m_strideB );
127 }
128 } else {
129 return m_mappedElemCount;
130 }
131 }
132
133 glmemsize_t byteCount() const noexcept override {
134 if( usesClientMem() ) {
135 if( m_sealed ) {
136 return m_bufferptr->limit() * m_bytesPerComp ;
137 } else {
138 return m_bufferptr->position() * m_bytesPerComp ;
139 }
140 } else {
142 }
143 }
144
145 glmemsize_t byteCapacity() const noexcept override {
146 if( usesClientMem() ) {
147 return m_bufferptr->capacity() * m_bytesPerComp;
148 } else {
149 return 0;
150 }
151 }
152
153 std::string elemStatsToString() const noexcept override {
154 const size_t elem_limit = ( (usesClientMem() ? m_bufferptr->limit() : 0) * m_bytesPerComp ) / m_strideB;
155 return jau::format_string("sealed %d, elements %s cnt, [%s pos .. %s rem .. %s lim .. %s cap]",
156 sealed(),
157 jau::to_decstring(elemCount()).c_str(),
160 jau::to_decstring(elem_limit).c_str(),
162 }
163
164 void destroy(GL& gl) override {
166 if( usesClientMem() ) {
167 m_bufferptr->clear();
168 }
169 }
170
171 std::string toString() const noexcept override { return toString(false); }
172 virtual std::string toString(bool withData) const noexcept {
173 std::string r(className());
174 r.append("[").append(m_name)
175 .append(", location ").append(std::to_string(m_location))
176 .append(", isVertexAttribute ").append(std::to_string(m_isVertexAttr))
177 .append(", dataType ").append(jau::to_hexstring(m_compType))
178 .append(", compsPerElem ").append(std::to_string(m_compsPerElement))
179 .append(", stride ").append(std::to_string(m_strideB)).append("b ").append(std::to_string(m_strideL)).append("c")
180 .append(", mappedElemCount ").append(std::to_string(m_mappedElemCount))
181 .append(", ").append(elemStatsToString());
182 if( withData ) {
183 r.append(", buffer ").append(usesClientMem()?m_bufferptr->toString():"nil");
184 }
185 r.append(", vboEnabled ").append(std::to_string(m_vboEnabled))
186 .append(", vboName ").append(std::to_string(m_vboName))
187 .append(", vboUsage ").append(jau::to_hexstring(m_vboUsage))
188 .append(", vboTarget ").append(jau::to_hexstring(m_vboTarget))
189 .append(", vboOffset ").append(std::to_string(m_vboOffset))
190 .append(", alive ").append(std::to_string(m_alive)).append("]");
191 return r;
192 }
193
194 /** The Buffer holding the data, nullptr if !usesClientMem() */
195 const buffer_t* buffer_ptr() const noexcept { return m_bufferptr; }
196 /** The Buffer holding the data, nullptr if !usesClientMem() */
197 buffer_t* buffer_ptr() noexcept { return m_bufferptr; }
198
199 /**
200 * Returns the element position.
201 * <p>
202 * On element consist out of {@link #getCompsPerElem()} components.
203 * </p>
204 * @see bytePosition()
205 * @see getElemCount()
206 * @see remainingElems()
207 * @see getElemCapacity()
208 */
209 glmemsize_t elemPosition() const noexcept {
210 if( usesClientMem() ) {
211 return ( m_bufferptr->position() * m_bytesPerComp ) / m_strideB ;
212 } else {
213 return m_mappedElemCount;
214 }
215 }
216
217 /**
218 * The current number of remaining elements.
219 * <p>
220 * On element consist out of {@link #getCompsPerElem()} components.
221 * </p>
222 * Returns the number of elements between the current position and the limit, i.e. remaining elements to write in this buffer.
223 * @see remainingBytes()
224 * @see getElemCount()
225 * @see elemPosition()
226 * @see getElemCapacity()
227 */
228 glmemsize_t remainingElems() const noexcept {
229 if( usesClientMem() ) {
230 return (m_bufferptr->remaining() * m_bytesPerComp) / m_strideB;
231 } else {
232 return 0;
233 }
234 }
235
236 /**
237 * Return the element capacity.
238 * <p>
239 * On element consist out of {@link #getCompsPerElem()} components.
240 * </p>
241 * @see getByteCapacity()
242 * @see getElemCount()
243 * @see elemPosition()
244 * @see remainingElems()
245 */
246 glmemsize_t elemCapacity() const noexcept {
247 if( usesClientMem() ) {
248 return ( m_bufferptr->capacity() * m_bytesPerComp ) / m_strideB ;
249 } else {
250 return 0;
251 }
252 }
253
254 /**
255 * Returns the bytes position.
256 * @see elemPosition
257 * @see getByteCount
258 * @see remainingElems
259 * @see getElemCapacity
260 */
261 glmemsize_t bytePosition() const noexcept {
262 if( usesClientMem() ) {
263 return m_bufferptr->position() * m_bytesPerComp;
264 } else {
266 }
267 }
268
269 /**
270 * The current number of remaining bytes.
271 * <p>
272 * Returns the number of bytes between the current position and the limit, i.e. remaining bytes to write in this buffer.
273 * </p>
274 * @see remainingElems
275 * @see getByteCount
276 * @see bytePosition
277 * @see getByteCapacity
278 */
279 glmemsize_t remainingBytes() const noexcept {
280 if( usesClientMem() ) {
281 return m_bufferptr->remaining() * m_bytesPerComp;
282 } else {
283 return 0;
284 }
285 }
286
287 /**
288 * Returns a string with detailed buffer fill stats.
289 */
290 std::string fillStatsToString() const noexcept {
291 const size_t cnt_bytes = byteCount();
292 const size_t cap_bytes = byteCapacity();
293 const float filled = (float)cnt_bytes / (float)cap_bytes;
294 return jau::format_string("elements %s cnt / %s cap, bytes %s cnt / %s cap, filled %.1f%%, left %.1f%%",
295 jau::to_decstring(elemCount()).c_str(),
297 jau::to_decstring(cnt_bytes).c_str(),
298 jau::to_decstring(cap_bytes).c_str(),
299 filled * 100.0f, (1.0f - filled) * 100.0f);
300 }
301
302 //
303 // OpenGL pass through funcs
304 //
305
306 void glBufferData(const GL& gl) const noexcept {
308 }
309
310 protected:
311 struct Private{ explicit Private() = default; };
312
313 public:
314 // using passing through client buffer_t
316 const std::string& name, GLsizei componentsPerElement,
317 bool normalized, GLsizei stride, buffer_t& buffer,
318 bool isVertexAttribute, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
319 : GLArrayData(GLArrayData::Private(), name, componentsPerElement, glType<value_type>(), jau::make_ctti<value_type>(),
320 normalized, stride, /*mappedElementCount=*/0, isVertexAttribute, vboName, vboOffset, vboUsage, vboTarget),
321 m_bufferptr( &buffer )
322 { }
323
324 // using memory mapped elements
326 const std::string& name, GLsizei componentsPerElement,
327 bool normalized, GLsizei stride, GLsizei mappedElementCount,
328 bool isVertexAttribute, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
329 : GLArrayData(GLArrayData::Private(), name, componentsPerElement, glType<value_type>(), jau::make_ctti<value_type>(),
331 m_bufferptr(nullptr)
332 {
333 if( 0 == mappedElementCount ) {
334 throw jau::IllegalArgumentError("mappedElementCount:=" + std::to_string(mappedElementCount)
335 + " specified for memory map:\n\t" + toStringImpl(), E_FILE_LINE);
336 }
337 }
338
339 protected:
340 // mutable
342 };
343
344 /**@}*/
345
346} // namespace gamp::render::gl
347
348#endif /* GAMP_GLARRAYDATAPROXY_HPP_ */
#define E_FILE_LINE
Proxying a data buffer for GLArrayData usage of given template-type Value_type.
glmemsize_t elemCapacity() const noexcept
Return the element capacity.
constexpr bool usesMappedMem() const noexcept
Returns true if using mapped memory, otherwise false for client-data.
const buffer_t * buffer_ptr() const noexcept
The Buffer holding the data, nullptr if !usesClientMem()
glmemsize_t byteCount() const noexcept override
Returns the byte position (written elements) if not sealed() or the byte limit (available to read) af...
std::string_view className() const noexcept override
Returns class name of implementing class.
glmemsize_t remainingElems() const noexcept
The current number of remaining elements.
static proxy_ref createGLSL(const std::string &name, GLsizei componentsPerElement, bool normalized, GLsizei stride, buffer_t &buffer, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
static proxy_ref createGLSL(const std::string &name, GLsizei componentsPerElement, bool normalized, GLsizei stride, GLsizei mappedElementCount, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
Create a VBO, using a custom GLSL array attribute name, proxying the mapped data characteristics.
glmemsize_t bytePosition() const noexcept
Returns the bytes position.
const GLArrayDataProxyRef< value_type > shared()
virtual std::string toString(bool withData) const noexcept
buffer_t * buffer_ptr() noexcept
The Buffer holding the data, nullptr if !usesClientMem()
std::string fillStatsToString() const noexcept
Returns a string with detailed buffer fill stats.
std::string elemStatsToString() const noexcept override
Returns a string with detailed buffer element stats, i.e.
GLsizei elemCount() const noexcept override
Returns the element position (written elements) if not sealed() or the element limit (available to re...
GLArrayDataProxy(Private, const std::string &name, GLsizei componentsPerElement, bool normalized, GLsizei stride, buffer_t &buffer, bool isVertexAttribute, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
jau::darray< value_type, glmemsize_t > buffer_t
std::string toString() const noexcept override
glmemsize_t elemPosition() const noexcept
Returns the element position.
glmemsize_t remainingBytes() const noexcept
The current number of remaining bytes.
const void * data() const noexcept override
Returns client-data pointer at current position if usesClientMem(), otherwise nullptr.
~GLArrayDataProxy() noexcept override=default
const jau::type_info & classSignature() const noexcept override
Returns type signature of implementing class.
void glBufferData(const GL &gl) const noexcept
constexpr bool usesClientMem() const noexcept
Returns true if using client-data, otherwise false for mapped memory.
GLArrayDataProxy(Private, const std::string &name, GLsizei componentsPerElement, bool normalized, GLsizei stride, GLsizei mappedElementCount, bool isVertexAttribute, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
glmemsize_t byteCapacity() const noexcept override
Return the capacity in bytes.
constexpr bool sealed() const noexcept
Returns true if data has been sealed (flipped to read), otherwise false (writing mode).
void glBufferData(const GL &, glmemsize_t size) const noexcept
Sends (creates, updates) the data to the bound vboName buffer, see glBindBuffer().
constexpr uintptr_t vboOffset() const noexcept
The VBO buffer offset if isVBO()
constexpr const std::string & name() const noexcept
The name of the reflecting shader array attribute.
constexpr GLuint vboName() const noexcept
The VBO name or 0 if not a VBO.
constexpr bool isVertexAttribute() const noexcept
Returns true if this data set is intended for a GLSL vertex shader attribute, otherwise false,...
GLsizei m_strideB
stride in bytes; strideB >= compsPerElement * bytesPerComp
constexpr GLenum vboUsage() const noexcept
The VBO usage or 0 if not a VBO.
GLArrayData(Private, const std::string &name, GLsizei componentsPerElement, GLenum componentType, jau::type_info compTypeSignature, bool normalized, GLsizei stride, GLsizei mappedElementCount, bool isVertexAttribute, GLuint vboName, uintptr_t vboOffset, GLenum vboUsage, GLenum vboTarget)
Private ctor for shared_ptr.
std::string toStringImpl() const noexcept
GLsizei stride() const noexcept
constexpr bool normalized() const noexcept
True, if GL shall normalize fixed point data while converting them into float.
GLsizei m_strideL
stride in logical components
std::shared_ptr< ChildT > shared_from_base()
constexpr GLenum vboTarget() const noexcept
The VBO target or 0 if not a VBO.
Implementation of a dynamic linear array storage, aka vector, including relative positional access.
Definition darray.hpp:153
Generic type information using either Runtime type information (RTTI) or Compile time type informatio...
const jau::type_info & static_ctti() noexcept
Returns a static global reference of make_ctti<T>(true) w/ identity instance.
GLsizeiptr glmemsize_t
Compatible with ssize_t.
Definition GLBuffers.hpp:26
std::shared_ptr< GLArrayDataProxy< Value_type > > GLArrayDataProxyRef
constexpr GLenum glType() noexcept
Definition GLTypes.hpp:46
std::string to_hexstring(value_type const &v, const bool skipLeading0x=false) noexcept
Produce a lower-case hexadecimal string representation with leading 0x in MSB of the given pointer.
std::string format_string(const char *format,...)
Returns a string according to printf() formatting rules and variable number of arguments following th...
std::string to_decstring(const value_type &v, const char separator=',', const nsize_t width=0) noexcept
Produce a decimal string representation of an integral integer value.
__pack(...): Produces MSVC, clang and gcc compatible lead-in and -out macros.
Definition backtrace.hpp:32
STL namespace.
uint8_t Value_type