2014-09-12 17:06:13 -07:00
|
|
|
// Copyright 2014 Citra Emulator Project
|
2014-12-16 21:38:14 -08:00
|
|
|
// Licensed under GPLv2 or any later version
|
2014-09-12 17:06:13 -07:00
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-08-26 19:47:45 +02:00
|
|
|
#include <iomanip>
|
2016-04-05 13:29:55 +01:00
|
|
|
#include <memory>
|
2018-08-26 19:47:45 +02:00
|
|
|
#include <sstream>
|
2023-03-27 14:29:17 +03:00
|
|
|
#include <type_traits>
|
2023-10-17 03:31:56 +01:00
|
|
|
#include <INIReader.h>
|
2016-03-01 17:24:18 +00:00
|
|
|
#include <SDL.h>
|
2017-05-27 18:14:00 -07:00
|
|
|
#include "citra/config.h"
|
2014-09-12 17:06:13 -07:00
|
|
|
#include "citra/default_ini.h"
|
|
|
|
#include "common/file_util.h"
|
2023-07-03 02:17:03 +03:00
|
|
|
#include "common/logging/backend.h"
|
2015-05-06 04:06:12 -03:00
|
|
|
#include "common/logging/log.h"
|
2022-12-08 13:27:25 +02:00
|
|
|
#include "common/settings.h"
|
2018-08-09 14:10:11 -05:00
|
|
|
#include "core/hle/service/service.h"
|
2017-01-21 11:53:03 +02:00
|
|
|
#include "input_common/main.h"
|
2018-01-18 08:43:17 -07:00
|
|
|
#include "input_common/udp/client.h"
|
2020-08-20 04:46:35 -03:00
|
|
|
#include "network/network_settings.h"
|
2014-09-12 17:06:13 -07:00
|
|
|
|
|
|
|
Config::Config() {
|
|
|
|
// TODO: Don't hardcode the path; let the frontend decide where to put the config files.
|
2018-07-21 15:52:42 -04:00
|
|
|
sdl2_config_loc = FileUtil::GetUserPath(FileUtil::UserPath::ConfigDir) + "sdl2-config.ini";
|
2016-04-05 13:29:55 +01:00
|
|
|
sdl2_config = std::make_unique<INIReader>(sdl2_config_loc);
|
2014-09-12 17:06:13 -07:00
|
|
|
|
|
|
|
Reload();
|
|
|
|
}
|
|
|
|
|
2017-05-27 18:14:00 -07:00
|
|
|
Config::~Config() = default;
|
|
|
|
|
2016-03-02 12:49:30 +00:00
|
|
|
bool Config::LoadINI(const std::string& default_contents, bool retry) {
|
Port yuzu-emu/yuzu#2511: "common/file_util: Minor cleanup" (#4782)
* common/file_util: Make IOFile's WriteString take a std::string_view
We don't need to force the usage of a std::string here, and can instead
use a std::string_view, which allows writing out other forms of strings
(e.g. C-style strings) without any unnecessary heap allocations.
* common/file_util: Remove unnecessary c_str() calls
The file stream open functions have supported std::string overloads
since C++11, so we don't need to use c_str() here. Same behavior, less
code.
* common/file_util: Make ReadFileToString and WriteStringToFile consistent
Makes the parameter ordering consistent, and also makes the filename
parameter a std::string. A std::string would be constructed anyways with
the previous code, as IOFile's only constructor with a filepath is one
taking a std::string.
We can also make WriteStringToFile's string parameter utilize a
std::string_view for the string, making use of our previous changes to
IOFile.
* common/file_util: Remove duplicated documentation comments
These are already present within the header, so they don't need to be
repeated in the cpp file.
* common/file_util: Make GetCurrentDir() return a std::optional
nullptr was being returned in the error case, which, at a glance may
seem perfectly OK... until you realize that std::string has the
invariant that it may not be constructed from a null pointer. This
means that if this error case was ever hit, then the application would
most likely crash from a thrown exception in std::string's constructor.
Instead, we can change the function to return an optional value,
indicating if a failure occurred.
* common/file_util: Remove unnecessary return at end of void StripTailDirSlashes()
While we're at it, also invert the conditional into a guard clause.
2019-06-08 00:23:57 +02:00
|
|
|
const std::string& location = this->sdl2_config_loc;
|
2016-03-02 12:49:30 +00:00
|
|
|
if (sdl2_config->ParseError() < 0) {
|
2014-09-12 17:06:13 -07:00
|
|
|
if (retry) {
|
2018-06-29 14:18:07 +03:00
|
|
|
LOG_WARNING(Config, "Failed to load {}. Creating file from defaults...", location);
|
2014-09-12 17:06:13 -07:00
|
|
|
FileUtil::CreateFullPath(location);
|
Port yuzu-emu/yuzu#2511: "common/file_util: Minor cleanup" (#4782)
* common/file_util: Make IOFile's WriteString take a std::string_view
We don't need to force the usage of a std::string here, and can instead
use a std::string_view, which allows writing out other forms of strings
(e.g. C-style strings) without any unnecessary heap allocations.
* common/file_util: Remove unnecessary c_str() calls
The file stream open functions have supported std::string overloads
since C++11, so we don't need to use c_str() here. Same behavior, less
code.
* common/file_util: Make ReadFileToString and WriteStringToFile consistent
Makes the parameter ordering consistent, and also makes the filename
parameter a std::string. A std::string would be constructed anyways with
the previous code, as IOFile's only constructor with a filepath is one
taking a std::string.
We can also make WriteStringToFile's string parameter utilize a
std::string_view for the string, making use of our previous changes to
IOFile.
* common/file_util: Remove duplicated documentation comments
These are already present within the header, so they don't need to be
repeated in the cpp file.
* common/file_util: Make GetCurrentDir() return a std::optional
nullptr was being returned in the error case, which, at a glance may
seem perfectly OK... until you realize that std::string has the
invariant that it may not be constructed from a null pointer. This
means that if this error case was ever hit, then the application would
most likely crash from a thrown exception in std::string's constructor.
Instead, we can change the function to return an optional value,
indicating if a failure occurred.
* common/file_util: Remove unnecessary return at end of void StripTailDirSlashes()
While we're at it, also invert the conditional into a guard clause.
2019-06-08 00:23:57 +02:00
|
|
|
FileUtil::WriteStringToFile(true, location, default_contents);
|
2016-04-05 13:29:55 +01:00
|
|
|
sdl2_config = std::make_unique<INIReader>(location); // Reopen file
|
2014-09-12 17:06:13 -07:00
|
|
|
|
2016-03-02 12:49:30 +00:00
|
|
|
return LoadINI(default_contents, false);
|
2014-09-12 17:06:13 -07:00
|
|
|
}
|
2018-06-29 14:18:07 +03:00
|
|
|
LOG_ERROR(Config, "Failed.");
|
2014-09-12 17:06:13 -07:00
|
|
|
return false;
|
|
|
|
}
|
2018-06-29 14:18:07 +03:00
|
|
|
LOG_INFO(Config, "Successfully loaded {}", location);
|
2014-09-12 17:06:13 -07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-01-21 11:53:03 +02:00
|
|
|
static const std::array<int, Settings::NativeButton::NumButtons> default_buttons = {
|
2018-12-28 15:13:16 -06:00
|
|
|
SDL_SCANCODE_A, SDL_SCANCODE_S, SDL_SCANCODE_Z, SDL_SCANCODE_X, SDL_SCANCODE_T, SDL_SCANCODE_G,
|
|
|
|
SDL_SCANCODE_F, SDL_SCANCODE_H, SDL_SCANCODE_Q, SDL_SCANCODE_W, SDL_SCANCODE_M, SDL_SCANCODE_N,
|
2018-12-28 17:13:37 -06:00
|
|
|
SDL_SCANCODE_O, SDL_SCANCODE_P, SDL_SCANCODE_1, SDL_SCANCODE_2, SDL_SCANCODE_B,
|
2015-06-20 04:34:45 +01:00
|
|
|
};
|
|
|
|
|
2017-01-21 13:04:00 +02:00
|
|
|
static const std::array<std::array<int, 5>, Settings::NativeAnalog::NumAnalogs> default_analogs{{
|
|
|
|
{
|
2018-02-19 17:51:27 -07:00
|
|
|
SDL_SCANCODE_UP,
|
|
|
|
SDL_SCANCODE_DOWN,
|
|
|
|
SDL_SCANCODE_LEFT,
|
|
|
|
SDL_SCANCODE_RIGHT,
|
|
|
|
SDL_SCANCODE_D,
|
2017-01-21 13:04:00 +02:00
|
|
|
},
|
|
|
|
{
|
2018-02-19 17:51:27 -07:00
|
|
|
SDL_SCANCODE_I,
|
|
|
|
SDL_SCANCODE_K,
|
|
|
|
SDL_SCANCODE_J,
|
|
|
|
SDL_SCANCODE_L,
|
|
|
|
SDL_SCANCODE_D,
|
2017-01-21 13:04:00 +02:00
|
|
|
},
|
|
|
|
}};
|
|
|
|
|
2023-03-27 14:29:17 +03:00
|
|
|
template <>
|
|
|
|
void Config::ReadSetting(const std::string& group, Settings::Setting<std::string>& setting) {
|
|
|
|
std::string setting_value = sdl2_config->Get(group, setting.GetLabel(), setting.GetDefault());
|
|
|
|
if (setting_value.empty()) {
|
|
|
|
setting_value = setting.GetDefault();
|
|
|
|
}
|
|
|
|
setting = std::move(setting_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
|
|
|
void Config::ReadSetting(const std::string& group, Settings::Setting<bool>& setting) {
|
|
|
|
setting = sdl2_config->GetBoolean(group, setting.GetLabel(), setting.GetDefault());
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename Type, bool ranged>
|
|
|
|
void Config::ReadSetting(const std::string& group, Settings::Setting<Type, ranged>& setting) {
|
|
|
|
if constexpr (std::is_floating_point_v<Type>) {
|
2023-05-02 01:08:58 +05:30
|
|
|
setting = static_cast<Type>(
|
|
|
|
sdl2_config->GetReal(group, setting.GetLabel(), setting.GetDefault()));
|
2023-03-27 14:29:17 +03:00
|
|
|
} else {
|
|
|
|
setting = static_cast<Type>(sdl2_config->GetInteger(
|
|
|
|
group, setting.GetLabel(), static_cast<long>(setting.GetDefault())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-15 11:56:18 -08:00
|
|
|
void Config::ReadValues() {
|
|
|
|
// Controls
|
2018-12-28 19:24:56 -05:00
|
|
|
// TODO: add multiple input profile support
|
2017-01-21 11:53:03 +02:00
|
|
|
for (int i = 0; i < Settings::NativeButton::NumButtons; ++i) {
|
|
|
|
std::string default_param = InputCommon::GenerateKeyboardParam(default_buttons[i]);
|
2018-12-28 23:31:55 -05:00
|
|
|
Settings::values.current_input_profile.buttons[i] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Controls", Settings::NativeButton::mapping[i], default_param);
|
2018-12-28 23:31:55 -05:00
|
|
|
if (Settings::values.current_input_profile.buttons[i].empty())
|
|
|
|
Settings::values.current_input_profile.buttons[i] = default_param;
|
2015-06-20 04:34:45 +01:00
|
|
|
}
|
2014-09-12 17:06:13 -07:00
|
|
|
|
2017-01-21 13:04:00 +02:00
|
|
|
for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) {
|
|
|
|
std::string default_param = InputCommon::GenerateAnalogParamFromKeys(
|
|
|
|
default_analogs[i][0], default_analogs[i][1], default_analogs[i][2],
|
|
|
|
default_analogs[i][3], default_analogs[i][4], 0.5f);
|
2018-12-28 23:31:55 -05:00
|
|
|
Settings::values.current_input_profile.analogs[i] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Controls", Settings::NativeAnalog::mapping[i], default_param);
|
2018-12-28 23:31:55 -05:00
|
|
|
if (Settings::values.current_input_profile.analogs[i].empty())
|
|
|
|
Settings::values.current_input_profile.analogs[i] = default_param;
|
2017-01-21 13:04:00 +02:00
|
|
|
}
|
|
|
|
|
2018-12-28 23:31:55 -05:00
|
|
|
Settings::values.current_input_profile.motion_device = sdl2_config->GetString(
|
2018-10-12 22:58:30 +08:00
|
|
|
"Controls", "motion_device",
|
|
|
|
"engine:motion_emu,update_period:100,sensitivity:0.01,tilt_clamp:90.0");
|
2018-12-28 23:31:55 -05:00
|
|
|
Settings::values.current_input_profile.touch_device =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Controls", "touch_device", "engine:emu_window");
|
2018-12-28 23:31:55 -05:00
|
|
|
Settings::values.current_input_profile.udp_input_address = sdl2_config->GetString(
|
2018-10-12 22:58:30 +08:00
|
|
|
"Controls", "udp_input_address", InputCommon::CemuhookUDP::DEFAULT_ADDR);
|
2018-12-28 23:31:55 -05:00
|
|
|
Settings::values.current_input_profile.udp_input_port =
|
|
|
|
static_cast<u16>(sdl2_config->GetInteger("Controls", "udp_input_port",
|
|
|
|
InputCommon::CemuhookUDP::DEFAULT_PORT));
|
2024-07-21 12:36:00 +02:00
|
|
|
ReadSetting("Controls", Settings::values.use_artic_base_controller);
|
2017-08-07 00:04:06 +03:00
|
|
|
|
2014-11-15 11:56:18 -08:00
|
|
|
// Core
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Core", Settings::values.use_cpu_jit);
|
|
|
|
ReadSetting("Core", Settings::values.cpu_clock_percentage);
|
2014-10-25 12:54:44 -07:00
|
|
|
|
2015-04-03 15:35:51 -07:00
|
|
|
// Renderer
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Renderer", Settings::values.graphics_api);
|
2023-09-13 01:28:50 +03:00
|
|
|
ReadSetting("Renderer", Settings::values.physical_device);
|
|
|
|
ReadSetting("Renderer", Settings::values.spirv_shader_gen);
|
|
|
|
ReadSetting("Renderer", Settings::values.async_shader_compilation);
|
|
|
|
ReadSetting("Renderer", Settings::values.async_presentation);
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Renderer", Settings::values.use_gles);
|
|
|
|
ReadSetting("Renderer", Settings::values.use_hw_shader);
|
|
|
|
ReadSetting("Renderer", Settings::values.shaders_accurate_mul);
|
|
|
|
ReadSetting("Renderer", Settings::values.use_shader_jit);
|
|
|
|
ReadSetting("Renderer", Settings::values.resolution_factor);
|
|
|
|
ReadSetting("Renderer", Settings::values.use_disk_shader_cache);
|
|
|
|
ReadSetting("Renderer", Settings::values.frame_limit);
|
|
|
|
ReadSetting("Renderer", Settings::values.use_vsync_new);
|
2023-04-21 10:14:55 +03:00
|
|
|
ReadSetting("Renderer", Settings::values.texture_filter);
|
2024-01-02 12:05:22 -08:00
|
|
|
ReadSetting("Renderer", Settings::values.texture_sampling);
|
2024-07-13 00:57:03 +02:00
|
|
|
ReadSetting("Renderer", Settings::values.delay_game_render_thread_us);
|
2023-03-27 14:29:17 +03:00
|
|
|
|
|
|
|
ReadSetting("Renderer", Settings::values.mono_render_option);
|
|
|
|
ReadSetting("Renderer", Settings::values.render_3d);
|
|
|
|
ReadSetting("Renderer", Settings::values.factor_3d);
|
|
|
|
ReadSetting("Renderer", Settings::values.pp_shader_name);
|
|
|
|
ReadSetting("Renderer", Settings::values.anaglyph_shader_name);
|
|
|
|
ReadSetting("Renderer", Settings::values.filter_mode);
|
|
|
|
|
|
|
|
ReadSetting("Renderer", Settings::values.bg_red);
|
|
|
|
ReadSetting("Renderer", Settings::values.bg_green);
|
|
|
|
ReadSetting("Renderer", Settings::values.bg_blue);
|
2015-04-03 15:35:51 -07:00
|
|
|
|
2016-05-03 00:07:17 -06:00
|
|
|
// Layout
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Layout", Settings::values.layout_option);
|
|
|
|
ReadSetting("Layout", Settings::values.swap_screen);
|
|
|
|
ReadSetting("Layout", Settings::values.upright_screen);
|
|
|
|
ReadSetting("Layout", Settings::values.large_screen_proportion);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_layout);
|
2024-05-26 21:17:37 +01:00
|
|
|
ReadSetting("Layout", Settings::values.custom_top_x);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_top_y);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_top_width);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_top_height);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_bottom_x);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_bottom_y);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_bottom_width);
|
|
|
|
ReadSetting("Layout", Settings::values.custom_bottom_height);
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Layout", Settings::values.custom_second_layer_opacity);
|
2016-05-03 00:07:17 -06:00
|
|
|
|
2019-04-02 20:57:13 -04:00
|
|
|
// Utility
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Utility", Settings::values.dump_textures);
|
|
|
|
ReadSetting("Utility", Settings::values.custom_textures);
|
|
|
|
ReadSetting("Utility", Settings::values.preload_textures);
|
2023-04-27 07:38:28 +03:00
|
|
|
ReadSetting("Utility", Settings::values.async_custom_loading);
|
2019-04-02 20:57:13 -04:00
|
|
|
|
2016-04-27 13:53:23 +01:00
|
|
|
// Audio
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Audio", Settings::values.audio_emulation);
|
|
|
|
ReadSetting("Audio", Settings::values.enable_audio_stretching);
|
2024-05-16 12:33:00 +01:00
|
|
|
ReadSetting("Audio", Settings::values.enable_realtime_audio);
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Audio", Settings::values.volume);
|
2023-05-01 12:17:45 -07:00
|
|
|
ReadSetting("Audio", Settings::values.output_type);
|
|
|
|
ReadSetting("Audio", Settings::values.output_device);
|
|
|
|
ReadSetting("Audio", Settings::values.input_type);
|
|
|
|
ReadSetting("Audio", Settings::values.input_device);
|
2016-04-27 13:53:23 +01:00
|
|
|
|
2014-11-15 11:56:18 -08:00
|
|
|
// Data Storage
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Data Storage", Settings::values.use_virtual_sd);
|
|
|
|
ReadSetting("Data Storage", Settings::values.use_custom_storage);
|
2022-10-22 10:39:47 -03:00
|
|
|
|
|
|
|
if (Settings::values.use_custom_storage) {
|
|
|
|
FileUtil::UpdateUserPath(FileUtil::UserPath::NANDDir,
|
|
|
|
sdl2_config->GetString("Data Storage", "nand_directory", ""));
|
|
|
|
FileUtil::UpdateUserPath(FileUtil::UserPath::SDMCDir,
|
|
|
|
sdl2_config->GetString("Data Storage", "sdmc_directory", ""));
|
|
|
|
}
|
2014-10-09 19:43:40 -07:00
|
|
|
|
2016-04-20 18:12:05 +08:00
|
|
|
// System
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("System", Settings::values.is_new_3ds);
|
2024-01-20 12:13:06 -08:00
|
|
|
ReadSetting("System", Settings::values.lle_applets);
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("System", Settings::values.region_value);
|
|
|
|
ReadSetting("System", Settings::values.init_clock);
|
2018-08-26 19:47:45 +02:00
|
|
|
{
|
|
|
|
std::tm t;
|
|
|
|
t.tm_sec = 1;
|
|
|
|
t.tm_min = 0;
|
|
|
|
t.tm_hour = 0;
|
|
|
|
t.tm_mday = 1;
|
|
|
|
t.tm_mon = 0;
|
|
|
|
t.tm_year = 100;
|
|
|
|
t.tm_isdst = 0;
|
|
|
|
std::istringstream string_stream(
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("System", "init_time", "2000-01-01 00:00:01"));
|
2018-08-26 19:47:45 +02:00
|
|
|
string_stream >> std::get_time(&t, "%Y-%m-%d %H:%M:%S");
|
|
|
|
if (string_stream.fail()) {
|
|
|
|
LOG_ERROR(Config, "Failed To parse init_time. Using 2000-01-01 00:00:01");
|
|
|
|
}
|
|
|
|
Settings::values.init_time =
|
|
|
|
std::chrono::duration_cast<std::chrono::seconds>(
|
|
|
|
std::chrono::system_clock::from_time_t(std::mktime(&t)).time_since_epoch())
|
|
|
|
.count();
|
|
|
|
}
|
2024-01-07 09:38:02 -08:00
|
|
|
ReadSetting("System", Settings::values.init_ticks_type);
|
|
|
|
ReadSetting("System", Settings::values.init_ticks_override);
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("System", Settings::values.plugin_loader_enabled);
|
|
|
|
ReadSetting("System", Settings::values.allow_plugin_loader);
|
2024-08-20 22:04:29 +03:00
|
|
|
ReadSetting("System", Settings::values.steps_per_hour);
|
2015-01-31 15:11:51 -08:00
|
|
|
|
2022-11-20 16:34:53 +01:00
|
|
|
{
|
|
|
|
constexpr const char* default_init_time_offset = "0 00:00:00";
|
|
|
|
|
|
|
|
std::string offset_string =
|
|
|
|
sdl2_config->GetString("System", "init_time_offset", default_init_time_offset);
|
|
|
|
|
2024-01-07 17:37:42 -03:00
|
|
|
std::size_t sep_index = offset_string.find(' ');
|
2022-11-20 16:34:53 +01:00
|
|
|
|
|
|
|
if (sep_index == std::string::npos) {
|
|
|
|
LOG_ERROR(Config, "Failed to parse init_time_offset. Using 0 00:00:00");
|
|
|
|
offset_string = default_init_time_offset;
|
|
|
|
|
|
|
|
sep_index = offset_string.find(' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string day_string = offset_string.substr(0, sep_index);
|
|
|
|
long long days = 0;
|
|
|
|
|
|
|
|
try {
|
|
|
|
days = std::stoll(day_string);
|
2022-11-20 22:41:33 -03:00
|
|
|
} catch (std::logic_error&) {
|
2022-11-20 16:34:53 +01:00
|
|
|
LOG_ERROR(Config, "Failed to parse days in init_time_offset. Using 0");
|
|
|
|
days = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
long long days_in_seconds = days * 86400;
|
|
|
|
|
|
|
|
std::tm t;
|
|
|
|
t.tm_sec = 0;
|
|
|
|
t.tm_min = 0;
|
|
|
|
t.tm_hour = 0;
|
|
|
|
t.tm_mday = 1;
|
|
|
|
t.tm_mon = 0;
|
|
|
|
t.tm_year = 100;
|
|
|
|
t.tm_isdst = 0;
|
|
|
|
|
|
|
|
std::istringstream string_stream(offset_string.substr(sep_index + 1));
|
|
|
|
string_stream >> std::get_time(&t, "%H:%M:%S");
|
|
|
|
|
|
|
|
if (string_stream.fail()) {
|
|
|
|
LOG_ERROR(Config,
|
|
|
|
"Failed to parse hours, minutes and seconds in init_time_offset. 00:00:00");
|
|
|
|
}
|
|
|
|
|
|
|
|
auto time_offset =
|
|
|
|
std::chrono::system_clock::from_time_t(std::mktime(&t)).time_since_epoch();
|
|
|
|
|
|
|
|
auto secs = std::chrono::duration_cast<std::chrono::seconds>(time_offset).count();
|
|
|
|
|
|
|
|
Settings::values.init_time_offset = static_cast<long long>(secs) + days_in_seconds;
|
|
|
|
}
|
|
|
|
|
2016-12-21 20:05:56 +02:00
|
|
|
// Camera
|
|
|
|
using namespace Service::CAM;
|
|
|
|
Settings::values.camera_name[OuterRightCamera] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Camera", "camera_outer_right_name", "blank");
|
2016-12-21 20:05:56 +02:00
|
|
|
Settings::values.camera_config[OuterRightCamera] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Camera", "camera_outer_right_config", "");
|
2018-05-20 09:07:37 +08:00
|
|
|
Settings::values.camera_flip[OuterRightCamera] =
|
|
|
|
sdl2_config->GetInteger("Camera", "camera_outer_right_flip", 0);
|
2016-12-21 20:05:56 +02:00
|
|
|
Settings::values.camera_name[InnerCamera] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Camera", "camera_inner_name", "blank");
|
2016-12-21 20:05:56 +02:00
|
|
|
Settings::values.camera_config[InnerCamera] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Camera", "camera_inner_config", "");
|
2018-05-20 09:07:37 +08:00
|
|
|
Settings::values.camera_flip[InnerCamera] =
|
|
|
|
sdl2_config->GetInteger("Camera", "camera_inner_flip", 0);
|
2016-12-21 20:05:56 +02:00
|
|
|
Settings::values.camera_name[OuterLeftCamera] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Camera", "camera_outer_left_name", "blank");
|
2016-12-21 20:05:56 +02:00
|
|
|
Settings::values.camera_config[OuterLeftCamera] =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("Camera", "camera_outer_left_config", "");
|
2018-05-20 09:07:37 +08:00
|
|
|
Settings::values.camera_flip[OuterLeftCamera] =
|
|
|
|
sdl2_config->GetInteger("Camera", "camera_outer_left_flip", 0);
|
2016-12-21 20:05:56 +02:00
|
|
|
|
2014-11-15 11:56:18 -08:00
|
|
|
// Miscellaneous
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Miscellaneous", Settings::values.log_filter);
|
2024-03-05 14:51:58 +01:00
|
|
|
ReadSetting("Miscellaneous", Settings::values.log_regex_filter);
|
2015-09-02 08:56:38 -04:00
|
|
|
|
2023-07-03 02:17:03 +03:00
|
|
|
// Apply the log_filter setting as the logger has already been initialized
|
|
|
|
// and doesn't pick up the filter on its own.
|
|
|
|
Common::Log::Filter filter;
|
|
|
|
filter.ParseFilterString(Settings::values.log_filter.GetValue());
|
|
|
|
Common::Log::SetGlobalFilter(filter);
|
2024-03-05 14:51:58 +01:00
|
|
|
Common::Log::SetRegexFilter(Settings::values.log_regex_filter.GetValue());
|
2023-07-03 02:17:03 +03:00
|
|
|
|
2015-10-11 20:07:58 -04:00
|
|
|
// Debugging
|
2019-08-12 22:15:00 -06:00
|
|
|
Settings::values.record_frame_times =
|
|
|
|
sdl2_config->GetBoolean("Debugging", "record_frame_times", false);
|
2023-03-27 14:29:17 +03:00
|
|
|
ReadSetting("Debugging", Settings::values.renderer_debug);
|
|
|
|
ReadSetting("Debugging", Settings::values.use_gdbstub);
|
|
|
|
ReadSetting("Debugging", Settings::values.gdbstub_port);
|
2024-07-21 12:36:00 +02:00
|
|
|
ReadSetting("Debugging", Settings::values.instant_debug_log);
|
2017-06-27 22:46:52 -04:00
|
|
|
|
2018-08-09 14:10:11 -05:00
|
|
|
for (const auto& service_module : Service::service_module_map) {
|
|
|
|
bool use_lle = sdl2_config->GetBoolean("Debugging", "LLE\\" + service_module.name, false);
|
|
|
|
Settings::values.lle_modules.emplace(service_module.name, use_lle);
|
|
|
|
}
|
|
|
|
|
2017-06-27 22:46:52 -04:00
|
|
|
// Web Service
|
2020-08-20 04:46:35 -03:00
|
|
|
NetSettings::values.web_api_url =
|
2018-10-12 22:58:30 +08:00
|
|
|
sdl2_config->GetString("WebService", "web_api_url", "https://api.citra-emu.org");
|
2020-08-20 04:46:35 -03:00
|
|
|
NetSettings::values.citra_username = sdl2_config->GetString("WebService", "citra_username", "");
|
|
|
|
NetSettings::values.citra_token = sdl2_config->GetString("WebService", "citra_token", "");
|
2020-01-29 14:57:16 +08:00
|
|
|
|
|
|
|
// Video Dumping
|
|
|
|
Settings::values.output_format =
|
|
|
|
sdl2_config->GetString("Video Dumping", "output_format", "webm");
|
|
|
|
Settings::values.format_options = sdl2_config->GetString("Video Dumping", "format_options", "");
|
|
|
|
|
|
|
|
Settings::values.video_encoder =
|
|
|
|
sdl2_config->GetString("Video Dumping", "video_encoder", "libvpx-vp9");
|
|
|
|
|
|
|
|
// Options for variable bit rate live streaming taken from here:
|
|
|
|
// https://developers.google.com/media/vp9/live-encoding
|
|
|
|
std::string default_video_options;
|
|
|
|
if (Settings::values.video_encoder == "libvpx-vp9") {
|
|
|
|
default_video_options =
|
|
|
|
"quality:realtime,speed:6,tile-columns:4,frame-parallel:1,threads:8,row-mt:1";
|
|
|
|
}
|
|
|
|
Settings::values.video_encoder_options =
|
|
|
|
sdl2_config->GetString("Video Dumping", "video_encoder_options", default_video_options);
|
|
|
|
Settings::values.video_bitrate =
|
|
|
|
sdl2_config->GetInteger("Video Dumping", "video_bitrate", 2500000);
|
|
|
|
|
|
|
|
Settings::values.audio_encoder =
|
|
|
|
sdl2_config->GetString("Video Dumping", "audio_encoder", "libvorbis");
|
|
|
|
Settings::values.audio_encoder_options =
|
|
|
|
sdl2_config->GetString("Video Dumping", "audio_encoder_options", "");
|
|
|
|
Settings::values.audio_bitrate =
|
|
|
|
sdl2_config->GetInteger("Video Dumping", "audio_bitrate", 64000);
|
2014-10-27 17:18:28 -04:00
|
|
|
}
|
|
|
|
|
2014-09-12 17:06:13 -07:00
|
|
|
void Config::Reload() {
|
2016-03-02 12:49:30 +00:00
|
|
|
LoadINI(DefaultINI::sdl2_config_file);
|
2014-11-15 11:56:18 -08:00
|
|
|
ReadValues();
|
2014-09-12 17:06:13 -07:00
|
|
|
}
|