diff options
author | Marco Nelissen <marcone@google.com> | 2012-08-24 09:55:44 -0700 |
---|---|---|
committer | Marco Nelissen <marcone@google.com> | 2012-08-24 10:36:51 -0700 |
commit | c209a06cfdcf633f12a299245312e3ac32bff27c (patch) | |
tree | 036c6847186f84384127378e914da23326f3a969 /media/jni | |
parent | 6715d1effaa70abf261112d2771d4d555cc109c2 (diff) | |
download | frameworks_base-c209a06cfdcf633f12a299245312e3ac32bff27c.zip frameworks_base-c209a06cfdcf633f12a299245312e3ac32bff27c.tar.gz frameworks_base-c209a06cfdcf633f12a299245312e3ac32bff27c.tar.bz2 |
Let apps provide a custom data source for extractors
Adds android.media.DataSource, which is modeled after its native namesake,
and a new method on MediaExtractor that lets apps specify their implementation
of a DataSource as the source of data for the extractor.
Change-Id: If1b169bd18d2691ebc4f8996494dfc8ee0894b6c
Diffstat (limited to 'media/jni')
-rw-r--r-- | media/jni/android_media_MediaExtractor.cpp | 100 | ||||
-rw-r--r-- | media/jni/android_media_MediaExtractor.h | 2 |
2 files changed, 102 insertions, 0 deletions
diff --git a/media/jni/android_media_MediaExtractor.cpp b/media/jni/android_media_MediaExtractor.cpp index 351ff04..23949fa 100644 --- a/media/jni/android_media_MediaExtractor.cpp +++ b/media/jni/android_media_MediaExtractor.cpp @@ -44,6 +44,72 @@ struct fields_t { static fields_t gFields; +class JavaDataSourceBridge : public DataSource { + jmethodID mReadMethod; + jmethodID mGetSizeMethod; + jmethodID mCloseMethod; + jobject mDataSource; + public: + JavaDataSourceBridge(JNIEnv *env, jobject source) { + mDataSource = env->NewGlobalRef(source); + + jclass datasourceclass = env->GetObjectClass(mDataSource); + CHECK(datasourceclass != NULL); + + mReadMethod = env->GetMethodID(datasourceclass, "readAt", "(J[BI)I"); + CHECK(mReadMethod != NULL); + + mGetSizeMethod = env->GetMethodID(datasourceclass, "getSize", "()J"); + CHECK(mGetSizeMethod != NULL); + + mCloseMethod = env->GetMethodID(datasourceclass, "close", "()V"); + CHECK(mCloseMethod != NULL); + } + + ~JavaDataSourceBridge() { + JNIEnv *env = AndroidRuntime::getJNIEnv(); + env->CallVoidMethod(mDataSource, mCloseMethod); + env->DeleteGlobalRef(mDataSource); + } + + virtual status_t initCheck() const { + return OK; + } + + virtual ssize_t readAt(off64_t offset, void* buffer, size_t size) { + JNIEnv *env = AndroidRuntime::getJNIEnv(); + + // XXX could optimize this by reusing the same array + jbyteArray byteArrayObj = env->NewByteArray(size); + env->DeleteLocalRef(env->GetObjectClass(mDataSource)); + env->DeleteLocalRef(env->GetObjectClass(byteArrayObj)); + ssize_t numread = env->CallIntMethod(mDataSource, mReadMethod, offset, byteArrayObj, size); + env->GetByteArrayRegion(byteArrayObj, 0, size, (jbyte*) buffer); + env->DeleteLocalRef(byteArrayObj); + if (env->ExceptionCheck()) { + ALOGW("Exception occurred while reading %d at %lld", size, offset); + LOGW_EX(env); + env->ExceptionClear(); + return -1; + } + return numread; + } + + virtual status_t getSize(off64_t *size) { + JNIEnv *env = AndroidRuntime::getJNIEnv(); + + CHECK(size != NULL); + + int64_t len = env->CallLongMethod(mDataSource, mGetSizeMethod); + if (len < 0) { + *size = ERROR_UNSUPPORTED; + } else { + *size = len; + } + return OK; + } +}; + //////////////////////////////////////////////////////////////////////////////// JMediaExtractor::JMediaExtractor(JNIEnv *env, jobject thiz) @@ -76,6 +142,10 @@ status_t JMediaExtractor::setDataSource(int fd, off64_t offset, off64_t size) { return mImpl->setDataSource(fd, offset, size); } +status_t JMediaExtractor::setDataSource(const sp<DataSource> &datasource) { + return mImpl->setDataSource(datasource); +} + size_t JMediaExtractor::countTracks() const { return mImpl->countTracks(); } @@ -625,6 +695,33 @@ static void android_media_MediaExtractor_setDataSourceFd( } } +static void android_media_MediaExtractor_setDataSourceCallback( + JNIEnv *env, jobject thiz, + jobject callbackObj) { + sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz); + + if (extractor == NULL) { + jniThrowException(env, "java/lang/IllegalStateException", NULL); + return; + } + + if (callbackObj == NULL) { + jniThrowException(env, "java/lang/IllegalArgumentException", NULL); + return; + } + + sp<JavaDataSourceBridge> bridge = new JavaDataSourceBridge(env, callbackObj); + status_t err = extractor->setDataSource(bridge); + + if (err != OK) { + jniThrowException( + env, + "java/io/IOException", + "Failed to instantiate extractor."); + return; + } +} + static jlong android_media_MediaExtractor_getCachedDurationUs( JNIEnv *env, jobject thiz) { sp<JMediaExtractor> extractor = getMediaExtractor(env, thiz); @@ -713,6 +810,9 @@ static JNINativeMethod gMethods[] = { { "setDataSource", "(Ljava/io/FileDescriptor;JJ)V", (void *)android_media_MediaExtractor_setDataSourceFd }, + { "setDataSource", "(Landroid/media/DataSource;)V", + (void *)android_media_MediaExtractor_setDataSourceCallback }, + { "getCachedDuration", "()J", (void *)android_media_MediaExtractor_getCachedDurationUs }, diff --git a/media/jni/android_media_MediaExtractor.h b/media/jni/android_media_MediaExtractor.h index 2d4627e..03900db 100644 --- a/media/jni/android_media_MediaExtractor.h +++ b/media/jni/android_media_MediaExtractor.h @@ -19,6 +19,7 @@ #include <media/stagefright/foundation/ABase.h> #include <media/stagefright/MediaSource.h> +#include <media/stagefright/DataSource.h> #include <utils/Errors.h> #include <utils/KeyedVector.h> #include <utils/RefBase.h> @@ -39,6 +40,7 @@ struct JMediaExtractor : public RefBase { const KeyedVector<String8, String8> *headers); status_t setDataSource(int fd, off64_t offset, off64_t size); + status_t setDataSource(const sp<DataSource> &source); size_t countTracks() const; status_t getTrackFormat(size_t index, jobject *format) const; |