2016-12-17 21:51:05 -05:00
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
|
|
|
// Copyright (C) 2016 SuperTuxKart-Team
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License
|
|
|
|
// as published by the Free Software Foundation; either version 3
|
|
|
|
// of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
|
|
|
#ifndef SERVER_ONLY
|
|
|
|
|
|
|
|
#include "graphics/shader_files_manager.hpp"
|
|
|
|
#include "graphics/central_settings.hpp"
|
|
|
|
#include "graphics/shared_gpu_objects.hpp"
|
|
|
|
#include "io/file_manager.hpp"
|
|
|
|
#include "utils/log.hpp"
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
#include <sstream>
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
/** Returns a string with the content of header.txt (which contains basic
|
|
|
|
* shader defines).
|
|
|
|
*/
|
|
|
|
const std::string& ShaderFilesManager::getHeader()
|
|
|
|
{
|
2016-12-18 06:30:49 -05:00
|
|
|
// Stores the content of header.txt, to avoid reading this file repeatedly.
|
|
|
|
static std::string shader_header;
|
|
|
|
|
2016-12-17 21:51:05 -05:00
|
|
|
// Only read file first time
|
2016-12-18 06:30:49 -05:00
|
|
|
if (shader_header.empty())
|
2016-12-17 21:51:05 -05:00
|
|
|
{
|
2016-12-18 06:30:49 -05:00
|
|
|
std::ifstream stream(file_manager->getShader("header.txt"), std::ios::in);
|
2016-12-17 21:51:05 -05:00
|
|
|
if (stream.is_open())
|
|
|
|
{
|
|
|
|
std::string line = "";
|
2016-12-18 06:30:49 -05:00
|
|
|
while (std::getline(stream, line))
|
|
|
|
shader_header += "\n" + line;
|
2016-12-17 21:51:05 -05:00
|
|
|
stream.close();
|
|
|
|
}
|
2016-12-18 06:30:49 -05:00
|
|
|
} // if shader_header.empty()
|
2016-12-17 21:51:05 -05:00
|
|
|
|
2016-12-18 06:30:49 -05:00
|
|
|
return shader_header;
|
2016-12-17 21:51:05 -05:00
|
|
|
} // getHeader
|
|
|
|
|
2017-03-21 13:41:18 -04:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void ShaderFilesManager::readFile(const std::string& file,
|
|
|
|
std::ostringstream& code)
|
|
|
|
{
|
|
|
|
std::ifstream stream(file_manager->getShader(file), std::ios::in);
|
|
|
|
|
|
|
|
if (!stream.is_open())
|
|
|
|
{
|
|
|
|
Log::error("ShaderFilesManager", "Can not open '%s'.", file.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string stk_include = "#stk_include";
|
|
|
|
std::string line;
|
|
|
|
|
|
|
|
while (std::getline(stream, line))
|
|
|
|
{
|
|
|
|
const std::size_t pos = line.find(stk_include);
|
|
|
|
|
|
|
|
// load the custom file pointed by the #stk_include directive
|
|
|
|
if (pos != std::string::npos)
|
|
|
|
{
|
|
|
|
// find the start "
|
|
|
|
std::size_t pos = line.find("\"");
|
|
|
|
if (pos == std::string::npos)
|
|
|
|
{
|
|
|
|
Log::error("ShaderFilesManager", "Invalid #stk_include"
|
|
|
|
" line: '%s'.", line.c_str());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string filename = line.substr(pos + 1);
|
|
|
|
|
|
|
|
// find the end "
|
|
|
|
pos = filename.find("\"");
|
|
|
|
if (pos == std::string::npos)
|
|
|
|
{
|
|
|
|
Log::error("ShaderFilesManager", "Invalid #stk_include"
|
|
|
|
" line: '%s'.", line.c_str());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
filename = filename.substr(0, pos);
|
|
|
|
|
|
|
|
// read the whole include file
|
|
|
|
readFile(filename, code);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
code << "\n" << line;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stream.close();
|
|
|
|
}
|
|
|
|
|
2016-12-17 21:51:05 -05:00
|
|
|
// ----------------------------------------------------------------------------
|
2016-12-18 06:30:49 -05:00
|
|
|
/** Loads a single shader. This is NOT cached, use addShaderFile for that.
|
2016-12-17 21:51:05 -05:00
|
|
|
* \param file Filename of the shader to load.
|
|
|
|
* \param type Type of the shader.
|
|
|
|
*/
|
2016-12-18 06:30:49 -05:00
|
|
|
GLuint ShaderFilesManager::loadShader(const std::string &file, unsigned type)
|
2016-12-17 21:51:05 -05:00
|
|
|
{
|
2016-12-18 06:30:49 -05:00
|
|
|
const GLuint id = glCreateShader(type);
|
2016-12-17 21:51:05 -05:00
|
|
|
|
|
|
|
std::ostringstream code;
|
|
|
|
#if !defined(USE_GLES2)
|
|
|
|
code << "#version " << CVS->getGLSLVersion()<<"\n";
|
|
|
|
#else
|
|
|
|
if (CVS->isGLSL())
|
|
|
|
code << "#version 300 es\n";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(USE_GLES2)
|
|
|
|
// Some drivers report that the compute shaders extension is available,
|
|
|
|
// but they report only OpenGL 3.x version, and thus these extensions
|
|
|
|
// must be enabled manually. Otherwise the shaders compilation will fail
|
|
|
|
// because STK tries to use extensions which are available, but disabled
|
|
|
|
// by default.
|
|
|
|
if (type == GL_COMPUTE_SHADER)
|
|
|
|
{
|
|
|
|
if (CVS->isARBComputeShaderUsable())
|
|
|
|
code << "#extension GL_ARB_compute_shader : enable\n";
|
|
|
|
if (CVS->isARBImageLoadStoreUsable())
|
|
|
|
code << "#extension GL_ARB_shader_image_load_store : enable\n";
|
|
|
|
if (CVS->isARBArraysOfArraysUsable())
|
|
|
|
code << "#extension GL_ARB_arrays_of_arrays : enable\n";
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (CVS->isAMDVertexShaderLayerUsable())
|
|
|
|
code << "#extension GL_AMD_vertex_shader_layer : enable\n";
|
|
|
|
|
|
|
|
if (CVS->isARBExplicitAttribLocationUsable())
|
2017-02-01 15:58:10 -05:00
|
|
|
{
|
2016-12-17 21:51:05 -05:00
|
|
|
code << "#extension GL_ARB_explicit_attrib_location : enable\n";
|
2017-02-01 15:58:10 -05:00
|
|
|
code << "#define Explicit_Attrib_Location_Usable\n";
|
|
|
|
}
|
2016-12-17 21:51:05 -05:00
|
|
|
|
|
|
|
if (CVS->isAZDOEnabled())
|
|
|
|
{
|
|
|
|
code << "#extension GL_ARB_bindless_texture : enable\n";
|
|
|
|
code << "#define Use_Bindless_Texture\n";
|
|
|
|
}
|
|
|
|
code << "//" << file << "\n";
|
|
|
|
if (!CVS->isARBUniformBufferObjectUsable())
|
|
|
|
code << "#define UBO_DISABLED\n";
|
|
|
|
if (CVS->isAMDVertexShaderLayerUsable())
|
|
|
|
code << "#define VSLayer\n";
|
|
|
|
if (CVS->needsRGBBindlessWorkaround())
|
|
|
|
code << "#define SRGBBindlessFix\n";
|
|
|
|
|
|
|
|
#if !defined(USE_GLES2)
|
|
|
|
// shader compilation fails with some drivers if there is no precision
|
|
|
|
// qualifier
|
|
|
|
if (type == GL_FRAGMENT_SHADER)
|
|
|
|
code << "precision mediump float;\n";
|
|
|
|
#else
|
|
|
|
int range[2], precision;
|
|
|
|
glGetShaderPrecisionFormat(GL_FRAGMENT_SHADER, GL_HIGH_FLOAT, range,
|
|
|
|
&precision);
|
|
|
|
|
|
|
|
if (precision > 0)
|
|
|
|
code << "precision highp float;\n";
|
|
|
|
else
|
|
|
|
code << "precision mediump float;\n";
|
|
|
|
#endif
|
|
|
|
code << "#define MAX_BONES " << SharedGPUObjects::getMaxMat4Size() << "\n";
|
|
|
|
|
|
|
|
code << getHeader();
|
|
|
|
|
2017-03-21 13:41:18 -04:00
|
|
|
readFile(file, code);
|
2016-12-17 21:51:05 -05:00
|
|
|
|
|
|
|
Log::info("ShaderFilesManager", "Compiling shader : %s", file.c_str());
|
|
|
|
const std::string &source = code.str();
|
|
|
|
char const *source_pointer = source.c_str();
|
|
|
|
int len = source.size();
|
|
|
|
glShaderSource(id, 1, &source_pointer, &len);
|
|
|
|
glCompileShader(id);
|
|
|
|
|
|
|
|
GLint result = GL_FALSE;
|
|
|
|
glGetShaderiv(id, GL_COMPILE_STATUS, &result);
|
|
|
|
if (result == GL_FALSE)
|
|
|
|
{
|
2016-12-18 06:30:49 -05:00
|
|
|
// failed to compile
|
2016-12-17 21:51:05 -05:00
|
|
|
int info_length;
|
|
|
|
Log::error("ShaderFilesManager", "Error in shader %s", file.c_str());
|
|
|
|
glGetShaderiv(id, GL_INFO_LOG_LENGTH, &info_length);
|
|
|
|
if (info_length < 0)
|
|
|
|
info_length = 1024;
|
|
|
|
char *error_message = new char[info_length];
|
|
|
|
error_message[0] = 0;
|
|
|
|
glGetShaderInfoLog(id, info_length, NULL, error_message);
|
|
|
|
Log::error("ShaderFilesManager", error_message);
|
|
|
|
delete[] error_message;
|
|
|
|
}
|
|
|
|
glGetError();
|
|
|
|
|
2016-12-18 06:30:49 -05:00
|
|
|
return id;
|
|
|
|
} // loadShader
|
2016-12-17 21:51:05 -05:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2016-12-18 06:30:49 -05:00
|
|
|
/** Loads a single shader file, and add it to the loaded (cached) list
|
|
|
|
* \param file Filename of the shader to load.
|
|
|
|
* \param type Type of the shader.
|
|
|
|
*/
|
|
|
|
GLuint ShaderFilesManager::addShaderFile(const std::string &file, unsigned type)
|
2016-12-17 21:51:05 -05:00
|
|
|
{
|
2016-12-18 06:30:49 -05:00
|
|
|
#ifdef DEBUG
|
|
|
|
// Make sure no duplicated shader is added somewhere else
|
2016-12-17 21:51:05 -05:00
|
|
|
std::unordered_map<std::string, GLuint>::const_iterator i =
|
|
|
|
m_shader_files_loaded.find(file);
|
2016-12-18 06:30:49 -05:00
|
|
|
assert(i == m_shader_files_loaded.end());
|
|
|
|
#endif
|
|
|
|
|
|
|
|
const GLuint id = loadShader(file, type);
|
|
|
|
m_shader_files_loaded[file] = id;
|
|
|
|
return id;
|
|
|
|
} // addShaderFile
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
/** Get a shader file. If the shader is not already in the cache it will be loaded and cached.
|
|
|
|
* \param file Filename of the shader to load.
|
|
|
|
* \param type Type of the shader.
|
|
|
|
*/
|
|
|
|
GLuint ShaderFilesManager::getShaderFile(const std::string &file, unsigned type)
|
|
|
|
{
|
|
|
|
// found in cache
|
|
|
|
auto it = m_shader_files_loaded.find(file);
|
|
|
|
if (it != m_shader_files_loaded.end())
|
|
|
|
return it->second;
|
2016-12-17 21:51:05 -05:00
|
|
|
|
2016-12-18 06:30:49 -05:00
|
|
|
// add to the cache now
|
|
|
|
return addShaderFile(file, type);
|
2016-12-17 21:51:05 -05:00
|
|
|
} // getShaderFile
|
|
|
|
|
|
|
|
#endif // !SERVER_ONLY
|