blob: 15e529e169fc0a2efb28edf54ac60f784c0ad71d [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001#include "CreateJavaOutputStreamAdaptor.h"
Leon Scroggins IIIca320212013-08-20 17:59:39 -04002#include "SkData.h"
Mike Reed31274c82018-01-05 10:25:08 -05003#include "SkMalloc.h"
Leon Scroggins IIIca320212013-08-20 17:59:39 -04004#include "SkRefCnt.h"
5#include "SkStream.h"
6#include "SkTypes.h"
7#include "Utils.h"
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08008
Steven Moreland2279b252017-07-19 09:50:45 -07009#include <nativehelper/JNIHelp.h>
Derek Sollenbergereec1b862019-10-24 09:44:55 -040010#include <log/log.h>
Ben Wagner60126ef2015-08-07 12:13:48 -040011#include <memory>
12
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080013static jmethodID gInputStream_readMethodID;
14static jmethodID gInputStream_skipMethodID;
15
Leon Scroggins IIIca320212013-08-20 17:59:39 -040016/**
Leon Scroggins III7315f1b2013-09-10 20:26:05 -040017 * Wrapper for a Java InputStream.
Leon Scroggins IIIca320212013-08-20 17:59:39 -040018 */
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080019class JavaInputStreamAdaptor : public SkStream {
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050020 JavaInputStreamAdaptor(JavaVM* jvm, jobject js, jbyteArray ar, jint capacity,
21 bool swallowExceptions)
22 : fJvm(jvm)
23 , fJavaInputStream(js)
24 , fJavaByteArray(ar)
25 , fCapacity(capacity)
26 , fBytesRead(0)
27 , fIsAtEnd(false)
28 , fSwallowExceptions(swallowExceptions) {}
29
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080030public:
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050031 static JavaInputStreamAdaptor* Create(JNIEnv* env, jobject js, jbyteArray ar,
32 bool swallowExceptions) {
33 JavaVM* jvm;
34 LOG_ALWAYS_FATAL_IF(env->GetJavaVM(&jvm) != JNI_OK);
35
36 js = env->NewGlobalRef(js);
37 if (!js) {
38 return nullptr;
39 }
40
41 ar = (jbyteArray) env->NewGlobalRef(ar);
42 if (!ar) {
43 env->DeleteGlobalRef(js);
44 return nullptr;
45 }
46
47 jint capacity = env->GetArrayLength(ar);
48 return new JavaInputStreamAdaptor(jvm, js, ar, capacity, swallowExceptions);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080049 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -070050
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050051 ~JavaInputStreamAdaptor() override {
Leon Scroggins IIIf97b29d22020-04-06 12:01:01 -040052 auto* env = android::requireEnv(fJvm);
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050053 env->DeleteGlobalRef(fJavaInputStream);
54 env->DeleteGlobalRef(fJavaByteArray);
55 }
56
57 size_t read(void* buffer, size_t size) override {
Leon Scroggins IIIf97b29d22020-04-06 12:01:01 -040058 auto* env = android::requireEnv(fJvm);
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050059 if (!fSwallowExceptions && checkException(env)) {
60 // Just in case the caller did not clear from a previous exception.
61 return 0;
62 }
Leon Scroggins IIIca320212013-08-20 17:59:39 -040063 if (NULL == buffer) {
64 if (0 == size) {
65 return 0;
66 } else {
67 /* InputStream.skip(n) can return <=0 but still not be at EOF
68 If we see that value, we need to call read(), which will
69 block if waiting for more data, or return -1 at EOF
70 */
71 size_t amountSkipped = 0;
72 do {
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050073 size_t amount = this->doSkip(size - amountSkipped, env);
Leon Scroggins IIIca320212013-08-20 17:59:39 -040074 if (0 == amount) {
75 char tmp;
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050076 amount = this->doRead(&tmp, 1, env);
Leon Scroggins IIIca320212013-08-20 17:59:39 -040077 if (0 == amount) {
78 // if read returned 0, we're at EOF
79 fIsAtEnd = true;
80 break;
81 }
82 }
83 amountSkipped += amount;
84 } while (amountSkipped < size);
85 return amountSkipped;
86 }
87 }
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050088 return this->doRead(buffer, size, env);
Leon Scroggins IIIca320212013-08-20 17:59:39 -040089 }
90
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050091 bool isAtEnd() const override { return fIsAtEnd; }
Leon Scroggins IIIca320212013-08-20 17:59:39 -040092
93private:
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -050094 size_t doRead(void* buffer, size_t size, JNIEnv* env) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080095 size_t bytesRead = 0;
96 // read the bytes
97 do {
Ashok Bhat2bb39d72014-03-05 12:40:53 +000098 jint requested = 0;
99 if (size > static_cast<size_t>(fCapacity)) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800100 requested = fCapacity;
Ashok Bhat2bb39d72014-03-05 12:40:53 +0000101 } else {
102 // This is safe because requested is clamped to (jint)
103 // fCapacity.
104 requested = static_cast<jint>(size);
105 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700106
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800107 jint n = env->CallIntMethod(fJavaInputStream,
108 gInputStream_readMethodID, fJavaByteArray, 0, requested);
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500109 if (checkException(env)) {
Sally Qi7e3f93b2021-07-15 00:00:54 +0000110 ALOGD("---- read threw an exception\n");
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500111 return bytesRead;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800112 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700113
Gilles Debunne8cd48572010-07-15 18:06:36 -0700114 if (n < 0) { // n == 0 should not be possible, see InputStream read() specifications.
Leon Scroggins IIIca320212013-08-20 17:59:39 -0400115 fIsAtEnd = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800116 break; // eof
117 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700118
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800119 env->GetByteArrayRegion(fJavaByteArray, 0, n,
120 reinterpret_cast<jbyte*>(buffer));
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500121 if (checkException(env)) {
Sally Qi7e3f93b2021-07-15 00:00:54 +0000122 ALOGD("---- read:GetByteArrayRegion threw an exception\n");
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500123 return bytesRead;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800124 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700125
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800126 buffer = (void*)((char*)buffer + n);
127 bytesRead += n;
128 size -= n;
129 fBytesRead += n;
130 } while (size != 0);
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700131
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800132 return bytesRead;
133 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700134
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500135 size_t doSkip(size_t size, JNIEnv* env) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800136 jlong skipped = env->CallLongMethod(fJavaInputStream,
137 gInputStream_skipMethodID, (jlong)size);
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500138 if (checkException(env)) {
Sally Qi7e3f93b2021-07-15 00:00:54 +0000139 ALOGD("------- skip threw an exception\n");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800140 return 0;
141 }
142 if (skipped < 0) {
143 skipped = 0;
144 }
Gilles Debunne8cd48572010-07-15 18:06:36 -0700145
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800146 return (size_t)skipped;
147 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700148
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500149 bool checkException(JNIEnv* env) {
150 if (!env->ExceptionCheck()) {
151 return false;
152 }
153
154 env->ExceptionDescribe();
155 if (fSwallowExceptions) {
156 env->ExceptionClear();
157 }
158
159 // There is no way to recover from the error, so consider the stream
160 // to be at the end.
161 fIsAtEnd = true;
162
163 return true;
164 }
165
166 JavaVM* fJvm;
167 jobject fJavaInputStream;
168 jbyteArray fJavaByteArray;
169 const jint fCapacity;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800170 size_t fBytesRead;
Leon Scroggins IIIca320212013-08-20 17:59:39 -0400171 bool fIsAtEnd;
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500172 const bool fSwallowExceptions;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800173};
174
Leon Scroggins IIIed074fd2017-12-11 13:47:23 -0500175SkStream* CreateJavaInputStreamAdaptor(JNIEnv* env, jobject stream, jbyteArray storage,
176 bool swallowExceptions) {
177 return JavaInputStreamAdaptor::Create(env, stream, storage, swallowExceptions);
Leon Scroggins IIIca320212013-08-20 17:59:39 -0400178}
179
Leon Scroggins III23ac0362020-05-04 15:38:58 -0400180sk_sp<SkData> CopyJavaInputStream(JNIEnv* env, jobject inputStream, jbyteArray storage) {
181 std::unique_ptr<SkStream> stream(CreateJavaInputStreamAdaptor(env, inputStream, storage));
182 if (!stream) {
183 return nullptr;
184 }
185
Leon Scroggins III37b82e32013-09-12 20:00:46 -0400186 size_t bufferSize = 4096;
187 size_t streamLen = 0;
188 size_t len;
189 char* data = (char*)sk_malloc_throw(bufferSize);
190
191 while ((len = stream->read(data + streamLen,
192 bufferSize - streamLen)) != 0) {
193 streamLen += len;
194 if (streamLen == bufferSize) {
195 bufferSize *= 2;
196 data = (char*)sk_realloc_throw(data, bufferSize);
197 }
198 }
199 data = (char*)sk_realloc_throw(data, streamLen);
200
Leon Scroggins III23ac0362020-05-04 15:38:58 -0400201 return SkData::MakeFromMalloc(data, streamLen);
Leon Scroggins IIIca320212013-08-20 17:59:39 -0400202}
203
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800204///////////////////////////////////////////////////////////////////////////////
205
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800206static jmethodID gOutputStream_writeMethodID;
207static jmethodID gOutputStream_flushMethodID;
208
209class SkJavaOutputStream : public SkWStream {
210public:
211 SkJavaOutputStream(JNIEnv* env, jobject stream, jbyteArray storage)
Leon Scrogginscc11f152014-03-31 16:52:13 -0400212 : fEnv(env), fJavaOutputStream(stream), fJavaByteArray(storage), fBytesWritten(0) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800213 fCapacity = env->GetArrayLength(storage);
214 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700215
Leon Scrogginscc11f152014-03-31 16:52:13 -0400216 virtual size_t bytesWritten() const {
217 return fBytesWritten;
218 }
219
Ashok Bhat2bb39d72014-03-05 12:40:53 +0000220 virtual bool write(const void* buffer, size_t size) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800221 JNIEnv* env = fEnv;
222 jbyteArray storage = fJavaByteArray;
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700223
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800224 while (size > 0) {
Ashok Bhat2bb39d72014-03-05 12:40:53 +0000225 jint requested = 0;
226 if (size > static_cast<size_t>(fCapacity)) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800227 requested = fCapacity;
Ashok Bhat2bb39d72014-03-05 12:40:53 +0000228 } else {
229 // This is safe because requested is clamped to (jint)
230 // fCapacity.
231 requested = static_cast<jint>(size);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800232 }
233
234 env->SetByteArrayRegion(storage, 0, requested,
235 reinterpret_cast<const jbyte*>(buffer));
236 if (env->ExceptionCheck()) {
237 env->ExceptionDescribe();
238 env->ExceptionClear();
Sally Qi7e3f93b2021-07-15 00:00:54 +0000239 ALOGD("--- write:SetByteArrayElements threw an exception\n");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800240 return false;
241 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700242
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800243 fEnv->CallVoidMethod(fJavaOutputStream, gOutputStream_writeMethodID,
244 storage, 0, requested);
245 if (env->ExceptionCheck()) {
246 env->ExceptionDescribe();
247 env->ExceptionClear();
Sally Qi7e3f93b2021-07-15 00:00:54 +0000248 ALOGD("------- write threw an exception\n");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800249 return false;
250 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700251
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800252 buffer = (void*)((char*)buffer + requested);
253 size -= requested;
Leon Scrogginscc11f152014-03-31 16:52:13 -0400254 fBytesWritten += requested;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800255 }
256 return true;
257 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700258
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800259 virtual void flush() {
260 fEnv->CallVoidMethod(fJavaOutputStream, gOutputStream_flushMethodID);
261 }
Elliott Hughesdd66bcb2011-04-12 11:28:59 -0700262
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800263private:
264 JNIEnv* fEnv;
265 jobject fJavaOutputStream; // the caller owns this object
266 jbyteArray fJavaByteArray; // the caller owns this object
Ashok Bhat2bb39d72014-03-05 12:40:53 +0000267 jint fCapacity;
Leon Scrogginscc11f152014-03-31 16:52:13 -0400268 size_t fBytesWritten;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800269};
270
271SkWStream* CreateJavaOutputStreamAdaptor(JNIEnv* env, jobject stream,
272 jbyteArray storage) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800273 return new SkJavaOutputStream(env, stream, storage);
274}
Leon Scroggins IIId0d7eaf2013-09-06 16:46:57 -0400275
276static jclass findClassCheck(JNIEnv* env, const char classname[]) {
277 jclass clazz = env->FindClass(classname);
278 SkASSERT(!env->ExceptionCheck());
279 return clazz;
280}
281
Leon Scroggins IIId0d7eaf2013-09-06 16:46:57 -0400282static jmethodID getMethodIDCheck(JNIEnv* env, jclass clazz,
283 const char methodname[], const char type[]) {
284 jmethodID id = env->GetMethodID(clazz, methodname, type);
285 SkASSERT(!env->ExceptionCheck());
286 return id;
287}
288
289int register_android_graphics_CreateJavaOutputStreamAdaptor(JNIEnv* env) {
290 jclass inputStream_Clazz = findClassCheck(env, "java/io/InputStream");
Leon Scroggins IIId0d7eaf2013-09-06 16:46:57 -0400291 gInputStream_readMethodID = getMethodIDCheck(env, inputStream_Clazz, "read", "([BII)I");
292 gInputStream_skipMethodID = getMethodIDCheck(env, inputStream_Clazz, "skip", "(J)J");
293
Leon Scroggins IIId0d7eaf2013-09-06 16:46:57 -0400294 jclass outputStream_Clazz = findClassCheck(env, "java/io/OutputStream");
295 gOutputStream_writeMethodID = getMethodIDCheck(env, outputStream_Clazz, "write", "([BII)V");
296 gOutputStream_flushMethodID = getMethodIDCheck(env, outputStream_Clazz, "flush", "()V");
297
298 return 0;
299}