123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282 |
- /**
- * Copyright (c) 2006-2020 LOVE Development Team
- *
- * This software is provided 'as-is', without any express or implied
- * warranty. In no event will the authors be held liable for any damages
- * arising from the use of this software.
- *
- * Permission is granted to anyone to use this software for any purpose,
- * including commercial applications, and to alter it and redistribute it
- * freely, subject to the following restrictions:
- *
- * 1. The origin of this software must not be misrepresented; you must not
- * claim that you wrote the original software. If you use this software
- * in a product, an acknowledgment in the product documentation would be
- * appreciated but is not required.
- * 2. Altered source versions must be plainly marked as such, and must not be
- * misrepresented as being the original software.
- * 3. This notice may not be removed or altered from any source distribution.
- **/
- #include "Buffer.h"
- #include "Graphics.h"
- #include "common/memory.h"
- namespace love
- {
- namespace graphics
- {
- love::Type Buffer::type("GraphicsBuffer", &Object::type);
- Buffer::Buffer(Graphics *gfx, const Settings &settings, const std::vector<DataDeclaration> &bufferformat, size_t size, size_t arraylength)
- : arrayLength(0)
- , arrayStride(0)
- , size(size)
- , usageFlags(settings.usageFlags)
- , dataUsage(settings.dataUsage)
- , mapped(false)
- {
- if (size == 0 && arraylength == 0)
- throw love::Exception("Size or array length must be specified.");
- if (bufferformat.size() == 0)
- throw love::Exception("Data format must contain values.");
- const auto &caps = gfx->getCapabilities();
- bool supportsGLSL3 = caps.features[Graphics::FEATURE_GLSL3];
- bool indexbuffer = settings.usageFlags & BUFFERUSAGEFLAG_INDEX;
- bool vertexbuffer = settings.usageFlags & BUFFERUSAGEFLAG_VERTEX;
- bool texelbuffer = settings.usageFlags & BUFFERUSAGEFLAG_TEXEL;
- bool storagebuffer = settings.usageFlags & BUFFERUSAGEFLAG_SHADER_STORAGE;
- bool copydest = settings.usageFlags & BUFFERUSAGEFLAG_COPY_DEST;
- if (!indexbuffer && !vertexbuffer && !texelbuffer && !storagebuffer)
- throw love::Exception("Buffer must be created with at least one buffer type (index, vertex, texel, or shaderstorage).");
- if (texelbuffer && !caps.features[Graphics::FEATURE_TEXEL_BUFFER])
- throw love::Exception("Texel buffers are not supported on this system.");
- if (storagebuffer && !caps.features[Graphics::FEATURE_GLSL4])
- throw love::Exception("Shader Storage buffers are not supported on this system (GLSL 4 support is necessary.)");
- if (copydest && dataUsage == BUFFERDATAUSAGE_STREAM)
- throw love::Exception("Buffers created with 'stream' data usage cannot be used as a copy destination.");
- size_t offset = 0;
- size_t stride = 0;
- size_t structurealignment = 1;
- for (const DataDeclaration &decl : bufferformat)
- {
- DataMember member(decl);
- DataFormat format = member.decl.format;
- const DataFormatInfo &info = member.info;
- if (indexbuffer)
- {
- if (format != DATAFORMAT_UINT16 && format != DATAFORMAT_UINT32)
- throw love::Exception("Index buffers only support uint16 and uint32 data types.");
- if (bufferformat.size() > 1)
- throw love::Exception("Index buffers only support a single value per element.");
- if (decl.arrayLength > 0)
- throw love::Exception("Arrays are not supported in index buffers.");
- }
- if (vertexbuffer)
- {
- if (decl.arrayLength > 0)
- throw love::Exception("Arrays are not supported in vertex buffers.");
- if (info.isMatrix)
- throw love::Exception("Matrix types are not supported in vertex buffers.");
- if (info.baseType == DATA_BASETYPE_BOOL)
- throw love::Exception("Bool types are not supported in vertex buffers.");
- if ((info.baseType == DATA_BASETYPE_INT || info.baseType == DATA_BASETYPE_UINT) && !supportsGLSL3)
- throw love::Exception("Integer vertex attribute data types require GLSL 3 support.");
- if (decl.name.empty())
- throw love::Exception("Vertex buffer attributes must have a name.");
- }
- if (texelbuffer)
- {
- if (format != bufferformat[0].format)
- throw love::Exception("All values in a texel buffer must have the same format.");
- if (decl.arrayLength > 0)
- throw love::Exception("Arrays are not supported in texel buffers.");
- if (info.isMatrix)
- throw love::Exception("Matrix types are not supported in texel buffers.");
- if (info.baseType == DATA_BASETYPE_BOOL)
- throw love::Exception("Bool types are not supported in texel buffers.");
- if (info.components == 3)
- throw love::Exception("3-component formats are not supported in texel buffers.");
- if (info.baseType == DATA_BASETYPE_SNORM)
- throw love::Exception("Signed normalized formats are not supported in texel buffers.");
- }
- size_t memberoffset = offset;
- size_t membersize = member.info.size;
- // Storage buffers are always treated as being an array of a structure.
- // The structure's contents are the buffer format declaration.
- if (storagebuffer)
- {
- // TODO: We can support these.
- if (decl.arrayLength > 0)
- throw love::Exception("Arrays are not currently supported in shader storage buffers.");
- if (info.baseType == DATA_BASETYPE_BOOL)
- throw love::Exception("Bool types are not supported in shader storage buffers.");
- if (info.baseType == DATA_BASETYPE_UNORM || info.baseType == DATA_BASETYPE_SNORM)
- throw love::Exception("Normalized formats are not supported in shader storage buffers.");
- size_t alignment = 1;
- // GLSL's std430 packing rules. We also assume all matrices are
- // column-major.
- if (info.isMatrix)
- alignment = info.matrixRows * info.componentSize;
- else
- alignment = info.components * info.componentSize;
- structurealignment = std::max(structurealignment, alignment);
- memberoffset = alignUp(memberoffset, alignment);
- if (memberoffset != offset && (indexbuffer || vertexbuffer || texelbuffer))
- throw love::Exception("Cannot create Buffer:\nInternal alignment of member '%s' is preventing Buffer from being created as both a shader storage buffer and other buffer types\nMember byte offset needed for shader storage buffer: %d\nMember byte offset needed for other buffer types: %d",
- member.decl.name.c_str(), memberoffset, offset);
- }
- member.offset = memberoffset;
- member.size = membersize;
- offset = member.offset + member.size;
- dataMembers.push_back(member);
- }
- stride = alignUp(offset, structurealignment);
- if (storagebuffer && (indexbuffer || vertexbuffer || texelbuffer))
- {
- if (stride != offset)
- throw love::Exception("Cannot create Buffer:\nBuffer used as a shader storage buffer would have a different number of bytes per array element (%d) than when used as other buffer types (%d)",
- stride, offset);
- }
- if (storagebuffer && stride > SHADER_STORAGE_BUFFER_MAX_STRIDE)
- throw love::Exception("Shader storage buffers cannot have more than %d bytes within each array element.", SHADER_STORAGE_BUFFER_MAX_STRIDE);
- if (size != 0)
- {
- size_t remainder = size % stride;
- if (remainder > 0)
- size += stride - remainder;
- arraylength = size / stride;
- }
- else
- {
- size = arraylength * stride;
- }
- this->arrayStride = stride;
- this->arrayLength = arraylength;
- this->size = size;
- if (texelbuffer && arraylength * dataMembers.size() > caps.limits[Graphics::LIMIT_TEXEL_BUFFER_SIZE])
- throw love::Exception("Cannot create texel buffer: total number of values in the buffer (%d * %d) is too large for this system (maximum %d).",
- (int) dataMembers.size(), (int) arraylength, caps.limits[Graphics::LIMIT_TEXEL_BUFFER_SIZE]);
- }
- Buffer::~Buffer()
- {
- }
- int Buffer::getDataMemberIndex(const std::string &name) const
- {
- for (size_t i = 0; i < dataMembers.size(); i++)
- {
- if (dataMembers[i].decl.name == name)
- return (int) i;
- }
- return -1;
- }
- std::vector<Buffer::DataDeclaration> Buffer::getCommonFormatDeclaration(CommonFormat format)
- {
- switch (format)
- {
- case CommonFormat::NONE:
- return {};
- case CommonFormat::XYf:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC2 }
- };
- case CommonFormat::XYZf:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC3 }
- };
- case CommonFormat::RGBAub:
- return {
- { getConstant(ATTRIB_COLOR), DATAFORMAT_UNORM8_VEC4 }
- };
- case CommonFormat::STf_RGBAub:
- return {
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_COLOR), DATAFORMAT_UNORM8_VEC4 },
- };
- case CommonFormat::STPf_RGBAub:
- return {
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_FLOAT_VEC3 },
- { getConstant(ATTRIB_COLOR), DATAFORMAT_UNORM8_VEC4 },
- };
- case CommonFormat::XYf_STf:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_FLOAT_VEC2 },
- };
- case CommonFormat::XYf_STPf:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_FLOAT_VEC3 },
- };
- case CommonFormat::XYf_STf_RGBAub:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_COLOR), DATAFORMAT_UNORM8_VEC4 },
- };
- case CommonFormat::XYf_STus_RGBAub:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_UNORM16_VEC2 },
- { getConstant(ATTRIB_COLOR), DATAFORMAT_UNORM8_VEC4 },
- };
- case CommonFormat::XYf_STPf_RGBAub:
- return {
- { getConstant(ATTRIB_POS), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_TEXCOORD), DATAFORMAT_FLOAT_VEC2 },
- { getConstant(ATTRIB_COLOR), DATAFORMAT_UNORM8_VEC4 },
- };
- }
- return {};
- }
- } // graphics
- } // love
|