mirror of
https://github.com/Lime3DS/Lime3DS.git
synced 2026-02-18 15:24:40 -07:00
313 lines
12 KiB
C++
313 lines
12 KiB
C++
// Copyright Citra Emulator Project / Azahar Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#ifdef ANDROID
|
|
#include <boost/uuid/uuid_generators.hpp>
|
|
#include <boost/uuid/uuid_io.hpp>
|
|
#include "common/android_storage.h"
|
|
#include "common/file_util.h"
|
|
#include "common/logging/log.h"
|
|
|
|
namespace AndroidStorage {
|
|
JNIEnv* GetEnvForThread() {
|
|
thread_local static struct OwnedEnv {
|
|
OwnedEnv() {
|
|
status = g_jvm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6);
|
|
if (status == JNI_EDETACHED)
|
|
g_jvm->AttachCurrentThread(&env, nullptr);
|
|
}
|
|
|
|
~OwnedEnv() {
|
|
if (status == JNI_EDETACHED)
|
|
g_jvm->DetachCurrentThread();
|
|
}
|
|
|
|
int status;
|
|
JNIEnv* env = nullptr;
|
|
} owned;
|
|
return owned.env;
|
|
}
|
|
|
|
AndroidOpenMode ParseOpenmode(const std::string_view openmode) {
|
|
AndroidOpenMode android_open_mode = AndroidOpenMode::NEVER;
|
|
const char* mode = openmode.data();
|
|
int o = 0;
|
|
switch (*mode++) {
|
|
case 'r':
|
|
android_open_mode = AndroidStorage::AndroidOpenMode::READ;
|
|
break;
|
|
case 'w':
|
|
android_open_mode = AndroidStorage::AndroidOpenMode::WRITE;
|
|
o = O_TRUNC;
|
|
break;
|
|
case 'a':
|
|
android_open_mode = AndroidStorage::AndroidOpenMode::WRITE;
|
|
o = O_APPEND;
|
|
break;
|
|
}
|
|
|
|
// [rwa]\+ or [rwa]b\+ means read and write
|
|
if (*mode == '+' || (*mode == 'b' && mode[1] == '+')) {
|
|
android_open_mode = AndroidStorage::AndroidOpenMode::READ_WRITE;
|
|
}
|
|
|
|
return android_open_mode | o;
|
|
}
|
|
|
|
void InitJNI(JNIEnv* env, jclass clazz) {
|
|
env->GetJavaVM(&g_jvm);
|
|
native_library = clazz;
|
|
|
|
#define FR(FunctionName, ReturnValue, JMethodID, Caller, JMethodName, Signature) \
|
|
F(JMethodID, JMethodName, Signature)
|
|
#define FS(FunctionName, ReturnValue, Parameters, JMethodID, JMethodName, Signature) \
|
|
F(JMethodID, JMethodName, Signature)
|
|
#define F(JMethodID, JMethodName, Signature) \
|
|
JMethodID = env->GetStaticMethodID(native_library, JMethodName, Signature);
|
|
ANDROID_SINGLE_PATH_DETERMINE_FUNCTIONS(FR)
|
|
ANDROID_STORAGE_FUNCTIONS(FS)
|
|
#undef F
|
|
#undef FS
|
|
#undef FR
|
|
}
|
|
|
|
void CleanupJNI() {
|
|
#define FR(FunctionName, ReturnValue, JMethodID, Caller, JMethodName, Signature) F(JMethodID)
|
|
#define FS(FunctionName, ReturnValue, Parameters, JMethodID, JMethodName, Signature) F(JMethodID)
|
|
#define F(JMethodID) JMethodID = nullptr;
|
|
ANDROID_SINGLE_PATH_DETERMINE_FUNCTIONS(FR)
|
|
ANDROID_STORAGE_FUNCTIONS(FS)
|
|
#undef F
|
|
#undef FS
|
|
#undef FR
|
|
}
|
|
|
|
bool CreateFile(const std::string& directory, const std::string& filename) {
|
|
if (create_file == nullptr) {
|
|
return false;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_directory = env->NewStringUTF(directory.c_str());
|
|
jstring j_filename = env->NewStringUTF(filename.c_str());
|
|
return env->CallStaticBooleanMethod(native_library, create_file, j_directory, j_filename);
|
|
}
|
|
|
|
bool CreateDir(const std::string& directory, const std::string& filename) {
|
|
if (create_dir == nullptr) {
|
|
return false;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_directory = env->NewStringUTF(directory.c_str());
|
|
jstring j_directory_name = env->NewStringUTF(filename.c_str());
|
|
return env->CallStaticBooleanMethod(native_library, create_dir, j_directory, j_directory_name);
|
|
}
|
|
|
|
int OpenContentUri(const std::string& filepath, AndroidOpenMode openmode) {
|
|
if (open_content_uri == nullptr) {
|
|
return -1;
|
|
}
|
|
|
|
const char* mode = "";
|
|
switch (openmode) {
|
|
case AndroidOpenMode::READ:
|
|
mode = "r";
|
|
break;
|
|
case AndroidOpenMode::WRITE:
|
|
mode = "w";
|
|
break;
|
|
case AndroidOpenMode::READ_WRITE:
|
|
mode = "rw";
|
|
break;
|
|
case AndroidOpenMode::WRITE_TRUNCATE:
|
|
mode = "wt";
|
|
break;
|
|
case AndroidOpenMode::WRITE_APPEND:
|
|
mode = "wa";
|
|
break;
|
|
case AndroidOpenMode::READ_WRITE_APPEND:
|
|
mode = "rwa";
|
|
break;
|
|
case AndroidOpenMode::READ_WRITE_TRUNCATE:
|
|
mode = "rwt";
|
|
break;
|
|
case AndroidOpenMode::NEVER:
|
|
return -1;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_filepath = env->NewStringUTF(filepath.c_str());
|
|
jstring j_mode = env->NewStringUTF(mode);
|
|
return env->CallStaticIntMethod(native_library, open_content_uri, j_filepath, j_mode);
|
|
}
|
|
|
|
std::vector<std::string> GetFilesName(const std::string& filepath) {
|
|
auto vector = std::vector<std::string>();
|
|
if (get_files_name == nullptr) {
|
|
return vector;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_filepath = env->NewStringUTF(filepath.c_str());
|
|
auto j_object =
|
|
(jobjectArray)env->CallStaticObjectMethod(native_library, get_files_name, j_filepath);
|
|
const jsize j_size = env->GetArrayLength(j_object);
|
|
vector.reserve(j_size);
|
|
for (int i = 0; i < j_size; i++) {
|
|
auto string = (jstring)(env->GetObjectArrayElement(j_object, i));
|
|
vector.emplace_back(env->GetStringUTFChars(string, nullptr));
|
|
}
|
|
return vector;
|
|
}
|
|
|
|
std::optional<std::string> GetUserDirectory() {
|
|
if (get_user_directory == nullptr) {
|
|
throw std::runtime_error(
|
|
"Unable to locate user directory: Function with ID 'get_user_directory' is missing");
|
|
}
|
|
auto env = GetEnvForThread();
|
|
auto j_user_directory =
|
|
(jstring)(env->CallStaticObjectMethod(native_library, get_user_directory, nullptr));
|
|
auto result = env->GetStringUTFChars(j_user_directory, nullptr);
|
|
if (result == "") {
|
|
return std::nullopt;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
std::string GetBuildFlavor() {
|
|
if (get_build_flavor == nullptr) {
|
|
throw std::runtime_error(
|
|
"Unable get build flavor: Function with ID 'get_build_flavor' is missing");
|
|
}
|
|
auto env = GetEnvForThread();
|
|
const auto jflavor =
|
|
(jstring)(env->CallStaticObjectMethod(native_library, get_build_flavor, nullptr));
|
|
return env->GetStringUTFChars(jflavor, nullptr);
|
|
}
|
|
|
|
bool CopyFile(const std::string& source, const std::string& destination_path,
|
|
const std::string& destination_filename) {
|
|
if (copy_file == nullptr) {
|
|
return false;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_source_path = env->NewStringUTF(source.c_str());
|
|
jstring j_destination_path = env->NewStringUTF(destination_path.c_str());
|
|
jstring j_destination_filename = env->NewStringUTF(destination_filename.c_str());
|
|
return env->CallStaticBooleanMethod(native_library, copy_file, j_source_path,
|
|
j_destination_path, j_destination_filename);
|
|
}
|
|
|
|
bool RenameFile(const std::string& source, const std::string& filename) {
|
|
if (rename_file == nullptr) {
|
|
return false;
|
|
}
|
|
if (std::string(FileUtil::GetFilename(source)) ==
|
|
std::string(FileUtil::GetFilename(filename))) {
|
|
// TODO: Should this be treated as a success or failure?
|
|
return false;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_source_path = env->NewStringUTF(source.c_str());
|
|
jstring j_destination_path = env->NewStringUTF(filename.c_str());
|
|
return env->CallStaticBooleanMethod(native_library, rename_file, j_source_path,
|
|
j_destination_path);
|
|
}
|
|
|
|
bool UpdateDocumentLocation(const std::string& source_path, const std::string& destination_path) {
|
|
if (update_document_location == nullptr) {
|
|
return false;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_source_path = env->NewStringUTF(source_path.c_str());
|
|
jstring j_destination_path = env->NewStringUTF(destination_path.c_str());
|
|
return env->CallStaticBooleanMethod(native_library, update_document_location, j_source_path,
|
|
j_destination_path);
|
|
}
|
|
|
|
bool MoveFile(const std::string& filename, const std::string& source_dir_path,
|
|
const std::string& destination_dir_path) {
|
|
if (move_file == nullptr) {
|
|
return false;
|
|
}
|
|
if (source_dir_path == destination_dir_path) {
|
|
// TODO: Should this be treated as a success or failure?
|
|
return false;
|
|
}
|
|
auto env = GetEnvForThread();
|
|
jstring j_filename = env->NewStringUTF(filename.c_str());
|
|
jstring j_source_dir_path = env->NewStringUTF(source_dir_path.c_str());
|
|
jstring j_destination_dir_path = env->NewStringUTF(destination_dir_path.c_str());
|
|
return env->CallStaticBooleanMethod(native_library, move_file, j_filename, j_source_dir_path,
|
|
j_destination_dir_path);
|
|
}
|
|
|
|
bool MoveAndRenameFile(const std::string& src_full_path, const std::string& dest_full_path) {
|
|
if (src_full_path == dest_full_path) {
|
|
// TODO: Should this be treated as a success or failure?
|
|
return false;
|
|
}
|
|
const auto src_filename = std::string(FileUtil::GetFilename(src_full_path));
|
|
const auto src_parent_path = std::string(FileUtil::GetParentPath(src_full_path));
|
|
const auto dest_filename = std::string(FileUtil::GetFilename(dest_full_path));
|
|
const auto dest_parent_path = std::string(FileUtil::GetParentPath(dest_full_path));
|
|
bool result;
|
|
|
|
const std::string tmp_path = "/tmp";
|
|
AndroidStorage::CreateDir("/", "tmp");
|
|
|
|
// If a simultaneous move and rename are not necessary, use individual methods
|
|
if (src_filename == dest_filename || src_parent_path == dest_parent_path) {
|
|
if (src_filename != dest_filename) {
|
|
return AndroidStorage::RenameFile(src_full_path, dest_filename);
|
|
} else if (src_parent_path != dest_parent_path) {
|
|
return AndroidStorage::MoveFile(src_filename, src_parent_path, dest_parent_path);
|
|
}
|
|
}
|
|
|
|
// Step 1: Create directory named after UUID inside /tmp to house the moved file.
|
|
// This prevents clashes if files with the same name are moved simultaneously.
|
|
const auto uuid = boost::uuids::to_string(boost::uuids::time_generator_v7()());
|
|
const auto allocated_tmp_path = tmp_path + "/" + uuid;
|
|
AndroidStorage::CreateDir(tmp_path, uuid);
|
|
|
|
// Step 2: Attempt to move to allocated temporary directory.
|
|
// If this step fails, skip everything except the cleanup.
|
|
result = AndroidStorage::MoveFile(src_filename, src_parent_path, allocated_tmp_path);
|
|
if (result == true) {
|
|
// Step 3: Rename to desired file name.
|
|
AndroidStorage::RenameFile((allocated_tmp_path + "/" + src_filename), dest_filename);
|
|
|
|
// Step 4: If a file with the desired name in the destination exists, remove it.
|
|
AndroidStorage::DeleteDocument(dest_full_path);
|
|
|
|
// Step 5: Attempt to move file to desired location.
|
|
// If this step fails, move the file back to where it came from.
|
|
result = AndroidStorage::MoveFile(dest_filename, allocated_tmp_path, dest_parent_path);
|
|
if (result == false) {
|
|
AndroidStorage::MoveAndRenameFile((allocated_tmp_path + "/" + dest_filename),
|
|
src_full_path);
|
|
}
|
|
}
|
|
// Step 6: Clean up the allocated temp directory.
|
|
AndroidStorage::DeleteDocument(allocated_tmp_path);
|
|
return result;
|
|
}
|
|
|
|
#define FR(FunctionName, ReturnValue, JMethodID, Caller, JMethodName, Signature) \
|
|
F(FunctionName, ReturnValue, JMethodID, Caller)
|
|
#define F(FunctionName, ReturnValue, JMethodID, Caller) \
|
|
ReturnValue FunctionName(const std::string& filepath) { \
|
|
if (JMethodID == nullptr) { \
|
|
return 0; \
|
|
} \
|
|
auto env = GetEnvForThread(); \
|
|
jstring j_filepath = env->NewStringUTF(filepath.c_str()); \
|
|
return env->Caller(native_library, JMethodID, j_filepath); \
|
|
}
|
|
ANDROID_SINGLE_PATH_DETERMINE_FUNCTIONS(FR)
|
|
#undef F
|
|
#undef FR
|
|
|
|
} // namespace AndroidStorage
|
|
#endif
|