| Index: webrtc/api/android/java/src/org/webrtc/MediaCodecVideoDecoder.java | 
| diff --git a/webrtc/api/android/java/src/org/webrtc/MediaCodecVideoDecoder.java b/webrtc/api/android/java/src/org/webrtc/MediaCodecVideoDecoder.java | 
| deleted file mode 100644 | 
| index 02235fe3c74a5559c0980b6448ee57e1bf2f74d3..0000000000000000000000000000000000000000 | 
| --- a/webrtc/api/android/java/src/org/webrtc/MediaCodecVideoDecoder.java | 
| +++ /dev/null | 
| @@ -1,714 +0,0 @@ | 
| -/* | 
| - *  Copyright 2014 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. | 
| - */ | 
| - | 
| -package org.webrtc; | 
| - | 
| -import android.media.MediaCodec; | 
| -import android.media.MediaCodecInfo; | 
| -import android.media.MediaCodecInfo.CodecCapabilities; | 
| -import android.media.MediaCodecList; | 
| -import android.media.MediaFormat; | 
| -import android.os.Build; | 
| -import android.os.SystemClock; | 
| -import android.view.Surface; | 
| - | 
| -import org.webrtc.Logging; | 
| - | 
| -import java.nio.ByteBuffer; | 
| -import java.util.Arrays; | 
| -import java.util.HashSet; | 
| -import java.util.LinkedList; | 
| -import java.util.List; | 
| -import java.util.Set; | 
| -import java.util.concurrent.CountDownLatch; | 
| -import java.util.Queue; | 
| -import java.util.concurrent.TimeUnit; | 
| - | 
| -// Java-side of peerconnection_jni.cc:MediaCodecVideoDecoder. | 
| -// This class is an implementation detail of the Java PeerConnection API. | 
| -@SuppressWarnings("deprecation") | 
| -public class MediaCodecVideoDecoder { | 
| -  // This class is constructed, operated, and destroyed by its C++ incarnation, | 
| -  // so the class and its methods have non-public visibility.  The API this | 
| -  // class exposes aims to mimic the webrtc::VideoDecoder API as closely as | 
| -  // possibly to minimize the amount of translation work necessary. | 
| - | 
| -  private static final String TAG = "MediaCodecVideoDecoder"; | 
| -  private static final long MAX_DECODE_TIME_MS = 200; | 
| - | 
| -  // Tracks webrtc::VideoCodecType. | 
| -  public enum VideoCodecType { | 
| -    VIDEO_CODEC_VP8, | 
| -    VIDEO_CODEC_VP9, | 
| -    VIDEO_CODEC_H264 | 
| -  } | 
| - | 
| -  // Timeout for input buffer dequeue. | 
| -  private static final int DEQUEUE_INPUT_TIMEOUT = 500000; | 
| -  // Timeout for codec releasing. | 
| -  private static final int MEDIA_CODEC_RELEASE_TIMEOUT_MS = 5000; | 
| -  // Max number of output buffers queued before starting to drop decoded frames. | 
| -  private static final int MAX_QUEUED_OUTPUTBUFFERS = 3; | 
| -  // Active running decoder instance. Set in initDecode() (called from native code) | 
| -  // and reset to null in release() call. | 
| -  private static MediaCodecVideoDecoder runningInstance = null; | 
| -  private static MediaCodecVideoDecoderErrorCallback errorCallback = null; | 
| -  private static int codecErrors = 0; | 
| -  // List of disabled codec types - can be set from application. | 
| -  private static Set<String> hwDecoderDisabledTypes = new HashSet<String>(); | 
| - | 
| -  private Thread mediaCodecThread; | 
| -  private MediaCodec mediaCodec; | 
| -  private ByteBuffer[] inputBuffers; | 
| -  private ByteBuffer[] outputBuffers; | 
| -  private static final String VP8_MIME_TYPE = "video/x-vnd.on2.vp8"; | 
| -  private static final String VP9_MIME_TYPE = "video/x-vnd.on2.vp9"; | 
| -  private static final String H264_MIME_TYPE = "video/avc"; | 
| -  // List of supported HW VP8 decoders. | 
| -  private static final String[] supportedVp8HwCodecPrefixes = | 
| -    {"OMX.qcom.", "OMX.Nvidia.", "OMX.Exynos.", "OMX.Intel." }; | 
| -  // List of supported HW VP9 decoders. | 
| -  private static final String[] supportedVp9HwCodecPrefixes = | 
| -    {"OMX.qcom.", "OMX.Exynos." }; | 
| -  // List of supported HW H.264 decoders. | 
| -  private static final String[] supportedH264HwCodecPrefixes = | 
| -    {"OMX.qcom.", "OMX.Intel.", "OMX.Exynos." }; | 
| -  // NV12 color format supported by QCOM codec, but not declared in MediaCodec - | 
| -  // see /hardware/qcom/media/mm-core/inc/OMX_QCOMExtns.h | 
| -  private static final int | 
| -    COLOR_QCOM_FORMATYUV420PackedSemiPlanar32m = 0x7FA30C04; | 
| -  // Allowable color formats supported by codec - in order of preference. | 
| -  private static final List<Integer> supportedColorList = Arrays.asList( | 
| -    CodecCapabilities.COLOR_FormatYUV420Planar, | 
| -    CodecCapabilities.COLOR_FormatYUV420SemiPlanar, | 
| -    CodecCapabilities.COLOR_QCOM_FormatYUV420SemiPlanar, | 
| -    COLOR_QCOM_FORMATYUV420PackedSemiPlanar32m); | 
| -  private int colorFormat; | 
| -  private int width; | 
| -  private int height; | 
| -  private int stride; | 
| -  private int sliceHeight; | 
| -  private boolean hasDecodedFirstFrame; | 
| -  private final Queue<TimeStamps> decodeStartTimeMs = new LinkedList<TimeStamps>(); | 
| -  private boolean useSurface; | 
| - | 
| -  // The below variables are only used when decoding to a Surface. | 
| -  private TextureListener textureListener; | 
| -  private int droppedFrames; | 
| -  private Surface surface = null; | 
| -  private final Queue<DecodedOutputBuffer> | 
| -      dequeuedSurfaceOutputBuffers = new LinkedList<DecodedOutputBuffer>(); | 
| - | 
| -  // MediaCodec error handler - invoked when critical error happens which may prevent | 
| -  // further use of media codec API. Now it means that one of media codec instances | 
| -  // is hanging and can no longer be used in the next call. | 
| -  public static interface MediaCodecVideoDecoderErrorCallback { | 
| -    void onMediaCodecVideoDecoderCriticalError(int codecErrors); | 
| -  } | 
| - | 
| -  public static void setErrorCallback(MediaCodecVideoDecoderErrorCallback errorCallback) { | 
| -    Logging.d(TAG, "Set error callback"); | 
| -    MediaCodecVideoDecoder.errorCallback = errorCallback; | 
| -  } | 
| - | 
| -  // Functions to disable HW decoding - can be called from applications for platforms | 
| -  // which have known HW decoding problems. | 
| -  public static void disableVp8HwCodec() { | 
| -    Logging.w(TAG, "VP8 decoding is disabled by application."); | 
| -    hwDecoderDisabledTypes.add(VP8_MIME_TYPE); | 
| -  } | 
| - | 
| -  public static void disableVp9HwCodec() { | 
| -    Logging.w(TAG, "VP9 decoding is disabled by application."); | 
| -    hwDecoderDisabledTypes.add(VP9_MIME_TYPE); | 
| -  } | 
| - | 
| -  public static void disableH264HwCodec() { | 
| -    Logging.w(TAG, "H.264 decoding is disabled by application."); | 
| -    hwDecoderDisabledTypes.add(H264_MIME_TYPE); | 
| -  } | 
| - | 
| -  // Functions to query if HW decoding is supported. | 
| -  public static boolean isVp8HwSupported() { | 
| -    return !hwDecoderDisabledTypes.contains(VP8_MIME_TYPE) && | 
| -        (findDecoder(VP8_MIME_TYPE, supportedVp8HwCodecPrefixes) != null); | 
| -  } | 
| - | 
| -  public static boolean isVp9HwSupported() { | 
| -    return !hwDecoderDisabledTypes.contains(VP9_MIME_TYPE) && | 
| -        (findDecoder(VP9_MIME_TYPE, supportedVp9HwCodecPrefixes) != null); | 
| -  } | 
| - | 
| -  public static boolean isH264HwSupported() { | 
| -    return !hwDecoderDisabledTypes.contains(H264_MIME_TYPE) && | 
| -        (findDecoder(H264_MIME_TYPE, supportedH264HwCodecPrefixes) != null); | 
| -  } | 
| - | 
| -  public static void printStackTrace() { | 
| -    if (runningInstance != null && runningInstance.mediaCodecThread != null) { | 
| -      StackTraceElement[] mediaCodecStackTraces = runningInstance.mediaCodecThread.getStackTrace(); | 
| -      if (mediaCodecStackTraces.length > 0) { | 
| -        Logging.d(TAG, "MediaCodecVideoDecoder stacks trace:"); | 
| -        for (StackTraceElement stackTrace : mediaCodecStackTraces) { | 
| -          Logging.d(TAG, stackTrace.toString()); | 
| -        } | 
| -      } | 
| -    } | 
| -  } | 
| - | 
| -  // Helper struct for findDecoder() below. | 
| -  private static class DecoderProperties { | 
| -    public DecoderProperties(String codecName, int colorFormat) { | 
| -      this.codecName = codecName; | 
| -      this.colorFormat = colorFormat; | 
| -    } | 
| -    public final String codecName; // OpenMax component name for VP8 codec. | 
| -    public final int colorFormat;  // Color format supported by codec. | 
| -  } | 
| - | 
| -  private static DecoderProperties findDecoder( | 
| -      String mime, String[] supportedCodecPrefixes) { | 
| -    if (Build.VERSION.SDK_INT < Build.VERSION_CODES.KITKAT) { | 
| -      return null; // MediaCodec.setParameters is missing. | 
| -    } | 
| -    Logging.d(TAG, "Trying to find HW decoder for mime " + mime); | 
| -    for (int i = 0; i < MediaCodecList.getCodecCount(); ++i) { | 
| -      MediaCodecInfo info = MediaCodecList.getCodecInfoAt(i); | 
| -      if (info.isEncoder()) { | 
| -        continue; | 
| -      } | 
| -      String name = null; | 
| -      for (String mimeType : info.getSupportedTypes()) { | 
| -        if (mimeType.equals(mime)) { | 
| -          name = info.getName(); | 
| -          break; | 
| -        } | 
| -      } | 
| -      if (name == null) { | 
| -        continue;  // No HW support in this codec; try the next one. | 
| -      } | 
| -      Logging.d(TAG, "Found candidate decoder " + name); | 
| - | 
| -      // Check if this is supported decoder. | 
| -      boolean supportedCodec = false; | 
| -      for (String codecPrefix : supportedCodecPrefixes) { | 
| -        if (name.startsWith(codecPrefix)) { | 
| -          supportedCodec = true; | 
| -          break; | 
| -        } | 
| -      } | 
| -      if (!supportedCodec) { | 
| -        continue; | 
| -      } | 
| - | 
| -      // Check if codec supports either yuv420 or nv12. | 
| -      CodecCapabilities capabilities = | 
| -          info.getCapabilitiesForType(mime); | 
| -      for (int colorFormat : capabilities.colorFormats) { | 
| -        Logging.v(TAG, "   Color: 0x" + Integer.toHexString(colorFormat)); | 
| -      } | 
| -      for (int supportedColorFormat : supportedColorList) { | 
| -        for (int codecColorFormat : capabilities.colorFormats) { | 
| -          if (codecColorFormat == supportedColorFormat) { | 
| -            // Found supported HW decoder. | 
| -            Logging.d(TAG, "Found target decoder " + name + | 
| -                ". Color: 0x" + Integer.toHexString(codecColorFormat)); | 
| -            return new DecoderProperties(name, codecColorFormat); | 
| -          } | 
| -        } | 
| -      } | 
| -    } | 
| -    Logging.d(TAG, "No HW decoder found for mime " + mime); | 
| -    return null;  // No HW decoder. | 
| -  } | 
| - | 
| -  private void checkOnMediaCodecThread() throws IllegalStateException { | 
| -    if (mediaCodecThread.getId() != Thread.currentThread().getId()) { | 
| -      throw new IllegalStateException( | 
| -          "MediaCodecVideoDecoder previously operated on " + mediaCodecThread + | 
| -          " but is now called on " + Thread.currentThread()); | 
| -    } | 
| -  } | 
| - | 
| -  // Pass null in |surfaceTextureHelper| to configure the codec for ByteBuffer output. | 
| -  private boolean initDecode( | 
| -      VideoCodecType type, int width, int height, | 
| -      SurfaceTextureHelper surfaceTextureHelper) { | 
| -    if (mediaCodecThread != null) { | 
| -      throw new RuntimeException("initDecode: Forgot to release()?"); | 
| -    } | 
| - | 
| -    String mime = null; | 
| -    useSurface = (surfaceTextureHelper != null); | 
| -    String[] supportedCodecPrefixes = null; | 
| -    if (type == VideoCodecType.VIDEO_CODEC_VP8) { | 
| -      mime = VP8_MIME_TYPE; | 
| -      supportedCodecPrefixes = supportedVp8HwCodecPrefixes; | 
| -    } else if (type == VideoCodecType.VIDEO_CODEC_VP9) { | 
| -      mime = VP9_MIME_TYPE; | 
| -      supportedCodecPrefixes = supportedVp9HwCodecPrefixes; | 
| -    } else if (type == VideoCodecType.VIDEO_CODEC_H264) { | 
| -      mime = H264_MIME_TYPE; | 
| -      supportedCodecPrefixes = supportedH264HwCodecPrefixes; | 
| -    } else { | 
| -      throw new RuntimeException("initDecode: Non-supported codec " + type); | 
| -    } | 
| -    DecoderProperties properties = findDecoder(mime, supportedCodecPrefixes); | 
| -    if (properties == null) { | 
| -      throw new RuntimeException("Cannot find HW decoder for " + type); | 
| -    } | 
| - | 
| -    Logging.d(TAG, "Java initDecode: " + type + " : "+ width + " x " + height + | 
| -        ". Color: 0x" + Integer.toHexString(properties.colorFormat) + | 
| -        ". Use Surface: " + useSurface); | 
| - | 
| -    runningInstance = this; // Decoder is now running and can be queried for stack traces. | 
| -    mediaCodecThread = Thread.currentThread(); | 
| -    try { | 
| -      this.width = width; | 
| -      this.height = height; | 
| -      stride = width; | 
| -      sliceHeight = height; | 
| - | 
| -      if (useSurface) { | 
| -        textureListener = new TextureListener(surfaceTextureHelper); | 
| -        surface = new Surface(surfaceTextureHelper.getSurfaceTexture()); | 
| -      } | 
| - | 
| -      MediaFormat format = MediaFormat.createVideoFormat(mime, width, height); | 
| -      if (!useSurface) { | 
| -        format.setInteger(MediaFormat.KEY_COLOR_FORMAT, properties.colorFormat); | 
| -      } | 
| -      Logging.d(TAG, "  Format: " + format); | 
| -      mediaCodec = MediaCodecVideoEncoder.createByCodecName(properties.codecName); | 
| -      if (mediaCodec == null) { | 
| -        Logging.e(TAG, "Can not create media decoder"); | 
| -        return false; | 
| -      } | 
| -      mediaCodec.configure(format, surface, null, 0); | 
| -      mediaCodec.start(); | 
| - | 
| -      colorFormat = properties.colorFormat; | 
| -      outputBuffers = mediaCodec.getOutputBuffers(); | 
| -      inputBuffers = mediaCodec.getInputBuffers(); | 
| -      decodeStartTimeMs.clear(); | 
| -      hasDecodedFirstFrame = false; | 
| -      dequeuedSurfaceOutputBuffers.clear(); | 
| -      droppedFrames = 0; | 
| -      Logging.d(TAG, "Input buffers: " + inputBuffers.length + | 
| -          ". Output buffers: " + outputBuffers.length); | 
| -      return true; | 
| -    } catch (IllegalStateException e) { | 
| -      Logging.e(TAG, "initDecode failed", e); | 
| -      return false; | 
| -    } | 
| -  } | 
| - | 
| -  // Resets the decoder so it can start decoding frames with new resolution. | 
| -  // Flushes MediaCodec and clears decoder output buffers. | 
| -  private void reset(int width, int height) { | 
| -    if (mediaCodecThread == null || mediaCodec == null) { | 
| -      throw new RuntimeException("Incorrect reset call for non-initialized decoder."); | 
| -    } | 
| -    Logging.d(TAG, "Java reset: " + width + " x " + height); | 
| - | 
| -    mediaCodec.flush(); | 
| - | 
| -    this.width = width; | 
| -    this.height = height; | 
| -    decodeStartTimeMs.clear(); | 
| -    dequeuedSurfaceOutputBuffers.clear(); | 
| -    hasDecodedFirstFrame = false; | 
| -    droppedFrames = 0; | 
| -  } | 
| - | 
| -  private void release() { | 
| -    Logging.d(TAG, "Java releaseDecoder. Total number of dropped frames: " + droppedFrames); | 
| -    checkOnMediaCodecThread(); | 
| - | 
| -    // Run Mediacodec stop() and release() on separate thread since sometime | 
| -    // Mediacodec.stop() may hang. | 
| -    final CountDownLatch releaseDone = new CountDownLatch(1); | 
| - | 
| -    Runnable runMediaCodecRelease = new Runnable() { | 
| -      @Override | 
| -      public void run() { | 
| -        try { | 
| -          Logging.d(TAG, "Java releaseDecoder on release thread"); | 
| -          mediaCodec.stop(); | 
| -          mediaCodec.release(); | 
| -          Logging.d(TAG, "Java releaseDecoder on release thread done"); | 
| -        } catch (Exception e) { | 
| -          Logging.e(TAG, "Media decoder release failed", e); | 
| -        } | 
| -        releaseDone.countDown(); | 
| -      } | 
| -    }; | 
| -    new Thread(runMediaCodecRelease).start(); | 
| - | 
| -    if (!ThreadUtils.awaitUninterruptibly(releaseDone, MEDIA_CODEC_RELEASE_TIMEOUT_MS)) { | 
| -      Logging.e(TAG, "Media decoder release timeout"); | 
| -      codecErrors++; | 
| -      if (errorCallback != null) { | 
| -        Logging.e(TAG, "Invoke codec error callback. Errors: " + codecErrors); | 
| -        errorCallback.onMediaCodecVideoDecoderCriticalError(codecErrors); | 
| -      } | 
| -    } | 
| - | 
| -    mediaCodec = null; | 
| -    mediaCodecThread = null; | 
| -    runningInstance = null; | 
| -    if (useSurface) { | 
| -      surface.release(); | 
| -      surface = null; | 
| -      textureListener.release(); | 
| -    } | 
| -    Logging.d(TAG, "Java releaseDecoder done"); | 
| -  } | 
| - | 
| -  // Dequeue an input buffer and return its index, -1 if no input buffer is | 
| -  // available, or -2 if the codec is no longer operative. | 
| -  private int dequeueInputBuffer() { | 
| -    checkOnMediaCodecThread(); | 
| -    try { | 
| -      return mediaCodec.dequeueInputBuffer(DEQUEUE_INPUT_TIMEOUT); | 
| -    } catch (IllegalStateException e) { | 
| -      Logging.e(TAG, "dequeueIntputBuffer failed", e); | 
| -      return -2; | 
| -    } | 
| -  } | 
| - | 
| -  private boolean queueInputBuffer(int inputBufferIndex, int size, long presentationTimeStamUs, | 
| -      long timeStampMs, long ntpTimeStamp) { | 
| -    checkOnMediaCodecThread(); | 
| -    try { | 
| -      inputBuffers[inputBufferIndex].position(0); | 
| -      inputBuffers[inputBufferIndex].limit(size); | 
| -      decodeStartTimeMs.add(new TimeStamps(SystemClock.elapsedRealtime(), timeStampMs, | 
| -          ntpTimeStamp)); | 
| -      mediaCodec.queueInputBuffer(inputBufferIndex, 0, size, presentationTimeStamUs, 0); | 
| -      return true; | 
| -    } | 
| -    catch (IllegalStateException e) { | 
| -      Logging.e(TAG, "decode failed", e); | 
| -      return false; | 
| -    } | 
| -  } | 
| - | 
| -  private static class TimeStamps { | 
| -    public TimeStamps(long decodeStartTimeMs, long timeStampMs, long ntpTimeStampMs) { | 
| -      this.decodeStartTimeMs = decodeStartTimeMs; | 
| -      this.timeStampMs = timeStampMs; | 
| -      this.ntpTimeStampMs = ntpTimeStampMs; | 
| -    } | 
| -    // Time when this frame was queued for decoding. | 
| -    private final long decodeStartTimeMs; | 
| -    // Only used for bookkeeping in Java. Stores C++ inputImage._timeStamp value for input frame. | 
| -    private final long timeStampMs; | 
| -    // Only used for bookkeeping in Java. Stores C++ inputImage.ntp_time_ms_ value for input frame. | 
| -    private final long ntpTimeStampMs; | 
| -  } | 
| - | 
| -  // Helper struct for dequeueOutputBuffer() below. | 
| -  private static class DecodedOutputBuffer { | 
| -    public DecodedOutputBuffer(int index, int offset, int size, long presentationTimeStampMs, | 
| -        long timeStampMs, long ntpTimeStampMs, long decodeTime, long endDecodeTime) { | 
| -      this.index = index; | 
| -      this.offset = offset; | 
| -      this.size = size; | 
| -      this.presentationTimeStampMs = presentationTimeStampMs; | 
| -      this.timeStampMs = timeStampMs; | 
| -      this.ntpTimeStampMs = ntpTimeStampMs; | 
| -      this.decodeTimeMs = decodeTime; | 
| -      this.endDecodeTimeMs = endDecodeTime; | 
| -    } | 
| - | 
| -    private final int index; | 
| -    private final int offset; | 
| -    private final int size; | 
| -    // Presentation timestamp returned in dequeueOutputBuffer call. | 
| -    private final long presentationTimeStampMs; | 
| -    // C++ inputImage._timeStamp value for output frame. | 
| -    private final long timeStampMs; | 
| -    // C++ inputImage.ntp_time_ms_ value for output frame. | 
| -    private final long ntpTimeStampMs; | 
| -    // Number of ms it took to decode this frame. | 
| -    private final long decodeTimeMs; | 
| -    // System time when this frame decoding finished. | 
| -    private final long endDecodeTimeMs; | 
| -  } | 
| - | 
| -  // Helper struct for dequeueTextureBuffer() below. | 
| -  private static class DecodedTextureBuffer { | 
| -    private final int textureID; | 
| -    private final float[] transformMatrix; | 
| -    // Presentation timestamp returned in dequeueOutputBuffer call. | 
| -    private final long presentationTimeStampMs; | 
| -    // C++ inputImage._timeStamp value for output frame. | 
| -    private final long timeStampMs; | 
| -    // C++ inputImage.ntp_time_ms_ value for output frame. | 
| -    private final long ntpTimeStampMs; | 
| -    // Number of ms it took to decode this frame. | 
| -    private final long decodeTimeMs; | 
| -    // Interval from when the frame finished decoding until this buffer has been created. | 
| -    // Since there is only one texture, this interval depend on the time from when | 
| -    // a frame is decoded and provided to C++ and until that frame is returned to the MediaCodec | 
| -    // so that the texture can be updated with the next decoded frame. | 
| -    private final long frameDelayMs; | 
| - | 
| -    // A DecodedTextureBuffer with zero |textureID| has special meaning and represents a frame | 
| -    // that was dropped. | 
| -    public DecodedTextureBuffer(int textureID, float[] transformMatrix, | 
| -        long presentationTimeStampMs, long timeStampMs, long ntpTimeStampMs, long decodeTimeMs, | 
| -        long frameDelay) { | 
| -      this.textureID = textureID; | 
| -      this.transformMatrix = transformMatrix; | 
| -      this.presentationTimeStampMs = presentationTimeStampMs; | 
| -      this.timeStampMs = timeStampMs; | 
| -      this.ntpTimeStampMs = ntpTimeStampMs; | 
| -      this.decodeTimeMs = decodeTimeMs; | 
| -      this.frameDelayMs = frameDelay; | 
| -    } | 
| -  } | 
| - | 
| -  // Poll based texture listener. | 
| -  private static class TextureListener | 
| -      implements SurfaceTextureHelper.OnTextureFrameAvailableListener { | 
| -    private final SurfaceTextureHelper surfaceTextureHelper; | 
| -    // |newFrameLock| is used to synchronize arrival of new frames with wait()/notifyAll(). | 
| -    private final Object newFrameLock = new Object(); | 
| -    // |bufferToRender| is non-null when waiting for transition between addBufferToRender() to | 
| -    // onTextureFrameAvailable(). | 
| -    private DecodedOutputBuffer bufferToRender; | 
| -    private DecodedTextureBuffer renderedBuffer; | 
| - | 
| -    public TextureListener(SurfaceTextureHelper surfaceTextureHelper) { | 
| -      this.surfaceTextureHelper = surfaceTextureHelper; | 
| -      surfaceTextureHelper.startListening(this); | 
| -    } | 
| - | 
| -    public void addBufferToRender(DecodedOutputBuffer buffer) { | 
| -      if (bufferToRender != null) { | 
| -        Logging.e(TAG, | 
| -            "Unexpected addBufferToRender() called while waiting for a texture."); | 
| -        throw new IllegalStateException("Waiting for a texture."); | 
| -      } | 
| -      bufferToRender = buffer; | 
| -    } | 
| - | 
| -    public boolean isWaitingForTexture() { | 
| -      synchronized (newFrameLock) { | 
| -        return bufferToRender != null; | 
| -      } | 
| -    } | 
| - | 
| -    // Callback from |surfaceTextureHelper|. May be called on an arbitrary thread. | 
| -    @Override | 
| -    public void onTextureFrameAvailable( | 
| -        int oesTextureId, float[] transformMatrix, long timestampNs) { | 
| -      synchronized (newFrameLock) { | 
| -        if (renderedBuffer != null) { | 
| -          Logging.e(TAG, | 
| -              "Unexpected onTextureFrameAvailable() called while already holding a texture."); | 
| -          throw new IllegalStateException("Already holding a texture."); | 
| -        } | 
| -        // |timestampNs| is always zero on some Android versions. | 
| -        renderedBuffer = new DecodedTextureBuffer(oesTextureId, transformMatrix, | 
| -            bufferToRender.presentationTimeStampMs, bufferToRender.timeStampMs, | 
| -            bufferToRender.ntpTimeStampMs, bufferToRender.decodeTimeMs, | 
| -            SystemClock.elapsedRealtime() - bufferToRender.endDecodeTimeMs); | 
| -        bufferToRender = null; | 
| -        newFrameLock.notifyAll(); | 
| -      } | 
| -    } | 
| - | 
| -    // Dequeues and returns a DecodedTextureBuffer if available, or null otherwise. | 
| -    public DecodedTextureBuffer dequeueTextureBuffer(int timeoutMs) { | 
| -      synchronized (newFrameLock) { | 
| -        if (renderedBuffer == null && timeoutMs > 0 && isWaitingForTexture()) { | 
| -          try { | 
| -            newFrameLock.wait(timeoutMs); | 
| -          } catch(InterruptedException e) { | 
| -            // Restore the interrupted status by reinterrupting the thread. | 
| -            Thread.currentThread().interrupt(); | 
| -          } | 
| -        } | 
| -        DecodedTextureBuffer returnedBuffer = renderedBuffer; | 
| -        renderedBuffer = null; | 
| -        return returnedBuffer; | 
| -      } | 
| -    } | 
| - | 
| -    public void release() { | 
| -      // SurfaceTextureHelper.stopListening() will block until any onTextureFrameAvailable() in | 
| -      // progress is done. Therefore, the call must be outside any synchronized | 
| -      // statement that is also used in the onTextureFrameAvailable() above to avoid deadlocks. | 
| -      surfaceTextureHelper.stopListening(); | 
| -      synchronized (newFrameLock) { | 
| -        if (renderedBuffer != null) { | 
| -          surfaceTextureHelper.returnTextureFrame(); | 
| -          renderedBuffer = null; | 
| -        } | 
| -      } | 
| -    } | 
| -  } | 
| - | 
| -  // Returns null if no decoded buffer is available, and otherwise a DecodedByteBuffer. | 
| -  // Throws IllegalStateException if call is made on the wrong thread, if color format changes to an | 
| -  // unsupported format, or if |mediaCodec| is not in the Executing state. Throws CodecException | 
| -  // upon codec error. | 
| -  private DecodedOutputBuffer dequeueOutputBuffer(int dequeueTimeoutMs) { | 
| -    checkOnMediaCodecThread(); | 
| -    if (decodeStartTimeMs.isEmpty()) { | 
| -      return null; | 
| -    } | 
| -    // Drain the decoder until receiving a decoded buffer or hitting | 
| -    // MediaCodec.INFO_TRY_AGAIN_LATER. | 
| -    final MediaCodec.BufferInfo info = new MediaCodec.BufferInfo(); | 
| -    while (true) { | 
| -      final int result = mediaCodec.dequeueOutputBuffer( | 
| -          info, TimeUnit.MILLISECONDS.toMicros(dequeueTimeoutMs)); | 
| -      switch (result) { | 
| -        case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED: | 
| -          outputBuffers = mediaCodec.getOutputBuffers(); | 
| -          Logging.d(TAG, "Decoder output buffers changed: " + outputBuffers.length); | 
| -          if (hasDecodedFirstFrame) { | 
| -            throw new RuntimeException("Unexpected output buffer change event."); | 
| -          } | 
| -          break; | 
| -        case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED: | 
| -          MediaFormat format = mediaCodec.getOutputFormat(); | 
| -          Logging.d(TAG, "Decoder format changed: " + format.toString()); | 
| -          int new_width = format.getInteger(MediaFormat.KEY_WIDTH); | 
| -          int new_height = format.getInteger(MediaFormat.KEY_HEIGHT); | 
| -          if (hasDecodedFirstFrame && (new_width != width || new_height != height)) { | 
| -            throw new RuntimeException("Unexpected size change. Configured " + width + "*" + | 
| -                height + ". New " + new_width + "*" + new_height); | 
| -          } | 
| -          width = format.getInteger(MediaFormat.KEY_WIDTH); | 
| -          height = format.getInteger(MediaFormat.KEY_HEIGHT); | 
| - | 
| -          if (!useSurface && format.containsKey(MediaFormat.KEY_COLOR_FORMAT)) { | 
| -            colorFormat = format.getInteger(MediaFormat.KEY_COLOR_FORMAT); | 
| -            Logging.d(TAG, "Color: 0x" + Integer.toHexString(colorFormat)); | 
| -            if (!supportedColorList.contains(colorFormat)) { | 
| -              throw new IllegalStateException("Non supported color format: " + colorFormat); | 
| -            } | 
| -          } | 
| -          if (format.containsKey("stride")) { | 
| -            stride = format.getInteger("stride"); | 
| -          } | 
| -          if (format.containsKey("slice-height")) { | 
| -            sliceHeight = format.getInteger("slice-height"); | 
| -          } | 
| -          Logging.d(TAG, "Frame stride and slice height: " + stride + " x " + sliceHeight); | 
| -          stride = Math.max(width, stride); | 
| -          sliceHeight = Math.max(height, sliceHeight); | 
| -          break; | 
| -        case MediaCodec.INFO_TRY_AGAIN_LATER: | 
| -          return null; | 
| -        default: | 
| -          hasDecodedFirstFrame = true; | 
| -          TimeStamps timeStamps = decodeStartTimeMs.remove(); | 
| -          long decodeTimeMs = SystemClock.elapsedRealtime() - timeStamps.decodeStartTimeMs; | 
| -          if (decodeTimeMs > MAX_DECODE_TIME_MS) { | 
| -            Logging.e(TAG, "Very high decode time: " + decodeTimeMs + "ms" | 
| -                + ". Q size: " + decodeStartTimeMs.size() | 
| -                + ". Might be caused by resuming H264 decoding after a pause."); | 
| -            decodeTimeMs = MAX_DECODE_TIME_MS; | 
| -          } | 
| -          return new DecodedOutputBuffer(result, | 
| -              info.offset, | 
| -              info.size, | 
| -              TimeUnit.MICROSECONDS.toMillis(info.presentationTimeUs), | 
| -              timeStamps.timeStampMs, | 
| -              timeStamps.ntpTimeStampMs, | 
| -              decodeTimeMs, | 
| -              SystemClock.elapsedRealtime()); | 
| -        } | 
| -    } | 
| -  } | 
| - | 
| -  // Returns null if no decoded buffer is available, and otherwise a DecodedTextureBuffer. | 
| -  // Throws IllegalStateException if call is made on the wrong thread, if color format changes to an | 
| -  // unsupported format, or if |mediaCodec| is not in the Executing state. Throws CodecException | 
| -  // upon codec error. If |dequeueTimeoutMs| > 0, the oldest decoded frame will be dropped if | 
| -  // a frame can't be returned. | 
| -  private DecodedTextureBuffer dequeueTextureBuffer(int dequeueTimeoutMs) { | 
| -    checkOnMediaCodecThread(); | 
| -    if (!useSurface) { | 
| -      throw new IllegalStateException("dequeueTexture() called for byte buffer decoding."); | 
| -    } | 
| -    DecodedOutputBuffer outputBuffer = dequeueOutputBuffer(dequeueTimeoutMs); | 
| -    if (outputBuffer != null) { | 
| -      dequeuedSurfaceOutputBuffers.add(outputBuffer); | 
| -    } | 
| - | 
| -    MaybeRenderDecodedTextureBuffer(); | 
| -    // Check if there is texture ready now by waiting max |dequeueTimeoutMs|. | 
| -    DecodedTextureBuffer renderedBuffer = textureListener.dequeueTextureBuffer(dequeueTimeoutMs); | 
| -    if (renderedBuffer != null) { | 
| -      MaybeRenderDecodedTextureBuffer(); | 
| -      return renderedBuffer; | 
| -    } | 
| - | 
| -    if ((dequeuedSurfaceOutputBuffers.size() | 
| -         >= Math.min(MAX_QUEUED_OUTPUTBUFFERS, outputBuffers.length) | 
| -         || (dequeueTimeoutMs > 0 && !dequeuedSurfaceOutputBuffers.isEmpty()))) { | 
| -      ++droppedFrames; | 
| -      // Drop the oldest frame still in dequeuedSurfaceOutputBuffers. | 
| -      // The oldest frame is owned by |textureListener| and can't be dropped since | 
| -      // mediaCodec.releaseOutputBuffer has already been called. | 
| -      final DecodedOutputBuffer droppedFrame = dequeuedSurfaceOutputBuffers.remove(); | 
| -      if (dequeueTimeoutMs > 0) { | 
| -        // TODO(perkj): Re-add the below log when VideoRenderGUI has been removed or fixed to | 
| -        // return the one and only texture even if it does not render. | 
| -        Logging.w(TAG, "Draining decoder. Dropping frame with TS: " | 
| -            + droppedFrame.presentationTimeStampMs + | 
| -            ". Total number of dropped frames: " + droppedFrames); | 
| -      } else { | 
| -        Logging.w(TAG, "Too many output buffers " + dequeuedSurfaceOutputBuffers.size() + | 
| -            ". Dropping frame with TS: " + droppedFrame.presentationTimeStampMs + | 
| -            ". Total number of dropped frames: " + droppedFrames); | 
| -      } | 
| - | 
| -      mediaCodec.releaseOutputBuffer(droppedFrame.index, false /* render */); | 
| -      return new DecodedTextureBuffer(0, null, | 
| -          droppedFrame.presentationTimeStampMs, droppedFrame.timeStampMs, | 
| -          droppedFrame.ntpTimeStampMs, droppedFrame.decodeTimeMs, | 
| -          SystemClock.elapsedRealtime() - droppedFrame.endDecodeTimeMs); | 
| -    } | 
| -    return null; | 
| -  } | 
| - | 
| -  private void MaybeRenderDecodedTextureBuffer() { | 
| -    if (dequeuedSurfaceOutputBuffers.isEmpty() || textureListener.isWaitingForTexture()) { | 
| -      return; | 
| -    } | 
| -    // Get the first frame in the queue and render to the decoder output surface. | 
| -    final DecodedOutputBuffer buffer = dequeuedSurfaceOutputBuffers.remove(); | 
| -    textureListener.addBufferToRender(buffer); | 
| -    mediaCodec.releaseOutputBuffer(buffer.index, true /* render */); | 
| -  } | 
| - | 
| -  // Release a dequeued output byte buffer back to the codec for re-use. Should only be called for | 
| -  // non-surface decoding. | 
| -  // Throws IllegalStateException if the call is made on the wrong thread, if codec is configured | 
| -  // for surface decoding, or if |mediaCodec| is not in the Executing state. Throws | 
| -  // MediaCodec.CodecException upon codec error. | 
| -  private void returnDecodedOutputBuffer(int index) | 
| -      throws IllegalStateException, MediaCodec.CodecException { | 
| -    checkOnMediaCodecThread(); | 
| -    if (useSurface) { | 
| -      throw new IllegalStateException("returnDecodedOutputBuffer() called for surface decoding."); | 
| -    } | 
| -    mediaCodec.releaseOutputBuffer(index, false /* render */); | 
| -  } | 
| -} | 
|  |