| Left: | ||
| Right: |
| LEFT | RIGHT |
|---|---|
| 1 /* | 1 /* |
| 2 * This file is part of Adblock Plus <https://adblockplus.org/>, | 2 * This file is part of Adblock Plus <https://adblockplus.org/>, |
| 3 * Copyright (C) 2006-2017 eyeo GmbH | 3 * Copyright (C) 2006-2017 eyeo GmbH |
| 4 * | 4 * |
| 5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
| 6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
| 7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
| 8 * | 8 * |
| 9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
| 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 12 * GNU General Public License for more details. | 12 * GNU General Public License for more details. |
| 13 * | 13 * |
| 14 * You should have received a copy of the GNU General Public License | 14 * You should have received a copy of the GNU General Public License |
| 15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. | 15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. |
| 16 */ | 16 */ |
| 17 | 17 |
| 18 #include "JniCallbacks.h" | 18 #include "JniCallbacks.h" |
| 19 #include "AdblockPlus/FileSystem.h" | 19 #include "AdblockPlus/FileSystem.h" |
| 20 #include "Utils.h" | 20 #include "Utils.h" |
| 21 #include <sstream> | 21 #include <istream> |
| 22 #include <streambuf> | |
| 23 | |
| 24 class RuntimeErrorWithErrno : public std::runtime_error | |
| 25 { | |
| 26 public: | |
| 27 explicit RuntimeErrorWithErrno(const std::string& message) | |
| 28 : std::runtime_error(message + " (" + strerror(errno) + ")") | |
| 29 { | |
| 30 } | |
| 31 }; | |
| 32 | |
| 33 struct membuf: std::streambuf { | |
| 34 membuf(char const* base, size_t size) { | |
| 35 char* p(const_cast<char*>(base)); | |
| 36 this->setg(p, p, p + size); | |
| 37 } | |
| 38 }; | |
| 39 | |
| 40 struct imemstream: virtual membuf, std::istream { | |
| 41 imemstream(char const* base, size_t size) | |
| 42 : membuf(base, size) | |
| 43 , std::istream(static_cast<std::streambuf*>(this)) { | |
| 44 } | |
| 45 }; | |
|
sergei
2017/08/07 12:50:00
All these things should be in the anonymous namesp
| |
| 22 | 46 |
| 23 // precached in JNI_OnLoad and released in JNI_OnUnload | 47 // precached in JNI_OnLoad and released in JNI_OnUnload |
| 24 JniGlobalReference<jclass>* statResultClass; | 48 JniGlobalReference<jclass>* statResultClass; |
| 25 jmethodID existsMethod; | 49 jmethodID existsMethod; |
| 26 jmethodID isDirectoryMethod; | 50 jmethodID isDirectoryMethod; |
| 27 jmethodID isFileMethod; | 51 jmethodID isFileMethod; |
| 28 jmethodID getLastModifiedMethod; | 52 jmethodID getLastModifiedMethod; |
| 29 | 53 |
| 30 void JniFileSystem_OnLoad(JavaVM* vm, JNIEnv* env, void* reserved) | 54 void JniFileSystem_OnLoad(JavaVM* vm, JNIEnv* env, void* reserved) |
| 31 { | 55 { |
| 32 statResultClass = new JniGlobalReference<jclass>(env, env->FindClass(PKG("File System$StatResult"))); | 56 statResultClass = new JniGlobalReference<jclass>(env, env->FindClass(PKG("File System$StatResult"))); |
| 33 existsMethod = env->GetMethodID(statResultClass->Get(), "exists", "()Z"); | 57 existsMethod = env->GetMethodID(statResultClass->Get(), "exists", "()Z"); |
| 34 isDirectoryMethod = env->GetMethodID(statResultClass->Get(), "isDirectory", "( )Z"); | 58 isDirectoryMethod = env->GetMethodID(statResultClass->Get(), "isDirectory", "( )Z"); |
| 35 isFileMethod = env->GetMethodID(statResultClass->Get(), "isFile", "()Z"); | 59 isFileMethod = env->GetMethodID(statResultClass->Get(), "isFile", "()Z"); |
| 36 getLastModifiedMethod = env->GetMethodID(statResultClass->Get(), "getLastModif ied", "()J"); | 60 getLastModifiedMethod = env->GetMethodID(statResultClass->Get(), "getLastModif ied", "()J"); |
| 37 } | 61 } |
| 38 | 62 |
| 39 void JniFileSystem_OnUnload(JavaVM* vm, JNIEnv* env, void* reserved) | 63 void JniFileSystem_OnUnload(JavaVM* vm, JNIEnv* env, void* reserved) |
| 40 { | 64 { |
| 41 if (statResultClass) | 65 if (statResultClass) |
| 42 { | 66 { |
| 43 delete statResultClass; | 67 delete statResultClass; |
| 44 statResultClass = NULL; | 68 statResultClass = NULL; |
|
sergei
2017/08/07 12:50:00
nullptr?
| |
| 45 } | 69 } |
| 46 } | 70 } |
| 47 | 71 |
| 48 static jlong JNICALL JniCtor(JNIEnv* env, jclass clazz, jobject callbackObject) | 72 static jlong JNICALL JniCtor(JNIEnv* env, jclass clazz, jobject callbackObject) |
|
sergei
2017/08/07 12:49:59
I think that it should be done a little bit differ
| |
| 49 { | 73 { |
| 50 try | 74 try |
| 51 { | 75 { |
| 52 return JniPtrToLong(new JniFileSystemCallback(env, callbackObject)); | 76 return JniPtrToLong(new AdblockPlus::FileSystemPtr(new JniFileSystemCallback (env, callbackObject))); |
| 53 } | 77 } |
| 54 CATCH_THROW_AND_RETURN(env, 0) | 78 CATCH_THROW_AND_RETURN(env, 0) |
| 55 } | 79 } |
| 56 | 80 |
| 57 static void JNICALL JniDtor(JNIEnv* env, jclass clazz, jlong ptr) | 81 static void JNICALL JniDtor(JNIEnv* env, jclass clazz, jlong ptr) |
| 58 { | 82 { |
| 59 delete JniLongToTypePtr<JniFileSystemCallback>(ptr); | 83 delete JniLongToTypePtr<AdblockPlus::FileSystemPtr>(ptr); |
| 60 } | 84 } |
| 61 | 85 |
| 62 JniFileSystemCallback::JniFileSystemCallback(JNIEnv* env, jobject callbackObject ) | 86 JniFileSystemCallback::JniFileSystemCallback(JNIEnv* env, jobject callbackObject ) |
| 63 : JniCallbackBase(env, callbackObject), | 87 : JniCallbackBase(env, callbackObject) |
| 64 AdblockPlus::FileSystem() | |
| 65 { | 88 { |
| 66 } | 89 } |
| 67 | 90 |
| 68 std::shared_ptr<std::istream> JniFileSystemCallback::Read(const std::string& pat h) const | 91 std::shared_ptr<std::istream> JniFileSystemCallback::Read(const std::string& pat h) const |
| 69 { | 92 { |
| 70 JNIEnvAcquire env(GetJavaVM()); | 93 JNIEnvAcquire env(GetJavaVM()); |
| 71 | 94 |
| 72 jmethodID method = env->GetMethodID( | 95 jmethodID method = env->GetMethodID( |
| 73 *JniLocalReference<jclass>(*env, env->GetObjectClass(GetCallbackObject())), | 96 *JniLocalReference<jclass>(*env, env->GetObjectClass(GetCallbackObject())), |
| 74 "read", | 97 "read", |
| 75 "(Ljava/lang/String;)Ljava/lang/String;"); | 98 "(Ljava/lang/String;)[B"); |
| 76 | 99 |
| 77 JniLocalReference<jstring> jPath(*env, env->NewStringUTF(path.c_str())); | 100 JniLocalReference<jstring> jPath(*env, env->NewStringUTF(path.c_str())); |
| 78 jstring jData = (jstring)env->CallObjectMethod(GetCallbackObject(), method, *j Path); | 101 jbyteArray jData = (jbyteArray)env->CallObjectMethod(GetCallbackObject(), meth od, *jPath); |
| 79 CheckAndLogJavaException(*env); | 102 if (CheckAndLogJavaException(*env)) |
| 80 | 103 throw new RuntimeErrorWithErrno("Failed to open file (File not found)"); |
| 81 if (!jData) | 104 |
| 82 return NULL; | 105 int dataLength = env->GetArrayLength(jData); |
| 83 | 106 char* cData = new char[dataLength]; |
| 84 std::string cData = JniJavaToStdString(*env, jData); | 107 env->GetByteArrayRegion(jData, 0, dataLength, reinterpret_cast<jbyte*>(cData)) ; |
| 85 std::shared_ptr<std::istream> cSharedStream(new std::istringstream(cData)); | 108 |
| 109 std::shared_ptr<std::istream> cSharedStream(new imemstream(cData, dataLength)) ; | |
| 86 return cSharedStream; | 110 return cSharedStream; |
| 87 } | 111 } |
| 88 | 112 |
| 89 void JniFileSystemCallback::Write(const std::string& path, std::shared_ptr<std:: istream> data) | 113 void JniFileSystemCallback::Write(const std::string& path, std::shared_ptr<std:: istream> dataStreamPtr) |
| 90 { | 114 { |
| 91 JNIEnvAcquire env(GetJavaVM()); | 115 JNIEnvAcquire env(GetJavaVM()); |
| 92 | 116 |
| 93 jmethodID method = env->GetMethodID( | 117 jmethodID method = env->GetMethodID( |
| 94 *JniLocalReference<jclass>(*env, env->GetObjectClass(GetCallbackObject())), | 118 *JniLocalReference<jclass>(*env, env->GetObjectClass(GetCallbackObject())), |
| 95 "write", | 119 "write", |
| 96 "(Ljava/lang/String;Ljava/lang/String;)V"); | 120 "(Ljava/lang/String;[B)V"); |
| 97 | 121 |
| 98 JniLocalReference<jstring> jPath(*env, env->NewStringUTF(path.c_str())); | 122 JniLocalReference<jstring> jPath(*env, env->NewStringUTF(path.c_str())); |
| 99 | 123 |
| 100 // read all the data from the stream into buffer (no appropriate way to pass s treams over JNI) | 124 // read all the data from the stream into buffer (no appropriate way to pass s treams over JNI) |
| 101 std::string cData = JniStdStreamToStdString(data.get()); | 125 std::istream* dataStream = dataStreamPtr.get(); |
| 102 JniLocalReference<jstring> jData(*env, env->NewStringUTF(cData.c_str())); | 126 dataStream->seekg(0, std::ios::end); |
| 103 | 127 int dataLength = dataStream->tellg(); |
| 104 env->CallVoidMethod(GetCallbackObject(), method, *jPath, *jData); | 128 char* cData = new char[dataLength]; |
| 105 CheckAndLogJavaException(*env); | 129 dataStream->seekg(0, std::ios::beg); |
| 130 dataStream->read(cData, dataLength); | |
| 131 | |
| 132 jbyteArray jData = env->NewByteArray(dataLength); | |
| 133 env->SetByteArrayRegion(jData, 0, dataLength, reinterpret_cast<jbyte*>(cData)) ; | |
| 134 | |
| 135 env->CallVoidMethod(GetCallbackObject(), method, *jPath, jData); | |
| 136 CheckAndLogJavaException(*env); | |
| 137 delete[] cData; | |
| 106 } | 138 } |
| 107 | 139 |
| 108 void JniFileSystemCallback::Move(const std::string& fromPath, const std::string& toPath) | 140 void JniFileSystemCallback::Move(const std::string& fromPath, const std::string& toPath) |
| 109 { | 141 { |
| 110 JNIEnvAcquire env(GetJavaVM()); | 142 JNIEnvAcquire env(GetJavaVM()); |
| 111 | 143 |
| 112 jmethodID method = env->GetMethodID( | 144 jmethodID method = env->GetMethodID( |
| 113 *JniLocalReference<jclass>(*env, env->GetObjectClass(GetCallbackObject())), | 145 *JniLocalReference<jclass>(*env, env->GetObjectClass(GetCallbackObject())), |
| 114 "move", | 146 "move", |
| 115 "(Ljava/lang/String;Ljava/lang/String;)V"); | 147 "(Ljava/lang/String;Ljava/lang/String;)V"); |
| (...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 187 static JNINativeMethod methods[] = | 219 static JNINativeMethod methods[] = |
| 188 { | 220 { |
| 189 { (char*)"ctor", (char*)"(Ljava/lang/Object;)J", (void*)JniCtor }, | 221 { (char*)"ctor", (char*)"(Ljava/lang/Object;)J", (void*)JniCtor }, |
| 190 { (char*)"dtor", (char*)"(J)V", (void*)JniDtor } | 222 { (char*)"dtor", (char*)"(J)V", (void*)JniDtor } |
| 191 }; | 223 }; |
| 192 | 224 |
| 193 extern "C" JNIEXPORT void JNICALL Java_org_adblockplus_libadblockplus_FileSystem _registerNatives(JNIEnv *env, jclass clazz) | 225 extern "C" JNIEXPORT void JNICALL Java_org_adblockplus_libadblockplus_FileSystem _registerNatives(JNIEnv *env, jclass clazz) |
| 194 { | 226 { |
| 195 env->RegisterNatives(clazz, methods, sizeof(methods) / sizeof(methods[0])); | 227 env->RegisterNatives(clazz, methods, sizeof(methods) / sizeof(methods[0])); |
| 196 } | 228 } |
| LEFT | RIGHT |