| /* |
| * Copyright 2012, The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| //#define LOG_NDEBUG 0 |
| #define LOG_TAG "MediaCodec-JNI" |
| #include <utils/Log.h> |
| |
| #include "android_media_MediaCodec.h" |
| |
| #include "android_media_MediaCrypto.h" |
| #include "android_media_Utils.h" |
| #include "android_runtime/AndroidRuntime.h" |
| #include "android_runtime/android_view_Surface.h" |
| #include "jni.h" |
| #include "JNIHelp.h" |
| |
| #include <gui/Surface.h> |
| #include <gui/SurfaceTextureClient.h> |
| |
| #include <media/ICrypto.h> |
| #include <media/stagefright/MediaCodec.h> |
| #include <media/stagefright/foundation/ABuffer.h> |
| #include <media/stagefright/foundation/ADebug.h> |
| #include <media/stagefright/foundation/ALooper.h> |
| #include <media/stagefright/foundation/AMessage.h> |
| #include <media/stagefright/foundation/AString.h> |
| #include <media/stagefright/MediaErrors.h> |
| |
| #include <system/window.h> |
| |
| namespace android { |
| |
| // Keep these in sync with their equivalents in MediaCodec.java !!! |
| enum { |
| DEQUEUE_INFO_TRY_AGAIN_LATER = -1, |
| DEQUEUE_INFO_OUTPUT_FORMAT_CHANGED = -2, |
| DEQUEUE_INFO_OUTPUT_BUFFERS_CHANGED = -3, |
| }; |
| |
| struct fields_t { |
| jfieldID context; |
| |
| jfieldID cryptoInfoNumSubSamplesID; |
| jfieldID cryptoInfoNumBytesOfClearDataID; |
| jfieldID cryptoInfoNumBytesOfEncryptedDataID; |
| jfieldID cryptoInfoKeyID; |
| jfieldID cryptoInfoIVID; |
| jfieldID cryptoInfoModeID; |
| }; |
| |
| static fields_t gFields; |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| |
| JMediaCodec::JMediaCodec( |
| JNIEnv *env, jobject thiz, |
| const char *name, bool nameIsType, bool encoder) |
| : mClass(NULL), |
| mObject(NULL) { |
| jclass clazz = env->GetObjectClass(thiz); |
| CHECK(clazz != NULL); |
| |
| mClass = (jclass)env->NewGlobalRef(clazz); |
| mObject = env->NewWeakGlobalRef(thiz); |
| |
| mLooper = new ALooper; |
| mLooper->setName("MediaCodec_looper"); |
| |
| mLooper->start( |
| false, // runOnCallingThread |
| false, // canCallJava |
| PRIORITY_DEFAULT); |
| |
| if (nameIsType) { |
| mCodec = MediaCodec::CreateByType(mLooper, name, encoder); |
| } else { |
| mCodec = MediaCodec::CreateByComponentName(mLooper, name); |
| } |
| } |
| |
| status_t JMediaCodec::initCheck() const { |
| return mCodec != NULL ? OK : NO_INIT; |
| } |
| |
| JMediaCodec::~JMediaCodec() { |
| if (mCodec != NULL) { |
| mCodec->release(); |
| mCodec.clear(); |
| } |
| |
| JNIEnv *env = AndroidRuntime::getJNIEnv(); |
| |
| env->DeleteWeakGlobalRef(mObject); |
| mObject = NULL; |
| env->DeleteGlobalRef(mClass); |
| mClass = NULL; |
| } |
| |
| status_t JMediaCodec::configure( |
| const sp<AMessage> &format, |
| const sp<ISurfaceTexture> &surfaceTexture, |
| const sp<ICrypto> &crypto, |
| int flags) { |
| sp<SurfaceTextureClient> client; |
| if (surfaceTexture != NULL) { |
| mSurfaceTextureClient = new SurfaceTextureClient(surfaceTexture); |
| } else { |
| mSurfaceTextureClient.clear(); |
| } |
| |
| return mCodec->configure(format, mSurfaceTextureClient, crypto, flags); |
| } |
| |
| status_t JMediaCodec::start() { |
| return mCodec->start(); |
| } |
| |
| status_t JMediaCodec::stop() { |
| mSurfaceTextureClient.clear(); |
| |
| return mCodec->stop(); |
| } |
| |
| status_t JMediaCodec::flush() { |
| return mCodec->flush(); |
| } |
| |
| status_t JMediaCodec::queueInputBuffer( |
| size_t index, |
| size_t offset, size_t size, int64_t timeUs, uint32_t flags, |
| AString *errorDetailMsg) { |
| return mCodec->queueInputBuffer( |
| index, offset, size, timeUs, flags, errorDetailMsg); |
| } |
| |
| status_t JMediaCodec::queueSecureInputBuffer( |
| size_t index, |
| size_t offset, |
| const CryptoPlugin::SubSample *subSamples, |
| size_t numSubSamples, |
| const uint8_t key[16], |
| const uint8_t iv[16], |
| CryptoPlugin::Mode mode, |
| int64_t presentationTimeUs, |
| uint32_t flags, |
| AString *errorDetailMsg) { |
| return mCodec->queueSecureInputBuffer( |
| index, offset, subSamples, numSubSamples, key, iv, mode, |
| presentationTimeUs, flags, errorDetailMsg); |
| } |
| |
| status_t JMediaCodec::dequeueInputBuffer(size_t *index, int64_t timeoutUs) { |
| return mCodec->dequeueInputBuffer(index, timeoutUs); |
| } |
| |
| status_t JMediaCodec::dequeueOutputBuffer( |
| JNIEnv *env, jobject bufferInfo, size_t *index, int64_t timeoutUs) { |
| size_t size, offset; |
| int64_t timeUs; |
| uint32_t flags; |
| status_t err; |
| if ((err = mCodec->dequeueOutputBuffer( |
| index, &offset, &size, &timeUs, &flags, timeoutUs)) != OK) { |
| return err; |
| } |
| |
| jclass clazz = env->FindClass("android/media/MediaCodec$BufferInfo"); |
| |
| jmethodID method = env->GetMethodID(clazz, "set", "(IIJI)V"); |
| env->CallVoidMethod(bufferInfo, method, offset, size, timeUs, flags); |
| |
| return OK; |
| } |
| |
| status_t JMediaCodec::releaseOutputBuffer(size_t index, bool render) { |
| return render |
| ? mCodec->renderOutputBufferAndRelease(index) |
| : mCodec->releaseOutputBuffer(index); |
| } |
| |
| status_t JMediaCodec::getOutputFormat(JNIEnv *env, jobject *format) const { |
| sp<AMessage> msg; |
| status_t err; |
| if ((err = mCodec->getOutputFormat(&msg)) != OK) { |
| return err; |
| } |
| |
| return ConvertMessageToMap(env, msg, format); |
| } |
| |
| status_t JMediaCodec::getBuffers( |
| JNIEnv *env, bool input, jobjectArray *bufArray) const { |
| Vector<sp<ABuffer> > buffers; |
| |
| status_t err = |
| input |
| ? mCodec->getInputBuffers(&buffers) |
| : mCodec->getOutputBuffers(&buffers); |
| |
| if (err != OK) { |
| return err; |
| } |
| |
| jclass byteBufferClass = env->FindClass("java/nio/ByteBuffer"); |
| CHECK(byteBufferClass != NULL); |
| |
| jmethodID orderID = env->GetMethodID( |
| byteBufferClass, |
| "order", |
| "(Ljava/nio/ByteOrder;)Ljava/nio/ByteBuffer;"); |
| |
| CHECK(orderID != NULL); |
| |
| jclass byteOrderClass = env->FindClass("java/nio/ByteOrder"); |
| CHECK(byteOrderClass != NULL); |
| |
| jmethodID nativeOrderID = env->GetStaticMethodID( |
| byteOrderClass, "nativeOrder", "()Ljava/nio/ByteOrder;"); |
| CHECK(nativeOrderID != NULL); |
| |
| jobject nativeByteOrderObj = |
| env->CallStaticObjectMethod(byteOrderClass, nativeOrderID); |
| CHECK(nativeByteOrderObj != NULL); |
| |
| *bufArray = (jobjectArray)env->NewObjectArray( |
| buffers.size(), byteBufferClass, NULL); |
| |
| for (size_t i = 0; i < buffers.size(); ++i) { |
| const sp<ABuffer> &buffer = buffers.itemAt(i); |
| |
| jobject byteBuffer = |
| env->NewDirectByteBuffer( |
| buffer->base(), |
| buffer->capacity()); |
| |
| jobject me = env->CallObjectMethod( |
| byteBuffer, orderID, nativeByteOrderObj); |
| env->DeleteLocalRef(me); |
| me = NULL; |
| |
| env->SetObjectArrayElement( |
| *bufArray, i, byteBuffer); |
| |
| env->DeleteLocalRef(byteBuffer); |
| byteBuffer = NULL; |
| } |
| |
| env->DeleteLocalRef(nativeByteOrderObj); |
| nativeByteOrderObj = NULL; |
| |
| return OK; |
| } |
| |
| void JMediaCodec::setVideoScalingMode(int mode) { |
| if (mSurfaceTextureClient != NULL) { |
| native_window_set_scaling_mode(mSurfaceTextureClient.get(), mode); |
| } |
| } |
| |
| } // namespace android |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| |
| using namespace android; |
| |
| static sp<JMediaCodec> setMediaCodec( |
| JNIEnv *env, jobject thiz, const sp<JMediaCodec> &codec) { |
| sp<JMediaCodec> old = (JMediaCodec *)env->GetIntField(thiz, gFields.context); |
| if (codec != NULL) { |
| codec->incStrong(thiz); |
| } |
| if (old != NULL) { |
| old->decStrong(thiz); |
| } |
| env->SetIntField(thiz, gFields.context, (int)codec.get()); |
| |
| return old; |
| } |
| |
| static sp<JMediaCodec> getMediaCodec(JNIEnv *env, jobject thiz) { |
| return (JMediaCodec *)env->GetIntField(thiz, gFields.context); |
| } |
| |
| static void android_media_MediaCodec_release(JNIEnv *env, jobject thiz) { |
| setMediaCodec(env, thiz, NULL); |
| } |
| |
| static void throwCryptoException(JNIEnv *env, status_t err, const char *msg) { |
| jclass clazz = env->FindClass("android/media/MediaCodec$CryptoException"); |
| CHECK(clazz != NULL); |
| |
| jmethodID constructID = |
| env->GetMethodID(clazz, "<init>", "(ILjava/lang/String;)V"); |
| CHECK(constructID != NULL); |
| |
| jstring msgObj = env->NewStringUTF(msg != NULL ? msg : "Unknown Error"); |
| |
| jthrowable exception = |
| (jthrowable)env->NewObject(clazz, constructID, err, msgObj); |
| |
| env->Throw(exception); |
| } |
| |
| static jint throwExceptionAsNecessary( |
| JNIEnv *env, status_t err, const char *msg = NULL) { |
| if (err >= ERROR_DRM_WV_VENDOR_MIN && err <= ERROR_DRM_WV_VENDOR_MAX) { |
| // We'll throw our custom MediaCodec.CryptoException |
| |
| throwCryptoException(env, err, msg); |
| return 0; |
| } |
| |
| switch (err) { |
| case OK: |
| return 0; |
| |
| case -EAGAIN: |
| return DEQUEUE_INFO_TRY_AGAIN_LATER; |
| |
| case INFO_FORMAT_CHANGED: |
| return DEQUEUE_INFO_OUTPUT_FORMAT_CHANGED; |
| |
| case INFO_OUTPUT_BUFFERS_CHANGED: |
| return DEQUEUE_INFO_OUTPUT_BUFFERS_CHANGED; |
| |
| default: |
| { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| break; |
| } |
| } |
| |
| return 0; |
| } |
| |
| static void android_media_MediaCodec_native_configure( |
| JNIEnv *env, |
| jobject thiz, |
| jobjectArray keys, jobjectArray values, |
| jobject jsurface, |
| jobject jcrypto, |
| jint flags) { |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| sp<AMessage> format; |
| status_t err = ConvertKeyValueArraysToMessage(env, keys, values, &format); |
| |
| if (err != OK) { |
| jniThrowException(env, "java/lang/IllegalArgumentException", NULL); |
| return; |
| } |
| |
| sp<ISurfaceTexture> surfaceTexture; |
| if (jsurface != NULL) { |
| sp<Surface> surface(android_view_Surface_getSurface(env, jsurface)); |
| if (surface != NULL) { |
| surfaceTexture = surface->getSurfaceTexture(); |
| } else { |
| jniThrowException( |
| env, |
| "java/lang/IllegalArgumentException", |
| "The surface has been released"); |
| return; |
| } |
| } |
| |
| sp<ICrypto> crypto; |
| if (jcrypto != NULL) { |
| crypto = JCrypto::GetCrypto(env, jcrypto); |
| } |
| |
| err = codec->configure(format, surfaceTexture, crypto, flags); |
| |
| throwExceptionAsNecessary(env, err); |
| } |
| |
| static void android_media_MediaCodec_start(JNIEnv *env, jobject thiz) { |
| ALOGV("android_media_MediaCodec_start"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| status_t err = codec->start(); |
| |
| throwExceptionAsNecessary(env, err); |
| } |
| |
| static void android_media_MediaCodec_stop(JNIEnv *env, jobject thiz) { |
| ALOGV("android_media_MediaCodec_stop"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| status_t err = codec->stop(); |
| |
| throwExceptionAsNecessary(env, err); |
| } |
| |
| static void android_media_MediaCodec_flush(JNIEnv *env, jobject thiz) { |
| ALOGV("android_media_MediaCodec_flush"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| status_t err = codec->flush(); |
| |
| throwExceptionAsNecessary(env, err); |
| } |
| |
| static void android_media_MediaCodec_queueInputBuffer( |
| JNIEnv *env, |
| jobject thiz, |
| jint index, |
| jint offset, |
| jint size, |
| jlong timestampUs, |
| jint flags) { |
| ALOGV("android_media_MediaCodec_queueInputBuffer"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| AString errorDetailMsg; |
| |
| status_t err = codec->queueInputBuffer( |
| index, offset, size, timestampUs, flags, &errorDetailMsg); |
| |
| throwExceptionAsNecessary( |
| env, err, errorDetailMsg.empty() ? NULL : errorDetailMsg.c_str()); |
| } |
| |
| static void android_media_MediaCodec_queueSecureInputBuffer( |
| JNIEnv *env, |
| jobject thiz, |
| jint index, |
| jint offset, |
| jobject cryptoInfoObj, |
| jlong timestampUs, |
| jint flags) { |
| ALOGV("android_media_MediaCodec_queueSecureInputBuffer"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| jint numSubSamples = |
| env->GetIntField(cryptoInfoObj, gFields.cryptoInfoNumSubSamplesID); |
| |
| jintArray numBytesOfClearDataObj = |
| (jintArray)env->GetObjectField( |
| cryptoInfoObj, gFields.cryptoInfoNumBytesOfClearDataID); |
| |
| jintArray numBytesOfEncryptedDataObj = |
| (jintArray)env->GetObjectField( |
| cryptoInfoObj, gFields.cryptoInfoNumBytesOfEncryptedDataID); |
| |
| jbyteArray keyObj = |
| (jbyteArray)env->GetObjectField(cryptoInfoObj, gFields.cryptoInfoKeyID); |
| |
| jbyteArray ivObj = |
| (jbyteArray)env->GetObjectField(cryptoInfoObj, gFields.cryptoInfoIVID); |
| |
| jint mode = env->GetIntField(cryptoInfoObj, gFields.cryptoInfoModeID); |
| |
| status_t err = OK; |
| |
| CryptoPlugin::SubSample *subSamples = NULL; |
| jbyte *key = NULL; |
| jbyte *iv = NULL; |
| |
| if (numSubSamples <= 0) { |
| err = -EINVAL; |
| } else if (numBytesOfClearDataObj == NULL |
| && numBytesOfEncryptedDataObj == NULL) { |
| err = -EINVAL; |
| } else if (numBytesOfEncryptedDataObj != NULL |
| && env->GetArrayLength(numBytesOfEncryptedDataObj) < numSubSamples) { |
| err = -ERANGE; |
| } else if (numBytesOfClearDataObj != NULL |
| && env->GetArrayLength(numBytesOfClearDataObj) < numSubSamples) { |
| err = -ERANGE; |
| } else { |
| jboolean isCopy; |
| |
| jint *numBytesOfClearData = |
| (numBytesOfClearDataObj == NULL) |
| ? NULL |
| : env->GetIntArrayElements(numBytesOfClearDataObj, &isCopy); |
| |
| jint *numBytesOfEncryptedData = |
| (numBytesOfEncryptedDataObj == NULL) |
| ? NULL |
| : env->GetIntArrayElements(numBytesOfEncryptedDataObj, &isCopy); |
| |
| subSamples = new CryptoPlugin::SubSample[numSubSamples]; |
| |
| for (jint i = 0; i < numSubSamples; ++i) { |
| subSamples[i].mNumBytesOfClearData = |
| (numBytesOfClearData == NULL) ? 0 : numBytesOfClearData[i]; |
| |
| subSamples[i].mNumBytesOfEncryptedData = |
| (numBytesOfEncryptedData == NULL) |
| ? 0 : numBytesOfEncryptedData[i]; |
| } |
| |
| if (numBytesOfEncryptedData != NULL) { |
| env->ReleaseIntArrayElements( |
| numBytesOfEncryptedDataObj, numBytesOfEncryptedData, 0); |
| numBytesOfEncryptedData = NULL; |
| } |
| |
| if (numBytesOfClearData != NULL) { |
| env->ReleaseIntArrayElements( |
| numBytesOfClearDataObj, numBytesOfClearData, 0); |
| numBytesOfClearData = NULL; |
| } |
| } |
| |
| if (err == OK && keyObj != NULL) { |
| if (env->GetArrayLength(keyObj) != 16) { |
| err = -EINVAL; |
| } else { |
| jboolean isCopy; |
| key = env->GetByteArrayElements(keyObj, &isCopy); |
| } |
| } |
| |
| if (err == OK && ivObj != NULL) { |
| if (env->GetArrayLength(ivObj) != 16) { |
| err = -EINVAL; |
| } else { |
| jboolean isCopy; |
| iv = env->GetByteArrayElements(ivObj, &isCopy); |
| } |
| } |
| |
| AString errorDetailMsg; |
| |
| if (err == OK) { |
| err = codec->queueSecureInputBuffer( |
| index, offset, |
| subSamples, numSubSamples, |
| (const uint8_t *)key, (const uint8_t *)iv, |
| (CryptoPlugin::Mode)mode, |
| timestampUs, |
| flags, |
| &errorDetailMsg); |
| } |
| |
| if (iv != NULL) { |
| env->ReleaseByteArrayElements(ivObj, iv, 0); |
| iv = NULL; |
| } |
| |
| if (key != NULL) { |
| env->ReleaseByteArrayElements(keyObj, key, 0); |
| key = NULL; |
| } |
| |
| delete[] subSamples; |
| subSamples = NULL; |
| |
| throwExceptionAsNecessary( |
| env, err, errorDetailMsg.empty() ? NULL : errorDetailMsg.c_str()); |
| } |
| |
| static jint android_media_MediaCodec_dequeueInputBuffer( |
| JNIEnv *env, jobject thiz, jlong timeoutUs) { |
| ALOGV("android_media_MediaCodec_dequeueInputBuffer"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return -1; |
| } |
| |
| size_t index; |
| status_t err = codec->dequeueInputBuffer(&index, timeoutUs); |
| |
| if (err == OK) { |
| return index; |
| } |
| |
| return throwExceptionAsNecessary(env, err); |
| } |
| |
| static jint android_media_MediaCodec_dequeueOutputBuffer( |
| JNIEnv *env, jobject thiz, jobject bufferInfo, jlong timeoutUs) { |
| ALOGV("android_media_MediaCodec_dequeueOutputBuffer"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return 0; |
| } |
| |
| size_t index; |
| status_t err = codec->dequeueOutputBuffer( |
| env, bufferInfo, &index, timeoutUs); |
| |
| if (err == OK) { |
| return index; |
| } |
| |
| return throwExceptionAsNecessary(env, err); |
| } |
| |
| static void android_media_MediaCodec_releaseOutputBuffer( |
| JNIEnv *env, jobject thiz, jint index, jboolean render) { |
| ALOGV("android_media_MediaCodec_renderOutputBufferAndRelease"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| status_t err = codec->releaseOutputBuffer(index, render); |
| |
| throwExceptionAsNecessary(env, err); |
| } |
| |
| static jobject android_media_MediaCodec_getOutputFormatNative( |
| JNIEnv *env, jobject thiz) { |
| ALOGV("android_media_MediaCodec_getOutputFormatNative"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return NULL; |
| } |
| |
| jobject format; |
| status_t err = codec->getOutputFormat(env, &format); |
| |
| if (err == OK) { |
| return format; |
| } |
| |
| throwExceptionAsNecessary(env, err); |
| |
| return NULL; |
| } |
| |
| static jobjectArray android_media_MediaCodec_getBuffers( |
| JNIEnv *env, jobject thiz, jboolean input) { |
| ALOGV("android_media_MediaCodec_getBuffers"); |
| |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return NULL; |
| } |
| |
| jobjectArray buffers; |
| status_t err = codec->getBuffers(env, input, &buffers); |
| |
| if (err == OK) { |
| return buffers; |
| } |
| |
| throwExceptionAsNecessary(env, err); |
| |
| return NULL; |
| } |
| |
| static void android_media_MediaCodec_setVideoScalingMode( |
| JNIEnv *env, jobject thiz, jint mode) { |
| sp<JMediaCodec> codec = getMediaCodec(env, thiz); |
| |
| if (codec == NULL) { |
| jniThrowException(env, "java/lang/IllegalStateException", NULL); |
| return; |
| } |
| |
| if (mode != NATIVE_WINDOW_SCALING_MODE_SCALE_TO_WINDOW |
| && mode != NATIVE_WINDOW_SCALING_MODE_SCALE_CROP) { |
| jniThrowException(env, "java/lang/InvalidArgumentException", NULL); |
| return; |
| } |
| |
| codec->setVideoScalingMode(mode); |
| } |
| |
| static void android_media_MediaCodec_native_init(JNIEnv *env) { |
| jclass clazz = env->FindClass("android/media/MediaCodec"); |
| CHECK(clazz != NULL); |
| |
| gFields.context = env->GetFieldID(clazz, "mNativeContext", "I"); |
| CHECK(gFields.context != NULL); |
| |
| clazz = env->FindClass("android/media/MediaCodec$CryptoInfo"); |
| CHECK(clazz != NULL); |
| |
| gFields.cryptoInfoNumSubSamplesID = |
| env->GetFieldID(clazz, "numSubSamples", "I"); |
| CHECK(gFields.cryptoInfoNumSubSamplesID != NULL); |
| |
| gFields.cryptoInfoNumBytesOfClearDataID = |
| env->GetFieldID(clazz, "numBytesOfClearData", "[I"); |
| CHECK(gFields.cryptoInfoNumBytesOfClearDataID != NULL); |
| |
| gFields.cryptoInfoNumBytesOfEncryptedDataID = |
| env->GetFieldID(clazz, "numBytesOfEncryptedData", "[I"); |
| CHECK(gFields.cryptoInfoNumBytesOfEncryptedDataID != NULL); |
| |
| gFields.cryptoInfoKeyID = env->GetFieldID(clazz, "key", "[B"); |
| CHECK(gFields.cryptoInfoKeyID != NULL); |
| |
| gFields.cryptoInfoIVID = env->GetFieldID(clazz, "iv", "[B"); |
| CHECK(gFields.cryptoInfoIVID != NULL); |
| |
| gFields.cryptoInfoModeID = env->GetFieldID(clazz, "mode", "I"); |
| CHECK(gFields.cryptoInfoModeID != NULL); |
| } |
| |
| static void android_media_MediaCodec_native_setup( |
| JNIEnv *env, jobject thiz, |
| jstring name, jboolean nameIsType, jboolean encoder) { |
| if (name == NULL) { |
| jniThrowException(env, "java/lang/IllegalArgumentException", NULL); |
| return; |
| } |
| |
| const char *tmp = env->GetStringUTFChars(name, NULL); |
| |
| if (tmp == NULL) { |
| return; |
| } |
| |
| sp<JMediaCodec> codec = new JMediaCodec(env, thiz, tmp, nameIsType, encoder); |
| |
| status_t err = codec->initCheck(); |
| |
| env->ReleaseStringUTFChars(name, tmp); |
| tmp = NULL; |
| |
| if (err != OK) { |
| jniThrowException( |
| env, |
| "java/io/IOException", |
| "Failed to allocate component instance"); |
| return; |
| } |
| |
| setMediaCodec(env,thiz, codec); |
| } |
| |
| static void android_media_MediaCodec_native_finalize( |
| JNIEnv *env, jobject thiz) { |
| android_media_MediaCodec_release(env, thiz); |
| } |
| |
| static JNINativeMethod gMethods[] = { |
| { "release", "()V", (void *)android_media_MediaCodec_release }, |
| |
| { "native_configure", |
| "([Ljava/lang/String;[Ljava/lang/Object;Landroid/view/Surface;" |
| "Landroid/media/MediaCrypto;I)V", |
| (void *)android_media_MediaCodec_native_configure }, |
| |
| { "start", "()V", (void *)android_media_MediaCodec_start }, |
| { "stop", "()V", (void *)android_media_MediaCodec_stop }, |
| { "flush", "()V", (void *)android_media_MediaCodec_flush }, |
| |
| { "queueInputBuffer", "(IIIJI)V", |
| (void *)android_media_MediaCodec_queueInputBuffer }, |
| |
| { "queueSecureInputBuffer", "(IILandroid/media/MediaCodec$CryptoInfo;JI)V", |
| (void *)android_media_MediaCodec_queueSecureInputBuffer }, |
| |
| { "dequeueInputBuffer", "(J)I", |
| (void *)android_media_MediaCodec_dequeueInputBuffer }, |
| |
| { "dequeueOutputBuffer", "(Landroid/media/MediaCodec$BufferInfo;J)I", |
| (void *)android_media_MediaCodec_dequeueOutputBuffer }, |
| |
| { "releaseOutputBuffer", "(IZ)V", |
| (void *)android_media_MediaCodec_releaseOutputBuffer }, |
| |
| { "getOutputFormatNative", "()Ljava/util/Map;", |
| (void *)android_media_MediaCodec_getOutputFormatNative }, |
| |
| { "getBuffers", "(Z)[Ljava/nio/ByteBuffer;", |
| (void *)android_media_MediaCodec_getBuffers }, |
| |
| { "setVideoScalingMode", "(I)V", |
| (void *)android_media_MediaCodec_setVideoScalingMode }, |
| |
| { "native_init", "()V", (void *)android_media_MediaCodec_native_init }, |
| |
| { "native_setup", "(Ljava/lang/String;ZZ)V", |
| (void *)android_media_MediaCodec_native_setup }, |
| |
| { "native_finalize", "()V", |
| (void *)android_media_MediaCodec_native_finalize }, |
| }; |
| |
| int register_android_media_MediaCodec(JNIEnv *env) { |
| return AndroidRuntime::registerNativeMethods(env, |
| "android/media/MediaCodec", gMethods, NELEM(gMethods)); |
| } |