#ifndef OPENMW_COMPONENTS_FX_TECHNIQUE_H #define OPENMW_COMPONENTS_FX_TECHNIQUE_H #include #include #include #include #include #include #include #include #include #include #include "lexer.hpp" #include "pass.hpp" #include "types.hpp" namespace Resource { class ImageManager; } namespace VFS { class Manager; } namespace fx { using FlagsType = size_t; struct DispatchNode { DispatchNode() = default; DispatchNode(const DispatchNode& other, const osg::CopyOp& copyOp = osg::CopyOp::SHALLOW_COPY) : mHandle(other.mHandle) , mFlags(other.mFlags) , mRootStateSet(other.mRootStateSet) { mPasses.reserve(other.mPasses.size()); for (const auto& subpass : other.mPasses) mPasses.emplace_back(subpass, copyOp); } struct SubPass { SubPass() = default; osg::ref_ptr mStateSet = new osg::StateSet; osg::ref_ptr mRenderTarget; osg::ref_ptr mRenderTexture; bool mResolve = false; Types::SizeProxy mSize; bool mMipMap = false; SubPass(const SubPass& other, const osg::CopyOp& copyOp = osg::CopyOp::SHALLOW_COPY) : mStateSet(new osg::StateSet(*other.mStateSet, copyOp)) , mResolve(other.mResolve) , mSize(other.mSize) , mMipMap(other.mMipMap) { if (other.mRenderTarget) mRenderTarget = new osg::FrameBufferObject(*other.mRenderTarget, copyOp); if (other.mRenderTexture) mRenderTexture = new osg::Texture2D(*other.mRenderTexture, copyOp); } }; void compile() { for (auto rit = mPasses.rbegin(); rit != mPasses.rend(); ++rit) { if (!rit->mRenderTarget) { rit->mResolve = true; break; } } } // not safe to read/write in draw thread std::shared_ptr mHandle = nullptr; FlagsType mFlags = 0; std::vector mPasses; osg::ref_ptr mRootStateSet = new osg::StateSet; }; using DispatchArray = std::vector; class Technique { public: using PassList = std::vector>; using TexList = std::vector>; using UniformMap = std::vector>; using RenderTargetMap = std::unordered_map; inline static std::string sExt = ".omwfx"; inline static std::string sSubdir = "shaders"; enum class Status { Success, Uncompiled, File_Not_exists, Parse_Error }; static constexpr FlagsType Flag_Disable_Interiors = (1 << 0); static constexpr FlagsType Flag_Disable_Exteriors = (1 << 1); static constexpr FlagsType Flag_Disable_Underwater = (1 << 2); static constexpr FlagsType Flag_Disable_Abovewater = (1 << 3); static constexpr FlagsType Flag_Disable_SunGlare = (1 << 4); static constexpr FlagsType Flag_Hidden = (1 << 5); Technique(const VFS::Manager& vfs, Resource::ImageManager& imageManager, std::string name, int width, int height, bool ubo, bool supportsNormals); bool compile(); std::string getName() const; std::string getFileName() const; bool setLastModificationTime(std::filesystem::file_time_type timeStamp); bool isValid() const { return mValid; } bool getHDR() const { return mHDR; } bool getNormals() const { return mNormals && mSupportsNormals; } bool getLights() const { return mLights; } const PassList& getPasses() { return mPasses; } const TexList& getTextures() const { return mTextures; } Status getStatus() const { return mStatus; } std::string_view getAuthor() const { return mAuthor; } std::string_view getDescription() const { return mDescription; } std::string_view getVersion() const { return mVersion; } int getGLSLVersion() const { return mGLSLVersion; } std::string getGLSLProfile() const { return mGLSLProfile; } const std::unordered_set& getGLSLExtensions() const { return mGLSLExtensions; } FlagsType getFlags() const { return mFlags; } bool getHidden() const { return mFlags & Flag_Hidden; } UniformMap& getUniformMap() { return mDefinedUniforms; } RenderTargetMap& getRenderTargetsMap() { return mRenderTargets; } std::string getLastError() const { return mLastError; } UniformMap::iterator findUniform(const std::string& name); bool getDynamic() const { return mDynamic; } void setLocked(bool locked) { mLocked = locked; } bool getLocked() const { return mLocked; } void setInternal(bool internal) { mInternal = internal; } bool getInternal() const { return mInternal; } private: [[noreturn]] void error(const std::string& msg); void clear(); std::string_view asLiteral() const; template void expect(const std::string& err = ""); template void expect(const std::string& err = ""); template bool isNext(); void parse(std::string&& buffer); template void parseUniform(); template void parseSampler(); template void parseBlock(bool named = true); template void parseBlockImp() { } void parseBlockHeader(); bool parseBool(); std::string_view parseString(); float parseFloat(); int parseInteger(); int parseInternalFormat(); int parseSourceType(); int parseSourceFormat(); osg::BlendEquation::Equation parseBlendEquation(); osg::BlendFunc::BlendFuncMode parseBlendFuncMode(); osg::Texture::WrapMode parseWrapMode(); osg::Texture::InternalFormatMode parseCompression(); FlagsType parseFlags(); osg::Texture::FilterMode parseFilterMode(); template std::vector parseLiteralList(); template OSGVec parseVec(); std::string getBlockWithLineDirective(); std::unique_ptr mLexer; Lexer::Token mToken; std::string mShared; std::string mName; std::string mFileName; std::string_view mBlockName; std::string_view mAuthor; std::string_view mDescription; std::string_view mVersion; std::unordered_set mGLSLExtensions; int mGLSLVersion; std::string mGLSLProfile; FlagsType mFlags; Status mStatus; bool mEnabled; std::filesystem::file_time_type mLastModificationTime; bool mValid; bool mHDR; bool mNormals; bool mLights; int mWidth; int mHeight; RenderTargetMap mRenderTargets; TexList mTextures; PassList mPasses; std::unordered_map> mPassMap; std::vector mPassKeys; Pass::Type mLastAppliedType; UniformMap mDefinedUniforms; const VFS::Manager& mVFS; Resource::ImageManager& mImageManager; bool mUBO; bool mSupportsNormals; std::string mBuffer; std::string mLastError; bool mDynamic = false; bool mLocked = false; bool mInternal = false; }; template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); template <> void Technique::parseBlockImp(); } #endif