|
@@ -71,25 +71,6 @@ namespace filesystem
|
|
namespace physfs
|
|
namespace physfs
|
|
{
|
|
{
|
|
|
|
|
|
-static size_t getDriveDelim(const std::string &input)
|
|
|
|
-{
|
|
|
|
- for (size_t i = 0; i < input.size(); ++i)
|
|
|
|
- if (input[i] == '/' || input[i] == '\\')
|
|
|
|
- return i;
|
|
|
|
- // Something's horribly wrong
|
|
|
|
- return 0;
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static std::string getDriveRoot(const std::string &input)
|
|
|
|
-{
|
|
|
|
- return input.substr(0, getDriveDelim(input)+1);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
-static std::string skipDriveRoot(const std::string &input)
|
|
|
|
-{
|
|
|
|
- return input.substr(getDriveDelim(input)+1);
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
static std::string normalize(const std::string &input)
|
|
static std::string normalize(const std::string &input)
|
|
{
|
|
{
|
|
std::stringstream out;
|
|
std::stringstream out;
|
|
@@ -105,13 +86,18 @@ static std::string normalize(const std::string &input)
|
|
return out.str();
|
|
return out.str();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static const Filesystem::CommonPath appCommonPaths[] =
|
|
|
|
+{
|
|
|
|
+ Filesystem::COMMONPATH_APP_SAVEDIR,
|
|
|
|
+ Filesystem::COMMONPATH_APP_DOCUMENTS
|
|
|
|
+};
|
|
|
|
+
|
|
static bool isAppCommonPath(Filesystem::CommonPath path)
|
|
static bool isAppCommonPath(Filesystem::CommonPath path)
|
|
{
|
|
{
|
|
switch (path)
|
|
switch (path)
|
|
{
|
|
{
|
|
- case Filesystem::COMMONPATH_APP_IDENTITY:
|
|
|
|
|
|
+ case Filesystem::COMMONPATH_APP_SAVEDIR:
|
|
case Filesystem::COMMONPATH_APP_DOCUMENTS:
|
|
case Filesystem::COMMONPATH_APP_DOCUMENTS:
|
|
- case Filesystem::COMMONPATH_APP_TEMP:
|
|
|
|
return true;
|
|
return true;
|
|
default:
|
|
default:
|
|
return false;
|
|
return false;
|
|
@@ -119,8 +105,12 @@ static bool isAppCommonPath(Filesystem::CommonPath path)
|
|
}
|
|
}
|
|
|
|
|
|
Filesystem::Filesystem()
|
|
Filesystem::Filesystem()
|
|
- : fused(false)
|
|
|
|
|
|
+ : appendIdentityToPath(false)
|
|
|
|
+ , fused(false)
|
|
, fusedSet(false)
|
|
, fusedSet(false)
|
|
|
|
+ , fullPaths()
|
|
|
|
+ , commonPathMountInfo()
|
|
|
|
+ , saveDirectoryNeedsMounting(false)
|
|
{
|
|
{
|
|
requirePath = {"?.lua", "?/init.lua"};
|
|
requirePath = {"?.lua", "?/init.lua"};
|
|
cRequirePath = {"??"};
|
|
cRequirePath = {"??"};
|
|
@@ -166,65 +156,65 @@ bool Filesystem::setIdentity(const char *ident, bool appendToPath)
|
|
if (!PHYSFS_isInit())
|
|
if (!PHYSFS_isInit())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- std::string old_save_path = save_path_full;
|
|
|
|
-
|
|
|
|
- // Store the save directory.
|
|
|
|
- save_identity = std::string(ident);
|
|
|
|
-
|
|
|
|
- // Generate the relative path to the game save folder.
|
|
|
|
- if (fused)
|
|
|
|
- save_path_relative = std::string(LOVE_APPDATA_PREFIX) + save_identity;
|
|
|
|
- else
|
|
|
|
- save_path_relative = std::string(LOVE_APPDATA_PREFIX LOVE_APPDATA_FOLDER LOVE_PATH_SEPARATOR) + save_identity;
|
|
|
|
-
|
|
|
|
- // Generate the full path to the game save folder.
|
|
|
|
- save_path_full = std::string(getAppdataDirectory()) + std::string(LOVE_PATH_SEPARATOR) + save_path_relative;
|
|
|
|
- save_path_full = normalize(save_path_full);
|
|
|
|
-
|
|
|
|
-#ifdef LOVE_ANDROID
|
|
|
|
- if (save_identity == "")
|
|
|
|
- save_identity = "unnamed";
|
|
|
|
-
|
|
|
|
- std::string storage_path;
|
|
|
|
- if (isAndroidSaveExternal())
|
|
|
|
- storage_path = SDL_AndroidGetExternalStoragePath();
|
|
|
|
- else
|
|
|
|
- storage_path = SDL_AndroidGetInternalStoragePath();
|
|
|
|
-
|
|
|
|
- std::string save_directory = storage_path + "/save";
|
|
|
|
-
|
|
|
|
- save_path_full = storage_path + std::string("/save/") + save_identity;
|
|
|
|
|
|
+ if (ident == nullptr || strlen(ident) == 0)
|
|
|
|
+ return false;
|
|
|
|
|
|
- if (!love::android::directoryExists(save_path_full.c_str()) &&
|
|
|
|
- !love::android::mkdir(save_path_full.c_str()))
|
|
|
|
- SDL_Log("Error: Could not create save directory %s!", save_path_full.c_str());
|
|
|
|
-#endif
|
|
|
|
|
|
+ // Validate whether re-mounting will work.
|
|
|
|
+ for (CommonPath p : appCommonPaths)
|
|
|
|
+ {
|
|
|
|
+ if (!commonPathMountInfo[p].mounted)
|
|
|
|
+ continue;
|
|
|
|
|
|
- // We now have something like:
|
|
|
|
- // save_identity: game
|
|
|
|
- // save_path_relative: ./LOVE/game
|
|
|
|
- // save_path_full: C:\Documents and Settings\user\Application Data/LOVE/game
|
|
|
|
|
|
+ // If a file is still open, unmount will fail.
|
|
|
|
+ std::string fullPath = getFullCommonPath(p);
|
|
|
|
+ if (!fullPath.empty() && !PHYSFS_canUnmount(fullPath.c_str()))
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
|
|
- // We don't want old read-only save paths to accumulate when we set a new
|
|
|
|
- // identity.
|
|
|
|
- if (!old_save_path.empty())
|
|
|
|
- PHYSFS_unmount(old_save_path.c_str());
|
|
|
|
|
|
+ bool oldMountedCommonPaths[COMMONPATH_MAX_ENUM] = {false};
|
|
|
|
|
|
- // Try to add the save directory to the search path.
|
|
|
|
- // (No error on fail, it means that the path doesn't exist).
|
|
|
|
- PHYSFS_mount(save_path_full.c_str(), nullptr, appendToPath);
|
|
|
|
|
|
+ // We don't want old save paths to accumulate when we set a new identity.
|
|
|
|
+ for (CommonPath p : appCommonPaths)
|
|
|
|
+ {
|
|
|
|
+ oldMountedCommonPaths[p] = commonPathMountInfo[p].mounted;
|
|
|
|
+ if (commonPathMountInfo[p].mounted)
|
|
|
|
+ unmount(p);
|
|
|
|
+ }
|
|
|
|
|
|
- // HACK: This forces setupWriteDirectory to be called the next time a file
|
|
|
|
- // is opened for writing - otherwise it won't be called at all if it was
|
|
|
|
- // already called at least once before.
|
|
|
|
- PHYSFS_setWriteDir(nullptr);
|
|
|
|
|
|
+ // These will be re-populated by getFullCommonPath.
|
|
|
|
+ for (CommonPath p : appCommonPaths)
|
|
|
|
+ fullPaths[p].clear();
|
|
|
|
+
|
|
|
|
+ // Store the save directory. getFullCommonPath(COMMONPATH_APP_*) uses this.
|
|
|
|
+ saveIdentity = std::string(ident);
|
|
|
|
+ appendIdentityToPath = appendToPath;
|
|
|
|
+
|
|
|
|
+ // Try to mount as readwrite without creating missing directories in the
|
|
|
|
+ // path hierarchy. If this fails, setupWriteDirectory will attempt to create
|
|
|
|
+ // them and try again.
|
|
|
|
+ // This is done so the save directory is only created on-demand.
|
|
|
|
+ if (!mountCommonPathInternal(COMMONPATH_APP_SAVEDIR, nullptr, MOUNT_PERMISSIONS_READWRITE, appendToPath, false))
|
|
|
|
+ saveDirectoryNeedsMounting = true;
|
|
|
|
+
|
|
|
|
+ // Mount any other app common paths with directory creation immediately
|
|
|
|
+ // instead of on-demand, since to get to this point they would have to be
|
|
|
|
+ // explicitly mounted already beforehand.
|
|
|
|
+ for (CommonPath p : appCommonPaths)
|
|
|
|
+ {
|
|
|
|
+ if (oldMountedCommonPaths[p] && p != COMMONPATH_APP_SAVEDIR)
|
|
|
|
+ {
|
|
|
|
+ // TODO: error handling?
|
|
|
|
+ auto info = commonPathMountInfo[p];
|
|
|
|
+ mountCommonPathInternal(p, info.mountPoint.c_str(), info.permissions, appendToPath, true);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
const char *Filesystem::getIdentity() const
|
|
const char *Filesystem::getIdentity() const
|
|
{
|
|
{
|
|
- return save_identity.c_str();
|
|
|
|
|
|
+ return saveIdentity.c_str();
|
|
}
|
|
}
|
|
|
|
|
|
bool Filesystem::setSource(const char *source)
|
|
bool Filesystem::setSource(const char *source)
|
|
@@ -233,7 +223,7 @@ bool Filesystem::setSource(const char *source)
|
|
return false;
|
|
return false;
|
|
|
|
|
|
// Check whether directory is already set.
|
|
// Check whether directory is already set.
|
|
- if (!game_source.empty())
|
|
|
|
|
|
+ if (!gameSource.empty())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
std::string new_search_path = source;
|
|
std::string new_search_path = source;
|
|
@@ -275,14 +265,14 @@ bool Filesystem::setSource(const char *source)
|
|
#endif
|
|
#endif
|
|
|
|
|
|
// Save the game source.
|
|
// Save the game source.
|
|
- game_source = new_search_path;
|
|
|
|
|
|
+ gameSource = new_search_path;
|
|
|
|
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
const char *Filesystem::getSource() const
|
|
const char *Filesystem::getSource() const
|
|
{
|
|
{
|
|
- return game_source.c_str();
|
|
|
|
|
|
+ return gameSource.c_str();
|
|
}
|
|
}
|
|
|
|
|
|
bool Filesystem::setupWriteDirectory()
|
|
bool Filesystem::setupWriteDirectory()
|
|
@@ -290,54 +280,19 @@ bool Filesystem::setupWriteDirectory()
|
|
if (!PHYSFS_isInit())
|
|
if (!PHYSFS_isInit())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- // These must all be set.
|
|
|
|
- if (save_identity.empty() || save_path_full.empty() || save_path_relative.empty())
|
|
|
|
- return false;
|
|
|
|
-
|
|
|
|
- // We need to make sure the write directory is created. To do that, we also
|
|
|
|
- // need to make sure all its parent directories are also created.
|
|
|
|
- std::string temp_writedir = getDriveRoot(save_path_full);
|
|
|
|
- std::string temp_createdir = skipDriveRoot(save_path_full);
|
|
|
|
-
|
|
|
|
- // On some sandboxed platforms, physfs will break when its write directory
|
|
|
|
- // is the root of the drive and it tries to create a folder (even if the
|
|
|
|
- // folder's path is in a writable location.) If the user's home folder is
|
|
|
|
- // in the save path, we'll try starting from there instead.
|
|
|
|
- if (save_path_full.find(getUserDirectory()) == 0)
|
|
|
|
- {
|
|
|
|
- temp_writedir = getUserDirectory();
|
|
|
|
- temp_createdir = save_path_full.substr(getUserDirectory().length());
|
|
|
|
-
|
|
|
|
- // Strip leading '/' characters from the path we want to create.
|
|
|
|
- size_t startpos = temp_createdir.find_first_not_of('/');
|
|
|
|
- if (startpos != std::string::npos)
|
|
|
|
- temp_createdir = temp_createdir.substr(startpos);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // Set either '/' or the user's home as a writable directory.
|
|
|
|
- // (We must create the save folder before mounting it).
|
|
|
|
- if (!PHYSFS_setWriteDir(temp_writedir.c_str()))
|
|
|
|
- return false;
|
|
|
|
-
|
|
|
|
- // Create the save folder. (We're now "at" either '/' or the user's home).
|
|
|
|
- if (!createDirectory(temp_createdir.c_str()))
|
|
|
|
- {
|
|
|
|
- // Clear the write directory in case of error.
|
|
|
|
- PHYSFS_setWriteDir(nullptr);
|
|
|
|
- return false;
|
|
|
|
- }
|
|
|
|
|
|
+ if (!saveDirectoryNeedsMounting)
|
|
|
|
+ return true;
|
|
|
|
|
|
- // Set the final write directory.
|
|
|
|
- if (!PHYSFS_setWriteDir(save_path_full.c_str()))
|
|
|
|
|
|
+ if (saveIdentity.empty())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- // Add the directory. (Will not be readded if already present).
|
|
|
|
- if (!PHYSFS_mount(save_path_full.c_str(), nullptr, 0))
|
|
|
|
- {
|
|
|
|
- PHYSFS_setWriteDir(nullptr); // Clear the write directory in case of error.
|
|
|
|
|
|
+ // Only the save directory is mounted on-demand if it doesn't exist yet.
|
|
|
|
+ // Other app common paths are immediately re-mounted in setIdentity.
|
|
|
|
+ bool createdir = true;
|
|
|
|
+ if (!mountCommonPathInternal(COMMONPATH_APP_SAVEDIR, nullptr, MOUNT_PERMISSIONS_READWRITE, appendIdentityToPath, createdir))
|
|
return false;
|
|
return false;
|
|
- }
|
|
|
|
|
|
|
|
|
|
+ saveDirectoryNeedsMounting = false;
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -374,7 +329,7 @@ bool Filesystem::mount(const char *archive, const char *mountpoint, bool appendT
|
|
|
|
|
|
// Always disallow mounting of files inside the game source, since it
|
|
// Always disallow mounting of files inside the game source, since it
|
|
// won't work anyway if the game source is a zipped .love file.
|
|
// won't work anyway if the game source is a zipped .love file.
|
|
- if (realPath.find(game_source) == 0)
|
|
|
|
|
|
+ if (realPath.find(gameSource) == 0)
|
|
return false;
|
|
return false;
|
|
|
|
|
|
realPath += LOVE_PATH_SEPARATOR;
|
|
realPath += LOVE_PATH_SEPARATOR;
|
|
@@ -386,30 +341,40 @@ bool Filesystem::mount(const char *archive, const char *mountpoint, bool appendT
|
|
|
|
|
|
bool Filesystem::mountFullPath(const char *archive, const char *mountpoint, MountPermissions permissions, bool appendToPath)
|
|
bool Filesystem::mountFullPath(const char *archive, const char *mountpoint, MountPermissions permissions, bool appendToPath)
|
|
{
|
|
{
|
|
- if (!PHYSFS_isInit() || !archive || !mountpoint)
|
|
|
|
|
|
+ if (!PHYSFS_isInit() || !archive)
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- if (permissions == MOUNT_PERMISSIONS_READWRITE && strlen(mountpoint) == 0)
|
|
|
|
- return false;
|
|
|
|
|
|
+ if (permissions == MOUNT_PERMISSIONS_READWRITE)
|
|
|
|
+ return PHYSFS_mountRW(archive, mountpoint, appendToPath) != 0;
|
|
|
|
|
|
- // TODO: readwrite mount
|
|
|
|
return PHYSFS_mount(archive, mountpoint, appendToPath) != 0;
|
|
return PHYSFS_mount(archive, mountpoint, appendToPath) != 0;
|
|
}
|
|
}
|
|
|
|
|
|
-bool Filesystem::mountCommonPath(CommonPath path, const char *mountpoint, MountPermissions permissions, bool appendToPath)
|
|
|
|
|
|
+bool Filesystem::mountCommonPathInternal(CommonPath path, const char *mountpoint, MountPermissions permissions, bool appendToPath, bool createDir)
|
|
{
|
|
{
|
|
std::string fullpath = getFullCommonPath(path);
|
|
std::string fullpath = getFullCommonPath(path);
|
|
if (fullpath.empty())
|
|
if (fullpath.empty())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- bool success = mountFullPath(fullpath.c_str(), mountpoint, permissions, appendToPath);
|
|
|
|
|
|
+ if (createDir && isAppCommonPath(path) && !isRealDirectory(fullpath))
|
|
|
|
+ {
|
|
|
|
+ if (!createRealDirectory(fullpath))
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
|
|
- if (!success && isAppCommonPath(path))
|
|
|
|
|
|
+ if (mountFullPath(fullpath.c_str(), mountpoint, permissions, appendToPath))
|
|
{
|
|
{
|
|
-
|
|
|
|
|
|
+ std::string mp = mountpoint != nullptr ? mountpoint : "/";
|
|
|
|
+ commonPathMountInfo[path] = {true, mp, permissions};
|
|
|
|
+ return true;
|
|
}
|
|
}
|
|
|
|
|
|
- return success;
|
|
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+bool Filesystem::mountCommonPath(CommonPath path, const char *mountpoint, MountPermissions permissions, bool appendToPath)
|
|
|
|
+{
|
|
|
|
+ return mountCommonPathInternal(path, mountpoint, permissions, appendToPath, true);
|
|
}
|
|
}
|
|
|
|
|
|
bool Filesystem::mount(Data *data, const char *archivename, const char *mountpoint, bool appendToPath)
|
|
bool Filesystem::mount(Data *data, const char *archivename, const char *mountpoint, bool appendToPath)
|
|
@@ -439,8 +404,13 @@ bool Filesystem::unmount(const char *archive)
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
- if (PHYSFS_getRealDir(archive) != nullptr)
|
|
|
|
- return PHYSFS_unmount(archive) != 0;
|
|
|
|
|
|
+ auto it = std::find(allowedMountPaths.begin(), allowedMountPaths.end(), archive);
|
|
|
|
+ if (it != allowedMountPaths.end())
|
|
|
|
+ return unmountFullPath(archive);
|
|
|
|
+
|
|
|
|
+ std::string sourceBase = getSourceBaseDirectory();
|
|
|
|
+ if (isFused() && sourceBase.compare(archive) == 0)
|
|
|
|
+ return unmountFullPath(archive);
|
|
|
|
|
|
if (strlen(archive) == 0 || strstr(archive, "..") || strcmp(archive, "/") == 0)
|
|
if (strlen(archive) == 0 || strstr(archive, "..") || strcmp(archive, "/") == 0)
|
|
return false;
|
|
return false;
|
|
@@ -459,13 +429,25 @@ bool Filesystem::unmount(const char *archive)
|
|
return PHYSFS_unmount(realPath.c_str()) != 0;
|
|
return PHYSFS_unmount(realPath.c_str()) != 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+bool Filesystem::unmountFullPath(const char *fullpath)
|
|
|
|
+{
|
|
|
|
+ if (!PHYSFS_isInit() || !fullpath)
|
|
|
|
+ return false;
|
|
|
|
+
|
|
|
|
+ return PHYSFS_unmount(fullpath) != 0;
|
|
|
|
+}
|
|
|
|
+
|
|
bool Filesystem::unmount(CommonPath path)
|
|
bool Filesystem::unmount(CommonPath path)
|
|
{
|
|
{
|
|
std::string fullpath = getFullCommonPath(path);
|
|
std::string fullpath = getFullCommonPath(path);
|
|
- if (fullpath.empty())
|
|
|
|
- return false;
|
|
|
|
|
|
|
|
- return unmount(fullpath.c_str());
|
|
|
|
|
|
+ if (!fullpath.empty() && unmountFullPath(fullpath.c_str()))
|
|
|
|
+ {
|
|
|
|
+ commonPathMountInfo[path].mounted = false;
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return false;
|
|
}
|
|
}
|
|
|
|
|
|
bool Filesystem::unmount(Data *data)
|
|
bool Filesystem::unmount(Data *data)
|
|
@@ -489,34 +471,60 @@ love::filesystem::File *Filesystem::newFile(const char *filename) const
|
|
|
|
|
|
std::string Filesystem::getFullCommonPath(CommonPath path)
|
|
std::string Filesystem::getFullCommonPath(CommonPath path)
|
|
{
|
|
{
|
|
- if (!fullCommonPaths[path].empty())
|
|
|
|
- return fullCommonPaths[path];
|
|
|
|
|
|
+ if (!fullPaths[path].empty())
|
|
|
|
+ return fullPaths[path];
|
|
|
|
|
|
- if (path == COMMONPATH_APP_IDENTITY || path == COMMONPATH_APP_DOCUMENTS || path == COMMONPATH_APP_TEMP)
|
|
|
|
|
|
+ if (isAppCommonPath(path))
|
|
{
|
|
{
|
|
-
|
|
|
|
|
|
+ if (saveIdentity.empty())
|
|
|
|
+ return fullPaths[path];
|
|
|
|
+
|
|
|
|
+ std::string rootpath;
|
|
|
|
+ switch (path)
|
|
|
|
+ {
|
|
|
|
+ case COMMONPATH_APP_SAVEDIR:
|
|
|
|
+ rootpath = getFullCommonPath(COMMONPATH_USER_APPDATA);
|
|
|
|
+ break;
|
|
|
|
+ case COMMONPATH_APP_DOCUMENTS:
|
|
|
|
+ rootpath = getFullCommonPath(COMMONPATH_USER_DOCUMENTS);
|
|
|
|
+ break;
|
|
|
|
+ default:
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (rootpath.empty())
|
|
|
|
+ return fullPaths[path];
|
|
|
|
+
|
|
|
|
+ std::string suffix;
|
|
|
|
+ if (isFused())
|
|
|
|
+ suffix = std::string(LOVE_PATH_SEPARATOR) + saveIdentity;
|
|
|
|
+ else
|
|
|
|
+ suffix = std::string(LOVE_PATH_SEPARATOR LOVE_APPDATA_FOLDER LOVE_PATH_SEPARATOR) + saveIdentity;
|
|
|
|
+
|
|
|
|
+ fullPaths[path] = normalize(rootpath + suffix);
|
|
|
|
+
|
|
|
|
+ return fullPaths[path];
|
|
}
|
|
}
|
|
|
|
|
|
#if defined(LOVE_MACOS) || defined(LOVE_IOS)
|
|
#if defined(LOVE_MACOS) || defined(LOVE_IOS)
|
|
|
|
|
|
switch (path)
|
|
switch (path)
|
|
{
|
|
{
|
|
- case COMMONPATH_APP_IDENTITY:
|
|
|
|
|
|
+ case COMMONPATH_APP_SAVEDIR:
|
|
case COMMONPATH_APP_DOCUMENTS:
|
|
case COMMONPATH_APP_DOCUMENTS:
|
|
- case COMMONPATH_APP_TEMP:
|
|
|
|
// Handled above.
|
|
// Handled above.
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_HOME:
|
|
case COMMONPATH_USER_HOME:
|
|
- fullCommonPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_HOME);
|
|
|
|
|
|
+ fullPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_HOME);
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_APPDATA:
|
|
case COMMONPATH_USER_APPDATA:
|
|
- fullCommonPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_APPSUPPORT);
|
|
|
|
|
|
+ fullPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_APPSUPPORT);
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_DESKTOP:
|
|
case COMMONPATH_USER_DESKTOP:
|
|
- fullCommonPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_DESKTOP);
|
|
|
|
|
|
+ fullPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_DESKTOP);
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_DOCUMENTS:
|
|
case COMMONPATH_USER_DOCUMENTS:
|
|
- fullCommonPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_DOCUMENTS);
|
|
|
|
|
|
+ fullPaths[path] = apple::getUserDirectory(apple::USER_DIRECTORY_DOCUMENTS);
|
|
break;
|
|
break;
|
|
case COMMONPATH_MAX_ENUM:
|
|
case COMMONPATH_MAX_ENUM:
|
|
break;
|
|
break;
|
|
@@ -529,9 +537,8 @@ std::string Filesystem::getFullCommonPath(CommonPath path)
|
|
|
|
|
|
switch (path)
|
|
switch (path)
|
|
{
|
|
{
|
|
- case COMMONPATH_APP_IDENTITY:
|
|
|
|
|
|
+ case COMMONPATH_APP_SAVEDIR:
|
|
case COMMONPATH_APP_DOCUMENTS:
|
|
case COMMONPATH_APP_DOCUMENTS:
|
|
- case COMMONPATH_APP_TEMP:
|
|
|
|
// Handled above.
|
|
// Handled above.
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_HOME:
|
|
case COMMONPATH_USER_HOME:
|
|
@@ -552,40 +559,66 @@ std::string Filesystem::getFullCommonPath(CommonPath path)
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
{
|
|
- fullCommonPaths[path] = to_utf8(winpath);
|
|
|
|
|
|
+ fullPaths[path] = to_utf8(winpath);
|
|
CoTaskMemFree(winpath);
|
|
CoTaskMemFree(winpath);
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+#elif defined(LOVE_ANDROID)
|
|
|
|
+
|
|
|
|
+ std::string storagepath;
|
|
|
|
+ if (isAndroidSaveExternal())
|
|
|
|
+ storagepath = SDL_AndroidGetExternalStoragePath();
|
|
else
|
|
else
|
|
- {
|
|
|
|
|
|
+ storagepath = SDL_AndroidGetInternalStoragePath();
|
|
|
|
|
|
|
|
+ switch (path)
|
|
|
|
+ {
|
|
|
|
+ case COMMONPATH_APP_SAVEDIR:
|
|
|
|
+ case COMMONPATH_APP_DOCUMENTS:
|
|
|
|
+ // Handled above.
|
|
|
|
+ break;
|
|
|
|
+ case COMMONPATH_USER_HOME:
|
|
|
|
+ fullPaths[path] = normalize(PHYSFS_getUserDir());
|
|
|
|
+ break;
|
|
|
|
+ case COMMONPATH_USER_APPDATA:
|
|
|
|
+ fullPaths[path] = normalize(storagepath + "/save/");
|
|
|
|
+ break;
|
|
|
|
+ case COMMONPATH_USER_DESKTOP:
|
|
|
|
+ // No such thing on Android?
|
|
|
|
+ break;
|
|
|
|
+ case COMMONPATH_USER_DOCUMENTS:
|
|
|
|
+ // TODO: something more idiomatic / useful?
|
|
|
|
+ fullPaths[path] = normalize(storagepath + "/Documents/");
|
|
|
|
+ break;
|
|
|
|
+ case COMMONPATH_MAX_ENUM:
|
|
|
|
+ break;
|
|
}
|
|
}
|
|
|
|
|
|
-#elif defined(LOVE_ANDROID)
|
|
|
|
-
|
|
|
|
#elif defined(LOVE_LINUX)
|
|
#elif defined(LOVE_LINUX)
|
|
|
|
|
|
const char *xdgdir = nullptr;
|
|
const char *xdgdir = nullptr;
|
|
|
|
|
|
switch (path)
|
|
switch (path)
|
|
{
|
|
{
|
|
- case COMMONPATH_APP_IDENTITY:
|
|
|
|
|
|
+ case COMMONPATH_APP_SAVEDIR:
|
|
case COMMONPATH_APP_DOCUMENTS:
|
|
case COMMONPATH_APP_DOCUMENTS:
|
|
- case COMMONPATH_APP_TEMP:
|
|
|
|
// Handled above.
|
|
// Handled above.
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_HOME:
|
|
case COMMONPATH_USER_HOME:
|
|
- fullCommonPaths[path] = normalize(PHYSFS_getUserDir());
|
|
|
|
|
|
+ fullPaths[path] = normalize(PHYSFS_getUserDir());
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_APPDATA:
|
|
case COMMONPATH_USER_APPDATA:
|
|
xdgdir = getenv("XDG_DATA_HOME");
|
|
xdgdir = getenv("XDG_DATA_HOME");
|
|
if (!xdgdir)
|
|
if (!xdgdir)
|
|
- fullCommonPaths[path] = normalize(std::string(getUserDirectory()) + "/.local/share/");
|
|
|
|
|
|
+ fullPaths[path] = normalize(std::string(getUserDirectory()) + "/.local/share/");
|
|
else
|
|
else
|
|
- fullCommonPaths[path] = xdgdir;
|
|
|
|
|
|
+ fullPaths[path] = xdgdir;
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_DESKTOP:
|
|
case COMMONPATH_USER_DESKTOP:
|
|
|
|
+ fullPaths[path] = normalize(std::string(getUserDirectory()) + "/Desktop/");
|
|
break;
|
|
break;
|
|
case COMMONPATH_USER_DOCUMENTS:
|
|
case COMMONPATH_USER_DOCUMENTS:
|
|
|
|
+ fullPaths[path] = normalize(std::string(getUserDirectory()) + "/Documents/");
|
|
break;
|
|
break;
|
|
case COMMONPATH_MAX_ENUM:
|
|
case COMMONPATH_MAX_ENUM:
|
|
break;
|
|
break;
|
|
@@ -593,7 +626,7 @@ std::string Filesystem::getFullCommonPath(CommonPath path)
|
|
|
|
|
|
#endif
|
|
#endif
|
|
|
|
|
|
- return fullCommonPaths[path];
|
|
|
|
|
|
+ return fullPaths[path];
|
|
}
|
|
}
|
|
|
|
|
|
const char *Filesystem::getWorkingDirectory()
|
|
const char *Filesystem::getWorkingDirectory()
|
|
@@ -621,58 +654,22 @@ const char *Filesystem::getWorkingDirectory()
|
|
|
|
|
|
std::string Filesystem::getUserDirectory()
|
|
std::string Filesystem::getUserDirectory()
|
|
{
|
|
{
|
|
-#if defined(LOVE_IOS) || defined(LOVE_MACOS)
|
|
|
|
- // PHYSFS_getUserDir doesn't give exactly the path we want on iOS.
|
|
|
|
- static std::string userDir = normalize(apple::getUserDirectory(apple::USER_DIRECTORY_HOME));
|
|
|
|
-#else
|
|
|
|
- static std::string userDir = normalize(PHYSFS_getUserDir());
|
|
|
|
-#endif
|
|
|
|
-
|
|
|
|
- return userDir;
|
|
|
|
|
|
+ return getFullCommonPath(COMMONPATH_USER_HOME);
|
|
}
|
|
}
|
|
|
|
|
|
std::string Filesystem::getAppdataDirectory()
|
|
std::string Filesystem::getAppdataDirectory()
|
|
{
|
|
{
|
|
- if (appdata.empty())
|
|
|
|
- {
|
|
|
|
-#ifdef LOVE_WINDOWS_UWP
|
|
|
|
- appdata = getUserDirectory();
|
|
|
|
-#elif defined(LOVE_WINDOWS)
|
|
|
|
- PWSTR path = nullptr;
|
|
|
|
- if (SUCCEEDED(SHGetKnownFolderPath(FOLDERID_RoamingAppData, 0, nullptr, &path)))
|
|
|
|
- {
|
|
|
|
- appdata = to_utf8(path);
|
|
|
|
- CoTaskMemFree(path);
|
|
|
|
- }
|
|
|
|
- else
|
|
|
|
- {
|
|
|
|
- wchar_t *w_appdata = _wgetenv(L"APPDATA");
|
|
|
|
- appdata = to_utf8(w_appdata);
|
|
|
|
- }
|
|
|
|
- replace_char(appdata, '\\', '/');
|
|
|
|
-#elif defined(LOVE_MACOS) || defined(LOVE_IOS)
|
|
|
|
- appdata = normalize(apple::getUserDirectory(apple::USER_DIRECTORY_APPSUPPORT));
|
|
|
|
-#elif defined(LOVE_LINUX)
|
|
|
|
- char *xdgdatahome = getenv("XDG_DATA_HOME");
|
|
|
|
- if (!xdgdatahome)
|
|
|
|
- appdata = normalize(std::string(getUserDirectory()) + "/.local/share/");
|
|
|
|
- else
|
|
|
|
- appdata = xdgdatahome;
|
|
|
|
-#else
|
|
|
|
- appdata = getUserDirectory();
|
|
|
|
-#endif
|
|
|
|
- }
|
|
|
|
- return appdata;
|
|
|
|
|
|
+ return getFullCommonPath(COMMONPATH_USER_APPDATA);
|
|
}
|
|
}
|
|
|
|
|
|
-const char *Filesystem::getSaveDirectory()
|
|
|
|
|
|
+std::string Filesystem::getSaveDirectory()
|
|
{
|
|
{
|
|
- return save_path_full.c_str();
|
|
|
|
|
|
+ return getFullCommonPath(COMMONPATH_APP_SAVEDIR);
|
|
}
|
|
}
|
|
|
|
|
|
std::string Filesystem::getSourceBaseDirectory() const
|
|
std::string Filesystem::getSourceBaseDirectory() const
|
|
{
|
|
{
|
|
- size_t source_len = game_source.length();
|
|
|
|
|
|
+ size_t source_len = gameSource.length();
|
|
|
|
|
|
if (source_len == 0)
|
|
if (source_len == 0)
|
|
return "";
|
|
return "";
|
|
@@ -681,9 +678,9 @@ std::string Filesystem::getSourceBaseDirectory() const
|
|
// symbols (i.e. '..' and '.')
|
|
// symbols (i.e. '..' and '.')
|
|
#ifdef LOVE_WINDOWS
|
|
#ifdef LOVE_WINDOWS
|
|
// In windows, delimiters can be either '/' or '\'.
|
|
// In windows, delimiters can be either '/' or '\'.
|
|
- size_t base_end_pos = game_source.find_last_of("/\\", source_len - 2);
|
|
|
|
|
|
+ size_t base_end_pos = gameSource.find_last_of("/\\", source_len - 2);
|
|
#else
|
|
#else
|
|
- size_t base_end_pos = game_source.find_last_of('/', source_len - 2);
|
|
|
|
|
|
+ size_t base_end_pos = gameSource.find_last_of('/', source_len - 2);
|
|
#endif
|
|
#endif
|
|
|
|
|
|
if (base_end_pos == std::string::npos)
|
|
if (base_end_pos == std::string::npos)
|
|
@@ -693,7 +690,7 @@ std::string Filesystem::getSourceBaseDirectory() const
|
|
if (base_end_pos == 0)
|
|
if (base_end_pos == 0)
|
|
base_end_pos = 1;
|
|
base_end_pos = 1;
|
|
|
|
|
|
- return game_source.substr(0, base_end_pos);
|
|
|
|
|
|
+ return gameSource.substr(0, base_end_pos);
|
|
}
|
|
}
|
|
|
|
|
|
std::string Filesystem::getRealDirectory(const char *filename) const
|
|
std::string Filesystem::getRealDirectory(const char *filename) const
|
|
@@ -739,7 +736,7 @@ bool Filesystem::createDirectory(const char *dir)
|
|
if (!PHYSFS_isInit())
|
|
if (!PHYSFS_isInit())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- if (PHYSFS_getWriteDir() == 0 && !setupWriteDirectory())
|
|
|
|
|
|
+ if (!setupWriteDirectory())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
if (!PHYSFS_mkdir(dir))
|
|
if (!PHYSFS_mkdir(dir))
|
|
@@ -753,7 +750,7 @@ bool Filesystem::remove(const char *file)
|
|
if (!PHYSFS_isInit())
|
|
if (!PHYSFS_isInit())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
- if (PHYSFS_getWriteDir() == 0 && !setupWriteDirectory())
|
|
|
|
|
|
+ if (!setupWriteDirectory())
|
|
return false;
|
|
return false;
|
|
|
|
|
|
if (!PHYSFS_delete(file))
|
|
if (!PHYSFS_delete(file))
|