mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2025-01-22 22:56:11 +00:00
c7026b9094
* Implement NCM * Modernize ncm_main * Remove unnecessary smExit * Give access to svcCallSecureMonitor * Stack size bump * Fix incorrect setup for NandUser's content storage entry * Fix a potential data abort when flushing the placeholder accessor cache * Fix HasFile and HasDirectory * Use r+b, not w+b * Misc fixes * errno begone * Fixed more stdio error handling * More main fixes * Various command improvements * Make dispatch tables great again * Fix logic inversion * Fixed content path generation * Bump heap size, fix CleanupAllPlaceHolder * Various fixes. Note: This contains debug stuff which will be removed later. I was getting tired of having to cherrypick tiny changes * Fixed placeholder/content deletion * Fixed incorrect content manager destruction * Prevent automatic placeholder creation on open * Fixed List implementation. Also lots of debug logging. * Removed debug code * Added a scope guard for WritePlaceHolder * Manually prevent placeholder/content appending * Revert "Removed debug code" This reverts commitd6ff261fcc
. * Always cache placeholder file. Switch to ftell for preventing appending * Universally use EnsureEnabled * Abstract away file writing logic * Misc cleanup * Refactor placeholder cacheing * Remove debug code (again) * Revert "Remove debug code (again)" This reverts commit168447d80e
. * Misc changes * Fixed file modes * Fixed ContentId/PlaceHolderId alignment * Improved type safety * Fixed reinitialization * Fixed doubleup on path creation * Remove debug code * Fixed 1.0.0 booting * Correct amount of add on content * Correct main thread stack size * lr: Introducing registered data * Reorder stratosphere Makefile * Move results to libstrat * lr: Cleanup lr_redirection * lr: lr_manager tweaks * lr: Imrpoved path handling and adjust ResolveAddOnContentPath order * lr: Organise types * Add eof newlines * lr: Eliminate unnecessary vars * lr: Unnecessary vars 2 electric boogaloo * lr: Various helpers * lr: RegisteredLocationResolver helpers * ncm: Move ncm_types to libstrat * ncm: Misc cleanup * Implement NCM * Modernize ncm_main * Remove unnecessary smExit * Give access to svcCallSecureMonitor * Stack size bump * Fix incorrect setup for NandUser's content storage entry * Fix a potential data abort when flushing the placeholder accessor cache * Fix HasFile and HasDirectory * Use r+b, not w+b * Misc fixes * errno begone * Fixed more stdio error handling * More main fixes * Various command improvements * Make dispatch tables great again * Fix logic inversion * Fixed content path generation * Bump heap size, fix CleanupAllPlaceHolder * Various fixes. Note: This contains debug stuff which will be removed later. I was getting tired of having to cherrypick tiny changes * Fixed placeholder/content deletion * Fixed incorrect content manager destruction * Prevent automatic placeholder creation on open * Fixed List implementation. Also lots of debug logging. * Removed debug code * Added a scope guard for WritePlaceHolder * Manually prevent placeholder/content appending * Revert "Removed debug code" This reverts commitd6ff261fcc
. * Always cache placeholder file. Switch to ftell for preventing appending * Universally use EnsureEnabled * Abstract away file writing logic * Misc cleanup * Refactor placeholder cacheing * Remove debug code (again) * Revert "Remove debug code (again)" This reverts commit168447d80e
. * Misc changes * Fixed file modes * Fixed ContentId/PlaceHolderId alignment * Improved type safety * Fixed reinitialization * Fixed doubleup on path creation * Remove debug code * Fixed 1.0.0 booting * Correct amount of add on content * Correct main thread stack size * lr: Introducing registered data * Reorder stratosphere Makefile * Move results to libstrat * lr: Cleanup lr_redirection * lr: lr_manager tweaks * lr: Imrpoved path handling and adjust ResolveAddOnContentPath order * lr: Organise types * Add eof newlines * lr: Eliminate unnecessary vars * lr: Unnecessary vars 2 electric boogaloo * lr: Various helpers * lr: RegisteredLocationResolver helpers * ncm: Move ncm_types to libstrat * ncm: Misc cleanup * Updated AddOnContentLocationResolver and RegisteredLocationResolver to 9.0.0 * Finished updating lr to 9.0.0 * Updated NCM to 9.0.0 * Fix libstrat includes * Fixed application launching * title_id_2 -> owner_tid * Updated to new-ipc * Change to using pure virtuals * Title Id -> Program Id * Fixed compilation against master * std::scoped_lock<> -> std::scoped_lock * Adopted R_UNLESS and R_CONVERT * Prefix namespace to Results * Adopt std::numeric_limits * Fixed incorrect error handling in ReadFile * Adopted AMS_ABORT_UNLESS * Adopt util::GenerateUuid() * Syntax improvements * ncm_types: Address review * Address more review comments * Updated copyrights * Address more feedback * More feedback addressed * More changes * Move dispatch tables out of interface files * Addressed remaining comments * lr: move into libstratosphere * ncm: Fix logic inversion * lr: Add comments * lr: Remove whitespace * ncm: Start addressing feedback * ncm: Cleanup InitializeContentManager * lr: support client-side usage * lr_service -> lr_api * ncm: Begin refactoring content manager * ncm: More content manager improvements * ncm: Content manager mount improvements * ldr: use lr bindings * lr bindings usage: minor fixes * ncm/lr: Pointer placement * ncm: placeholder accessor cleanup * ncm: minor fixes * ncm: refactor rights cache * ncm: content meta database cleanup * ncm: move content meta database impl out of interface file * ncm: Use const ContentMetaKey & * ncm: fix other non-const ContentMetaKey references * ncm: content meta database cleanup * ncm: content storage fixes for 2.0.0 * ncm: add missing end of file newlines * ncm: implement ContentMetaReader * ncm: client-side api * ncm: trim trailing spaces * ncm: FS_MAX_PATH-1 -> fs::EntryNameLengthMax * ncm: Use PathString and Path * fs: implement accessor wrappers for ncm * fs: implement user fs wrappers * fs: add MountSdCard * ncm: move to content manager impl * ncm: fix up main * kvdb: use fs:: * fs: Add wrappers needed for ncm * ncm: use fs bindings, other refactoring * ncm: minor fixes * fsa: fix ReadFile without size output * fs: add substorage, rom path tool * ncm: fix dangling fsdev usage * fs: fix bug in Commit * fs: fixed incorrect mode check * fs: implement Mount(System)Data * ncm: don't delete hos * results: add R_SUCCEED_IF * ams-except-ncm: use R_SUCCEED_IF * ncm: added comments * ncm: fix api definitions * ncm: use R_SUCCEED_IF * pm: think of the savings * ncm: employ kernel strats * ncm: Nintendo has 5 MiB of heap. Give ourselves 4 to be safe, pending analysis * ncm: refactor IDs, split types header into many headers * ams.mitm: use fs bindings instead of stdio * fs: SystemData uses SystemDataId * ncm: improve meta-db accuracy * ncm: inline getlatestkey * fs: improve UnsupportedOperation results * fs: modernize mount utils * ams: misc fixes for merge-errors * fs: improve unsupportedoperation results * git subrepo pull emummc subrepo: subdir: "emummc" merged: "d12dd546" upstream: origin: "https://github.com/m4xw/emuMMC" branch: "develop" commit: "d12dd546" git-subrepo: version: "0.4.1" origin: "???" commit: "???" * util: add boundedmap * ncm: minor style fixes * ncm: don't unmount if mounting fails * lr: bug fixes * ncm: implement ncm.for-initialize + ncm.for-safemode * lr: ncm::ProgramId::Invalid -> ncm::InvalidProgramId * ncm: fix open directory mode on 1.0.0 * ncm: fix fs use, implement more of < 4.0.0 for-initialize/safemode * ncm: implement packagedcontent -> content for building metadb * ncm: fix save data flag management * ncm: address some review suggestions (thanks @leoetlino!) * updater: use fs bindings * fs: implement MountCode * fs: prefer make_unique to operator new * ncm: implement remaining ContentMetaDatabaseBuilder functionality Co-authored-by: Michael Scire <SciresM@gmail.com>
536 lines
26 KiB
C++
536 lines
26 KiB
C++
/*
|
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
|
|
namespace ams::fs {
|
|
|
|
namespace {
|
|
|
|
Result ConvertNcaCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultNcaCorrupted::Includes(res));
|
|
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultInvalidNcaFileSystemType, fs::ResultInvalidRomNcaFileSystemType())
|
|
R_CONVERT(fs::ResultInvalidAcidFileSize, fs::ResultInvalidRomAcidFileSize())
|
|
R_CONVERT(fs::ResultInvalidAcidSize, fs::ResultInvalidRomAcidSize())
|
|
R_CONVERT(fs::ResultInvalidAcid, fs::ResultInvalidRomAcid())
|
|
R_CONVERT(fs::ResultAcidVerificationFailed, fs::ResultRomAcidVerificationFailed())
|
|
R_CONVERT(fs::ResultInvalidNcaSignature, fs::ResultInvalidRomNcaSignature())
|
|
R_CONVERT(fs::ResultNcaHeaderSignature1VerificationFailed, fs::ResultRomNcaHeaderSignature1VerificationFailed())
|
|
R_CONVERT(fs::ResultNcaHeaderSignature2VerificationFailed, fs::ResultRomNcaHeaderSignature2VerificationFailed())
|
|
R_CONVERT(fs::ResultNcaFsHeaderHashVerificationFailed, fs::ResultRomNcaFsHeaderHashVerificationFailed())
|
|
R_CONVERT(fs::ResultInvalidNcaKeyIndex, fs::ResultInvalidRomNcaKeyIndex())
|
|
R_CONVERT(fs::ResultInvalidNcaFsHeaderHashType, fs::ResultInvalidRomNcaFsHeaderHashType())
|
|
R_CONVERT(fs::ResultInvalidNcaFsHeaderEncryptionType, fs::ResultInvalidRomNcaFsHeaderEncryptionType())
|
|
R_CONVERT(fs::ResultInvalidHierarchicalSha256BlockSize, fs::ResultInvalidRomHierarchicalSha256BlockSize())
|
|
R_CONVERT(fs::ResultInvalidHierarchicalSha256LayerCount, fs::ResultInvalidRomHierarchicalSha256LayerCount())
|
|
R_CONVERT(fs::ResultHierarchicalSha256BaseStorageTooLarge, fs::ResultRomHierarchicalSha256BaseStorageTooLarge())
|
|
R_CONVERT(fs::ResultHierarchicalSha256HashVerificationFailed, fs::ResultRomHierarchicalSha256HashVerificationFailed())
|
|
R_CATCH_ALL() { /* ... */ }
|
|
} R_END_TRY_CATCH;
|
|
|
|
AMS_ASSERT(false);
|
|
return fs::ResultNcaCorrupted();
|
|
}
|
|
|
|
Result ConvertIntegrityVerificationStorageCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultIntegrityVerificationStorageCorrupted::Includes(res));
|
|
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultIncorrectIntegrityVerificationMagic, fs::ResultIncorrectRomIntegrityVerificationMagic())
|
|
R_CONVERT(fs::ResultInvalidZeroHash, fs::ResultInvalidRomZeroHash())
|
|
R_CONVERT(fs::ResultNonRealDataVerificationFailed, fs::ResultRomNonRealDataVerificationFailed())
|
|
R_CONVERT(fs::ResultInvalidHierarchicalIntegrityVerificationLayerCount, fs::ResultInvalidRomHierarchicalIntegrityVerificationLayerCount())
|
|
R_CONVERT(fs::ResultClearedRealDataVerificationFailed, fs::ResultClearedRomRealDataVerificationFailed())
|
|
R_CONVERT(fs::ResultUnclearedRealDataVerificationFailed, fs::ResultUnclearedRomRealDataVerificationFailed())
|
|
R_CATCH_ALL() { /* ... */ }
|
|
} R_END_TRY_CATCH;
|
|
|
|
AMS_ASSERT(false);
|
|
return fs::ResultIntegrityVerificationStorageCorrupted();
|
|
}
|
|
|
|
Result ConvertBuiltInStorageCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultBuiltInStorageCorrupted::Includes(res));
|
|
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultGptHeaderVerificationFailed, fs::ResultRomGptHeaderVerificationFailed())
|
|
R_CATCH_ALL() { /* ... */ }
|
|
} R_END_TRY_CATCH;
|
|
|
|
AMS_ASSERT(false);
|
|
return fs::ResultBuiltInStorageCorrupted();
|
|
}
|
|
|
|
Result ConvertPartitionFileSystemCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultPartitionFileSystemCorrupted::Includes(res));
|
|
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultInvalidSha256PartitionHashTarget, fs::ResultInvalidRomSha256PartitionHashTarget())
|
|
R_CONVERT(fs::ResultSha256PartitionHashVerificationFailed, fs::ResultRomSha256PartitionHashVerificationFailed())
|
|
R_CONVERT(fs::ResultPartitionSignatureVerificationFailed, fs::ResultRomPartitionSignatureVerificationFailed())
|
|
R_CONVERT(fs::ResultSha256PartitionSignatureVerificationFailed, fs::ResultRomSha256PartitionSignatureVerificationFailed())
|
|
R_CONVERT(fs::ResultInvalidPartitionEntryOffset, fs::ResultInvalidRomPartitionEntryOffset())
|
|
R_CONVERT(fs::ResultInvalidSha256PartitionMetaDataSize, fs::ResultInvalidRomSha256PartitionMetaDataSize())
|
|
R_CATCH_ALL() { /* ... */ }
|
|
} R_END_TRY_CATCH;
|
|
|
|
AMS_ASSERT(false);
|
|
return fs::ResultPartitionFileSystemCorrupted();
|
|
}
|
|
|
|
Result ConvertFatFileSystemCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultFatFileSystemCorrupted::Includes(res));
|
|
|
|
return res;
|
|
}
|
|
|
|
Result ConvertHostFileSystemCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultHostFileSystemCorrupted::Includes(res));
|
|
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultHostEntryCorrupted, fs::ResultRomHostEntryCorrupted())
|
|
R_CONVERT(fs::ResultHostFileDataCorrupted, fs::ResultRomHostFileDataCorrupted())
|
|
R_CONVERT(fs::ResultHostFileCorrupted, fs::ResultRomHostFileCorrupted())
|
|
R_CONVERT(fs::ResultInvalidHostHandle, fs::ResultInvalidRomHostHandle())
|
|
R_CATCH_ALL() { /* ... */ }
|
|
} R_END_TRY_CATCH;
|
|
|
|
AMS_ASSERT(false);
|
|
return fs::ResultHostFileSystemCorrupted();
|
|
}
|
|
|
|
Result ConvertDatabaseCorruptedResult(Result res) {
|
|
AMS_ASSERT(fs::ResultDatabaseCorrupted::Includes(res));
|
|
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultInvalidAllocationTableBlock, fs::ResultInvalidRomAllocationTableBlock())
|
|
R_CONVERT(fs::ResultInvalidKeyValueListElementIndex, fs::ResultInvalidRomKeyValueListElementIndex())
|
|
R_CATCH_ALL() { /* ... */ }
|
|
} R_END_TRY_CATCH;
|
|
|
|
AMS_ASSERT(false);
|
|
return fs::ResultDatabaseCorrupted();
|
|
}
|
|
|
|
Result ConvertRomFsResult(Result res) {
|
|
R_TRY_CATCH(res) {
|
|
R_CONVERT(fs::ResultUnsupportedVersion, fs::ResultUnsupportedRomVersion())
|
|
R_CONVERT(fs::ResultNcaCorrupted, ConvertNcaCorruptedResult(res))
|
|
R_CONVERT(fs::ResultIntegrityVerificationStorageCorrupted, ConvertIntegrityVerificationStorageCorruptedResult(res))
|
|
R_CONVERT(fs::ResultBuiltInStorageCorrupted, ConvertBuiltInStorageCorruptedResult(res))
|
|
R_CONVERT(fs::ResultPartitionFileSystemCorrupted, ConvertPartitionFileSystemCorruptedResult(res))
|
|
R_CONVERT(fs::ResultFatFileSystemCorrupted, ConvertFatFileSystemCorruptedResult(res))
|
|
R_CONVERT(fs::ResultHostFileSystemCorrupted, ConvertHostFileSystemCorruptedResult(res))
|
|
R_CONVERT(fs::ResultDatabaseCorrupted, ConvertDatabaseCorruptedResult(res))
|
|
R_CONVERT(fs::ResultNotFound, fs::ResultPathNotFound())
|
|
R_CONVERT(fs::ResultPermissionDenied, fs::ResultTargetLocked())
|
|
R_CONVERT(fs::ResultIncompatiblePath, fs::ResultPathNotFound())
|
|
} R_END_TRY_CATCH;
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result ReadFile(IStorage *storage, s64 offset, void *buffer, size_t size) {
|
|
AMS_ASSERT(storage != nullptr);
|
|
AMS_ASSERT(offset >= 0);
|
|
AMS_ASSERT(buffer != nullptr || size == 0);
|
|
|
|
return ConvertRomFsResult(storage->Read(offset, buffer, size));
|
|
}
|
|
|
|
Result ReadFileHeader(IStorage *storage, RomFileSystemInformation *out) {
|
|
AMS_ASSERT(storage != nullptr);
|
|
AMS_ASSERT(out != nullptr);
|
|
|
|
return ReadFile(storage, 0, out, sizeof(*out));
|
|
}
|
|
|
|
constexpr size_t CalculateRequiredWorkingMemorySize(const RomFileSystemInformation &header) {
|
|
const size_t needed_size = header.directory_bucket_size + header.directory_entry_size + header.file_bucket_size + header.file_entry_size;
|
|
return util::AlignUp(needed_size, 8);
|
|
}
|
|
|
|
class RomFsFile : public fsa::IFile, public impl::Newable {
|
|
private:
|
|
RomFsFileSystem *parent;
|
|
s64 start;
|
|
s64 end;
|
|
public:
|
|
RomFsFile(RomFsFileSystem *p, s64 s, s64 e) : parent(p), start(s), end(e) { /* ... */ }
|
|
virtual ~RomFsFile() { /* ... */ }
|
|
|
|
Result VerifyArguments(size_t *out, s64 offset, void *buf, size_t size, const fs::ReadOption &option) {
|
|
R_TRY(DryRead(out, offset, size, option, fs::OpenMode_Read));
|
|
|
|
AMS_ASSERT(this->GetStorage() != nullptr);
|
|
AMS_ASSERT(offset >= 0);
|
|
AMS_ASSERT(buf != nullptr || size == 0);
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result ConvertResult(Result res) const {
|
|
return ConvertRomFsResult(res);
|
|
}
|
|
|
|
s64 GetOffset() const {
|
|
return this->start;
|
|
}
|
|
|
|
s64 GetSize() const {
|
|
return this->end - this->start;
|
|
}
|
|
|
|
IStorage *GetStorage() {
|
|
return this->parent->GetBaseStorage();
|
|
}
|
|
public:
|
|
virtual Result ReadImpl(size_t *out, s64 offset, void *buffer, size_t size, const fs::ReadOption &option) override {
|
|
size_t read_size = 0;
|
|
R_TRY(this->VerifyArguments(std::addressof(read_size), offset, buffer, size, option));
|
|
|
|
R_TRY(this->ConvertResult(this->GetStorage()->Read(offset + this->start, buffer, size)));
|
|
*out = read_size;
|
|
|
|
return ResultSuccess();
|
|
}
|
|
|
|
virtual Result GetSizeImpl(s64 *out) override {
|
|
*out = this->GetSize();
|
|
return ResultSuccess();
|
|
}
|
|
|
|
virtual Result FlushImpl() override {
|
|
return ResultSuccess();
|
|
}
|
|
|
|
virtual Result WriteImpl(s64 offset, const void *buffer, size_t size, const fs::WriteOption &option) override {
|
|
return fs::ResultUnsupportedOperationInRomFsFileA();
|
|
}
|
|
|
|
virtual Result SetSizeImpl(s64 size) override {
|
|
return fs::ResultUnsupportedOperationInRomFsFileA();
|
|
}
|
|
|
|
virtual Result OperateRangeImpl(void *dst, size_t dst_size, fs::OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override {
|
|
switch (op_id) {
|
|
case OperationId::InvalidateCache:
|
|
case OperationId::QueryRange:
|
|
{
|
|
R_UNLESS(offset >= 0, fs::ResultOutOfRange());
|
|
R_UNLESS(this->GetSize() >= 0, fs::ResultOutOfRange());
|
|
}
|
|
default:
|
|
return fs::ResultUnsupportedOperationInRomFsFileB();
|
|
}
|
|
}
|
|
public:
|
|
virtual sf::cmif::DomainObjectId GetDomainObjectId() const override {
|
|
AMS_ABORT();
|
|
}
|
|
};
|
|
|
|
class RomFsDirectory : public fsa::IDirectory, public impl::Newable {
|
|
private:
|
|
using FindPosition = RomFsFileSystem::RomFileTable::FindPosition;
|
|
private:
|
|
RomFsFileSystem *parent;
|
|
FindPosition current_find;
|
|
FindPosition first_find;
|
|
fs::OpenDirectoryMode mode;
|
|
public:
|
|
RomFsDirectory(RomFsFileSystem *p, const FindPosition &f, fs::OpenDirectoryMode m) : parent(p), current_find(f), first_find(f), mode(m) { /* ... */ }
|
|
virtual ~RomFsDirectory() override { /* ... */ }
|
|
public:
|
|
virtual Result ReadImpl(s64 *out_count, DirectoryEntry *out_entries, s64 max_entries) {
|
|
return this->ReadImpl(out_count, std::addressof(this->current_find), out_entries, max_entries);
|
|
}
|
|
|
|
virtual Result GetEntryCountImpl(s64 *out) {
|
|
FindPosition find = this->first_find;
|
|
return this->ReadImpl(out, std::addressof(find), nullptr, 0);
|
|
}
|
|
private:
|
|
Result ReadImpl(s64 *out_count, FindPosition *find, DirectoryEntry *out_entries, s64 max_entries) {
|
|
AMS_ASSERT(out_count != nullptr);
|
|
AMS_ASSERT(find != nullptr);
|
|
|
|
constexpr size_t NameBufferSize = fs::EntryNameLengthMax + 1;
|
|
char *name_buf = static_cast<char *>(::ams::fs::impl::Allocate(NameBufferSize));
|
|
R_UNLESS(name_buf != nullptr, fs::ResultAllocationFailureInRomFsFileSystemE());
|
|
ON_SCOPE_EXIT { ::ams::fs::impl::Deallocate(name_buf, NameBufferSize); };
|
|
|
|
s32 i = 0;
|
|
|
|
if (this->mode & fs::OpenDirectoryMode_Directory) {
|
|
while (i < max_entries || out_entries == nullptr) {
|
|
R_TRY_CATCH(this->parent->GetRomFileTable()->FindNextDirectory(name_buf, find, NameBufferSize)) {
|
|
R_CATCH(fs::ResultDbmFindFinished) { break; }
|
|
} R_END_TRY_CATCH;
|
|
|
|
if (out_entries) {
|
|
R_UNLESS(strnlen(name_buf, NameBufferSize) < NameBufferSize, fs::ResultTooLongPath());
|
|
strncpy(out_entries[i].name, name_buf, fs::EntryNameLengthMax);
|
|
out_entries[i].name[fs::EntryNameLengthMax] = '\x00';
|
|
out_entries[i].type = fs::DirectoryEntryType_Directory;
|
|
out_entries[i].file_size = 0;
|
|
}
|
|
|
|
i++;
|
|
}
|
|
}
|
|
|
|
if (this->mode & fs::OpenDirectoryMode_File) {
|
|
while (i < max_entries || out_entries == nullptr) {
|
|
auto file_pos = find->next_file;
|
|
|
|
R_TRY_CATCH(this->parent->GetRomFileTable()->FindNextFile(name_buf, find, NameBufferSize)) {
|
|
R_CATCH(fs::ResultDbmFindFinished) { break; }
|
|
} R_END_TRY_CATCH;
|
|
|
|
if (out_entries) {
|
|
R_UNLESS(strnlen(name_buf, NameBufferSize) < NameBufferSize, fs::ResultTooLongPath());
|
|
strncpy(out_entries[i].name, name_buf, fs::EntryNameLengthMax);
|
|
out_entries[i].name[fs::EntryNameLengthMax] = '\x00';
|
|
out_entries[i].type = fs::DirectoryEntryType_File;
|
|
|
|
RomFsFileSystem::RomFileTable::FileInfo file_info;
|
|
R_TRY(this->parent->GetRomFileTable()->OpenFile(std::addressof(file_info), this->parent->GetRomFileTable()->ConvertToFileId(file_pos)));
|
|
out_entries[i].file_size = file_info.size.Get();
|
|
}
|
|
|
|
i++;
|
|
}
|
|
}
|
|
|
|
*out_count = i;
|
|
return ResultSuccess();
|
|
}
|
|
public:
|
|
virtual sf::cmif::DomainObjectId GetDomainObjectId() const override {
|
|
AMS_ABORT();
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
|
|
RomFsFileSystem::RomFsFileSystem() : base_storage() {
|
|
/* ... */
|
|
}
|
|
|
|
RomFsFileSystem::~RomFsFileSystem() {
|
|
/* ... */
|
|
}
|
|
|
|
Result RomFsFileSystem::GetRequiredWorkingMemorySize(size_t *out, IStorage *storage) {
|
|
RomFileSystemInformation header;
|
|
R_TRY(ReadFileHeader(storage, std::addressof(header)));
|
|
|
|
*out = CalculateRequiredWorkingMemorySize(header);
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::Initialize(IStorage *base, void *work, size_t work_size, bool use_cache) {
|
|
AMS_ABORT_UNLESS(!use_cache || work != nullptr);
|
|
AMS_ABORT_UNLESS(base != nullptr);
|
|
|
|
/* Read the header. */
|
|
RomFileSystemInformation header;
|
|
R_TRY(ReadFileHeader(base, std::addressof(header)));
|
|
|
|
/* Set up our storages. */
|
|
if (use_cache) {
|
|
const size_t needed_size = CalculateRequiredWorkingMemorySize(header);
|
|
R_UNLESS(work_size >= needed_size, fs::ResultPreconditionViolation());
|
|
|
|
u8 *buf = static_cast<u8 *>(work);
|
|
auto dir_bucket_buf = buf; buf += header.directory_bucket_size;
|
|
auto dir_entry_buf = buf; buf += header.directory_entry_size;
|
|
auto file_bucket_buf = buf; buf += header.file_bucket_size;
|
|
auto file_entry_buf = buf; buf += header.file_entry_size;
|
|
|
|
R_TRY(ReadFile(base, header.directory_bucket_offset, dir_bucket_buf, header.directory_bucket_size));
|
|
R_TRY(ReadFile(base, header.directory_entry_offset, dir_entry_buf, header.directory_entry_size));
|
|
R_TRY(ReadFile(base, header.file_bucket_offset, file_bucket_buf, header.file_bucket_size));
|
|
R_TRY(ReadFile(base, header.file_entry_offset, file_entry_buf, header.file_entry_size));
|
|
|
|
this->dir_bucket_storage.reset(new MemoryStorage(dir_bucket_buf, header.directory_bucket_size));
|
|
this->dir_entry_storage.reset(new MemoryStorage(dir_entry_buf, header.directory_entry_size));
|
|
this->file_bucket_storage.reset(new MemoryStorage(file_bucket_buf, header.file_bucket_size));
|
|
this->file_entry_storage.reset(new MemoryStorage(file_entry_buf, header.file_entry_size));
|
|
} else {
|
|
this->dir_bucket_storage.reset(new SubStorage(base, header.directory_bucket_offset, header.directory_bucket_size));
|
|
this->dir_entry_storage.reset(new SubStorage(base, header.directory_entry_offset, header.directory_entry_size));
|
|
this->file_bucket_storage.reset(new SubStorage(base, header.file_bucket_offset, header.file_bucket_size));
|
|
this->file_entry_storage.reset(new SubStorage(base, header.file_entry_offset, header.file_entry_size));
|
|
}
|
|
|
|
/* Ensure we allocated storages successfully. */
|
|
R_UNLESS(this->dir_bucket_storage != nullptr, fs::ResultAllocationFailureInRomFsFileSystemA());
|
|
R_UNLESS(this->dir_entry_storage != nullptr, fs::ResultAllocationFailureInRomFsFileSystemA());
|
|
R_UNLESS(this->file_bucket_storage != nullptr, fs::ResultAllocationFailureInRomFsFileSystemA());
|
|
R_UNLESS(this->file_entry_storage != nullptr, fs::ResultAllocationFailureInRomFsFileSystemA());
|
|
|
|
/* Initialize the rom table. */
|
|
{
|
|
|
|
SubStorage db(this->dir_bucket_storage.get(), 0, header.directory_bucket_size);
|
|
SubStorage de(this->dir_entry_storage.get(), 0, header.directory_entry_size);
|
|
SubStorage fb(this->file_bucket_storage.get(), 0, header.file_bucket_size);
|
|
SubStorage fe(this->file_entry_storage.get(), 0, header.file_entry_size);
|
|
R_TRY(this->rom_file_table.Initialize(db, de, fb, fe));
|
|
}
|
|
|
|
/* Set members. */
|
|
this->entry_size = header.body_offset;
|
|
this->base_storage = base;
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::Initialize(std::unique_ptr<IStorage>&& base, void *work, size_t work_size, bool use_cache) {
|
|
this->unique_storage = std::move(base);
|
|
return this->Initialize(this->unique_storage.get(), work, work_size, use_cache);
|
|
}
|
|
|
|
Result RomFsFileSystem::GetFileInfo(RomFileTable::FileInfo *out, const char *path) {
|
|
R_TRY_CATCH(this->rom_file_table.OpenFile(out, path)) {
|
|
R_CONVERT(fs::ResultDbmNotFound, fs::ResultPathNotFound());
|
|
R_CONVERT(fs::ResultDbmInvalidOperation, fs::ResultPathNotFound());
|
|
} R_END_TRY_CATCH;
|
|
return ResultSuccess();
|
|
}
|
|
|
|
IStorage *RomFsFileSystem::GetBaseStorage() {
|
|
return this->base_storage;
|
|
}
|
|
|
|
RomFsFileSystem::RomFileTable *RomFsFileSystem::GetRomFileTable() {
|
|
return std::addressof(this->rom_file_table);
|
|
}
|
|
|
|
Result RomFsFileSystem::GetFileBaseOffset(s64 *out, const char *path) {
|
|
AMS_ABORT_UNLESS(out != nullptr);
|
|
AMS_ABORT_UNLESS(path != nullptr);
|
|
|
|
RomFileTable::FileInfo info;
|
|
R_TRY(this->GetFileInfo(std::addressof(info), path));
|
|
*out = this->entry_size + info.offset.Get();
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::CreateFileImpl(const char *path, s64 size, int flags) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::DeleteFileImpl(const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::CreateDirectoryImpl(const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::DeleteDirectoryImpl(const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::DeleteDirectoryRecursivelyImpl(const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::RenameFileImpl(const char *old_path, const char *new_path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::RenameDirectoryImpl(const char *old_path, const char *new_path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::GetEntryTypeImpl(fs::DirectoryEntryType *out, const char *path) {
|
|
RomDirectoryInfo dir_info;
|
|
R_TRY_CATCH(this->rom_file_table.GetDirectoryInformation(std::addressof(dir_info), path)) {
|
|
R_CONVERT(fs::ResultDbmNotFound, fs::ResultPathNotFound())
|
|
R_CATCH(fs::ResultDbmInvalidOperation) {
|
|
RomFileTable::FileInfo file_info;
|
|
R_TRY(this->GetFileInfo(std::addressof(file_info), path));
|
|
*out = fs::DirectoryEntryType_File;
|
|
return ResultSuccess();
|
|
}
|
|
} R_END_TRY_CATCH;
|
|
|
|
*out = fs::DirectoryEntryType_Directory;
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::OpenFileImpl(std::unique_ptr<fs::fsa::IFile> *out_file, const char *path, fs::OpenMode mode) {
|
|
AMS_ASSERT(out_file != nullptr);
|
|
AMS_ASSERT(path != nullptr);
|
|
|
|
R_UNLESS((mode & fs::OpenMode_All) == fs::OpenMode_Read, fs::ResultInvalidArgument());
|
|
|
|
RomFileTable::FileInfo file_info;
|
|
R_TRY(this->GetFileInfo(std::addressof(file_info), path));
|
|
|
|
auto file = std::make_unique<RomFsFile>(this, this->entry_size + file_info.offset.Get(), this->entry_size + file_info.offset.Get() + file_info.size.Get());
|
|
R_UNLESS(file != nullptr, fs::ResultAllocationFailureInRomFsFileSystemB());
|
|
|
|
*out_file = std::move(file);
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::OpenDirectoryImpl(std::unique_ptr<fs::fsa::IDirectory> *out_dir, const char *path, fs::OpenDirectoryMode mode) {
|
|
AMS_ASSERT(out_dir != nullptr);
|
|
AMS_ASSERT(path != nullptr);
|
|
|
|
RomFileTable::FindPosition find;
|
|
R_TRY_CATCH(this->rom_file_table.FindOpen(std::addressof(find), path)) {
|
|
R_CONVERT(fs::ResultDbmNotFound, fs::ResultPathNotFound())
|
|
R_CONVERT(fs::ResultDbmInvalidOperation, fs::ResultPathNotFound())
|
|
} R_END_TRY_CATCH;
|
|
|
|
auto dir = std::make_unique<RomFsDirectory>(this, find, mode);
|
|
R_UNLESS(dir != nullptr, fs::ResultAllocationFailureInRomFsFileSystemC());
|
|
|
|
*out_dir = std::move(dir);
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::CommitImpl() {
|
|
return ResultSuccess();
|
|
}
|
|
|
|
Result RomFsFileSystem::GetFreeSpaceSizeImpl(s64 *out, const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemC();
|
|
}
|
|
|
|
Result RomFsFileSystem::GetTotalSpaceSizeImpl(s64 *out, const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemC();
|
|
}
|
|
|
|
Result RomFsFileSystem::CleanDirectoryRecursivelyImpl(const char *path) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemA();
|
|
}
|
|
|
|
Result RomFsFileSystem::CommitProvisionallyImpl(s64 counter) {
|
|
return fs::ResultUnsupportedOperationInRomFsFileSystemB();
|
|
}
|
|
|
|
Result RomFsFileSystem::RollbackImpl() {
|
|
return ResultSuccess();
|
|
}
|
|
}
|