2014-01-07 11:25:07 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2011 Laschweinski
|
2018-01-29 16:25:32 +08:00
|
|
|
Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2022-04-28 11:43:59 +08:00
|
|
|
Copyright (c) 2021-2022 Bytedance Inc.
|
2014-01-07 11:25:07 +08:00
|
|
|
|
2022-10-01 16:24:52 +08:00
|
|
|
https://axmolengine.github.io/
|
2014-01-07 11:25:07 +08:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
2023-06-11 13:08:08 +08:00
|
|
|
#include "platform/linux/FileUtils-linux.h"
|
|
|
|
#include "platform/linux/Application-linux.h"
|
|
|
|
#include "platform/Common.h"
|
|
|
|
#include "base/Macros.h"
|
|
|
|
#include "base/UTF8.h"
|
2013-04-10 11:45:55 +08:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
#define DECLARE_GUARD (void)0 // std::lock_guard<std::recursive_mutex> mutexGuard(_mutex)
|
2018-09-17 10:47:41 +08:00
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_BEGIN
|
2013-04-10 11:45:55 +08:00
|
|
|
|
2023-01-10 23:06:04 +08:00
|
|
|
static std::string _checkPath(const char* path) {
|
|
|
|
std::string ret;
|
|
|
|
ret.resize(PATH_MAX - 1);
|
|
|
|
int n = readlink(path, &ret.front(), PATH_MAX);
|
|
|
|
if (n > 0) {
|
|
|
|
ret.resize(n);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
return std::string{};
|
|
|
|
}
|
|
|
|
|
2013-07-12 12:03:39 +08:00
|
|
|
FileUtils* FileUtils::getInstance()
|
2013-04-10 11:45:55 +08:00
|
|
|
{
|
2016-06-23 11:39:23 +08:00
|
|
|
if (s_sharedFileUtils == nullptr)
|
2013-04-10 11:45:55 +08:00
|
|
|
{
|
2013-06-20 14:13:12 +08:00
|
|
|
s_sharedFileUtils = new FileUtilsLinux();
|
2021-12-25 10:04:45 +08:00
|
|
|
if (!s_sharedFileUtils->init())
|
2013-07-23 20:44:08 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
delete s_sharedFileUtils;
|
|
|
|
s_sharedFileUtils = nullptr;
|
2022-07-16 10:43:05 +08:00
|
|
|
AXLOG("ERROR: Could not init CCFileUtilsLinux");
|
2013-07-23 20:44:08 +08:00
|
|
|
}
|
2013-04-10 11:45:55 +08:00
|
|
|
}
|
|
|
|
return s_sharedFileUtils;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
FileUtilsLinux::FileUtilsLinux() {}
|
2013-04-10 11:45:55 +08:00
|
|
|
|
2013-06-20 14:13:12 +08:00
|
|
|
bool FileUtilsLinux::init()
|
2013-04-10 11:45:55 +08:00
|
|
|
{
|
2018-09-17 10:47:41 +08:00
|
|
|
DECLARE_GUARD;
|
2013-04-10 18:39:01 +08:00
|
|
|
|
2023-01-10 23:06:04 +08:00
|
|
|
// application path
|
|
|
|
std::string exePath = _checkPath("/proc/self/exe");
|
|
|
|
std::string_view exePathSV{exePath};
|
|
|
|
auto slash = exePath.find_last_of('/');
|
|
|
|
assert(slash != std::string::npos);
|
|
|
|
auto exeDir = exePathSV.substr(0, slash + 1);
|
|
|
|
|
|
|
|
std::string workingDir = _checkPath("/proc/self/cwd");
|
|
|
|
workingDir += '/';
|
|
|
|
bool startedFromSelfLocation = workingDir == exeDir;
|
2023-02-23 21:57:30 +08:00
|
|
|
if (!startedFromSelfLocation || !isDirectoryExistInternal(AX_CONTENT_DIR))
|
2023-01-10 23:06:04 +08:00
|
|
|
_defaultResRootPath = workingDir;
|
|
|
|
else
|
2021-12-25 10:04:45 +08:00
|
|
|
{
|
2023-02-23 21:57:30 +08:00
|
|
|
_defaultResRootPath.reserve(exeDir.size() + AX_CONTENT_DIR_LEN);
|
|
|
|
_defaultResRootPath.append(exeDir).append(AX_CONTENT_DIR, AX_CONTENT_DIR_LEN);
|
2013-04-10 18:39:01 +08:00
|
|
|
}
|
|
|
|
|
2013-04-11 09:20:25 +08:00
|
|
|
// Set writable path to $XDG_CONFIG_HOME or ~/.config/<app name>/ if $XDG_CONFIG_HOME not exists.
|
|
|
|
const char* xdg_config_path = getenv("XDG_CONFIG_HOME");
|
|
|
|
std::string xdgConfigPath;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (xdg_config_path == NULL)
|
|
|
|
{
|
2013-04-11 09:20:25 +08:00
|
|
|
xdgConfigPath = getenv("HOME");
|
|
|
|
xdgConfigPath += "/.config";
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
xdgConfigPath = xdg_config_path;
|
2013-04-11 09:20:25 +08:00
|
|
|
}
|
2013-06-15 14:03:30 +08:00
|
|
|
_writablePath = xdgConfigPath;
|
2023-01-10 23:06:04 +08:00
|
|
|
_writablePath += exePathSV.substr(slash);
|
2013-06-15 14:03:30 +08:00
|
|
|
_writablePath += "/";
|
2013-04-10 11:45:55 +08:00
|
|
|
|
2023-01-10 23:06:04 +08:00
|
|
|
bool ret = FileUtils::init();
|
|
|
|
|
|
|
|
// make sure any path relative to exe dir can be found when app working directory location not exe path
|
|
|
|
if (!startedFromSelfLocation)
|
|
|
|
addSearchPath(exeDir);
|
|
|
|
|
|
|
|
return ret;
|
2013-04-10 11:45:55 +08:00
|
|
|
}
|
|
|
|
|
2013-09-07 13:52:23 +08:00
|
|
|
string FileUtilsLinux::getWritablePath() const
|
2021-04-22 23:00:00 +08:00
|
|
|
{
|
|
|
|
DECLARE_GUARD;
|
|
|
|
return getNativeWritableAbsolutePath();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string FileUtilsLinux::getNativeWritableAbsolutePath() const
|
2013-04-10 11:45:55 +08:00
|
|
|
{
|
2013-04-11 08:58:48 +08:00
|
|
|
struct stat st;
|
2013-06-15 14:03:30 +08:00
|
|
|
stat(_writablePath.c_str(), &st);
|
2021-12-25 10:04:45 +08:00
|
|
|
if (!S_ISDIR(st.st_mode))
|
|
|
|
{
|
2013-06-15 14:03:30 +08:00
|
|
|
mkdir(_writablePath.c_str(), 0744);
|
2013-04-11 08:58:48 +08:00
|
|
|
}
|
|
|
|
|
2013-06-15 14:03:30 +08:00
|
|
|
return _writablePath;
|
2013-04-10 11:45:55 +08:00
|
|
|
}
|
|
|
|
|
2021-12-28 19:01:28 +08:00
|
|
|
bool FileUtilsLinux::isFileExistInternal(std::string_view path) const
|
2013-04-10 11:45:55 +08:00
|
|
|
{
|
|
|
|
struct stat sts;
|
2021-12-28 19:01:28 +08:00
|
|
|
return (stat(path.data(), &sts) == 0) && S_ISREG(sts.st_mode);
|
2013-04-10 11:45:55 +08:00
|
|
|
}
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_END
|