// Copyright (C) 2015 Jérôme Leclercq // This file is part of the "Nazara Engine - Renderer module" // For conditions of distribution and use, see copyright notice in Config.hpp #include #include #include #include #include #include #include NzUberShaderPreprocessor::~NzUberShaderPreprocessor() { OnUberShaderPreprocessorRelease(this); } NzUberShaderInstance* NzUberShaderPreprocessor::Get(const NzParameterList& parameters) const { // Première étape, transformer les paramètres en un flag nzUInt32 flags = 0; for (auto it = m_flags.begin(); it != m_flags.end(); ++it) { if (parameters.HasParameter(it->first)) { bool value; if (parameters.GetBooleanParameter(it->first, &value) && value) flags |= it->second; } } // Le shader fait-il partie du cache ? auto shaderIt = m_cache.find(flags); // Si non, il nous faut le construire if (shaderIt == m_cache.end()) { try { // Une exception sera lancée à la moindre erreur et celle-ci ne sera pas enregistrée dans le log (car traitée dans le bloc catch) NzErrorFlags errFlags(nzErrorFlag_Silent | nzErrorFlag_ThrowException, true); NzShaderRef shader = NzShader::New(); shader->Create(); for (unsigned int i = 0; i <= nzShaderStage_Max; ++i) { const Shader& shaderStage = m_shaders[i]; // Le shader stage est-il activé dans cette version du shader ? if (shaderStage.present && (flags & shaderStage.requiredFlags) == shaderStage.requiredFlags) { nzUInt32 stageFlags = 0; for (auto it = shaderStage.flags.begin(); it != shaderStage.flags.end(); ++it) { if (parameters.HasParameter(it->first)) { bool value; if (parameters.GetBooleanParameter(it->first, &value) && value) stageFlags |= it->second; } } auto stageIt = shaderStage.cache.find(stageFlags); if (stageIt == shaderStage.cache.end()) { NzShaderStage stage; stage.Create(static_cast(i)); unsigned int glslVersion = NzOpenGL::GetGLSLVersion(); NzStringStream code; code << "#version " << glslVersion << "\n\n"; code << "#define GLSL_VERSION " << glslVersion << "\n\n"; code << "#define EARLY_FRAGMENT_TEST " << (glslVersion >= 420 || NzOpenGL::IsSupported(nzOpenGLExtension_Shader_ImageLoadStore)) << "\n\n"; for (auto it = shaderStage.flags.begin(); it != shaderStage.flags.end(); ++it) code << "#define " << it->first << ' ' << ((stageFlags & it->second) ? '1' : '0') << '\n'; code << "\n#line 1\n"; // Pour que les éventuelles erreurs du shader se réfèrent à la bonne ligne code << shaderStage.source; stage.SetSource(code); stage.Compile(); stageIt = shaderStage.cache.emplace(flags, std::move(stage)).first; } shader->AttachStage(static_cast(i), stageIt->second); } } shader->Link(); // On construit l'instant shaderIt = m_cache.emplace(flags, shader.Get()).first; } catch (const std::exception&) { NzErrorFlags errFlags(nzErrorFlag_ThrowExceptionDisabled); NazaraError("Failed to build UberShader instance: " + NzError::GetLastError()); throw; } } return &shaderIt->second; } void NzUberShaderPreprocessor::SetShader(nzShaderStage stage, const NzString& source, const NzString& shaderFlags, const NzString& requiredFlags) { Shader& shader = m_shaders[stage]; shader.present = true; shader.source = source; // On extrait les flags de la chaîne std::vector flags; shaderFlags.Split(flags, ' '); for (NzString& flag : flags) { auto it = m_flags.find(flag); if (it == m_flags.end()) m_flags[flag] = 1U << m_flags.size(); auto it2 = shader.flags.find(flag); if (it2 == shader.flags.end()) shader.flags[flag] = 1U << shader.flags.size(); } // On construit les flags requis pour l'activation du shader shader.requiredFlags = 0; flags.clear(); requiredFlags.Split(flags, ' '); for (NzString& flag : flags) { nzUInt32 flagVal; auto it = m_flags.find(flag); if (it == m_flags.end()) { flagVal = 1U << m_flags.size(); m_flags[flag] = flagVal; } else flagVal = it->second; shader.requiredFlags |= flagVal; } } bool NzUberShaderPreprocessor::SetShaderFromFile(nzShaderStage stage, const NzString& filePath, const NzString& shaderFlags, const NzString& requiredFlags) { NzFile file(filePath); if (!file.Open(NzFile::ReadOnly | NzFile::Text)) { NazaraError("Failed to open \"" + filePath + '"'); return false; } unsigned int length = static_cast(file.GetSize()); NzString source(length, '\0'); if (file.Read(&source[0], length) != length) { NazaraError("Failed to read program file"); return false; } file.Close(); SetShader(stage, source, shaderFlags, requiredFlags); return true; } bool NzUberShaderPreprocessor::IsSupported() { return true; // Forcément supporté }