blob: 51f30953c1945fc15e6c61bddd5d324b3163d730 [file] [log] [blame]
// Copyright 2015 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/base/android/media_codec_util.h"
#include <stddef.h>
#include <algorithm>
#include <vector>
#include "base/android/build_info.h"
#include "base/android/jni_android.h"
#include "base/android/jni_array.h"
#include "base/android/jni_string.h"
#include "base/logging.h"
#include "base/strings/string_piece.h"
#include "base/strings/string_util.h"
#include "jni/CodecProfileLevelList_jni.h"
#include "jni/MediaCodecUtil_jni.h"
#include "media/base/android/media_codec_bridge.h"
#include "media/base/video_codecs.h"
#include "url/gurl.h"
using base::android::AttachCurrentThread;
using base::android::ConvertJavaStringToUTF8;
using base::android::ConvertUTF8ToJavaString;
using base::android::JavaIntArrayToIntVector;
using base::android::JavaRef;
using base::android::ScopedJavaLocalRef;
using base::android::SDK_VERSION_JELLY_BEAN_MR2;
using base::android::SDK_VERSION_KITKAT;
using base::android::SDK_VERSION_LOLLIPOP;
using base::android::SDK_VERSION_LOLLIPOP_MR1;
namespace media {
namespace {
const char kMp3MimeType[] = "audio/mpeg";
const char kAacMimeType[] = "audio/mp4a-latm";
const char kOpusMimeType[] = "audio/opus";
const char kVorbisMimeType[] = "audio/vorbis";
const char kAc3MimeType[] = "audio/ac3";
const char kEac3MimeType[] = "audio/eac3";
const char kBitstreamAudioMimeType[] = "audio/raw";
const char kAvcMimeType[] = "video/avc";
const char kHevcMimeType[] = "video/hevc";
const char kVp8MimeType[] = "video/x-vnd.on2.vp8";
const char kVp9MimeType[] = "video/x-vnd.on2.vp9";
} // namespace
static CodecProfileLevel MediaCodecProfileLevelToChromiumProfileLevel(
JNIEnv* env,
const JavaRef<jobject>& j_codec_profile_level) {
VideoCodec codec = static_cast<VideoCodec>(
Java_CodecProfileLevelAdapter_getCodec(env, j_codec_profile_level));
VideoCodecProfile profile = static_cast<VideoCodecProfile>(
Java_CodecProfileLevelAdapter_getProfile(env, j_codec_profile_level));
int level =
Java_CodecProfileLevelAdapter_getLevel(env, j_codec_profile_level);
return {codec, profile, level};
}
static bool IsSupportedAndroidMimeType(const std::string& mime_type) {
std::vector<std::string> supported{
kMp3MimeType, kAacMimeType, kOpusMimeType, kVorbisMimeType,
kAvcMimeType, kHevcMimeType, kVp8MimeType, kVp9MimeType};
return std::find(supported.begin(), supported.end(), mime_type) !=
supported.end();
}
static std::string GetDefaultCodecName(const std::string& mime_type,
MediaCodecDirection direction,
bool requires_software_codec) {
DCHECK(MediaCodecUtil::IsMediaCodecAvailable());
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jstring> j_mime = ConvertUTF8ToJavaString(env, mime_type);
ScopedJavaLocalRef<jstring> j_codec_name =
Java_MediaCodecUtil_getDefaultCodecName(
env, j_mime, static_cast<int>(direction), requires_software_codec);
return ConvertJavaStringToUTF8(env, j_codec_name.obj());
}
static bool IsDecoderSupportedByDevice(const std::string& android_mime_type) {
DCHECK(MediaCodecUtil::IsMediaCodecAvailable());
DCHECK(IsSupportedAndroidMimeType(android_mime_type));
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jstring> j_mime =
ConvertUTF8ToJavaString(env, android_mime_type);
return Java_MediaCodecUtil_isDecoderSupportedForDevice(env, j_mime);
}
static bool IsEncoderSupportedByDevice(const std::string& android_mime_type) {
DCHECK(MediaCodecUtil::IsMediaCodecAvailable());
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jstring> j_mime =
ConvertUTF8ToJavaString(env, android_mime_type);
return Java_MediaCodecUtil_isEncoderSupportedByDevice(env, j_mime);
}
// static
std::string MediaCodecUtil::CodecToAndroidMimeType(AudioCodec codec) {
if (IsPassthroughAudioFormat(codec))
return kBitstreamAudioMimeType;
switch (codec) {
case kCodecMP3:
return kMp3MimeType;
case kCodecVorbis:
return kVorbisMimeType;
case kCodecOpus:
return kOpusMimeType;
case kCodecAAC:
return kAacMimeType;
case kCodecAC3:
return kAc3MimeType;
case kCodecEAC3:
return kEac3MimeType;
default:
return std::string();
}
}
// static
std::string MediaCodecUtil::CodecToAndroidMimeType(VideoCodec codec) {
switch (codec) {
case kCodecH264:
return kAvcMimeType;
case kCodecHEVC:
return kHevcMimeType;
case kCodecVP8:
return kVp8MimeType;
case kCodecVP9:
return kVp9MimeType;
default:
return std::string();
}
}
// static
bool MediaCodecUtil::IsMediaCodecAvailable() {
return IsMediaCodecAvailableFor(
base::android::BuildInfo::GetInstance()->sdk_int(),
base::android::BuildInfo::GetInstance()->model());
}
// static
bool MediaCodecUtil::IsMediaCodecAvailableFor(int sdk, const char* model) {
// We will blacklist the model on any sdk that is as old or older than
// |last_bad_sdk| for the given model.
struct BlacklistEntry {
BlacklistEntry(const char* m, int s) : model(m), last_bad_sdk(s) {}
base::StringPiece model;
int last_bad_sdk;
bool operator==(const BlacklistEntry& other) const {
// Search on name only. Ignore |last_bad_sdk|.
return model == other.model;
}
};
static const BlacklistEntry blacklist[] = {
// crbug.com/653905
{"LGMS330", SDK_VERSION_LOLLIPOP_MR1},
// crbug.com/615872
{"GT-I9100", SDK_VERSION_KITKAT},
{"GT-I9300", SDK_VERSION_KITKAT},
{"GT-N7000", SDK_VERSION_KITKAT},
{"GT-N7100", SDK_VERSION_KITKAT},
// crbug.com/628509
{"A6600", SDK_VERSION_KITKAT},
{"A6800", SDK_VERSION_KITKAT},
// crbug.com/634920
{"GT-S7262", SDK_VERSION_KITKAT},
{"GT-S5282", SDK_VERSION_KITKAT},
{"GT-I8552", SDK_VERSION_KITKAT},
// crbug.com/365494, crbug.com/615872
{"GT-P3113", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-P5110", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-P5100", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-P5113", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-P3110", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-N5110", SDK_VERSION_JELLY_BEAN_MR2},
{"e-tab4", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-I8200Q", SDK_VERSION_JELLY_BEAN_MR2},
// crbug.com/693216
{"GT-I8552B", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-I8262", SDK_VERSION_JELLY_BEAN_MR2},
{"GT-I8262B", SDK_VERSION_JELLY_BEAN_MR2},
};
const BlacklistEntry* iter = std::find(
std::begin(blacklist), std::end(blacklist), BlacklistEntry(model, 0));
return iter == std::end(blacklist) || sdk > iter->last_bad_sdk;
}
// static
bool MediaCodecUtil::SupportsSetParameters() {
// MediaCodec.setParameters() is only available starting with KitKat.
return base::android::BuildInfo::GetInstance()->sdk_int() >=
SDK_VERSION_KITKAT;
}
// static
bool MediaCodecUtil::PlatformSupportsCbcsEncryption(int sdk) {
JNIEnv* env = AttachCurrentThread();
return Java_MediaCodecUtil_platformSupportsCbcsEncryption(env, sdk);
}
// static
std::set<int> MediaCodecUtil::GetEncoderColorFormats(
const std::string& mime_type) {
std::set<int> color_formats;
if (!IsMediaCodecAvailable())
return color_formats;
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jstring> j_mime = ConvertUTF8ToJavaString(env, mime_type);
ScopedJavaLocalRef<jintArray> j_color_format_array =
Java_MediaCodecUtil_getEncoderColorFormatsForMime(env, j_mime);
if (!j_color_format_array.is_null()) {
std::vector<int> formats;
JavaIntArrayToIntVector(env, j_color_format_array, &formats);
color_formats = std::set<int>(formats.begin(), formats.end());
}
return color_formats;
}
// static
bool MediaCodecUtil::CanDecode(VideoCodec codec, bool is_secure) {
return CanDecodeInternal(CodecToAndroidMimeType(codec), is_secure);
}
// static
bool MediaCodecUtil::CanDecode(AudioCodec codec) {
return CanDecodeInternal(CodecToAndroidMimeType(codec), false);
}
// static
bool MediaCodecUtil::CanDecodeInternal(const std::string& mime,
bool is_secure) {
if (!IsMediaCodecAvailable())
return false;
if (mime.empty())
return false;
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jstring> j_mime = ConvertUTF8ToJavaString(env, mime);
return Java_MediaCodecUtil_canDecode(env, j_mime, is_secure);
}
// static
bool MediaCodecUtil::AddSupportedCodecProfileLevels(
std::vector<CodecProfileLevel>* result) {
DCHECK(result);
if (!IsMediaCodecAvailable())
return false;
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jobjectArray> j_codec_profile_levels(
Java_MediaCodecUtil_getSupportedCodecProfileLevels(env));
int java_array_length = env->GetArrayLength(j_codec_profile_levels.obj());
for (int i = 0; i < java_array_length; ++i) {
ScopedJavaLocalRef<jobject> java_codec_profile_level(
env, env->GetObjectArrayElement(j_codec_profile_levels.obj(), i));
result->push_back(MediaCodecProfileLevelToChromiumProfileLevel(
env, java_codec_profile_level));
}
return true;
}
// static
bool MediaCodecUtil::IsKnownUnaccelerated(VideoCodec codec,
MediaCodecDirection direction) {
if (!IsMediaCodecAvailable())
return true;
std::string codec_name =
GetDefaultCodecName(CodecToAndroidMimeType(codec), direction, false);
DVLOG(1) << __func__ << "Default codec for " << GetCodecName(codec) << " : "
<< codec_name << ", direction: " << static_cast<int>(direction);
if (codec_name.empty())
return true;
// MediaTek hardware vp8 is known slower than the software implementation.
// MediaTek hardware vp9 is known crashy, see http://crbug.com/446974 and
// http://crbug.com/597836.
if (base::StartsWith(codec_name, "OMX.MTK.", base::CompareCase::SENSITIVE)) {
if (codec == kCodecVP8)
return true;
if (codec == kCodecVP9)
return base::android::BuildInfo::GetInstance()->sdk_int() <
SDK_VERSION_LOLLIPOP;
return false;
}
// It would be nice if MediaCodecInfo externalized some notion of
// HW-acceleration but it doesn't. Android Media guidance is that the
// "OMX.google" prefix is always used for SW decoders, so that's what we
// use. "OMX.SEC.*" codec is Samsung software implementation - report it
// as unaccelerated as well.
return base::StartsWith(codec_name, "OMX.google.",
base::CompareCase::SENSITIVE) ||
base::StartsWith(codec_name, "OMX.SEC.", base::CompareCase::SENSITIVE);
}
// static
bool MediaCodecUtil::IsHLSPath(const GURL& url) {
return (url.SchemeIsHTTPOrHTTPS() || url.SchemeIsFile()) &&
base::EndsWith(url.path(), ".m3u8",
base::CompareCase::INSENSITIVE_ASCII);
}
// static
bool MediaCodecUtil::IsHLSURL(const GURL& url) {
return (url.SchemeIsHTTPOrHTTPS() || url.SchemeIsFile()) &&
url.spec().find("m3u8") != std::string::npos;
}
// static
bool MediaCodecUtil::IsVp8DecoderAvailable() {
return IsMediaCodecAvailable() && IsDecoderSupportedByDevice(kVp8MimeType);
}
// static
bool MediaCodecUtil::IsVp8EncoderAvailable() {
// Currently the vp8 encoder and decoder blacklists cover the same devices,
// but we have a second method for clarity in future issues.
return IsVp8DecoderAvailable();
}
// static
bool MediaCodecUtil::IsVp9DecoderAvailable() {
return IsMediaCodecAvailable() && IsDecoderSupportedByDevice(kVp9MimeType);
}
// static
bool MediaCodecUtil::IsH264EncoderAvailable() {
return IsMediaCodecAvailable() && IsEncoderSupportedByDevice(kAvcMimeType);
}
// static
bool MediaCodecUtil::IsSurfaceViewOutputSupported() {
// Disable SurfaceView output for the Samsung Galaxy S3; it does not work
// well enough for even 360p24 H264 playback. http://crbug.com/602870.
//
// Notably this is codec agnostic at present, so any devices added to
// the blacklist will avoid trying to play any codecs on SurfaceView. If
// needed in the future this can be expanded to be codec specific.
const char* model_prefixes[] = {// Exynos 4 (Mali-400)
"GT-I9300", "GT-I9305", "SHV-E210",
// Snapdragon S4 (Adreno-225)
"SCH-I535", "SCH-J201", "SCH-R530",
"SCH-I960", "SCH-S968", "SGH-T999",
"SGH-I747", "SGH-N064", 0};
std::string model(base::android::BuildInfo::GetInstance()->model());
for (int i = 0; model_prefixes[i]; ++i) {
if (base::StartsWith(model, model_prefixes[i],
base::CompareCase::INSENSITIVE_ASCII)) {
return false;
}
}
return true;
}
// static
bool MediaCodecUtil::IsSetOutputSurfaceSupported() {
JNIEnv* env = AttachCurrentThread();
return Java_MediaCodecUtil_isSetOutputSurfaceSupported(env);
}
// static
bool MediaCodecUtil::IsPassthroughAudioFormat(AudioCodec codec) {
return codec == kCodecAC3 || codec == kCodecEAC3;
}
// static
bool MediaCodecUtil::CodecNeedsFlushWorkaround(MediaCodecBridge* codec) {
int sdk_int = base::android::BuildInfo::GetInstance()->sdk_int();
std::string codec_name = codec->GetName();
return sdk_int < SDK_VERSION_JELLY_BEAN_MR2 ||
(sdk_int == SDK_VERSION_JELLY_BEAN_MR2 &&
("OMX.SEC.avc.dec" == codec_name ||
"OMX.SEC.avc.dec.secure" == codec_name)) ||
(sdk_int == SDK_VERSION_KITKAT &&
base::StartsWith(base::android::BuildInfo::GetInstance()->model(),
"SM-G800", base::CompareCase::INSENSITIVE_ASCII) &&
("OMX.Exynos.avc.dec" == codec_name ||
"OMX.Exynos.avc.dec.secure" == codec_name));
}
} // namespace media