mirror of
https://github.com/OpenMW/openmw.git
synced 2025-10-24 04:26:37 +00:00
1122 lines
36 KiB
C++
1122 lines
36 KiB
C++
#include "technique.hpp"
|
|
|
|
#include <array>
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
#include <osg/Texture1D>
|
|
#include <osg/Texture2D>
|
|
#include <osg/Texture3D>
|
|
|
|
#include <SDL_opengl_glext.h>
|
|
|
|
#include <components/debug/debuglog.hpp>
|
|
#include <components/files/conversion.hpp>
|
|
#include <components/misc/strings/algorithm.hpp>
|
|
#include <components/resource/imagemanager.hpp>
|
|
#include <components/stereo/multiview.hpp>
|
|
#include <components/vfs/manager.hpp>
|
|
|
|
#include "parseconstants.hpp"
|
|
|
|
namespace
|
|
{
|
|
struct ProxyTextureData
|
|
{
|
|
osg::Texture::WrapMode wrap_s = osg::Texture::CLAMP_TO_EDGE;
|
|
osg::Texture::WrapMode wrap_t = osg::Texture::CLAMP_TO_EDGE;
|
|
osg::Texture::WrapMode wrap_r = osg::Texture::CLAMP_TO_EDGE;
|
|
osg::Texture::FilterMode min_filter = osg::Texture::LINEAR_MIPMAP_LINEAR;
|
|
osg::Texture::FilterMode mag_filter = osg::Texture::LINEAR;
|
|
osg::Texture::InternalFormatMode compression = osg::Texture::USE_IMAGE_DATA_FORMAT;
|
|
std::optional<int> source_format;
|
|
std::optional<int> source_type;
|
|
std::optional<int> internal_format;
|
|
};
|
|
}
|
|
|
|
namespace fx
|
|
{
|
|
VFS::Path::Normalized Technique::makeFileName(std::string_view name)
|
|
{
|
|
std::string fileName(name);
|
|
fileName += '.';
|
|
fileName += Technique::sExt;
|
|
VFS::Path::Normalized result(Technique::sSubdir);
|
|
result /= fileName;
|
|
return result;
|
|
}
|
|
|
|
Technique::Technique(const VFS::Manager& vfs, Resource::ImageManager& imageManager,
|
|
VFS::Path::NormalizedView fileName, std::string name, int width, int height, bool ubo, bool supportsNormals)
|
|
: mName(std::move(name))
|
|
, mFilePath(fileName)
|
|
, mLastModificationTime(std::filesystem::file_time_type::clock::now())
|
|
, mWidth(width)
|
|
, mHeight(height)
|
|
, mVFS(vfs)
|
|
, mImageManager(imageManager)
|
|
, mUBO(ubo)
|
|
, mSupportsNormals(supportsNormals)
|
|
{
|
|
clear();
|
|
}
|
|
|
|
void Technique::clear()
|
|
{
|
|
mTextures.clear();
|
|
mStatus = Status::Uncompiled;
|
|
mValid = false;
|
|
mHDR = false;
|
|
mNormals = false;
|
|
mLights = false;
|
|
mEnabled = true;
|
|
mPassMap.clear();
|
|
mPasses.clear();
|
|
mPassKeys.clear();
|
|
mDefinedUniforms.clear();
|
|
mRenderTargets.clear();
|
|
mLastAppliedType = Pass::Type::None;
|
|
mFlags = 0;
|
|
mShared.clear();
|
|
mAuthor = {};
|
|
mDescription = {};
|
|
mVersion = {};
|
|
mGLSLExtensions.clear();
|
|
mGLSLVersion = (mUBO || Stereo::getMultiview()) ? 330 : 120;
|
|
mGLSLProfile.clear();
|
|
mDynamic = false;
|
|
}
|
|
|
|
std::string Technique::getBlockWithLineDirective()
|
|
{
|
|
auto block = mLexer->getLastJumpBlock();
|
|
std::string content = std::string(block.content);
|
|
|
|
content = "\n#line " + std::to_string(block.line + 1) + "\n" + std::string(block.content) + "\n";
|
|
return content;
|
|
}
|
|
|
|
Technique::UniformMap::iterator Technique::findUniform(const std::string& name)
|
|
{
|
|
return std::find_if(mDefinedUniforms.begin(), mDefinedUniforms.end(),
|
|
[&name](const auto& uniform) { return uniform->mName == name; });
|
|
}
|
|
|
|
bool Technique::compile()
|
|
{
|
|
clear();
|
|
|
|
if (std::ranges::count(mFilePath.value(), '/') > 1)
|
|
{
|
|
Log(Debug::Error) << "Could not load technique, invalid location '" << mFilePath << "'";
|
|
|
|
mStatus = Status::File_Not_exists;
|
|
return false;
|
|
}
|
|
|
|
if (!mVFS.exists(mFilePath))
|
|
{
|
|
Log(Debug::Error) << "Could not load technique, file does not exist '" << mFilePath << "'";
|
|
|
|
mStatus = Status::File_Not_exists;
|
|
return false;
|
|
}
|
|
|
|
try
|
|
{
|
|
std::string source(std::istreambuf_iterator<char>(*mVFS.get(getFileName())), {});
|
|
|
|
parse(std::move(source));
|
|
|
|
if (mPassKeys.empty())
|
|
error("no pass list found, ensure you define one in a 'technique' block");
|
|
|
|
int swaps = 0;
|
|
|
|
for (auto& name : mPassKeys)
|
|
{
|
|
auto it = mPassMap.find(name);
|
|
|
|
if (it == mPassMap.end())
|
|
error(
|
|
Misc::StringUtils::format("pass '%s' was found in the pass list, but there was no matching "
|
|
"'fragment', 'vertex' or 'compute' block",
|
|
std::string(name)));
|
|
|
|
if (mLastAppliedType != Pass::Type::None && mLastAppliedType != it->second->mType)
|
|
{
|
|
swaps++;
|
|
if (swaps == 2)
|
|
Log(Debug::Warning) << "compute and pixel shaders are being swapped multiple times in shader "
|
|
"chain, this can lead to serious performance drain.";
|
|
}
|
|
else
|
|
mLastAppliedType = it->second->mType;
|
|
|
|
if (Stereo::getMultiview())
|
|
{
|
|
mGLSLExtensions.insert("GL_OVR_multiview");
|
|
mGLSLExtensions.insert("GL_OVR_multiview2");
|
|
mGLSLExtensions.insert("GL_EXT_texture_array");
|
|
}
|
|
|
|
it->second->compile(*this, mShared);
|
|
|
|
if (!it->second->mTarget.empty())
|
|
{
|
|
auto rtIt = mRenderTargets.find(it->second->mTarget);
|
|
if (rtIt == mRenderTargets.end())
|
|
error(Misc::StringUtils::format("target '%s' not defined", std::string(it->second->mTarget)));
|
|
}
|
|
|
|
mPasses.emplace_back(it->second);
|
|
}
|
|
|
|
if (mPasses.empty())
|
|
error("invalid pass list, no passes defined for technique");
|
|
|
|
mValid = true;
|
|
}
|
|
catch (const std::runtime_error& e)
|
|
{
|
|
clear();
|
|
mStatus = Status::Parse_Error;
|
|
|
|
mLastError = "Failed parsing technique '" + getName() + "' " + e.what();
|
|
|
|
Log(Debug::Error) << mLastError;
|
|
}
|
|
|
|
return mValid;
|
|
}
|
|
|
|
std::string Technique::getName() const
|
|
{
|
|
return mName;
|
|
}
|
|
|
|
bool Technique::setLastModificationTime(std::filesystem::file_time_type timeStamp)
|
|
{
|
|
const bool isDirty = timeStamp != mLastModificationTime;
|
|
mLastModificationTime = timeStamp;
|
|
return isDirty;
|
|
}
|
|
|
|
[[noreturn]] void Technique::error(const std::string& msg)
|
|
{
|
|
mLexer->error(msg);
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Shared>()
|
|
{
|
|
if (!mLexer->jump())
|
|
error(Misc::StringUtils::format("unterminated 'shared' block, expected closing brackets"));
|
|
|
|
if (!mShared.empty())
|
|
error("repeated 'shared' block, only one allowed per technique file");
|
|
|
|
mShared = getBlockWithLineDirective();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Technique>()
|
|
{
|
|
if (!mPassKeys.empty())
|
|
error("exactly one 'technique' block can appear per file");
|
|
|
|
while (!isNext<Lexer::Close_bracket>() && !isNext<Lexer::Eof>())
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
auto key = std::get<Lexer::Literal>(mToken).value;
|
|
|
|
expect<Lexer::Equal>();
|
|
|
|
if (key == "passes")
|
|
mPassKeys = parseLiteralList<Lexer::Comma>();
|
|
else if (key == "version")
|
|
mVersion = parseString();
|
|
else if (key == "description")
|
|
mDescription = parseString();
|
|
else if (key == "author")
|
|
mAuthor = parseString();
|
|
else if (key == "glsl_version")
|
|
mGLSLVersion = std::max(mGLSLVersion, parseInteger());
|
|
else if (key == "flags")
|
|
mFlags = parseFlags();
|
|
else if (key == "hdr")
|
|
mHDR = parseBool();
|
|
else if (key == "pass_normals")
|
|
mNormals = parseBool() && mSupportsNormals;
|
|
else if (key == "pass_lights")
|
|
mLights = parseBool();
|
|
else if (key == "glsl_profile")
|
|
{
|
|
expect<Lexer::String>();
|
|
mGLSLProfile = std::string(std::get<Lexer::String>(mToken).value);
|
|
}
|
|
else if (key == "glsl_extensions")
|
|
{
|
|
for (const auto& ext : parseLiteralList<Lexer::Comma>())
|
|
mGLSLExtensions.emplace(ext);
|
|
}
|
|
else if (key == "dynamic")
|
|
mDynamic = parseBool();
|
|
else
|
|
error(Misc::StringUtils::format("unexpected key '%s'", std::string{ key }));
|
|
|
|
expect<Lexer::SemiColon>();
|
|
}
|
|
|
|
if (mPassKeys.empty())
|
|
error("pass list in 'technique' block cannot be empty.");
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Render_Target>()
|
|
{
|
|
if (mRenderTargets.count(mBlockName))
|
|
error(Misc::StringUtils::format("redeclaration of render target '%s'", std::string(mBlockName)));
|
|
|
|
fx::Types::RenderTarget rt;
|
|
rt.mTarget->setTextureSize(mWidth, mHeight);
|
|
rt.mTarget->setSourceFormat(GL_RGB);
|
|
rt.mTarget->setInternalFormat(GL_RGB);
|
|
rt.mTarget->setSourceType(GL_UNSIGNED_BYTE);
|
|
rt.mTarget->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
|
|
rt.mTarget->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
|
|
rt.mTarget->setName(std::string(mBlockName));
|
|
|
|
while (!isNext<Lexer::Close_bracket>() && !isNext<Lexer::Eof>())
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
auto key = std::get<Lexer::Literal>(mToken).value;
|
|
|
|
expect<Lexer::Equal>();
|
|
|
|
if (key == "min_filter")
|
|
rt.mTarget->setFilter(osg::Texture2D::MIN_FILTER, parseFilterMode());
|
|
else if (key == "mag_filter")
|
|
rt.mTarget->setFilter(osg::Texture2D::MAG_FILTER, parseFilterMode());
|
|
else if (key == "wrap_s")
|
|
rt.mTarget->setWrap(osg::Texture2D::WRAP_S, parseWrapMode());
|
|
else if (key == "wrap_t")
|
|
rt.mTarget->setWrap(osg::Texture2D::WRAP_T, parseWrapMode());
|
|
else if (key == "width_ratio")
|
|
rt.mSize.mWidthRatio = parseFloat();
|
|
else if (key == "height_ratio")
|
|
rt.mSize.mHeightRatio = parseFloat();
|
|
else if (key == "width")
|
|
rt.mSize.mWidth = parseInteger();
|
|
else if (key == "height")
|
|
rt.mSize.mHeight = parseInteger();
|
|
else if (key == "internal_format")
|
|
rt.mTarget->setInternalFormat(parseInternalFormat());
|
|
else if (key == "source_type")
|
|
rt.mTarget->setSourceType(parseSourceType());
|
|
else if (key == "source_format")
|
|
rt.mTarget->setSourceFormat(parseSourceFormat());
|
|
else if (key == "mipmaps")
|
|
rt.mMipMap = parseBool();
|
|
else if (key == "clear_color")
|
|
rt.mClearColor = parseVec<osg::Vec4f, Lexer::Vec4>();
|
|
else
|
|
error(Misc::StringUtils::format("unexpected key '%s'", std::string(key)));
|
|
|
|
expect<Lexer::SemiColon>();
|
|
}
|
|
|
|
mRenderTargets.emplace(mBlockName, std::move(rt));
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Vertex>()
|
|
{
|
|
if (!mLexer->jump())
|
|
error(Misc::StringUtils::format("unterminated 'vertex' block, expected closing brackets"));
|
|
|
|
auto& pass = mPassMap[mBlockName];
|
|
|
|
if (!pass)
|
|
pass = std::make_shared<fx::Pass>();
|
|
|
|
pass->mName = mBlockName;
|
|
|
|
if (pass->mCompute)
|
|
error(Misc::StringUtils::format("'compute' block already defined. Usage is ambiguous."));
|
|
else if (!pass->mVertex)
|
|
pass->mVertex = new osg::Shader(osg::Shader::VERTEX, getBlockWithLineDirective());
|
|
else
|
|
error(Misc::StringUtils::format("duplicate vertex shader for block '%s'", std::string(mBlockName)));
|
|
|
|
pass->mType = Pass::Type::Pixel;
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Fragment>()
|
|
{
|
|
if (!mLexer->jump())
|
|
error(Misc::StringUtils::format("unterminated 'fragment' block, expected closing brackets"));
|
|
|
|
auto& pass = mPassMap[mBlockName];
|
|
|
|
if (!pass)
|
|
pass = std::make_shared<fx::Pass>();
|
|
|
|
pass->mUBO = mUBO;
|
|
pass->mName = mBlockName;
|
|
|
|
if (pass->mCompute)
|
|
error(Misc::StringUtils::format("'compute' block already defined. Usage is ambiguous."));
|
|
else if (!pass->mFragment)
|
|
pass->mFragment = new osg::Shader(osg::Shader::FRAGMENT, getBlockWithLineDirective());
|
|
else
|
|
error(Misc::StringUtils::format("duplicate vertex shader for block '%s'", std::string(mBlockName)));
|
|
|
|
pass->mType = Pass::Type::Pixel;
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Compute>()
|
|
{
|
|
if (!mLexer->jump())
|
|
error(Misc::StringUtils::format("unterminated 'compute' block, expected closing brackets"));
|
|
|
|
auto& pass = mPassMap[mBlockName];
|
|
|
|
if (!pass)
|
|
pass = std::make_shared<fx::Pass>();
|
|
|
|
pass->mName = mBlockName;
|
|
|
|
if (pass->mFragment)
|
|
error(Misc::StringUtils::format("'fragment' block already defined. Usage is ambiguous."));
|
|
else if (pass->mVertex)
|
|
error(Misc::StringUtils::format("'vertex' block already defined. Usage is ambiguous."));
|
|
else if (!pass->mFragment)
|
|
pass->mCompute = new osg::Shader(osg::Shader::COMPUTE, getBlockWithLineDirective());
|
|
else
|
|
error(Misc::StringUtils::format("duplicate vertex shader for block '%s'", std::string(mBlockName)));
|
|
|
|
pass->mType = Pass::Type::Compute;
|
|
}
|
|
|
|
template <class T>
|
|
void Technique::parseSampler()
|
|
{
|
|
if (findUniform(std::string(mBlockName)) != mDefinedUniforms.end())
|
|
error(Misc::StringUtils::format("redeclaration of uniform '%s'", std::string(mBlockName)));
|
|
|
|
ProxyTextureData proxy;
|
|
osg::ref_ptr<osg::Texture> sampler;
|
|
|
|
constexpr bool is1D = std::is_same_v<Lexer::Sampler_1D, T>;
|
|
constexpr bool is3D = std::is_same_v<Lexer::Sampler_3D, T>;
|
|
|
|
Types::SamplerType type;
|
|
|
|
while (!isNext<Lexer::Close_bracket>() && !isNext<Lexer::Eof>())
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
auto key = asLiteral();
|
|
|
|
expect<Lexer::Equal>();
|
|
|
|
if (!is1D && key == "min_filter")
|
|
proxy.min_filter = parseFilterMode();
|
|
else if (!is1D && key == "mag_filter")
|
|
proxy.mag_filter = parseFilterMode();
|
|
else if (key == "wrap_s")
|
|
proxy.wrap_s = parseWrapMode();
|
|
else if (key == "wrap_t")
|
|
proxy.wrap_t = parseWrapMode();
|
|
else if (is3D && key == "wrap_r")
|
|
proxy.wrap_r = parseWrapMode();
|
|
else if (key == "compression")
|
|
proxy.compression = parseCompression();
|
|
else if (key == "source_type")
|
|
proxy.source_type = parseSourceType();
|
|
else if (key == "source_format")
|
|
proxy.source_format = parseSourceFormat();
|
|
else if (key == "internal_format")
|
|
proxy.internal_format = parseInternalFormat();
|
|
else if (key == "source")
|
|
{
|
|
expect<Lexer::String>();
|
|
const osg::ref_ptr<osg::Image> image
|
|
= mImageManager.getImage(VFS::Path::Normalized(std::get<Lexer::String>(mToken).value), is3D);
|
|
if constexpr (is1D)
|
|
{
|
|
type = Types::SamplerType::Texture_1D;
|
|
sampler = new osg::Texture1D(image);
|
|
}
|
|
else if constexpr (is3D)
|
|
{
|
|
type = Types::SamplerType::Texture_3D;
|
|
sampler = new osg::Texture3D(image);
|
|
}
|
|
else
|
|
{
|
|
type = Types::SamplerType::Texture_2D;
|
|
sampler = new osg::Texture2D(image);
|
|
}
|
|
}
|
|
else
|
|
error(Misc::StringUtils::format("unexpected key '%s'", std::string{ key }));
|
|
|
|
expect<Lexer::SemiColon>();
|
|
}
|
|
if (!sampler)
|
|
error(Misc::StringUtils::format(
|
|
"%s '%s' requires a filename", std::string(T::repr), std::string{ mBlockName }));
|
|
|
|
if (!is1D)
|
|
{
|
|
sampler->setFilter(osg::Texture::MIN_FILTER, proxy.min_filter);
|
|
sampler->setFilter(osg::Texture::MAG_FILTER, proxy.mag_filter);
|
|
}
|
|
if (is3D)
|
|
sampler->setWrap(osg::Texture::WRAP_R, proxy.wrap_r);
|
|
sampler->setWrap(osg::Texture::WRAP_S, proxy.wrap_s);
|
|
sampler->setWrap(osg::Texture::WRAP_T, proxy.wrap_t);
|
|
sampler->setInternalFormatMode(proxy.compression);
|
|
if (proxy.internal_format.has_value())
|
|
sampler->setInternalFormat(proxy.internal_format.value());
|
|
if (proxy.source_type.has_value())
|
|
sampler->setSourceType(proxy.source_type.value());
|
|
if (proxy.internal_format.has_value())
|
|
sampler->setSourceFormat(proxy.internal_format.value());
|
|
sampler->setName(std::string{ mBlockName });
|
|
sampler->setResizeNonPowerOfTwoHint(false);
|
|
|
|
mTextures.emplace_back(sampler);
|
|
|
|
std::shared_ptr<Types::UniformBase> uniform = std::make_shared<Types::UniformBase>();
|
|
uniform->mSamplerType = type;
|
|
uniform->mName = std::string(mBlockName);
|
|
mDefinedUniforms.emplace_back(std::move(uniform));
|
|
}
|
|
|
|
template <class SrcT, class T>
|
|
SrcT Technique::getUniformValue()
|
|
{
|
|
constexpr bool isVec
|
|
= std::is_same_v<osg::Vec2f, SrcT> || std::is_same_v<osg::Vec3f, SrcT> || std::is_same_v<osg::Vec4f, SrcT>;
|
|
constexpr bool isFloat = std::is_same_v<float, SrcT>;
|
|
constexpr bool isInt = std::is_same_v<int, SrcT>;
|
|
constexpr bool isBool = std::is_same_v<bool, SrcT>;
|
|
|
|
static_assert(isVec || isFloat || isInt || isBool, "Unsupported type");
|
|
|
|
if constexpr (isVec)
|
|
{
|
|
return parseVec<SrcT, T>();
|
|
}
|
|
else if constexpr (isFloat)
|
|
{
|
|
return parseFloat();
|
|
}
|
|
else if constexpr (isInt)
|
|
{
|
|
return parseInteger();
|
|
}
|
|
else if constexpr (isBool)
|
|
{
|
|
return parseBool();
|
|
}
|
|
|
|
error(Misc::StringUtils::format("failed setting uniform type"));
|
|
}
|
|
|
|
template <class SrcT, class T>
|
|
void Technique::parseUniform()
|
|
{
|
|
if (findUniform(std::string(mBlockName)) != mDefinedUniforms.end())
|
|
error(Misc::StringUtils::format("redeclaration of uniform '%s'", std::string(mBlockName)));
|
|
|
|
std::shared_ptr<Types::UniformBase> uniform = std::make_shared<Types::UniformBase>();
|
|
Types::Uniform<SrcT> data = Types::Uniform<SrcT>();
|
|
|
|
while (!isNext<Lexer::Close_bracket>() && !isNext<Lexer::Eof>())
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
auto key = asLiteral();
|
|
|
|
expect<Lexer::Equal>("error parsing config for uniform block");
|
|
|
|
if (key == "default")
|
|
{
|
|
data.mDefault = getUniformValue<SrcT, T>();
|
|
}
|
|
else if (key == "size")
|
|
{
|
|
if constexpr (std::is_same_v<bool, SrcT>)
|
|
error("bool arrays currently unsupported");
|
|
|
|
int size = parseInteger();
|
|
if (size > 1)
|
|
data.mArray = std::vector<SrcT>(size);
|
|
}
|
|
else if (key == "min")
|
|
{
|
|
data.mMin = getUniformValue<SrcT, T>();
|
|
}
|
|
else if (key == "max")
|
|
{
|
|
data.mMax = getUniformValue<SrcT, T>();
|
|
}
|
|
else if (key == "step")
|
|
uniform->mStep = parseFloat();
|
|
else if (key == "static")
|
|
uniform->mStatic = parseBool();
|
|
else if (key == "description")
|
|
{
|
|
uniform->mDescription = parseString();
|
|
}
|
|
else if (key == "header")
|
|
{
|
|
uniform->mHeader = parseString();
|
|
}
|
|
else if (key == "display_name")
|
|
{
|
|
uniform->mDisplayName = parseString();
|
|
}
|
|
else if (key == "widget_type")
|
|
{
|
|
parseWidgetType<SrcT, T>(data);
|
|
}
|
|
else
|
|
error(Misc::StringUtils::format("unexpected key '%s'", std::string{ key }));
|
|
|
|
expect<Lexer::SemiColon>();
|
|
}
|
|
|
|
if (data.isArray())
|
|
uniform->mStatic = false;
|
|
|
|
uniform->mName = std::string(mBlockName);
|
|
uniform->mData = data;
|
|
uniform->mTechniqueName = mName;
|
|
|
|
if (data.mArray)
|
|
{
|
|
if constexpr (!std::is_same_v<bool, SrcT>)
|
|
{
|
|
if (auto cached = Settings::ShaderManager::get().getValue<std::vector<SrcT>>(mName, uniform->mName))
|
|
uniform->setValue(cached.value());
|
|
}
|
|
}
|
|
else if (auto cached = Settings::ShaderManager::get().getValue<SrcT>(mName, uniform->mName))
|
|
{
|
|
uniform->setValue(cached.value());
|
|
}
|
|
|
|
mDefinedUniforms.emplace_back(std::move(uniform));
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Sampler_1D>()
|
|
{
|
|
parseSampler<Lexer::Sampler_1D>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Sampler_2D>()
|
|
{
|
|
parseSampler<Lexer::Sampler_2D>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Sampler_3D>()
|
|
{
|
|
parseSampler<Lexer::Sampler_3D>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Uniform_Bool>()
|
|
{
|
|
parseUniform<bool, bool>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Uniform_Float>()
|
|
{
|
|
parseUniform<float, float>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Uniform_Int>()
|
|
{
|
|
parseUniform<int, int>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Uniform_Vec2>()
|
|
{
|
|
parseUniform<osg::Vec2f, Lexer::Vec2>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Uniform_Vec3>()
|
|
{
|
|
parseUniform<osg::Vec3f, Lexer::Vec3>();
|
|
}
|
|
|
|
template <>
|
|
void Technique::parseBlockImp<Lexer::Uniform_Vec4>()
|
|
{
|
|
parseUniform<osg::Vec4f, Lexer::Vec4>();
|
|
}
|
|
|
|
template <class T>
|
|
void Technique::expect(const std::string& err)
|
|
{
|
|
mToken = mLexer->next();
|
|
if (!std::holds_alternative<T>(mToken))
|
|
{
|
|
if (err.empty())
|
|
error(Misc::StringUtils::format("Expected %s", std::string(T::repr)));
|
|
else
|
|
error(Misc::StringUtils::format("%s. Expected %s", err, std::string(T::repr)));
|
|
}
|
|
}
|
|
|
|
template <class T, class T2>
|
|
void Technique::expect(const std::string& err)
|
|
{
|
|
mToken = mLexer->next();
|
|
if (!std::holds_alternative<T>(mToken) && !std::holds_alternative<T2>(mToken))
|
|
{
|
|
if (err.empty())
|
|
error(Misc::StringUtils::format(
|
|
"%s. Expected %s or %s", err, std::string(T::repr), std::string(T2::repr)));
|
|
else
|
|
error(Misc::StringUtils::format("Expected %s or %s", std::string(T::repr), std::string(T2::repr)));
|
|
}
|
|
}
|
|
|
|
template <class T>
|
|
bool Technique::isNext()
|
|
{
|
|
return std::holds_alternative<T>(mLexer->peek());
|
|
}
|
|
|
|
void Technique::parse(std::string&& buffer)
|
|
{
|
|
mBuffer = std::move(buffer);
|
|
Misc::StringUtils::replaceAll(mBuffer, "\r\n", "\n");
|
|
mLexer = std::make_unique<Lexer::Lexer>(mBuffer);
|
|
|
|
for (auto t = mLexer->next(); !std::holds_alternative<Lexer::Eof>(t); t = mLexer->next())
|
|
{
|
|
std::visit(
|
|
[this](auto&& arg) {
|
|
using T = std::decay_t<decltype(arg)>;
|
|
|
|
if constexpr (std::is_same_v<Lexer::Shared, T>)
|
|
parseBlock<Lexer::Shared>(false);
|
|
else if constexpr (std::is_same_v<Lexer::Technique, T>)
|
|
parseBlock<Lexer::Technique>(false);
|
|
else if constexpr (std::is_same_v<Lexer::Render_Target, T>)
|
|
parseBlock<Lexer::Render_Target>();
|
|
else if constexpr (std::is_same_v<Lexer::Vertex, T>)
|
|
parseBlock<Lexer::Vertex>();
|
|
else if constexpr (std::is_same_v<Lexer::Fragment, T>)
|
|
parseBlock<Lexer::Fragment>();
|
|
else if constexpr (std::is_same_v<Lexer::Compute, T>)
|
|
parseBlock<Lexer::Compute>();
|
|
else if constexpr (std::is_same_v<Lexer::Sampler_1D, T>)
|
|
parseBlock<Lexer::Sampler_1D>();
|
|
else if constexpr (std::is_same_v<Lexer::Sampler_2D, T>)
|
|
parseBlock<Lexer::Sampler_2D>();
|
|
else if constexpr (std::is_same_v<Lexer::Sampler_3D, T>)
|
|
parseBlock<Lexer::Sampler_3D>();
|
|
else if constexpr (std::is_same_v<Lexer::Uniform_Bool, T>)
|
|
parseBlock<Lexer::Uniform_Bool>();
|
|
else if constexpr (std::is_same_v<Lexer::Uniform_Float, T>)
|
|
parseBlock<Lexer::Uniform_Float>();
|
|
else if constexpr (std::is_same_v<Lexer::Uniform_Int, T>)
|
|
parseBlock<Lexer::Uniform_Int>();
|
|
else if constexpr (std::is_same_v<Lexer::Uniform_Vec2, T>)
|
|
parseBlock<Lexer::Uniform_Vec2>();
|
|
else if constexpr (std::is_same_v<Lexer::Uniform_Vec3, T>)
|
|
parseBlock<Lexer::Uniform_Vec3>();
|
|
else if constexpr (std::is_same_v<Lexer::Uniform_Vec4, T>)
|
|
parseBlock<Lexer::Uniform_Vec4>();
|
|
else
|
|
error("invalid top level block");
|
|
},
|
|
t);
|
|
}
|
|
}
|
|
|
|
template <class T>
|
|
void Technique::parseBlock(bool named)
|
|
{
|
|
mBlockName = T::repr;
|
|
|
|
if (named)
|
|
{
|
|
expect<Lexer::Literal>("name is required for preceeding block decleration");
|
|
|
|
mBlockName = std::get<Lexer::Literal>(mToken).value;
|
|
|
|
if (isNext<Lexer::Open_Parenthesis>())
|
|
parseBlockHeader();
|
|
}
|
|
|
|
expect<Lexer::Open_bracket>();
|
|
|
|
parseBlockImp<T>();
|
|
|
|
expect<Lexer::Close_bracket>();
|
|
}
|
|
|
|
template <class TDelimeter>
|
|
std::vector<std::string_view> Technique::parseLiteralList()
|
|
{
|
|
std::vector<std::string_view> data;
|
|
|
|
while (!isNext<Lexer::Eof>())
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
data.emplace_back(std::get<Lexer::Literal>(mToken).value);
|
|
|
|
if (!isNext<TDelimeter>())
|
|
break;
|
|
|
|
mLexer->next();
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
void Technique::parseBlockHeader()
|
|
{
|
|
expect<Lexer::Open_Parenthesis>();
|
|
|
|
if (isNext<Lexer::Close_Parenthesis>())
|
|
{
|
|
mLexer->next();
|
|
return;
|
|
}
|
|
|
|
auto& pass = mPassMap[mBlockName];
|
|
|
|
if (!pass)
|
|
pass = std::make_shared<fx::Pass>();
|
|
|
|
while (!isNext<Lexer::Eof>())
|
|
{
|
|
expect<Lexer::Literal>("invalid key in block header");
|
|
|
|
std::string_view key = std::get<Lexer::Literal>(mToken).value;
|
|
|
|
expect<Lexer::Equal>();
|
|
|
|
if (key == "target")
|
|
{
|
|
expect<Lexer::Literal>();
|
|
pass->mTarget = std::get<Lexer::Literal>(mToken).value;
|
|
}
|
|
else if (key == "rt1")
|
|
{
|
|
expect<Lexer::Literal>();
|
|
pass->mRenderTargets[0] = std::get<Lexer::Literal>(mToken).value;
|
|
}
|
|
else if (key == "rt2")
|
|
{
|
|
expect<Lexer::Literal>();
|
|
pass->mRenderTargets[1] = std::get<Lexer::Literal>(mToken).value;
|
|
}
|
|
else if (key == "rt3")
|
|
{
|
|
expect<Lexer::Literal>();
|
|
pass->mRenderTargets[2] = std::get<Lexer::Literal>(mToken).value;
|
|
}
|
|
else if (key == "blend")
|
|
{
|
|
expect<Lexer::Open_Parenthesis>();
|
|
osg::BlendEquation::Equation blendEq = parseBlendEquation();
|
|
expect<Lexer::Comma>();
|
|
osg::BlendFunc::BlendFuncMode blendSrc = parseBlendFuncMode();
|
|
expect<Lexer::Comma>();
|
|
osg::BlendFunc::BlendFuncMode blendDest = parseBlendFuncMode();
|
|
expect<Lexer::Close_Parenthesis>();
|
|
|
|
pass->mBlendSource = blendSrc;
|
|
pass->mBlendDest = blendDest;
|
|
if (blendEq != osg::BlendEquation::FUNC_ADD)
|
|
pass->mBlendEq = blendEq;
|
|
}
|
|
else
|
|
error(Misc::StringUtils::format("unrecognized key '%s' in block header", std::string(key)));
|
|
|
|
mToken = mLexer->next();
|
|
|
|
if (std::holds_alternative<Lexer::Comma>(mToken))
|
|
{
|
|
if (std::holds_alternative<Lexer::Close_Parenthesis>(mLexer->peek()))
|
|
error(Misc::StringUtils::format("leading comma in '%s' is not allowed", std::string(mBlockName)));
|
|
else
|
|
continue;
|
|
}
|
|
|
|
if (std::holds_alternative<Lexer::Close_Parenthesis>(mToken))
|
|
return;
|
|
}
|
|
|
|
error("malformed block header");
|
|
}
|
|
|
|
std::string_view Technique::asLiteral() const
|
|
{
|
|
return std::get<Lexer::Literal>(mToken).value;
|
|
}
|
|
|
|
FlagsType Technique::parseFlags()
|
|
{
|
|
auto parseBit = [this](std::string_view term) {
|
|
for (const auto& [identifer, bit] : constants::TechniqueFlag)
|
|
{
|
|
if (Misc::StringUtils::ciEqual(term, identifer))
|
|
return bit;
|
|
}
|
|
error(Misc::StringUtils::format("unrecognized flag '%s'", std::string(term)));
|
|
};
|
|
|
|
FlagsType flag = 0;
|
|
for (const auto& bit : parseLiteralList<Lexer::Comma>())
|
|
flag |= parseBit(bit);
|
|
|
|
return flag;
|
|
}
|
|
|
|
osg::Texture::FilterMode Technique::parseFilterMode()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::FilterMode)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized filter mode '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
osg::Texture::WrapMode Technique::parseWrapMode()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::WrapMode)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
if (asLiteral() == "clamp")
|
|
error(
|
|
"unsupported wrap mode 'clamp'; 'clamp_to_edge' was likely intended, look for an updated shader or "
|
|
"contact author");
|
|
|
|
error(Misc::StringUtils::format("unrecognized wrap mode '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
osg::Texture::InternalFormatMode Technique::parseCompression()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::Compression)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized compression '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
int Technique::parseInternalFormat()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::InternalFormat)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized internal format '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
int Technique::parseSourceType()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::SourceType)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized source type '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
int Technique::parseSourceFormat()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::SourceFormat)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized source format '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
osg::BlendEquation::Equation Technique::parseBlendEquation()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::BlendEquation)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized blend equation '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
osg::BlendFunc::BlendFuncMode Technique::parseBlendFuncMode()
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
for (const auto& [identifer, mode] : constants::BlendFunc)
|
|
{
|
|
if (asLiteral() == identifer)
|
|
return mode;
|
|
}
|
|
|
|
error(Misc::StringUtils::format("unrecognized blend function '%s'", std::string{ asLiteral() }));
|
|
}
|
|
|
|
template <class SrcT, class T>
|
|
void Technique::parseWidgetType(Types::Uniform<SrcT>& uniform)
|
|
{
|
|
expect<Lexer::Literal>();
|
|
|
|
if (asLiteral() == "choice")
|
|
{
|
|
/* Example usage
|
|
|
|
widget_type = choice(
|
|
"Option A": <T>,
|
|
"Option B": <T>,
|
|
"Option C": <T>
|
|
);
|
|
|
|
*/
|
|
expect<Lexer::Open_Parenthesis>();
|
|
|
|
std::vector<fx::Types::Choice<SrcT>> choices;
|
|
|
|
while (!isNext<Lexer::Eof>())
|
|
{
|
|
fx::Types::Choice<SrcT> choice;
|
|
choice.mLabel = parseString();
|
|
expect<Lexer::Equal>();
|
|
choice.mValue = getUniformValue<SrcT, T>();
|
|
choices.push_back(choice);
|
|
|
|
if (isNext<Lexer::Comma>())
|
|
{
|
|
mToken = mLexer->next();
|
|
|
|
// Handle leading comma
|
|
if (isNext<Lexer::Close_Parenthesis>())
|
|
{
|
|
break;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
uniform.mChoices = std::move(choices);
|
|
|
|
expect<Lexer::Close_Parenthesis>();
|
|
}
|
|
else
|
|
{
|
|
error(Misc::StringUtils::format("unrecognized widget type '%s'", std::string{ asLiteral() }));
|
|
}
|
|
}
|
|
|
|
bool Technique::parseBool()
|
|
{
|
|
mToken = mLexer->next();
|
|
|
|
if (std::holds_alternative<Lexer::True>(mToken))
|
|
return true;
|
|
if (std::holds_alternative<Lexer::False>(mToken))
|
|
return false;
|
|
|
|
error("expected 'true' or 'false' as boolean value");
|
|
}
|
|
|
|
std::string_view Technique::parseString()
|
|
{
|
|
expect<Lexer::String>();
|
|
|
|
return std::get<Lexer::String>(mToken).value;
|
|
}
|
|
|
|
float Technique::parseFloat()
|
|
{
|
|
mToken = mLexer->next();
|
|
|
|
if (std::holds_alternative<Lexer::Float>(mToken))
|
|
return std::get<Lexer::Float>(mToken).value;
|
|
if (std::holds_alternative<Lexer::Integer>(mToken))
|
|
return static_cast<float>(std::get<Lexer::Integer>(mToken).value);
|
|
|
|
error("expected float value");
|
|
}
|
|
|
|
int Technique::parseInteger()
|
|
{
|
|
expect<Lexer::Integer>();
|
|
|
|
return std::get<Lexer::Integer>(mToken).value;
|
|
}
|
|
|
|
template <class OSGVec, class T>
|
|
OSGVec Technique::parseVec()
|
|
{
|
|
expect<T>();
|
|
expect<Lexer::Open_Parenthesis>();
|
|
|
|
OSGVec value;
|
|
|
|
for (int i = 0; i < OSGVec::num_components; ++i)
|
|
{
|
|
value[i] = parseFloat();
|
|
|
|
if (i < OSGVec::num_components - 1)
|
|
expect<Lexer::Comma>();
|
|
}
|
|
|
|
expect<Lexer::Close_Parenthesis>("check definition of the vector");
|
|
|
|
return value;
|
|
}
|
|
}
|