mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-24 17:09:06 +00:00 
			
		
		
		
	Also make the `Decrypt` method private. As far as I can tell, the only motivation for exposing the `SetBytes` and `Reset` methods is to allow `CBoot::SetupWiiMemory` to use the same `SettingsHandler` instance to read settings data and then write it back. It seems cleaner to just use two separate instances, and require a given `SettingsHandler` instance to be used for either writing data to a buffer or reading data from a buffer, but not both. A natural next step is to split the `SettingsHandler` class into two classes, one for writing data and one for reading data. I've deferred that change for a future PR.
		
			
				
	
	
		
			128 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2012 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| // Thanks to Treeki for writing the original class - 29/01/2012
 | |
| 
 | |
| #include "Common/SettingsHandler.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <cstddef>
 | |
| #include <ctime>
 | |
| #include <iomanip>
 | |
| #include <string>
 | |
| 
 | |
| #include <fmt/chrono.h>
 | |
| 
 | |
| #include "Common/CommonTypes.h"
 | |
| 
 | |
| namespace Common
 | |
| {
 | |
| SettingsHandler::SettingsHandler() : m_buffer{}, m_position{0}, m_key{INITIAL_SEED}, decoded{""}
 | |
| {
 | |
| }
 | |
| 
 | |
| SettingsHandler::SettingsHandler(const Buffer& buffer) : SettingsHandler()
 | |
| {
 | |
|   m_buffer = buffer;
 | |
|   Decrypt();
 | |
| }
 | |
| 
 | |
| const SettingsHandler::Buffer& SettingsHandler::GetBytes() const
 | |
| {
 | |
|   return m_buffer;
 | |
| }
 | |
| 
 | |
| std::string SettingsHandler::GetValue(std::string_view key) const
 | |
| {
 | |
|   constexpr char delim[] = "\n";
 | |
|   std::string toFind = std::string(delim).append(key).append("=");
 | |
|   size_t found = decoded.find(toFind);
 | |
| 
 | |
|   if (found != std::string_view::npos)
 | |
|   {
 | |
|     size_t delimFound = decoded.find(delim, found + toFind.length());
 | |
|     if (delimFound == std::string_view::npos)
 | |
|       delimFound = decoded.length() - 1;
 | |
|     return decoded.substr(found + toFind.length(), delimFound - (found + toFind.length()));
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     toFind = std::string(key).append("=");
 | |
|     found = decoded.find(toFind);
 | |
|     if (found == 0)
 | |
|     {
 | |
|       size_t delimFound = decoded.find(delim, found + toFind.length());
 | |
|       if (delimFound == std::string_view::npos)
 | |
|         delimFound = decoded.length() - 1;
 | |
|       return decoded.substr(found + toFind.length(), delimFound - (found + toFind.length()));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return "";
 | |
| }
 | |
| 
 | |
| void SettingsHandler::Decrypt()
 | |
| {
 | |
|   while (m_position < m_buffer.size())
 | |
|   {
 | |
|     decoded.push_back((u8)(m_buffer[m_position] ^ m_key));
 | |
|     m_position++;
 | |
|     m_key = (m_key >> 31) | (m_key << 1);
 | |
|   }
 | |
| 
 | |
|   // The decoded data normally uses CRLF line endings, but occasionally
 | |
|   // (see the comment in WriteLine), lines can be separated by CRLFLF.
 | |
|   // To handle this, we remove every CR and treat LF as the line ending.
 | |
|   // (We ignore empty lines.)
 | |
|   std::erase(decoded, '\x0d');
 | |
| }
 | |
| 
 | |
| void SettingsHandler::AddSetting(std::string_view key, std::string_view value)
 | |
| {
 | |
|   WriteLine(fmt::format("{}={}\r\n", key, value));
 | |
| }
 | |
| 
 | |
| void SettingsHandler::WriteLine(std::string_view str)
 | |
| {
 | |
|   const u32 old_position = m_position;
 | |
|   const u32 old_key = m_key;
 | |
| 
 | |
|   // Encode and write the line
 | |
|   for (char c : str)
 | |
|     WriteByte(c);
 | |
| 
 | |
|   // If the encoded data contains a null byte, Nintendo's decoder will stop at that null byte
 | |
|   // instead of decoding all the data. To avoid this: If the data we just wrote contains
 | |
|   // a null byte, add an LF right before the line to prod the values into being different,
 | |
|   // just like Nintendo does. Due to the chosen key, LF itself never encodes into a null byte.
 | |
|   const auto begin = m_buffer.cbegin() + old_position;
 | |
|   const auto end = m_buffer.cbegin() + m_position;
 | |
|   if (std::find(begin, end, 0) != end)
 | |
|   {
 | |
|     m_key = old_key;
 | |
|     m_position = old_position;
 | |
|     WriteByte('\n');
 | |
|     WriteLine(str);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void SettingsHandler::WriteByte(u8 b)
 | |
| {
 | |
|   if (m_position >= m_buffer.size())
 | |
|     return;
 | |
| 
 | |
|   m_buffer[m_position] = b ^ m_key;
 | |
|   m_position++;
 | |
|   m_key = (m_key >> 31) | (m_key << 1);
 | |
| }
 | |
| 
 | |
| std::string SettingsHandler::GenerateSerialNumber()
 | |
| {
 | |
|   const std::time_t t = std::time(nullptr);
 | |
| 
 | |
|   // Must be 9 characters at most; otherwise the serial number will be rejected by SDK libraries,
 | |
|   // as there is a check to ensure the string length is strictly lower than 10.
 | |
|   // 3 for %j, 2 for %H, 2 for %M, 2 for %S.
 | |
|   return fmt::format("{:%j%H%M%S}", fmt::localtime(t));
 | |
| }
 | |
| }  // namespace Common
 |