mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 09:29:43 +00:00 
			
		
		
		
	The way Config::Get works in master, it first calls Config::GetActiveLayerForConfig which searches for the setting in all layers, and then calls Config::Layer::Get which searches for the same setting again within the given layer. We can remove this second search by combining the logic of Config::GetActiveLayerForConfig and Config::Layer::Get into one function.
		
			
				
	
	
		
			213 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2016 Dolphin Emulator Project
 | |
| // Licensed under GPLv2+
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <list>
 | |
| #include <map>
 | |
| #include <mutex>
 | |
| #include <shared_mutex>
 | |
| 
 | |
| #include "Common/Config/Config.h"
 | |
| 
 | |
| namespace Config
 | |
| {
 | |
| using Layers = std::map<LayerType, std::shared_ptr<Layer>>;
 | |
| 
 | |
| static Layers s_layers;
 | |
| static std::list<ConfigChangedCallback> s_callbacks;
 | |
| static u32 s_callback_guards = 0;
 | |
| 
 | |
| static std::shared_mutex s_layers_rw_lock;
 | |
| 
 | |
| using ReadLock = std::shared_lock<std::shared_mutex>;
 | |
| using WriteLock = std::unique_lock<std::shared_mutex>;
 | |
| 
 | |
| static void AddLayerInternal(std::shared_ptr<Layer> layer)
 | |
| {
 | |
|   {
 | |
|     WriteLock lock(s_layers_rw_lock);
 | |
| 
 | |
|     const Config::LayerType layer_type = layer->GetLayer();
 | |
|     s_layers.insert_or_assign(layer_type, std::move(layer));
 | |
|   }
 | |
|   InvokeConfigChangedCallbacks();
 | |
| }
 | |
| 
 | |
| void AddLayer(std::unique_ptr<ConfigLayerLoader> loader)
 | |
| {
 | |
|   AddLayerInternal(std::make_shared<Layer>(std::move(loader)));
 | |
| }
 | |
| 
 | |
| std::shared_ptr<Layer> GetLayer(LayerType layer)
 | |
| {
 | |
|   ReadLock lock(s_layers_rw_lock);
 | |
| 
 | |
|   std::shared_ptr<Layer> result;
 | |
|   const auto it = s_layers.find(layer);
 | |
|   if (it != s_layers.end())
 | |
|   {
 | |
|     result = it->second;
 | |
|   }
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| void RemoveLayer(LayerType layer)
 | |
| {
 | |
|   {
 | |
|     WriteLock lock(s_layers_rw_lock);
 | |
| 
 | |
|     s_layers.erase(layer);
 | |
|   }
 | |
|   InvokeConfigChangedCallbacks();
 | |
| }
 | |
| 
 | |
| void AddConfigChangedCallback(ConfigChangedCallback func)
 | |
| {
 | |
|   s_callbacks.emplace_back(std::move(func));
 | |
| }
 | |
| 
 | |
| void InvokeConfigChangedCallbacks()
 | |
| {
 | |
|   if (s_callback_guards)
 | |
|     return;
 | |
| 
 | |
|   for (const auto& callback : s_callbacks)
 | |
|     callback();
 | |
| }
 | |
| 
 | |
| // Explicit load and save of layers
 | |
| void Load()
 | |
| {
 | |
|   {
 | |
|     ReadLock lock(s_layers_rw_lock);
 | |
| 
 | |
|     for (auto& layer : s_layers)
 | |
|       layer.second->Load();
 | |
|   }
 | |
|   InvokeConfigChangedCallbacks();
 | |
| }
 | |
| 
 | |
| void Save()
 | |
| {
 | |
|   {
 | |
|     ReadLock lock(s_layers_rw_lock);
 | |
| 
 | |
|     for (auto& layer : s_layers)
 | |
|       layer.second->Save();
 | |
|   }
 | |
|   InvokeConfigChangedCallbacks();
 | |
| }
 | |
| 
 | |
| void Init()
 | |
| {
 | |
|   // These layers contain temporary values
 | |
|   ClearCurrentRunLayer();
 | |
| }
 | |
| 
 | |
| void Shutdown()
 | |
| {
 | |
|   WriteLock lock(s_layers_rw_lock);
 | |
| 
 | |
|   s_layers.clear();
 | |
|   s_callbacks.clear();
 | |
| }
 | |
| 
 | |
| void ClearCurrentRunLayer()
 | |
| {
 | |
|   WriteLock lock(s_layers_rw_lock);
 | |
| 
 | |
|   s_layers.insert_or_assign(LayerType::CurrentRun, std::make_shared<Layer>(LayerType::CurrentRun));
 | |
| }
 | |
| 
 | |
| static const std::map<System, std::string> system_to_name = {
 | |
|     {System::Main, "Dolphin"},
 | |
|     {System::GCPad, "GCPad"},
 | |
|     {System::WiiPad, "Wiimote"},
 | |
|     {System::GCKeyboard, "GCKeyboard"},
 | |
|     {System::GFX, "Graphics"},
 | |
|     {System::Logger, "Logger"},
 | |
|     {System::Debugger, "Debugger"},
 | |
|     {System::SYSCONF, "SYSCONF"},
 | |
|     {System::DualShockUDPClient, "DualShockUDPClient"}};
 | |
| 
 | |
| const std::string& GetSystemName(System system)
 | |
| {
 | |
|   return system_to_name.at(system);
 | |
| }
 | |
| 
 | |
| std::optional<System> GetSystemFromName(const std::string& name)
 | |
| {
 | |
|   const auto system = std::find_if(system_to_name.begin(), system_to_name.end(),
 | |
|                                    [&name](const auto& entry) { return entry.second == name; });
 | |
|   if (system != system_to_name.end())
 | |
|     return system->first;
 | |
| 
 | |
|   return {};
 | |
| }
 | |
| 
 | |
| const std::string& GetLayerName(LayerType layer)
 | |
| {
 | |
|   static const std::map<LayerType, std::string> layer_to_name = {
 | |
|       {LayerType::Base, "Base"},
 | |
|       {LayerType::GlobalGame, "Global GameINI"},
 | |
|       {LayerType::LocalGame, "Local GameINI"},
 | |
|       {LayerType::Netplay, "Netplay"},
 | |
|       {LayerType::Movie, "Movie"},
 | |
|       {LayerType::CommandLine, "Command Line"},
 | |
|       {LayerType::CurrentRun, "Current Run"},
 | |
|   };
 | |
|   return layer_to_name.at(layer);
 | |
| }
 | |
| 
 | |
| LayerType GetActiveLayerForConfig(const Location& config)
 | |
| {
 | |
|   ReadLock lock(s_layers_rw_lock);
 | |
| 
 | |
|   for (auto layer : SEARCH_ORDER)
 | |
|   {
 | |
|     const auto it = s_layers.find(layer);
 | |
|     if (it != s_layers.end())
 | |
|     {
 | |
|       if (it->second->Exists(config))
 | |
|         return layer;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If config is not present in any layer, base layer is considered active.
 | |
|   return LayerType::Base;
 | |
| }
 | |
| 
 | |
| std::optional<std::string> GetAsString(const Location& config)
 | |
| {
 | |
|   std::optional<std::string> result;
 | |
|   ReadLock lock(s_layers_rw_lock);
 | |
| 
 | |
|   for (auto layer : SEARCH_ORDER)
 | |
|   {
 | |
|     const auto it = s_layers.find(layer);
 | |
|     if (it != s_layers.end())
 | |
|     {
 | |
|       result = it->second->Get<std::string>(config);
 | |
|       if (result.has_value())
 | |
|         break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| ConfigChangeCallbackGuard::ConfigChangeCallbackGuard()
 | |
| {
 | |
|   ++s_callback_guards;
 | |
| }
 | |
| 
 | |
| ConfigChangeCallbackGuard::~ConfigChangeCallbackGuard()
 | |
| {
 | |
|   if (--s_callback_guards)
 | |
|     return;
 | |
| 
 | |
|   InvokeConfigChangedCallbacks();
 | |
| }
 | |
| 
 | |
| }  // namespace Config
 |