HostFileSystem: Set all NAND folders to be saved in save states when a movie is active

This commit is contained in:
Lobsterzelda 2022-10-19 23:40:43 -04:00
parent 0210d115c2
commit ed54e1905a
4 changed files with 155 additions and 81 deletions

View File

@ -229,6 +229,22 @@ public:
return current; return current;
} }
// The reserved u32 is set to 0, and a pointer to it is returned.
// The caller needs to fill in the reserved u32 with the appropriate value later on, if they
// want a non-zero value there.
[[nodiscard]] u8* ReserveU32()
{
u32 temp = 0;
u8* previous_pointer = *m_ptr_current;
Do(temp);
return previous_pointer;
}
u32 GetOffsetFromPreviousPosition(u8* previous_pointer)
{
return static_cast<u32>((*m_ptr_current) - previous_pointer);
}
void Do(Common::Flag& flag) void Do(Common::Flag& flag)
{ {
bool s = flag.IsSet(); bool s = flag.IsSet();

View File

@ -20,9 +20,13 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/IOS/ES/ES.h" #include "Core/IOS/ES/ES.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/Movie.h"
#include "Core/WiiRoot.h"
namespace IOS::HLE::FS namespace IOS::HLE::FS
{ {
constexpr u32 BUFFER_CHUNK_SIZE = 65536;
HostFileSystem::HostFilename HostFileSystem::BuildFilename(const std::string& wii_path) const HostFileSystem::HostFilename HostFileSystem::BuildFilename(const std::string& wii_path) const
{ {
for (const auto& redirect : m_nand_redirects) for (const auto& redirect : m_nand_redirects)
@ -252,18 +256,11 @@ HostFileSystem::FstEntry* HostFileSystem::GetFstEntryForPath(const std::string&
return entry; return entry;
} }
void HostFileSystem::DoState(PointerWrap& p) void HostFileSystem::DoStateRead(PointerWrap& p, std::string start_directory_path)
{ {
// Temporarily close the file, to prevent any issues with the savestating of /tmp std::string path = BuildFilename(start_directory_path).host_path;
for (Handle& handle : m_handles) File::DeleteDirRecursively(path);
handle.host_file.reset(); File::CreateDir(path);
// handle /tmp
std::string Path = BuildFilename("/tmp").host_path;
if (p.IsReadMode())
{
File::DeleteDirRecursively(Path);
File::CreateDir(Path);
// now restore from the stream // now restore from the stream
while (1) while (1)
@ -274,7 +271,7 @@ void HostFileSystem::DoState(PointerWrap& p)
break; break;
std::string file_name; std::string file_name;
p.Do(file_name); p.Do(file_name);
std::string name = Path + "/" + file_name; std::string name = path + "/" + file_name;
switch (type) switch (type)
{ {
case 'd': case 'd':
@ -288,13 +285,13 @@ void HostFileSystem::DoState(PointerWrap& p)
p.Do(size); p.Do(size);
File::IOFile handle(name, "wb"); File::IOFile handle(name, "wb");
char buf[65536]; char buf[BUFFER_CHUNK_SIZE];
u32 count = size; u32 count = size;
while (count > 65536) while (count > BUFFER_CHUNK_SIZE)
{ {
p.DoArray(buf); p.DoArray(buf);
handle.WriteArray(&buf[0], 65536); handle.WriteArray(&buf[0], BUFFER_CHUNK_SIZE);
count -= 65536; count -= BUFFER_CHUNK_SIZE;
} }
p.DoArray(&buf[0], count); p.DoArray(&buf[0], count);
handle.WriteArray(&buf[0], count); handle.WriteArray(&buf[0], count);
@ -302,12 +299,12 @@ void HostFileSystem::DoState(PointerWrap& p)
} }
} }
} }
} }
else
{
// recurse through tmp and save dirs and files
File::FSTEntry parent_entry = File::ScanDirectoryTree(Path, true); void HostFileSystem::DoStateWriteOrMeasure(PointerWrap& p, std::string start_directory_path)
{
std::string path = BuildFilename(start_directory_path).host_path;
File::FSTEntry parent_entry = File::ScanDirectoryTree(path, true);
std::deque<File::FSTEntry> todo; std::deque<File::FSTEntry> todo;
todo.insert(todo.end(), parent_entry.children.begin(), parent_entry.children.end()); todo.insert(todo.end(), parent_entry.children.begin(), parent_entry.children.end());
@ -315,7 +312,7 @@ void HostFileSystem::DoState(PointerWrap& p)
{ {
File::FSTEntry& entry = todo.front(); File::FSTEntry& entry = todo.front();
std::string name = entry.physicalName; std::string name = entry.physicalName;
name.erase(0, Path.length() + 1); name.erase(0, path.length() + 1);
char type = entry.isDirectory ? 'd' : 'f'; char type = entry.isDirectory ? 'd' : 'f';
p.Do(type); p.Do(type);
p.Do(name); p.Do(name);
@ -329,13 +326,13 @@ void HostFileSystem::DoState(PointerWrap& p)
p.Do(size); p.Do(size);
File::IOFile handle(entry.physicalName, "rb"); File::IOFile handle(entry.physicalName, "rb");
char buf[65536]; char buf[BUFFER_CHUNK_SIZE];
u32 count = size; u32 count = size;
while (count > 65536) while (count > BUFFER_CHUNK_SIZE)
{ {
handle.ReadArray(&buf[0], 65536); handle.ReadArray(&buf[0], BUFFER_CHUNK_SIZE);
p.DoArray(buf); p.DoArray(buf);
count -= 65536; count -= BUFFER_CHUNK_SIZE;
} }
handle.ReadArray(&buf[0], count); handle.ReadArray(&buf[0], count);
p.DoArray(&buf[0], count); p.DoArray(&buf[0], count);
@ -345,6 +342,65 @@ void HostFileSystem::DoState(PointerWrap& p)
char type = 0; char type = 0;
p.Do(type); p.Do(type);
}
void HostFileSystem::DoState(PointerWrap& p)
{
// Temporarily close the file, to prevent any issues with the savestating of files/folders.
for (Handle& handle : m_handles)
handle.host_file.reset();
// The format for the next part of the save state is follows:
// 1. bool Movie::WasMovieActiveWhenStateSaved() &&
// WiiRoot::WasWiiRootTemporaryDirectoryWhenStateSaved()
// 2. Contents of the "/tmp" directory recursively.
// 3. u32 size_of_nand_folder_saved_below (or 0, if the root
// of the NAND folder is not savestated below).
// 4. Contents of the "/" directory recursively (or nothing, if the
// root of the NAND folder is not save stated).
// The "/" directory is only saved when a savestate is made during a movie recording
// and when the directory root is temporary (i.e. WiiSession).
// If a save state is made during a movie recording and is loaded when no movie is active,
// then a call to p.DoExternal() will be used to skip over reading the contents of the "/"
// directory (it skips over the number of bytes specified by size_of_nand_folder_saved)
bool original_save_state_made_during_movie_recording =
Movie::IsMovieActive() && Core::WiiRootIsTemporary();
p.Do(original_save_state_made_during_movie_recording);
u32 temp_val = 0;
if (!p.IsReadMode())
{
DoStateWriteOrMeasure(p, "/tmp");
u8* previous_position = p.ReserveU32();
if (original_save_state_made_during_movie_recording)
{
DoStateWriteOrMeasure(p, "/");
if (p.IsWriteMode())
{
u32 size_of_nand = p.GetOffsetFromPreviousPosition(previous_position) - sizeof(u32);
memcpy(previous_position, &size_of_nand, sizeof(u32));
}
}
}
else // case where we're in read mode.
{
DoStateRead(p, "/tmp");
if (!Movie::IsMovieActive() || !original_save_state_made_during_movie_recording ||
!Core::WiiRootIsTemporary() ||
(original_save_state_made_during_movie_recording !=
(Movie::IsMovieActive() && Core::WiiRootIsTemporary())))
{
(void)p.DoExternal(temp_val);
}
else
{
p.Do(temp_val);
if (Movie::IsMovieActive() && Core::WiiRootIsTemporary())
DoStateRead(p, "/");
}
} }
for (Handle& handle : m_handles) for (Handle& handle : m_handles)

View File

@ -59,10 +59,12 @@ public:
void SetNandRedirects(std::vector<NandRedirect> nand_redirects) override; void SetNandRedirects(std::vector<NandRedirect> nand_redirects) override;
private: private:
void DoStateWriteOrMeasure(PointerWrap& p, std::string start_directory_path);
void DoStateRead(PointerWrap& p, std::string start_directory_path);
struct FstEntry struct FstEntry
{ {
bool CheckPermission(Uid uid, Gid gid, Mode requested_mode) const; bool CheckPermission(Uid uid, Gid gid, Mode requested_mode) const;
std::string name; std::string name;
Metadata data{}; Metadata data{};
/// Children of this FST entry. Only valid for directories. /// Children of this FST entry. Only valid for directories.

View File

@ -94,7 +94,7 @@ static size_t s_state_writes_in_queue;
static std::condition_variable s_state_write_queue_is_empty; static std::condition_variable s_state_write_queue_is_empty;
// Don't forget to increase this after doing changes on the savestate system // Don't forget to increase this after doing changes on the savestate system
constexpr u32 STATE_VERSION = 155; // Last changed in PR 10890 constexpr u32 STATE_VERSION = 156; // Last changed in PR 11184
// Maps savestate versions to Dolphin versions. // Maps savestate versions to Dolphin versions.
// Versions after 42 don't need to be added to this list, // Versions after 42 don't need to be added to this list,