ncm: Updated ListContentId for 11.0.0

This commit is contained in:
Adubbz 2021-03-20 20:54:44 +11:00 committed by SciresM
parent c99ce36d7d
commit 5666c59657
2 changed files with 200 additions and 31 deletions

View file

@ -166,6 +166,134 @@ namespace ams::ncm {
} }
ContentStorageImpl::ContentIterator::~ContentIterator() {
for (size_t i = 0; i < this->depth; i++) {
fs::CloseDirectory(this->handles[i]);
}
}
Result ContentStorageImpl::ContentIterator::Initialize(const char *root_path, size_t max_depth) {
/* Initialize tracking variables. */
this->depth = 0;
this->max_depth = max_depth;
this->entry_count = 0;
/* Create the base content directory path. */
MakeBaseContentDirectoryPath(std::addressof(this->path), root_path);
/* Open the base directory. */
R_TRY(this->OpenCurrentDirectory());
return ResultSuccess();
}
Result ContentStorageImpl::ContentIterator::OpenCurrentDirectory() {
/* Determine valid directory mode (prior to 2.0.0, NotRequireFileSize was not valid). */
const auto open_mode = hos::GetVersion() >= hos::Version_2_0_0 ? (fs::OpenDirectoryMode_All | fs::OpenDirectoryMode_NotRequireFileSize) : (fs::OpenDirectoryMode_All);
/* Open the directory for our current path. */
R_TRY(fs::OpenDirectory(std::addressof(this->handles[this->depth]), this->path, open_mode));
/* Increase our depth. */
++this->depth;
return ResultSuccess();
}
Result ContentStorageImpl::ContentIterator::OpenDirectory(const char *dir) {
/* Set our current path. */
this->path.Set(dir);
/* Open the directory. */
return this->OpenCurrentDirectory();
}
Result ContentStorageImpl::ContentIterator::GetNext(std::optional<fs::DirectoryEntry> *out) {
/* Iterate until we get the next entry. */
while (true) {
/* Ensure that we have entries loaded. */
R_TRY(this->LoadEntries());
/* If we failed to load any entries, there's nothing to get. */
if (this->entry_count <= 0) {
*out = std::nullopt;
return ResultSuccess();
}
/* Get the next entry. */
const auto &entry = this->entries[--this->entry_count];
/* Process the current entry. */
switch (entry.type) {
case fs::DirectoryEntryType_Directory:
/* If the entry if a directory, we want to recurse into it if we can. */
if (this->depth < this->max_depth) {
/* Construct the full path for the subdirectory. */
PathString entry_path;
entry_path.SetFormat("%s/%s", this->path.Get(), entry.name);
/* Open the subdirectory. */
R_TRY(this->OpenDirectory(entry_path.Get()));
}
break;
case fs::DirectoryEntryType_File:
/* Otherwise, if the entry is a file, return it. */
*out = entry;
return ResultSuccess();
AMS_UNREACHABLE_DEFAULT_CASE();
}
}
return ResultSuccess();
}
Result ContentStorageImpl::ContentIterator::LoadEntries() {
/* If we already have entries loaded, we don't need to do anything. */
R_SUCCEED_IF(this->entry_count != 0);
/* If we have no directories open, there's nothing for us to load. */
if (this->depth == 0) {
this->entry_count = 0;
return ResultSuccess();
}
/* Determine the maximum entries that we can load. */
const s64 max_entries = this->depth == this->max_depth ? MaxDirectoryEntries : 1;
/* Read entries from the current directory. */
s64 num_entries;
R_TRY(fs::ReadDirectory(std::addressof(num_entries), this->entries, this->handles[this->depth - 1], max_entries));
/* If we successfully read entries, load them. */
if (num_entries > 0) {
/* Reverse the order of the loaded entries, for our future convenience. */
for (fs::DirectoryEntry *start_entry = this->entries, *end_entry = this->entries + num_entries - 1; start_entry < end_entry; ++start_entry, --end_entry) {
std::swap(*start_entry, *end_entry);
}
/* Set our entry count. */
this->entry_count = num_entries;
return ResultSuccess();
}
/* We didn't read any entries, so we need to advance to the next directory. */
fs::CloseDirectory(this->handles[--this->depth]);
/* Find the index of the parent directory's substring. */
size_t i = this->path.GetLength() - 1;
while (this->path.Get()[i--] != '/') {
AMS_ABORT_UNLESS(i > 0);
}
/* Set the path to the parent directory. */
this->path.Set(this->path.GetSubstring(0, i));
/* Try to load again from the parent directory. */
return this->LoadEntries();
}
ContentStorageImpl::~ContentStorageImpl() { ContentStorageImpl::~ContentStorageImpl() {
this->InvalidateFileCache(); this->InvalidateFileCache();
} }
@ -225,6 +353,7 @@ namespace ams::ncm {
fs::CloseFile(this->cached_file_handle); fs::CloseFile(this->cached_file_handle);
this->cached_content_id = InvalidContentId; this->cached_content_id = InvalidContentId;
} }
this->content_iterator = std::nullopt;
} }
Result ContentStorageImpl::OpenContentIdFile(ContentId content_id) { Result ContentStorageImpl::OpenContentIdFile(ContentId content_id) {
@ -445,48 +574,59 @@ namespace ams::ncm {
return ResultSuccess(); return ResultSuccess();
} }
Result ContentStorageImpl::ListContentId(sf::Out<s32> out_count, const sf::OutArray<ContentId> &out_buf, s32 offset) { Result ContentStorageImpl::ListContentId(sf::Out<s32> out_count, const sf::OutArray<ContentId> &out, s32 offset) {
R_UNLESS(offset >= 0, ncm::ResultInvalidOffset()); R_UNLESS(offset >= 0, ncm::ResultInvalidOffset());
R_TRY(this->EnsureEnabled()); R_TRY(this->EnsureEnabled());
/* Obtain the content base directory path. */ if (!this->content_iterator.has_value() || !this->last_content_offset.has_value() || this->last_content_offset != offset) {
PathString path; /* Create and initialize the content cache. */
MakeBaseContentDirectoryPath(std::addressof(path), this->root_path); this->content_iterator.emplace();
R_TRY(this->content_iterator->Initialize(this->root_path, GetHierarchicalContentDirectoryDepth(this->make_content_path_func)));
const auto depth = GetHierarchicalContentDirectoryDepth(this->make_content_path_func); /* Advance to the desired offset. */
size_t entry_count = 0; for (auto current_offset = 0; current_offset < offset; /* ... */) {
/* Get the next directory entry. */
std::optional<fs::DirectoryEntry> dir_entry;
R_TRY(this->content_iterator->GetNext(std::addressof(dir_entry)));
/* Traverse the content base directory finding all valid content. */ /* If we run out of entries before reaching the desired offset, we're done. */
R_TRY(TraverseDirectory(path, depth, [&](bool *should_continue, bool *should_retry_dir_read, const char *current_path, const fs::DirectoryEntry &entry) { if (!dir_entry) {
*should_retry_dir_read = false; out_count.SetValue(0);
*should_continue = true; return ResultSuccess();
}
/* We have nothing to do if not working with a file. */ /* If the current entry is a valid content id, advance. */
if (entry.type != fs::DirectoryEntryType_File) { if (GetContentIdFromString(dir_entry->name, std::strlen(dir_entry->name))) {
return ResultSuccess(); ++current_offset;
}
}
}
/* Iterate, reading as many entries as we can. */
s32 count = 0;
while (count < static_cast<s32>(out.GetSize())) {
/* Get the next directory entry. */
std::optional<fs::DirectoryEntry> dir_entry;
R_TRY(this->content_iterator->GetNext(std::addressof(dir_entry)));
/* Don't continue if the directory entry is absent. */
if (!dir_entry) {
break;
} }
/* Skip entries until we reach the start offset. */ /* Process the entry, if it's a valid content id. */
if (offset > 0) { if (auto content_id = GetContentIdFromString(dir_entry->name, std::strlen(dir_entry->name)); content_id.has_value()) {
--offset; /* Output the content id. */
return ResultSuccess(); out[count++] = *content_id;
/* Update our last content offset. */
this->last_content_offset = offset + count;
} }
}
/* We don't necessarily expect to be able to completely fill the output buffer. */ /* Set the output count. */
if (entry_count >= out_buf.GetSize()) { *out_count = count;
*should_continue = false;
return ResultSuccess();
}
auto content_id = GetContentIdFromString(entry.name, std::strlen(entry.name));
if (content_id) {
out_buf[entry_count++] = *content_id;
}
return ResultSuccess();
}));
out_count.SetValue(static_cast<s32>(entry_count));
return ResultSuccess(); return ResultSuccess();
} }

View file

@ -22,16 +22,45 @@
namespace ams::ncm { namespace ams::ncm {
class ContentStorageImpl : public ContentStorageImplBase { class ContentStorageImpl : public ContentStorageImplBase {
private:
class ContentIterator {
NON_COPYABLE(ContentIterator);
NON_MOVEABLE(ContentIterator);
static constexpr size_t MaxDirectoryHandles = 0x8;
static constexpr size_t MaxDirectoryEntries = 0x10;
public:
fs::DirectoryHandle handles[MaxDirectoryHandles]{};
size_t depth{};
size_t max_depth{};
PathString path{};
fs::DirectoryEntry entries[MaxDirectoryEntries]{};
s64 entry_count{};
public:
constexpr ContentIterator() = default;
~ContentIterator();
Result Initialize(const char *root_path, size_t max_depth);
Result GetNext(std::optional<fs::DirectoryEntry> *out);
private:
Result OpenCurrentDirectory();
Result OpenDirectory(const char *dir);
Result LoadEntries();
};
protected: protected:
PlaceHolderAccessor placeholder_accessor; PlaceHolderAccessor placeholder_accessor;
ContentId cached_content_id; ContentId cached_content_id;
fs::FileHandle cached_file_handle; fs::FileHandle cached_file_handle;
RightsIdCache *rights_id_cache; RightsIdCache *rights_id_cache;
std::optional<ContentIterator> content_iterator;
std::optional<s32> last_content_offset;
public: public:
static Result InitializeBase(const char *root_path); static Result InitializeBase(const char *root_path);
static Result CleanupBase(const char *root_path); static Result CleanupBase(const char *root_path);
static Result VerifyBase(const char *root_path); static Result VerifyBase(const char *root_path);
public: public:
ContentStorageImpl() : rights_id_cache(nullptr), content_iterator(std::nullopt), last_content_offset(std::nullopt) { /* ... */ }
~ContentStorageImpl(); ~ContentStorageImpl();
Result Initialize(const char *root_path, MakeContentPathFunction content_path_func, MakePlaceHolderPathFunction placeholder_path_func, bool delay_flush, RightsIdCache *rights_id_cache); Result Initialize(const char *root_path, MakeContentPathFunction content_path_func, MakePlaceHolderPathFunction placeholder_path_func, bool delay_flush, RightsIdCache *rights_id_cache);