| Index: webrtc/api/android/jni/androidmediadecoder_jni.cc | 
| diff --git a/webrtc/api/android/jni/androidmediadecoder_jni.cc b/webrtc/api/android/jni/androidmediadecoder_jni.cc | 
| deleted file mode 100644 | 
| index 2a7e68976aff881386d4ab363cebf30311080782..0000000000000000000000000000000000000000 | 
| --- a/webrtc/api/android/jni/androidmediadecoder_jni.cc | 
| +++ /dev/null | 
| @@ -1,995 +0,0 @@ | 
| -/* | 
| - *  Copyright 2015 The WebRTC project authors. All Rights Reserved. | 
| - * | 
| - *  Use of this source code is governed by a BSD-style license | 
| - *  that can be found in the LICENSE file in the root of the source | 
| - *  tree. An additional intellectual property rights grant can be found | 
| - *  in the file PATENTS.  All contributing project authors may | 
| - *  be found in the AUTHORS file in the root of the source tree. | 
| - */ | 
| - | 
| -#include <algorithm> | 
| -#include <memory> | 
| -#include <vector> | 
| - | 
| -// NOTICE: androidmediadecoder_jni.h must be included before | 
| -// androidmediacodeccommon.h to avoid build errors. | 
| -#include "webrtc/api/android/jni/androidmediadecoder_jni.h" | 
| - | 
| -#include "third_party/libyuv/include/libyuv/convert.h" | 
| -#include "third_party/libyuv/include/libyuv/convert_from.h" | 
| -#include "third_party/libyuv/include/libyuv/video_common.h" | 
| -#include "webrtc/api/android/jni/androidmediacodeccommon.h" | 
| -#include "webrtc/api/android/jni/classreferenceholder.h" | 
| -#include "webrtc/api/android/jni/native_handle_impl.h" | 
| -#include "webrtc/api/android/jni/surfacetexturehelper_jni.h" | 
| -#include "webrtc/base/bind.h" | 
| -#include "webrtc/base/checks.h" | 
| -#include "webrtc/base/logging.h" | 
| -#include "webrtc/base/scoped_ref_ptr.h" | 
| -#include "webrtc/base/thread.h" | 
| -#include "webrtc/base/timeutils.h" | 
| -#include "webrtc/common_video/include/i420_buffer_pool.h" | 
| -#include "webrtc/modules/video_coding/include/video_codec_interface.h" | 
| -#include "webrtc/system_wrappers/include/logcat_trace_context.h" | 
| - | 
| -using rtc::Bind; | 
| -using rtc::Thread; | 
| -using rtc::ThreadManager; | 
| - | 
| -using webrtc::CodecSpecificInfo; | 
| -using webrtc::DecodedImageCallback; | 
| -using webrtc::EncodedImage; | 
| -using webrtc::VideoFrame; | 
| -using webrtc::RTPFragmentationHeader; | 
| -using webrtc::VideoCodec; | 
| -using webrtc::VideoCodecType; | 
| -using webrtc::kVideoCodecH264; | 
| -using webrtc::kVideoCodecVP8; | 
| -using webrtc::kVideoCodecVP9; | 
| - | 
| -namespace webrtc_jni { | 
| - | 
| -// Logging macros. | 
| -#define TAG_DECODER "MediaCodecVideoDecoder" | 
| -#ifdef TRACK_BUFFER_TIMING | 
| -#define ALOGV(...) | 
| -  __android_log_print(ANDROID_LOG_VERBOSE, TAG_DECODER, __VA_ARGS__) | 
| -#else | 
| -#define ALOGV(...) | 
| -#endif | 
| -#define ALOGD LOG_TAG(rtc::LS_INFO, TAG_DECODER) | 
| -#define ALOGW LOG_TAG(rtc::LS_WARNING, TAG_DECODER) | 
| -#define ALOGE LOG_TAG(rtc::LS_ERROR, TAG_DECODER) | 
| - | 
| -enum { kMaxWarningLogFrames = 2 }; | 
| - | 
| -class MediaCodecVideoDecoder : public webrtc::VideoDecoder, | 
| -                               public rtc::MessageHandler { | 
| - public: | 
| -  explicit MediaCodecVideoDecoder( | 
| -      JNIEnv* jni, VideoCodecType codecType, jobject render_egl_context); | 
| -  virtual ~MediaCodecVideoDecoder(); | 
| - | 
| -  int32_t InitDecode(const VideoCodec* codecSettings, int32_t numberOfCores) | 
| -      override; | 
| - | 
| -  int32_t Decode( | 
| -      const EncodedImage& inputImage, bool missingFrames, | 
| -      const RTPFragmentationHeader* fragmentation, | 
| -      const CodecSpecificInfo* codecSpecificInfo = NULL, | 
| -      int64_t renderTimeMs = -1) override; | 
| - | 
| -  int32_t RegisterDecodeCompleteCallback(DecodedImageCallback* callback) | 
| -      override; | 
| - | 
| -  int32_t Release() override; | 
| - | 
| -  bool PrefersLateDecoding() const override { return true; } | 
| - | 
| -  // rtc::MessageHandler implementation. | 
| -  void OnMessage(rtc::Message* msg) override; | 
| - | 
| -  const char* ImplementationName() const override; | 
| - | 
| - private: | 
| -  // CHECK-fail if not running on |codec_thread_|. | 
| -  void CheckOnCodecThread(); | 
| - | 
| -  int32_t InitDecodeOnCodecThread(); | 
| -  int32_t ResetDecodeOnCodecThread(); | 
| -  int32_t ReleaseOnCodecThread(); | 
| -  int32_t DecodeOnCodecThread(const EncodedImage& inputImage); | 
| -  // Deliver any outputs pending in the MediaCodec to our |callback_| and return | 
| -  // true on success. | 
| -  bool DeliverPendingOutputs(JNIEnv* jni, int dequeue_timeout_us); | 
| -  int32_t ProcessHWErrorOnCodecThread(); | 
| -  void EnableFrameLogOnWarning(); | 
| -  void ResetVariables(); | 
| - | 
| -  // Type of video codec. | 
| -  VideoCodecType codecType_; | 
| - | 
| -  // Render EGL context - owned by factory, should not be allocated/destroyed | 
| -  // by VideoDecoder. | 
| -  jobject render_egl_context_; | 
| - | 
| -  bool key_frame_required_; | 
| -  bool inited_; | 
| -  bool sw_fallback_required_; | 
| -  bool use_surface_; | 
| -  VideoCodec codec_; | 
| -  webrtc::I420BufferPool decoded_frame_pool_; | 
| -  rtc::scoped_refptr<SurfaceTextureHelper> surface_texture_helper_; | 
| -  DecodedImageCallback* callback_; | 
| -  int frames_received_;  // Number of frames received by decoder. | 
| -  int frames_decoded_;  // Number of frames decoded by decoder. | 
| -  // Number of decoded frames for which log information is displayed. | 
| -  int frames_decoded_logged_; | 
| -  int64_t start_time_ms_;  // Start time for statistics. | 
| -  int current_frames_;  // Number of frames in the current statistics interval. | 
| -  int current_bytes_;  // Encoded bytes in the current statistics interval. | 
| -  int current_decoding_time_ms_;  // Overall decoding time in the current second | 
| -  int current_delay_time_ms_;  // Overall delay time in the current second. | 
| -  uint32_t max_pending_frames_;  // Maximum number of pending input frames. | 
| - | 
| -  // State that is constant for the lifetime of this object once the ctor | 
| -  // returns. | 
| -  std::unique_ptr<Thread> | 
| -      codec_thread_;  // Thread on which to operate MediaCodec. | 
| -  ScopedGlobalRef<jclass> j_media_codec_video_decoder_class_; | 
| -  ScopedGlobalRef<jobject> j_media_codec_video_decoder_; | 
| -  jmethodID j_init_decode_method_; | 
| -  jmethodID j_reset_method_; | 
| -  jmethodID j_release_method_; | 
| -  jmethodID j_dequeue_input_buffer_method_; | 
| -  jmethodID j_queue_input_buffer_method_; | 
| -  jmethodID j_dequeue_byte_buffer_method_; | 
| -  jmethodID j_dequeue_texture_buffer_method_; | 
| -  jmethodID j_return_decoded_byte_buffer_method_; | 
| -  // MediaCodecVideoDecoder fields. | 
| -  jfieldID j_input_buffers_field_; | 
| -  jfieldID j_output_buffers_field_; | 
| -  jfieldID j_color_format_field_; | 
| -  jfieldID j_width_field_; | 
| -  jfieldID j_height_field_; | 
| -  jfieldID j_stride_field_; | 
| -  jfieldID j_slice_height_field_; | 
| -  // MediaCodecVideoDecoder.DecodedTextureBuffer fields. | 
| -  jfieldID j_texture_id_field_; | 
| -  jfieldID j_transform_matrix_field_; | 
| -  jfieldID j_texture_presentation_timestamp_ms_field_; | 
| -  jfieldID j_texture_timestamp_ms_field_; | 
| -  jfieldID j_texture_ntp_timestamp_ms_field_; | 
| -  jfieldID j_texture_decode_time_ms_field_; | 
| -  jfieldID j_texture_frame_delay_ms_field_; | 
| -  // MediaCodecVideoDecoder.DecodedOutputBuffer fields. | 
| -  jfieldID j_info_index_field_; | 
| -  jfieldID j_info_offset_field_; | 
| -  jfieldID j_info_size_field_; | 
| -  jfieldID j_presentation_timestamp_ms_field_; | 
| -  jfieldID j_timestamp_ms_field_; | 
| -  jfieldID j_ntp_timestamp_ms_field_; | 
| -  jfieldID j_byte_buffer_decode_time_ms_field_; | 
| - | 
| -  // Global references; must be deleted in Release(). | 
| -  std::vector<jobject> input_buffers_; | 
| -}; | 
| - | 
| -MediaCodecVideoDecoder::MediaCodecVideoDecoder( | 
| -    JNIEnv* jni, VideoCodecType codecType, jobject render_egl_context) : | 
| -    codecType_(codecType), | 
| -    render_egl_context_(render_egl_context), | 
| -    key_frame_required_(true), | 
| -    inited_(false), | 
| -    sw_fallback_required_(false), | 
| -    codec_thread_(new Thread()), | 
| -    j_media_codec_video_decoder_class_( | 
| -        jni, | 
| -        FindClass(jni, "org/webrtc/MediaCodecVideoDecoder")), | 
| -          j_media_codec_video_decoder_( | 
| -              jni, | 
| -              jni->NewObject(*j_media_codec_video_decoder_class_, | 
| -                   GetMethodID(jni, | 
| -                              *j_media_codec_video_decoder_class_, | 
| -                              "<init>", | 
| -                              "()V"))) { | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| -  codec_thread_->SetName("MediaCodecVideoDecoder", NULL); | 
| -  RTC_CHECK(codec_thread_->Start()) << "Failed to start MediaCodecVideoDecoder"; | 
| - | 
| -  j_init_decode_method_ = GetMethodID( | 
| -      jni, *j_media_codec_video_decoder_class_, "initDecode", | 
| -      "(Lorg/webrtc/MediaCodecVideoDecoder$VideoCodecType;" | 
| -      "IILorg/webrtc/SurfaceTextureHelper;)Z"); | 
| -  j_reset_method_ = | 
| -      GetMethodID(jni, *j_media_codec_video_decoder_class_, "reset", "(II)V"); | 
| -  j_release_method_ = | 
| -      GetMethodID(jni, *j_media_codec_video_decoder_class_, "release", "()V"); | 
| -  j_dequeue_input_buffer_method_ = GetMethodID( | 
| -      jni, *j_media_codec_video_decoder_class_, "dequeueInputBuffer", "()I"); | 
| -  j_queue_input_buffer_method_ = GetMethodID( | 
| -      jni, *j_media_codec_video_decoder_class_, "queueInputBuffer", "(IIJJJ)Z"); | 
| -  j_dequeue_byte_buffer_method_ = GetMethodID( | 
| -      jni, *j_media_codec_video_decoder_class_, "dequeueOutputBuffer", | 
| -      "(I)Lorg/webrtc/MediaCodecVideoDecoder$DecodedOutputBuffer;"); | 
| -  j_dequeue_texture_buffer_method_ = GetMethodID( | 
| -      jni, *j_media_codec_video_decoder_class_, "dequeueTextureBuffer", | 
| -      "(I)Lorg/webrtc/MediaCodecVideoDecoder$DecodedTextureBuffer;"); | 
| -  j_return_decoded_byte_buffer_method_ = | 
| -      GetMethodID(jni, *j_media_codec_video_decoder_class_, | 
| -                  "returnDecodedOutputBuffer", "(I)V"); | 
| - | 
| -  j_input_buffers_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, | 
| -      "inputBuffers", "[Ljava/nio/ByteBuffer;"); | 
| -  j_output_buffers_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, | 
| -      "outputBuffers", "[Ljava/nio/ByteBuffer;"); | 
| -  j_color_format_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, "colorFormat", "I"); | 
| -  j_width_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, "width", "I"); | 
| -  j_height_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, "height", "I"); | 
| -  j_stride_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, "stride", "I"); | 
| -  j_slice_height_field_ = GetFieldID( | 
| -      jni, *j_media_codec_video_decoder_class_, "sliceHeight", "I"); | 
| - | 
| -  jclass j_decoded_texture_buffer_class = FindClass(jni, | 
| -      "org/webrtc/MediaCodecVideoDecoder$DecodedTextureBuffer"); | 
| -  j_texture_id_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "textureID", "I"); | 
| -  j_transform_matrix_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "transformMatrix", "[F"); | 
| -  j_texture_presentation_timestamp_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "presentationTimeStampMs", "J"); | 
| -  j_texture_timestamp_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "timeStampMs", "J"); | 
| -  j_texture_ntp_timestamp_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "ntpTimeStampMs", "J"); | 
| -  j_texture_decode_time_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "decodeTimeMs", "J"); | 
| -  j_texture_frame_delay_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_texture_buffer_class, "frameDelayMs", "J"); | 
| - | 
| -  jclass j_decoded_output_buffer_class = FindClass(jni, | 
| -      "org/webrtc/MediaCodecVideoDecoder$DecodedOutputBuffer"); | 
| -  j_info_index_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "index", "I"); | 
| -  j_info_offset_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "offset", "I"); | 
| -  j_info_size_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "size", "I"); | 
| -  j_presentation_timestamp_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "presentationTimeStampMs", "J"); | 
| -  j_timestamp_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "timeStampMs", "J"); | 
| -  j_ntp_timestamp_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "ntpTimeStampMs", "J"); | 
| -  j_byte_buffer_decode_time_ms_field_ = GetFieldID( | 
| -      jni, j_decoded_output_buffer_class, "decodeTimeMs", "J"); | 
| - | 
| -  CHECK_EXCEPTION(jni) << "MediaCodecVideoDecoder ctor failed"; | 
| -  use_surface_ = (render_egl_context_ != NULL); | 
| -  ALOGD << "MediaCodecVideoDecoder ctor. Use surface: " << use_surface_; | 
| -  memset(&codec_, 0, sizeof(codec_)); | 
| -  AllowBlockingCalls(); | 
| -} | 
| - | 
| -MediaCodecVideoDecoder::~MediaCodecVideoDecoder() { | 
| -  // Call Release() to ensure no more callbacks to us after we are deleted. | 
| -  Release(); | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::InitDecode(const VideoCodec* inst, | 
| -    int32_t numberOfCores) { | 
| -  ALOGD << "InitDecode."; | 
| -  if (inst == NULL) { | 
| -    ALOGE << "NULL VideoCodec instance"; | 
| -    return WEBRTC_VIDEO_CODEC_ERR_PARAMETER; | 
| -  } | 
| -  // Factory should guard against other codecs being used with us. | 
| -  RTC_CHECK(inst->codecType == codecType_) | 
| -      << "Unsupported codec " << inst->codecType << " for " << codecType_; | 
| - | 
| -  if (sw_fallback_required_) { | 
| -    ALOGE << "InitDecode() - fallback to SW decoder"; | 
| -    return WEBRTC_VIDEO_CODEC_OK; | 
| -  } | 
| -  // Save VideoCodec instance for later. | 
| -  if (&codec_ != inst) { | 
| -    codec_ = *inst; | 
| -  } | 
| -  // If maxFramerate is not set then assume 30 fps. | 
| -  codec_.maxFramerate = (codec_.maxFramerate >= 1) ? codec_.maxFramerate : 30; | 
| - | 
| -  // Call Java init. | 
| -  return codec_thread_->Invoke<int32_t>( | 
| -      RTC_FROM_HERE, | 
| -      Bind(&MediaCodecVideoDecoder::InitDecodeOnCodecThread, this)); | 
| -} | 
| - | 
| -void MediaCodecVideoDecoder::ResetVariables() { | 
| -  CheckOnCodecThread(); | 
| - | 
| -  key_frame_required_ = true; | 
| -  frames_received_ = 0; | 
| -  frames_decoded_ = 0; | 
| -  frames_decoded_logged_ = kMaxDecodedLogFrames; | 
| -  start_time_ms_ = rtc::TimeMillis(); | 
| -  current_frames_ = 0; | 
| -  current_bytes_ = 0; | 
| -  current_decoding_time_ms_ = 0; | 
| -  current_delay_time_ms_ = 0; | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::InitDecodeOnCodecThread() { | 
| -  CheckOnCodecThread(); | 
| -  JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| -  ALOGD << "InitDecodeOnCodecThread Type: " << (int)codecType_ << ". " | 
| -      << codec_.width << " x " << codec_.height << ". Fps: " << | 
| -      (int)codec_.maxFramerate; | 
| - | 
| -  // Release previous codec first if it was allocated before. | 
| -  int ret_val = ReleaseOnCodecThread(); | 
| -  if (ret_val < 0) { | 
| -    ALOGE << "Release failure: " << ret_val << " - fallback to SW codec"; | 
| -    sw_fallback_required_ = true; | 
| -    return WEBRTC_VIDEO_CODEC_ERROR; | 
| -  } | 
| - | 
| -  ResetVariables(); | 
| - | 
| -  if (use_surface_) { | 
| -    surface_texture_helper_ = SurfaceTextureHelper::create( | 
| -        jni, "Decoder SurfaceTextureHelper", render_egl_context_); | 
| -    if (!surface_texture_helper_) { | 
| -      ALOGE << "Couldn't create SurfaceTextureHelper - fallback to SW codec"; | 
| -      sw_fallback_required_ = true; | 
| -      return WEBRTC_VIDEO_CODEC_ERROR; | 
| -    } | 
| -  } | 
| - | 
| -  jobject j_video_codec_enum = JavaEnumFromIndexAndClassName( | 
| -      jni, "MediaCodecVideoDecoder$VideoCodecType", codecType_); | 
| -  bool success = jni->CallBooleanMethod( | 
| -      *j_media_codec_video_decoder_, | 
| -      j_init_decode_method_, | 
| -      j_video_codec_enum, | 
| -      codec_.width, | 
| -      codec_.height, | 
| -      use_surface_ ? surface_texture_helper_->GetJavaSurfaceTextureHelper() | 
| -                   : nullptr); | 
| - | 
| -  if (CheckException(jni) || !success) { | 
| -    ALOGE << "Codec initialization error - fallback to SW codec."; | 
| -    sw_fallback_required_ = true; | 
| -    return WEBRTC_VIDEO_CODEC_ERROR; | 
| -  } | 
| -  inited_ = true; | 
| - | 
| -  switch (codecType_) { | 
| -    case kVideoCodecVP8: | 
| -      max_pending_frames_ = kMaxPendingFramesVp8; | 
| -      break; | 
| -    case kVideoCodecVP9: | 
| -      max_pending_frames_ = kMaxPendingFramesVp9; | 
| -      break; | 
| -    case kVideoCodecH264: | 
| -      max_pending_frames_ = kMaxPendingFramesH264; | 
| -      break; | 
| -    default: | 
| -      max_pending_frames_ = 0; | 
| -  } | 
| -  ALOGD << "Maximum amount of pending frames: " << max_pending_frames_; | 
| - | 
| -  jobjectArray input_buffers = (jobjectArray)GetObjectField( | 
| -      jni, *j_media_codec_video_decoder_, j_input_buffers_field_); | 
| -  size_t num_input_buffers = jni->GetArrayLength(input_buffers); | 
| -  input_buffers_.resize(num_input_buffers); | 
| -  for (size_t i = 0; i < num_input_buffers; ++i) { | 
| -    input_buffers_[i] = | 
| -        jni->NewGlobalRef(jni->GetObjectArrayElement(input_buffers, i)); | 
| -    if (CheckException(jni)) { | 
| -      ALOGE << "NewGlobalRef error - fallback to SW codec."; | 
| -      sw_fallback_required_ = true; | 
| -      return WEBRTC_VIDEO_CODEC_ERROR; | 
| -    } | 
| -  } | 
| - | 
| -  codec_thread_->PostDelayed(RTC_FROM_HERE, kMediaCodecPollMs, this); | 
| - | 
| -  return WEBRTC_VIDEO_CODEC_OK; | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::ResetDecodeOnCodecThread() { | 
| -  CheckOnCodecThread(); | 
| -  JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| -  ALOGD << "ResetDecodeOnCodecThread Type: " << (int)codecType_ << ". " | 
| -      << codec_.width << " x " << codec_.height; | 
| -  ALOGD << "  Frames received: " << frames_received_ << | 
| -      ". Frames decoded: " << frames_decoded_; | 
| - | 
| -  inited_ = false; | 
| -  rtc::MessageQueueManager::Clear(this); | 
| -  ResetVariables(); | 
| - | 
| -  jni->CallVoidMethod( | 
| -      *j_media_codec_video_decoder_, | 
| -      j_reset_method_, | 
| -      codec_.width, | 
| -      codec_.height); | 
| - | 
| -  if (CheckException(jni)) { | 
| -    ALOGE << "Soft reset error - fallback to SW codec."; | 
| -    sw_fallback_required_ = true; | 
| -    return WEBRTC_VIDEO_CODEC_ERROR; | 
| -  } | 
| -  inited_ = true; | 
| - | 
| -  codec_thread_->PostDelayed(RTC_FROM_HERE, kMediaCodecPollMs, this); | 
| - | 
| -  return WEBRTC_VIDEO_CODEC_OK; | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::Release() { | 
| -  ALOGD << "DecoderRelease request"; | 
| -  return codec_thread_->Invoke<int32_t>( | 
| -      RTC_FROM_HERE, Bind(&MediaCodecVideoDecoder::ReleaseOnCodecThread, this)); | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::ReleaseOnCodecThread() { | 
| -  if (!inited_) { | 
| -    return WEBRTC_VIDEO_CODEC_OK; | 
| -  } | 
| -  CheckOnCodecThread(); | 
| -  JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -  ALOGD << "DecoderReleaseOnCodecThread: Frames received: " << | 
| -      frames_received_ << ". Frames decoded: " << frames_decoded_; | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| -  for (size_t i = 0; i < input_buffers_.size(); i++) { | 
| -    jni->DeleteGlobalRef(input_buffers_[i]); | 
| -  } | 
| -  input_buffers_.clear(); | 
| -  jni->CallVoidMethod(*j_media_codec_video_decoder_, j_release_method_); | 
| -  surface_texture_helper_ = nullptr; | 
| -  inited_ = false; | 
| -  rtc::MessageQueueManager::Clear(this); | 
| -  if (CheckException(jni)) { | 
| -    ALOGE << "Decoder release exception"; | 
| -    return WEBRTC_VIDEO_CODEC_ERROR; | 
| -  } | 
| -  ALOGD << "DecoderReleaseOnCodecThread done"; | 
| -  return WEBRTC_VIDEO_CODEC_OK; | 
| -} | 
| - | 
| -void MediaCodecVideoDecoder::CheckOnCodecThread() { | 
| -  RTC_CHECK(codec_thread_.get() == ThreadManager::Instance()->CurrentThread()) | 
| -      << "Running on wrong thread!"; | 
| -} | 
| - | 
| -void MediaCodecVideoDecoder::EnableFrameLogOnWarning() { | 
| -  // Log next 2 output frames. | 
| -  frames_decoded_logged_ = std::max( | 
| -      frames_decoded_logged_, frames_decoded_ + kMaxWarningLogFrames); | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::ProcessHWErrorOnCodecThread() { | 
| -  CheckOnCodecThread(); | 
| -  int ret_val = ReleaseOnCodecThread(); | 
| -  if (ret_val < 0) { | 
| -    ALOGE << "ProcessHWError: Release failure"; | 
| -  } | 
| -  if (codecType_ == kVideoCodecH264) { | 
| -    // For now there is no SW H.264 which can be used as fallback codec. | 
| -    // So try to restart hw codec for now. | 
| -    ret_val = InitDecodeOnCodecThread(); | 
| -    ALOGE << "Reset H.264 codec done. Status: " << ret_val; | 
| -    if (ret_val == WEBRTC_VIDEO_CODEC_OK) { | 
| -      // H.264 codec was succesfully reset - return regular error code. | 
| -      return WEBRTC_VIDEO_CODEC_ERROR; | 
| -    } else { | 
| -      // Fail to restart H.264 codec - return error code which should stop the | 
| -      // call. | 
| -      return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE; | 
| -    } | 
| -  } else { | 
| -    sw_fallback_required_ = true; | 
| -    ALOGE << "Return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE"; | 
| -    return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE; | 
| -  } | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::Decode( | 
| -    const EncodedImage& inputImage, | 
| -    bool missingFrames, | 
| -    const RTPFragmentationHeader* fragmentation, | 
| -    const CodecSpecificInfo* codecSpecificInfo, | 
| -    int64_t renderTimeMs) { | 
| -  if (sw_fallback_required_) { | 
| -    ALOGE << "Decode() - fallback to SW codec"; | 
| -    return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE; | 
| -  } | 
| -  if (callback_ == NULL) { | 
| -    ALOGE << "Decode() - callback_ is NULL"; | 
| -    return WEBRTC_VIDEO_CODEC_UNINITIALIZED; | 
| -  } | 
| -  if (inputImage._buffer == NULL && inputImage._length > 0) { | 
| -    ALOGE << "Decode() - inputImage is incorrect"; | 
| -    return WEBRTC_VIDEO_CODEC_ERR_PARAMETER; | 
| -  } | 
| -  if (!inited_) { | 
| -    ALOGE << "Decode() - decoder is not initialized"; | 
| -    return WEBRTC_VIDEO_CODEC_UNINITIALIZED; | 
| -  } | 
| - | 
| -  // Check if encoded frame dimension has changed. | 
| -  if ((inputImage._encodedWidth * inputImage._encodedHeight > 0) && | 
| -      (inputImage._encodedWidth != codec_.width || | 
| -      inputImage._encodedHeight != codec_.height)) { | 
| -    ALOGW << "Input resolution changed from " << | 
| -        codec_.width << " x " << codec_.height << " to " << | 
| -        inputImage._encodedWidth << " x " << inputImage._encodedHeight; | 
| -    codec_.width = inputImage._encodedWidth; | 
| -    codec_.height = inputImage._encodedHeight; | 
| -    int32_t ret; | 
| -    if (use_surface_ && | 
| -        (codecType_ == kVideoCodecVP8 || codecType_ == kVideoCodecH264)) { | 
| -      // Soft codec reset - only for surface decoding. | 
| -      ret = codec_thread_->Invoke<int32_t>( | 
| -          RTC_FROM_HERE, | 
| -          Bind(&MediaCodecVideoDecoder::ResetDecodeOnCodecThread, this)); | 
| -    } else { | 
| -      // Hard codec reset. | 
| -      ret = InitDecode(&codec_, 1); | 
| -    } | 
| -    if (ret < 0) { | 
| -      ALOGE << "InitDecode failure: " << ret << " - fallback to SW codec"; | 
| -      sw_fallback_required_ = true; | 
| -      return WEBRTC_VIDEO_CODEC_FALLBACK_SOFTWARE; | 
| -    } | 
| -  } | 
| - | 
| -  // Always start with a complete key frame. | 
| -  if (key_frame_required_) { | 
| -    if (inputImage._frameType != webrtc::kVideoFrameKey) { | 
| -      ALOGE << "Decode() - key frame is required"; | 
| -      return WEBRTC_VIDEO_CODEC_ERROR; | 
| -    } | 
| -    if (!inputImage._completeFrame) { | 
| -      ALOGE << "Decode() - complete frame is required"; | 
| -      return WEBRTC_VIDEO_CODEC_ERROR; | 
| -    } | 
| -    key_frame_required_ = false; | 
| -  } | 
| -  if (inputImage._length == 0) { | 
| -    return WEBRTC_VIDEO_CODEC_ERROR; | 
| -  } | 
| - | 
| -  return codec_thread_->Invoke<int32_t>( | 
| -      RTC_FROM_HERE, | 
| -      Bind(&MediaCodecVideoDecoder::DecodeOnCodecThread, this, inputImage)); | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::DecodeOnCodecThread( | 
| -    const EncodedImage& inputImage) { | 
| -  CheckOnCodecThread(); | 
| -  JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| - | 
| -  // Try to drain the decoder and wait until output is not too | 
| -  // much behind the input. | 
| -  if (codecType_ == kVideoCodecH264 && | 
| -      frames_received_ > frames_decoded_ + max_pending_frames_) { | 
| -    // Print warning for H.264 only - for VP8/VP9 one frame delay is ok. | 
| -    ALOGW << "Decoder is too far behind. Try to drain. Received: " << | 
| -        frames_received_ << ". Decoded: " << frames_decoded_; | 
| -    EnableFrameLogOnWarning(); | 
| -  } | 
| -  const int64 drain_start = rtc::TimeMillis(); | 
| -  while ((frames_received_ > frames_decoded_ + max_pending_frames_) && | 
| -         (rtc::TimeMillis() - drain_start) < kMediaCodecTimeoutMs) { | 
| -    if (!DeliverPendingOutputs(jni, kMediaCodecPollMs)) { | 
| -      ALOGE << "DeliverPendingOutputs error. Frames received: " << | 
| -          frames_received_ << ". Frames decoded: " << frames_decoded_; | 
| -      return ProcessHWErrorOnCodecThread(); | 
| -    } | 
| -  } | 
| -  if (frames_received_ > frames_decoded_ + max_pending_frames_) { | 
| -    ALOGE << "Output buffer dequeue timeout. Frames received: " << | 
| -        frames_received_ << ". Frames decoded: " << frames_decoded_; | 
| -    return ProcessHWErrorOnCodecThread(); | 
| -  } | 
| - | 
| -  // Get input buffer. | 
| -  int j_input_buffer_index = jni->CallIntMethod( | 
| -      *j_media_codec_video_decoder_, j_dequeue_input_buffer_method_); | 
| -  if (CheckException(jni) || j_input_buffer_index < 0) { | 
| -    ALOGE << "dequeueInputBuffer error: " << j_input_buffer_index << | 
| -        ". Retry DeliverPendingOutputs."; | 
| -    EnableFrameLogOnWarning(); | 
| -    // Try to drain the decoder. | 
| -    if (!DeliverPendingOutputs(jni, kMediaCodecPollMs)) { | 
| -      ALOGE << "DeliverPendingOutputs error. Frames received: " << | 
| -          frames_received_ << ". Frames decoded: " << frames_decoded_; | 
| -      return ProcessHWErrorOnCodecThread(); | 
| -    } | 
| -    // Try dequeue input buffer one last time. | 
| -    j_input_buffer_index = jni->CallIntMethod( | 
| -        *j_media_codec_video_decoder_, j_dequeue_input_buffer_method_); | 
| -    if (CheckException(jni) || j_input_buffer_index < 0) { | 
| -      ALOGE << "dequeueInputBuffer critical error: " << j_input_buffer_index; | 
| -      return ProcessHWErrorOnCodecThread(); | 
| -    } | 
| -  } | 
| - | 
| -  // Copy encoded data to Java ByteBuffer. | 
| -  jobject j_input_buffer = input_buffers_[j_input_buffer_index]; | 
| -  uint8_t* buffer = | 
| -      reinterpret_cast<uint8_t*>(jni->GetDirectBufferAddress(j_input_buffer)); | 
| -  RTC_CHECK(buffer) << "Indirect buffer??"; | 
| -  int64_t buffer_capacity = jni->GetDirectBufferCapacity(j_input_buffer); | 
| -  if (CheckException(jni) || buffer_capacity < inputImage._length) { | 
| -    ALOGE << "Input frame size "<<  inputImage._length << | 
| -        " is bigger than buffer size " << buffer_capacity; | 
| -    return ProcessHWErrorOnCodecThread(); | 
| -  } | 
| -  jlong presentation_timestamp_us = static_cast<jlong>( | 
| -      static_cast<int64_t>(frames_received_) * 1000000 / codec_.maxFramerate); | 
| -  memcpy(buffer, inputImage._buffer, inputImage._length); | 
| - | 
| -  if (frames_decoded_ < frames_decoded_logged_) { | 
| -    ALOGD << "Decoder frame in # " << frames_received_ << | 
| -        ". Type: " << inputImage._frameType << | 
| -        ". Buffer # " << j_input_buffer_index << | 
| -        ". TS: " << presentation_timestamp_us / 1000 << | 
| -        ". Size: " << inputImage._length; | 
| -  } | 
| - | 
| -  // Save input image timestamps for later output. | 
| -  frames_received_++; | 
| -  current_bytes_ += inputImage._length; | 
| - | 
| -  // Feed input to decoder. | 
| -  bool success = jni->CallBooleanMethod( | 
| -      *j_media_codec_video_decoder_, | 
| -      j_queue_input_buffer_method_, | 
| -      j_input_buffer_index, | 
| -      inputImage._length, | 
| -      presentation_timestamp_us, | 
| -      static_cast<int64_t> (inputImage._timeStamp), | 
| -      inputImage.ntp_time_ms_); | 
| -  if (CheckException(jni) || !success) { | 
| -    ALOGE << "queueInputBuffer error"; | 
| -    return ProcessHWErrorOnCodecThread(); | 
| -  } | 
| - | 
| -  // Try to drain the decoder | 
| -  if (!DeliverPendingOutputs(jni, 0)) { | 
| -    ALOGE << "DeliverPendingOutputs error"; | 
| -    return ProcessHWErrorOnCodecThread(); | 
| -  } | 
| - | 
| -  return WEBRTC_VIDEO_CODEC_OK; | 
| -} | 
| - | 
| -bool MediaCodecVideoDecoder::DeliverPendingOutputs( | 
| -    JNIEnv* jni, int dequeue_timeout_ms) { | 
| -  if (frames_received_ <= frames_decoded_) { | 
| -    // No need to query for output buffers - decoder is drained. | 
| -    return true; | 
| -  } | 
| -  // Get decoder output. | 
| -  jobject j_decoder_output_buffer = | 
| -      jni->CallObjectMethod(*j_media_codec_video_decoder_, | 
| -          use_surface_ ? j_dequeue_texture_buffer_method_ | 
| -                       : j_dequeue_byte_buffer_method_, | 
| -          dequeue_timeout_ms); | 
| - | 
| -  if (CheckException(jni)) { | 
| -    ALOGE << "dequeueOutputBuffer() error"; | 
| -    return false; | 
| -  } | 
| -  if (IsNull(jni, j_decoder_output_buffer)) { | 
| -    // No decoded frame ready. | 
| -    return true; | 
| -  } | 
| - | 
| -  // Get decoded video frame properties. | 
| -  int color_format = GetIntField(jni, *j_media_codec_video_decoder_, | 
| -      j_color_format_field_); | 
| -  int width = GetIntField(jni, *j_media_codec_video_decoder_, j_width_field_); | 
| -  int height = GetIntField(jni, *j_media_codec_video_decoder_, j_height_field_); | 
| -  int stride = GetIntField(jni, *j_media_codec_video_decoder_, j_stride_field_); | 
| -  int slice_height = GetIntField(jni, *j_media_codec_video_decoder_, | 
| -      j_slice_height_field_); | 
| - | 
| -  rtc::scoped_refptr<webrtc::VideoFrameBuffer> frame_buffer; | 
| -  int64_t presentation_timestamps_ms = 0; | 
| -  int64_t output_timestamps_ms = 0; | 
| -  int64_t output_ntp_timestamps_ms = 0; | 
| -  int decode_time_ms = 0; | 
| -  int64_t frame_delayed_ms = 0; | 
| -  if (use_surface_) { | 
| -    // Extract data from Java DecodedTextureBuffer. | 
| -    presentation_timestamps_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, | 
| -        j_texture_presentation_timestamp_ms_field_); | 
| -    output_timestamps_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, j_texture_timestamp_ms_field_); | 
| -    output_ntp_timestamps_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, j_texture_ntp_timestamp_ms_field_); | 
| -    decode_time_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, j_texture_decode_time_ms_field_); | 
| - | 
| -    const int texture_id = | 
| -        GetIntField(jni, j_decoder_output_buffer, j_texture_id_field_); | 
| -    if (texture_id != 0) {  // |texture_id| == 0 represents a dropped frame. | 
| -      const jfloatArray j_transform_matrix = | 
| -          reinterpret_cast<jfloatArray>(GetObjectField( | 
| -              jni, j_decoder_output_buffer, j_transform_matrix_field_)); | 
| -      frame_delayed_ms = GetLongField( | 
| -          jni, j_decoder_output_buffer, j_texture_frame_delay_ms_field_); | 
| - | 
| -      // Create webrtc::VideoFrameBuffer with native texture handle. | 
| -      frame_buffer = surface_texture_helper_->CreateTextureFrame( | 
| -          width, height, NativeHandleImpl(jni, texture_id, j_transform_matrix)); | 
| -    } else { | 
| -      EnableFrameLogOnWarning(); | 
| -    } | 
| -  } else { | 
| -    // Extract data from Java ByteBuffer and create output yuv420 frame - | 
| -    // for non surface decoding only. | 
| -    const int output_buffer_index = GetIntField( | 
| -        jni, j_decoder_output_buffer, j_info_index_field_); | 
| -    const int output_buffer_offset = GetIntField( | 
| -        jni, j_decoder_output_buffer, j_info_offset_field_); | 
| -    const int output_buffer_size = GetIntField( | 
| -        jni, j_decoder_output_buffer, j_info_size_field_); | 
| -    presentation_timestamps_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, j_presentation_timestamp_ms_field_); | 
| -    output_timestamps_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, j_timestamp_ms_field_); | 
| -    output_ntp_timestamps_ms = GetLongField( | 
| -        jni, j_decoder_output_buffer, j_ntp_timestamp_ms_field_); | 
| - | 
| -    decode_time_ms = GetLongField(jni, j_decoder_output_buffer, | 
| -                                  j_byte_buffer_decode_time_ms_field_); | 
| - | 
| -    if (output_buffer_size < width * height * 3 / 2) { | 
| -      ALOGE << "Insufficient output buffer size: " << output_buffer_size; | 
| -      return false; | 
| -    } | 
| -    if (output_buffer_size < stride * height * 3 / 2 && | 
| -        slice_height == height && stride > width) { | 
| -      // Some codecs (Exynos) incorrectly report stride information for | 
| -      // output byte buffer, so actual stride value need to be corrected. | 
| -      stride = output_buffer_size * 2 / (height * 3); | 
| -    } | 
| -    jobjectArray output_buffers = reinterpret_cast<jobjectArray>(GetObjectField( | 
| -        jni, *j_media_codec_video_decoder_, j_output_buffers_field_)); | 
| -    jobject output_buffer = | 
| -        jni->GetObjectArrayElement(output_buffers, output_buffer_index); | 
| -    uint8_t* payload = reinterpret_cast<uint8_t*>(jni->GetDirectBufferAddress( | 
| -        output_buffer)); | 
| -    if (CheckException(jni)) { | 
| -      return false; | 
| -    } | 
| -    payload += output_buffer_offset; | 
| - | 
| -    // Create yuv420 frame. | 
| -    frame_buffer = decoded_frame_pool_.CreateBuffer(width, height); | 
| -    if (color_format == COLOR_FormatYUV420Planar) { | 
| -      RTC_CHECK_EQ(0, stride % 2); | 
| -      RTC_CHECK_EQ(0, slice_height % 2); | 
| -      const int uv_stride = stride / 2; | 
| -      const int u_slice_height = slice_height / 2; | 
| -      const uint8_t* y_ptr = payload; | 
| -      const uint8_t* u_ptr = y_ptr + stride * slice_height; | 
| -      const uint8_t* v_ptr = u_ptr + uv_stride * u_slice_height; | 
| -      libyuv::I420Copy(y_ptr, stride, | 
| -                       u_ptr, uv_stride, | 
| -                       v_ptr, uv_stride, | 
| -                       frame_buffer->MutableDataY(), | 
| -                       frame_buffer->StrideY(), | 
| -                       frame_buffer->MutableDataU(), | 
| -                       frame_buffer->StrideU(), | 
| -                       frame_buffer->MutableDataV(), | 
| -                       frame_buffer->StrideV(), | 
| -                       width, height); | 
| -    } else { | 
| -      // All other supported formats are nv12. | 
| -      const uint8_t* y_ptr = payload; | 
| -      const uint8_t* uv_ptr = y_ptr + stride * slice_height; | 
| -      libyuv::NV12ToI420( | 
| -          y_ptr, stride, | 
| -          uv_ptr, stride, | 
| -          frame_buffer->MutableDataY(), | 
| -          frame_buffer->StrideY(), | 
| -          frame_buffer->MutableDataU(), | 
| -          frame_buffer->StrideU(), | 
| -          frame_buffer->MutableDataV(), | 
| -          frame_buffer->StrideV(), | 
| -          width, height); | 
| -    } | 
| -    // Return output byte buffer back to codec. | 
| -    jni->CallVoidMethod( | 
| -        *j_media_codec_video_decoder_, | 
| -        j_return_decoded_byte_buffer_method_, | 
| -        output_buffer_index); | 
| -    if (CheckException(jni)) { | 
| -      ALOGE << "returnDecodedOutputBuffer error"; | 
| -      return false; | 
| -    } | 
| -  } | 
| -  if (frames_decoded_ < frames_decoded_logged_) { | 
| -    ALOGD << "Decoder frame out # " << frames_decoded_ << | 
| -        ". " << width << " x " << height << | 
| -        ". " << stride << " x " <<  slice_height << | 
| -        ". Color: " << color_format << | 
| -        ". TS: " << presentation_timestamps_ms << | 
| -        ". DecTime: " << (int)decode_time_ms << | 
| -        ". DelayTime: " << (int)frame_delayed_ms; | 
| -  } | 
| - | 
| -  // Calculate and print decoding statistics - every 3 seconds. | 
| -  frames_decoded_++; | 
| -  current_frames_++; | 
| -  current_decoding_time_ms_ += decode_time_ms; | 
| -  current_delay_time_ms_ += frame_delayed_ms; | 
| -  int statistic_time_ms = rtc::TimeMillis() - start_time_ms_; | 
| -  if (statistic_time_ms >= kMediaCodecStatisticsIntervalMs && | 
| -      current_frames_ > 0) { | 
| -    int current_bitrate = current_bytes_ * 8 / statistic_time_ms; | 
| -    int current_fps = | 
| -        (current_frames_ * 1000 + statistic_time_ms / 2) / statistic_time_ms; | 
| -    ALOGD << "Frames decoded: " << frames_decoded_ << | 
| -        ". Received: " <<  frames_received_ << | 
| -        ". Bitrate: " << current_bitrate << " kbps" << | 
| -        ". Fps: " << current_fps << | 
| -        ". DecTime: " << (current_decoding_time_ms_ / current_frames_) << | 
| -        ". DelayTime: " << (current_delay_time_ms_ / current_frames_) << | 
| -        " for last " << statistic_time_ms << " ms."; | 
| -    start_time_ms_ = rtc::TimeMillis(); | 
| -    current_frames_ = 0; | 
| -    current_bytes_ = 0; | 
| -    current_decoding_time_ms_ = 0; | 
| -    current_delay_time_ms_ = 0; | 
| -  } | 
| - | 
| -  // If the frame was dropped, frame_buffer is left as nullptr. | 
| -  if (frame_buffer) { | 
| -    VideoFrame decoded_frame(frame_buffer, 0, 0, webrtc::kVideoRotation_0); | 
| -    decoded_frame.set_timestamp(output_timestamps_ms); | 
| -    decoded_frame.set_ntp_time_ms(output_ntp_timestamps_ms); | 
| - | 
| -    const int32_t callback_status = | 
| -        callback_->Decoded(decoded_frame, decode_time_ms); | 
| -    if (callback_status > 0) { | 
| -      ALOGE << "callback error"; | 
| -    } | 
| -  } | 
| -  return true; | 
| -} | 
| - | 
| -int32_t MediaCodecVideoDecoder::RegisterDecodeCompleteCallback( | 
| -    DecodedImageCallback* callback) { | 
| -  callback_ = callback; | 
| -  return WEBRTC_VIDEO_CODEC_OK; | 
| -} | 
| - | 
| -void MediaCodecVideoDecoder::OnMessage(rtc::Message* msg) { | 
| -  JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| -  if (!inited_) { | 
| -    return; | 
| -  } | 
| -  // We only ever send one message to |this| directly (not through a Bind()'d | 
| -  // functor), so expect no ID/data. | 
| -  RTC_CHECK(!msg->message_id) << "Unexpected message!"; | 
| -  RTC_CHECK(!msg->pdata) << "Unexpected message!"; | 
| -  CheckOnCodecThread(); | 
| - | 
| -  if (!DeliverPendingOutputs(jni, 0)) { | 
| -    ALOGE << "OnMessage: DeliverPendingOutputs error"; | 
| -    ProcessHWErrorOnCodecThread(); | 
| -    return; | 
| -  } | 
| -  codec_thread_->PostDelayed(RTC_FROM_HERE, kMediaCodecPollMs, this); | 
| -} | 
| - | 
| -MediaCodecVideoDecoderFactory::MediaCodecVideoDecoderFactory() | 
| -    : egl_context_(nullptr) { | 
| -  ALOGD << "MediaCodecVideoDecoderFactory ctor"; | 
| -  JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -  ScopedLocalRefFrame local_ref_frame(jni); | 
| -  jclass j_decoder_class = FindClass(jni, "org/webrtc/MediaCodecVideoDecoder"); | 
| -  supported_codec_types_.clear(); | 
| - | 
| -  bool is_vp8_hw_supported = jni->CallStaticBooleanMethod( | 
| -      j_decoder_class, | 
| -      GetStaticMethodID(jni, j_decoder_class, "isVp8HwSupported", "()Z")); | 
| -  if (CheckException(jni)) { | 
| -    is_vp8_hw_supported = false; | 
| -  } | 
| -  if (is_vp8_hw_supported) { | 
| -    ALOGD << "VP8 HW Decoder supported."; | 
| -    supported_codec_types_.push_back(kVideoCodecVP8); | 
| -  } | 
| - | 
| -  bool is_vp9_hw_supported = jni->CallStaticBooleanMethod( | 
| -      j_decoder_class, | 
| -      GetStaticMethodID(jni, j_decoder_class, "isVp9HwSupported", "()Z")); | 
| -  if (CheckException(jni)) { | 
| -    is_vp9_hw_supported = false; | 
| -  } | 
| -  if (is_vp9_hw_supported) { | 
| -    ALOGD << "VP9 HW Decoder supported."; | 
| -    supported_codec_types_.push_back(kVideoCodecVP9); | 
| -  } | 
| - | 
| -  bool is_h264_hw_supported = jni->CallStaticBooleanMethod( | 
| -      j_decoder_class, | 
| -      GetStaticMethodID(jni, j_decoder_class, "isH264HwSupported", "()Z")); | 
| -  if (CheckException(jni)) { | 
| -    is_h264_hw_supported = false; | 
| -  } | 
| -  if (is_h264_hw_supported) { | 
| -    ALOGD << "H264 HW Decoder supported."; | 
| -    supported_codec_types_.push_back(kVideoCodecH264); | 
| -  } | 
| -} | 
| - | 
| -MediaCodecVideoDecoderFactory::~MediaCodecVideoDecoderFactory() { | 
| -  ALOGD << "MediaCodecVideoDecoderFactory dtor"; | 
| -  if (egl_context_) { | 
| -    JNIEnv* jni = AttachCurrentThreadIfNeeded(); | 
| -    jni->DeleteGlobalRef(egl_context_); | 
| -  } | 
| -} | 
| - | 
| -void MediaCodecVideoDecoderFactory::SetEGLContext( | 
| -    JNIEnv* jni, jobject egl_context) { | 
| -  ALOGD << "MediaCodecVideoDecoderFactory::SetEGLContext"; | 
| -  if (egl_context_) { | 
| -    jni->DeleteGlobalRef(egl_context_); | 
| -    egl_context_ = nullptr; | 
| -  } | 
| -  egl_context_ = jni->NewGlobalRef(egl_context); | 
| -  if (CheckException(jni)) { | 
| -    ALOGE << "error calling NewGlobalRef for EGL Context."; | 
| -  } | 
| -} | 
| - | 
| -webrtc::VideoDecoder* MediaCodecVideoDecoderFactory::CreateVideoDecoder( | 
| -    VideoCodecType type) { | 
| -  if (supported_codec_types_.empty()) { | 
| -    ALOGW << "No HW video decoder for type " << (int)type; | 
| -    return nullptr; | 
| -  } | 
| -  for (VideoCodecType codec_type : supported_codec_types_) { | 
| -    if (codec_type == type) { | 
| -      ALOGD << "Create HW video decoder for type " << (int)type; | 
| -      return new MediaCodecVideoDecoder(AttachCurrentThreadIfNeeded(), type, | 
| -                                        egl_context_); | 
| -    } | 
| -  } | 
| -  ALOGW << "Can not find HW video decoder for type " << (int)type; | 
| -  return nullptr; | 
| -} | 
| - | 
| -void MediaCodecVideoDecoderFactory::DestroyVideoDecoder( | 
| -    webrtc::VideoDecoder* decoder) { | 
| -  ALOGD << "Destroy video decoder."; | 
| -  delete decoder; | 
| -} | 
| - | 
| -const char* MediaCodecVideoDecoder::ImplementationName() const { | 
| -  return "MediaCodec"; | 
| -} | 
| - | 
| -}  // namespace webrtc_jni | 
|  |