mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-22 07:59:44 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			124 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
	
		
			3.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2017 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <mutex>
 | |
| #include <shared_mutex>
 | |
| #include <string>
 | |
| #include <type_traits>
 | |
| #include <utility>
 | |
| 
 | |
| #include "Common/CommonTypes.h"
 | |
| #include "Common/Config/Enums.h"
 | |
| 
 | |
| namespace Config
 | |
| {
 | |
| namespace detail
 | |
| {
 | |
| // std::underlying_type may only be used with enum types, so make sure T is an enum type first.
 | |
| template <typename T>
 | |
| using UnderlyingType = typename std::enable_if_t<std::is_enum<T>{}, std::underlying_type<T>>::type;
 | |
| }  // namespace detail
 | |
| 
 | |
| struct Location
 | |
| {
 | |
|   System system{};
 | |
|   std::string section;
 | |
|   std::string key;
 | |
| 
 | |
|   bool operator==(const Location& other) const;
 | |
|   bool operator!=(const Location& other) const;
 | |
|   bool operator<(const Location& other) const;
 | |
| };
 | |
| 
 | |
| template <typename T>
 | |
| struct CachedValue
 | |
| {
 | |
|   T value;
 | |
|   u64 config_version;
 | |
| };
 | |
| 
 | |
| template <typename T>
 | |
| class Info
 | |
| {
 | |
| public:
 | |
|   constexpr Info(const Location& location, const T& default_value)
 | |
|       : m_location{location}, m_default_value{default_value}, m_cached_value{default_value, 0}
 | |
|   {
 | |
|   }
 | |
| 
 | |
|   Info(const Info<T>& other) { *this = other; }
 | |
| 
 | |
|   // Not thread-safe
 | |
|   Info(Info<T>&& other) { *this = std::move(other); }
 | |
| 
 | |
|   // Make it easy to convert Info<Enum> into Info<UnderlyingType<Enum>>
 | |
|   // so that enum settings can still easily work with code that doesn't care about the enum values.
 | |
|   template <typename Enum,
 | |
|             std::enable_if_t<std::is_same<T, detail::UnderlyingType<Enum>>::value>* = nullptr>
 | |
|   Info(const Info<Enum>& other)
 | |
|   {
 | |
|     *this = other;
 | |
|   }
 | |
| 
 | |
|   Info<T>& operator=(const Info<T>& other)
 | |
|   {
 | |
|     m_location = other.GetLocation();
 | |
|     m_default_value = other.GetDefaultValue();
 | |
|     m_cached_value = other.GetCachedValue();
 | |
|     return *this;
 | |
|   }
 | |
| 
 | |
|   // Not thread-safe
 | |
|   Info<T>& operator=(Info<T>&& other)
 | |
|   {
 | |
|     m_location = std::move(other.m_location);
 | |
|     m_default_value = std::move(other.m_default_value);
 | |
|     m_cached_value = std::move(other.m_cached_value);
 | |
|     return *this;
 | |
|   }
 | |
| 
 | |
|   // Make it easy to convert Info<Enum> into Info<UnderlyingType<Enum>>
 | |
|   // so that enum settings can still easily work with code that doesn't care about the enum values.
 | |
|   template <typename Enum,
 | |
|             std::enable_if_t<std::is_same<T, detail::UnderlyingType<Enum>>::value>* = nullptr>
 | |
|   Info<T>& operator=(const Info<Enum>& other)
 | |
|   {
 | |
|     m_location = other.GetLocation();
 | |
|     m_default_value = static_cast<T>(other.GetDefaultValue());
 | |
|     m_cached_value = other.template GetCachedValueCasted<T>();
 | |
|     return *this;
 | |
|   }
 | |
| 
 | |
|   constexpr const Location& GetLocation() const { return m_location; }
 | |
|   constexpr const T& GetDefaultValue() const { return m_default_value; }
 | |
| 
 | |
|   CachedValue<T> GetCachedValue() const
 | |
|   {
 | |
|     std::shared_lock lock(m_cached_value_mutex);
 | |
|     return m_cached_value;
 | |
|   }
 | |
| 
 | |
|   template <typename U>
 | |
|   CachedValue<U> GetCachedValueCasted() const
 | |
|   {
 | |
|     std::shared_lock lock(m_cached_value_mutex);
 | |
|     return CachedValue<U>{static_cast<U>(m_cached_value.value), m_cached_value.config_version};
 | |
|   }
 | |
| 
 | |
|   void SetCachedValue(const CachedValue<T>& cached_value) const
 | |
|   {
 | |
|     std::unique_lock lock(m_cached_value_mutex);
 | |
|     if (m_cached_value.config_version < cached_value.config_version)
 | |
|       m_cached_value = cached_value;
 | |
|   }
 | |
| 
 | |
| private:
 | |
|   Location m_location;
 | |
|   T m_default_value;
 | |
| 
 | |
|   mutable CachedValue<T> m_cached_value;
 | |
|   mutable std::shared_mutex m_cached_value_mutex;
 | |
| };
 | |
| }  // namespace Config
 |