pcsx2/pcsx2/gui/MemoryCardFolder.cpp

2197 lines
64 KiB
C++

/* PCSX2 - PS2 Emulator for PCs
* Copyright (C) 2002-2015 PCSX2 Dev Team
*
* PCSX2 is free software: you can redistribute it and/or modify it under the terms
* of the GNU Lesser General Public License as published by the Free Software Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* PCSX2 is distributed in the hope that 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 PCSX2.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include "PrecompiledHeader.h"
#include "Utilities/SafeArray.inl"
#include "MemoryCardFile.h"
#include "MemoryCardFolder.h"
#include "System.h"
#include "AppConfig.h"
#include "yaml-cpp/yaml.h"
#include "svnrev.h"
bool RemoveDirectory(const wxString& dirname);
// A helper function to parse the YAML file
static YAML::Node LoadYAMLFromFile(const wxString& fileName)
{
YAML::Node index;
wxFFile indexFile;
bool result;
{
// Suppress "file does not exist" errors
wxLogNull noLog;
result = indexFile.Open(fileName, L"r");
}
if (result)
{
wxString fileContents;
if (indexFile.ReadAll(&fileContents))
{
index = YAML::Load(fileContents.mbc_str());
}
}
return index;
}
FolderMemoryCard::FolderMemoryCard()
{
m_slot = 0;
m_isEnabled = false;
m_performFileWrites = false;
m_framesUntilFlush = 0;
m_timeLastWritten = 0;
m_filteringEnabled = false;
m_filteringString = L"";
}
void FolderMemoryCard::InitializeInternalData()
{
memset(&m_superBlock, 0xFF, sizeof(m_superBlock));
memset(&m_indirectFat, 0xFF, sizeof(m_indirectFat));
memset(&m_fat, 0xFF, sizeof(m_fat));
memset(&m_backupBlock1, 0xFF, sizeof(m_backupBlock1));
memset(&m_backupBlock2, 0xFF, sizeof(m_backupBlock2));
m_cache.clear();
m_oldDataCache.clear();
m_lastAccessedFile.CloseAll();
m_fileMetadataQuickAccess.clear();
m_timeLastWritten = 0;
m_isEnabled = false;
m_framesUntilFlush = 0;
m_performFileWrites = true;
m_filteringEnabled = false;
m_filteringString = L"";
}
bool FolderMemoryCard::IsFormatted() const
{
// this should be a good enough arbitrary check, if someone can think of a case where this doesn't work feel free to change
return m_superBlock.raw[0x16] == 0x6F;
}
void FolderMemoryCard::Open(const bool enableFiltering, const wxString& filter)
{
Open(g_Conf->FullpathToMcd(m_slot), g_Conf->Mcd[m_slot], 0, enableFiltering, filter, false);
}
void FolderMemoryCard::Open(const wxString& fullPath, const AppConfig::McdOptions& mcdOptions, const u32 sizeInClusters, const bool enableFiltering, const wxString& filter, bool simulateFileWrites)
{
InitializeInternalData();
m_performFileWrites = !simulateFileWrites;
wxFileName configuredFileName(fullPath);
m_folderName = wxFileName(configuredFileName.GetFullPath() + L"/");
wxString str(configuredFileName.GetFullPath());
bool disabled = false;
if (mcdOptions.Enabled && mcdOptions.Type == MemoryCardType::MemoryCard_Folder)
{
if (configuredFileName.GetFullName().IsEmpty())
{
str = L"[empty filename]";
disabled = true;
}
if (!disabled && configuredFileName.FileExists())
{
str = L"[is file, should be folder]";
disabled = true;
}
// if nothing exists at a valid location, create a directory for the memory card
if (!disabled && m_performFileWrites && !m_folderName.DirExists())
{
if (!m_folderName.Mkdir())
{
str = L"[couldn't create folder]";
disabled = true;
}
}
}
else
{
// if the user has disabled this slot or is using a different memory card type, just return without a console log
return;
}
Console.WriteLn(disabled ? Color_Gray : Color_Green, L"McdSlot %u: [Folder] " + str, m_slot);
if (disabled)
return;
m_isEnabled = true;
m_filteringEnabled = enableFiltering;
m_filteringString = filter;
LoadMemoryCardData(sizeInClusters, enableFiltering, filter);
SetTimeLastWrittenToNow();
m_framesUntilFlush = 0;
}
void FolderMemoryCard::Close(bool flush)
{
if (!m_isEnabled)
{
return;
}
if (flush)
{
Flush();
}
m_cache.clear();
m_oldDataCache.clear();
m_lastAccessedFile.CloseAll();
m_fileMetadataQuickAccess.clear();
}
bool FolderMemoryCard::ReIndex(bool enableFiltering, const wxString& filter)
{
if (!m_isEnabled)
{
return false;
}
if (m_filteringEnabled != enableFiltering || m_filteringString != filter)
{
Close();
Open(enableFiltering, filter);
return true;
}
return false;
}
void FolderMemoryCard::LoadMemoryCardData(const u32 sizeInClusters, const bool enableFiltering, const wxString& filter)
{
bool formatted = false;
// read superblock if it exists
wxFileName superBlockFileName(m_folderName.GetPath(), L"_pcsx2_superblock");
if (superBlockFileName.FileExists())
{
wxFFile superBlockFile(superBlockFileName.GetFullPath().c_str(), L"rb");
if (superBlockFile.IsOpened() && superBlockFile.Read(&m_superBlock.raw, sizeof(m_superBlock.raw)) >= sizeof(m_superBlock.data))
{
formatted = IsFormatted();
}
}
if (sizeInClusters > 0 && sizeInClusters != GetSizeInClusters())
{
SetSizeInClusters(sizeInClusters);
FlushBlock(0);
}
// if superblock was valid, load folders and files
if (formatted)
{
if (enableFiltering)
{
Console.WriteLn(Color_Green, L"(FolderMcd) Indexing slot %u with filter \"%s\".", m_slot, WX_STR(filter));
}
else
{
Console.WriteLn(Color_Green, L"(FolderMcd) Indexing slot %u without filter.", m_slot);
}
CreateFat();
CreateRootDir();
MemoryCardFileEntry* const rootDirEntry = &m_fileEntryDict[m_superBlock.data.rootdir_cluster].entries[0];
AddFolder(rootDirEntry, m_folderName.GetPath(), nullptr, enableFiltering, filter);
#ifdef DEBUG_WRITE_FOLDER_CARD_IN_MEMORY_TO_FILE_ON_CHANGE
WriteToFile(m_folderName.GetFullPath().RemoveLast() + L"-debug_" + wxDateTime::Now().Format(L"%Y-%m-%d-%H-%M-%S") + L"_load.ps2");
#endif
}
}
void FolderMemoryCard::CreateFat()
{
const u32 totalClusters = m_superBlock.data.clusters_per_card;
const u32 clusterSize = m_superBlock.data.page_len * m_superBlock.data.pages_per_cluster;
const u32 fatEntriesPerCluster = clusterSize / 4;
const u32 countFatClusters = (totalClusters % fatEntriesPerCluster) != 0 ? (totalClusters / fatEntriesPerCluster + 1) : (totalClusters / fatEntriesPerCluster);
const u32 countDataClusters = m_superBlock.data.alloc_end;
// create indirect FAT
for (unsigned int i = 0; i < countFatClusters; ++i)
{
m_indirectFat.data[0][i] = GetFreeSystemCluster();
}
// fill FAT with default values
for (unsigned int i = 0; i < countDataClusters; ++i)
{
m_fat.data[0][0][i] = 0x7FFFFFFFu;
}
}
void FolderMemoryCard::CreateRootDir()
{
MemoryCardFileEntryCluster* const rootCluster = &m_fileEntryDict[m_superBlock.data.rootdir_cluster];
memset(rootCluster->entries[0].entry.raw, 0x00, sizeof(rootCluster->entries[0].entry.raw));
rootCluster->entries[0].entry.data.mode = MemoryCardFileEntry::Mode_Read | MemoryCardFileEntry::Mode_Write | MemoryCardFileEntry::Mode_Execute | MemoryCardFileEntry::Mode_Directory | MemoryCardFileEntry::Mode_Unknown0x0400 | MemoryCardFileEntry::Mode_Used;
rootCluster->entries[0].entry.data.length = 2;
rootCluster->entries[0].entry.data.name[0] = '.';
memset(rootCluster->entries[1].entry.raw, 0x00, sizeof(rootCluster->entries[1].entry.raw));
rootCluster->entries[1].entry.data.mode = MemoryCardFileEntry::Mode_Write | MemoryCardFileEntry::Mode_Execute | MemoryCardFileEntry::Mode_Directory | MemoryCardFileEntry::Mode_Unknown0x0400 | MemoryCardFileEntry::Mode_Unknown0x2000 | MemoryCardFileEntry::Mode_Used;
rootCluster->entries[1].entry.data.name[0] = '.';
rootCluster->entries[1].entry.data.name[1] = '.';
// mark root dir cluster as used
m_fat.data[0][0][m_superBlock.data.rootdir_cluster] = LastDataCluster | DataClusterInUseMask;
}
u32 FolderMemoryCard::GetFreeSystemCluster() const
{
// first block is reserved for superblock
u32 highestUsedCluster = (m_superBlock.data.pages_per_block / m_superBlock.data.pages_per_cluster) - 1;
// can't use any of the indirect fat clusters
for (int i = 0; i < IndirectFatClusterCount; ++i)
{
highestUsedCluster = std::max(highestUsedCluster, m_superBlock.data.ifc_list[i]);
}
// or fat clusters
for (int i = 0; i < IndirectFatClusterCount; ++i)
{
for (int j = 0; j < ClusterSize / 4; ++j)
{
if (m_indirectFat.data[i][j] != IndirectFatUnused)
{
highestUsedCluster = std::max(highestUsedCluster, m_indirectFat.data[i][j]);
}
}
}
return highestUsedCluster + 1;
}
u32 FolderMemoryCard::GetAmountDataClusters() const
{
// BIOS reports different cluster values than what the memory card actually has, match that when adding files
// 8mb card -> BIOS: 7999 clusters / Superblock: 8135 clusters
// 16mb card -> BIOS: 15999 clusters / Superblock: 16295 clusters
// 32mb card -> BIOS: 31999 clusters / Superblock: 32615 clusters
// 64mb card -> BIOS: 64999 clusters / Superblock: 65255 clusters
return (m_superBlock.data.alloc_end / 1000) * 1000 - 1;
}
u32 FolderMemoryCard::GetFreeDataCluster() const
{
const u32 countDataClusters = GetAmountDataClusters();
for (unsigned int i = 0; i < countDataClusters; ++i)
{
const u32 cluster = m_fat.data[0][0][i];
if ((cluster & DataClusterInUseMask) == 0)
{
return i;
}
}
return 0xFFFFFFFFu;
}
u32 FolderMemoryCard::GetAmountFreeDataClusters() const
{
const u32 countDataClusters = GetAmountDataClusters();
u32 countFreeDataClusters = 0;
for (unsigned int i = 0; i < countDataClusters; ++i)
{
const u32 cluster = m_fat.data[0][0][i];
if ((cluster & DataClusterInUseMask) == 0)
{
++countFreeDataClusters;
}
}
return countFreeDataClusters;
}
u32 FolderMemoryCard::GetLastClusterOfData(const u32 cluster) const
{
u32 entryCluster;
u32 nextCluster = cluster;
do
{
entryCluster = nextCluster;
nextCluster = m_fat.data[0][0][entryCluster] & NextDataClusterMask;
} while (nextCluster != LastDataCluster);
return entryCluster;
}
MemoryCardFileEntry* FolderMemoryCard::AppendFileEntryToDir(const MemoryCardFileEntry* const dirEntry)
{
u32 entryCluster = GetLastClusterOfData(dirEntry->entry.data.cluster);
MemoryCardFileEntry* newFileEntry;
if (dirEntry->entry.data.length % 2 == 0)
{
// need new cluster
u32 newCluster = GetFreeDataCluster();
if (newCluster == 0xFFFFFFFFu)
{
return nullptr;
}
m_fat.data[0][0][entryCluster] = newCluster | DataClusterInUseMask;
m_fat.data[0][0][newCluster] = LastDataCluster | DataClusterInUseMask;
newFileEntry = &m_fileEntryDict[newCluster].entries[0];
}
else
{
// can use last page of existing clusters
newFileEntry = &m_fileEntryDict[entryCluster].entries[1];
}
return newFileEntry;
}
bool FilterMatches(const wxString& fileName, const wxString& filter)
{
size_t start = 0;
size_t len = filter.Len();
while (start < len)
{
size_t end = filter.find('/', start);
if (end == wxString::npos)
{
end = len;
}
wxString singleFilter = filter.Mid(start, end - start);
if (fileName.Contains(singleFilter))
{
return true;
}
start = end + 1;
}
return false;
}
bool FolderMemoryCard::AddFolder(MemoryCardFileEntry* const dirEntry, const wxString& dirPath, MemoryCardFileMetadataReference* parent, const bool enableFiltering, const wxString& filter)
{
if (wxDir::Exists(dirPath))
{
wxString localFilter;
if (enableFiltering)
{
bool hasFilter = !filter.IsEmpty();
if (hasFilter)
{
localFilter = L"DATA-SYSTEM/BWNETCNF/" + filter;
}
else
{
localFilter = L"DATA-SYSTEM/BWNETCNF";
}
}
int entryNumber = 2; // include . and ..
for (const auto& file : GetOrderedFiles(dirPath))
{
wxFileName fileInfo(dirPath, file.m_fileName);
if (file.m_isFile)
{
// don't load files in the root dir if we're filtering; no official software stores files there
if (enableFiltering && parent == nullptr)
{
continue;
}
if (AddFile(dirEntry, dirPath, file, parent))
{
++entryNumber;
}
}
else
{
// if possible filter added directories by game serial
// this has the effective result of only files relevant to the current game being loaded into the memory card
// which means every game essentially sees the memory card as if no other files exist
if (enableFiltering && !FilterMatches(file.m_fileName, localFilter))
{
continue;
}
// make sure we have enough space on the memcard for the directory
const u32 newNeededClusters = CalculateRequiredClustersOfDirectory(dirPath + L"/" + file.m_fileName) + ((dirEntry->entry.data.length % 2) == 0 ? 1 : 0);
if (newNeededClusters > GetAmountFreeDataClusters())
{
Console.Warning(GetCardFullMessage(file.m_fileName));
continue;
}
// is a subdirectory
fileInfo.AppendDir(fileInfo.GetFullName());
fileInfo.SetName(L"");
fileInfo.ClearExt();
// add entry for subdir in parent dir
MemoryCardFileEntry* newDirEntry = AppendFileEntryToDir(dirEntry);
dirEntry->entry.data.length++;
// set metadata
wxFileName metaFileName(dirPath, L"_pcsx2_meta_directory");
metaFileName.AppendDir(file.m_fileName);
wxFFile metaFile;
if (metaFileName.FileExists() && metaFile.Open(metaFileName.GetFullPath(), L"rb"))
{
size_t bytesRead = metaFile.Read(&newDirEntry->entry.raw, sizeof(newDirEntry->entry.raw));
metaFile.Close();
if (bytesRead < 0x60)
{
strcpy(reinterpret_cast<char*>(newDirEntry->entry.data.name), file.m_fileName.mbc_str());
}
}
else
{
newDirEntry->entry.data.mode = MemoryCardFileEntry::DefaultDirMode;
newDirEntry->entry.data.timeCreated = MemoryCardFileEntryDateTime::FromTime(file.m_timeCreated);
newDirEntry->entry.data.timeModified = MemoryCardFileEntryDateTime::FromTime(file.m_timeModified);
strcpy(reinterpret_cast<char*>(newDirEntry->entry.data.name), file.m_fileName.mbc_str());
}
// create new cluster for . and .. entries
newDirEntry->entry.data.length = 2;
u32 newCluster = GetFreeDataCluster();
m_fat.data[0][0][newCluster] = LastDataCluster | DataClusterInUseMask;
newDirEntry->entry.data.cluster = newCluster;
MemoryCardFileEntryCluster* const subDirCluster = &m_fileEntryDict[newCluster];
memset(subDirCluster->entries[0].entry.raw, 0x00, sizeof(subDirCluster->entries[0].entry.raw));
subDirCluster->entries[0].entry.data.mode = MemoryCardFileEntry::DefaultDirMode;
subDirCluster->entries[0].entry.data.dirEntry = entryNumber;
subDirCluster->entries[0].entry.data.name[0] = '.';
memset(subDirCluster->entries[1].entry.raw, 0x00, sizeof(subDirCluster->entries[1].entry.raw));
subDirCluster->entries[1].entry.data.mode = MemoryCardFileEntry::DefaultDirMode;
subDirCluster->entries[1].entry.data.name[0] = '.';
subDirCluster->entries[1].entry.data.name[1] = '.';
MemoryCardFileMetadataReference* dirRef = AddDirEntryToMetadataQuickAccess(newDirEntry, parent);
++entryNumber;
// and add all files in subdir
AddFolder(newDirEntry, fileInfo.GetFullPath(), dirRef);
}
}
return true;
}
return false;
}
bool FolderMemoryCard::AddFile(MemoryCardFileEntry* const dirEntry, const wxString& dirPath, const EnumeratedFileEntry& fileEntry, MemoryCardFileMetadataReference* parent)
{
wxFileName relativeFilePath(dirPath, fileEntry.m_fileName);
relativeFilePath.MakeRelativeTo(m_folderName.GetPath());
wxFileName fileInfo(dirPath, fileEntry.m_fileName);
wxFFile file(fileInfo.GetFullPath(), L"rb");
if (file.IsOpened())
{
// make sure we have enough space on the memcard to hold the data
const u32 clusterSize = m_superBlock.data.pages_per_cluster * m_superBlock.data.page_len;
const u32 filesize = file.Length();
const u32 countClusters = (filesize % clusterSize) != 0 ? (filesize / clusterSize + 1) : (filesize / clusterSize);
const u32 newNeededClusters = (dirEntry->entry.data.length % 2) == 0 ? countClusters + 1 : countClusters;
if (newNeededClusters > GetAmountFreeDataClusters())
{
Console.Warning(GetCardFullMessage(relativeFilePath.GetFullPath()));
return false;
}
MemoryCardFileEntry* newFileEntry = AppendFileEntryToDir(dirEntry);
// set file entry metadata
memset(newFileEntry->entry.raw, 0x00, sizeof(newFileEntry->entry.raw));
wxFileName metaFileName(dirPath, fileEntry.m_fileName);
metaFileName.AppendDir(L"_pcsx2_meta");
wxFFile metaFile;
if (metaFileName.FileExists() && metaFile.Open(metaFileName.GetFullPath(), L"rb"))
{
size_t bytesRead = metaFile.Read(&newFileEntry->entry.raw, sizeof(newFileEntry->entry.raw));
metaFile.Close();
if (bytesRead < 0x60)
{
strcpy(reinterpret_cast<char*>(newFileEntry->entry.data.name), fileEntry.m_fileName.mbc_str());
}
}
else
{
newFileEntry->entry.data.mode = MemoryCardFileEntry::DefaultFileMode;
newFileEntry->entry.data.timeCreated = MemoryCardFileEntryDateTime::FromTime(fileEntry.m_timeCreated);
newFileEntry->entry.data.timeModified = MemoryCardFileEntryDateTime::FromTime(fileEntry.m_timeModified);
strcpy(reinterpret_cast<char*>(newFileEntry->entry.data.name), fileEntry.m_fileName.mbc_str());
}
newFileEntry->entry.data.length = filesize;
if (filesize != 0)
{
u32 fileDataStartingCluster = GetFreeDataCluster();
newFileEntry->entry.data.cluster = fileDataStartingCluster;
// mark the appropriate amount of clusters as used
u32 dataCluster = fileDataStartingCluster;
m_fat.data[0][0][dataCluster] = LastDataCluster | DataClusterInUseMask;
for (unsigned int i = 0; i < countClusters - 1; ++i)
{
u32 newCluster = GetFreeDataCluster();
m_fat.data[0][0][dataCluster] = newCluster | DataClusterInUseMask;
m_fat.data[0][0][newCluster] = LastDataCluster | DataClusterInUseMask;
dataCluster = newCluster;
}
}
else
{
newFileEntry->entry.data.cluster = MemoryCardFileEntry::EmptyFileCluster;
}
file.Close();
MemoryCardFileMetadataReference* fileRef = AddFileEntryToMetadataQuickAccess(newFileEntry, parent);
if (fileRef != nullptr)
{
// acquire a handle on the file so nothing else can change the file contents while the memory card is open
m_lastAccessedFile.ReOpen(m_folderName, fileRef);
}
// and finally, increase file count in the directory entry
dirEntry->entry.data.length++;
return true;
}
else
{
Console.WriteLn(L"(FolderMcd) Could not open file: %s", WX_STR(relativeFilePath.GetFullPath()));
return false;
}
}
u32 FolderMemoryCard::CalculateRequiredClustersOfDirectory(const wxString& dirPath) const
{
const u32 clusterSize = m_superBlock.data.pages_per_cluster * m_superBlock.data.page_len;
u32 requiredFileEntryPages = 2;
u32 requiredClusters = 0;
// No need to read the index file as we are only counting space required; order of files is irrelevant.
wxDir dir(dirPath);
wxString fileName;
bool hasNext = dir.GetFirst(&fileName);
while (hasNext)
{
if (fileName.StartsWith(L"_pcsx2_"))
{
hasNext = dir.GetNext(&fileName);
continue;
}
++requiredFileEntryPages;
wxFileName file(dirPath, fileName);
wxString path = file.GetFullPath();
bool isFile = wxFile::Exists(path);
if (isFile)
{
const u32 filesize = file.GetSize().GetValue();
const u32 countClusters = (filesize % clusterSize) != 0 ? (filesize / clusterSize + 1) : (filesize / clusterSize);
requiredClusters += countClusters;
}
else
{
requiredClusters += CalculateRequiredClustersOfDirectory(path);
}
hasNext = dir.GetNext(&fileName);
}
return requiredClusters + requiredFileEntryPages / 2 + (requiredFileEntryPages % 2 == 0 ? 0 : 1);
}
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;
}
MemoryCardFileMetadataReference* FolderMemoryCard::AddFileEntryToMetadataQuickAccess(MemoryCardFileEntry* const entry, MemoryCardFileMetadataReference* const parent)
{
const u32 firstFileCluster = entry->entry.data.cluster;
u32 fileCluster = firstFileCluster;
// zero-length files have no file clusters
if (fileCluster == 0xFFFFFFFFu)
{
return nullptr;
}
u32 clusterNumber = 0;
do
{
MemoryCardFileMetadataReference* ref = &m_fileMetadataQuickAccess[fileCluster & NextDataClusterMask];
ref->parent = parent;
ref->entry = entry;
ref->consecutiveCluster = clusterNumber;
++clusterNumber;
} while ((fileCluster = m_fat.data[0][0][fileCluster & NextDataClusterMask]) != (LastDataCluster | DataClusterInUseMask));
return &m_fileMetadataQuickAccess[firstFileCluster & NextDataClusterMask];
}
s32 FolderMemoryCard::IsPresent() const
{
return m_isEnabled;
}
void FolderMemoryCard::GetSizeInfo(McdSizeInfo& outways) const
{
outways.SectorSize = PageSize;
outways.EraseBlockSizeInSectors = BlockSize / PageSize;
outways.McdSizeInSectors = GetSizeInClusters() * 2;
u8* pdata = (u8*)&outways.McdSizeInSectors;
outways.Xor = 18;
outways.Xor ^= pdata[0] ^ pdata[1] ^ pdata[2] ^ pdata[3];
}
bool FolderMemoryCard::IsPSX() const
{
return false;
}
u8* FolderMemoryCard::GetSystemBlockPointer(const u32 adr)
{
const u32 block = adr / BlockSizeRaw;
const u32 page = adr / PageSizeRaw;
const u32 offset = adr % PageSizeRaw;
const u32 cluster = adr / ClusterSizeRaw;
const u32 startDataCluster = m_superBlock.data.alloc_offset;
const u32 endDataCluster = startDataCluster + m_superBlock.data.alloc_end;
if (cluster >= startDataCluster && cluster < endDataCluster)
{
// trying to access a file entry?
const u32 fatCluster = cluster - m_superBlock.data.alloc_offset;
// if this cluster is unused according to FAT, we can assume we won't find anything
if ((m_fat.data[0][0][fatCluster] & DataClusterInUseMask) == 0)
{
return nullptr;
}
return GetFileEntryPointer(fatCluster, page % 2, offset);
}
if (block == 0)
{
return &m_superBlock.raw[page * PageSize + offset];
}
else if (block == m_superBlock.data.backup_block1)
{
return &m_backupBlock1[(page % 16) * PageSize + offset];
}
else if (block == m_superBlock.data.backup_block2)
{
return &m_backupBlock2.raw[(page % 16) * PageSize + offset];
}
else
{
// trying to access indirect FAT?
for (int i = 0; i < IndirectFatClusterCount; ++i)
{
if (cluster == m_superBlock.data.ifc_list[i])
{
return &m_indirectFat.raw[i][(page % 2) * PageSize + offset];
}
}
// trying to access FAT?
for (int i = 0; i < IndirectFatClusterCount; ++i)
{
for (int j = 0; j < ClusterSize / 4; ++j)
{
const u32 fatCluster = m_indirectFat.data[i][j];
if (fatCluster != IndirectFatUnused && fatCluster == cluster)
{
return &m_fat.raw[i][j][(page % 2) * PageSize + offset];
}
}
}
}
return nullptr;
}
u8* FolderMemoryCard::GetFileEntryPointer(const u32 searchCluster, const u32 entryNumber, const u32 offset)
{
const u32 fileCount = m_fileEntryDict[m_superBlock.data.rootdir_cluster].entries[0].entry.data.length;
MemoryCardFileEntryCluster* ptr = GetFileEntryCluster(m_superBlock.data.rootdir_cluster, searchCluster, fileCount);
if (ptr != nullptr)
{
return &ptr->entries[entryNumber].entry.raw[offset];
}
return nullptr;
}
MemoryCardFileEntryCluster* FolderMemoryCard::GetFileEntryCluster(const u32 currentCluster, const u32 searchCluster, const u32 fileCount)
{
// we found the correct cluster, return pointer to it
if (currentCluster == searchCluster)
{
return &m_fileEntryDict[currentCluster];
}
// check other clusters of this directory
const u32 nextCluster = m_fat.data[0][0][currentCluster] & NextDataClusterMask;
if (nextCluster != LastDataCluster)
{
MemoryCardFileEntryCluster* ptr = GetFileEntryCluster(nextCluster, searchCluster, fileCount - 2);
if (ptr != nullptr)
{
return ptr;
}
}
// check subdirectories
auto it = m_fileEntryDict.find(currentCluster);
if (it != m_fileEntryDict.end())
{
const u32 filesInThisCluster = std::min(fileCount, 2u);
for (unsigned int i = 0; i < filesInThisCluster; ++i)
{
const MemoryCardFileEntry* const entry = &it->second.entries[i];
if (entry->IsValid() && entry->IsUsed() && entry->IsDir() && !entry->IsDotDir())
{
const u32 newFileCount = entry->entry.data.length;
MemoryCardFileEntryCluster* ptr = GetFileEntryCluster(entry->entry.data.cluster, searchCluster, newFileCount);
if (ptr != nullptr)
{
return ptr;
}
}
}
}
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)
{
// 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)
{
MemoryCardFileEntry* const entry = &m_fileEntryDict[currentCluster].entries[i];
if (entry->IsValid() && entry->IsUsed() && entry->IsFile())
{
u32 fileCluster = entry->entry.data.cluster;
u32 clusterNumber = 0;
do
{
if (fileCluster == searchCluster)
{
fileName->SetName(wxString::FromAscii((const char*)entry->entry.data.name));
*outClusterNumber = clusterNumber;
return entry;
}
++clusterNumber;
} while ((fileCluster = m_fat.data[0][0][fileCluster] & NextDataClusterMask) != LastDataCluster);
}
}
// check other clusters of this directory
// this can probably be solved more efficiently by looping through nextClusters instead of recursively calling
const u32 nextCluster = m_fat.data[0][0][currentCluster] & NextDataClusterMask;
if (nextCluster != LastDataCluster)
{
MemoryCardFileEntry* ptr = GetFileEntryFromFileDataCluster(nextCluster, searchCluster, fileName, originalDirCount, outClusterNumber);
if (ptr != nullptr)
{
return ptr;
}
}
// check subdirectories
for (int i = 0; i < 2; ++i)
{
MemoryCardFileEntry* const entry = &m_fileEntryDict[currentCluster].entries[i];
if (entry->IsValid() && entry->IsUsed() && entry->IsDir() && !entry->IsDotDir())
{
MemoryCardFileEntry* ptr = GetFileEntryFromFileDataCluster(entry->entry.data.cluster, searchCluster, fileName, originalDirCount, outClusterNumber);
if (ptr != nullptr)
{
fileName->InsertDir(originalDirCount, wxString::FromAscii((const char*)entry->entry.data.name));
return ptr;
}
}
}
return nullptr;
}
bool FolderMemoryCard::ReadFromFile(u8* dest, u32 adr, u32 dataLength)
{
const u32 page = adr / PageSizeRaw;
const u32 offset = adr % PageSizeRaw;
const u32 cluster = adr / ClusterSizeRaw;
const u32 fatCluster = cluster - m_superBlock.data.alloc_offset;
// if the cluster is unused according to FAT, just return
if ((m_fat.data[0][0][fatCluster] & DataClusterInUseMask) == 0)
{
return false;
}
// figure out which file to read from
auto it = m_fileMetadataQuickAccess.find(fatCluster);
if (it != m_fileMetadataQuickAccess.end())
{
const u32 clusterNumber = it->second.consecutiveCluster;
wxFFile* file = m_lastAccessedFile.ReOpen(m_folderName, &it->second);
if (file->IsOpened())
{
const u32 clusterOffset = (page % 2) * PageSize + offset;
const u32 fileOffset = clusterNumber * ClusterSize + clusterOffset;
if (fileOffset != file->Tell())
{
file->Seek(fileOffset);
}
size_t bytesRead = file->Read(dest, dataLength);
// if more bytes were requested than actually exist, fill the rest with 0xFF
if (bytesRead < dataLength)
{
memset(&dest[bytesRead], 0xFF, dataLength - bytesRead);
}
return bytesRead > 0;
}
}
return false;
}
s32 FolderMemoryCard::Read(u8* dest, u32 adr, int size)
{
//const u32 block = adr / BlockSizeRaw;
const u32 page = adr / PageSizeRaw;
const u32 offset = adr % PageSizeRaw;
//const u32 cluster = adr / ClusterSizeRaw;
const u32 end = offset + size;
if (end > PageSizeRaw)
{
// is trying to read more than one page at a time
// do this recursively so that each function call only has to care about one page
const u32 toNextPage = PageSizeRaw - offset;
Read(dest + toNextPage, adr + toNextPage, size - toNextPage);
size = toNextPage;
}
if (offset < PageSize)
{
// is trying to read (part of) an actual data block
const u32 dataLength = std::min((u32)size, (u32)(PageSize - offset));
// if we have a cache for this page, just load from that
auto it = m_cache.find(page);
if (it != m_cache.end())
{
memcpy(dest, &it->second.raw[offset], dataLength);
}
else
{
ReadDataWithoutCache(dest, adr, dataLength);
}
}
if (end > PageSize)
{
// is trying to (partially) read the ECC
const u32 eccOffset = PageSize - offset;
const u32 eccLength = std::min((u32)(size - offset), (u32)EccSize);
const u32 adrStart = page * PageSizeRaw;
u8 data[PageSize];
Read(data, adrStart, PageSize);
u8 ecc[EccSize];
memset(ecc, 0xFF, EccSize);
for (int i = 0; i < PageSize / 0x80; ++i)
{
FolderMemoryCard::CalculateECC(ecc + (i * 3), &data[i * 0x80]);
}
memcpy(dest + eccOffset, ecc, eccLength);
}
SetTimeLastReadToNow();
// return 0 on fail, 1 on success?
return 1;
}
void FolderMemoryCard::ReadDataWithoutCache(u8* const dest, const u32 adr, const u32 dataLength)
{
u8* src = GetSystemBlockPointer(adr);
if (src != nullptr)
{
memcpy(dest, src, dataLength);
}
else
{
if (!ReadFromFile(dest, adr, dataLength))
{
memset(dest, 0xFF, dataLength);
}
}
}
s32 FolderMemoryCard::Save(const u8* src, u32 adr, int size)
{
//const u32 block = adr / BlockSizeRaw;
//const u32 cluster = adr / ClusterSizeRaw;
const u32 page = adr / PageSizeRaw;
const u32 offset = adr % PageSizeRaw;
const u32 end = offset + size;
if (end > PageSizeRaw)
{
// is trying to store more than one page at a time
// do this recursively so that each function call only has to care about one page
const u32 toNextPage = PageSizeRaw - offset;
Save(src + toNextPage, adr + toNextPage, size - toNextPage);
size = toNextPage;
}
if (offset < PageSize)
{
// is trying to store (part of) an actual data block
const u32 dataLength = std::min((u32)size, PageSize - offset);
// if cache page has not yet been touched, fill it with the data from our memory card
auto it = m_cache.find(page);
MemoryCardPage* cachePage;
if (it == m_cache.end())
{
cachePage = &m_cache[page];
const u32 adrLoad = page * PageSizeRaw;
ReadDataWithoutCache(&cachePage->raw[0], adrLoad, PageSize);
memcpy(&m_oldDataCache[page].raw[0], &cachePage->raw[0], PageSize);
}
else
{
cachePage = &it->second;
}
// then just write to the cache
memcpy(&cachePage->raw[offset], src, dataLength);
SetTimeLastWrittenToNow();
}
return 1;
}
void FolderMemoryCard::NextFrame()
{
if (m_framesUntilFlush > 0 && --m_framesUntilFlush == 0)
{
Flush();
}
}
void FolderMemoryCard::Flush()
{
if (m_cache.empty())
{
return;
}
#ifdef DEBUG_WRITE_FOLDER_CARD_IN_MEMORY_TO_FILE_ON_CHANGE
WriteToFile(m_folderName.GetFullPath().RemoveLast() + L"-debug_" + wxDateTime::Now().Format(L"%Y-%m-%d-%H-%M-%S") + L"_pre-flush.ps2");
#endif
Console.WriteLn(L"(FolderMcd) Writing data for slot %u to file system...", m_slot);
const u64 timeFlushStart = wxGetLocalTimeMillis().GetValue();
// Keep a copy of the old file entries so we can figure out which files and directories, if any, have been deleted from the memory card.
std::vector<MemoryCardFileEntryTreeNode> oldFileEntryTree;
if (IsFormatted())
{
CopyEntryDictIntoTree(&oldFileEntryTree, m_superBlock.data.rootdir_cluster, m_fileEntryDict[m_superBlock.data.rootdir_cluster].entries[0].entry.data.length);
}
// first write the superblock if necessary
FlushSuperBlock();
if (!IsFormatted())
{
return;
}
// check if we were interrupted in the middle of a save operation, if yes abort
FlushBlock(m_superBlock.data.backup_block1);
FlushBlock(m_superBlock.data.backup_block2);
if (m_backupBlock2.programmedBlock != 0xFFFFFFFFu)
{
Console.Warning(L"(FolderMcd) Aborting flush of slot %u, emulation was interrupted during save process!", m_slot);
return;
}
const u32 clusterCount = GetSizeInClusters();
const u32 pageCount = clusterCount * 2;
// then write the indirect FAT
for (int i = 0; i < IndirectFatClusterCount; ++i)
{
const u32 cluster = m_superBlock.data.ifc_list[i];
if (cluster > 0 && cluster < clusterCount)
{
FlushCluster(cluster);
}
}
// and the FAT
for (int i = 0; i < IndirectFatClusterCount; ++i)
{
for (int j = 0; j < ClusterSize / 4; ++j)
{
const u32 cluster = m_indirectFat.data[i][j];
if (cluster > 0 && cluster < clusterCount)
{
FlushCluster(cluster);
}
}
}
// then all directory and file entries
FlushFileEntries();
// Now we have the new file system, compare it to the old one and "delete" any files that were in it before but aren't anymore.
FlushDeletedFilesAndRemoveUnchangedDataFromCache(oldFileEntryTree);
// and finally, flush everything that hasn't been flushed yet
for (uint i = 0; i < pageCount; ++i)
{
FlushPage(i);
}
m_lastAccessedFile.FlushAll();
m_lastAccessedFile.ClearMetadataWriteState();
m_oldDataCache.clear();
const u64 timeFlushEnd = wxGetLocalTimeMillis().GetValue();
Console.WriteLn(L"(FolderMcd) Done! Took %u ms.", timeFlushEnd - timeFlushStart);
#ifdef DEBUG_WRITE_FOLDER_CARD_IN_MEMORY_TO_FILE_ON_CHANGE
WriteToFile(m_folderName.GetFullPath().RemoveLast() + L"-debug_" + wxDateTime::Now().Format(L"%Y-%m-%d-%H-%M-%S") + L"_post-flush.ps2");
#endif
}
bool FolderMemoryCard::FlushPage(const u32 page)
{
auto it = m_cache.find(page);
if (it != m_cache.end())
{
WriteWithoutCache(&it->second.raw[0], page * PageSizeRaw, PageSize);
m_cache.erase(it);
return true;
}
return false;
}
bool FolderMemoryCard::FlushCluster(const u32 cluster)
{
const u32 page = cluster * 2;
bool flushed = false;
if (FlushPage(page))
{
flushed = true;
}
if (FlushPage(page + 1))
{
flushed = true;
}
return flushed;
}
bool FolderMemoryCard::FlushBlock(const u32 block)
{
const u32 page = block * 16;
bool flushed = false;
for (int i = 0; i < 16; ++i)
{
if (FlushPage(page + i))
{
flushed = true;
}
}
return flushed;
}
void FolderMemoryCard::FlushSuperBlock()
{
if (FlushBlock(0) && m_performFileWrites)
{
wxFileName superBlockFileName(m_folderName.GetPath(), L"_pcsx2_superblock");
wxFFile superBlockFile(superBlockFileName.GetFullPath().c_str(), L"wb");
if (superBlockFile.IsOpened())
{
superBlockFile.Write(&m_superBlock.raw, sizeof(m_superBlock.raw));
}
}
}
void FolderMemoryCard::FlushFileEntries()
{
// Flush all file entry data from the cache into m_fileEntryDict.
const u32 rootDirCluster = m_superBlock.data.rootdir_cluster;
FlushCluster(rootDirCluster + m_superBlock.data.alloc_offset);
MemoryCardFileEntryCluster* rootEntries = &m_fileEntryDict[rootDirCluster];
if (rootEntries->entries[0].IsValid() && rootEntries->entries[0].IsUsed())
{
FlushFileEntries(rootDirCluster, rootEntries->entries[0].entry.data.length);
}
}
void FolderMemoryCard::FlushFileEntries(const u32 dirCluster, const u32 remainingFiles, const wxString& dirPath, MemoryCardFileMetadataReference* parent)
{
// flush the current cluster
FlushCluster(dirCluster + m_superBlock.data.alloc_offset);
// if either of the current entries is a subdir, flush that too
MemoryCardFileEntryCluster* entries = &m_fileEntryDict[dirCluster];
const u32 filesInThisCluster = std::min(remainingFiles, 2u);
for (unsigned int i = 0; i < filesInThisCluster; ++i)
{
MemoryCardFileEntry* entry = &entries->entries[i];
if (entry->IsValid() && entry->IsUsed())
{
if (entry->IsDir())
{
if (!entry->IsDotDir())
{
char cleanName[sizeof(entry->entry.data.name)];
memcpy(cleanName, (const char*)entry->entry.data.name, sizeof(cleanName));
bool filenameCleaned = FileAccessHelper::CleanMemcardFilename(cleanName);
const wxString subDirName = wxString::FromAscii((const char*)cleanName);
const wxString subDirPath = dirPath + L"/" + subDirName;
if (m_performFileWrites)
{
// if this directory has nonstandard metadata, write that to the file system
wxFileName metaFileName(m_folderName.GetFullPath() + subDirPath, L"_pcsx2_meta_directory");
if (!metaFileName.DirExists())
{
metaFileName.Mkdir();
}
if (filenameCleaned || entry->entry.data.mode != MemoryCardFileEntry::DefaultDirMode || entry->entry.data.attr != 0)
{
wxFFile metaFile(metaFileName.GetFullPath(), L"wb");
if (metaFile.IsOpened())
{
metaFile.Write(entry->entry.raw, sizeof(entry->entry.raw));
}
}
else
{
// if metadata is standard make sure to remove a possibly existing metadata file
if (metaFileName.FileExists())
{
wxRemoveFile(metaFileName.GetFullPath());
}
}
// write the directory index
metaFileName.SetName(L"_pcsx2_index");
YAML::Node index = LoadYAMLFromFile(metaFileName.GetFullPath());
YAML::Node entryNode = index["%ROOT"];
entryNode["timeCreated"] = entry->entry.data.timeCreated.ToTime();
entryNode["timeModified"] = entry->entry.data.timeModified.ToTime();
// Write out the changes
wxFFile indexFile;
if (indexFile.Open(metaFileName.GetFullPath(), L"w"))
{
indexFile.Write(YAML::Dump(index));
}
}
MemoryCardFileMetadataReference* dirRef = AddDirEntryToMetadataQuickAccess(entry, parent);
FlushFileEntries(entry->entry.data.cluster, entry->entry.data.length, subDirPath, dirRef);
}
}
else if (entry->IsFile())
{
AddFileEntryToMetadataQuickAccess(entry, parent);
if (entry->entry.data.length == 0)
{
// empty files need to be explicitly created, as there will be no data cluster referencing it later
if (m_performFileWrites)
{
char cleanName[sizeof(entry->entry.data.name)];
memcpy(cleanName, (const char*)entry->entry.data.name, sizeof(cleanName));
FileAccessHelper::CleanMemcardFilename(cleanName);
const wxString filePath = dirPath + L"/" + wxString::FromAscii((const char*)cleanName);
wxFileName fn(m_folderName.GetFullPath() + filePath);
if (!fn.FileExists())
{
if (!fn.DirExists())
{
fn.Mkdir(0777, wxPATH_MKDIR_FULL);
}
wxFFile createEmptyFile(fn.GetFullPath(), L"wb");
createEmptyFile.Close();
}
}
}
if (m_performFileWrites)
{
FileAccessHelper::WriteIndex(m_folderName.GetFullPath() + dirPath, entry, parent);
}
}
}
}
// continue to the next cluster of this directory
const u32 nextCluster = m_fat.data[0][0][dirCluster];
if (nextCluster != (LastDataCluster | DataClusterInUseMask))
{
FlushFileEntries(nextCluster & NextDataClusterMask, remainingFiles - 2, dirPath, parent);
}
}
void FolderMemoryCard::FlushDeletedFilesAndRemoveUnchangedDataFromCache(const std::vector<MemoryCardFileEntryTreeNode>& oldFileEntries)
{
const u32 newRootDirCluster = m_superBlock.data.rootdir_cluster;
const u32 newFileCount = m_fileEntryDict[newRootDirCluster].entries[0].entry.data.length;
FlushDeletedFilesAndRemoveUnchangedDataFromCache(oldFileEntries, newRootDirCluster, newFileCount, "");
}
void FolderMemoryCard::FlushDeletedFilesAndRemoveUnchangedDataFromCache(const std::vector<MemoryCardFileEntryTreeNode>& oldFileEntries, const u32 newCluster, const u32 newFileCount, const wxString& dirPath)
{
// go through all file entires of the current directory of the old data
for (auto it = oldFileEntries.cbegin(); it != oldFileEntries.cend(); ++it)
{
const MemoryCardFileEntry* entry = &it->entry;
if (entry->IsValid() && entry->IsUsed() && !entry->IsDotDir())
{
// check if an equivalent entry exists in m_fileEntryDict
const MemoryCardFileEntry* newEntry = FindEquivalent(entry, newCluster, newFileCount);
if (newEntry == nullptr)
{
// file/dir doesn't exist anymore, remove!
char cleanName[sizeof(entry->entry.data.name)];
memcpy(cleanName, (const char*)entry->entry.data.name, sizeof(cleanName));
FileAccessHelper::CleanMemcardFilename(cleanName);
const wxString fileName = wxString::FromAscii(cleanName);
const wxString filePath = m_folderName.GetFullPath() + dirPath + L"/" + fileName;
m_lastAccessedFile.CloseMatching(filePath);
const wxString newFilePath = m_folderName.GetFullPath() + dirPath + L"/_pcsx2_deleted_" + fileName;
if (wxFileName::DirExists(newFilePath))
{
// wxRenameFile doesn't overwrite directories, so we have to remove the old one first
RemoveDirectory(newFilePath);
}
wxRenameFile(filePath, newFilePath);
DeleteFromIndex(m_folderName.GetFullPath() + dirPath, fileName);
}
else if (entry->IsDir())
{
// still exists and is a directory, recursive call for subdir
char cleanName[sizeof(entry->entry.data.name)];
memcpy(cleanName, (const char*)entry->entry.data.name, sizeof(cleanName));
FileAccessHelper::CleanMemcardFilename(cleanName);
const wxString subDirName = wxString::FromAscii(cleanName);
const wxString subDirPath = dirPath + L"/" + subDirName;
FlushDeletedFilesAndRemoveUnchangedDataFromCache(it->subdir, newEntry->entry.data.cluster, newEntry->entry.data.length, subDirPath);
}
else if (entry->IsFile())
{
// still exists and is a file, see if we can remove unchanged data from m_cache
RemoveUnchangedDataFromCache(entry, newEntry);
}
}
}
}
void FolderMemoryCard::RemoveUnchangedDataFromCache(const MemoryCardFileEntry* const oldEntry, const MemoryCardFileEntry* const newEntry)
{
// Disclaimer: Technically, to actually prove that file data has not changed and still belongs to the same file, we'd need to keep a copy
// of the old FAT cluster chain and compare that as well, and only acknowledge the file as unchanged if none of those have changed. However,
// the chain of events that leads to a file having the exact same file contents as a deleted old file while also being placed in the same
// data clusters as the deleted file AND matching this condition here, in a quick enough succession that no flush has occurred yet since the
// deletion of that old file is incredibly unlikely, so I'm not sure if it's actually worth coding for.
if (oldEntry->entry.data.timeModified != newEntry->entry.data.timeModified || oldEntry->entry.data.timeCreated != newEntry->entry.data.timeCreated || oldEntry->entry.data.length != newEntry->entry.data.length || oldEntry->entry.data.cluster != newEntry->entry.data.cluster)
{
return;
}
u32 cluster = newEntry->entry.data.cluster & NextDataClusterMask;
const u32 alloc_offset = m_superBlock.data.alloc_offset;
while (cluster != LastDataCluster)
{
for (int i = 0; i < 2; ++i)
{
const u32 page = (cluster + alloc_offset) * 2 + i;
auto newIt = m_cache.find(page);
if (newIt == m_cache.end())
{
continue;
}
auto oldIt = m_oldDataCache.find(page);
if (oldIt == m_oldDataCache.end())
{
continue;
}
if (memcmp(&oldIt->second.raw[0], &newIt->second.raw[0], PageSize) == 0)
{
m_cache.erase(newIt);
}
}
cluster = m_fat.data[0][0][cluster] & NextDataClusterMask;
}
}
s32 FolderMemoryCard::WriteWithoutCache(const u8* src, u32 adr, int size)
{
//const u32 block = adr / BlockSizeRaw;
//const u32 cluster = adr / ClusterSizeRaw;
//const u32 page = adr / PageSizeRaw;
const u32 offset = adr % PageSizeRaw;
const u32 end = offset + size;
if (end > PageSizeRaw)
{
// is trying to store more than one page at a time
// do this recursively so that each function call only has to care about one page
const u32 toNextPage = PageSizeRaw - offset;
Save(src + toNextPage, adr + toNextPage, size - toNextPage);
size = toNextPage;
}
if (offset < PageSize)
{
// is trying to store (part of) an actual data block
const u32 dataLength = std::min((u32)size, PageSize - offset);
u8* dest = GetSystemBlockPointer(adr);
if (dest != nullptr)
{
memcpy(dest, src, dataLength);
}
else
{
WriteToFile(src, adr, dataLength);
}
}
if (end > PageSize)
{
// is trying to store ECC
// simply ignore this, is automatically generated when reading
}
// return 0 on fail, 1 on success?
return 1;
}
bool FolderMemoryCard::WriteToFile(const u8* src, u32 adr, u32 dataLength)
{
const u32 cluster = adr / ClusterSizeRaw;
const u32 page = adr / PageSizeRaw;
const u32 offset = adr % PageSizeRaw;
const u32 fatCluster = cluster - m_superBlock.data.alloc_offset;
// if the cluster is unused according to FAT, just skip all this, we're not gonna find anything anyway
if ((m_fat.data[0][0][fatCluster] & DataClusterInUseMask) == 0)
{
return false;
}
// figure out which file to write to
auto it = m_fileMetadataQuickAccess.find(fatCluster);
if (it != m_fileMetadataQuickAccess.end())
{
const MemoryCardFileEntry* const entry = it->second.entry;
const u32 clusterNumber = it->second.consecutiveCluster;
if (m_performFileWrites)
{
wxFFile* file = m_lastAccessedFile.ReOpen(m_folderName, &it->second, true);
if (file->IsOpened())
{
const u32 clusterOffset = (page % 2) * PageSize + offset;
const u32 fileSize = entry->entry.data.length;
const u32 fileOffsetStart = std::min(clusterNumber * ClusterSize + clusterOffset, fileSize);
const u32 fileOffsetEnd = std::min(fileOffsetStart + dataLength, fileSize);
const u32 bytesToWrite = fileOffsetEnd - fileOffsetStart;
wxFileOffset actualFileSize = file->Length();
if (actualFileSize < fileOffsetStart)
{
file->Seek(actualFileSize);
const u32 diff = fileOffsetStart - actualFileSize;
u8 temp = 0xFF;
for (u32 i = 0; i < diff; ++i)
{
file->Write(&temp, 1);
}
}
const wxFileOffset fileOffset = file->Tell();
if (fileOffset != fileOffsetStart)
{
file->Seek(fileOffsetStart);
}
if (bytesToWrite > 0)
{
file->Write(src, bytesToWrite);
}
}
else
{
return false;
}
}
return true;
}
return false;
}
void FolderMemoryCard::CopyEntryDictIntoTree(std::vector<MemoryCardFileEntryTreeNode>* fileEntryTree, const u32 cluster, const u32 fileCount)
{
const MemoryCardFileEntryCluster* entryCluster = &m_fileEntryDict[cluster];
u32 fileCluster = cluster;
for (size_t i = 0; i < fileCount; ++i)
{
const MemoryCardFileEntry* entry = &entryCluster->entries[i % 2];
if (entry->IsValid() && entry->IsUsed())
{
fileEntryTree->emplace_back(*entry);
if (entry->IsDir() && !entry->IsDotDir())
{
MemoryCardFileEntryTreeNode* treeEntry = &fileEntryTree->back();
CopyEntryDictIntoTree(&treeEntry->subdir, entry->entry.data.cluster, entry->entry.data.length);
}
}
if (i % 2 == 1)
{
fileCluster = m_fat.data[0][0][fileCluster] & 0x7FFFFFFFu;
if (fileCluster == 0x7FFFFFFFu)
{
return;
}
entryCluster = &m_fileEntryDict[fileCluster];
}
}
}
const MemoryCardFileEntry* FolderMemoryCard::FindEquivalent(const MemoryCardFileEntry* searchEntry, const u32 cluster, const u32 fileCount)
{
const MemoryCardFileEntryCluster* entryCluster = &m_fileEntryDict[cluster];
u32 fileCluster = cluster;
for (size_t i = 0; i < fileCount; ++i)
{
const MemoryCardFileEntry* entry = &entryCluster->entries[i % 2];
if (entry->IsValid() && entry->IsUsed())
{
if (entry->IsFile() == searchEntry->IsFile() && entry->IsDir() == searchEntry->IsDir() && strncmp((const char*)searchEntry->entry.data.name, (const char*)entry->entry.data.name, sizeof(entry->entry.data.name)) == 0)
{
return entry;
}
}
if (i % 2 == 1)
{
fileCluster = m_fat.data[0][0][fileCluster] & 0x7FFFFFFFu;
if (fileCluster == 0x7FFFFFFFu)
{
return nullptr;
}
entryCluster = &m_fileEntryDict[fileCluster];
}
}
return nullptr;
}
s32 FolderMemoryCard::EraseBlock(u32 adr)
{
const u32 block = adr / BlockSizeRaw;
u8 eraseData[PageSize];
memset(eraseData, 0xFF, PageSize);
for (int page = 0; page < 16; ++page)
{
const u32 adr = block * BlockSizeRaw + page * PageSizeRaw;
Save(eraseData, adr, PageSize);
}
// return 0 on fail, 1 on success?
return 1;
}
u64 FolderMemoryCard::GetCRC() const
{
// Since this is just used as integrity check for savestate loading,
// give a timestamp of the last time the memory card was written to
return m_timeLastWritten;
}
void FolderMemoryCard::SetSlot(uint slot)
{
pxAssert(slot < 8);
m_slot = slot;
}
u32 FolderMemoryCard::GetSizeInClusters() const
{
const u32 clusters = m_superBlock.data.clusters_per_card;
if (clusters > 0 && clusters < 0xFFFFFFFFu)
{
return clusters;
}
else
{
return TotalClusters;
}
}
void FolderMemoryCard::SetSizeInClusters(u32 clusters)
{
superBlockUnion newSuperBlock;
memcpy(&newSuperBlock.raw[0], &m_superBlock.raw[0], sizeof(newSuperBlock.raw));
newSuperBlock.data.clusters_per_card = clusters;
const u32 alloc_offset = clusters / 0x100 + 9;
newSuperBlock.data.alloc_offset = alloc_offset;
newSuperBlock.data.alloc_end = clusters - 0x10 - alloc_offset;
const u32 blocks = clusters / 8;
newSuperBlock.data.backup_block1 = blocks - 1;
newSuperBlock.data.backup_block2 = blocks - 2;
for (size_t i = 0; i < sizeof(newSuperBlock.raw) / PageSize; ++i)
{
Save(&newSuperBlock.raw[i * PageSize], i * PageSizeRaw, PageSize);
}
}
void FolderMemoryCard::SetSizeInMB(u32 megaBytes)
{
SetSizeInClusters((megaBytes * 1024 * 1024) / ClusterSize);
}
void FolderMemoryCard::SetTimeLastReadToNow()
{
m_framesUntilFlush = FramesAfterWriteUntilFlush;
}
void FolderMemoryCard::SetTimeLastWrittenToNow()
{
m_timeLastWritten = wxGetLocalTimeMillis().GetValue();
m_framesUntilFlush = FramesAfterWriteUntilFlush;
}
std::vector<FolderMemoryCard::EnumeratedFileEntry> FolderMemoryCard::GetOrderedFiles(const wxString& dirPath) const
{
std::vector<EnumeratedFileEntry> result;
wxDir dir(dirPath);
if (dir.IsOpened())
{
const YAML::Node index = LoadYAMLFromFile(wxFileName(dirPath, "_pcsx2_index").GetFullPath());
// We must be able to support legacy folder memcards without the index file, so for those
// track an order variable and make it negative - this way new files get their order preserved
// and old files are listed first.
// In the YAML File order is stored as an unsigned int, so use a signed int64_t to accommodate for
// all possible values without cutting them off
// Also exploit the fact pairs sort lexicographically to ensure directories are listed first
// (since they don't carry their own order in the index file)
std::map<std::pair<bool, int64_t>, EnumeratedFileEntry> sortContainer;
int64_t orderForDirectories = 1;
int64_t orderForLegacyFiles = -1;
const auto getOptionalNodeAttribute = [](const YAML::Node& node, const char* attribName, auto def) {
auto result = std::move(def);
if (node.IsDefined())
{
result = node[attribName].as<decltype(def)>(def);
}
return result;
};
wxString fileName;
bool hasNext = dir.GetFirst(&fileName);
while (hasNext)
{
if (fileName.StartsWith(L"_pcsx2_"))
{
hasNext = dir.GetNext(&fileName);
continue;
}
wxFileName fileInfo(dirPath, fileName);
if (wxFile::Exists(fileInfo.GetFullPath()))
{
wxDateTime creationTime, modificationTime;
fileInfo.GetTimes(nullptr, &modificationTime, &creationTime);
const wxCharTypeBuffer fileNameUTF8(fileName.ToUTF8());
const YAML::Node& node = index[fileNameUTF8.data()];
// orderForLegacyFiles will decrement even if it ends up being unused, but that's fine
auto key = std::make_pair(true, getOptionalNodeAttribute(node, "order", orderForLegacyFiles--));
EnumeratedFileEntry entry{fileName, getOptionalNodeAttribute(node, "timeCreated", creationTime.GetTicks()),
getOptionalNodeAttribute(node, "timeModified", modificationTime.GetTicks()), true};
sortContainer.try_emplace(std::move(key), std::move(entry));
}
else
{
fileInfo.AppendDir(fileInfo.GetFullName());
fileInfo.SetName(L"");
wxDateTime creationTime, modificationTime;
fileInfo.GetTimes(nullptr, &modificationTime, &creationTime);
const YAML::Node indexForDirectory = LoadYAMLFromFile(wxFileName(fileInfo.GetFullPath(), "_pcsx2_index").GetFullPath());
const YAML::Node& node = indexForDirectory["%ROOT"];
// orderForDirectories will increment even if it ends up being unused, but that's fine
auto key = std::make_pair(false, orderForDirectories++);
EnumeratedFileEntry entry{fileName, getOptionalNodeAttribute(node, "timeCreated", creationTime.GetTicks()),
getOptionalNodeAttribute(node, "timeModified", modificationTime.GetTicks()), false};
sortContainer.try_emplace(std::move(key), std::move(entry));
}
hasNext = dir.GetNext(&fileName);
}
// Move items from the intermediate map to a final vector
result.reserve(sortContainer.size());
for (auto& e : sortContainer)
{
result.push_back(std::move(e.second));
}
}
return result;
}
void FolderMemoryCard::DeleteFromIndex(const wxString& filePath, const wxString& entry) const
{
const wxString indexName = wxFileName(filePath, "_pcsx2_index").GetFullPath();
YAML::Node index = LoadYAMLFromFile(indexName);
const wxCharTypeBuffer entryUTF8(entry.ToUTF8());
index.remove(entryUTF8.data());
// Write out the changes
wxFFile indexFile;
if (indexFile.Open(indexName, L"w"))
{
indexFile.Write(YAML::Dump(index));
}
}
// from http://www.oocities.org/siliconvalley/station/8269/sma02/sma02.html#ECC
void FolderMemoryCard::CalculateECC(u8* ecc, const u8* data)
{
static const u8 Table[] = {
0x00, 0x87, 0x96, 0x11, 0xa5, 0x22, 0x33, 0xb4, 0xb4, 0x33, 0x22, 0xa5, 0x11, 0x96, 0x87, 0x00,
0xc3, 0x44, 0x55, 0xd2, 0x66, 0xe1, 0xf0, 0x77, 0x77, 0xf0, 0xe1, 0x66, 0xd2, 0x55, 0x44, 0xc3,
0xd2, 0x55, 0x44, 0xc3, 0x77, 0xf0, 0xe1, 0x66, 0x66, 0xe1, 0xf0, 0x77, 0xc3, 0x44, 0x55, 0xd2,
0x11, 0x96, 0x87, 0x00, 0xb4, 0x33, 0x22, 0xa5, 0xa5, 0x22, 0x33, 0xb4, 0x00, 0x87, 0x96, 0x11,
0xe1, 0x66, 0x77, 0xf0, 0x44, 0xc3, 0xd2, 0x55, 0x55, 0xd2, 0xc3, 0x44, 0xf0, 0x77, 0x66, 0xe1,
0x22, 0xa5, 0xb4, 0x33, 0x87, 0x00, 0x11, 0x96, 0x96, 0x11, 0x00, 0x87, 0x33, 0xb4, 0xa5, 0x22,
0x33, 0xb4, 0xa5, 0x22, 0x96, 0x11, 0x00, 0x87, 0x87, 0x00, 0x11, 0x96, 0x22, 0xa5, 0xb4, 0x33,
0xf0, 0x77, 0x66, 0xe1, 0x55, 0xd2, 0xc3, 0x44, 0x44, 0xc3, 0xd2, 0x55, 0xe1, 0x66, 0x77, 0xf0,
0xf0, 0x77, 0x66, 0xe1, 0x55, 0xd2, 0xc3, 0x44, 0x44, 0xc3, 0xd2, 0x55, 0xe1, 0x66, 0x77, 0xf0,
0x33, 0xb4, 0xa5, 0x22, 0x96, 0x11, 0x00, 0x87, 0x87, 0x00, 0x11, 0x96, 0x22, 0xa5, 0xb4, 0x33,
0x22, 0xa5, 0xb4, 0x33, 0x87, 0x00, 0x11, 0x96, 0x96, 0x11, 0x00, 0x87, 0x33, 0xb4, 0xa5, 0x22,
0xe1, 0x66, 0x77, 0xf0, 0x44, 0xc3, 0xd2, 0x55, 0x55, 0xd2, 0xc3, 0x44, 0xf0, 0x77, 0x66, 0xe1,
0x11, 0x96, 0x87, 0x00, 0xb4, 0x33, 0x22, 0xa5, 0xa5, 0x22, 0x33, 0xb4, 0x00, 0x87, 0x96, 0x11,
0xd2, 0x55, 0x44, 0xc3, 0x77, 0xf0, 0xe1, 0x66, 0x66, 0xe1, 0xf0, 0x77, 0xc3, 0x44, 0x55, 0xd2,
0xc3, 0x44, 0x55, 0xd2, 0x66, 0xe1, 0xf0, 0x77, 0x77, 0xf0, 0xe1, 0x66, 0xd2, 0x55, 0x44, 0xc3,
0x00, 0x87, 0x96, 0x11, 0xa5, 0x22, 0x33, 0xb4, 0xb4, 0x33, 0x22, 0xa5, 0x11, 0x96, 0x87, 0x00};
int i, c;
ecc[0] = ecc[1] = ecc[2] = 0;
for (i = 0; i < 0x80; i++)
{
c = Table[data[i]];
ecc[0] ^= c;
if (c & 0x80)
{
ecc[1] ^= ~i;
ecc[2] ^= i;
}
}
ecc[0] = ~ecc[0];
ecc[0] &= 0x77;
ecc[1] = ~ecc[1];
ecc[1] &= 0x7f;
ecc[2] = ~ecc[2];
ecc[2] &= 0x7f;
return;
}
void FolderMemoryCard::WriteToFile(const wxString& filename)
{
wxFFile targetFile(filename, L"wb");
u8 buffer[FolderMemoryCard::PageSizeRaw];
u32 adr = 0;
while (adr < GetSizeInClusters() * FolderMemoryCard::ClusterSizeRaw)
{
Read(buffer, adr, FolderMemoryCard::PageSizeRaw);
targetFile.Write(buffer, FolderMemoryCard::PageSizeRaw);
adr += FolderMemoryCard::PageSizeRaw;
}
}
FileAccessHelper::FileAccessHelper()
{
}
FileAccessHelper::~FileAccessHelper()
{
this->CloseAll();
}
wxFFile* FileAccessHelper::Open(const wxFileName& folderName, MemoryCardFileMetadataReference* fileRef, bool writeMetadata)
{
wxFileName fn(folderName);
fileRef->GetPath(&fn);
wxString filename(fn.GetFullPath());
if (!fn.FileExists())
{
if (!fn.DirExists())
{
fn.Mkdir(0777, wxPATH_MKDIR_FULL);
}
wxFFile createEmptyFile(filename, L"wb");
createEmptyFile.Close();
}
wxFFile* file = new wxFFile(filename, L"r+b");
std::string internalPath;
fileRef->GetInternalPath(&internalPath);
MemoryCardFileHandleStructure handleStruct;
handleStruct.fileHandle = file;
handleStruct.fileRef = fileRef;
m_files.emplace(std::move(internalPath), std::move(handleStruct));
if (writeMetadata)
{
WriteMetadata(folderName, fileRef);
}
return file;
}
void FileAccessHelper::WriteMetadata(wxFileName folderName, const MemoryCardFileMetadataReference* fileRef)
{
const bool cleanedFilename = fileRef->GetPath(&folderName);
folderName.AppendDir(L"_pcsx2_meta");
const auto* entry = &fileRef->entry->entry;
const bool metadataIsNonstandard = cleanedFilename || entry->data.mode != MemoryCardFileEntry::DefaultFileMode || entry->data.attr != 0;
if (metadataIsNonstandard)
{
// write metadata of file if it's nonstandard
if (!folderName.DirExists())
{
folderName.Mkdir();
}
wxFFile metaFile(folderName.GetFullPath(), L"wb");
if (metaFile.IsOpened())
{
metaFile.Write(entry->raw, sizeof(entry->raw));
}
}
else
{
// if metadata is standard remove metadata file if it exists
if (folderName.FileExists())
{
wxRemoveFile(folderName.GetFullPath());
// and remove the metadata dir if it's now empty
wxDir metaDir(folderName.GetPath());
if (metaDir.IsOpened() && !metaDir.HasFiles())
{
wxRmdir(folderName.GetPath());
}
}
}
}
void FileAccessHelper::WriteIndex(wxFileName folderName, MemoryCardFileEntry* const entry, MemoryCardFileMetadataReference* const parent)
{
parent->GetPath(&folderName);
char cleanName[sizeof(entry->entry.data.name)];
memcpy(cleanName, (const char*)entry->entry.data.name, sizeof(cleanName));
FileAccessHelper::CleanMemcardFilename(cleanName);
if (entry->IsDir())
{
folderName.AppendDir(wxString::FromAscii(cleanName));
}
else if (entry->IsFile())
{
folderName.SetName(wxString::FromAscii(cleanName));
}
const wxCharTypeBuffer fileName(folderName.GetName().ToUTF8());
folderName.SetName(L"_pcsx2_index");
YAML::Node index = LoadYAMLFromFile(folderName.GetFullPath());
YAML::Node entryNode = index[fileName.data()];
if (!entryNode.IsDefined())
{
// Newly added file - figure out the sort order as the entry should be added to the end of the list
unsigned int order = 0;
for (const auto& node : index)
{
order = std::max(order, node.second["order"].as<unsigned int>(0));
}
entryNode["order"] = order + 1;
}
// Update timestamps basing on internal data
const auto* e = &entry->entry.data;
entryNode["timeCreated"] = e->timeCreated.ToTime();
entryNode["timeModified"] = e->timeModified.ToTime();
// Write out the changes
wxFFile indexFile;
if (indexFile.Open(folderName.GetFullPath(), L"w"))
{
indexFile.Write(YAML::Dump(index));
}
}
wxFFile* FileAccessHelper::ReOpen(const wxFileName& folderName, MemoryCardFileMetadataReference* fileRef, bool writeMetadata)
{
std::string internalPath;
fileRef->GetInternalPath(&internalPath);
auto it = m_files.find(internalPath);
if (it != m_files.end())
{
// we already have a handle to this file
// if the caller wants to write metadata and we haven't done this recently, do so and remember that we did
if (writeMetadata)
{
if (m_lastWrittenFileRef != fileRef)
{
WriteMetadata(folderName, fileRef);
m_lastWrittenFileRef = fileRef;
}
}
else
{
if (m_lastWrittenFileRef != nullptr)
{
m_lastWrittenFileRef = nullptr;
}
}
// update the fileRef in the map since it might have been modified or deleted
it->second.fileRef = fileRef;
return it->second.fileHandle;
}
else
{
return this->Open(folderName, fileRef, writeMetadata);
}
}
void FileAccessHelper::CloseFileHandle(wxFFile* file, const MemoryCardFileEntry* entry)
{
file->Close();
delete file;
}
void FileAccessHelper::CloseMatching(const wxString& path)
{
wxFileName fn(path);
fn.Normalize();
wxString pathNormalized = fn.GetFullPath();
for (auto it = m_files.begin(); it != m_files.end();)
{
wxString openPath = it->second.fileHandle->GetName();
if (openPath.StartsWith(pathNormalized))
{
CloseFileHandle(it->second.fileHandle, it->second.fileRef->entry);
it = m_files.erase(it);
}
else
{
++it;
}
}
}
void FileAccessHelper::CloseAll()
{
for (auto it = m_files.begin(); it != m_files.end(); ++it)
{
CloseFileHandle(it->second.fileHandle, it->second.fileRef->entry);
}
m_files.clear();
}
void FileAccessHelper::FlushAll()
{
for (auto it = m_files.begin(); it != m_files.end(); ++it)
{
it->second.fileHandle->Flush();
}
}
void FileAccessHelper::ClearMetadataWriteState()
{
m_lastWrittenFileRef = nullptr;
}
bool FileAccessHelper::CleanMemcardFilename(char* name)
{
// invalid characters for filenames in the PS2 file system: { '/', '?', '*' }
// the following characters are valid in a PS2 memcard file system but invalid in Windows
// there's less restrictions on Linux but by cleaning them always we keep the folders cross-compatible
const char illegalChars[] = {'\\', '%', ':', '|', '"', '<', '>'};
bool cleaned = false;
const size_t filenameLength = strlen(name);
for (size_t i = 0; i < sizeof(illegalChars); ++i)
{
for (size_t j = 0; j < filenameLength; ++j)
{
if (name[j] == illegalChars[i])
{
name[j] = '_';
cleaned = true;
}
}
}
cleaned |= CleanMemcardFilenameEndDotOrSpace(name, filenameLength);
return cleaned;
}
bool FileAccessHelper::CleanMemcardFilenameEndDotOrSpace(char* name, size_t length)
{
// Windows truncates dots and spaces at the end of filenames, so make sure that doesn't happen
bool cleaned = false;
for (size_t j = length; j > 0; --j)
{
switch (name[j - 1])
{
case ' ':
case '.':
name[j - 1] = '_';
cleaned = true;
break;
default:
return cleaned;
}
}
return cleaned;
}
bool MemoryCardFileMetadataReference::GetPath(wxFileName* fileName) const
{
bool parentCleaned = false;
if (parent)
{
parentCleaned = parent->GetPath(fileName);
}
char cleanName[sizeof(entry->entry.data.name)];
memcpy(cleanName, (const char*)entry->entry.data.name, sizeof(cleanName));
bool localCleaned = FileAccessHelper::CleanMemcardFilename(cleanName);
if (entry->IsDir())
{
fileName->AppendDir(wxString::FromAscii(cleanName));
}
else if (entry->IsFile())
{
fileName->SetName(wxString::FromAscii(cleanName));
}
return parentCleaned || localCleaned;
}
void MemoryCardFileMetadataReference::GetInternalPath(std::string* fileName) const
{
if (parent)
{
parent->GetInternalPath(fileName);
}
fileName->append((const char*)entry->entry.data.name);
if (entry->IsDir())
{
fileName->append("/");
}
}
FolderMemoryCardAggregator::FolderMemoryCardAggregator()
{
for (uint i = 0; i < TotalCardSlots; ++i)
{
m_cards[i].SetSlot(i);
}
}
void FolderMemoryCardAggregator::Open()
{
for (int i = 0; i < TotalCardSlots; ++i)
{
m_cards[i].Open(m_enableFiltering, m_lastKnownFilter);
}
}
void FolderMemoryCardAggregator::Close()
{
for (int i = 0; i < TotalCardSlots; ++i)
{
m_cards[i].Close();
}
}
void FolderMemoryCardAggregator::SetFiltering(const bool enableFiltering)
{
m_enableFiltering = enableFiltering;
}
s32 FolderMemoryCardAggregator::IsPresent(uint slot)
{
return m_cards[slot].IsPresent();
}
void FolderMemoryCardAggregator::GetSizeInfo(uint slot, McdSizeInfo& outways)
{
m_cards[slot].GetSizeInfo(outways);
}
bool FolderMemoryCardAggregator::IsPSX(uint slot)
{
return m_cards[slot].IsPSX();
}
s32 FolderMemoryCardAggregator::Read(uint slot, u8* dest, u32 adr, int size)
{
return m_cards[slot].Read(dest, adr, size);
}
s32 FolderMemoryCardAggregator::Save(uint slot, const u8* src, u32 adr, int size)
{
return m_cards[slot].Save(src, adr, size);
}
s32 FolderMemoryCardAggregator::EraseBlock(uint slot, u32 adr)
{
return m_cards[slot].EraseBlock(adr);
}
u64 FolderMemoryCardAggregator::GetCRC(uint slot)
{
return m_cards[slot].GetCRC();
}
void FolderMemoryCardAggregator::NextFrame(uint slot)
{
m_cards[slot].NextFrame();
}
bool FolderMemoryCardAggregator::ReIndex(uint slot, const bool enableFiltering, const wxString& filter)
{
if (m_cards[slot].ReIndex(enableFiltering, filter))
{
SetFiltering(enableFiltering);
m_lastKnownFilter = filter;
return true;
}
return false;
}