Riyyi
5 months ago
12 changed files with 374 additions and 52 deletions
@ -0,0 +1,195 @@
|
||||
/*
|
||||
* Copyright (C) 2024 Riyyi |
||||
* |
||||
* SPDX-License-Identifier: MIT |
||||
*/ |
||||
|
||||
#include <cstdint> // int32_t, uint8_t, uintt32_t |
||||
|
||||
#include "glad/glad.h" |
||||
#include "glm/ext/matrix_float2x2.hpp" // glm::mat2 |
||||
#include "glm/ext/matrix_float3x3.hpp" // glm::mat3 |
||||
#include "glm/ext/vector_float4.hpp" // glm::vec4 |
||||
|
||||
#include "inferno/render/buffer.h" |
||||
#include "inferno/render/shader-storage-buffer.h" |
||||
|
||||
namespace Inferno { |
||||
|
||||
ShaderStorageBuffer::ShaderStorageBuffer(s) |
||||
{ |
||||
// Get maximum uniformbuffer bindings the GPU supports
|
||||
int32_t maxBindingPoints = 0; |
||||
glGetIntegerv(GL_MAX_SHADER_STORAGE_BUFFER_BINDINGS, &maxBindingPoints); |
||||
m_maxBindingPoints = static_cast<uint8_t>(maxBindingPoints); |
||||
} |
||||
|
||||
ShaderStorageBuffer::~ShaderStorageBuffer() |
||||
{ |
||||
} |
||||
|
||||
// -----------------------------------------
|
||||
|
||||
// https://stackoverflow.com/questions/56512216#answer-56513136
|
||||
void ShaderStorageBuffer::setLayout(std::string_view blockName, uint8_t bindingPoint, uint32_t shaderID) |
||||
{ |
||||
VERIFY(bindingPoint < m_maxBindingPoints, |
||||
"shader storage buffer exceeded binding points: {}/{}", bindingPoint, m_maxBindingPoints); |
||||
|
||||
if (!exists(blockName)) { |
||||
m_blocks[blockName.data()] = {}; |
||||
} |
||||
|
||||
BufferBlock& block = m_blocks[blockName.data()]; |
||||
block.bindingPoint = bindingPoint; |
||||
block.memberOffsets.clear(); |
||||
|
||||
glBindBuffer(GL_SHADER_STORAGE_BUFFER, block.id); |
||||
|
||||
// Get the shader block index
|
||||
uint32_t resourceIndex = glGetProgramResourceIndex(shaderID, GL_SHADER_STORAGE_BLOCK, blockName.data()); |
||||
VERIFY(resourceIndex != GL_INVALID_INDEX, "block doesnt exist in shader: {}::{}", blockName, shaderID); |
||||
|
||||
// Get the amount of member variables
|
||||
uint32_t prop = GL_NUM_ACTIVE_VARIABLES; |
||||
int32_t memberCount; |
||||
glGetProgramResourceiv(shaderID, GL_SHADER_STORAGE_BLOCK, resourceIndex, 1, &prop, 1, nullptr, &memberCount); |
||||
|
||||
// Get the indices of the members
|
||||
prop = GL_ACTIVE_VARIABLES; |
||||
std::vector<int32_t> members(memberCount); |
||||
glGetProgramResourceiv(shaderID, GL_SHADER_STORAGE_BLOCK, resourceIndex, 1, &prop, (int32_t)members.size(), nullptr, members.data()); |
||||
|
||||
// Reserve memory for the names of the members
|
||||
int32_t memberNameSize; |
||||
glGetProgramInterfaceiv(shaderID, GL_BUFFER_VARIABLE, GL_MAX_NAME_LENGTH, &memberNameSize); |
||||
std::vector<char> memberNameBuffer(memberNameSize); |
||||
|
||||
int32_t lastOffset = 0; |
||||
int32_t lastType = 0; |
||||
for (int32_t i = 0; i < memberCount; i++) { |
||||
|
||||
// Get name of buffer variable
|
||||
int32_t stringLength; |
||||
glGetProgramResourceName(shaderID, GL_BUFFER_VARIABLE, members[i], memberNameSize, &stringLength, memberNameBuffer.data()); |
||||
std::string memberName = std::string(memberNameBuffer.begin(), memberNameBuffer.begin() + stringLength); |
||||
|
||||
// Get the other data needed for computing
|
||||
|
||||
uint32_t props[7] = { |
||||
GL_OFFSET, // 0
|
||||
GL_TYPE, // 1
|
||||
GL_ARRAY_SIZE, // 2
|
||||
GL_ARRAY_STRIDE, // 3
|
||||
GL_MATRIX_STRIDE, // 4
|
||||
GL_TOP_LEVEL_ARRAY_SIZE, // 5
|
||||
GL_TOP_LEVEL_ARRAY_STRIDE, // 6
|
||||
}; |
||||
int32_t params[7]; |
||||
glGetProgramResourceiv(shaderID, GL_BUFFER_VARIABLE, members[i], 7, props, 7, nullptr, params); |
||||
|
||||
// ruc::error("{}", memberName);
|
||||
// ruc::error("\n Offset: {}, type: {:#x}, arraySize: {}, arrayStride: {},\n matrixStride: {}, top level size: {}, top level stride: {}\n",
|
||||
// params[0], params[1], params[2], params[3], params[4], params[5], params[6]);
|
||||
|
||||
// Array of structs
|
||||
if (params[5] != 1 && params[6 != 0]) { |
||||
size_t bracketOpen = memberName.find_first_of('['); |
||||
size_t bracketClose = memberName.find_first_of(']'); |
||||
std::string memberNameBegin = memberName.substr(0, bracketOpen); // name: myArray[0].member -> myArray
|
||||
std::string memberNameMember = memberName.substr(bracketClose + 1); // name: myArray[0].member -> .member
|
||||
for (int32_t j = 0; j < params[5]; ++j) { |
||||
lastOffset = params[0] + (j * params[6]); // calc offset
|
||||
lastType = params[1]; |
||||
|
||||
// Only add the member variant the first time its encountered
|
||||
if (j == 0 && block.memberOffsets.find(memberNameBegin) == block.memberOffsets.end()) { |
||||
block.memberOffsets.emplace(memberNameBegin, lastOffset); // name: myArray
|
||||
} |
||||
|
||||
// Only add the index variant the first time its encountered
|
||||
std::string memberNameIndex = memberNameBegin + "[" + std::to_string(j) + "]"; // name: myArray -> myArray[i]
|
||||
if (block.memberOffsets.find(memberNameIndex) == block.memberOffsets.end()) { |
||||
block.memberOffsets.emplace(memberNameIndex, lastOffset); |
||||
} |
||||
|
||||
block.memberOffsets.emplace(memberNameIndex + memberNameMember, // name: myArray -> myArray[i].member
|
||||
lastOffset); |
||||
} |
||||
} |
||||
// Array of primitives
|
||||
else if (params[2] != 1 && params[3] != 0) { |
||||
std::string memberNameBegin = memberName.substr(0, memberName.find_first_of('[')); // name: myArray[0] -> myArray
|
||||
for (int32_t j = 0; j < params[2]; ++j) { |
||||
lastOffset = params[0] + (j * params[3]); // calc offset
|
||||
lastType = params[1]; |
||||
|
||||
// Only add the member variant the first time its encountered
|
||||
if (j == 0) { |
||||
block.memberOffsets.emplace(memberNameBegin, lastOffset); // name: myArray
|
||||
} |
||||
|
||||
block.memberOffsets.emplace(memberNameBegin + "[" + std::to_string(j) + "]", // name: myArray -> myArray[i]
|
||||
lastOffset); |
||||
} |
||||
} |
||||
// Matrix case
|
||||
else if (params[4] != 0) { |
||||
lastType = params[1]; |
||||
lastOffset = params[0]; |
||||
block.memberOffsets.emplace(memberName, params[0]); |
||||
} |
||||
else { |
||||
lastType = params[1]; |
||||
lastOffset = params[0]; |
||||
block.memberOffsets.emplace(memberName, params[0]); |
||||
} |
||||
} |
||||
|
||||
block.size = lastOffset + BufferElement::getGLTypeSize(lastType); |
||||
|
||||
glBindBuffer(GL_SHADER_STORAGE_BUFFER, 0); |
||||
|
||||
// Results
|
||||
// for (auto [k, v] : block.memberOffsets) {
|
||||
// ruc::error("{}:{}", k, v);
|
||||
// }
|
||||
} |
||||
|
||||
void ShaderStorageBuffer::create(std::string_view blockName) |
||||
{ |
||||
VERIFY(exists(blockName), "shader storage buffer block doesnt exist"); |
||||
|
||||
BufferBlock& block = m_blocks[blockName.data()]; |
||||
|
||||
if (block.id != 0) { |
||||
glDeleteBuffers(1, &block.id); |
||||
} |
||||
|
||||
// Allocate buffer
|
||||
block.id = UINT_MAX; |
||||
glGenBuffers(1, &block.id); |
||||
glBindBuffer(GL_SHADER_STORAGE_BUFFER, block.id); |
||||
glBufferData(GL_SHADER_STORAGE_BUFFER, block.size, NULL, GL_DYNAMIC_DRAW); |
||||
glBindBuffer(GL_SHADER_STORAGE_BUFFER, 0); |
||||
|
||||
// Bind buffer to binding point
|
||||
glBindBufferBase(GL_SHADER_STORAGE_BUFFER, block.bindingPoint, block.id); |
||||
} |
||||
|
||||
void ShaderStorageBuffer::setValue(std::string_view blockName, std::string_view member, bool value) |
||||
{ |
||||
setValue(blockName, member, static_cast<uint32_t>(value), sizeof(uint32_t)); |
||||
} |
||||
|
||||
void ShaderStorageBuffer::setValue(std::string_view blockName, std::string_view member, glm::mat2 value) |
||||
{ |
||||
setValue(blockName, member, static_cast<glm::mat4>(value), sizeof(glm::vec4) * 2); |
||||
} |
||||
|
||||
void ShaderStorageBuffer::setValue(std::string_view blockName, std::string_view member, glm::mat3 value) |
||||
{ |
||||
setValue(blockName, member, static_cast<glm::mat4>(value), sizeof(glm::vec4) * 3); |
||||
} |
||||
|
||||
} // namespace Inferno
|
@ -0,0 +1,63 @@
|
||||
/*
|
||||
* Copyright (C) 2024 Riyyi |
||||
* |
||||
* SPDX-License-Identifier: MIT |
||||
*/ |
||||
|
||||
#pragma once |
||||
|
||||
#include <cstddef> // size_t |
||||
#include <cstdint> // uint8_t, uint32_t |
||||
#include <string_view> |
||||
#include <unordered_map> |
||||
|
||||
#include "glad/glad.h" |
||||
#include "glm/ext/matrix_float2x2.hpp" // glm::mat2 |
||||
#include "glm/ext/matrix_float3x3.hpp" // glm::mat3 |
||||
#include "ruc/singleton.h" |
||||
|
||||
#define CHECK_SET_CALL(blockName, member) \ |
||||
VERIFY(exists(blockName), "shader storage buffer block doesnt exist: {}", blockName); \
|
||||
const BufferBlock& block = m_blocks[blockName.data()]; \
|
||||
VERIFY(block.memberOffsets.find(member.data()) != block.memberOffsets.end(), \
|
||||
"shader storage buffer member doesnt exist: {}", member); |
||||
|
||||
namespace Inferno { |
||||
|
||||
struct BufferBlock { |
||||
uint32_t id { 0 }; |
||||
uint32_t size { 0 }; |
||||
uint8_t bindingPoint { 0 }; |
||||
std::unordered_map<std::string, uint32_t> memberOffsets {}; |
||||
}; |
||||
|
||||
class ShaderStorageBuffer final : public ruc::Singleton<ShaderStorageBuffer> { // Shader Storage Buffer Object, SSBO
|
||||
public: |
||||
ShaderStorageBuffer(s); |
||||
virtual ~ShaderStorageBuffer(); |
||||
|
||||
void setLayout(std::string_view blockName, uint8_t bindingPoint, uint32_t shaderID); |
||||
void create(std::string_view blockName); |
||||
|
||||
bool exists(std::string_view blockName) const { return m_blocks.find(blockName.data()) != m_blocks.end(); } |
||||
|
||||
template<typename T> // Capture value by reference, instead of decaying to pointer
|
||||
void setValue(std::string_view blockName, std::string_view member, T&& value, size_t size = 0) |
||||
{ |
||||
CHECK_SET_CALL(blockName, member); |
||||
|
||||
glBindBuffer(GL_SHADER_STORAGE_BUFFER, block.id); |
||||
glBufferSubData(GL_SHADER_STORAGE_BUFFER, block.memberOffsets.at(member.data()), (size) ? size : sizeof(T), &value); |
||||
glBindBuffer(GL_SHADER_STORAGE_BUFFER, 0); |
||||
} |
||||
// Exceptions:
|
||||
void setValue(std::string_view blockName, std::string_view member, bool value); |
||||
void setValue(std::string_view blockName, std::string_view member, glm::mat2 value); |
||||
void setValue(std::string_view blockName, std::string_view member, glm::mat3 value); |
||||
|
||||
private: |
||||
uint8_t m_maxBindingPoints { 0 }; |
||||
std::unordered_map<std::string, BufferBlock> m_blocks; |
||||
}; |
||||
|
||||
} // namespace Inferno
|
@ -0,0 +1,24 @@
|
||||
/*
|
||||
* Copyright (C) 2024 Riyyi |
||||
* |
||||
* SPDX-License-Identifier: MIT |
||||
*/ |
||||
|
||||
#pragma once |
||||
|
||||
#include "glm/ext/vector_float3.hpp" // glm::vec3 |
||||
|
||||
namespace Inferno { |
||||
|
||||
// Shader storage block layouts, using std430 memory layout rules
|
||||
|
||||
#define MAX_DIRECTIONAL_LIGHTS 4 |
||||
struct alignas(16) DirectionalLightBlock { |
||||
alignas(16) glm::vec3 direction { 0 }; |
||||
|
||||
alignas(16) glm::vec3 ambient { 0 }; |
||||
alignas(16) glm::vec3 diffuse { 0 }; |
||||
alignas(16) glm::vec3 specular { 0 }; |
||||
}; |
||||
|
||||
} // namespace Inferno
|
Loading…
Reference in new issue