/* * 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 <media/stagefright/foundation/ADebug.h> #include <media/stagefright/foundation/AMessage.h> #include <media/stagefright/MediaCodecList.h> #include <media/IMediaCodecList.h> #include <media/MediaCodecInfo.h> #include "android_runtime/AndroidRuntime.h" #include "jni.h" #include "JNIHelp.h" #include "android_media_Utils.h" using namespace android; static sp<IMediaCodecList> getCodecList(JNIEnv *env) { sp<IMediaCodecList> mcl = MediaCodecList::getInstance(); if (mcl == NULL) { // This should never happen unless something is really wrong jniThrowException( env, "java/lang/RuntimeException", "cannot get MediaCodecList"); } return mcl; } static jint android_media_MediaCodecList_getCodecCount( JNIEnv *env, jobject thiz) { sp<IMediaCodecList> mcl = getCodecList(env); if (mcl == NULL) { // Runtime exception already pending. return 0; } return mcl->countCodecs(); } static jstring android_media_MediaCodecList_getCodecName( JNIEnv *env, jobject thiz, jint index) { sp<IMediaCodecList> mcl = getCodecList(env); if (mcl == NULL) { // Runtime exception already pending. return NULL; } const sp<MediaCodecInfo> &info = mcl->getCodecInfo(index); if (info == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return NULL; } const char *name = info->getCodecName(); return env->NewStringUTF(name); } static jint android_media_MediaCodecList_findCodecByName( JNIEnv *env, jobject thiz, jstring name) { if (name == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return -ENOENT; } const char *nameStr = env->GetStringUTFChars(name, NULL); if (nameStr == NULL) { // Out of memory exception already pending. return -ENOENT; } sp<IMediaCodecList> mcl = getCodecList(env); if (mcl == NULL) { // Runtime exception already pending. env->ReleaseStringUTFChars(name, nameStr); return -ENOENT; } jint ret = mcl->findCodecByName(nameStr); env->ReleaseStringUTFChars(name, nameStr); return ret; } static jboolean android_media_MediaCodecList_isEncoder( JNIEnv *env, jobject thiz, jint index) { sp<IMediaCodecList> mcl = getCodecList(env); if (mcl == NULL) { // Runtime exception already pending. return false; } const sp<MediaCodecInfo> &info = mcl->getCodecInfo(index); if (info == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return false; } return info->isEncoder(); } static jarray android_media_MediaCodecList_getSupportedTypes( JNIEnv *env, jobject thiz, jint index) { sp<IMediaCodecList> mcl = getCodecList(env); if (mcl == NULL) { // Runtime exception already pending. return NULL; } const sp<MediaCodecInfo> &info = mcl->getCodecInfo(index); if (info == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return NULL; } Vector<AString> types; info->getSupportedMimes(&types); jclass clazz = env->FindClass("java/lang/String"); CHECK(clazz != NULL); jobjectArray array = env->NewObjectArray(types.size(), clazz, NULL); for (size_t i = 0; i < types.size(); ++i) { jstring obj = env->NewStringUTF(types.itemAt(i).c_str()); env->SetObjectArrayElement(array, i, obj); env->DeleteLocalRef(obj); obj = NULL; } return array; } static jobject android_media_MediaCodecList_getCodecCapabilities( JNIEnv *env, jobject thiz, jint index, jstring type) { if (type == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return NULL; } sp<IMediaCodecList> mcl = getCodecList(env); if (mcl == NULL) { // Runtime exception already pending. return NULL; } const sp<MediaCodecInfo> &info = mcl->getCodecInfo(index); if (info == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return NULL; } const char *typeStr = env->GetStringUTFChars(type, NULL); if (typeStr == NULL) { // Out of memory exception already pending. return NULL; } Vector<MediaCodecInfo::ProfileLevel> profileLevels; Vector<uint32_t> colorFormats; sp<AMessage> defaultFormat = new AMessage(); defaultFormat->setString("mime", typeStr); // TODO query default-format also from codec/codec list const sp<MediaCodecInfo::Capabilities> &capabilities = info->getCapabilitiesFor(typeStr); env->ReleaseStringUTFChars(type, typeStr); typeStr = NULL; if (capabilities == NULL) { jniThrowException(env, "java/lang/IllegalArgumentException", NULL); return NULL; } capabilities->getSupportedColorFormats(&colorFormats); capabilities->getSupportedProfileLevels(&profileLevels); uint32_t flags = capabilities->getFlags(); sp<AMessage> details = capabilities->getDetails(); bool isEncoder = info->isEncoder(); jobject defaultFormatObj = NULL; if (ConvertMessageToMap(env, defaultFormat, &defaultFormatObj)) { return NULL; } jobject infoObj = NULL; if (ConvertMessageToMap(env, details, &infoObj)) { env->DeleteLocalRef(defaultFormatObj); return NULL; } jclass capsClazz = env->FindClass("android/media/MediaCodecInfo$CodecCapabilities"); CHECK(capsClazz != NULL); jclass profileLevelClazz = env->FindClass("android/media/MediaCodecInfo$CodecProfileLevel"); CHECK(profileLevelClazz != NULL); jobjectArray profileLevelArray = env->NewObjectArray(profileLevels.size(), profileLevelClazz, NULL); jfieldID profileField = env->GetFieldID(profileLevelClazz, "profile", "I"); jfieldID levelField = env->GetFieldID(profileLevelClazz, "level", "I"); for (size_t i = 0; i < profileLevels.size(); ++i) { const MediaCodecInfo::ProfileLevel &src = profileLevels.itemAt(i); jobject profileLevelObj = env->AllocObject(profileLevelClazz); env->SetIntField(profileLevelObj, profileField, src.mProfile); env->SetIntField(profileLevelObj, levelField, src.mLevel); env->SetObjectArrayElement(profileLevelArray, i, profileLevelObj); env->DeleteLocalRef(profileLevelObj); profileLevelObj = NULL; } jintArray colorFormatsArray = env->NewIntArray(colorFormats.size()); for (size_t i = 0; i < colorFormats.size(); ++i) { jint val = colorFormats.itemAt(i); env->SetIntArrayRegion(colorFormatsArray, i, 1, &val); } jmethodID capsConstructID = env->GetMethodID(capsClazz, "<init>", "([Landroid/media/MediaCodecInfo$CodecProfileLevel;[IZI" "Ljava/util/Map;Ljava/util/Map;)V"); jobject caps = env->NewObject(capsClazz, capsConstructID, profileLevelArray, colorFormatsArray, isEncoder, flags, defaultFormatObj, infoObj); env->DeleteLocalRef(profileLevelArray); profileLevelArray = NULL; env->DeleteLocalRef(colorFormatsArray); colorFormatsArray = NULL; env->DeleteLocalRef(defaultFormatObj); defaultFormatObj = NULL; env->DeleteLocalRef(infoObj); infoObj = NULL; return caps; } static void android_media_MediaCodecList_native_init(JNIEnv *env) { } static JNINativeMethod gMethods[] = { { "native_getCodecCount", "()I", (void *)android_media_MediaCodecList_getCodecCount }, { "getCodecName", "(I)Ljava/lang/String;", (void *)android_media_MediaCodecList_getCodecName }, { "isEncoder", "(I)Z", (void *)android_media_MediaCodecList_isEncoder }, { "getSupportedTypes", "(I)[Ljava/lang/String;", (void *)android_media_MediaCodecList_getSupportedTypes }, { "getCodecCapabilities", "(ILjava/lang/String;)Landroid/media/MediaCodecInfo$CodecCapabilities;", (void *)android_media_MediaCodecList_getCodecCapabilities }, { "findCodecByName", "(Ljava/lang/String;)I", (void *)android_media_MediaCodecList_findCodecByName }, { "native_init", "()V", (void *)android_media_MediaCodecList_native_init }, }; int register_android_media_MediaCodecList(JNIEnv *env) { return AndroidRuntime::registerNativeMethods(env, "android/media/MediaCodecList", gMethods, NELEM(gMethods)); }