#ifndef COMPONENTS_STD140_UBO_H #define COMPONENTS_STD140_UBO_H #include #include #include #include #include #include #include #include namespace std140 { struct Mat4 { using Value = osg::Matrixf; Value mValue; static constexpr size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "mat4"; }; struct Vec4 { using Value = osg::Vec4f; Value mValue; static constexpr size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "vec4"; }; struct Vec3 { using Value = osg::Vec3f; Value mValue; static constexpr std::size_t sAlign = 4 * sizeof(osg::Vec3f::value_type); static constexpr std::string_view sTypeName = "vec3"; }; struct Vec2 { using Value = osg::Vec2f; Value mValue; static constexpr std::size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "vec2"; }; struct Float { using Value = float; Value mValue; static constexpr std::size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "float"; }; struct Int { using Value = std::int32_t; Value mValue; static constexpr std::size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "int"; }; struct UInt { using Value = std::uint32_t; Value mValue; static constexpr std::size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "uint"; }; struct Bool { using Value = std::int32_t; Value mValue; static constexpr std::size_t sAlign = sizeof(Value); static constexpr std::string_view sTypeName = "bool"; }; template class UBO { private: template struct contains : std::bool_constant<(std::is_base_of_v || ...)> { }; static_assert((contains() && ...)); static constexpr size_t roundUpRemainder(size_t x, size_t multiple) { size_t remainder = x % multiple; if (remainder == 0) return 0; return multiple - remainder; } template static constexpr std::size_t getOffset() { bool found = false; std::size_t size = 0; ((found = found || std::is_same_v, size += (found ? 0 : sizeof(typename CArgs::Value) + roundUpRemainder(size, CArgs::sAlign))), ...); return size + roundUpRemainder(size, T::sAlign); } public: static constexpr size_t getGPUSize() { std::size_t size = 0; ((size += (sizeof(typename CArgs::Value) + roundUpRemainder(size, CArgs::sAlign))), ...); return size; } static std::string getDefinition(const std::string& name) { std::string structDefinition = "struct " + name + " {\n"; ((structDefinition += (" " + std::string(CArgs::sTypeName) + " " + std::string(CArgs::sName) + ";\n")), ...); return structDefinition + "};"; } using BufferType = std::array; using TupleType = std::tuple; template typename T::Value& get() { return std::get(mData).mValue; } template const typename T::Value& get() const { return std::get(mData).mValue; } void copyTo(BufferType& buffer) const { const auto copy = [&](const auto& v) { static_assert(std::is_standard_layout_v>); constexpr std::size_t offset = getOffset>(); std::memcpy(buffer.data() + offset, &v.mValue, sizeof(v.mValue)); }; std::apply([&](const auto&... v) { (copy(v), ...); }, mData); } const auto& getData() const { return mData; } private: std::tuple mData; }; } #endif