jGL
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
vertexBuffer.h
Go to the documentation of this file.
1#ifndef VERTEXBUFFER
2#define VERTEXBUFFER
3
5#include <jGL/Vulkan/buffer.h>
6#include <memory>
7
8#include <cstddef>
9
10#define GLM_FORCE_RADIANS
11// will mostly auto align buffer data
12#define GLM_FORCE_DEFAULT_ALIGNED_GENTYPES
13#include <glm/glm.hpp>
14#include <glm/gtc/matrix_transform.hpp>
15
16namespace jGL::Vulkan
17{
18
20 {
21
22 /*
23
24 Abstract base class representing buffer objects
25
26 Use derived types, e.g. VertexBuffer<glm::vec4> to
27 represent a vertex buffer of a single vec4, that
28 can be used as e.g. layout(location=0) vec4 colour
29 or something
30
31 Templated concrete types are specialised, following the
32 constructor signature:
33
34 VertexBuffer<T>
35 (
36 const Device & device,
37 const Command & command,
38 std::vector<T> data,
39 VkVertexInputRate rate,
40 uint32_t binding = 0,
41 uint32_t location = 0
42 );
43
44 */
45
46 public:
47
49 :
51 {}
52
53 // an implementation must define these two layout properties
54 virtual std::vector<VkVertexInputBindingDescription> getBindingDescription() = 0;
55 virtual std::vector<VkVertexInputAttributeDescription> getArrtibuteDescriptions() = 0;
56
57 void setBinding(uint32_t b) { binding = b; }
58 void setLocation(uint32_t l) { location = l; }
59
60 const VkBuffer & getVkBuffer() const { return buffer->getVkBuffer(); }
61
62 protected:
63
64 const Device & device;
65
66 VkVertexInputRate inputRate;
67
68 uint32_t binding, location;
69
70 std::unique_ptr<Buffer> buffer;
71
72 };
73
74 template <class T>
76 {
77
78 public:
79
81 (
82 const Device & device,
83 const Command & command,
84 std::vector<T> data,
85 VkVertexInputRate rate,
86 uint32_t binding = 0,
87 uint32_t location = 0
88 )
90 {
91
92 this->inputRate = rate;
93 this->binding = binding;
94 this->location = location;
95
96 VkDeviceSize bufferSize = sizeof(T)*data.size();
97
98 Buffer stagingBuffer
99 (
100 device,
101 bufferSize,
102 VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
103 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT
104 );
105
106 stagingBuffer.copyMemory
107 (
108 0,
109 bufferSize,
110 data.data()
111 );
112
113 // create normal buffer
114
115 buffer = std::make_unique<Buffer>
116 (
117 device,
118 bufferSize,
119 VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,
120 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT
121 );
122
123 buffer->copy(stagingBuffer, command);
124 }
125
127 (
128 const Device & device,
129 const Command & command,
130 std::vector<T> data
131 )
132 {
133
134 VkDeviceSize bufferSize = sizeof(T)*data.size();
135
136 if (bufferSize > buffer->getVkDeviceSize())
137 {
138 buffer = std::make_unique<Buffer>
139 (
140 device,
141 bufferSize,
142 VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,
143 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT
144 );
145 }
146
147 Buffer stagingBuffer
148 (
149 device,
150 bufferSize,
151 VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
152 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT
153 );
154
155 stagingBuffer.copyMemory
156 (
157 0,
158 bufferSize,
159 data.data()
160 );
161
162 buffer->copy(stagingBuffer, command);
163 }
164
165 std::vector<VkVertexInputBindingDescription> getBindingDescription();
166 std::vector<VkVertexInputAttributeDescription> getArrtibuteDescriptions();
167
168 private:
169
170
171
172 };
173
174 template <>
175 class VertexBuffer<glm::vec4> : public VertexBufferObject
176 {
177
178 public:
179
181 (
182 const Device & device,
183 const Command & command,
184 std::vector<glm::vec4> data,
185 VkVertexInputRate rate,
186 uint32_t binding = 0,
187 uint32_t location = 0
188 )
190 {
191
192 // hate this, take a look to abstract
193
194 this->inputRate = rate;
195 this->binding = binding;
196 this->location = location;
197
198 VkDeviceSize bufferSize = sizeof(glm::vec4)*data.size();
199
200 Buffer stagingBuffer
201 (
202 device,
203 bufferSize,
204 VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
205 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT
206 );
207
208 stagingBuffer.copyMemory
209 (
210 0,
211 bufferSize,
212 data.data()
213 );
214
215 // create normal buffer
216
217 buffer = std::make_unique<Buffer>
218 (
219 device,
220 bufferSize,
221 VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,
222 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT
223 );
224
225 buffer->copy(stagingBuffer, command);
226 }
227
229 (
230 const Device & device,
231 const Command & command,
232 std::vector<glm::vec4> data
233 )
234 {
235
236 VkDeviceSize bufferSize = sizeof(glm::vec4)*data.size();
237
238 if (bufferSize > buffer->getVkDeviceSize())
239 {
240 buffer = std::make_unique<Buffer>
241 (
242 device,
243 bufferSize,
244 VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,
245 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT
246 );
247 }
248
249 Buffer stagingBuffer
250 (
251 device,
252 bufferSize,
253 VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
254 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT
255 );
256
257 stagingBuffer.copyMemory
258 (
259 0,
260 bufferSize,
261 data.data()
262 );
263
264 buffer->copy(stagingBuffer, command);
265 }
266
267
268 std::vector<VkVertexInputBindingDescription> getBindingDescription()
269 {
270 std::vector<VkVertexInputBindingDescription> bindingDescription{};
271 bindingDescription.resize(1);
272 bindingDescription[0].binding = binding;
273 bindingDescription[0].stride = sizeof(glm::vec4);
274 bindingDescription[0].inputRate = inputRate;
275 return bindingDescription;
276 }
277
278 std::vector<VkVertexInputAttributeDescription> getArrtibuteDescriptions()
279 {
280 std::vector<VkVertexInputAttributeDescription> attributeDescriptions{};
281 attributeDescriptions.resize(1);
282 // vertex binding, 0
283 attributeDescriptions[0].binding = binding;
284 // layout(location = 0)
285 attributeDescriptions[0].location = location;
286 // we have a vec4, so rgba
287 attributeDescriptions[0].format = VK_FORMAT_R32G32B32A32_SFLOAT;
288 // simply a glm::vec4, so no offset here
289 attributeDescriptions[0].offset = 0;
290
291 return attributeDescriptions;
292 }
293
294 };
295
296 template <>
297 class VertexBuffer<glm::vec2> : public VertexBufferObject
298 {
299
300 public:
301
303 (
304 const Device & device,
305 const Command & command,
306 std::vector<glm::vec2> data,
307 VkVertexInputRate rate,
308 uint32_t binding = 0,
309 uint32_t location = 0
310 )
312 {
313
314 // hate this, take a look to abstract
315
316 this->inputRate = rate;
317 this->binding = binding;
318 this->location = location;
319
320 VkDeviceSize bufferSize = sizeof(glm::vec2)*data.size();
321
322 Buffer stagingBuffer
323 (
324 device,
325 bufferSize,
326 VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
327 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT
328 );
329
330 stagingBuffer.copyMemory
331 (
332 0,
333 bufferSize,
334 data.data()
335 );
336
337 // create normal buffer
338
339 buffer = std::make_unique<Buffer>
340 (
341 device,
342 bufferSize,
343 VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,
344 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT
345 );
346
347 buffer->copy(stagingBuffer, command);
348 }
349
351 (
352 const Device & device,
353 const Command & command,
354 std::vector<glm::vec2> data
355 )
356 {
357
358 VkDeviceSize bufferSize = sizeof(glm::vec2)*data.size();
359
360 if (bufferSize > buffer->getVkDeviceSize())
361 {
362 buffer = std::make_unique<Buffer>
363 (
364 device,
365 bufferSize,
366 VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT,
367 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT
368 );
369 }
370
371 Buffer stagingBuffer
372 (
373 device,
374 bufferSize,
375 VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
376 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT
377 );
378
379 stagingBuffer.copyMemory
380 (
381 0,
382 bufferSize,
383 data.data()
384 );
385
386 buffer->copy(stagingBuffer, command);
387 }
388
389
390 std::vector<VkVertexInputBindingDescription> getBindingDescription()
391 {
392 std::vector<VkVertexInputBindingDescription> bindingDescription{};
393 bindingDescription.resize(1);
394 bindingDescription[0].binding = binding;
395 bindingDescription[0].stride = sizeof(glm::vec2);
396 bindingDescription[0].inputRate = inputRate;
397 return bindingDescription;
398 }
399
400 std::vector<VkVertexInputAttributeDescription> getArrtibuteDescriptions()
401 {
402 std::vector<VkVertexInputAttributeDescription> attributeDescriptions{};
403 attributeDescriptions.resize(1);
404 // vertex binding, 0
405 attributeDescriptions[0].binding = binding;
406 // layout(location = 0)
407 attributeDescriptions[0].location = location;
408 // we have a vec2, so rg
409 attributeDescriptions[0].format = VK_FORMAT_R32G32_SFLOAT;
410 // simply a glm::vec2, so no offset here
411 attributeDescriptions[0].offset = 0;
412
413 return attributeDescriptions;
414 }
415
416 };
417}
418
419#endif /* VERTEXBUFFER */
420
421/*
422
423 A note on how attributes and bindings function
424
425 https://gist.github.com/SaschaWillems/428d15ed4b5d71ead462bc63adffa93a
426
427 Shader
428
429 layout (location = 0) in vec3 inPos;
430 layout (location = 1) in vec3 inColor;
431 layout (location = 2) in vec3 inPosB;
432 layout (location = 3) in vec3 inColorB;
433
434 ...
435
436 Application
437
438 // Input state
439 std::array<VkVertexInputBindingDescription,2> vertexInputBinding = {};
440 vertexInputBinding[0].binding = 0;
441 vertexInputBinding[0].stride = sizeof(Vertex);
442 vertexInputBinding[0].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
443 vertexInputBinding[1].binding = 1;
444 vertexInputBinding[1].stride = sizeof(Vertex);
445 vertexInputBinding[1].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
446
447 // Inpute attribute bindings describe shader attribute locations and memory layouts
448 std::array<VkVertexInputAttributeDescription, 4> vertexInputAttributs;
449 vertexInputAttributs[0].binding = 0;
450 vertexInputAttributs[0].location = 0;
451 vertexInputAttributs[0].format = VK_FORMAT_R32G32B32_SFLOAT;
452 vertexInputAttributs[0].offset = offsetof(Vertex, position);
453 vertexInputAttributs[1].binding = 0;
454 vertexInputAttributs[1].location = 1;
455 vertexInputAttributs[1].format = VK_FORMAT_R32G32B32_SFLOAT;
456 vertexInputAttributs[1].offset = offsetof(Vertex, color);
457
458 vertexInputAttributs[2].binding = 1;
459 vertexInputAttributs[2].location = 2;
460 vertexInputAttributs[2].format = VK_FORMAT_R32G32B32_SFLOAT;
461 vertexInputAttributs[2].offset = offsetof(Vertex, position);
462 vertexInputAttributs[3].binding = 1;
463 vertexInputAttributs[3].location = 3;
464 vertexInputAttributs[3].format = VK_FORMAT_R32G32B32_SFLOAT;
465 vertexInputAttributs[3].offset = offsetof(Vertex, color);
466
467 // Vertex input state used for pipeline creation
468 VkPipelineVertexInputStateCreateInfo vertexInputState = {};
469 vertexInputState.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
470 vertexInputState.vertexBindingDescriptionCount = 2;
471 vertexInputState.pVertexBindingDescriptions = vertexInputBinding.data();
472 vertexInputState.vertexAttributeDescriptionCount = 4;
473 vertexInputState.pVertexAttributeDescriptions = vertexInputAttributs.data();
474
475 // Bind
476 vkCmdBindVertexBuffers(drawCmdBuffers[i], 0, 1, &vertices.buffer, offsets);
477 vkCmdBindVertexBuffers(drawCmdBuffers[i], 1, 1, &verticesB.buffer, offsets);
478
479*/
Definition buffer.h:12
void copyMemory(const VkDeviceSize offset, const VkDeviceSize size, void *data, const VkMemoryMapFlags=0)
Definition buffer.cpp:12
Definition command.h:12
Definition device.h:10
Definition vertexBuffer.h:20
void setLocation(uint32_t l)
Definition vertexBuffer.h:58
uint32_t binding
Definition vertexBuffer.h:68
void setBinding(uint32_t b)
Definition vertexBuffer.h:57
virtual std::vector< VkVertexInputAttributeDescription > getArrtibuteDescriptions()=0
const VkBuffer & getVkBuffer() const
Definition vertexBuffer.h:60
std::unique_ptr< Buffer > buffer
Definition vertexBuffer.h:70
virtual std::vector< VkVertexInputBindingDescription > getBindingDescription()=0
VertexBufferObject(const Device &device)
Definition vertexBuffer.h:48
uint32_t location
Definition vertexBuffer.h:68
const Device & device
Definition vertexBuffer.h:64
VkVertexInputRate inputRate
Definition vertexBuffer.h:66
void subData(const Device &device, const Command &command, std::vector< glm::vec2 > data)
Definition vertexBuffer.h:351
std::vector< VkVertexInputBindingDescription > getBindingDescription()
Definition vertexBuffer.h:390
std::vector< VkVertexInputAttributeDescription > getArrtibuteDescriptions()
Definition vertexBuffer.h:400
std::vector< VkVertexInputAttributeDescription > getArrtibuteDescriptions()
Definition vertexBuffer.h:278
std::vector< VkVertexInputBindingDescription > getBindingDescription()
Definition vertexBuffer.h:268
void subData(const Device &device, const Command &command, std::vector< glm::vec4 > data)
Definition vertexBuffer.h:229
Definition vertexBuffer.h:76
std::vector< VkVertexInputBindingDescription > getBindingDescription()
std::vector< VkVertexInputAttributeDescription > getArrtibuteDescriptions()
VertexBuffer(const Device &device, const Command &command, std::vector< T > data, VkVertexInputRate rate, uint32_t binding=0, uint32_t location=0)
Definition vertexBuffer.h:81
void subData(const Device &device, const Command &command, std::vector< T > data)
Definition vertexBuffer.h:127
Definition buffer.h:10