use of C++23 code
Signed-off-by: Peter Siegmund <mars3142@noreply.mars3142.dev>
This commit is contained in:
@@ -10,8 +10,9 @@ set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
|||||||
|
|
||||||
set(wxBUILD_SHARED OFF)
|
set(wxBUILD_SHARED OFF)
|
||||||
|
|
||||||
add_subdirectory(libs/lua-5.1.4)
|
add_subdirectory(components/lua-5.1.4)
|
||||||
add_subdirectory(libs/cartridge)
|
add_subdirectory(components/cartridge)
|
||||||
|
add_subdirectory(components/storage)
|
||||||
|
|
||||||
message(STATUS "Fetching wxWidgets...")
|
message(STATUS "Fetching wxWidgets...")
|
||||||
|
|
||||||
@@ -65,6 +66,7 @@ if (APPLE)
|
|||||||
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
COMMAND ${CMAKE_COMMAND} -E copy_if_different
|
||||||
$<TARGET_FILE:lua>
|
$<TARGET_FILE:lua>
|
||||||
$<TARGET_FILE:cartridge>
|
$<TARGET_FILE:cartridge>
|
||||||
|
$<TARGET_FILE:storage>
|
||||||
"$<TARGET_BUNDLE_DIR:${PROJECT_NAME}>/Contents/MacOS/"
|
"$<TARGET_BUNDLE_DIR:${PROJECT_NAME}>/Contents/MacOS/"
|
||||||
)
|
)
|
||||||
endif ()
|
endif ()
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ This project is a desktop application for simulating and running Wherigo geocach
|
|||||||
## Project Structure
|
## Project Structure
|
||||||
- `main/` - Application entry point and UI
|
- `main/` - Application entry point and UI
|
||||||
- `libs/cartridge/` - Cartridge parsing and media handling
|
- `libs/cartridge/` - Cartridge parsing and media handling
|
||||||
|
- `libs/storage/` - Storage management
|
||||||
- `libs/lua-5.1.4/` - Lua interpreter
|
- `libs/lua-5.1.4/` - Lua interpreter
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|||||||
@@ -21,6 +21,8 @@ add_library(cartridge SHARED ${CARTRIDGE_SRC})
|
|||||||
set_target_properties(cartridge PROPERTIES
|
set_target_properties(cartridge PROPERTIES
|
||||||
VERSION ${CARTRIDGE_VERSION}
|
VERSION ${CARTRIDGE_VERSION}
|
||||||
SOVERSION ${CARTRIDGE_VERSION_MAJOR}
|
SOVERSION ${CARTRIDGE_VERSION_MAJOR}
|
||||||
|
CXX_STANDARD 23
|
||||||
|
CXX_STANDARD_REQUIRED ON
|
||||||
)
|
)
|
||||||
|
|
||||||
target_include_directories(cartridge
|
target_include_directories(cartridge
|
||||||
@@ -28,3 +30,8 @@ target_include_directories(cartridge
|
|||||||
${CMAKE_CURRENT_SOURCE_DIR}/include
|
${CMAKE_CURRENT_SOURCE_DIR}/include
|
||||||
${CMAKE_CURRENT_BINARY_DIR}/include
|
${CMAKE_CURRENT_BINARY_DIR}/include
|
||||||
)
|
)
|
||||||
|
|
||||||
|
target_link_libraries(cartridge
|
||||||
|
PRIVATE
|
||||||
|
storage
|
||||||
|
)
|
||||||
@@ -17,9 +17,11 @@
|
|||||||
// SOFTWARE.
|
// SOFTWARE.
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#include <cstdint>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <span>
|
||||||
|
|
||||||
namespace cartridge {
|
namespace cartridge {
|
||||||
|
|
||||||
@@ -29,25 +31,25 @@ enum class Endian { Little, Big };
|
|||||||
|
|
||||||
class BinaryReader {
|
class BinaryReader {
|
||||||
public:
|
public:
|
||||||
explicit BinaryReader(const std::vector<uint8_t> &data,
|
explicit BinaryReader(std::span<const uint8_t> data,
|
||||||
Endian endian = Endian::Little);
|
Endian endian = Endian::Little);
|
||||||
|
|
||||||
void seek(int offset, SeekOrigin origin);
|
void seek(int offset, SeekOrigin origin);
|
||||||
uint8_t getByte();
|
[[nodiscard]] uint8_t getByte();
|
||||||
int16_t getShort();
|
[[nodiscard]] int16_t getShort();
|
||||||
uint16_t getUShort();
|
[[nodiscard]] uint16_t getUShort();
|
||||||
int32_t getLong();
|
[[nodiscard]] int32_t getLong();
|
||||||
uint32_t getULong();
|
[[nodiscard]] uint32_t getULong();
|
||||||
double getDouble();
|
[[nodiscard]] double getDouble();
|
||||||
std::string getASCIIZ();
|
[[nodiscard]] std::string getASCIIZ();
|
||||||
[[nodiscard]] const std::vector<uint8_t>& data() const { return _data; }
|
[[nodiscard]] std::span<const uint8_t> data() const { return _data; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
template <typename T> T readInt(size_t size);
|
template <typename T> [[nodiscard]] T readInt(size_t size);
|
||||||
static bool isSystemLittleEndian();
|
[[nodiscard]] static bool isSystemLittleEndian();
|
||||||
static void swapBytes(uint8_t *data, size_t size);
|
static void swapBytes(uint8_t *data, size_t size);
|
||||||
|
|
||||||
const std::vector<uint8_t> &_data;
|
std::span<const uint8_t> _data;
|
||||||
Endian _endian;
|
Endian _endian;
|
||||||
size_t _index;
|
size_t _index;
|
||||||
};
|
};
|
||||||
@@ -21,4 +21,8 @@ namespace cartridge {
|
|||||||
|
|
||||||
std::unique_ptr<Cartridge> parseData(const std::vector<uint8_t>& bytes);
|
std::unique_ptr<Cartridge> parseData(const std::vector<uint8_t>& bytes);
|
||||||
|
|
||||||
|
std::unique_ptr<Cartridge> parseFile(const std::string& filePath);
|
||||||
|
|
||||||
|
std::unique_ptr<Cartridge> parseCartridge();
|
||||||
|
|
||||||
} // namespace cartridge
|
} // namespace cartridge
|
||||||
@@ -19,10 +19,12 @@
|
|||||||
#include "cartridge/binary_reader.h"
|
#include "cartridge/binary_reader.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
#include <bit>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
namespace cartridge {
|
namespace cartridge {
|
||||||
|
|
||||||
BinaryReader::BinaryReader(const std::vector<uint8_t> &data,
|
BinaryReader::BinaryReader(std::span<const uint8_t> data,
|
||||||
const Endian endian)
|
const Endian endian)
|
||||||
: _data(data), _endian(endian), _index(0) {}
|
: _data(data), _endian(endian), _index(0) {}
|
||||||
|
|
||||||
@@ -78,14 +80,11 @@ std::string BinaryReader::getASCIIZ() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool BinaryReader::isSystemLittleEndian() {
|
bool BinaryReader::isSystemLittleEndian() {
|
||||||
uint16_t num = 1;
|
return std::endian::native == std::endian::little;
|
||||||
return *reinterpret_cast<uint8_t *>(&num) == 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void BinaryReader::swapBytes(uint8_t *data, size_t size) {
|
void BinaryReader::swapBytes(uint8_t *data, size_t size) {
|
||||||
for (size_t i = 0; i < size / 2; ++i) {
|
std::ranges::reverse(data, data + size);
|
||||||
std::swap(data[i], data[size - 1 - i]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace cartridge
|
} // namespace cartridge
|
||||||
@@ -46,39 +46,40 @@ Cartridge::Cartridge(std::string cartridgeGuid, double altitude,
|
|||||||
|
|
||||||
std::unique_ptr<Cartridge> Cartridge::create(BinaryReader &reader) {
|
std::unique_ptr<Cartridge> Cartridge::create(BinaryReader &reader) {
|
||||||
try {
|
try {
|
||||||
uint16_t count = reader.getUShort();
|
const uint16_t count = reader.getUShort();
|
||||||
std::map<int, int> references;
|
std::map<int, int> references;
|
||||||
for (uint16_t index = 0; index < count; ++index) {
|
for (uint16_t index = 0; index < count; ++index) {
|
||||||
int objectId = reader.getShort();
|
const int objectId = reader.getShort();
|
||||||
int address = reader.getLong();
|
const int address = reader.getLong();
|
||||||
references.emplace(objectId, address);
|
references.emplace(objectId, address);
|
||||||
}
|
}
|
||||||
|
|
||||||
reader.getLong(); // header length
|
reader.getLong(); // header length
|
||||||
double latitude = reader.getDouble();
|
const double latitude = reader.getDouble();
|
||||||
double longitude = reader.getDouble();
|
const double longitude = reader.getDouble();
|
||||||
double altitude = reader.getDouble();
|
const double altitude = reader.getDouble();
|
||||||
reader.getLong(); // unknown 0
|
reader.getLong(); // unknown 0
|
||||||
reader.getLong(); // unknown 1
|
reader.getLong(); // unknown 1
|
||||||
int splashScreenId = reader.getShort();
|
const int splashScreenId = reader.getShort();
|
||||||
int smallIconId = reader.getShort();
|
const int smallIconId = reader.getShort();
|
||||||
std::string typeOfCartridge = reader.getASCIIZ();
|
const std::string typeOfCartridge = reader.getASCIIZ();
|
||||||
std::string playerName = reader.getASCIIZ();
|
const std::string playerName = reader.getASCIIZ();
|
||||||
reader.getLong(); // unknown 2
|
reader.getLong(); // unknown 2
|
||||||
reader.getLong(); // unknown 3
|
reader.getLong(); // unknown 3
|
||||||
std::string cartridgeName = reader.getASCIIZ();
|
const std::string cartridgeName = reader.getASCIIZ();
|
||||||
std::string cartridgeGuid = reader.getASCIIZ();
|
const std::string cartridgeGuid = reader.getASCIIZ();
|
||||||
std::string cartridgeDesc = reader.getASCIIZ();
|
const std::string cartridgeDesc = reader.getASCIIZ();
|
||||||
std::string startLocationDesc = reader.getASCIIZ();
|
const std::string startLocationDesc = reader.getASCIIZ();
|
||||||
std::string version = reader.getASCIIZ();
|
const std::string version = reader.getASCIIZ();
|
||||||
std::string author = reader.getASCIIZ();
|
const std::string author = reader.getASCIIZ();
|
||||||
std::string company = reader.getASCIIZ();
|
const std::string company = reader.getASCIIZ();
|
||||||
std::string recommendedDevice = reader.getASCIIZ();
|
const std::string recommendedDevice = reader.getASCIIZ();
|
||||||
reader.getLong(); // unknown 4
|
reader.getLong(); // unknown 4
|
||||||
std::string completionCode = reader.getASCIIZ();
|
const std::string completionCode = reader.getASCIIZ();
|
||||||
|
|
||||||
// Rohdaten aus dem Reader extrahieren
|
// Rohdaten aus dem Reader extrahieren
|
||||||
const std::vector<uint8_t> &bytes = reader.data();
|
const auto bytes_span = reader.data();
|
||||||
|
const std::vector<uint8_t> bytes(bytes_span.begin(), bytes_span.end());
|
||||||
|
|
||||||
return std::make_unique<Cartridge>(
|
return std::make_unique<Cartridge>(
|
||||||
cartridgeGuid, altitude, author, cartridgeDesc, cartridgeName, company,
|
cartridgeGuid, altitude, author, cartridgeDesc, cartridgeName, company,
|
||||||
@@ -114,9 +115,8 @@ std::unique_ptr<Media> Cartridge::getMedia(const int objectId) {
|
|||||||
if (m_lastObject == objectId && m_lastMedia) {
|
if (m_lastObject == objectId && m_lastMedia) {
|
||||||
return std::make_unique<Media>(*m_lastMedia);
|
return std::make_unique<Media>(*m_lastMedia);
|
||||||
}
|
}
|
||||||
auto it = m_references.find(objectId);
|
if (auto it = m_references.find(objectId); it != m_references.end()) {
|
||||||
if (it != m_references.end()) {
|
const int address = it->second;
|
||||||
int address = it->second;
|
|
||||||
BinaryReader reader(m_bytes, Endian::Little);
|
BinaryReader reader(m_bytes, Endian::Little);
|
||||||
auto media = Media::create(reader, objectId, address);
|
auto media = Media::create(reader, objectId, address);
|
||||||
if (media && media->getData().size() < 128000) {
|
if (media && media->getData().size() < 128000) {
|
||||||
@@ -19,8 +19,7 @@
|
|||||||
#include "cartridge/lat_lng.h"
|
#include "cartridge/lat_lng.h"
|
||||||
|
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
#include <iomanip>
|
#include <format>
|
||||||
#include <sstream>
|
|
||||||
|
|
||||||
namespace cartridge {
|
namespace cartridge {
|
||||||
|
|
||||||
@@ -48,9 +47,8 @@ std::string LatLng::_format(double value, const std::string &suffix) {
|
|||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
const double minutes = (value - degrees) * 60.0;
|
const double minutes = (value - degrees) * 60.0;
|
||||||
std::ostringstream oss;
|
return std::format("{} {}\u00B0 {:.3f}",
|
||||||
oss << (isNegative ? suffix.substr(1, 1) : suffix.substr(0, 1)) << " "
|
isNegative ? suffix.substr(1, 1) : suffix.substr(0, 1),
|
||||||
<< degrees << "\u00B0 " << std::fixed << std::setprecision(3) << minutes;
|
degrees, minutes);
|
||||||
return oss.str();
|
|
||||||
}
|
}
|
||||||
} // namespace cartridge
|
} // namespace cartridge
|
||||||
@@ -19,6 +19,7 @@
|
|||||||
#include "cartridge/media.h"
|
#include "cartridge/media.h"
|
||||||
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
#include <format>
|
||||||
|
|
||||||
namespace cartridge {
|
namespace cartridge {
|
||||||
|
|
||||||
@@ -28,67 +29,48 @@ Media::Media(std::string objectType, const std::vector<uint8_t> &data)
|
|||||||
std::unique_ptr<Media> Media::create(BinaryReader &reader, const int objectId,
|
std::unique_ptr<Media> Media::create(BinaryReader &reader, const int objectId,
|
||||||
const int address) {
|
const int address) {
|
||||||
try {
|
try {
|
||||||
std::vector<uint8_t> data;
|
|
||||||
int objectType = 0;
|
int objectType = 0;
|
||||||
reader.seek(address, SeekOrigin::Begin);
|
reader.seek(address, SeekOrigin::Begin);
|
||||||
if (objectId == 0) {
|
std::vector<uint8_t> data = (objectId == 0)
|
||||||
data = readMediaData(reader);
|
? readMediaData(reader)
|
||||||
} else {
|
: (reader.getByte() != 0)
|
||||||
if (reader.getByte() != 0) {
|
? (objectType = reader.getLong(), readMediaData(reader))
|
||||||
objectType = reader.getLong();
|
: std::vector<uint8_t>{};
|
||||||
data = readMediaData(reader);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!data.empty()) {
|
if (!data.empty()) {
|
||||||
return std::make_unique<Media>(getObjectTypeString(objectType), data);
|
return std::make_unique<Media>(getObjectTypeString(objectType), data);
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
} catch ([[maybe_unused]] const std::exception &ex) {
|
} catch ([[maybe_unused]] const std::exception &ex) {
|
||||||
// Logging analog zu Dart: print('Exception: $ex');
|
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<uint8_t> Media::readMediaData(BinaryReader &reader) {
|
std::vector<uint8_t> Media::readMediaData(BinaryReader &reader) {
|
||||||
const int32_t length = reader.getLong();
|
const int32_t length = reader.getLong();
|
||||||
std::vector<uint8_t> data;
|
std::vector<uint8_t> data(length);
|
||||||
data.reserve(length);
|
|
||||||
for (int32_t i = 0; i < length; ++i) {
|
for (int32_t i = 0; i < length; ++i) {
|
||||||
data.push_back(reader.getByte());
|
data[i] = reader.getByte();
|
||||||
}
|
}
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string Media::getObjectTypeString(int objectType) {
|
std::string Media::getObjectTypeString(const int objectType) {
|
||||||
switch (objectType) {
|
switch (objectType) {
|
||||||
case -1:
|
case -1: return "deleted";
|
||||||
return "deleted";
|
case 0: return "luac";
|
||||||
case 0:
|
case 1: return "bmp";
|
||||||
return "luac";
|
case 2: return "png";
|
||||||
case 1:
|
case 3: return "jpg";
|
||||||
return "bmp";
|
case 4: return "gif";
|
||||||
case 2:
|
case 17: return "wav";
|
||||||
return "png";
|
case 18: return "mp3";
|
||||||
case 3:
|
case 19: return "fdl";
|
||||||
return "jpg";
|
case 20: return "snd";
|
||||||
case 4:
|
case 21: return "ogg";
|
||||||
return "gif";
|
case 33: return "swf";
|
||||||
case 17:
|
case 49: return "txt";
|
||||||
return "wav";
|
default: return std::format("invalid ({})", objectType);
|
||||||
case 18:
|
|
||||||
return "mp3";
|
|
||||||
case 19:
|
|
||||||
return "fdl";
|
|
||||||
case 20:
|
|
||||||
return "snd";
|
|
||||||
case 21:
|
|
||||||
return "ogg";
|
|
||||||
case 33:
|
|
||||||
return "swf";
|
|
||||||
case 49:
|
|
||||||
return "txt";
|
|
||||||
default:
|
|
||||||
return "invalid (" + std::to_string(objectType) + ")";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
101
components/cartridge/src/parser.cpp
Normal file
101
components/cartridge/src/parser.cpp
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
// MIT License
|
||||||
|
// Copyright (c) 2026 by Peter Siegmund (mars3142)
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
|
// in the Software without restriction, including without limitation the rights
|
||||||
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
// copies of the Software, and to permit persons to whom the Software is
|
||||||
|
// furnished to do so, subject to the following conditions:
|
||||||
|
// The above copyright notice and this permission notice shall be included in
|
||||||
|
// all copies or substantial portions of the Software.
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
// SOFTWARE.
|
||||||
|
|
||||||
|
#include "cartridge/parser.h"
|
||||||
|
#include "storage/storage.h"
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <filesystem>
|
||||||
|
#include <format>
|
||||||
|
#include <iostream>
|
||||||
|
#include <print>
|
||||||
|
#include <ranges>
|
||||||
|
|
||||||
|
namespace cartridge {
|
||||||
|
|
||||||
|
std::unique_ptr<Cartridge> parseData(const std::vector<uint8_t> &bytes) {
|
||||||
|
try {
|
||||||
|
constexpr std::array<uint8_t, 7> header = {0x02, 0x0a, 0x43, 0x41,
|
||||||
|
0x52, 0x54, 0x00};
|
||||||
|
BinaryReader reader(bytes, Endian::Little);
|
||||||
|
|
||||||
|
if (!std::ranges::all_of(header, [&reader](const uint8_t expected) {
|
||||||
|
return reader.getByte() == expected;
|
||||||
|
})) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Cartridge::create(reader);
|
||||||
|
} catch (...) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<Cartridge> parseFile(const std::string &filePath) {
|
||||||
|
storage::Storage storage;
|
||||||
|
auto result = storage.readFile(filePath);
|
||||||
|
if (!result) {
|
||||||
|
std::println(std::cerr, "Fehler beim Lesen der Datei: {}", filePath);
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
return parseData(*result);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<Cartridge> parseCartridge() {
|
||||||
|
auto cartridge = parseFile("/Volumes/Coding/git.mars3142.dev/mars3142/"
|
||||||
|
"wx_wherigo/cartridges/the_ombos_idol_-_c.gwc");
|
||||||
|
if (!cartridge) {
|
||||||
|
std::println(std::cerr, "Cartridge konnte nicht geladen werden.");
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const int count = cartridge->mediaCount();
|
||||||
|
const std::filesystem::path outDir =
|
||||||
|
"/Volumes/Coding/git.mars3142.dev/mars3142/wx_wherigo/cartridges";
|
||||||
|
storage::Storage storage;
|
||||||
|
|
||||||
|
for (const int i : std::views::iota(0, count)) {
|
||||||
|
auto media = cartridge->getMedia(i);
|
||||||
|
if (!media) {
|
||||||
|
std::println(std::cerr, "Media-Objekt an Index {} ist nullptr.", i);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto &data = media->getData();
|
||||||
|
if (data.empty()) {
|
||||||
|
std::println(std::cerr, "Media-Daten an Index {} sind leer.", i);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string ext = media->getObjectType();
|
||||||
|
if (ext.empty()) {
|
||||||
|
std::println(std::cerr, "Media-Extension an Index {} ist leer.", i);
|
||||||
|
ext = "bin";
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::filesystem::path outFile = outDir / std::format("file{}.{}", i, ext);
|
||||||
|
|
||||||
|
if (auto writeResult = storage.writeFile(outFile.string(), data); !writeResult) {
|
||||||
|
std::println(std::cerr, "Fehler beim Schreiben der Datei: {}", outFile.string());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return cartridge;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace cartridge
|
||||||
26
components/storage/CMakeLists.txt
Normal file
26
components/storage/CMakeLists.txt
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
set(STORAGE_VERSION_MAJOR 1)
|
||||||
|
set(STORAGE_VERSION_MINOR 0)
|
||||||
|
set(STORAGE_VERSION_PATCH 0)
|
||||||
|
set(STORAGE_VERSION "${STORAGE_VERSION_MAJOR}.${STORAGE_VERSION_MINOR}.${STORAGE_VERSION_PATCH}")
|
||||||
|
|
||||||
|
configure_file(
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/version.h.in
|
||||||
|
${CMAKE_CURRENT_BINARY_DIR}/include/storage/version.h
|
||||||
|
@ONLY
|
||||||
|
)
|
||||||
|
|
||||||
|
set(STORAGE_SRC
|
||||||
|
src/storage.cpp
|
||||||
|
)
|
||||||
|
|
||||||
|
add_library(storage SHARED ${STORAGE_SRC})
|
||||||
|
set_target_properties(storage PROPERTIES
|
||||||
|
VERSION ${STORAGE_VERSION}
|
||||||
|
SOVERSION ${STORAGE_VERSION_MAJOR}
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(storage
|
||||||
|
PUBLIC
|
||||||
|
${CMAKE_CURRENT_SOURCE_DIR}/include
|
||||||
|
${CMAKE_CURRENT_BINARY_DIR}/include
|
||||||
|
)
|
||||||
50
components/storage/include/storage/storage.h
Normal file
50
components/storage/include/storage/storage.h
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
// MIT License
|
||||||
|
// Copyright (c) 2026 by Peter Siegmund (mars3142)
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
|
// in the Software without restriction, including without limitation the rights
|
||||||
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
// copies of the Software, and to permit persons to whom the Software is
|
||||||
|
// furnished to do so, subject to the following conditions:
|
||||||
|
// The above copyright notice and this permission notice shall be included in
|
||||||
|
// all copies or substantial portions of the Software.
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
// SOFTWARE.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "storage/storage_error.h"
|
||||||
|
|
||||||
|
#include <expected>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace storage {
|
||||||
|
|
||||||
|
class Storage {
|
||||||
|
public:
|
||||||
|
/**
|
||||||
|
* Liest eine Datei von einem Pfad.
|
||||||
|
* @param path Der Dateipfad.
|
||||||
|
* @return Ein expected mit den Dateidaten oder einem StorageError.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] std::expected<std::vector<uint8_t>, StorageError> readFile(
|
||||||
|
const std::string &path) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Schreibt Daten in eine Datei.
|
||||||
|
* @param path Der Dateipfad.
|
||||||
|
* @param data Die zu schreibenden Daten.
|
||||||
|
* @return Ein expected<void> oder einem StorageError.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] std::expected<void, StorageError> writeFile(
|
||||||
|
const std::string &path,
|
||||||
|
const std::vector<uint8_t> &data) const;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace storage
|
||||||
@@ -16,29 +16,14 @@
|
|||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
// SOFTWARE.
|
// SOFTWARE.
|
||||||
|
|
||||||
#include "cartridge/parser.h"
|
#pragma once
|
||||||
|
|
||||||
#include <array>
|
namespace storage {
|
||||||
#include <filesystem>
|
|
||||||
#include <fstream>
|
|
||||||
#include <iostream>
|
|
||||||
|
|
||||||
namespace cartridge {
|
enum class StorageError {
|
||||||
|
FileNotFound,
|
||||||
|
ReadError,
|
||||||
|
WriteError,
|
||||||
|
};
|
||||||
|
|
||||||
std::unique_ptr<Cartridge> parseData(const std::vector<uint8_t> &bytes) {
|
} // namespace storage
|
||||||
try {
|
|
||||||
constexpr std::array<uint8_t, 7> header = {0x02, 0x0a, 0x43, 0x41,
|
|
||||||
0x52, 0x54, 0x00};
|
|
||||||
BinaryReader reader(bytes, Endian::Little);
|
|
||||||
for (const unsigned char i : header) {
|
|
||||||
if (reader.getByte() != i) {
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Cartridge::create(reader);
|
|
||||||
} catch (...) {
|
|
||||||
return nullptr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace cartridge
|
|
||||||
59
components/storage/src/storage.cpp
Normal file
59
components/storage/src/storage.cpp
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
// MIT License
|
||||||
|
// Copyright (c) 2026 by Peter Siegmund (mars3142)
|
||||||
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
// of this software and associated documentation files (the "Software"), to deal
|
||||||
|
// in the Software without restriction, including without limitation the rights
|
||||||
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
// copies of the Software, and to permit persons to whom the Software is
|
||||||
|
// furnished to do so, subject to the following conditions:
|
||||||
|
// The above copyright notice and this permission notice shall be included in
|
||||||
|
// all copies or substantial portions of the Software.
|
||||||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
// SOFTWARE.
|
||||||
|
|
||||||
|
#include "storage/storage.h"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
|
||||||
|
namespace storage {
|
||||||
|
|
||||||
|
std::expected<std::vector<uint8_t>, StorageError> Storage::readFile(
|
||||||
|
const std::string &path) const {
|
||||||
|
std::ifstream file(path, std::ios::binary | std::ios::ate);
|
||||||
|
if (!file) {
|
||||||
|
return std::unexpected(StorageError::FileNotFound);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::streamsize size = file.tellg();
|
||||||
|
if (size < 0) {
|
||||||
|
return std::unexpected(StorageError::ReadError);
|
||||||
|
}
|
||||||
|
file.seekg(0, std::ios::beg);
|
||||||
|
|
||||||
|
std::vector<uint8_t> data(static_cast<size_t>(size));
|
||||||
|
if (!file.read(reinterpret_cast<char *>(data.data()), size)) {
|
||||||
|
return std::unexpected(StorageError::ReadError);
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::expected<void, StorageError> Storage::writeFile(
|
||||||
|
const std::string &path,
|
||||||
|
const std::vector<uint8_t> &data) const {
|
||||||
|
std::ofstream file(path, std::ios::binary);
|
||||||
|
if (!file) {
|
||||||
|
return std::unexpected(StorageError::WriteError);
|
||||||
|
}
|
||||||
|
file.write(reinterpret_cast<const char *>(data.data()), data.size());
|
||||||
|
if (!file.good()) {
|
||||||
|
return std::unexpected(StorageError::WriteError);
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace storage
|
||||||
7
components/storage/version.h.in
Normal file
7
components/storage/version.h.in
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#define STORAGE_VERSION_MAJOR @STORAGE_VERSION_MAJOR@
|
||||||
|
#define STORAGE_VERSION_MINOR @STORAGE_VERSION_MINOR@
|
||||||
|
#define STORAGE_VERSION_PATCH @STORAGE_VERSION_PATCH@
|
||||||
|
#define STORAGE_VERSION "@STORAGE_VERSION@"
|
||||||
|
|
||||||
@@ -1,8 +1,11 @@
|
|||||||
#include "cApp.h"
|
#include "cApp.h"
|
||||||
#include "ui/cFrame.h"
|
#include "ui/cFrame.h"
|
||||||
|
|
||||||
|
#include <cartridge/parser.h>
|
||||||
|
|
||||||
bool cApp::OnInit()
|
bool cApp::OnInit()
|
||||||
{
|
{
|
||||||
|
auto cart = cartridge::parseCartridge();
|
||||||
auto *frame = new cFrame();
|
auto *frame = new cFrame();
|
||||||
return frame->Show(true);
|
return frame->Show(true);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user