rpcs3/Utilities/Config.cpp

391 lines
7.2 KiB
C++
Raw Normal View History

#include "stdafx.h"
2016-02-01 21:55:43 +00:00
#include "Config.h"
#include "Utilities/types.h"
2016-02-01 21:55:43 +00:00
#include "util/yaml.hpp"
2016-02-01 21:55:43 +00:00
2017-09-16 17:36:53 +00:00
#include <typeinfo>
#include <charconv>
2017-09-16 17:36:53 +00:00
[[noreturn]] void report_fatal_error(const std::string&);
LOG_CHANNEL(cfg_log, "CFG");
2016-02-01 21:55:43 +00:00
namespace cfg
{
2017-05-20 11:45:02 +00:00
_base::_base(type _type)
2016-02-01 21:55:43 +00:00
: m_type(_type)
{
if (_type != type::node)
{
cfg_log.fatal("Invalid root node" HERE);
2016-02-01 21:55:43 +00:00
}
}
_base::_base(type _type, node* owner, const std::string& name, bool dynamic)
: m_type(_type), m_dynamic(dynamic)
2016-02-01 21:55:43 +00:00
{
2017-05-20 11:45:02 +00:00
for (const auto& pair : owner->m_nodes)
2016-02-01 21:55:43 +00:00
{
2017-05-20 11:45:02 +00:00
if (pair.first == name)
{
cfg_log.fatal("Node already exists: %s" HERE, name);
2017-05-20 11:45:02 +00:00
}
2016-02-01 21:55:43 +00:00
}
2017-05-20 11:45:02 +00:00
owner->m_nodes.emplace_back(name, this);
}
bool _base::from_string(const std::string&, bool)
{
report_fatal_error("from_string() purecall" HERE);
}
2017-05-20 11:45:02 +00:00
bool _base::from_list(std::vector<std::string>&&)
{
report_fatal_error("from_list() purecall" HERE);
2016-02-01 21:55:43 +00:00
}
// Emit YAML
2017-05-20 11:45:02 +00:00
static void encode(YAML::Emitter& out, const class _base& rhs);
2016-02-01 21:55:43 +00:00
// Incrementally load config entries from YAML::Node.
// The config value is preserved if the corresponding YAML node doesn't exist.
static void decode(const YAML::Node& data, class _base& rhs, bool dynamic = false);
2016-02-01 21:55:43 +00:00
}
2017-08-02 10:22:53 +00:00
std::vector<std::string> cfg::make_int_range(s64 min, s64 max)
{
return {std::to_string(min), std::to_string(max)};
}
2016-02-01 21:55:43 +00:00
bool cfg::try_to_int64(s64* out, const std::string& value, s64 min, s64 max)
{
s64 result;
const char* start = &value.front();
const char* end = &value.back() + 1;
int base = 10;
2016-02-01 21:55:43 +00:00
if (start[0] == '0' && (start[1] == 'x' || start[1] == 'X'))
2016-02-01 21:55:43 +00:00
{
// Limited hex support
base = 16;
start += 2;
2016-02-01 21:55:43 +00:00
}
const auto ret = std::from_chars(start, end, result, base);
if (ret.ec != std::errc() || ret.ptr != end)
2016-02-01 21:55:43 +00:00
{
if (out) cfg_log.error("cfg::try_to_int('%s'): invalid integer", value);
2016-02-01 21:55:43 +00:00
return false;
}
if (result < min || result > max)
{
if (out) cfg_log.error("cfg::try_to_int('%s'): out of bounds (%d..%d)", value, min, max);
return false;
}
if (out) *out = result;
return true;
}
std::vector<std::string> cfg::make_uint_range(u64 min, u64 max)
{
return {std::to_string(min), std::to_string(max)};
}
bool cfg::try_to_uint64(u64* out, const std::string& value, u64 min, u64 max)
{
u64 result;
const char* start = &value.front();
const char* end = &value.back() + 1;
int base = 10;
if (start[0] == '0' && (start[1] == 'x' || start[1] == 'X'))
{
// Limited hex support
base = 16;
start += 2;
}
const auto ret = std::from_chars(start, end, result, base);
if (ret.ec != std::errc() || ret.ptr != end)
{
if (out) cfg_log.error("cfg::try_to_int('%s'): invalid integer", value);
return false;
}
if (result < min || result > max)
{
if (out) cfg_log.error("cfg::try_to_int('%s'): out of bounds (%u..%u)", value, min, max);
2016-02-01 21:55:43 +00:00
return false;
}
if (out) *out = result;
return true;
}
bool cfg::try_to_enum_value(u64* out, decltype(&fmt_class_string<int>::format) func, const std::string& value)
{
2017-08-13 15:30:44 +00:00
u64 max = -1;
for (u64 i = 0;; i++)
{
std::string var;
func(var, i);
if (var == value)
{
if (out) *out = i;
return true;
}
std::string hex;
fmt_class_string<u64>::format(hex, i);
if (var == hex)
{
break;
}
2017-08-13 15:30:44 +00:00
max = i;
}
u64 result;
const char* start = &value.front();
const char* end = &value.back() + 1;
int base = 10;
if (start[0] == '0' && (start[1] == 'x' || start[1] == 'X'))
{
// Limited hex support
base = 16;
start += 2;
}
2017-05-20 16:26:22 +00:00
const auto ret = std::from_chars(start, end, result, base);
2017-08-13 15:30:44 +00:00
if (ret.ec != std::errc() || ret.ptr != end)
{
if (out) cfg_log.error("cfg::try_to_enum_value('%s'): invalid enum or integer", value);
return false;
}
if (result > max)
{
if (out) cfg_log.error("cfg::try_to_enum_value('%s'): out of bounds(0..%u)", value, max);
return false;
}
if (out) *out = result;
return true;
}
std::vector<std::string> cfg::try_to_enum_list(decltype(&fmt_class_string<int>::format) func)
{
std::vector<std::string> result;
for (u64 i = 0;; i++)
{
std::string var;
func(var, i);
std::string hex;
fmt_class_string<u64>::format(hex, i);
if (var == hex)
{
break;
}
result.emplace_back(std::move(var));
}
return result;
}
2017-05-20 11:45:02 +00:00
void cfg::encode(YAML::Emitter& out, const cfg::_base& rhs)
2016-02-01 21:55:43 +00:00
{
switch (rhs.get_type())
{
case type::node:
{
out << YAML::BeginMap;
for (const auto& np : static_cast<const node&>(rhs).get_nodes())
{
out << YAML::Key << np.first;
2017-05-13 18:30:37 +00:00
out << YAML::Value;
encode(out, *np.second);
2016-02-01 21:55:43 +00:00
}
out << YAML::EndMap;
return;
}
case type::set:
{
out << YAML::BeginSeq;
for (const auto& str : static_cast<const set_entry&>(rhs).get_set())
{
out << str;
}
out << YAML::EndSeq;
return;
}
2017-05-13 18:30:37 +00:00
case type::log:
{
out << YAML::BeginMap;
for (const auto& np : static_cast<const log_entry&>(rhs).get_map())
{
if (np.second == logs::level::notice) continue;
out << YAML::Key << np.first;
out << YAML::Value << fmt::format("%s", np.second);
}
out << YAML::EndMap;
return;
}
default:
{
out << rhs.to_string();
return;
}
2016-02-01 21:55:43 +00:00
}
}
void cfg::decode(const YAML::Node& data, cfg::_base& rhs, bool dynamic)
2016-02-01 21:55:43 +00:00
{
if (dynamic && !rhs.get_is_dynamic())
{
return;
}
2016-02-01 21:55:43 +00:00
switch (rhs.get_type())
{
case type::node:
{
if (data.IsScalar() || data.IsSequence())
{
return; // ???
}
for (const auto& pair : data)
{
if (!pair.first.IsScalar()) continue;
// Find the key among existing nodes
2017-05-20 11:45:02 +00:00
for (const auto& _pair : static_cast<node&>(rhs).get_nodes())
2016-02-01 21:55:43 +00:00
{
2017-05-20 11:45:02 +00:00
if (_pair.first == pair.first.Scalar())
{
decode(pair.second, *_pair.second, dynamic);
2017-05-20 11:45:02 +00:00
}
2016-02-01 21:55:43 +00:00
}
}
break;
}
case type::set:
{
std::vector<std::string> values;
if (YAML::convert<decltype(values)>::decode(data, values))
{
rhs.from_list(std::move(values));
}
break;
}
2017-05-13 18:30:37 +00:00
case type::log:
{
if (data.IsScalar() || data.IsSequence())
{
return; // ???
}
std::map<std::string, logs::level> values;
for (const auto& pair : data)
{
if (!pair.first.IsScalar() || !pair.second.IsScalar()) continue;
u64 value;
if (cfg::try_to_enum_value(&value, &fmt_class_string<logs::level>::format, pair.second.Scalar()))
{
values.emplace(pair.first.Scalar(), static_cast<logs::level>(static_cast<int>(value)));
}
}
static_cast<log_entry&>(rhs).set_map(std::move(values));
break;
}
2016-02-01 21:55:43 +00:00
default:
{
std::string value;
if (YAML::convert<std::string>::decode(data, value))
{
2020-03-31 18:50:23 +00:00
rhs.from_string(value, dynamic);
2016-02-01 21:55:43 +00:00
}
break; // ???
}
}
}
std::string cfg::node::to_string() const
{
YAML::Emitter out;
cfg::encode(out, *this);
2017-08-02 10:22:53 +00:00
return {out.c_str(), out.size()};
2016-02-01 21:55:43 +00:00
}
bool cfg::node::from_string(const std::string& value, bool dynamic)
2017-08-02 10:23:12 +00:00
{
auto [result, error] = yaml_load(value);
if (error.empty())
{
cfg::decode(result, *this, dynamic);
return true;
}
cfg_log.fatal("Failed to load node: %s", error);
2017-08-02 10:23:12 +00:00
return false;
}
2016-02-01 21:55:43 +00:00
void cfg::node::from_default()
{
for (auto& node : m_nodes)
{
node.second->from_default();
}
}
2017-05-20 11:45:02 +00:00
void cfg::_bool::from_default()
2017-01-24 23:22:19 +00:00
{
2017-05-20 11:45:02 +00:00
m_value = def;
2017-01-24 23:22:19 +00:00
}
2017-05-20 11:45:02 +00:00
void cfg::string::from_default()
2017-01-24 23:22:19 +00:00
{
m_value = m_value.make(def);
2017-01-24 23:22:19 +00:00
}
void cfg::set_entry::from_default()
{
m_set = {};
}
2017-05-13 18:30:37 +00:00
void cfg::log_entry::set_map(std::map<std::string, logs::level>&& map)
{
2020-03-28 14:28:23 +00:00
m_map = std::move(map);
2017-05-13 18:30:37 +00:00
}
void cfg::log_entry::from_default()
{
set_map({});
}