FolderMemoryCard: Add a helper structure to quickly access a file entry from a file data FAT cluster. Speeds up file access, especially when a lot of files are loaded in the virtual memory card.

This commit is contained in:
Admiral H. Curtiss 2015-06-09 23:44:00 +02:00
parent bee4f0578d
commit 80feb1087c
2 changed files with 96 additions and 24 deletions

View File

@ -36,6 +36,7 @@ void FolderMemoryCard::InitializeInternalData() {
memset( &m_backupBlock1, 0xFF, sizeof( m_backupBlock1 ) ); memset( &m_backupBlock1, 0xFF, sizeof( m_backupBlock1 ) );
memset( &m_backupBlock2, 0xFF, sizeof( m_backupBlock2 ) ); memset( &m_backupBlock2, 0xFF, sizeof( m_backupBlock2 ) );
m_cache.clear(); m_cache.clear();
m_fileMetadataQuickAccess.clear();
m_timeLastWritten = 0; m_timeLastWritten = 0;
m_isEnabled = false; m_isEnabled = false;
m_framesUntilFlush = 0; m_framesUntilFlush = 0;
@ -102,6 +103,8 @@ void FolderMemoryCard::Close() {
superBlockFile.Write( &m_superBlock.raw, sizeof( m_superBlock.raw ) ); superBlockFile.Write( &m_superBlock.raw, sizeof( m_superBlock.raw ) );
} }
m_cache.clear();
m_fileMetadataQuickAccess.clear();
m_lastAccessedFile.Close(); m_lastAccessedFile.Close();
} }
@ -128,7 +131,7 @@ void FolderMemoryCard::LoadMemoryCardData( const bool enableFiltering, const wxS
CreateFat(); CreateFat();
CreateRootDir(); CreateRootDir();
MemoryCardFileEntry* const rootDirEntry = &m_fileEntryDict[m_superBlock.data.rootdir_cluster].entries[0]; MemoryCardFileEntry* const rootDirEntry = &m_fileEntryDict[m_superBlock.data.rootdir_cluster].entries[0];
AddFolder( rootDirEntry, m_folderName.GetPath(), enableFiltering, filter ); AddFolder( rootDirEntry, m_folderName.GetPath(), nullptr, enableFiltering, filter );
} }
} }
@ -270,7 +273,7 @@ bool FilterMatches( const wxString& fileName, const wxString& filter ) {
return false; return false;
} }
bool FolderMemoryCard::AddFolder( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const bool enableFiltering, const wxString& filter ) { bool FolderMemoryCard::AddFolder( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, MemoryCardFileMetadataReference* parent, const bool enableFiltering, const wxString& filter ) {
wxDir dir( dirPath ); wxDir dir( dirPath );
if ( dir.IsOpened() ) { if ( dir.IsOpened() ) {
Console.WriteLn( L"(FolderMcd) Adding folder: %s", WX_STR( dirPath ) ); Console.WriteLn( L"(FolderMcd) Adding folder: %s", WX_STR( dirPath ) );
@ -302,7 +305,7 @@ bool FolderMemoryCard::AddFolder( MemoryCardFileEntry* const dirEntry, const wxS
bool isFile = wxFile::Exists( fileInfo.GetFullPath() ); bool isFile = wxFile::Exists( fileInfo.GetFullPath() );
if ( isFile ) { if ( isFile ) {
if ( AddFile( dirEntry, dirPath, fileName ) ) { if ( AddFile( dirEntry, dirPath, fileName, parent ) ) {
++entryNumber; ++entryNumber;
} }
} else { } else {
@ -365,10 +368,12 @@ bool FolderMemoryCard::AddFolder( MemoryCardFileEntry* const dirEntry, const wxS
subDirCluster->entries[1].entry.data.name[0] = '.'; subDirCluster->entries[1].entry.data.name[0] = '.';
subDirCluster->entries[1].entry.data.name[1] = '.'; subDirCluster->entries[1].entry.data.name[1] = '.';
MemoryCardFileMetadataReference* dirRef = AddDirEntryToMetadataQuickAccess( newDirEntry, parent );
++entryNumber; ++entryNumber;
// and add all files in subdir // and add all files in subdir
AddFolder( newDirEntry, fileInfo.GetFullPath() ); AddFolder( newDirEntry, fileInfo.GetFullPath(), dirRef );
} }
hasNext = dir.GetNext( &fileName ); hasNext = dir.GetNext( &fileName );
@ -380,7 +385,7 @@ bool FolderMemoryCard::AddFolder( MemoryCardFileEntry* const dirEntry, const wxS
return false; return false;
} }
bool FolderMemoryCard::AddFile( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const wxString& fileName ) { bool FolderMemoryCard::AddFile( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const wxString& fileName, MemoryCardFileMetadataReference* parent ) {
wxFileName relativeFilePath( dirPath, fileName ); wxFileName relativeFilePath( dirPath, fileName );
relativeFilePath.MakeRelativeTo( m_folderName.GetPath() ); relativeFilePath.MakeRelativeTo( m_folderName.GetPath() );
Console.WriteLn( L"(FolderMcd) Adding file: %s", WX_STR( relativeFilePath.GetFullPath() ) ); Console.WriteLn( L"(FolderMcd) Adding file: %s", WX_STR( relativeFilePath.GetFullPath() ) );
@ -434,6 +439,8 @@ bool FolderMemoryCard::AddFile( MemoryCardFileEntry* const dirEntry, const wxStr
} }
file.Close(); file.Close();
AddFileEntryToMetadataQuickAccess( newFileEntry, parent );
} else { } else {
Console.WriteLn( L"(FolderMcd) Could not open file: %s", WX_STR( relativeFilePath.GetFullPath() ) ); Console.WriteLn( L"(FolderMcd) Could not open file: %s", WX_STR( relativeFilePath.GetFullPath() ) );
return false; return false;
@ -445,6 +452,44 @@ bool FolderMemoryCard::AddFile( MemoryCardFileEntry* const dirEntry, const wxStr
return true; return true;
} }
MemoryCardFileMetadataReference* FolderMemoryCard::AddDirEntryToMetadataQuickAccess( MemoryCardFileEntry* const entry, MemoryCardFileMetadataReference* const parent ) {
MemoryCardFileMetadataReference* ref = &m_fileMetadataQuickAccess[entry->entry.data.cluster];
ref->parent = parent;
ref->entry = entry;
ref->consecutiveCluster = 0xFFFFFFFFu;
return ref;
}
void FolderMemoryCard::AddFileEntryToMetadataQuickAccess( MemoryCardFileEntry* const entry, MemoryCardFileMetadataReference* const parent ) {
u32 fileCluster = entry->entry.data.cluster;
// zero-length files have no file clusters
if ( fileCluster == 0xFFFFFFFFu ) {
return;
}
u32 clusterNumber = 0;
do {
MemoryCardFileMetadataReference* ref = &m_fileMetadataQuickAccess[fileCluster & 0x7FFFFFFFu];
ref->parent = parent;
ref->entry = entry;
ref->consecutiveCluster = clusterNumber;
++clusterNumber;
} while ( ( fileCluster = m_fat.data[0][0][fileCluster] ) != 0xFFFFFFFFu );
}
void MemoryCardFileMetadataReference::GetPath( wxFileName* fileName ) {
if ( parent ) {
parent->GetPath( fileName );
}
if ( entry->IsDir() ) {
fileName->AppendDir( wxString::FromAscii( (const char*)entry->entry.data.name ) );
} else if ( entry->IsFile() ) {
fileName->SetName( wxString::FromAscii( (const char*)entry->entry.data.name ) );
}
}
s32 FolderMemoryCard::IsPresent() { s32 FolderMemoryCard::IsPresent() {
return m_isEnabled; return m_isEnabled;
} }
@ -536,6 +581,8 @@ u8* FolderMemoryCard::GetFileEntryPointer( const u32 currentCluster, const u32 s
return nullptr; return nullptr;
} }
// This method is actually unused since the introduction of m_fileMetadataQuickAccess.
// I'll leave it here anyway though to show how you traverse the file system.
MemoryCardFileEntry* FolderMemoryCard::GetFileEntryFromFileDataCluster( const u32 currentCluster, const u32 searchCluster, wxFileName* fileName, const size_t originalDirCount, u32* outClusterNumber ) { MemoryCardFileEntry* FolderMemoryCard::GetFileEntryFromFileDataCluster( const u32 currentCluster, const u32 searchCluster, wxFileName* fileName, const size_t originalDirCount, u32* outClusterNumber ) {
// check both entries of the current cluster if they're the file we're searching for, and if yes return it // check both entries of the current cluster if they're the file we're searching for, and if yes return it
for ( int i = 0; i < 2; ++i ) { for ( int i = 0; i < 2; ++i ) {
@ -551,11 +598,6 @@ MemoryCardFileEntry* FolderMemoryCard::GetFileEntryFromFileDataCluster( const u3
} }
++clusterNumber; ++clusterNumber;
} while ( ( fileCluster = m_fat.data[0][0][fileCluster] & 0x7FFFFFFF ) != 0x7FFFFFFF ); } while ( ( fileCluster = m_fat.data[0][0][fileCluster] & 0x7FFFFFFF ) != 0x7FFFFFFF );
// There's a lot of optimization work that can be done here, looping through all clusters of every single file
// is not very efficient, especially since files are going to be accessed from the start and in-order the vast
// majority of the time. You can probably cut a lot of the work by remembering the state of the last access
// and only checking if the current access is either the same or the next cluster according to the FAT.
//} while ( false );
} }
} }
@ -594,10 +636,11 @@ bool FolderMemoryCard::ReadFromFile( u8 *dest, u32 adr, u32 dataLength ) {
} }
// figure out which file to read from // figure out which file to read from
auto it = m_fileMetadataQuickAccess.find( fatCluster );
if ( it != m_fileMetadataQuickAccess.end() ) {
wxFileName fileName( m_folderName ); wxFileName fileName( m_folderName );
u32 clusterNumber; const u32 clusterNumber = it->second.consecutiveCluster;
const MemoryCardFileEntry* const entry = GetFileEntryFromFileDataCluster( m_superBlock.data.rootdir_cluster, fatCluster, &fileName, fileName.GetDirCount(), &clusterNumber ); it->second.GetPath( &fileName );
if ( entry != nullptr ) {
wxFFile* file = m_lastAccessedFile.ReOpen( fileName.GetFullPath(), L"rb" ); wxFFile* file = m_lastAccessedFile.ReOpen( fileName.GetFullPath(), L"rb" );
if ( file->IsOpened() ) { if ( file->IsOpened() ) {
const u32 clusterOffset = ( page % 2 ) * PageSize + offset; const u32 clusterOffset = ( page % 2 ) * PageSize + offset;
@ -806,7 +849,7 @@ void FolderMemoryCard::FlushBlock( const u32 block ) {
} }
} }
void FolderMemoryCard::FlushFileEntries( const u32 dirCluster, const u32 remainingFiles, const wxString& dirPath ) { void FolderMemoryCard::FlushFileEntries( const u32 dirCluster, const u32 remainingFiles, const wxString& dirPath, MemoryCardFileMetadataReference* parent ) {
// flush the current cluster // flush the current cluster
FlushCluster( dirCluster + m_superBlock.data.alloc_offset ); FlushCluster( dirCluster + m_superBlock.data.alloc_offset );
@ -839,15 +882,19 @@ void FolderMemoryCard::FlushFileEntries( const u32 dirCluster, const u32 remaini
} }
} }
FlushFileEntries( cluster, entry->entry.data.length, subDirPath ); MemoryCardFileMetadataReference* dirRef = AddDirEntryToMetadataQuickAccess( entry, parent );
FlushFileEntries( cluster, entry->entry.data.length, subDirPath, dirRef );
} }
} else if ( entry->IsValid() && entry->IsUsed() && entry->IsFile() ) {
AddFileEntryToMetadataQuickAccess( entry, parent );
} }
} }
// continue to the next cluster of this directory // continue to the next cluster of this directory
const u32 nextCluster = m_fat.data[0][0][dirCluster]; const u32 nextCluster = m_fat.data[0][0][dirCluster];
if ( nextCluster != 0xFFFFFFFF ) { if ( nextCluster != 0xFFFFFFFF ) {
FlushFileEntries( nextCluster & 0x7FFFFFFF, remainingFiles - 2, dirPath ); FlushFileEntries( nextCluster & 0x7FFFFFFF, remainingFiles - 2, dirPath, parent );
} }
} }
@ -899,10 +946,12 @@ bool FolderMemoryCard::WriteToFile( const u8* src, u32 adr, u32 dataLength ) {
} }
// figure out which file to write to // figure out which file to write to
auto it = m_fileMetadataQuickAccess.find( fatCluster );
if ( it != m_fileMetadataQuickAccess.end() ) {
wxFileName fileName( m_folderName ); wxFileName fileName( m_folderName );
u32 clusterNumber; const MemoryCardFileEntry* const entry = it->second.entry;
const MemoryCardFileEntry* const entry = GetFileEntryFromFileDataCluster( m_superBlock.data.rootdir_cluster, fatCluster, &fileName, fileName.GetDirCount(), &clusterNumber ); const u32 clusterNumber = it->second.consecutiveCluster;
if ( entry != nullptr ) { it->second.GetPath( &fileName );
wxFFile* file = m_lastAccessedFile.ReOpen( fileName.GetFullPath(), L"r+b" ); wxFFile* file = m_lastAccessedFile.ReOpen( fileName.GetFullPath(), L"r+b" );
if ( file->IsOpened() ) { if ( file->IsOpened() ) {
const u32 clusterOffset = ( page % 2 ) * PageSize + offset; const u32 clusterOffset = ( page % 2 ) * PageSize + offset;

View File

@ -131,6 +131,18 @@ struct MemoryCardPage {
}; };
#pragma pack(pop) #pragma pack(pop)
// --------------------------------------------------------------------------------------
// MemoryCardFileMetadataReference
// --------------------------------------------------------------------------------------
// Helper structure to quickly access file entries from any file data FAT cluster
struct MemoryCardFileMetadataReference {
MemoryCardFileMetadataReference* parent;
MemoryCardFileEntry* entry;
u32 consecutiveCluster;
void GetPath( wxFileName* fileName );
};
// -------------------------------------------------------------------------------------- // --------------------------------------------------------------------------------------
// FileAccessHelper // FileAccessHelper
// -------------------------------------------------------------------------------------- // --------------------------------------------------------------------------------------
@ -198,6 +210,8 @@ protected:
// stores directory and file metadata // stores directory and file metadata
std::map<u32, MemoryCardFileEntryCluster> m_fileEntryDict; std::map<u32, MemoryCardFileEntryCluster> m_fileEntryDict;
// quick-access map of related file entry metadata for each memory card FAT cluster that contains file data
std::map<u32, MemoryCardFileMetadataReference> m_fileMetadataQuickAccess;
// holds a copy of modified pages of the memory card before they're flushed to the file system // holds a copy of modified pages of the memory card before they're flushed to the file system
std::map<u32, MemoryCardPage> m_cache; std::map<u32, MemoryCardPage> m_cache;
@ -320,14 +334,23 @@ protected:
// adds a folder in the host file system to the memory card, including all files and subdirectories // adds a folder in the host file system to the memory card, including all files and subdirectories
// - dirEntry: the entry of the directory in the parent directory, or the root "." entry // - dirEntry: the entry of the directory in the parent directory, or the root "." entry
// - dirPath: the full path to the directory in the host file system // - dirPath: the full path to the directory in the host file system
// - parent: pointer to the parent dir's quick-access reference element
// - enableFiltering and filter: filter loaded contents, see LoadMemoryCardData() // - enableFiltering and filter: filter loaded contents, see LoadMemoryCardData()
bool AddFolder( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const bool enableFiltering = false, const wxString& filter = L"" ); bool AddFolder( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, MemoryCardFileMetadataReference* parent = nullptr, const bool enableFiltering = false, const wxString& filter = L"" );
// adds a file in the host file sytem to the memory card // adds a file in the host file sytem to the memory card
// - dirEntry: the entry of the directory in the parent directory, or the root "." entry // - dirEntry: the entry of the directory in the parent directory, or the root "." entry
// - dirPath: the full path to the directory containing the file in the host file system // - dirPath: the full path to the directory containing the file in the host file system
// - fileName: the name of the file, without path // - fileName: the name of the file, without path
bool AddFile( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const wxString& fileName ); // - parent: pointer to the parent dir's quick-access reference element
bool AddFile( MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const wxString& fileName, MemoryCardFileMetadataReference* parent = nullptr );
// adds a file to the quick-access dictionary, so it can be accessed more efficiently (ie, without searching through the entire file system) later
void AddFileEntryToMetadataQuickAccess( MemoryCardFileEntry* const entry, MemoryCardFileMetadataReference* const parent );
// creates a reference to a directory entry, so it can be passed as parent to other files/directories
MemoryCardFileMetadataReference* AddDirEntryToMetadataQuickAccess( MemoryCardFileEntry* const entry, MemoryCardFileMetadataReference* const parent );
bool ReadFromFile( u8 *dest, u32 adr, u32 dataLength ); bool ReadFromFile( u8 *dest, u32 adr, u32 dataLength );
@ -347,7 +370,7 @@ protected:
void FlushBlock( const u32 block ); void FlushBlock( const u32 block );
// flush a directory's file entries and all its subdirectories to the internal data // flush a directory's file entries and all its subdirectories to the internal data
void FlushFileEntries( const u32 dirCluster, const u32 remainingFiles, const wxString& dirPath = L"" ); void FlushFileEntries( const u32 dirCluster, const u32 remainingFiles, const wxString& dirPath = L"", MemoryCardFileMetadataReference* parent = nullptr );
// write data as Save() normally would, but ignore the cache; used for flushing // write data as Save() normally would, but ignore the cache; used for flushing
s32 WriteWithoutCache( const u8 *src, u32 adr, int size ); s32 WriteWithoutCache( const u8 *src, u32 adr, int size );