You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
269 lines
8.4 KiB
269 lines
8.4 KiB
/****************************************************************************
|
|
Copyright (c) 2022-2023 Xiamen Yaji Software Co., Ltd.
|
|
|
|
http://www.cocos.com
|
|
|
|
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.
|
|
****************************************************************************/
|
|
|
|
#include "platform/ohos/FileUtils-ohos.h"
|
|
#include <hilog/log.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/syscall.h>
|
|
#include <unistd.h>
|
|
#include <cstdio>
|
|
#include <regex>
|
|
#include "base/Log.h"
|
|
#include "base/memory/Memory.h"
|
|
#include "base/std/container/string.h"
|
|
#include "platform/java/jni/JniHelper.h"
|
|
|
|
#define ASSETS_FOLDER_NAME "@assets/"
|
|
|
|
#ifndef JCLS_HELPER
|
|
#define JCLS_HELPER "com/cocos/lib/CocosHelper"
|
|
#endif
|
|
|
|
namespace cc {
|
|
|
|
ResourceManager *FileUtilsOHOS::ohosResourceMgr = {};
|
|
ccstd::string FileUtilsOHOS::ohosAssetPath = {};
|
|
|
|
namespace {
|
|
|
|
ccstd::string rawfilePrefix = "entry/resources/rawfile/";
|
|
|
|
void printRawfiles(ResourceManager *mgr, const ccstd::string &path) {
|
|
auto *file = OpenRawFile(mgr, path.c_str());
|
|
if (file) {
|
|
HILOG_DEBUG(LOG_APP, "PrintRawfile %{public}s", path.c_str());
|
|
return;
|
|
}
|
|
|
|
RawDir *dir = OpenRawDir(mgr, path.c_str());
|
|
if (dir) {
|
|
auto fileCnt = GetRawFileCount(dir);
|
|
for (auto i = 0; i < fileCnt; i++) {
|
|
ccstd::string subFile = GetRawFileName(dir, i);
|
|
auto newPath = path + "/" + subFile; // NOLINT
|
|
auto debugPtr = newPath.c_str();
|
|
HILOG_ERROR(LOG_APP, " find path %{public}s", newPath.c_str());
|
|
printRawfiles(mgr, newPath);
|
|
}
|
|
} else {
|
|
HILOG_ERROR(LOG_APP, "Invalidate path %{public}s", path.c_str());
|
|
}
|
|
}
|
|
} // namespace
|
|
|
|
FileUtils *createFileUtils() {
|
|
return ccnew FileUtilsOHOS();
|
|
}
|
|
|
|
bool FileUtilsOHOS::initResourceManager(ResourceManager *mgr, const ccstd::string &assetPath, const ccstd::string &moduleName) {
|
|
CC_ASSERT(mgr);
|
|
ohosResourceMgr = mgr;
|
|
if (!assetPath.empty() && assetPath[assetPath.length() - 1] != '/') {
|
|
ohosAssetPath = assetPath + "/";
|
|
} else {
|
|
ohosAssetPath = assetPath;
|
|
}
|
|
if (!moduleName.empty()) {
|
|
setRawfilePrefix(moduleName + "/resources/rawfile/");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void FileUtilsOHOS::setRawfilePrefix(const ccstd::string &prefix) {
|
|
rawfilePrefix = prefix;
|
|
}
|
|
|
|
ResourceManager *FileUtilsOHOS::getResourceManager() {
|
|
return ohosResourceMgr;
|
|
}
|
|
|
|
FileUtilsOHOS::FileUtilsOHOS() {
|
|
init();
|
|
}
|
|
|
|
bool FileUtilsOHOS::init() {
|
|
_defaultResRootPath = ASSETS_FOLDER_NAME;
|
|
return FileUtils::init();
|
|
}
|
|
|
|
FileUtils::Status FileUtilsOHOS::getContents(const ccstd::string &filename, ResizableBuffer *buffer) {
|
|
if (filename.empty()) {
|
|
return FileUtils::Status::NOT_EXISTS;
|
|
}
|
|
|
|
ccstd::string fullPath = fullPathForFilename(filename);
|
|
if (fullPath.empty()) {
|
|
return FileUtils::Status::NOT_EXISTS;
|
|
}
|
|
|
|
if (fullPath[0] == '/') {
|
|
return FileUtils::getContents(fullPath, buffer);
|
|
}
|
|
|
|
ccstd::string relativePath;
|
|
size_t position = fullPath.find(ASSETS_FOLDER_NAME);
|
|
if (0 == position) {
|
|
// "@assets/" is at the beginning of the path and we don't want it
|
|
relativePath = rawfilePrefix + fullPath.substr(strlen(ASSETS_FOLDER_NAME));
|
|
} else {
|
|
relativePath = fullPath;
|
|
}
|
|
|
|
if (nullptr == ohosResourceMgr) {
|
|
HILOG_ERROR(LOG_APP, "... FileUtilsAndroid::assetmanager is nullptr");
|
|
return FileUtils::Status::NOT_INITIALIZED;
|
|
}
|
|
|
|
RawFile *asset = OpenRawFile(ohosResourceMgr, relativePath.c_str());
|
|
if (nullptr == asset) {
|
|
HILOG_DEBUG(LOG_APP, "asset (%{public}s) is nullptr", filename.c_str());
|
|
return FileUtils::Status::OPEN_FAILED;
|
|
}
|
|
|
|
auto size = GetRawFileSize(asset);
|
|
buffer->resize(size);
|
|
|
|
assert(buffer->buffer());
|
|
|
|
int readsize = ReadRawFile(asset, buffer->buffer(), size);
|
|
CloseRawFile(asset);
|
|
|
|
// TODO(unknown): read error
|
|
if (readsize < size) {
|
|
if (readsize >= 0) {
|
|
buffer->resize(readsize);
|
|
}
|
|
return FileUtils::Status::READ_FAILED;
|
|
}
|
|
|
|
return FileUtils::Status::OK;
|
|
}
|
|
|
|
bool FileUtilsOHOS::isAbsolutePath(const ccstd::string &strPath) const {
|
|
return !strPath.empty() && (strPath[0] == '/' || strPath.find(ASSETS_FOLDER_NAME) == 0);
|
|
}
|
|
|
|
ccstd::string FileUtilsOHOS::getWritablePath() const {
|
|
auto tmp = cc::JniHelper::callStaticStringMethod(JCLS_HELPER, "getWritablePath");
|
|
if (tmp.empty()) {
|
|
return "./";
|
|
}
|
|
return tmp.append("/");
|
|
}
|
|
|
|
bool FileUtilsOHOS::isFileExistInternal(const ccstd::string &strFilePath) const {
|
|
if (strFilePath.empty()) return false;
|
|
auto filePath = strFilePath;
|
|
auto fileFound = false;
|
|
|
|
if (strFilePath[0] == '/') { // absolute path
|
|
struct stat info;
|
|
return ::stat(filePath.c_str(), &info) == 0;
|
|
}
|
|
|
|
// relative path
|
|
if (strFilePath.find(_defaultResRootPath) == 0) {
|
|
filePath = rawfilePrefix + filePath.substr(_defaultResRootPath.length());
|
|
}
|
|
|
|
auto rawFile = OpenRawFile(ohosResourceMgr, filePath.c_str());
|
|
if (rawFile != nullptr) {
|
|
CloseRawFile(rawFile);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool FileUtilsOHOS::isDirectoryExistInternal(const ccstd::string &dirPath) const {
|
|
if (dirPath.empty()) return false;
|
|
ccstd::string dirPathMf = dirPath[dirPath.length() - 1] == '/' ? dirPath.substr(0, dirPath.length() - 1) : dirPath;
|
|
|
|
if (dirPathMf[0] == '/') {
|
|
struct stat st;
|
|
return stat(dirPathMf.c_str(), &st) == 0 && S_ISDIR(st.st_mode);
|
|
}
|
|
|
|
if (dirPathMf.find(_defaultResRootPath) == 0) {
|
|
dirPathMf = rawfilePrefix + dirPathMf.substr(_defaultResRootPath.length(), dirPathMf.length());
|
|
}
|
|
assert(ohosResourceMgr);
|
|
auto dir = OpenRawDir(ohosResourceMgr, dirPathMf.c_str());
|
|
if (dir != nullptr) {
|
|
CloseRawDir(dir);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
ccstd::string FileUtilsOHOS::expandPath(const ccstd::string &input, bool *isRawFile) const {
|
|
if (!input.empty() && input[0] == '/') {
|
|
if (isRawFile) *isRawFile = false;
|
|
return input;
|
|
}
|
|
const auto fullpath = fullPathForFilename(input);
|
|
|
|
if (fullpath.find(_defaultResRootPath) == 0) {
|
|
if (isRawFile) *isRawFile = true;
|
|
return rawfilePrefix + fullpath.substr(_defaultResRootPath.length(), fullpath.length());
|
|
}
|
|
|
|
if (isRawFile) *isRawFile = false;
|
|
|
|
return fullpath;
|
|
}
|
|
|
|
std::pair<int, std::function<void()>> FileUtilsOHOS::getFd(const ccstd::string &path) const {
|
|
bool isRawFile = false;
|
|
const auto fullpath = expandPath(path, &isRawFile);
|
|
if (isRawFile) {
|
|
RawFile *rf = OpenRawFile(ohosResourceMgr, fullpath.c_str());
|
|
// FIXME: try reuse file
|
|
const auto bufSize = GetRawFileSize(rf);
|
|
auto fileCache = ccstd::vector<char>(bufSize);
|
|
auto *buf = fileCache.data();
|
|
// Fill buffer
|
|
const auto readBytes = ReadRawFile(rf, buf, bufSize);
|
|
assert(readBytes == bufSize); // read failure ?
|
|
auto fd = syscall(__NR_memfd_create, fullpath.c_str(), 0);
|
|
{
|
|
auto writeBytes = ::write(fd, buf, bufSize); // Write can fail?
|
|
assert(writeBytes == bufSize);
|
|
::lseek(fd, 0, SEEK_SET);
|
|
}
|
|
if (errno != 0) {
|
|
const auto *errMsg = strerror(errno);
|
|
CC_LOG_ERROR("failed to open buffer fd %s", errMsg);
|
|
}
|
|
return std::make_pair(fd, [fd]() {
|
|
close(fd);
|
|
});
|
|
}
|
|
|
|
FILE *fp = fopen(fullpath.c_str(), "rb");
|
|
return std::make_pair(fileno(fp), [fp]() {
|
|
fclose(fp);
|
|
});
|
|
}
|
|
|
|
} // namespace cc
|