// Copyright 2018 Dolphin Emulator Project // SPDX-License-Identifier: GPL-2.0-or-later #include "jni/AndroidCommon/AndroidCommon.h" #include #include #include #include #include #include #include "Common/Assert.h" #include "Common/Logging/Log.h" #include "Common/StringUtil.h" #include "jni/AndroidCommon/IDCache.h" std::string GetJString(JNIEnv* env, jstring jstr) { const jchar* jchars = env->GetStringChars(jstr, nullptr); const jsize length = env->GetStringLength(jstr); const std::u16string_view string_view(reinterpret_cast(jchars), length); std::string converted_string = UTF16ToUTF8(string_view); env->ReleaseStringChars(jstr, jchars); return converted_string; } jstring ToJString(JNIEnv* env, std::string_view str) { const std::u16string converted_string = UTF8ToUTF16(str); return env->NewString(reinterpret_cast(converted_string.data()), converted_string.size()); } std::vector JStringArrayToVector(JNIEnv* env, jobjectArray array) { const jsize size = env->GetArrayLength(array); std::vector result; result.reserve(size); for (jsize i = 0; i < size; ++i) { jstring str = reinterpret_cast(env->GetObjectArrayElement(array, i)); result.push_back(GetJString(env, str)); env->DeleteLocalRef(str); } return result; } jobjectArray VectorToJStringArray(JNIEnv* env, const std::vector& vector) { return VectorToJObjectArray(env, vector, IDCache::GetStringClass(), ToJString); } bool IsPathAndroidContent(std::string_view uri) { return uri.starts_with("content://"); } std::string OpenModeToAndroid(std::string mode) { // The 'b' specifier is not supported by Android. Since we're on POSIX, it's fine to just skip it. std::erase(mode, 'b'); if (mode == "r") return "r"; else if (mode == "w") return "wt"; else if (mode == "a") return "wa"; else if (mode == "r+") return "rw"; else if (mode == "w+") return "rwt"; else if (mode == "a+") return "rwa"; ERROR_LOG_FMT(COMMON, "OpenModeToAndroid(std::string): Unsupported open mode: {}", mode); return ""; } std::string OpenModeToAndroid(std::ios_base::openmode mode) { // The 'b' specifier is not supported by Android. Since we're on POSIX, it's fine to just skip it. mode &= ~std::ios_base::binary; switch (mode) { case std::ios_base::in: return "r"; case std::ios_base::out: case std::ios_base::out | std::ios_base::trunc: return "wt"; case std::ios_base::app: case std::ios_base::out | std::ios_base::app: return "wa"; case std::ios_base::in | std::ios_base::out: return "rw"; case std::ios_base::in | std::ios_base::out | std::ios_base::trunc: return "rwt"; case std::ios_base::in | std::ios_base::app: case std::ios_base::in | std::ios_base::out | std::ios_base::app: return "rwa"; default: ERROR_LOG_FMT(COMMON, "OpenModeToAndroid(std::ios_base::openmode): Unsupported open mode: {:#x}", mode); return ""; } } int OpenAndroidContent(std::string_view uri, std::string_view mode) { JNIEnv* env = IDCache::GetEnvForThread(); jstring j_uri = ToJString(env, uri); jstring j_mode = ToJString(env, mode); jint result = env->CallStaticIntMethod(IDCache::GetContentHandlerClass(), IDCache::GetContentHandlerOpenFd(), j_uri, j_mode); env->DeleteLocalRef(j_uri); env->DeleteLocalRef(j_mode); return result; } bool DeleteAndroidContent(std::string_view uri) { JNIEnv* env = IDCache::GetEnvForThread(); jstring j_uri = ToJString(env, uri); jboolean result = env->CallStaticBooleanMethod(IDCache::GetContentHandlerClass(), IDCache::GetContentHandlerDelete(), j_uri); env->DeleteLocalRef(j_uri); return static_cast(result); } jlong GetAndroidContentSizeAndIsDirectory(std::string_view uri) { JNIEnv* env = IDCache::GetEnvForThread(); jstring j_uri = ToJString(env, uri); jlong result = env->CallStaticLongMethod( IDCache::GetContentHandlerClass(), IDCache::GetContentHandlerGetSizeAndIsDirectory(), j_uri); env->DeleteLocalRef(j_uri); return result; } std::string GetAndroidContentDisplayName(std::string_view uri) { JNIEnv* env = IDCache::GetEnvForThread(); jstring j_uri = ToJString(env, uri); jstring j_result = reinterpret_cast(env->CallStaticObjectMethod( IDCache::GetContentHandlerClass(), IDCache::GetContentHandlerGetDisplayName(), j_uri)); env->DeleteLocalRef(j_uri); if (!j_result) return ""; std::string result = GetJString(env, j_result); env->DeleteLocalRef(j_result); return result; } std::vector GetAndroidContentChildNames(std::string_view uri) { JNIEnv* env = IDCache::GetEnvForThread(); jstring j_uri = ToJString(env, uri); jobjectArray j_result = reinterpret_cast(env->CallStaticObjectMethod( IDCache::GetContentHandlerClass(), IDCache::GetContentHandlerGetChildNames(), j_uri, false)); std::vector result = JStringArrayToVector(env, j_result); env->DeleteLocalRef(j_uri); env->DeleteLocalRef(j_result); return result; } std::vector DoFileSearchAndroidContent(std::string_view directory, const std::vector& extensions, bool recursive) { JNIEnv* env = IDCache::GetEnvForThread(); jstring j_directory = ToJString(env, directory); jobjectArray j_extensions = VectorToJStringArray(env, extensions); jobjectArray j_result = reinterpret_cast(env->CallStaticObjectMethod( IDCache::GetContentHandlerClass(), IDCache::GetContentHandlerDoFileSearch(), j_directory, j_extensions, recursive)); std::vector result = JStringArrayToVector(env, j_result); env->DeleteLocalRef(j_directory); env->DeleteLocalRef(j_extensions); env->DeleteLocalRef(j_result); return result; } int GetNetworkIpAddress() { JNIEnv* env = IDCache::GetEnvForThread(); return env->CallStaticIntMethod(IDCache::GetNetworkHelperClass(), IDCache::GetNetworkHelperGetNetworkIpAddress()); } int GetNetworkPrefixLength() { JNIEnv* env = IDCache::GetEnvForThread(); return env->CallStaticIntMethod(IDCache::GetNetworkHelperClass(), IDCache::GetNetworkHelperGetNetworkPrefixLength()); } int GetNetworkGateway() { JNIEnv* env = IDCache::GetEnvForThread(); return env->CallStaticIntMethod(IDCache::GetNetworkHelperClass(), IDCache::GetNetworkHelperGetNetworkGateway()); }