Android: Make the handling of SAF open modes more robust
This commit is contained in:
parent
6a4ac74ec4
commit
70df5446d3
|
@ -17,7 +17,9 @@ public class ContentHandler
|
||||||
return DolphinApplication.getAppContext().getContentResolver()
|
return DolphinApplication.getAppContext().getContentResolver()
|
||||||
.openFileDescriptor(Uri.parse(uri), mode).detachFd();
|
.openFileDescriptor(Uri.parse(uri), mode).detachFd();
|
||||||
}
|
}
|
||||||
catch (FileNotFoundException | NullPointerException e)
|
// Some content providers throw IllegalArgumentException for invalid modes,
|
||||||
|
// despite the documentation saying that invalid modes result in a FileNotFoundException
|
||||||
|
catch (FileNotFoundException | IllegalArgumentException | NullPointerException e)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,6 +10,7 @@
|
||||||
|
|
||||||
#include <jni.h>
|
#include <jni.h>
|
||||||
|
|
||||||
|
#include "Common/Assert.h"
|
||||||
#include "Common/StringUtil.h"
|
#include "Common/StringUtil.h"
|
||||||
#include "jni/AndroidCommon/IDCache.h"
|
#include "jni/AndroidCommon/IDCache.h"
|
||||||
|
|
||||||
|
@ -42,21 +43,35 @@ std::vector<std::string> JStringArrayToVector(JNIEnv* env, jobjectArray array)
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool IsPathAndroidContent(const std::string& uri)
|
||||||
|
{
|
||||||
|
return StringBeginsWith(uri, "content://");
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string OpenModeToAndroid(std::string mode)
|
||||||
|
{
|
||||||
|
// The 'b' specifier is not supported. Since we're on POSIX, it's fine to just skip it.
|
||||||
|
if (!mode.empty() && mode.back() == 'b')
|
||||||
|
mode.pop_back();
|
||||||
|
|
||||||
|
if (mode == "r+")
|
||||||
|
mode = "rw";
|
||||||
|
else if (mode == "w+")
|
||||||
|
mode = "rwt";
|
||||||
|
else if (mode == "a+")
|
||||||
|
mode = "rwa";
|
||||||
|
else if (mode == "a")
|
||||||
|
mode = "wa";
|
||||||
|
|
||||||
|
return mode;
|
||||||
|
}
|
||||||
|
|
||||||
int OpenAndroidContent(const std::string& uri, const std::string& mode)
|
int OpenAndroidContent(const std::string& uri, const std::string& mode)
|
||||||
{
|
{
|
||||||
JNIEnv* env = IDCache::GetEnvForThread();
|
JNIEnv* env = IDCache::GetEnvForThread();
|
||||||
const jint fd = env->CallStaticIntMethod(IDCache::GetContentHandlerClass(),
|
return env->CallStaticIntMethod(IDCache::GetContentHandlerClass(),
|
||||||
IDCache::GetContentHandlerOpenFd(), ToJString(env, uri),
|
IDCache::GetContentHandlerOpenFd(), ToJString(env, uri),
|
||||||
ToJString(env, mode));
|
ToJString(env, mode));
|
||||||
|
|
||||||
// We can get an IllegalArgumentException when passing an invalid mode
|
|
||||||
if (env->ExceptionCheck())
|
|
||||||
{
|
|
||||||
env->ExceptionDescribe();
|
|
||||||
abort();
|
|
||||||
}
|
|
||||||
|
|
||||||
return fd;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DeleteAndroidContent(const std::string& uri)
|
bool DeleteAndroidContent(const std::string& uri)
|
||||||
|
|
|
@ -12,5 +12,14 @@ std::string GetJString(JNIEnv* env, jstring jstr);
|
||||||
jstring ToJString(JNIEnv* env, const std::string& str);
|
jstring ToJString(JNIEnv* env, const std::string& str);
|
||||||
std::vector<std::string> JStringArrayToVector(JNIEnv* env, jobjectArray array);
|
std::vector<std::string> JStringArrayToVector(JNIEnv* env, jobjectArray array);
|
||||||
|
|
||||||
|
// Returns true if the given path should be opened as Android content instead of a normal file.
|
||||||
|
bool IsPathAndroidContent(const std::string& uri);
|
||||||
|
|
||||||
|
// Turns a C/C++ style mode (e.g. "rb") into one which can be used with OpenAndroidContent.
|
||||||
|
std::string OpenModeToAndroid(std::string mode);
|
||||||
|
|
||||||
|
// Opens a given file and returns a file descriptor.
|
||||||
int OpenAndroidContent(const std::string& uri, const std::string& mode);
|
int OpenAndroidContent(const std::string& uri, const std::string& mode);
|
||||||
|
|
||||||
|
// Deletes a given file.
|
||||||
bool DeleteAndroidContent(const std::string& uri);
|
bool DeleteAndroidContent(const std::string& uri);
|
||||||
|
|
|
@ -18,7 +18,6 @@
|
||||||
#ifdef ANDROID
|
#ifdef ANDROID
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
|
||||||
#include "Common/StringUtil.h"
|
|
||||||
#include "jni/AndroidCommon/AndroidCommon.h"
|
#include "jni/AndroidCommon/AndroidCommon.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -66,24 +65,17 @@ void IOFile::Swap(IOFile& other) noexcept
|
||||||
bool IOFile::Open(const std::string& filename, const char openmode[])
|
bool IOFile::Open(const std::string& filename, const char openmode[])
|
||||||
{
|
{
|
||||||
Close();
|
Close();
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
m_good = _tfopen_s(&m_file, UTF8ToTStr(filename).c_str(), UTF8ToTStr(openmode).c_str()) == 0;
|
m_good = _tfopen_s(&m_file, UTF8ToTStr(filename).c_str(), UTF8ToTStr(openmode).c_str()) == 0;
|
||||||
#else
|
#else
|
||||||
#ifdef ANDROID
|
#ifdef ANDROID
|
||||||
if (StringBeginsWith(filename, "content://"))
|
if (IsPathAndroidContent(filename))
|
||||||
{
|
m_file = fdopen(OpenAndroidContent(filename, OpenModeToAndroid(openmode)), openmode);
|
||||||
// The Java method which OpenAndroidContent passes the mode to does not support the b specifier.
|
|
||||||
// Since we're on POSIX, it's fine to just remove the b.
|
|
||||||
std::string mode_without_b(openmode);
|
|
||||||
mode_without_b.erase(std::remove(mode_without_b.begin(), mode_without_b.end(), 'b'),
|
|
||||||
mode_without_b.end());
|
|
||||||
m_file = fdopen(OpenAndroidContent(filename, mode_without_b), mode_without_b.c_str());
|
|
||||||
}
|
|
||||||
else
|
else
|
||||||
#endif
|
#endif
|
||||||
{
|
|
||||||
m_file = std::fopen(filename.c_str(), openmode);
|
m_file = std::fopen(filename.c_str(), openmode);
|
||||||
}
|
|
||||||
m_good = m_file != nullptr;
|
m_good = m_file != nullptr;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue