1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-27 03:35:27 +00:00
OpenMW/components/shader/shadermanager.hpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

146 lines
5.5 KiB
C++
Raw Normal View History

#ifndef OPENMW_COMPONENTS_SHADERMANAGER_H
#define OPENMW_COMPONENTS_SHADERMANAGER_H
#include <array>
#include <map>
2022-08-03 20:17:53 +02:00
#include <memory>
2020-06-25 21:46:07 +02:00
#include <mutex>
2023-02-25 11:03:39 -08:00
#include <optional>
#include <string>
#include <vector>
#include <osg/ref_ptr>
#include <filesystem>
#include <osg/Program>
#include <osg/Shader>
2022-08-12 18:47:29 +02:00
namespace osgViewer
{
class Viewer;
}
namespace Shader
{
struct HotReloadManager;
/// @brief Reads shader template files and turns them into a concrete shader, based on a list of define's.
/// @par Shader templates can get the value of a define with the syntax @define.
class ShaderManager
{
public:
friend HotReloadManager;
2021-03-01 23:30:54 -08:00
ShaderManager();
~ShaderManager();
2021-02-21 10:38:15 -08:00
void setShaderPath(const std::filesystem::path& path);
typedef std::map<std::string, std::string> DefineMap;
/// Create or retrieve a shader instance.
2023-02-25 11:03:39 -08:00
/// @param templateName The path of the shader template.
/// @param defines Define values that can be retrieved by the shader template.
/// @param shaderType The type of shader (usually vertex or fragment shader).
2018-10-09 10:21:12 +04:00
/// @note May return nullptr on failure.
/// @note Thread safe.
2023-02-25 11:03:39 -08:00
osg::ref_ptr<osg::Shader> getShader(std::string templateName, const DefineMap& defines = {},
std::optional<osg::Shader::Type> type = std::nullopt);
osg::ref_ptr<osg::Program> getProgram(const std::string& templateName, const DefineMap& defines = {},
const osg::Program* programTemplate = nullptr);
osg::ref_ptr<osg::Program> getProgram(osg::ref_ptr<osg::Shader> vertexShader,
osg::ref_ptr<osg::Shader> fragmentShader, const osg::Program* programTemplate = nullptr);
const osg::Program* getProgramTemplate() const { return mProgramTemplate; }
void setProgramTemplate(const osg::Program* program) { mProgramTemplate = program; }
2016-02-17 15:04:05 +01:00
/// Clone an osg::Program including bindUniformBlocks that osg::Program::clone does not copy for some reason.
static osg::ref_ptr<osg::Program> cloneProgram(const osg::Program*);
/// Get (a copy of) the DefineMap used to construct all shaders
DefineMap getGlobalDefines();
/// Set the DefineMap used to construct all shaders
/// @param defines The DefineMap to use
/// @note This will change the source code for any shaders already created, potentially causing problems if
/// they're being used to render a frame. It is recommended that any associated Viewers have their threading
/// stopped while this function is running if any shaders are in use.
void setGlobalDefines(DefineMap& globalDefines);
void releaseGLObjects(osg::State* state);
2016-02-17 15:04:05 +01:00
bool createSourceFromTemplate(std::string& source, std::vector<std::string>& linkedShaderTemplateNames,
const std::string& templateName, const ShaderManager::DefineMap& defines);
2022-06-06 22:40:38 +02:00
void setMaxTextureUnits(int maxTextureUnits) { mMaxTextureUnits = maxTextureUnits; }
int getMaxTextureUnits() const { return mMaxTextureUnits; }
int getAvailableTextureUnits() const { return mMaxTextureUnits - mReservedTextureUnits; }
enum class Slot
{
OpaqueDepthTexture,
SkyTexture,
ShadowMaps,
SLOT_COUNT
};
int reserveGlobalTextureUnits(Slot slot, int count = 1);
2022-06-06 22:40:38 +02:00
2022-08-12 18:26:04 +02:00
void update(osgViewer::Viewer& viewer);
void setHotReloadEnabled(bool value);
void triggerShaderReload();
2022-09-22 21:26:05 +03:00
private:
void getLinkedShaders(osg::ref_ptr<osg::Shader> shader, const std::vector<std::string>& linkedShaderNames,
const DefineMap& defines);
void addLinkedShaders(osg::ref_ptr<osg::Shader> shader, osg::ref_ptr<osg::Program> program);
std::filesystem::path mPath;
DefineMap mGlobalDefines;
// <name, code>
typedef std::map<std::string, std::string> TemplateMap;
TemplateMap mShaderTemplates;
typedef std::pair<std::string, DefineMap> MapKey;
typedef std::map<MapKey, osg::ref_ptr<osg::Shader>> ShaderMap;
ShaderMap mShaders;
2016-02-17 15:04:05 +01:00
typedef std::map<std::pair<osg::ref_ptr<osg::Shader>, osg::ref_ptr<osg::Shader>>, osg::ref_ptr<osg::Program>>
ProgramMap;
ProgramMap mPrograms;
typedef std::vector<osg::ref_ptr<osg::Shader>> ShaderList;
typedef std::map<osg::ref_ptr<osg::Shader>, ShaderList> LinkedShadersMap;
LinkedShadersMap mLinkedShaders;
2020-06-25 21:46:07 +02:00
std::mutex mMutex;
osg::ref_ptr<const osg::Program> mProgramTemplate;
2022-06-06 22:40:38 +02:00
int mMaxTextureUnits = 0;
int mReservedTextureUnits = 0;
std::unique_ptr<HotReloadManager> mHotReloadManager;
struct ReservedTextureUnits
{
int index = -1;
int count = 0;
};
std::array<ReservedTextureUnits, static_cast<int>(Slot::SLOT_COUNT)> mReservedTextureUnitsBySlot = {};
};
bool parseForeachDirective(std::string& source, const std::string& templateName, size_t foundPos);
bool parseLinkDirective(
std::string& source, std::string& linkTarget, const std::string& templateName, size_t foundPos);
2020-05-01 23:03:13 +02:00
bool parseDefines(std::string& source, const ShaderManager::DefineMap& defines,
const ShaderManager::DefineMap& globalDefines, const std::string& templateName);
bool parseDirectives(std::string& source, std::vector<std::string>& linkedShaderTemplateNames,
const ShaderManager::DefineMap& defines, const ShaderManager::DefineMap& globalDefines,
const std::string& templateName);
}
#endif