mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-26 15:56:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			160 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			160 lines
		
	
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef COMPONENTS_STD140_UBO_H
 | |
| #define COMPONENTS_STD140_UBO_H
 | |
| 
 | |
| #include <osg/Matrixf>
 | |
| #include <osg/Vec2f>
 | |
| #include <osg/Vec4f>
 | |
| 
 | |
| #include <array>
 | |
| #include <cstdint>
 | |
| #include <cstring>
 | |
| #include <string>
 | |
| #include <string_view>
 | |
| #include <tuple>
 | |
| 
 | |
| 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... CArgs>
 | |
|     class UBO
 | |
|     {
 | |
|     private:
 | |
|         template <typename T, typename... Args>
 | |
|         struct contains : std::bool_constant<(std::is_base_of_v<Args, T> || ...)>
 | |
|         {
 | |
|         };
 | |
| 
 | |
|         static_assert((contains<CArgs, Mat4, Vec4, Vec3, Vec2, Float, Int, UInt, Bool>() && ...));
 | |
| 
 | |
|         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 <class T>
 | |
|         static constexpr std::size_t getOffset()
 | |
|         {
 | |
|             bool found = false;
 | |
|             std::size_t size = 0;
 | |
|             ((found = found || std::is_same_v<T, CArgs>,
 | |
|                  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<char, getGPUSize()>;
 | |
|         using TupleType = std::tuple<CArgs...>;
 | |
| 
 | |
|         template <class T>
 | |
|         typename T::Value& get()
 | |
|         {
 | |
|             return std::get<T>(mData).mValue;
 | |
|         }
 | |
| 
 | |
|         template <class T>
 | |
|         const typename T::Value& get() const
 | |
|         {
 | |
|             return std::get<T>(mData).mValue;
 | |
|         }
 | |
| 
 | |
|         void copyTo(BufferType& buffer) const
 | |
|         {
 | |
|             const auto copy = [&](const auto& v) {
 | |
|                 static_assert(std::is_standard_layout_v<std::decay_t<decltype(v.mValue)>>);
 | |
|                 constexpr std::size_t offset = getOffset<std::decay_t<decltype(v)>>();
 | |
|                 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<CArgs...> mData;
 | |
|     };
 | |
| }
 | |
| 
 | |
| #endif
 |