dolphin/Source/Core/VideoCommon/PostProcessing.cpp
Ryan Houdek 2d624780c0 Catch broken configurations inside of the Post Processing shaders.
This catches most instances of configuration failures that can happen in a post processing shader.
Gives a user a helpful error message that lets them know what they have failed to set up correctly
2014-08-17 23:59:21 -05:00

433 lines
14 KiB
C++

// Copyright 2013 Dolphin Emulator Project
// Licensed under GPLv2
// Refer to the license.txt file included.
#include <string>
#include "Common/CommonPaths.h"
#include "Common/FileUtil.h"
#include "Common/IniFile.h"
#include "Common/MsgHandler.h"
#include "Common/StringUtil.h"
#include "VideoCommon/PostProcessing.h"
#include "VideoCommon/VideoConfig.h"
PostProcessingShaderImplementation::PostProcessingShaderImplementation()
{
m_timer.Start();
}
PostProcessingShaderImplementation::~PostProcessingShaderImplementation()
{
m_timer.Stop();
}
std::string PostProcessingShaderConfiguration::LoadShader(std::string shader)
{
// Load the shader from the configuration if there isn't one sent to us.
if (shader == "")
shader = g_ActiveConfig.sPostProcessingShader;
m_current_shader = shader;
// loading shader code
std::string code;
std::string path = File::GetUserPath(D_SHADERS_IDX) + shader + ".glsl";
if (!File::Exists(path))
{
// Fallback to shared user dir
path = File::GetSysDirectory() + SHADERS_DIR DIR_SEP + shader + ".glsl";
}
if (!File::ReadFileToString(path, code))
{
ERROR_LOG(VIDEO, "Post-processing shader not found: %s", path.c_str());
return "";
}
LoadOptions(code);
std::string error = VerifyOptions(m_options);
if (error.size())
{
PanicAlert("%s", StringFromFormat("Shader %s has a configuration issue!\n%s", shader.c_str(), error.c_str()).c_str());
return "";
}
LoadOptionsConfiguration();
return code;
}
std::string PostProcessingShaderConfiguration::VerifyOptions(const ConfigMap& config_map)
{
for (const auto& option : config_map)
{
// Make sure we have an option name
if (!option.second.m_option_name.size())
return "Option doesn't have a unique identifier via 'OptionName'";
// Make sure we have a correct type
if (option.second.m_type == ConfigurationOption::OptionType::OPTION_INVALID)
return StringFromFormat("Option '%s' has invalid type", option.second.m_option_name.c_str());
// Make sure we have a GUI name
if (!option.second.m_gui_name.size())
return StringFromFormat("Option '%s' has no GUI name", option.second.m_option_name.c_str());
// If we have a dependent option, make sure that option actually exists
if (option.second.m_dependent_option.size())
{
if (config_map.find(option.second.m_dependent_option) == config_map.end())
return StringFromFormat("Option '%s' has a dependent option '%s' that doesn't exist",
option.second.m_option_name.c_str(), option.second.m_dependent_option.c_str());
}
switch (option.second.m_type)
{
case ConfigurationOption::OptionType::OPTION_INTEGER:
{
// Make sure our vectors are the same sizes
if (option.second.m_integer_values.size() != option.second.m_integer_min_values.size() ||
option.second.m_integer_min_values.size() != option.second.m_integer_max_values.size() ||
option.second.m_integer_max_values.size() != option.second.m_integer_step_values.size())
return StringFromFormat("Option '%s' has invalid set value amounts", option.second.m_option_name.c_str());
int option_size = option.second.m_integer_values.size();
// Make sure our minimums are lower than our maximums
for (int i = 0; i < option_size; ++i)
{
if (option.second.m_integer_min_values[i] > option.second.m_integer_max_values[i])
return StringFromFormat("Option '%s' minimum value index %d is greater than max(%d > %d)",
option.second.m_option_name.c_str(), i,
option.second.m_integer_min_values[i], option.second.m_integer_max_values[i]);
}
// Make sure the default value is between minimum and maximum
for (int i = 0; i < option_size; ++i)
{
if (option.second.m_integer_values[i] < option.second.m_integer_min_values[i] ||
option.second.m_integer_values[i] > option.second.m_integer_max_values[i])
return StringFromFormat("Option '%s' default value index %d is outside of available range(%d outside %d - %d)",
option.second.m_option_name.c_str(), i,
option.second.m_integer_values[i],
option.second.m_integer_min_values[i], option.second.m_integer_max_values[i]);
}
// Make sure our step size is smaller than the range acceptable values
for (int i = 0; i < option_size; ++i)
{
if (option.second.m_integer_step_values[i] >
(option.second.m_integer_max_values[i] - option.second.m_integer_min_values[i]))
return StringFromFormat("Option '%s' step value is larger than maximum range of %d",
option.second.m_option_name.c_str(), option.second.m_integer_max_values[i] - option.second.m_integer_min_values[i]);
}
}
break;
case ConfigurationOption::OptionType::OPTION_FLOAT:
{
// Make sure our vectors are the same sizes
if (option.second.m_float_values.size() != option.second.m_float_min_values.size() ||
option.second.m_float_min_values.size() != option.second.m_float_max_values.size() ||
option.second.m_float_max_values.size() != option.second.m_float_step_values.size())
return StringFromFormat("Option '%s' has invalid set value amounts", option.second.m_option_name.c_str());
int option_size = option.second.m_float_values.size();
// Make sure our minimums are lower than our maximums
for (int i = 0; i < option_size; ++i)
{
if (option.second.m_float_min_values[i] > option.second.m_float_max_values[i])
return StringFromFormat("Option '%s' minimum value index %d is greater than max(%f > %f)",
option.second.m_option_name.c_str(), i,
option.second.m_float_min_values[i], option.second.m_float_max_values[i]);
}
// Make sure the default value is between minimum and maximum
for (int i = 0; i < option_size; ++i)
{
if (option.second.m_float_values[i] < option.second.m_float_min_values[i] ||
option.second.m_float_values[i] > option.second.m_float_max_values[i])
return StringFromFormat("Option '%s' default value index %d is outside of available range(%f outside %f - %f)",
option.second.m_option_name.c_str(), i,
option.second.m_float_values[i],
option.second.m_float_min_values[i], option.second.m_float_max_values[i]);
}
// Make sure our step size is smaller than the range acceptable values
for (int i = 0; i < option_size; ++i)
{
if (option.second.m_float_step_values[i] >
(option.second.m_float_max_values[i] - option.second.m_float_min_values[i]))
return StringFromFormat("Option '%s' step value is larger than maximum range of %f",
option.second.m_option_name.c_str(), option.second.m_float_max_values[i] - option.second.m_float_min_values[i]);
}
}
break;
case ConfigurationOption::OptionType::OPTION_BOOL:
case ConfigurationOption::OptionType::OPTION_INVALID: // Won't be hit
default:
break;
}
}
return "";
}
void PostProcessingShaderConfiguration::LoadOptions(const std::string& code)
{
const std::string config_start_delimiter = "[configuration]";
const std::string config_end_delimiter = "[/configuration]";
size_t configuration_start = code.find(config_start_delimiter);
size_t configuration_end = code.find(config_end_delimiter);
m_options.clear();
m_any_options_dirty = true;
if (configuration_start == std::string::npos ||
configuration_end == std::string::npos)
{
// Issue loading configuration or there isn't one.
return;
}
std::string configuration_string = code.substr(configuration_start + config_start_delimiter.size(),
configuration_end - configuration_start - config_start_delimiter.size());
std::istringstream in(configuration_string);
struct GLSLStringOption
{
std::string m_type;
std::vector<std::pair<std::string, std::string>> m_options;
};
std::vector<GLSLStringOption> option_strings;
GLSLStringOption* current_strings = nullptr;
while (!in.eof())
{
std::string line;
if (std::getline(in, line))
{
#ifndef _WIN32
// Check for CRLF eol and convert it to LF
if (!line.empty() && line.at(line.size()-1) == '\r')
{
line.erase(line.size()-1);
}
#endif
if (line.size() > 0)
{
if (line[0] == '[')
{
size_t endpos = line.find("]");
if (endpos != std::string::npos)
{
// New section!
std::string sub = line.substr(1, endpos - 1);
option_strings.push_back({ sub });
current_strings = &option_strings.back();
}
}
else
{
if (current_strings)
{
std::string key, value;
IniFile::ParseLine(line, &key, &value);
if (!(key == "" && value == ""))
current_strings->m_options.push_back(std::make_pair(key, value));
}
}
}
}
}
for (const auto& it : option_strings)
{
ConfigurationOption option;
option.m_dirty = true;
if (it.m_type == "OptionBool")
option.m_type = ConfigurationOption::OptionType::OPTION_BOOL;
else if (it.m_type == "OptionRangeFloat")
option.m_type = ConfigurationOption::OptionType::OPTION_FLOAT;
else if (it.m_type == "OptionRangeInteger")
option.m_type = ConfigurationOption::OptionType::OPTION_INTEGER;
else
option.m_type = ConfigurationOption::OptionType::OPTION_INVALID;
for (const auto& string_option : it.m_options)
{
if (string_option.first == "GUIName")
{
option.m_gui_name = string_option.second;
}
else if (string_option.first == "OptionName")
{
option.m_option_name = string_option.second;
}
else if (string_option.first == "DependentOption")
{
option.m_dependent_option = string_option.second;
}
else if (string_option.first == "MinValue" ||
string_option.first == "MaxValue" ||
string_option.first == "DefaultValue" ||
string_option.first == "StepAmount")
{
std::vector<s32>* output_integer = nullptr;
std::vector<float>* output_float = nullptr;
if (string_option.first == "MinValue")
{
output_integer = &option.m_integer_min_values;
output_float = &option.m_float_min_values;
}
else if (string_option.first == "MaxValue")
{
output_integer = &option.m_integer_max_values;
output_float = &option.m_float_max_values;
}
else if (string_option.first == "DefaultValue")
{
output_integer = &option.m_integer_values;
output_float = &option.m_float_values;
}
else if (string_option.first == "StepAmount")
{
output_integer = &option.m_integer_step_values;
output_float = &option.m_float_step_values;
}
if (option.m_type == ConfigurationOption::OptionType::OPTION_BOOL)
{
TryParse(string_option.second, &option.m_bool_value);
}
else if (option.m_type == ConfigurationOption::OptionType::OPTION_INTEGER)
{
TryParseVector(string_option.second, output_integer);
if (output_integer->size() > 4)
output_integer->erase(output_integer->begin() + 4, output_integer->end());
}
else if (option.m_type == ConfigurationOption::OptionType::OPTION_FLOAT)
{
TryParseVector(string_option.second, output_float);
if (output_float->size() > 4)
output_float->erase(output_float->begin() + 4, output_float->end());
}
}
}
m_options[option.m_option_name] = option;
}
}
void PostProcessingShaderConfiguration::LoadOptionsConfiguration()
{
IniFile ini;
ini.Load(File::GetUserPath(F_DOLPHINCONFIG_IDX));
std::string section = m_current_shader + "-options";
for (auto& it : m_options)
{
switch (it.second.m_type)
{
case ConfigurationOption::OptionType::OPTION_BOOL:
ini.GetOrCreateSection(section)->Get(it.second.m_option_name, &it.second.m_bool_value, it.second.m_bool_value);
break;
case ConfigurationOption::OptionType::OPTION_INTEGER:
{
std::string value;
ini.GetOrCreateSection(section)->Get(it.second.m_option_name, &value);
if (value != "")
TryParseVector(value, &it.second.m_integer_values);
}
break;
case ConfigurationOption::OptionType::OPTION_FLOAT:
{
std::string value;
ini.GetOrCreateSection(section)->Get(it.second.m_option_name, &value);
if (value != "")
TryParseVector(value, &it.second.m_float_values);
}
break;
case ConfigurationOption::OptionType::OPTION_INVALID: // Won't get hit
break;
}
}
}
void PostProcessingShaderConfiguration::SaveOptionsConfiguration()
{
IniFile ini;
ini.Load(File::GetUserPath(F_DOLPHINCONFIG_IDX));
std::string section = m_current_shader + "-options";
for (auto& it : m_options)
{
switch (it.second.m_type)
{
case ConfigurationOption::OptionType::OPTION_BOOL:
{
ini.GetOrCreateSection(section)->Set(it.second.m_option_name, it.second.m_bool_value);
}
break;
case ConfigurationOption::OptionType::OPTION_INTEGER:
{
std::string value = "";
for (size_t i = 0; i < it.second.m_integer_values.size(); ++i)
value += StringFromFormat("%d%s", it.second.m_integer_values[i], i == (it.second.m_integer_values.size() - 1) ? "": ", ");
ini.GetOrCreateSection(section)->Set(it.second.m_option_name, value);
}
break;
case ConfigurationOption::OptionType::OPTION_FLOAT:
{
std::string value = "";
for (size_t i = 0; i < it.second.m_float_values.size(); ++i)
value += StringFromFormat("%f%s", it.second.m_float_values[i], i == (it.second.m_float_values.size() - 1) ? "": ", ");
ini.GetOrCreateSection(section)->Set(it.second.m_option_name, value);
}
break;
case ConfigurationOption::OptionType::OPTION_INVALID: // Won't get hit
break;
}
}
ini.Save(File::GetUserPath(F_DOLPHINCONFIG_IDX));
}
void PostProcessingShaderConfiguration::ReloadShader()
{
m_current_shader = "";
}
void PostProcessingShaderConfiguration::SetOptionf(std::string option, int index, float value)
{
auto it = m_options.find(option);
it->second.m_float_values[index] = value;
it->second.m_dirty = true;
m_any_options_dirty = true;
}
void PostProcessingShaderConfiguration::SetOptioni(std::string option, int index, s32 value)
{
auto it = m_options.find(option);
it->second.m_integer_values[index] = value;
it->second.m_dirty = true;
m_any_options_dirty = true;
}
void PostProcessingShaderConfiguration::SetOptionb(std::string option, bool value)
{
auto it = m_options.find(option);
it->second.m_bool_value = value;
it->second.m_dirty = true;
m_any_options_dirty = true;
}