2019-12-09 03:57:37 -08:00
|
|
|
/*
|
2021-10-04 12:59:10 -07:00
|
|
|
* Copyright (c) Atmosphère-NX
|
2019-12-09 03:57:37 -08:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#include <stratosphere.hpp>
|
|
|
|
|
|
|
|
namespace ams::kvdb {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
/* Convenience definitions. */
|
|
|
|
constexpr u8 ArchiveHeaderMagic[4] = {'I', 'M', 'K', 'V'};
|
|
|
|
constexpr u8 ArchiveEntryMagic[4] = {'I', 'M', 'E', 'N'};
|
|
|
|
|
|
|
|
/* Archive types. */
|
|
|
|
struct ArchiveHeader {
|
|
|
|
u8 magic[sizeof(ArchiveHeaderMagic)];
|
|
|
|
u32 pad;
|
|
|
|
u32 entry_count;
|
|
|
|
|
|
|
|
Result Validate() const {
|
2021-10-09 10:36:21 -07:00
|
|
|
R_UNLESS(std::memcmp(this->magic, ArchiveHeaderMagic, sizeof(ArchiveHeaderMagic)) == 0, kvdb::ResultInvalidKeyValue());
|
2019-12-09 03:57:37 -08:00
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
static ArchiveHeader Make(size_t entry_count) {
|
|
|
|
ArchiveHeader header = {};
|
|
|
|
std::memcpy(header.magic, ArchiveHeaderMagic, sizeof(ArchiveHeaderMagic));
|
|
|
|
header.entry_count = static_cast<u32>(entry_count);
|
|
|
|
return header;
|
|
|
|
}
|
|
|
|
};
|
2020-05-11 15:02:10 -07:00
|
|
|
static_assert(sizeof(ArchiveHeader) == 0xC && util::is_pod<ArchiveHeader>::value, "ArchiveHeader definition!");
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
struct ArchiveEntryHeader {
|
|
|
|
u8 magic[sizeof(ArchiveEntryMagic)];
|
|
|
|
u32 key_size;
|
|
|
|
u32 value_size;
|
|
|
|
|
|
|
|
Result Validate() const {
|
2021-10-09 10:36:21 -07:00
|
|
|
R_UNLESS(std::memcmp(this->magic, ArchiveEntryMagic, sizeof(ArchiveEntryMagic)) == 0, kvdb::ResultInvalidKeyValue());
|
2019-12-09 03:57:37 -08:00
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
static ArchiveEntryHeader Make(size_t ksz, size_t vsz) {
|
|
|
|
ArchiveEntryHeader header = {};
|
|
|
|
std::memcpy(header.magic, ArchiveEntryMagic, sizeof(ArchiveEntryMagic));
|
|
|
|
header.key_size = ksz;
|
|
|
|
header.value_size = vsz;
|
|
|
|
return header;
|
|
|
|
}
|
|
|
|
};
|
2020-05-11 15:02:10 -07:00
|
|
|
static_assert(sizeof(ArchiveEntryHeader) == 0xC && util::is_pod<ArchiveEntryHeader>::value, "ArchiveEntryHeader definition!");
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reader functionality. */
|
|
|
|
Result ArchiveReader::Peek(void *dst, size_t size) {
|
|
|
|
/* Bounds check. */
|
2021-10-10 00:14:06 -07:00
|
|
|
R_UNLESS(m_offset + size <= m_buffer.GetSize(), kvdb::ResultInvalidKeyValue());
|
|
|
|
R_UNLESS(m_offset < m_offset + size, kvdb::ResultInvalidKeyValue());
|
2019-12-09 03:57:37 -08:00
|
|
|
|
2021-10-10 00:14:06 -07:00
|
|
|
std::memcpy(dst, m_buffer.Get() + m_offset, size);
|
2019-12-09 03:57:37 -08:00
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ArchiveReader::Read(void *dst, size_t size) {
|
|
|
|
R_TRY(this->Peek(dst, size));
|
2021-10-10 00:14:06 -07:00
|
|
|
m_offset += size;
|
2019-12-09 03:57:37 -08:00
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ArchiveReader::ReadEntryCount(size_t *out) {
|
|
|
|
/* This should only be called at the start of reading stream. */
|
2021-10-10 00:14:06 -07:00
|
|
|
AMS_ABORT_UNLESS(m_offset == 0);
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
/* Read and validate header. */
|
|
|
|
ArchiveHeader header;
|
2021-10-09 14:49:53 -07:00
|
|
|
R_TRY(this->Read(std::addressof(header), sizeof(header)));
|
2019-12-09 03:57:37 -08:00
|
|
|
R_TRY(header.Validate());
|
|
|
|
|
|
|
|
*out = header.entry_count;
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ArchiveReader::GetEntrySize(size_t *out_key_size, size_t *out_value_size) {
|
|
|
|
/* This should only be called after ReadEntryCount. */
|
2021-10-10 00:14:06 -07:00
|
|
|
AMS_ABORT_UNLESS(m_offset != 0);
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
/* Peek the next entry header. */
|
|
|
|
ArchiveEntryHeader header;
|
2021-10-09 14:49:53 -07:00
|
|
|
R_TRY(this->Peek(std::addressof(header), sizeof(header)));
|
2019-12-09 03:57:37 -08:00
|
|
|
R_TRY(header.Validate());
|
|
|
|
|
|
|
|
*out_key_size = header.key_size;
|
|
|
|
*out_value_size = header.value_size;
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ArchiveReader::ReadEntry(void *out_key, size_t key_size, void *out_value, size_t value_size) {
|
|
|
|
/* This should only be called after ReadEntryCount. */
|
2021-10-10 00:14:06 -07:00
|
|
|
AMS_ABORT_UNLESS(m_offset != 0);
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
/* Read the next entry header. */
|
|
|
|
ArchiveEntryHeader header;
|
2021-10-09 14:49:53 -07:00
|
|
|
R_TRY(this->Read(std::addressof(header), sizeof(header)));
|
2019-12-09 03:57:37 -08:00
|
|
|
R_TRY(header.Validate());
|
|
|
|
|
|
|
|
/* Key size and Value size must be correct. */
|
2020-02-22 23:05:14 -08:00
|
|
|
AMS_ABORT_UNLESS(key_size == header.key_size);
|
|
|
|
AMS_ABORT_UNLESS(value_size == header.value_size);
|
2019-12-09 03:57:37 -08:00
|
|
|
|
2020-02-22 23:05:14 -08:00
|
|
|
R_ABORT_UNLESS(this->Read(out_key, key_size));
|
|
|
|
R_ABORT_UNLESS(this->Read(out_value, value_size));
|
2019-12-09 03:57:37 -08:00
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Writer functionality. */
|
|
|
|
Result ArchiveWriter::Write(const void *src, size_t size) {
|
|
|
|
/* Bounds check. */
|
2021-10-10 00:14:06 -07:00
|
|
|
R_UNLESS(m_offset + size <= m_buffer.GetSize(), kvdb::ResultInvalidKeyValue());
|
|
|
|
R_UNLESS(m_offset < m_offset + size, kvdb::ResultInvalidKeyValue());
|
2019-12-09 03:57:37 -08:00
|
|
|
|
2021-10-10 00:14:06 -07:00
|
|
|
std::memcpy(m_buffer.Get() + m_offset, src, size);
|
|
|
|
m_offset += size;
|
2019-12-09 03:57:37 -08:00
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArchiveWriter::WriteHeader(size_t entry_count) {
|
|
|
|
/* This should only be called at start of write. */
|
2021-10-10 00:14:06 -07:00
|
|
|
AMS_ABORT_UNLESS(m_offset == 0);
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
ArchiveHeader header = ArchiveHeader::Make(entry_count);
|
2021-10-09 14:49:53 -07:00
|
|
|
R_ABORT_UNLESS(this->Write(std::addressof(header), sizeof(header)));
|
2019-12-09 03:57:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void ArchiveWriter::WriteEntry(const void *key, size_t key_size, const void *value, size_t value_size) {
|
|
|
|
/* This should only be called after writing header. */
|
2021-10-10 00:14:06 -07:00
|
|
|
AMS_ABORT_UNLESS(m_offset != 0);
|
2019-12-09 03:57:37 -08:00
|
|
|
|
|
|
|
ArchiveEntryHeader header = ArchiveEntryHeader::Make(key_size, value_size);
|
2021-10-09 14:49:53 -07:00
|
|
|
R_ABORT_UNLESS(this->Write(std::addressof(header), sizeof(header)));
|
2020-02-22 23:05:14 -08:00
|
|
|
R_ABORT_UNLESS(this->Write(key, key_size));
|
|
|
|
R_ABORT_UNLESS(this->Write(value, value_size));
|
2019-12-09 03:57:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Size helper functionality. */
|
2021-10-10 00:14:06 -07:00
|
|
|
ArchiveSizeHelper::ArchiveSizeHelper() : m_size(sizeof(ArchiveHeader)) {
|
2019-12-09 03:57:37 -08:00
|
|
|
/* ... */
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArchiveSizeHelper::AddEntry(size_t key_size, size_t value_size) {
|
2021-10-10 00:14:06 -07:00
|
|
|
m_size += sizeof(ArchiveEntryHeader) + key_size + value_size;
|
2019-12-09 03:57:37 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|