blob: 854ee794ba1fe0a608a91ac4487a416fd2062e4f [file] [log] [blame]
Andreas Huber88572f72012-02-21 11:47:18 -08001/*
2 * Copyright 2012, The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17//#define LOG_NDEBUG 0
18#define LOG_TAG "MediaExtractor-JNI"
19#include <utils/Log.h>
20
21#include "android_media_MediaExtractor.h"
22
23#include "android_media_Utils.h"
24#include "android_runtime/AndroidRuntime.h"
Ruben Brunk87eac992013-09-09 17:44:59 -070025#include "android_runtime/Log.h"
Andreas Huber88572f72012-02-21 11:47:18 -080026#include "jni.h"
27#include "JNIHelp.h"
28
Andreas Huberab57d032012-04-18 15:59:43 -070029#include <media/hardware/CryptoAPI.h>
Andreas Huber88572f72012-02-21 11:47:18 -080030#include <media/stagefright/foundation/ABuffer.h>
31#include <media/stagefright/foundation/ADebug.h>
32#include <media/stagefright/foundation/AMessage.h>
33#include <media/stagefright/DataSource.h>
34#include <media/stagefright/MediaErrors.h>
Andreas Huber91befdc2012-04-18 12:19:51 -070035#include <media/stagefright/MetaData.h>
Andreas Huber88572f72012-02-21 11:47:18 -080036#include <media/stagefright/NuMediaExtractor.h>
37
38namespace android {
39
40struct fields_t {
41 jfieldID context;
Andreas Huber91befdc2012-04-18 12:19:51 -070042
43 jmethodID cryptoInfoSetID;
Andreas Huber88572f72012-02-21 11:47:18 -080044};
45
46static fields_t gFields;
47
Marco Nelissenc209a062012-08-24 09:55:44 -070048class JavaDataSourceBridge : public DataSource {
49 jmethodID mReadMethod;
50 jmethodID mGetSizeMethod;
51 jmethodID mCloseMethod;
52 jobject mDataSource;
53 public:
54 JavaDataSourceBridge(JNIEnv *env, jobject source) {
55 mDataSource = env->NewGlobalRef(source);
56
57 jclass datasourceclass = env->GetObjectClass(mDataSource);
58 CHECK(datasourceclass != NULL);
59
60 mReadMethod = env->GetMethodID(datasourceclass, "readAt", "(J[BI)I");
61 CHECK(mReadMethod != NULL);
62
63 mGetSizeMethod = env->GetMethodID(datasourceclass, "getSize", "()J");
64 CHECK(mGetSizeMethod != NULL);
65
66 mCloseMethod = env->GetMethodID(datasourceclass, "close", "()V");
67 CHECK(mCloseMethod != NULL);
68 }
69
70 ~JavaDataSourceBridge() {
71 JNIEnv *env = AndroidRuntime::getJNIEnv();
72 env->CallVoidMethod(mDataSource, mCloseMethod);
73 env->DeleteGlobalRef(mDataSource);
74 }
75
76 virtual status_t initCheck() const {
77 return OK;
78 }
79
80 virtual ssize_t readAt(off64_t offset, void* buffer, size_t size) {
81 JNIEnv *env = AndroidRuntime::getJNIEnv();
82
83 // XXX could optimize this by reusing the same array
84 jbyteArray byteArrayObj = env->NewByteArray(size);
85 env->DeleteLocalRef(env->GetObjectClass(mDataSource));
86 env->DeleteLocalRef(env->GetObjectClass(byteArrayObj));
Ashok Bhat136c08a2014-03-05 15:23:13 +000087 ssize_t numread = env->CallIntMethod(mDataSource, mReadMethod, offset, byteArrayObj, (jint)size);
Marco Nelissenc209a062012-08-24 09:55:44 -070088 env->GetByteArrayRegion(byteArrayObj, 0, size, (jbyte*) buffer);
89 env->DeleteLocalRef(byteArrayObj);
90 if (env->ExceptionCheck()) {
Ashok Bhat075e9a12014-01-06 13:45:09 +000091 ALOGW("Exception occurred while reading %zu at %lld", size, offset);
Marco Nelissenc209a062012-08-24 09:55:44 -070092 LOGW_EX(env);
93 env->ExceptionClear();
94 return -1;
95 }
96 return numread;
97 }
98
99 virtual status_t getSize(off64_t *size) {
100 JNIEnv *env = AndroidRuntime::getJNIEnv();
101
102 CHECK(size != NULL);
103
104 int64_t len = env->CallLongMethod(mDataSource, mGetSizeMethod);
105 if (len < 0) {
106 *size = ERROR_UNSUPPORTED;
107 } else {
108 *size = len;
109 }
110 return OK;
111 }
112};
113
Andreas Huber88572f72012-02-21 11:47:18 -0800114////////////////////////////////////////////////////////////////////////////////
115
116JMediaExtractor::JMediaExtractor(JNIEnv *env, jobject thiz)
117 : mClass(NULL),
118 mObject(NULL) {
119 jclass clazz = env->GetObjectClass(thiz);
120 CHECK(clazz != NULL);
121
122 mClass = (jclass)env->NewGlobalRef(clazz);
123 mObject = env->NewWeakGlobalRef(thiz);
124
125 mImpl = new NuMediaExtractor;
126}
127
128JMediaExtractor::~JMediaExtractor() {
129 JNIEnv *env = AndroidRuntime::getJNIEnv();
130
131 env->DeleteWeakGlobalRef(mObject);
132 mObject = NULL;
133 env->DeleteGlobalRef(mClass);
134 mClass = NULL;
135}
136
Andreas Huber07ea4262012-04-11 12:21:20 -0700137status_t JMediaExtractor::setDataSource(
138 const char *path, const KeyedVector<String8, String8> *headers) {
139 return mImpl->setDataSource(path, headers);
140}
141
142status_t JMediaExtractor::setDataSource(int fd, off64_t offset, off64_t size) {
143 return mImpl->setDataSource(fd, offset, size);
Andreas Huber88572f72012-02-21 11:47:18 -0800144}
145
Marco Nelissenc209a062012-08-24 09:55:44 -0700146status_t JMediaExtractor::setDataSource(const sp<DataSource> &datasource) {
147 return mImpl->setDataSource(datasource);
148}
149
Andreas Huber88572f72012-02-21 11:47:18 -0800150size_t JMediaExtractor::countTracks() const {
151 return mImpl->countTracks();
152}
153
154status_t JMediaExtractor::getTrackFormat(size_t index, jobject *format) const {
155 sp<AMessage> msg;
156 status_t err;
157 if ((err = mImpl->getTrackFormat(index, &msg)) != OK) {
158 return err;
159 }
160
161 JNIEnv *env = AndroidRuntime::getJNIEnv();
162
163 return ConvertMessageToMap(env, msg, format);
164}
165
Marco Nelissene20a6d52013-04-08 14:28:55 -0700166status_t JMediaExtractor::getFileFormat(jobject *format) const {
167 sp<AMessage> msg;
168 status_t err;
169 if ((err = mImpl->getFileFormat(&msg)) != OK) {
170 return err;
171 }
172
173 JNIEnv *env = AndroidRuntime::getJNIEnv();
174
175 return ConvertMessageToMap(env, msg, format);
176}
177
Andreas Huber88572f72012-02-21 11:47:18 -0800178status_t JMediaExtractor::selectTrack(size_t index) {
179 return mImpl->selectTrack(index);
180}
181
Andreas Huberf2855b32012-04-25 15:57:43 -0700182status_t JMediaExtractor::unselectTrack(size_t index) {
183 return mImpl->unselectTrack(index);
184}
185
186status_t JMediaExtractor::seekTo(
187 int64_t timeUs, MediaSource::ReadOptions::SeekMode mode) {
188 return mImpl->seekTo(timeUs, mode);
Andreas Huber88572f72012-02-21 11:47:18 -0800189}
190
191status_t JMediaExtractor::advance() {
192 return mImpl->advance();
193}
194
195status_t JMediaExtractor::readSampleData(
196 jobject byteBuf, size_t offset, size_t *sampleSize) {
197 JNIEnv *env = AndroidRuntime::getJNIEnv();
198
199 void *dst = env->GetDirectBufferAddress(byteBuf);
200
Ashok Bhat075e9a12014-01-06 13:45:09 +0000201 size_t dstSize;
Andreas Huberc52b9802012-03-12 14:04:01 -0700202 jbyteArray byteArray = NULL;
203
Andreas Huber88572f72012-02-21 11:47:18 -0800204 if (dst == NULL) {
Andreas Huberc52b9802012-03-12 14:04:01 -0700205 jclass byteBufClass = env->FindClass("java/nio/ByteBuffer");
206 CHECK(byteBufClass != NULL);
207
208 jmethodID arrayID =
209 env->GetMethodID(byteBufClass, "array", "()[B");
210 CHECK(arrayID != NULL);
211
212 byteArray =
213 (jbyteArray)env->CallObjectMethod(byteBuf, arrayID);
214
215 if (byteArray == NULL) {
216 return INVALID_OPERATION;
217 }
218
219 jboolean isCopy;
220 dst = env->GetByteArrayElements(byteArray, &isCopy);
221
Ashok Bhat075e9a12014-01-06 13:45:09 +0000222 dstSize = (size_t) env->GetArrayLength(byteArray);
Andreas Huberc52b9802012-03-12 14:04:01 -0700223 } else {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000224 dstSize = (size_t) env->GetDirectBufferCapacity(byteBuf);
Andreas Huber88572f72012-02-21 11:47:18 -0800225 }
226
Andreas Huber88572f72012-02-21 11:47:18 -0800227 if (dstSize < offset) {
Andreas Huberc52b9802012-03-12 14:04:01 -0700228 if (byteArray != NULL) {
229 env->ReleaseByteArrayElements(byteArray, (jbyte *)dst, 0);
230 }
231
Andreas Huber88572f72012-02-21 11:47:18 -0800232 return -ERANGE;
233 }
234
235 sp<ABuffer> buffer = new ABuffer((char *)dst + offset, dstSize - offset);
236
237 status_t err = mImpl->readSampleData(buffer);
238
Andreas Huberc52b9802012-03-12 14:04:01 -0700239 if (byteArray != NULL) {
240 env->ReleaseByteArrayElements(byteArray, (jbyte *)dst, 0);
241 }
242
Andreas Huber88572f72012-02-21 11:47:18 -0800243 if (err != OK) {
244 return err;
245 }
246
247 *sampleSize = buffer->size();
248
249 return OK;
250}
251
252status_t JMediaExtractor::getSampleTrackIndex(size_t *trackIndex) {
253 return mImpl->getSampleTrackIndex(trackIndex);
254}
255
256status_t JMediaExtractor::getSampleTime(int64_t *sampleTimeUs) {
257 return mImpl->getSampleTime(sampleTimeUs);
258}
259
Andreas Huber9b8e4962012-03-26 11:13:27 -0700260status_t JMediaExtractor::getSampleFlags(uint32_t *sampleFlags) {
Andreas Huber91befdc2012-04-18 12:19:51 -0700261 *sampleFlags = 0;
262
263 sp<MetaData> meta;
264 status_t err = mImpl->getSampleMeta(&meta);
265
266 if (err != OK) {
267 return err;
268 }
269
270 int32_t val;
271 if (meta->findInt32(kKeyIsSyncFrame, &val) && val != 0) {
272 (*sampleFlags) |= NuMediaExtractor::SAMPLE_FLAG_SYNC;
273 }
274
275 uint32_t type;
276 const void *data;
277 size_t size;
278 if (meta->findData(kKeyEncryptedSizes, &type, &data, &size)) {
279 (*sampleFlags) |= NuMediaExtractor::SAMPLE_FLAG_ENCRYPTED;
280 }
281
282 return OK;
283}
284
285status_t JMediaExtractor::getSampleMeta(sp<MetaData> *sampleMeta) {
286 return mImpl->getSampleMeta(sampleMeta);
Andreas Huber9b8e4962012-03-26 11:13:27 -0700287}
288
Andreas Huber74a78b02012-04-19 16:24:32 -0700289bool JMediaExtractor::getCachedDuration(int64_t *durationUs, bool *eos) const {
290 return mImpl->getCachedDuration(durationUs, eos);
291}
292
Andreas Huber88572f72012-02-21 11:47:18 -0800293} // namespace android
294
295////////////////////////////////////////////////////////////////////////////////
296
297using namespace android;
298
299static sp<JMediaExtractor> setMediaExtractor(
300 JNIEnv *env, jobject thiz, const sp<JMediaExtractor> &extractor) {
301 sp<JMediaExtractor> old =
Ashok Bhat075e9a12014-01-06 13:45:09 +0000302 (JMediaExtractor *)env->GetLongField(thiz, gFields.context);
Andreas Huber88572f72012-02-21 11:47:18 -0800303
304 if (extractor != NULL) {
305 extractor->incStrong(thiz);
306 }
307 if (old != NULL) {
308 old->decStrong(thiz);
309 }
Ashok Bhat075e9a12014-01-06 13:45:09 +0000310 env->SetLongField(thiz, gFields.context, (jlong)extractor.get());
Andreas Huber88572f72012-02-21 11:47:18 -0800311
312 return old;
313}
314
315static sp<JMediaExtractor> getMediaExtractor(JNIEnv *env, jobject thiz) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000316 return (JMediaExtractor *)env->GetLongField(thiz, gFields.context);
Andreas Huber88572f72012-02-21 11:47:18 -0800317}
318
319static void android_media_MediaExtractor_release(JNIEnv *env, jobject thiz) {
320 setMediaExtractor(env, thiz, NULL);
321}
322
Andreas Huber60d610b2012-05-02 16:06:09 -0700323static jint android_media_MediaExtractor_getTrackCount(
Andreas Huber88572f72012-02-21 11:47:18 -0800324 JNIEnv *env, jobject thiz) {
325 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
326
327 if (extractor == NULL) {
328 jniThrowException(env, "java/lang/IllegalStateException", NULL);
Andreas Huber07ea4262012-04-11 12:21:20 -0700329 return -1;
Andreas Huber88572f72012-02-21 11:47:18 -0800330 }
331
Ashok Bhat075e9a12014-01-06 13:45:09 +0000332 return (jint) extractor->countTracks();
Andreas Huber88572f72012-02-21 11:47:18 -0800333}
334
Andreas Huber60d610b2012-05-02 16:06:09 -0700335static jobject android_media_MediaExtractor_getTrackFormatNative(
Andreas Huber88572f72012-02-21 11:47:18 -0800336 JNIEnv *env, jobject thiz, jint index) {
337 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
338
339 if (extractor == NULL) {
340 jniThrowException(env, "java/lang/IllegalStateException", NULL);
341 return NULL;
342 }
343
344 jobject format;
345 status_t err = extractor->getTrackFormat(index, &format);
346
347 if (err != OK) {
348 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
349 return NULL;
350 }
351
352 return format;
353}
354
Marco Nelissene20a6d52013-04-08 14:28:55 -0700355static jobject android_media_MediaExtractor_getFileFormatNative(
356 JNIEnv *env, jobject thiz) {
357 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
358
359 if (extractor == NULL) {
360 jniThrowException(env, "java/lang/IllegalStateException", NULL);
361 return NULL;
362 }
363
364 jobject format;
365 status_t err = extractor->getFileFormat(&format);
366
367 if (err != OK) {
368 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
369 return NULL;
370 }
371
372 return format;
373}
374
Andreas Huber88572f72012-02-21 11:47:18 -0800375static void android_media_MediaExtractor_selectTrack(
376 JNIEnv *env, jobject thiz, jint index) {
377 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
378
379 if (extractor == NULL) {
380 jniThrowException(env, "java/lang/IllegalStateException", NULL);
381 return;
382 }
383
384 status_t err = extractor->selectTrack(index);
385
386 if (err != OK) {
387 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
388 return;
389 }
390}
391
Andreas Huberf2855b32012-04-25 15:57:43 -0700392static void android_media_MediaExtractor_unselectTrack(
393 JNIEnv *env, jobject thiz, jint index) {
Andreas Huber88572f72012-02-21 11:47:18 -0800394 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
395
396 if (extractor == NULL) {
397 jniThrowException(env, "java/lang/IllegalStateException", NULL);
398 return;
399 }
400
Andreas Huberf2855b32012-04-25 15:57:43 -0700401 status_t err = extractor->unselectTrack(index);
402
403 if (err != OK) {
404 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
405 return;
406 }
407}
408
409static void android_media_MediaExtractor_seekTo(
410 JNIEnv *env, jobject thiz, jlong timeUs, jint mode) {
411 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
412
413 if (extractor == NULL) {
414 jniThrowException(env, "java/lang/IllegalStateException", NULL);
415 return;
416 }
417
418 if (mode < MediaSource::ReadOptions::SEEK_PREVIOUS_SYNC
Andreas Huberf86374a2012-05-09 11:25:29 -0700419 || mode >= MediaSource::ReadOptions::SEEK_CLOSEST) {
Andreas Huberf2855b32012-04-25 15:57:43 -0700420 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
421 return;
422 }
423
424 extractor->seekTo(timeUs, (MediaSource::ReadOptions::SeekMode)mode);
Andreas Huber88572f72012-02-21 11:47:18 -0800425}
426
427static jboolean android_media_MediaExtractor_advance(
428 JNIEnv *env, jobject thiz) {
429 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
430
431 if (extractor == NULL) {
432 jniThrowException(env, "java/lang/IllegalStateException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000433 return JNI_FALSE;
Andreas Huber88572f72012-02-21 11:47:18 -0800434 }
435
436 status_t err = extractor->advance();
437
438 if (err == ERROR_END_OF_STREAM) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000439 return JNI_FALSE;
Andreas Huber88572f72012-02-21 11:47:18 -0800440 } else if (err != OK) {
441 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000442 return JNI_FALSE;
Andreas Huber88572f72012-02-21 11:47:18 -0800443 }
444
Ashok Bhat075e9a12014-01-06 13:45:09 +0000445 return JNI_TRUE;
Andreas Huber88572f72012-02-21 11:47:18 -0800446}
447
448static jint android_media_MediaExtractor_readSampleData(
449 JNIEnv *env, jobject thiz, jobject byteBuf, jint offset) {
450 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
451
452 if (extractor == NULL) {
453 jniThrowException(env, "java/lang/IllegalStateException", NULL);
454 return -1;
455 }
456
457 size_t sampleSize;
458 status_t err = extractor->readSampleData(byteBuf, offset, &sampleSize);
459
460 if (err == ERROR_END_OF_STREAM) {
461 return -1;
462 } else if (err != OK) {
463 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000464 return -1;
Andreas Huber88572f72012-02-21 11:47:18 -0800465 }
466
Ashok Bhat075e9a12014-01-06 13:45:09 +0000467 return (jint) sampleSize;
Andreas Huber88572f72012-02-21 11:47:18 -0800468}
469
470static jint android_media_MediaExtractor_getSampleTrackIndex(
471 JNIEnv *env, jobject thiz) {
472 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
473
474 if (extractor == NULL) {
475 jniThrowException(env, "java/lang/IllegalStateException", NULL);
476 return -1;
477 }
478
479 size_t trackIndex;
480 status_t err = extractor->getSampleTrackIndex(&trackIndex);
481
482 if (err == ERROR_END_OF_STREAM) {
483 return -1;
484 } else if (err != OK) {
485 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000486 return -1;
Andreas Huber88572f72012-02-21 11:47:18 -0800487 }
488
Ashok Bhat075e9a12014-01-06 13:45:09 +0000489 return (jint) trackIndex;
Andreas Huber88572f72012-02-21 11:47:18 -0800490}
491
492static jlong android_media_MediaExtractor_getSampleTime(
493 JNIEnv *env, jobject thiz) {
494 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
495
496 if (extractor == NULL) {
497 jniThrowException(env, "java/lang/IllegalStateException", NULL);
498 return -1ll;
499 }
500
501 int64_t sampleTimeUs;
502 status_t err = extractor->getSampleTime(&sampleTimeUs);
503
504 if (err == ERROR_END_OF_STREAM) {
505 return -1ll;
506 } else if (err != OK) {
507 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000508 return -1ll;
Andreas Huber88572f72012-02-21 11:47:18 -0800509 }
510
Ashok Bhat075e9a12014-01-06 13:45:09 +0000511 return (jlong) sampleTimeUs;
Andreas Huber88572f72012-02-21 11:47:18 -0800512}
513
Andreas Huber9b8e4962012-03-26 11:13:27 -0700514static jint android_media_MediaExtractor_getSampleFlags(
515 JNIEnv *env, jobject thiz) {
516 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
517
518 if (extractor == NULL) {
519 jniThrowException(env, "java/lang/IllegalStateException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000520 return -1;
Andreas Huber9b8e4962012-03-26 11:13:27 -0700521 }
522
523 uint32_t sampleFlags;
524 status_t err = extractor->getSampleFlags(&sampleFlags);
525
526 if (err == ERROR_END_OF_STREAM) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000527 return -1;
Andreas Huber9b8e4962012-03-26 11:13:27 -0700528 } else if (err != OK) {
529 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000530 return -1;
Andreas Huber9b8e4962012-03-26 11:13:27 -0700531 }
532
Ashok Bhat075e9a12014-01-06 13:45:09 +0000533 return (jint) sampleFlags;
Andreas Huber9b8e4962012-03-26 11:13:27 -0700534}
535
Andreas Huber91befdc2012-04-18 12:19:51 -0700536static jboolean android_media_MediaExtractor_getSampleCryptoInfo(
537 JNIEnv *env, jobject thiz, jobject cryptoInfoObj) {
538 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
539
540 if (extractor == NULL) {
541 jniThrowException(env, "java/lang/IllegalStateException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000542 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700543 }
544
545 sp<MetaData> meta;
546 status_t err = extractor->getSampleMeta(&meta);
547
548 if (err != OK) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000549 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700550 }
551
552 uint32_t type;
553 const void *data;
554 size_t size;
555 if (!meta->findData(kKeyEncryptedSizes, &type, &data, &size)) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000556 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700557 }
558
Ashok Bhate92416d2014-03-05 11:51:15 +0000559 size_t numSubSamples = size / sizeof(int32_t);
Andreas Huber91befdc2012-04-18 12:19:51 -0700560
561 if (numSubSamples == 0) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000562 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700563 }
564
565 jintArray numBytesOfEncryptedDataObj = env->NewIntArray(numSubSamples);
566 jboolean isCopy;
567 jint *dst = env->GetIntArrayElements(numBytesOfEncryptedDataObj, &isCopy);
568 for (size_t i = 0; i < numSubSamples; ++i) {
Ashok Bhate92416d2014-03-05 11:51:15 +0000569 dst[i] = ((const int32_t *)data)[i];
Andreas Huber91befdc2012-04-18 12:19:51 -0700570 }
571 env->ReleaseIntArrayElements(numBytesOfEncryptedDataObj, dst, 0);
572 dst = NULL;
573
574 size_t encSize = size;
575 jintArray numBytesOfPlainDataObj = NULL;
Andreas Huberab57d032012-04-18 15:59:43 -0700576 if (meta->findData(kKeyPlainSizes, &type, &data, &size)) {
Andreas Huber91befdc2012-04-18 12:19:51 -0700577 if (size != encSize) {
578 // The two must be of the same length.
Ashok Bhat075e9a12014-01-06 13:45:09 +0000579 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700580 }
581
582 numBytesOfPlainDataObj = env->NewIntArray(numSubSamples);
583 jboolean isCopy;
584 jint *dst = env->GetIntArrayElements(numBytesOfPlainDataObj, &isCopy);
585 for (size_t i = 0; i < numSubSamples; ++i) {
Ashok Bhate92416d2014-03-05 11:51:15 +0000586 dst[i] = ((const int32_t *)data)[i];
Andreas Huber91befdc2012-04-18 12:19:51 -0700587 }
588 env->ReleaseIntArrayElements(numBytesOfPlainDataObj, dst, 0);
589 dst = NULL;
590 }
591
592 jbyteArray keyObj = NULL;
593 if (meta->findData(kKeyCryptoKey, &type, &data, &size)) {
594 if (size != 16) {
595 // Keys must be 16 bytes in length.
Ashok Bhat075e9a12014-01-06 13:45:09 +0000596 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700597 }
598
599 keyObj = env->NewByteArray(size);
600 jboolean isCopy;
601 jbyte *dst = env->GetByteArrayElements(keyObj, &isCopy);
602 memcpy(dst, data, size);
603 env->ReleaseByteArrayElements(keyObj, dst, 0);
604 dst = NULL;
605 }
606
607 jbyteArray ivObj = NULL;
608 if (meta->findData(kKeyCryptoIV, &type, &data, &size)) {
609 if (size != 16) {
610 // IVs must be 16 bytes in length.
Ashok Bhat075e9a12014-01-06 13:45:09 +0000611 return JNI_FALSE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700612 }
613
614 ivObj = env->NewByteArray(size);
615 jboolean isCopy;
616 jbyte *dst = env->GetByteArrayElements(ivObj, &isCopy);
617 memcpy(dst, data, size);
618 env->ReleaseByteArrayElements(ivObj, dst, 0);
619 dst = NULL;
620 }
621
622 int32_t mode;
623 if (!meta->findInt32(kKeyCryptoMode, &mode)) {
Andreas Huberab57d032012-04-18 15:59:43 -0700624 mode = CryptoPlugin::kMode_AES_CTR;
Andreas Huber91befdc2012-04-18 12:19:51 -0700625 }
626
627 env->CallVoidMethod(
628 cryptoInfoObj,
629 gFields.cryptoInfoSetID,
Ashok Bhat136c08a2014-03-05 15:23:13 +0000630 (jint)numSubSamples,
Andreas Huber91befdc2012-04-18 12:19:51 -0700631 numBytesOfPlainDataObj,
632 numBytesOfEncryptedDataObj,
633 keyObj,
634 ivObj,
635 mode);
636
Ashok Bhat075e9a12014-01-06 13:45:09 +0000637 return JNI_TRUE;
Andreas Huber91befdc2012-04-18 12:19:51 -0700638}
639
Andreas Huber88572f72012-02-21 11:47:18 -0800640static void android_media_MediaExtractor_native_init(JNIEnv *env) {
641 jclass clazz = env->FindClass("android/media/MediaExtractor");
642 CHECK(clazz != NULL);
643
Ashok Bhat075e9a12014-01-06 13:45:09 +0000644 gFields.context = env->GetFieldID(clazz, "mNativeContext", "J");
Andreas Huber88572f72012-02-21 11:47:18 -0800645 CHECK(gFields.context != NULL);
646
Andreas Huber91befdc2012-04-18 12:19:51 -0700647 clazz = env->FindClass("android/media/MediaCodec$CryptoInfo");
648 CHECK(clazz != NULL);
649
650 gFields.cryptoInfoSetID =
651 env->GetMethodID(clazz, "set", "(I[I[I[B[BI)V");
652
Andreas Huber88572f72012-02-21 11:47:18 -0800653 DataSource::RegisterDefaultSniffers();
654}
655
656static void android_media_MediaExtractor_native_setup(
Andreas Huber07ea4262012-04-11 12:21:20 -0700657 JNIEnv *env, jobject thiz) {
Andreas Huber88572f72012-02-21 11:47:18 -0800658 sp<JMediaExtractor> extractor = new JMediaExtractor(env, thiz);
Andreas Huber07ea4262012-04-11 12:21:20 -0700659 setMediaExtractor(env,thiz, extractor);
660}
Andreas Huber88572f72012-02-21 11:47:18 -0800661
Andreas Huber07ea4262012-04-11 12:21:20 -0700662static void android_media_MediaExtractor_setDataSource(
663 JNIEnv *env, jobject thiz,
664 jstring pathObj, jobjectArray keysArray, jobjectArray valuesArray) {
665 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
666
667 if (extractor == NULL) {
668 jniThrowException(env, "java/lang/IllegalStateException", NULL);
669 return;
670 }
671
672 if (pathObj == NULL) {
Andreas Huber88572f72012-02-21 11:47:18 -0800673 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
674 return;
675 }
676
Andreas Huber07ea4262012-04-11 12:21:20 -0700677 KeyedVector<String8, String8> headers;
678 if (!ConvertKeyValueArraysToKeyedVector(
679 env, keysArray, valuesArray, &headers)) {
Andreas Huber88572f72012-02-21 11:47:18 -0800680 return;
681 }
682
Andreas Huber07ea4262012-04-11 12:21:20 -0700683 const char *path = env->GetStringUTFChars(pathObj, NULL);
Andreas Huber88572f72012-02-21 11:47:18 -0800684
Andreas Huber07ea4262012-04-11 12:21:20 -0700685 if (path == NULL) {
686 return;
687 }
688
689 status_t err = extractor->setDataSource(path, &headers);
690
691 env->ReleaseStringUTFChars(pathObj, path);
692 path = NULL;
Andreas Huber88572f72012-02-21 11:47:18 -0800693
694 if (err != OK) {
695 jniThrowException(
696 env,
697 "java/io/IOException",
698 "Failed to instantiate extractor.");
699 return;
700 }
Andreas Huber07ea4262012-04-11 12:21:20 -0700701}
Andreas Huber88572f72012-02-21 11:47:18 -0800702
Andreas Huber07ea4262012-04-11 12:21:20 -0700703static void android_media_MediaExtractor_setDataSourceFd(
704 JNIEnv *env, jobject thiz,
705 jobject fileDescObj, jlong offset, jlong length) {
706 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
707
708 if (extractor == NULL) {
709 jniThrowException(env, "java/lang/IllegalStateException", NULL);
710 return;
711 }
712
713 if (fileDescObj == NULL) {
714 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
715 return;
716 }
717
718 int fd = jniGetFDFromFileDescriptor(env, fileDescObj);
719
720 status_t err = extractor->setDataSource(fd, offset, length);
721
722 if (err != OK) {
723 jniThrowException(
724 env,
725 "java/io/IOException",
726 "Failed to instantiate extractor.");
727 return;
728 }
Andreas Huber88572f72012-02-21 11:47:18 -0800729}
730
Marco Nelissenc209a062012-08-24 09:55:44 -0700731static void android_media_MediaExtractor_setDataSourceCallback(
732 JNIEnv *env, jobject thiz,
733 jobject callbackObj) {
734 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
735
736 if (extractor == NULL) {
737 jniThrowException(env, "java/lang/IllegalStateException", NULL);
738 return;
739 }
740
741 if (callbackObj == NULL) {
742 jniThrowException(env, "java/lang/IllegalArgumentException", NULL);
743 return;
744 }
745
746 sp<JavaDataSourceBridge> bridge = new JavaDataSourceBridge(env, callbackObj);
747 status_t err = extractor->setDataSource(bridge);
748
749 if (err != OK) {
750 jniThrowException(
751 env,
752 "java/io/IOException",
753 "Failed to instantiate extractor.");
754 return;
755 }
756}
757
Andreas Huber74a78b02012-04-19 16:24:32 -0700758static jlong android_media_MediaExtractor_getCachedDurationUs(
759 JNIEnv *env, jobject thiz) {
760 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
761
762 if (extractor == NULL) {
763 jniThrowException(env, "java/lang/IllegalStateException", NULL);
764 return -1ll;
765 }
766
767 int64_t cachedDurationUs;
768 bool eos;
769 if (!extractor->getCachedDuration(&cachedDurationUs, &eos)) {
770 return -1ll;
771 }
772
Ashok Bhat075e9a12014-01-06 13:45:09 +0000773 return (jlong) cachedDurationUs;
Andreas Huber74a78b02012-04-19 16:24:32 -0700774}
775
776static jboolean android_media_MediaExtractor_hasCacheReachedEOS(
777 JNIEnv *env, jobject thiz) {
778 sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz);
779
780 if (extractor == NULL) {
781 jniThrowException(env, "java/lang/IllegalStateException", NULL);
Ashok Bhat075e9a12014-01-06 13:45:09 +0000782 return JNI_TRUE;
Andreas Huber74a78b02012-04-19 16:24:32 -0700783 }
784
785 int64_t cachedDurationUs;
786 bool eos;
787 if (!extractor->getCachedDuration(&cachedDurationUs, &eos)) {
Ashok Bhat075e9a12014-01-06 13:45:09 +0000788 return JNI_TRUE;
Andreas Huber74a78b02012-04-19 16:24:32 -0700789 }
790
Ashok Bhat075e9a12014-01-06 13:45:09 +0000791 return eos ? JNI_TRUE : JNI_FALSE;
Andreas Huber74a78b02012-04-19 16:24:32 -0700792}
793
Andreas Huber88572f72012-02-21 11:47:18 -0800794static void android_media_MediaExtractor_native_finalize(
795 JNIEnv *env, jobject thiz) {
796 android_media_MediaExtractor_release(env, thiz);
797}
798
799static JNINativeMethod gMethods[] = {
800 { "release", "()V", (void *)android_media_MediaExtractor_release },
801
Andreas Huber60d610b2012-05-02 16:06:09 -0700802 { "getTrackCount", "()I", (void *)android_media_MediaExtractor_getTrackCount },
Andreas Huber88572f72012-02-21 11:47:18 -0800803
Marco Nelissene20a6d52013-04-08 14:28:55 -0700804 { "getFileFormatNative", "()Ljava/util/Map;",
805 (void *)android_media_MediaExtractor_getFileFormatNative },
806
Andreas Huber60d610b2012-05-02 16:06:09 -0700807 { "getTrackFormatNative", "(I)Ljava/util/Map;",
808 (void *)android_media_MediaExtractor_getTrackFormatNative },
Andreas Huber88572f72012-02-21 11:47:18 -0800809
810 { "selectTrack", "(I)V", (void *)android_media_MediaExtractor_selectTrack },
811
Andreas Huberf2855b32012-04-25 15:57:43 -0700812 { "unselectTrack", "(I)V",
813 (void *)android_media_MediaExtractor_unselectTrack },
814
815 { "seekTo", "(JI)V", (void *)android_media_MediaExtractor_seekTo },
Andreas Huber88572f72012-02-21 11:47:18 -0800816
817 { "advance", "()Z", (void *)android_media_MediaExtractor_advance },
818
819 { "readSampleData", "(Ljava/nio/ByteBuffer;I)I",
820 (void *)android_media_MediaExtractor_readSampleData },
821
822 { "getSampleTrackIndex", "()I",
823 (void *)android_media_MediaExtractor_getSampleTrackIndex },
824
825 { "getSampleTime", "()J",
826 (void *)android_media_MediaExtractor_getSampleTime },
827
Andreas Huber9b8e4962012-03-26 11:13:27 -0700828 { "getSampleFlags", "()I",
829 (void *)android_media_MediaExtractor_getSampleFlags },
830
Andreas Huber91befdc2012-04-18 12:19:51 -0700831 { "getSampleCryptoInfo", "(Landroid/media/MediaCodec$CryptoInfo;)Z",
832 (void *)android_media_MediaExtractor_getSampleCryptoInfo },
833
Andreas Huber88572f72012-02-21 11:47:18 -0800834 { "native_init", "()V", (void *)android_media_MediaExtractor_native_init },
835
Andreas Huber07ea4262012-04-11 12:21:20 -0700836 { "native_setup", "()V",
Andreas Huber88572f72012-02-21 11:47:18 -0800837 (void *)android_media_MediaExtractor_native_setup },
838
839 { "native_finalize", "()V",
840 (void *)android_media_MediaExtractor_native_finalize },
Andreas Huber07ea4262012-04-11 12:21:20 -0700841
842 { "setDataSource", "(Ljava/lang/String;[Ljava/lang/String;"
843 "[Ljava/lang/String;)V",
844 (void *)android_media_MediaExtractor_setDataSource },
845
846 { "setDataSource", "(Ljava/io/FileDescriptor;JJ)V",
847 (void *)android_media_MediaExtractor_setDataSourceFd },
Andreas Huber74a78b02012-04-19 16:24:32 -0700848
Marco Nelissenc209a062012-08-24 09:55:44 -0700849 { "setDataSource", "(Landroid/media/DataSource;)V",
850 (void *)android_media_MediaExtractor_setDataSourceCallback },
851
Andreas Huber74a78b02012-04-19 16:24:32 -0700852 { "getCachedDuration", "()J",
853 (void *)android_media_MediaExtractor_getCachedDurationUs },
854
855 { "hasCacheReachedEndOfStream", "()Z",
856 (void *)android_media_MediaExtractor_hasCacheReachedEOS },
Andreas Huber88572f72012-02-21 11:47:18 -0800857};
858
859int register_android_media_MediaExtractor(JNIEnv *env) {
860 return AndroidRuntime::registerNativeMethods(env,
861 "android/media/MediaExtractor", gMethods, NELEM(gMethods));
862}