Skip to content

Commit

Permalink
straighten out path finding
Browse files Browse the repository at this point in the history
  • Loading branch information
nikitalita committed Nov 7, 2024
1 parent a164f76 commit 677f0ea
Show file tree
Hide file tree
Showing 3 changed files with 60 additions and 42 deletions.
88 changes: 52 additions & 36 deletions utility/file_access_gdre.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -38,10 +38,6 @@ Vector<Ref<PackedFileInfo>> GDREPackedData::get_file_info_list(const Vector<Stri
return ret;
}

bool GDREPackedData::has_mapped_file(const String &p_path) {
return files.has(PathMD5(p_path.md5_buffer()));
}

void GDREPackedData::add_pack_source(PackSource *p_source) {
if (p_source != nullptr) {
sources.push_back(p_source);
Expand Down Expand Up @@ -145,35 +141,23 @@ Error GDREPackedData::add_pack(const String &p_path, bool p_replace_files, uint6
}

Ref<FileAccess> GDREPackedData::try_open_path(const String &p_path) {
PathMD5 pmd5(p_path.md5_buffer());
String simplified_path = p_path.simplify_path();
PathMD5 pmd5(simplified_path.md5_buffer());
HashMap<PathMD5, PackedData::PackedFile, PathMD5>::Iterator E = files.find(pmd5);
if (!E || E->value.offset == 0) {
if (is_gdre_file(p_path)) {
WARN_PRINT("Attempted to load a GDRE resource while another pack is loaded...");
if (PackedData::get_singleton()) {
// This works even when PackedData is disabled
return PackedData::get_singleton()->try_open_path(p_path);
}
}
return Ref<FileAccess>();
if (!E) {
return nullptr; //not found
}
if (E->value.offset == 0) {
return nullptr; //was erased
}

return E->value.src->get_file(p_path, &E->value);
}

bool GDREPackedData::has_path(const String &p_path) {
bool ret = files.has(PathMD5(p_path.md5_buffer()));
if (!ret) {
// We return true if the path is a GDRE resource to ensure that this will be used to load it
if (is_gdre_file(p_path)) {
WARN_PRINT("Attempted to check for existence of GDRE resource while another pack is loaded...");
if (PackedData::get_singleton()) {
// This works even when PackedData is disabled
return PackedData::get_singleton()->has_path(p_path);
}
}
}
return ret;
return files.has(PathMD5(p_path.simplify_path().md5_buffer()));
}

Ref<DirAccess> GDREPackedData::try_open_directory(const String &p_path) {
Ref<DirAccess> da = memnew(DirAccessGDRE());
if (da->change_dir(p_path) != OK) {
Expand Down Expand Up @@ -232,21 +216,25 @@ Error FileAccessGDRE::open_internal(const String &p_path, int p_mode_flags) {
if (!(p_mode_flags & WRITE) && GDREPackedData::get_singleton() && !GDREPackedData::get_singleton()->is_disabled()) {
proxy = GDREPackedData::get_singleton()->try_open_path(p_path);
if (proxy.is_valid()) {
if (is_gdre_file(p_path)) {
WARN_PRINT(vformat("Opening gdre file %s from a loaded external pack???? PLEASE REPORT THIS!!!!", p_path));
};
return OK;
}
}
String path = p_path;
if (is_gdre_file(p_path)) {
WARN_PRINT("Attempted to open a gdre file while we have a pack loaded...");
if (GDREPackedData::real_packed_data_has_pack_loaded() && !PackedData::get_singleton()->is_disabled()) {
if (!(p_mode_flags & WRITE) && is_gdre_file(p_path)) {
WARN_PRINT(vformat("Attempted to open a gdre file %s while we have a pack loaded...", p_path));
if (PathFinder::real_packed_data_has_path(p_path)) {
// this works even when PackedData is disabled
proxy = PackedData::get_singleton()->try_open_path(p_path);
if (proxy.is_valid()) {
return OK;
}
}
}
// Otherwise, it's on the file system.
path = PathFinder::_fix_path_file_access(p_path);
path = PathFinder::_fix_path_file_access(p_path, p_mode_flags);
Error err;
proxy = _open_filesystem(path, p_mode_flags, &err);
if (err != OK) {
Expand Down Expand Up @@ -323,12 +311,17 @@ void FileAccessGDRE::store_buffer(const uint8_t *p_src, uint64_t p_length) {
}

bool FileAccessGDRE::file_exists(const String &p_name) {
if (GDREPackedData::get_singleton() && !GDREPackedData::get_singleton()->is_disabled() && GDREPackedData::get_singleton()->has_path(p_name)) {
if (PathFinder::gdre_packed_data_valid_path(p_name)) {
return true;
}
if (proxy.is_valid()) {
return proxy->file_exists(p_name);
}
// TODO: I don't think this is necessary and may screw things up; revisit this
// After proxy, check if it's in the real packed data
// if (PathFinder::real_packed_data_has_path(p_name)) {
// return true;
// }

return false;
}
Expand Down Expand Up @@ -804,10 +797,28 @@ Ref<FileAccess> FileAccessGDRE::_open_filesystem(const String &p_path, int p_mod
return file_proxy;
}

String PathFinder::_fix_path_file_access(const String &p_path) {
bool PathFinder::real_packed_data_has_path(const String &p_path, bool check_disabled) {
return PackedData::get_singleton() && (!check_disabled || !PackedData::get_singleton()->is_disabled()) && PackedData::get_singleton()->has_path(p_path);
}

bool PathFinder::gdre_packed_data_valid_path(const String &p_path) {
return GDREPackedData::get_singleton() && !GDREPackedData::get_singleton()->is_disabled() && GDREPackedData::get_singleton()->has_path(p_path);
}

String PathFinder::_fix_path_file_access(const String &p_path, int p_mode_flags) {
if (p_path.begins_with("res://")) {
if (p_path.get_file().begins_with("gdre_")) {
if (is_gdre_file(p_path)) {
WARN_PRINT("WARNING: Calling fix_path on a gdre file...");
if (!(p_mode_flags & FileAccess::WRITE)) {
if (gdre_packed_data_valid_path(p_path)) {
WARN_PRINT("WARNING: fix_path: gdre file is in a loaded external pack???? PLEASE REPORT THIS!!!!");
return p_path.replace_first("res://", "");
};
// PackedData is disabled if an external pack is loaded, so we don't check if it's disabled here
if (real_packed_data_has_path(p_path)) {
return p_path.replace_first("res://", "");
}
}
String res_path = GDRESettings::get_singleton()->get_gdre_resource_path();
if (res_path != "") {
return p_path.replace("res:/", res_path);
Expand All @@ -820,12 +831,17 @@ String PathFinder::_fix_path_file_access(const String &p_path) {
return p_path.replace_first("res://", "");
} else if (p_path.begins_with("user://")) { // Some packs have user files in them, so we need to check for those
if (p_path.get_file().begins_with("gdre_")) {
WARN_PRINT("WARNING: Calling fix_path on a gdre file...");
// TODO: Once we start adding user files, we'll have to handle this; for right now, it doesn't particularly matter
WARN_PRINT(vformat("WARNING: Calling fix_path on a gdre file %s...", p_path));
if (!(p_mode_flags & FileAccess::WRITE)) {
if (gdre_packed_data_valid_path(p_path)) {
WARN_PRINT(vformat("WARNING: fix_path: gdre file %s is in a loaded external pack???? PLEASE REPORT THIS!!!!", p_path));
return p_path.replace_first("user://", "");
};
}
}

// check if the file is in the PackedData first
if (GDREPackedData::get_singleton() && !GDREPackedData::get_singleton()->is_disabled() && GDREPackedData::get_singleton()->has_path(p_path)) {
if (!(p_mode_flags & FileAccess::WRITE) && gdre_packed_data_valid_path(p_path)) {
return p_path.replace_first("user://", "");
}
String data_dir = OS::get_singleton()->get_user_data_dir();
Expand Down
10 changes: 6 additions & 4 deletions utility/file_access_gdre.h
Original file line number Diff line number Diff line change
Expand Up @@ -60,8 +60,6 @@ class GDREPackedData {
void add_pack_source(PackSource *p_source);
void add_path(const String &p_pkg_path, const String &p_path, uint64_t p_ofs, uint64_t p_size, const uint8_t *p_md5, PackSource *p_src, bool p_replace_files, bool p_encrypted = false, bool p_pck_src = false); // for PackSource

bool has_mapped_file(const String &p_path);

void clear();
void set_disabled(bool p_disabled);
_FORCE_INLINE_ bool is_disabled() const;
Expand Down Expand Up @@ -180,19 +178,23 @@ class DirAccessGDRE : public DirAccess {

class PathFinder {
public:
static bool real_packed_data_has_path(const String &p_path, bool check_disabled = false);
static bool gdre_packed_data_valid_path(const String &p_path);
static String _fix_path_file_access(const String &p_path);
static String _fix_path_file_access(const String &p_path, int p_mode_flags = 0);
};

template <class T>
class FileAccessProxy : public T {
static_assert(std::is_base_of<FileAccess, T>::value, "T must derive from FileAccess");

int mode_flags;

public:
virtual String fix_path(const String &p_path) const override {
return PathFinder::_fix_path_file_access(p_path.replace("\\", "/"));
return PathFinder::_fix_path_file_access(p_path.replace("\\", "/"), mode_flags);
}
virtual Error open_internal(const String &p_path, int p_mode_flags) override {
mode_flags = p_mode_flags;
return T::open_internal(p_path, p_mode_flags);
}
};
Expand Down
4 changes: 2 additions & 2 deletions utility/gdre_settings.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1320,7 +1320,7 @@ Array GDRESettings::get_import_files(bool copy) {
}

bool GDRESettings::has_file(const String &p_path) {
return GDREPackedData::get_singleton()->has_mapped_file(p_path);
return GDREPackedData::get_singleton()->has_path(p_path);
}

Error GDRESettings::load_pack_uid_cache(bool p_reset) {
Expand Down Expand Up @@ -1447,7 +1447,7 @@ Error GDRESettings::_load_import_file(const String &p_path, bool should_load_md5
// sound.wav-<pathmd5>.smp -> sound.wav-<pathmd5>.md5
String md5 = src.get_basename() + ".md5";
while (true) {
if (GDREPackedData::get_singleton()->has_mapped_file(md5)) {
if (GDREPackedData::get_singleton()->has_path(md5)) {
break;
}

Expand Down

0 comments on commit 677f0ea

Please sign in to comment.