From 8f19ac2deeba1c37ecb269ff813aa438be881fd4 Mon Sep 17 00:00:00 2001 From: Stenzek Date: Sat, 18 Jan 2025 13:56:52 +1000 Subject: [PATCH] System: Set rapidyaml error handlers on startup That way it can be used in multiple places. --- src/common/ryml_helpers.h | 64 ++++++++++++++++++++++++++------------ src/core/game_database.cpp | 15 +-------- src/core/system.cpp | 4 +++ 3 files changed, 49 insertions(+), 34 deletions(-) diff --git a/src/common/ryml_helpers.h b/src/common/ryml_helpers.h index 3fad9de8d..fbc5c3a41 100644 --- a/src/common/ryml_helpers.h +++ b/src/common/ryml_helpers.h @@ -1,7 +1,9 @@ -// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin +// SPDX-FileCopyrightText: 2019-2025 Connor McLaughlin // SPDX-License-Identifier: CC-BY-NC-ND-4.0 +#include "log.h" #include "types.h" +#include "string_util.h" #include "ryml.hpp" @@ -10,23 +12,22 @@ // RapidYAML utility routines. -[[maybe_unused]] ALWAYS_INLINE std::string_view to_stringview(const c4::csubstr& s) +static inline std::string_view to_stringview(const c4::csubstr& s) { return std::string_view(s.data(), s.size()); } -[[maybe_unused]] ALWAYS_INLINE std::string_view to_stringview(const c4::substr& s) +static inline std::string_view to_stringview(const c4::substr& s) { return std::string_view(s.data(), s.size()); } -[[maybe_unused]] ALWAYS_INLINE c4::csubstr to_csubstr(std::string_view sv) +static inline c4::csubstr to_csubstr(std::string_view sv) { return c4::csubstr(sv.data(), sv.length()); } -[[maybe_unused]] static bool GetStringFromObject(const ryml::ConstNodeRef& object, std::string_view key, - std::string* dest) +static inline bool GetStringFromObject(const ryml::ConstNodeRef& object, std::string_view key, std::string* dest) { dest->clear(); @@ -41,8 +42,7 @@ return true; } -[[maybe_unused]] static bool GetStringFromObject(const ryml::ConstNodeRef& object, std::string_view key, - std::string_view* dest) +static inline bool GetStringFromObject(const ryml::ConstNodeRef& object, std::string_view key, std::string_view* dest) { const ryml::ConstNodeRef member = object.find_child(to_csubstr(key)); if (!member.valid()) @@ -56,7 +56,7 @@ } template -[[maybe_unused]] static bool GetUIntFromObject(const ryml::ConstNodeRef& object, std::string_view key, T* dest) +static inline bool GetUIntFromObject(const ryml::ConstNodeRef& object, std::string_view key, T* dest) { *dest = 0; @@ -67,14 +67,15 @@ template const c4::csubstr val = member.val(); if (val.empty()) { - ERROR_LOG("Unexpected empty value in {}", key); + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, "Unexpected empty value in {}", key); return false; } const std::optional opt_value = StringUtil::FromChars(to_stringview(val)); if (!opt_value.has_value()) { - ERROR_LOG("Unexpected non-uint value in {}", key); + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, "Unexpected non-uint value in {}", + key); return false; } @@ -83,7 +84,7 @@ template } template -[[maybe_unused]] static std::optional GetOptionalTFromObject(const ryml::ConstNodeRef& object, std::string_view key) +static inline std::optional GetOptionalTFromObject(const ryml::ConstNodeRef& object, std::string_view key) { std::optional ret; @@ -97,14 +98,21 @@ template if (!ret.has_value()) { if constexpr (std::is_floating_point_v) - ERROR_LOG("Unexpected non-float value in {}", key); + { + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, + "Unexpected non-float value in {}", key); + } else if constexpr (std::is_integral_v) - ERROR_LOG("Unexpected non-int value in {}", key); + { + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, + "Unexpected non-int value in {}", key); + } } } else { - ERROR_LOG("Unexpected empty value in {}", key); + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, "Unexpected empty value in {}", + key); } } @@ -112,9 +120,8 @@ template } template -[[maybe_unused]] static std::optional -ParseOptionalTFromObject(const ryml::ConstNodeRef& object, std::string_view key, - std::optional (*from_string_function)(const char* str)) +static inline std::optional ParseOptionalTFromObject(const ryml::ConstNodeRef& object, std::string_view key, + std::optional (*from_string_function)(const char* str)) { std::optional ret; @@ -126,13 +133,30 @@ ParseOptionalTFromObject(const ryml::ConstNodeRef& object, std::string_view key, { ret = from_string_function(TinyString(to_stringview(val))); if (!ret.has_value()) - ERROR_LOG("Unknown value for {}: {}", key, to_stringview(val)); + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, "Unknown value for {}: {}", key, + to_stringview(val)); } else { - ERROR_LOG("Unexpected empty value in {}", key); + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, "Unexpected empty value in {}", + key); } } return ret; } + +static inline void SetRymlCallbacks() +{ + ryml::Callbacks callbacks = ryml::get_callbacks(); + callbacks.m_error = [](const char* msg, size_t msg_len, ryml::Location loc, void* userdata) { + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, + "YAML parse error at {}:{} (bufpos={}): {}", loc.line, loc.col, loc.offset, + std::string_view(msg, msg_len)); + }; + ryml::set_callbacks(callbacks); + c4::set_error_callback([](const char* msg, size_t msg_size) { + Log::FastWrite(Log::Channel::Log, Log::Level::Error, Log::Color::StrongOrange, "C4 error: {}", + std::string_view(msg, msg_size)); + }); +} diff --git a/src/core/game_database.cpp b/src/core/game_database.cpp index fcb15c7ac..b10472bed 100644 --- a/src/core/game_database.cpp +++ b/src/core/game_database.cpp @@ -16,6 +16,7 @@ #include "common/heterogeneous_containers.h" #include "common/log.h" #include "common/path.h" +#include "common/ryml_helpers.h" #include "common/string_util.h" #include "common/timer.h" @@ -34,8 +35,6 @@ LOG_CHANNEL(GameDatabase); -#include "common/ryml_helpers.h" - namespace GameDatabase { enum : u32 @@ -49,7 +48,6 @@ static const Entry* GetEntryForId(std::string_view code); static bool LoadFromCache(); static bool SaveToCache(); -static void SetRymlCallbacks(); static bool LoadGameDBYaml(); static bool ParseYamlEntry(Entry* entry, const ryml::ConstNodeRef& value); static bool ParseYamlCodes(PreferUnorderedStringMap& lookup, const ryml::ConstNodeRef& value, @@ -1109,17 +1107,6 @@ bool GameDatabase::SaveToCache() return true; } -void GameDatabase::SetRymlCallbacks() -{ - ryml::Callbacks callbacks = ryml::get_callbacks(); - callbacks.m_error = [](const char* msg, size_t msg_len, ryml::Location loc, void* userdata) { - ERROR_LOG("Parse error at {}:{} (bufpos={}): {}", loc.line, loc.col, loc.offset, std::string_view(msg, msg_len)); - }; - ryml::set_callbacks(callbacks); - c4::set_error_callback( - [](const char* msg, size_t msg_size) { ERROR_LOG("C4 error: {}", std::string_view(msg, msg_size)); }); -} - bool GameDatabase::LoadGameDBYaml() { Error error; diff --git a/src/core/system.cpp b/src/core/system.cpp index bcead5cb1..bfe470444 100644 --- a/src/core/system.cpp +++ b/src/core/system.cpp @@ -62,6 +62,7 @@ #include "common/log.h" #include "common/memmap.h" #include "common/path.h" +#include "common/ryml_helpers.h" #include "common/string_util.h" #include "common/task_queue.h" #include "common/timer.h" @@ -486,6 +487,9 @@ bool System::ProcessStartup(Error* error) CheckCacheLineSize(); + // Initialize rapidyaml before anything can use it. + SetRymlCallbacks(); + return true; }