2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2012 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2012-01-31 17:16:31 +13:00
|
|
|
|
|
|
|
// Thanks to Treeki for writing the original class - 29/01/2012
|
|
|
|
|
2018-05-12 13:43:59 -04:00
|
|
|
#include "Common/SettingsHandler.h"
|
|
|
|
|
2020-03-24 16:58:54 +01:00
|
|
|
#include <algorithm>
|
2014-02-20 04:11:52 +01:00
|
|
|
#include <cstddef>
|
2014-02-19 01:54:11 +01:00
|
|
|
#include <ctime>
|
2017-08-08 23:23:46 +08:00
|
|
|
#include <iomanip>
|
2014-02-20 04:11:52 +01:00
|
|
|
#include <string>
|
2012-01-29 21:41:35 +13:00
|
|
|
|
2020-02-04 15:00:33 -05:00
|
|
|
#include <fmt/chrono.h>
|
2019-07-16 04:23:18 -04:00
|
|
|
|
2014-02-20 04:11:52 +01:00
|
|
|
#include "Common/CommonTypes.h"
|
2014-02-19 01:54:11 +01:00
|
|
|
|
2018-05-12 13:39:35 -04:00
|
|
|
namespace Common
|
|
|
|
{
|
2024-04-27 20:07:17 -07:00
|
|
|
namespace
|
2012-01-31 17:16:31 +13:00
|
|
|
{
|
2024-04-27 20:07:17 -07:00
|
|
|
// Key used to encrypt/decrypt setting.txt contents
|
|
|
|
constexpr u32 INITIAL_SEED = 0x73B5DBFA;
|
|
|
|
} // namespace
|
2012-01-31 17:16:31 +13:00
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
SettingsWriter::SettingsWriter() : m_buffer{}, m_position{0}, m_key{INITIAL_SEED}
|
2017-01-27 10:01:25 -05:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
const SettingsBuffer& SettingsWriter::GetBytes() const
|
2017-01-27 10:01:25 -05:00
|
|
|
{
|
2018-05-09 20:40:56 +02:00
|
|
|
return m_buffer;
|
2017-01-27 10:01:25 -05:00
|
|
|
}
|
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
std::string SettingsReader::GetValue(std::string_view key) const
|
2012-01-31 17:16:31 +13:00
|
|
|
{
|
2020-03-14 22:27:26 +01:00
|
|
|
constexpr char delim[] = "\n";
|
2019-07-16 04:11:53 -04:00
|
|
|
std::string toFind = std::string(delim).append(key).append("=");
|
2024-04-27 20:07:17 -07:00
|
|
|
size_t found = m_decoded.find(toFind);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-07-16 04:11:53 -04:00
|
|
|
if (found != std::string_view::npos)
|
2012-02-09 21:04:07 -08:00
|
|
|
{
|
2024-04-27 20:07:17 -07:00
|
|
|
size_t delimFound = m_decoded.find(delim, found + toFind.length());
|
2019-07-16 04:11:53 -04:00
|
|
|
if (delimFound == std::string_view::npos)
|
2024-04-27 20:07:17 -07:00
|
|
|
delimFound = m_decoded.length() - 1;
|
|
|
|
return m_decoded.substr(found + toFind.length(), delimFound - (found + toFind.length()));
|
2012-02-09 21:04:07 -08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-07-16 04:11:53 -04:00
|
|
|
toFind = std::string(key).append("=");
|
2024-04-27 20:07:17 -07:00
|
|
|
found = m_decoded.find(toFind);
|
2012-02-09 21:04:07 -08:00
|
|
|
if (found == 0)
|
|
|
|
{
|
2024-04-27 20:07:17 -07:00
|
|
|
size_t delimFound = m_decoded.find(delim, found + toFind.length());
|
2019-07-16 04:11:53 -04:00
|
|
|
if (delimFound == std::string_view::npos)
|
2024-04-27 20:07:17 -07:00
|
|
|
delimFound = m_decoded.length() - 1;
|
|
|
|
return m_decoded.substr(found + toFind.length(), delimFound - (found + toFind.length()));
|
2012-01-31 17:16:31 +13:00
|
|
|
}
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2012-01-31 17:16:31 +13:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
SettingsReader::SettingsReader(const SettingsBuffer& buffer) : m_decoded{""}
|
2012-01-31 17:16:31 +13:00
|
|
|
{
|
2024-04-27 20:07:17 -07:00
|
|
|
u32 key = INITIAL_SEED;
|
|
|
|
for (u32 position = 0; position < buffer.size(); ++position)
|
2013-08-25 02:49:58 +02:00
|
|
|
{
|
2024-04-27 20:07:17 -07:00
|
|
|
m_decoded.push_back((u8)(buffer[position] ^ key));
|
|
|
|
key = (key >> 31) | (key << 1);
|
2012-01-31 17:16:31 +13:00
|
|
|
}
|
2020-03-14 22:27:26 +01:00
|
|
|
|
2020-03-24 16:58:54 +01:00
|
|
|
// 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.)
|
2024-04-27 20:07:17 -07:00
|
|
|
std::erase(m_decoded, '\x0d');
|
2012-01-31 17:16:31 +13:00
|
|
|
}
|
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
void SettingsWriter::AddSetting(std::string_view key, std::string_view value)
|
2012-01-31 17:16:31 +13:00
|
|
|
{
|
2023-12-11 07:53:10 -05:00
|
|
|
WriteLine(fmt::format("{}={}\r\n", key, value));
|
2020-03-24 16:58:54 +01:00
|
|
|
}
|
2012-01-31 17:16:31 +13:00
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
void SettingsWriter::WriteLine(std::string_view str)
|
2020-03-24 16:58:54 +01:00
|
|
|
{
|
|
|
|
const u32 old_position = m_position;
|
|
|
|
const u32 old_key = m_key;
|
2012-01-31 17:16:31 +13:00
|
|
|
|
2020-03-24 16:58:54 +01:00
|
|
|
// Encode and write the line
|
|
|
|
for (char c : str)
|
2014-02-08 18:50:37 +13:00
|
|
|
WriteByte(c);
|
2012-01-31 17:16:31 +13:00
|
|
|
|
2020-03-24 16:58:54 +01:00
|
|
|
// 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);
|
|
|
|
}
|
2012-01-31 17:16:31 +13:00
|
|
|
}
|
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
void SettingsWriter::WriteByte(u8 b)
|
2012-01-31 17:16:31 +13:00
|
|
|
{
|
2017-01-27 10:01:25 -05:00
|
|
|
if (m_position >= m_buffer.size())
|
2012-01-31 17:16:31 +13:00
|
|
|
return;
|
|
|
|
|
|
|
|
m_buffer[m_position] = b ^ m_key;
|
|
|
|
m_position++;
|
2012-02-09 21:04:07 -08:00
|
|
|
m_key = (m_key >> 31) | (m_key << 1);
|
2012-01-31 17:16:31 +13:00
|
|
|
}
|
|
|
|
|
2024-04-27 20:07:17 -07:00
|
|
|
std::string SettingsWriter::GenerateSerialNumber()
|
2012-01-31 17:16:31 +13:00
|
|
|
{
|
2017-08-08 23:23:46 +08:00
|
|
|
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.
|
2021-10-15 22:49:13 +02:00
|
|
|
return fmt::format("{:%j%H%M%S}", fmt::localtime(t));
|
2012-02-09 21:04:07 -08:00
|
|
|
}
|
2018-05-12 13:39:35 -04:00
|
|
|
} // namespace Common
|