fd2e50086c
b/10855561 Change-Id: I29b046b835ce0ca3644e5c2e8f9bca5c0b380d4b
1033 lines
29 KiB
C++
1033 lines
29 KiB
C++
/*
|
|
* 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 <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 <nativehelper/ScopedLocalRef.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 CryptoErrorCodes {
|
|
jint cryptoErrorNoKey;
|
|
jint cryptoErrorKeyExpired;
|
|
jint cryptoErrorResourceBusy;
|
|
} gCryptoErrorCodes;
|
|
|
|
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_FOREGROUND);
|
|
|
|
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<IGraphicBufferProducer> &bufferProducer,
|
|
const sp<ICrypto> &crypto,
|
|
int flags) {
|
|
sp<Surface> client;
|
|
if (bufferProducer != NULL) {
|
|
mSurfaceTextureClient = new Surface(bufferProducer, true /* controlledByApp */);
|
|
} else {
|
|
mSurfaceTextureClient.clear();
|
|
}
|
|
|
|
return mCodec->configure(format, mSurfaceTextureClient, crypto, flags);
|
|
}
|
|
|
|
status_t JMediaCodec::createInputSurface(
|
|
sp<IGraphicBufferProducer>* bufferProducer) {
|
|
return mCodec->createInputSurface(bufferProducer);
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
ScopedLocalRef<jclass> clazz(
|
|
env, env->FindClass("android/media/MediaCodec$BufferInfo"));
|
|
|
|
jmethodID method = env->GetMethodID(clazz.get(), "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::signalEndOfInputStream() {
|
|
return mCodec->signalEndOfInputStream();
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
ScopedLocalRef<jclass> byteBufferClass(
|
|
env, env->FindClass("java/nio/ByteBuffer"));
|
|
|
|
CHECK(byteBufferClass.get() != NULL);
|
|
|
|
jmethodID orderID = env->GetMethodID(
|
|
byteBufferClass.get(),
|
|
"order",
|
|
"(Ljava/nio/ByteOrder;)Ljava/nio/ByteBuffer;");
|
|
|
|
CHECK(orderID != NULL);
|
|
|
|
ScopedLocalRef<jclass> byteOrderClass(
|
|
env, env->FindClass("java/nio/ByteOrder"));
|
|
|
|
CHECK(byteOrderClass.get() != NULL);
|
|
|
|
jmethodID nativeOrderID = env->GetStaticMethodID(
|
|
byteOrderClass.get(), "nativeOrder", "()Ljava/nio/ByteOrder;");
|
|
CHECK(nativeOrderID != NULL);
|
|
|
|
jobject nativeByteOrderObj =
|
|
env->CallStaticObjectMethod(byteOrderClass.get(), nativeOrderID);
|
|
CHECK(nativeByteOrderObj != NULL);
|
|
|
|
*bufArray = (jobjectArray)env->NewObjectArray(
|
|
buffers.size(), byteBufferClass.get(), NULL);
|
|
if (*bufArray == NULL) {
|
|
env->DeleteLocalRef(nativeByteOrderObj);
|
|
return NO_MEMORY;
|
|
}
|
|
|
|
for (size_t i = 0; i < buffers.size(); ++i) {
|
|
const sp<ABuffer> &buffer = buffers.itemAt(i);
|
|
|
|
// if this is an ABuffer that doesn't actually hold any accessible memory,
|
|
// use a null ByteBuffer
|
|
if (buffer->base() == NULL) {
|
|
continue;
|
|
}
|
|
jobject byteBuffer =
|
|
env->NewDirectByteBuffer(
|
|
buffer->base(),
|
|
buffer->capacity());
|
|
if (byteBuffer == NULL) {
|
|
env->DeleteLocalRef(nativeByteOrderObj);
|
|
return NO_MEMORY;
|
|
}
|
|
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;
|
|
}
|
|
|
|
status_t JMediaCodec::getName(JNIEnv *env, jstring *nameStr) const {
|
|
AString name;
|
|
|
|
status_t err = mCodec->getName(&name);
|
|
|
|
if (err != OK) {
|
|
return err;
|
|
}
|
|
|
|
*nameStr = env->NewStringUTF(name.c_str());
|
|
|
|
return OK;
|
|
}
|
|
|
|
status_t JMediaCodec::setParameters(const sp<AMessage> &msg) {
|
|
return mCodec->setParameters(msg);
|
|
}
|
|
|
|
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) {
|
|
ScopedLocalRef<jclass> clazz(
|
|
env, env->FindClass("android/media/MediaCodec$CryptoException"));
|
|
CHECK(clazz.get() != NULL);
|
|
|
|
jmethodID constructID =
|
|
env->GetMethodID(clazz.get(), "<init>", "(ILjava/lang/String;)V");
|
|
CHECK(constructID != NULL);
|
|
|
|
jstring msgObj = env->NewStringUTF(msg != NULL ? msg : "Unknown Error");
|
|
|
|
/* translate OS errors to Java API CryptoException errorCodes */
|
|
switch (err) {
|
|
case ERROR_DRM_NO_LICENSE:
|
|
err = gCryptoErrorCodes.cryptoErrorNoKey;
|
|
break;
|
|
case ERROR_DRM_LICENSE_EXPIRED:
|
|
err = gCryptoErrorCodes.cryptoErrorKeyExpired;
|
|
break;
|
|
case ERROR_DRM_RESOURCE_BUSY:
|
|
err = gCryptoErrorCodes.cryptoErrorResourceBusy;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
jthrowable exception =
|
|
(jthrowable)env->NewObject(clazz.get(), constructID, err, msgObj);
|
|
|
|
env->Throw(exception);
|
|
}
|
|
|
|
static jint throwExceptionAsNecessary(
|
|
JNIEnv *env, status_t err, const char *msg = NULL) {
|
|
if (err >= ERROR_DRM_VENDOR_MIN && err <= ERROR_DRM_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;
|
|
|
|
case ERROR_DRM_NO_LICENSE:
|
|
case ERROR_DRM_LICENSE_EXPIRED:
|
|
case ERROR_DRM_RESOURCE_BUSY:
|
|
throwCryptoException(env, err, msg);
|
|
break;
|
|
|
|
default:
|
|
{
|
|
jniThrowException(env, "java/lang/IllegalStateException", msg);
|
|
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<IGraphicBufferProducer> bufferProducer;
|
|
if (jsurface != NULL) {
|
|
sp<Surface> surface(android_view_Surface_getSurface(env, jsurface));
|
|
if (surface != NULL) {
|
|
bufferProducer = surface->getIGraphicBufferProducer();
|
|
} 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, bufferProducer, crypto, flags);
|
|
|
|
throwExceptionAsNecessary(env, err);
|
|
}
|
|
|
|
static jobject android_media_MediaCodec_createInputSurface(JNIEnv* env,
|
|
jobject thiz) {
|
|
ALOGV("android_media_MediaCodec_createInputSurface");
|
|
|
|
sp<JMediaCodec> codec = getMediaCodec(env, thiz);
|
|
if (codec == NULL) {
|
|
jniThrowException(env, "java/lang/IllegalStateException", NULL);
|
|
return NULL;
|
|
}
|
|
|
|
// Tell the MediaCodec that we want to use a Surface as input.
|
|
sp<IGraphicBufferProducer> bufferProducer;
|
|
status_t err = codec->createInputSurface(&bufferProducer);
|
|
if (err != NO_ERROR) {
|
|
throwExceptionAsNecessary(env, err);
|
|
return NULL;
|
|
}
|
|
|
|
// Wrap the IGBP in a Java-language Surface.
|
|
return android_view_Surface_createFromIGraphicBufferProducer(env,
|
|
bufferProducer);
|
|
}
|
|
|
|
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", "no codec found");
|
|
return;
|
|
}
|
|
|
|
status_t err = codec->start();
|
|
|
|
throwExceptionAsNecessary(env, err, "start failed");
|
|
}
|
|
|
|
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 void android_media_MediaCodec_signalEndOfInputStream(JNIEnv* env,
|
|
jobject thiz) {
|
|
ALOGV("android_media_MediaCodec_signalEndOfInputStream");
|
|
|
|
sp<JMediaCodec> codec = getMediaCodec(env, thiz);
|
|
if (codec == NULL) {
|
|
jniThrowException(env, "java/lang/IllegalStateException", NULL);
|
|
return;
|
|
}
|
|
|
|
status_t err = codec->signalEndOfInputStream();
|
|
|
|
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;
|
|
}
|
|
|
|
// if we're out of memory, an exception was already thrown
|
|
if (err != NO_MEMORY) {
|
|
throwExceptionAsNecessary(env, err);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static jobject android_media_MediaCodec_getName(
|
|
JNIEnv *env, jobject thiz) {
|
|
ALOGV("android_media_MediaCodec_getName");
|
|
|
|
sp<JMediaCodec> codec = getMediaCodec(env, thiz);
|
|
|
|
if (codec == NULL) {
|
|
jniThrowException(env, "java/lang/IllegalStateException", NULL);
|
|
return NULL;
|
|
}
|
|
|
|
jstring name;
|
|
status_t err = codec->getName(env, &name);
|
|
|
|
if (err == OK) {
|
|
return name;
|
|
}
|
|
|
|
throwExceptionAsNecessary(env, err);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void android_media_MediaCodec_setParameters(
|
|
JNIEnv *env, jobject thiz, jobjectArray keys, jobjectArray vals) {
|
|
ALOGV("android_media_MediaCodec_setParameters");
|
|
|
|
sp<JMediaCodec> codec = getMediaCodec(env, thiz);
|
|
|
|
if (codec == NULL) {
|
|
jniThrowException(env, "java/lang/IllegalStateException", NULL);
|
|
return;
|
|
}
|
|
|
|
sp<AMessage> params;
|
|
status_t err = ConvertKeyValueArraysToMessage(env, keys, vals, ¶ms);
|
|
|
|
if (err == OK) {
|
|
err = codec->setParameters(params);
|
|
}
|
|
|
|
throwExceptionAsNecessary(env, err);
|
|
}
|
|
|
|
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) {
|
|
ScopedLocalRef<jclass> clazz(
|
|
env, env->FindClass("android/media/MediaCodec"));
|
|
CHECK(clazz.get() != NULL);
|
|
|
|
gFields.context = env->GetFieldID(clazz.get(), "mNativeContext", "I");
|
|
CHECK(gFields.context != NULL);
|
|
|
|
clazz.reset(env->FindClass("android/media/MediaCodec$CryptoInfo"));
|
|
CHECK(clazz.get() != NULL);
|
|
|
|
gFields.cryptoInfoNumSubSamplesID =
|
|
env->GetFieldID(clazz.get(), "numSubSamples", "I");
|
|
CHECK(gFields.cryptoInfoNumSubSamplesID != NULL);
|
|
|
|
gFields.cryptoInfoNumBytesOfClearDataID =
|
|
env->GetFieldID(clazz.get(), "numBytesOfClearData", "[I");
|
|
CHECK(gFields.cryptoInfoNumBytesOfClearDataID != NULL);
|
|
|
|
gFields.cryptoInfoNumBytesOfEncryptedDataID =
|
|
env->GetFieldID(clazz.get(), "numBytesOfEncryptedData", "[I");
|
|
CHECK(gFields.cryptoInfoNumBytesOfEncryptedDataID != NULL);
|
|
|
|
gFields.cryptoInfoKeyID = env->GetFieldID(clazz.get(), "key", "[B");
|
|
CHECK(gFields.cryptoInfoKeyID != NULL);
|
|
|
|
gFields.cryptoInfoIVID = env->GetFieldID(clazz.get(), "iv", "[B");
|
|
CHECK(gFields.cryptoInfoIVID != NULL);
|
|
|
|
gFields.cryptoInfoModeID = env->GetFieldID(clazz.get(), "mode", "I");
|
|
CHECK(gFields.cryptoInfoModeID != NULL);
|
|
|
|
clazz.reset(env->FindClass("android/media/MediaCodec$CryptoException"));
|
|
CHECK(clazz.get() != NULL);
|
|
|
|
jfieldID field;
|
|
field = env->GetStaticFieldID(clazz.get(), "ERROR_NO_KEY", "I");
|
|
CHECK(field != NULL);
|
|
gCryptoErrorCodes.cryptoErrorNoKey =
|
|
env->GetStaticIntField(clazz.get(), field);
|
|
|
|
field = env->GetStaticFieldID(clazz.get(), "ERROR_KEY_EXPIRED", "I");
|
|
CHECK(field != NULL);
|
|
gCryptoErrorCodes.cryptoErrorKeyExpired =
|
|
env->GetStaticIntField(clazz.get(), field);
|
|
|
|
field = env->GetStaticFieldID(clazz.get(), "ERROR_RESOURCE_BUSY", "I");
|
|
CHECK(field != NULL);
|
|
gCryptoErrorCodes.cryptoErrorResourceBusy =
|
|
env->GetStaticIntField(clazz.get(), field);
|
|
}
|
|
|
|
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 },
|
|
|
|
{ "createInputSurface", "()Landroid/view/Surface;",
|
|
(void *)android_media_MediaCodec_createInputSurface },
|
|
|
|
{ "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 },
|
|
|
|
{ "signalEndOfInputStream", "()V",
|
|
(void *)android_media_MediaCodec_signalEndOfInputStream },
|
|
|
|
{ "getOutputFormatNative", "()Ljava/util/Map;",
|
|
(void *)android_media_MediaCodec_getOutputFormatNative },
|
|
|
|
{ "getBuffers", "(Z)[Ljava/nio/ByteBuffer;",
|
|
(void *)android_media_MediaCodec_getBuffers },
|
|
|
|
{ "getName", "()Ljava/lang/String;",
|
|
(void *)android_media_MediaCodec_getName },
|
|
|
|
{ "setParameters", "([Ljava/lang/String;[Ljava/lang/Object;)V",
|
|
(void *)android_media_MediaCodec_setParameters },
|
|
|
|
{ "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));
|
|
}
|