mirror of
https://github.com/TwilitRealm/dusklight
synced 2026-05-23 06:34:15 -04:00
state packs and partial states
This commit is contained in:
@@ -10,7 +10,11 @@
|
||||
#include "d/d_com_inf_game.h"
|
||||
#include "dusk/main.h"
|
||||
#include "dusk/io.hpp"
|
||||
#include "dusk/logging.h"
|
||||
#include "../file_select.hpp"
|
||||
#include "aurora/lib/window.hpp"
|
||||
|
||||
#include <unordered_set>
|
||||
#include <zstd.h>
|
||||
|
||||
namespace dusk {
|
||||
@@ -27,9 +31,21 @@ struct StateSharePacket {
|
||||
};
|
||||
#pragma pack(pop)
|
||||
|
||||
static constexpr size_t PACKET_TOTAL = sizeof(StateSharePacket) + sizeof(dSv_info_c);
|
||||
static constexpr size_t PACKET_TOTAL = sizeof(StateSharePacket) + sizeof(dSv_info_c);
|
||||
static constexpr size_t PACKET_SAVE_ONLY = sizeof(StateSharePacket) + sizeof(dSv_save_c);
|
||||
static constexpr auto STATES_FILENAME = "states.json";
|
||||
|
||||
static bool ValidateEncodedState(const std::string&);
|
||||
|
||||
void ImGuiStateShare::onMergeFileSelected(void* userdata, const char* path, const char* /*error*/) {
|
||||
auto* self = static_cast<ImGuiStateShare*>(userdata);
|
||||
if (path != nullptr) {
|
||||
self->m_pendingMergePath = path;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
static std::string GetStatesFilePath() {
|
||||
return (dusk::ConfigPath / STATES_FILENAME).string();
|
||||
}
|
||||
@@ -64,7 +80,7 @@ void ImGuiStateShare::loadStatesFile() {
|
||||
void ImGuiStateShare::saveStatesFile() {
|
||||
json j = json::array();
|
||||
for (const auto& s : m_states) {
|
||||
j.push_back({{"name", s.name}, {"data", s.encoded}});
|
||||
j.push_back(json{{"name", s.name}, {"data", s.encoded}});
|
||||
}
|
||||
try {
|
||||
io::FileStream::WriteAllText(GetStatesFilePath().c_str(), j.dump(2));
|
||||
@@ -99,7 +115,14 @@ bool ImGuiStateShare::applyEncodedState(const std::string& encoded, const std::s
|
||||
}
|
||||
|
||||
unsigned long long dSize = ZSTD_getFrameContentSize(decoded.data(), decoded.size());
|
||||
if (dSize == ZSTD_CONTENTSIZE_ERROR || dSize == ZSTD_CONTENTSIZE_UNKNOWN || dSize < PACKET_TOTAL) {
|
||||
if (dSize == ZSTD_CONTENTSIZE_ERROR || dSize == ZSTD_CONTENTSIZE_UNKNOWN) {
|
||||
m_statusMsg = "Not a valid state string.";
|
||||
return false;
|
||||
}
|
||||
|
||||
const bool isFull = (dSize == PACKET_TOTAL);
|
||||
const bool isPartial = (dSize == PACKET_SAVE_ONLY);
|
||||
if (!isFull && !isPartial) {
|
||||
m_statusMsg = "Not a valid state string.";
|
||||
return false;
|
||||
}
|
||||
@@ -115,15 +138,27 @@ bool ImGuiStateShare::applyEncodedState(const std::string& encoded, const std::s
|
||||
memcpy(&pkt, raw.data(), sizeof(pkt));
|
||||
pkt.stageName[7] = '\0';
|
||||
|
||||
memcpy(&g_dComIfG_gameInfo.info, raw.data() + sizeof(pkt), sizeof(dSv_info_c));
|
||||
if (isFull) {
|
||||
memcpy(&g_dComIfG_gameInfo.info, raw.data() + sizeof(pkt), sizeof(dSv_info_c));
|
||||
m_pendingInfo = g_dComIfG_gameInfo.info;
|
||||
m_pendingSavedata.reset();
|
||||
} else {
|
||||
memcpy(&g_dComIfG_gameInfo.info.mSavedata, raw.data() + sizeof(pkt), sizeof(dSv_save_c));
|
||||
m_pendingSavedata = g_dComIfG_gameInfo.info.mSavedata;
|
||||
m_pendingInfo.reset();
|
||||
}
|
||||
|
||||
s16 spawnPoint = pkt.startPoint == -4 ? -1 : pkt.startPoint;
|
||||
if (spawnPoint == -1) {
|
||||
dComIfGs_setRestartRoomParam(pkt.roomNo & 0x3F);
|
||||
}
|
||||
|
||||
DuskLog.info("StateShare: applying {} state - stage={} room={} layer={} point={} lastSceneMode={}",
|
||||
isFull ? "full" : "partial",
|
||||
pkt.stageName, (int)pkt.roomNo, (int)pkt.layer, (int)spawnPoint,
|
||||
dComIfGs_getLastSceneMode());
|
||||
|
||||
dComIfGp_setNextStage(pkt.stageName, spawnPoint, pkt.roomNo, pkt.layer);
|
||||
m_pendingInfo = g_dComIfG_gameInfo.info;
|
||||
|
||||
if (name.empty()) {
|
||||
m_statusMsg = fmt::format("{} room {} layer {}.", pkt.stageName, (int)pkt.roomNo, (int)pkt.layer);
|
||||
@@ -134,11 +169,21 @@ bool ImGuiStateShare::applyEncodedState(const std::string& encoded, const std::s
|
||||
}
|
||||
|
||||
void ImGuiStateShare::tickPendingApply() {
|
||||
if (!m_pendingInfo.has_value() || dComIfGp_isEnableNextStage()) {
|
||||
if (!m_pendingInfo.has_value() && !m_pendingSavedata.has_value()) {
|
||||
return;
|
||||
}
|
||||
g_dComIfG_gameInfo.info = *m_pendingInfo;
|
||||
m_pendingInfo.reset();
|
||||
if (dComIfGp_isEnableNextStage()) {
|
||||
return;
|
||||
}
|
||||
if (m_pendingInfo.has_value()) {
|
||||
DuskLog.info("StateShare: tickPendingApply full - lastSceneMode={}", dComIfGs_getLastSceneMode());
|
||||
g_dComIfG_gameInfo.info = *m_pendingInfo;
|
||||
m_pendingInfo.reset();
|
||||
} else {
|
||||
DuskLog.info("StateShare: tickPendingApply partial - lastSceneMode={}", dComIfGs_getLastSceneMode());
|
||||
g_dComIfG_gameInfo.info.mSavedata = *m_pendingSavedata;
|
||||
m_pendingSavedata.reset();
|
||||
}
|
||||
dComIfGp_offOxygenShowFlag();
|
||||
dComIfGp_setMaxOxygen(600);
|
||||
dComIfGp_setOxygen(600);
|
||||
@@ -150,7 +195,55 @@ static bool ValidateEncodedState(const std::string& encoded) {
|
||||
return false;
|
||||
}
|
||||
unsigned long long dSize = ZSTD_getFrameContentSize(decoded.data(), decoded.size());
|
||||
return dSize != ZSTD_CONTENTSIZE_ERROR && dSize != ZSTD_CONTENTSIZE_UNKNOWN && dSize >= PACKET_TOTAL;
|
||||
return dSize == PACKET_TOTAL || dSize == PACKET_SAVE_ONLY;
|
||||
}
|
||||
|
||||
void ImGuiStateShare::mergeFromFile(const std::string& path) {
|
||||
try {
|
||||
auto data = io::FileStream::ReadAllBytes(path.c_str());
|
||||
auto j = json::parse(data);
|
||||
if (!j.is_array()) {
|
||||
m_statusMsg = "File does not contain a JSON array.";
|
||||
return;
|
||||
}
|
||||
|
||||
std::unordered_set<std::string> existingNames;
|
||||
for (const auto& s : m_states) {
|
||||
existingNames.insert(s.name);
|
||||
}
|
||||
|
||||
int added = 0;
|
||||
int skipped = 0;
|
||||
for (const auto& entry : j) {
|
||||
if (!entry.contains("name") || !entry.contains("data")) {
|
||||
++skipped;
|
||||
continue;
|
||||
}
|
||||
const std::string name = entry["name"].get<std::string>();
|
||||
const std::string encoded = entry["data"].get<std::string>();
|
||||
if (!ValidateEncodedState(encoded)) {
|
||||
++skipped;
|
||||
continue;
|
||||
}
|
||||
if (existingNames.count(name)) {
|
||||
++skipped;
|
||||
continue;
|
||||
}
|
||||
SavedStateEntry s;
|
||||
s.name = name;
|
||||
s.encoded = encoded;
|
||||
existingNames.insert(s.name);
|
||||
m_states.push_back(std::move(s));
|
||||
++added;
|
||||
}
|
||||
|
||||
if (added > 0) {
|
||||
saveStatesFile();
|
||||
}
|
||||
m_statusMsg = fmt::format("Merged: {} added, {} skipped.", added, skipped);
|
||||
} catch (const std::exception& e) {
|
||||
m_statusMsg = fmt::format("Failed to load file: {}", e.what());
|
||||
}
|
||||
}
|
||||
|
||||
void ImGuiStateShare::draw(bool& open) {
|
||||
@@ -162,6 +255,11 @@ void ImGuiStateShare::draw(bool& open) {
|
||||
loadStatesFile();
|
||||
}
|
||||
|
||||
if (!m_pendingMergePath.empty()) {
|
||||
mergeFromFile(m_pendingMergePath);
|
||||
m_pendingMergePath.clear();
|
||||
}
|
||||
|
||||
if (!open) {
|
||||
return;
|
||||
}
|
||||
@@ -243,7 +341,7 @@ void ImGuiStateShare::draw(bool& open) {
|
||||
|
||||
// Toolbar
|
||||
if (!gameRunning) { ImGui::BeginDisabled(); }
|
||||
if (ImGui::Button("Save Current")) {
|
||||
if (ImGui::Button("Current")) {
|
||||
SavedStateEntry entry;
|
||||
entry.name = fmt::format("State {}", m_states.size() + 1);
|
||||
entry.encoded = encodeCurrentState();
|
||||
@@ -273,6 +371,12 @@ void ImGuiStateShare::draw(bool& open) {
|
||||
}
|
||||
}
|
||||
|
||||
ImGui::SameLine();
|
||||
if (ImGui::Button("Load Pack")) {
|
||||
static constexpr SDL_DialogFileFilter filter = {"State pack", "json"};
|
||||
ShowFileSelect(&onMergeFileSelected, this, aurora::window::get_sdl_window(), &filter, 1, nullptr, false);
|
||||
}
|
||||
|
||||
if (!m_states.empty()) {
|
||||
ImGui::SameLine();
|
||||
if (ImGui::Button("Clear All")) {
|
||||
|
||||
@@ -23,13 +23,17 @@ private:
|
||||
void tickPendingApply();
|
||||
void loadStatesFile();
|
||||
void saveStatesFile();
|
||||
void mergeFromFile(const std::string& path);
|
||||
static void onMergeFileSelected(void* userdata, const char* path, const char* error);
|
||||
|
||||
std::vector<SavedStateEntry> m_states;
|
||||
std::string m_statusMsg;
|
||||
std::optional<dSv_info_c> m_pendingInfo;
|
||||
std::optional<dSv_info_c> m_pendingInfo;
|
||||
std::optional<dSv_save_c> m_pendingSavedata;
|
||||
int m_renamingIndex = -1;
|
||||
char m_renameBuffer[128] = {};
|
||||
bool m_loaded = false;
|
||||
std::string m_pendingMergePath;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
@@ -0,0 +1,58 @@
|
||||
"""
|
||||
Convert a folder of TPGZ saves to a states.json
|
||||
|
||||
Usage:
|
||||
python saves_to_states_json.py path/to/saves [prefix]
|
||||
|
||||
Requirements:
|
||||
pip install zstandard
|
||||
"""
|
||||
|
||||
import base64
|
||||
import json
|
||||
import struct
|
||||
import sys
|
||||
import zstandard
|
||||
from pathlib import Path
|
||||
|
||||
SAVE_C_SIZE = 0x958
|
||||
|
||||
PACKET_FORMAT = "<8sbbh"
|
||||
|
||||
RETURN_PLACE_OFF = 0x058
|
||||
NAME_OFF = RETURN_PLACE_OFF + 0x00
|
||||
ROOM_OFF = RETURN_PLACE_OFF + 0x09
|
||||
SPAWN_POINT_OFF = RETURN_PLACE_OFF + 0x08
|
||||
|
||||
folder = Path(sys.argv[1]) if len(sys.argv) > 1 else Path(__file__).parent
|
||||
out_path = folder / "states.json"
|
||||
|
||||
if len(sys.argv) > 2:
|
||||
prefix = sys.argv[2]
|
||||
else:
|
||||
prefix = None
|
||||
|
||||
cctx = zstandard.ZstdCompressor(level=1)
|
||||
states = []
|
||||
|
||||
for bin_path in sorted(folder.glob("*.bin")):
|
||||
raw = bin_path.read_bytes()
|
||||
save_c = raw[:SAVE_C_SIZE]
|
||||
if len(save_c) < SAVE_C_SIZE:
|
||||
print(f" skip {bin_path.name}: too small ({len(save_c)} bytes)")
|
||||
continue
|
||||
|
||||
stage_name = save_c[NAME_OFF:NAME_OFF + 8]
|
||||
room_no = struct.unpack_from("b", save_c, ROOM_OFF)[0]
|
||||
spawn_point = struct.unpack_from("B", save_c, SPAWN_POINT_OFF)[0]
|
||||
|
||||
pkt = struct.pack(PACKET_FORMAT, stage_name, room_no, -1, spawn_point)
|
||||
payload = pkt + save_c
|
||||
encoded = base64.b64encode(cctx.compress(payload)).decode("ascii")
|
||||
|
||||
stage_str = stage_name.rstrip(b"\x00").decode("ascii", errors="replace")
|
||||
print(f" {bin_path.stem:30s} stage={stage_str!r} room={room_no} point={spawn_point}")
|
||||
states.append({"name": f"({prefix}) {bin_path.stem}" if prefix else bin_path.stem, "data": encoded})
|
||||
|
||||
out_path.write_text(json.dumps(states, indent=2))
|
||||
print(f"\nWrote {len(states)} states to {out_path}")
|
||||
Reference in New Issue
Block a user