Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2017 The WebRTC project authors. All Rights Reserved. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
| 9 | */ |
| 10 | |
| 11 | package org.webrtc; |
| 12 | |
| 13 | import android.annotation.TargetApi; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 14 | import android.media.MediaCodec; |
| 15 | import android.media.MediaCodecInfo.CodecCapabilities; |
| 16 | import android.media.MediaFormat; |
| 17 | import android.os.SystemClock; |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 18 | import android.view.Surface; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 19 | import java.io.IOException; |
| 20 | import java.nio.ByteBuffer; |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 21 | import java.util.concurrent.BlockingDeque; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 22 | import java.util.concurrent.LinkedBlockingDeque; |
sakal | e172d89 | 2017-08-31 09:37:28 | [diff] [blame] | 23 | import java.util.concurrent.TimeUnit; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 24 | import org.webrtc.ThreadUtils.ThreadChecker; |
| 25 | |
| 26 | /** Android hardware video decoder. */ |
| 27 | @TargetApi(16) |
| 28 | @SuppressWarnings("deprecation") // Cannot support API 16 without using deprecated methods. |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 29 | class HardwareVideoDecoder |
| 30 | implements VideoDecoder, SurfaceTextureHelper.OnTextureFrameAvailableListener { |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 31 | private static final String TAG = "HardwareVideoDecoder"; |
| 32 | |
| 33 | // TODO(magjed): Use MediaFormat.KEY_* constants when part of the public API. |
| 34 | private static final String MEDIA_FORMAT_KEY_STRIDE = "stride"; |
| 35 | private static final String MEDIA_FORMAT_KEY_SLICE_HEIGHT = "slice-height"; |
| 36 | private static final String MEDIA_FORMAT_KEY_CROP_LEFT = "crop-left"; |
| 37 | private static final String MEDIA_FORMAT_KEY_CROP_RIGHT = "crop-right"; |
| 38 | private static final String MEDIA_FORMAT_KEY_CROP_TOP = "crop-top"; |
| 39 | private static final String MEDIA_FORMAT_KEY_CROP_BOTTOM = "crop-bottom"; |
| 40 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 41 | // MediaCodec.release() occasionally hangs. Release stops waiting and reports failure after |
| 42 | // this timeout. |
| 43 | private static final int MEDIA_CODEC_RELEASE_TIMEOUT_MS = 5000; |
| 44 | |
Sami Kalliomäki | 3aa3ea7 | 2017-06-26 12:30:46 | [diff] [blame] | 45 | // WebRTC queues input frames quickly in the beginning on the call. Wait for input buffers with a |
| 46 | // long timeout (500 ms) to prevent this from causing the codec to return an error. |
| 47 | private static final int DEQUEUE_INPUT_TIMEOUT_US = 500000; |
| 48 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 49 | // Dequeuing an output buffer will block until a buffer is available (up to 100 milliseconds). |
| 50 | // If this timeout is exceeded, the output thread will unblock and check if the decoder is still |
| 51 | // running. If it is, it will block on dequeue again. Otherwise, it will stop and release the |
| 52 | // MediaCodec. |
| 53 | private static final int DEQUEUE_OUTPUT_BUFFER_TIMEOUT_US = 100000; |
| 54 | |
| 55 | private final String codecName; |
| 56 | private final VideoCodecType codecType; |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 57 | |
| 58 | private static class FrameInfo { |
| 59 | final long decodeStartTimeMs; |
| 60 | final int rotation; |
| 61 | |
| 62 | FrameInfo(long decodeStartTimeMs, int rotation) { |
| 63 | this.decodeStartTimeMs = decodeStartTimeMs; |
| 64 | this.rotation = rotation; |
| 65 | } |
| 66 | } |
| 67 | |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 68 | private final BlockingDeque<FrameInfo> frameInfos; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 69 | private int colorFormat; |
| 70 | |
| 71 | // Output thread runs a loop which polls MediaCodec for decoded output buffers. It reformats |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 72 | // those buffers into VideoFrames and delivers them to the callback. Variable is set on decoder |
| 73 | // thread and is immutable while the codec is running. |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 74 | private Thread outputThread; |
| 75 | |
| 76 | // Checker that ensures work is run on the output thread. |
| 77 | private ThreadChecker outputThreadChecker; |
| 78 | |
| 79 | // Checker that ensures work is run on the decoder thread. The decoder thread is owned by the |
| 80 | // caller and must be used to call initDecode, decode, and release. |
| 81 | private ThreadChecker decoderThreadChecker; |
| 82 | |
| 83 | private volatile boolean running = false; |
| 84 | private volatile Exception shutdownException = null; |
| 85 | |
Sami Kalliomäki | 8d08a92 | 2017-06-28 09:20:50 | [diff] [blame] | 86 | // Prevents the decoder from being released before all output buffers have been released. |
| 87 | private final Object activeOutputBuffersLock = new Object(); |
| 88 | private int activeOutputBuffers = 0; // Guarded by activeOutputBuffersLock |
| 89 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 90 | // Dimensions (width, height, stride, and sliceHeight) may be accessed by either the decode thread |
| 91 | // or the output thread. Accesses should be protected with this lock. |
| 92 | private final Object dimensionLock = new Object(); |
| 93 | private int width; |
| 94 | private int height; |
| 95 | private int stride; |
| 96 | private int sliceHeight; |
| 97 | |
| 98 | // Whether the decoder has finished the first frame. The codec may not change output dimensions |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 99 | // after delivering the first frame. Only accessed on the output thread while the decoder is |
| 100 | // running. |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 101 | private boolean hasDecodedFirstFrame; |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 102 | // Whether the decoder has seen a key frame. The first frame must be a key frame. Only accessed |
| 103 | // on the decoder thread. |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 104 | private boolean keyFrameRequired; |
| 105 | |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 106 | private final EglBase.Context sharedContext; |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 107 | // Valid and immutable while the decoder is running. |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 108 | private SurfaceTextureHelper surfaceTextureHelper; |
| 109 | private Surface surface = null; |
| 110 | |
| 111 | private static class DecodedTextureMetadata { |
| 112 | final int width; |
| 113 | final int height; |
| 114 | final int rotation; |
| 115 | final long presentationTimestampUs; |
| 116 | final Integer decodeTimeMs; |
| 117 | |
| 118 | DecodedTextureMetadata( |
| 119 | int width, int height, int rotation, long presentationTimestampUs, Integer decodeTimeMs) { |
| 120 | this.width = width; |
| 121 | this.height = height; |
| 122 | this.rotation = rotation; |
| 123 | this.presentationTimestampUs = presentationTimestampUs; |
| 124 | this.decodeTimeMs = decodeTimeMs; |
| 125 | } |
| 126 | } |
| 127 | |
| 128 | // Metadata for the last frame rendered to the texture. Only accessed on the texture helper's |
| 129 | // thread. |
| 130 | private DecodedTextureMetadata renderedTextureMetadata; |
| 131 | |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 132 | // Decoding proceeds asynchronously. This callback returns decoded frames to the caller. Valid |
| 133 | // and immutable while the decoder is running. |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 134 | private Callback callback; |
| 135 | |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 136 | // Valid and immutable while the decoder is running. |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 137 | private MediaCodec codec = null; |
| 138 | |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 139 | HardwareVideoDecoder( |
| 140 | String codecName, VideoCodecType codecType, int colorFormat, EglBase.Context sharedContext) { |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 141 | if (!isSupportedColorFormat(colorFormat)) { |
| 142 | throw new IllegalArgumentException("Unsupported color format: " + colorFormat); |
| 143 | } |
| 144 | this.codecName = codecName; |
| 145 | this.codecType = codecType; |
| 146 | this.colorFormat = colorFormat; |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 147 | this.sharedContext = sharedContext; |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 148 | this.frameInfos = new LinkedBlockingDeque<>(); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 149 | } |
| 150 | |
| 151 | @Override |
| 152 | public VideoCodecStatus initDecode(Settings settings, Callback callback) { |
| 153 | this.decoderThreadChecker = new ThreadChecker(); |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 154 | |
| 155 | this.callback = callback; |
| 156 | if (sharedContext != null) { |
| 157 | surfaceTextureHelper = SurfaceTextureHelper.create("decoder-texture-thread", sharedContext); |
| 158 | surface = new Surface(surfaceTextureHelper.getSurfaceTexture()); |
| 159 | surfaceTextureHelper.startListening(this); |
| 160 | } |
| 161 | return initDecodeInternal(settings.width, settings.height); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 162 | } |
| 163 | |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 164 | // Internal variant is used when restarting the codec due to reconfiguration. |
| 165 | private VideoCodecStatus initDecodeInternal(int width, int height) { |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 166 | decoderThreadChecker.checkIsOnValidThread(); |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 167 | Logging.d(TAG, "initDecodeInternal"); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 168 | if (outputThread != null) { |
| 169 | Logging.e(TAG, "initDecodeInternal called while the codec is already running"); |
| 170 | return VideoCodecStatus.ERROR; |
| 171 | } |
| 172 | |
| 173 | // Note: it is not necessary to initialize dimensions under the lock, since the output thread |
| 174 | // is not running. |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 175 | this.width = width; |
| 176 | this.height = height; |
| 177 | |
| 178 | stride = width; |
| 179 | sliceHeight = height; |
| 180 | hasDecodedFirstFrame = false; |
| 181 | keyFrameRequired = true; |
| 182 | |
| 183 | try { |
| 184 | codec = MediaCodec.createByCodecName(codecName); |
| 185 | } catch (IOException | IllegalArgumentException e) { |
Bjorn Mellem | 3814524 | 2017-06-20 18:35:04 | [diff] [blame] | 186 | Logging.e(TAG, "Cannot create media decoder " + codecName); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 187 | return VideoCodecStatus.ERROR; |
| 188 | } |
| 189 | try { |
| 190 | MediaFormat format = MediaFormat.createVideoFormat(codecType.mimeType(), width, height); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 191 | if (sharedContext == null) { |
| 192 | format.setInteger(MediaFormat.KEY_COLOR_FORMAT, colorFormat); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 193 | } |
| 194 | codec.configure(format, surface, null, 0); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 195 | codec.start(); |
| 196 | } catch (IllegalStateException e) { |
| 197 | Logging.e(TAG, "initDecode failed", e); |
| 198 | release(); |
| 199 | return VideoCodecStatus.ERROR; |
| 200 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 201 | running = true; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 202 | outputThread = createOutputThread(); |
| 203 | outputThread.start(); |
| 204 | |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 205 | Logging.d(TAG, "initDecodeInternal done"); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 206 | return VideoCodecStatus.OK; |
| 207 | } |
| 208 | |
| 209 | @Override |
| 210 | public VideoCodecStatus decode(EncodedImage frame, DecodeInfo info) { |
| 211 | decoderThreadChecker.checkIsOnValidThread(); |
| 212 | if (codec == null || callback == null) { |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 213 | Logging.d(TAG, "decode uninitalized, codec: " + codec + ", callback: " + callback); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 214 | return VideoCodecStatus.UNINITIALIZED; |
| 215 | } |
| 216 | |
| 217 | if (frame.buffer == null) { |
| 218 | Logging.e(TAG, "decode() - no input data"); |
| 219 | return VideoCodecStatus.ERR_PARAMETER; |
| 220 | } |
| 221 | |
| 222 | int size = frame.buffer.remaining(); |
| 223 | if (size == 0) { |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 224 | Logging.e(TAG, "decode() - input buffer empty"); |
| 225 | return VideoCodecStatus.ERR_PARAMETER; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 226 | } |
| 227 | |
| 228 | // Load dimensions from shared memory under the dimension lock. |
| 229 | int width, height; |
| 230 | synchronized (dimensionLock) { |
| 231 | width = this.width; |
| 232 | height = this.height; |
| 233 | } |
| 234 | |
| 235 | // Check if the resolution changed and reset the codec if necessary. |
| 236 | if (frame.encodedWidth * frame.encodedHeight > 0 |
| 237 | && (frame.encodedWidth != width || frame.encodedHeight != height)) { |
| 238 | VideoCodecStatus status = reinitDecode(frame.encodedWidth, frame.encodedHeight); |
| 239 | if (status != VideoCodecStatus.OK) { |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 240 | return status; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 241 | } |
| 242 | } |
| 243 | |
| 244 | if (keyFrameRequired) { |
| 245 | // Need to process a key frame first. |
| 246 | if (frame.frameType != EncodedImage.FrameType.VideoFrameKey) { |
| 247 | Logging.e(TAG, "decode() - key frame required first"); |
| 248 | return VideoCodecStatus.ERROR; |
| 249 | } |
| 250 | if (!frame.completeFrame) { |
| 251 | Logging.e(TAG, "decode() - complete frame required first"); |
| 252 | return VideoCodecStatus.ERROR; |
| 253 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 254 | } |
| 255 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 256 | int index; |
| 257 | try { |
Sami Kalliomäki | 3aa3ea7 | 2017-06-26 12:30:46 | [diff] [blame] | 258 | index = codec.dequeueInputBuffer(DEQUEUE_INPUT_TIMEOUT_US); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 259 | } catch (IllegalStateException e) { |
| 260 | Logging.e(TAG, "dequeueInputBuffer failed", e); |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 261 | return VideoCodecStatus.ERROR; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 262 | } |
| 263 | if (index < 0) { |
| 264 | // Decoder is falling behind. No input buffers available. |
| 265 | // The decoder can't simply drop frames; it might lose a key frame. |
| 266 | Logging.e(TAG, "decode() - no HW buffers available; decoder falling behind"); |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 267 | return VideoCodecStatus.ERROR; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 268 | } |
| 269 | |
| 270 | ByteBuffer buffer; |
| 271 | try { |
| 272 | buffer = codec.getInputBuffers()[index]; |
| 273 | } catch (IllegalStateException e) { |
| 274 | Logging.e(TAG, "getInputBuffers failed", e); |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 275 | return VideoCodecStatus.ERROR; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 276 | } |
| 277 | |
| 278 | if (buffer.capacity() < size) { |
| 279 | Logging.e(TAG, "decode() - HW buffer too small"); |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 280 | return VideoCodecStatus.ERROR; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 281 | } |
| 282 | buffer.put(frame.buffer); |
| 283 | |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 284 | frameInfos.offer(new FrameInfo(SystemClock.elapsedRealtime(), frame.rotation)); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 285 | try { |
sakal | e172d89 | 2017-08-31 09:37:28 | [diff] [blame] | 286 | codec.queueInputBuffer(index, 0 /* offset */, size, |
| 287 | TimeUnit.NANOSECONDS.toMicros(frame.captureTimeNs), 0 /* flags */); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 288 | } catch (IllegalStateException e) { |
| 289 | Logging.e(TAG, "queueInputBuffer failed", e); |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 290 | frameInfos.pollLast(); |
| 291 | return VideoCodecStatus.ERROR; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 292 | } |
| 293 | if (keyFrameRequired) { |
| 294 | keyFrameRequired = false; |
| 295 | } |
| 296 | return VideoCodecStatus.OK; |
| 297 | } |
| 298 | |
| 299 | @Override |
| 300 | public boolean getPrefersLateDecoding() { |
| 301 | return true; |
| 302 | } |
| 303 | |
| 304 | @Override |
| 305 | public String getImplementationName() { |
| 306 | return "HardwareVideoDecoder: " + codecName; |
| 307 | } |
| 308 | |
| 309 | @Override |
| 310 | public VideoCodecStatus release() { |
Sami Kalliomäki | 93ad1f7 | 2017-06-27 08:03:49 | [diff] [blame] | 311 | // TODO(sakal): This is not called on the correct thread but is still called synchronously. |
| 312 | // Re-enable the check once this is called on the correct thread. |
| 313 | // decoderThreadChecker.checkIsOnValidThread(); |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 314 | Logging.d(TAG, "release"); |
| 315 | VideoCodecStatus status = releaseInternal(); |
| 316 | if (surface != null) { |
| 317 | surface.release(); |
| 318 | surface = null; |
| 319 | surfaceTextureHelper.stopListening(); |
| 320 | surfaceTextureHelper.dispose(); |
| 321 | surfaceTextureHelper = null; |
| 322 | } |
| 323 | callback = null; |
| 324 | frameInfos.clear(); |
| 325 | return status; |
| 326 | } |
| 327 | |
| 328 | // Internal variant is used when restarting the codec due to reconfiguration. |
| 329 | private VideoCodecStatus releaseInternal() { |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 330 | if (!running) { |
| 331 | Logging.d(TAG, "release: Decoder is not running."); |
| 332 | return VideoCodecStatus.OK; |
| 333 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 334 | try { |
| 335 | // The outputThread actually stops and releases the codec once running is false. |
| 336 | running = false; |
| 337 | if (!ThreadUtils.joinUninterruptibly(outputThread, MEDIA_CODEC_RELEASE_TIMEOUT_MS)) { |
| 338 | // Log an exception to capture the stack trace and turn it into a TIMEOUT error. |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 339 | Logging.e(TAG, "Media decoder release timeout", new RuntimeException()); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 340 | return VideoCodecStatus.TIMEOUT; |
| 341 | } |
| 342 | if (shutdownException != null) { |
| 343 | // Log the exception and turn it into an error. Wrap the exception in a new exception to |
| 344 | // capture both the output thread's stack trace and this thread's stack trace. |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 345 | Logging.e(TAG, "Media decoder release error", new RuntimeException(shutdownException)); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 346 | shutdownException = null; |
| 347 | return VideoCodecStatus.ERROR; |
| 348 | } |
| 349 | } finally { |
| 350 | codec = null; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 351 | outputThread = null; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 352 | } |
| 353 | return VideoCodecStatus.OK; |
| 354 | } |
| 355 | |
| 356 | private VideoCodecStatus reinitDecode(int newWidth, int newHeight) { |
| 357 | decoderThreadChecker.checkIsOnValidThread(); |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 358 | VideoCodecStatus status = releaseInternal(); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 359 | if (status != VideoCodecStatus.OK) { |
| 360 | return status; |
| 361 | } |
sakal | 6c36dcb | 2017-08-24 13:03:59 | [diff] [blame] | 362 | return initDecodeInternal(newWidth, newHeight); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 363 | } |
| 364 | |
| 365 | private Thread createOutputThread() { |
| 366 | return new Thread("HardwareVideoDecoder.outputThread") { |
| 367 | @Override |
| 368 | public void run() { |
| 369 | outputThreadChecker = new ThreadChecker(); |
| 370 | while (running) { |
| 371 | deliverDecodedFrame(); |
| 372 | } |
| 373 | releaseCodecOnOutputThread(); |
| 374 | } |
| 375 | }; |
| 376 | } |
| 377 | |
| 378 | private void deliverDecodedFrame() { |
| 379 | outputThreadChecker.checkIsOnValidThread(); |
| 380 | try { |
| 381 | MediaCodec.BufferInfo info = new MediaCodec.BufferInfo(); |
| 382 | // Block until an output buffer is available (up to 100 milliseconds). If the timeout is |
| 383 | // exceeded, deliverDecodedFrame() will be called again on the next iteration of the output |
| 384 | // thread's loop. Blocking here prevents the output thread from busy-waiting while the codec |
| 385 | // is idle. |
| 386 | int result = codec.dequeueOutputBuffer(info, DEQUEUE_OUTPUT_BUFFER_TIMEOUT_US); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 387 | if (result == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED) { |
| 388 | reformat(codec.getOutputFormat()); |
| 389 | return; |
| 390 | } |
| 391 | |
| 392 | if (result < 0) { |
| 393 | Logging.v(TAG, "dequeueOutputBuffer returned " + result); |
| 394 | return; |
| 395 | } |
| 396 | |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 397 | FrameInfo frameInfo = frameInfos.poll(); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 398 | Integer decodeTimeMs = null; |
Bjorn Mellem | 852a560 | 2017-06-23 17:01:56 | [diff] [blame] | 399 | int rotation = 0; |
| 400 | if (frameInfo != null) { |
| 401 | decodeTimeMs = (int) (SystemClock.elapsedRealtime() - frameInfo.decodeStartTimeMs); |
| 402 | rotation = frameInfo.rotation; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 403 | } |
| 404 | |
| 405 | hasDecodedFirstFrame = true; |
| 406 | |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 407 | if (surfaceTextureHelper != null) { |
| 408 | deliverTextureFrame(result, info, rotation, decodeTimeMs); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 409 | } else { |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 410 | deliverByteFrame(result, info, rotation, decodeTimeMs); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 411 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 412 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 413 | } catch (IllegalStateException e) { |
| 414 | Logging.e(TAG, "deliverDecodedFrame failed", e); |
| 415 | } |
| 416 | } |
| 417 | |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 418 | private void deliverTextureFrame(final int index, final MediaCodec.BufferInfo info, |
| 419 | final int rotation, final Integer decodeTimeMs) { |
| 420 | // Load dimensions from shared memory under the dimension lock. |
| 421 | final int width, height; |
| 422 | synchronized (dimensionLock) { |
| 423 | width = this.width; |
| 424 | height = this.height; |
| 425 | } |
| 426 | |
| 427 | surfaceTextureHelper.getHandler().post(new Runnable() { |
| 428 | @Override |
| 429 | public void run() { |
| 430 | renderedTextureMetadata = new DecodedTextureMetadata( |
| 431 | width, height, rotation, info.presentationTimeUs, decodeTimeMs); |
| 432 | codec.releaseOutputBuffer(index, true); |
| 433 | } |
| 434 | }); |
| 435 | } |
| 436 | |
| 437 | @Override |
| 438 | public void onTextureFrameAvailable(int oesTextureId, float[] transformMatrix, long timestampNs) { |
| 439 | VideoFrame.TextureBuffer oesBuffer = surfaceTextureHelper.createTextureBuffer( |
sakal | 836f60c | 2017-07-28 14:12:23 | [diff] [blame] | 440 | renderedTextureMetadata.width, renderedTextureMetadata.height, |
| 441 | RendererCommon.convertMatrixToAndroidGraphicsMatrix(transformMatrix)); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 442 | |
| 443 | VideoFrame frame = new VideoFrame(oesBuffer, renderedTextureMetadata.rotation, |
sakal | 836f60c | 2017-07-28 14:12:23 | [diff] [blame] | 444 | renderedTextureMetadata.presentationTimestampUs * 1000); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 445 | callback.onDecodedFrame(frame, renderedTextureMetadata.decodeTimeMs, null /* qp */); |
| 446 | frame.release(); |
| 447 | } |
| 448 | |
| 449 | private void deliverByteFrame( |
| 450 | int result, MediaCodec.BufferInfo info, int rotation, Integer decodeTimeMs) { |
| 451 | // Load dimensions from shared memory under the dimension lock. |
| 452 | int width, height, stride, sliceHeight; |
| 453 | synchronized (dimensionLock) { |
| 454 | width = this.width; |
| 455 | height = this.height; |
| 456 | stride = this.stride; |
| 457 | sliceHeight = this.sliceHeight; |
| 458 | } |
| 459 | |
| 460 | // Output must be at least width * height bytes for Y channel, plus (width / 2) * (height / 2) |
| 461 | // bytes for each of the U and V channels. |
| 462 | if (info.size < width * height * 3 / 2) { |
| 463 | Logging.e(TAG, "Insufficient output buffer size: " + info.size); |
| 464 | return; |
| 465 | } |
| 466 | |
| 467 | if (info.size < stride * height * 3 / 2 && sliceHeight == height && stride > width) { |
| 468 | // Some codecs (Exynos) report an incorrect stride. Correct it here. |
| 469 | // Expected size == stride * height * 3 / 2. A bit of algebra gives the correct stride as |
| 470 | // 2 * size / (3 * height). |
| 471 | stride = info.size * 2 / (height * 3); |
| 472 | } |
| 473 | |
| 474 | ByteBuffer buffer = codec.getOutputBuffers()[result]; |
| 475 | buffer.position(info.offset); |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 476 | buffer.limit(info.offset + info.size); |
| 477 | buffer = buffer.slice(); |
| 478 | final VideoFrame.Buffer frameBuffer; |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 479 | |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 480 | if (colorFormat == CodecCapabilities.COLOR_FormatYUV420Planar) { |
| 481 | if (sliceHeight % 2 == 0) { |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 482 | frameBuffer = wrapI420Buffer(buffer, result, stride, sliceHeight, width, height); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 483 | } else { |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 484 | // WebRTC rounds chroma plane size conversions up so we have to repeat the last row. |
| 485 | frameBuffer = copyI420Buffer(buffer, result, stride, sliceHeight, width, height); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 486 | } |
| 487 | } else { |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 488 | // All other supported color formats are NV12. |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 489 | frameBuffer = wrapNV12Buffer(buffer, result, stride, sliceHeight, width, height); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 490 | } |
| 491 | |
| 492 | long presentationTimeNs = info.presentationTimeUs * 1000; |
sakal | 836f60c | 2017-07-28 14:12:23 | [diff] [blame] | 493 | VideoFrame frame = new VideoFrame(frameBuffer, rotation, presentationTimeNs); |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 494 | |
| 495 | // Note that qp is parsed on the C++ side. |
| 496 | callback.onDecodedFrame(frame, decodeTimeMs, null /* qp */); |
| 497 | frame.release(); |
| 498 | } |
| 499 | |
sakal | 84778c7 | 2017-08-18 10:03:29 | [diff] [blame] | 500 | private VideoFrame.Buffer wrapNV12Buffer(ByteBuffer buffer, int outputBufferIndex, int stride, |
| 501 | int sliceHeight, int width, int height) { |
| 502 | synchronized (activeOutputBuffersLock) { |
| 503 | activeOutputBuffers++; |
| 504 | } |
| 505 | |
| 506 | return new NV12Buffer(width, height, stride, sliceHeight, buffer, () -> { |
| 507 | codec.releaseOutputBuffer(outputBufferIndex, false); |
| 508 | synchronized (activeOutputBuffersLock) { |
| 509 | activeOutputBuffers--; |
| 510 | activeOutputBuffersLock.notifyAll(); |
| 511 | } |
| 512 | }); |
| 513 | } |
| 514 | |
| 515 | private VideoFrame.Buffer copyI420Buffer(ByteBuffer buffer, int outputBufferIndex, int stride, |
| 516 | int sliceHeight, int width, int height) { |
| 517 | final int uvStride = stride / 2; |
| 518 | |
| 519 | final int yPos = 0; |
| 520 | final int uPos = yPos + stride * sliceHeight; |
| 521 | final int uEnd = uPos + uvStride * (sliceHeight / 2); |
| 522 | final int vPos = uPos + uvStride * sliceHeight / 2; |
| 523 | final int vEnd = vPos + uvStride * (sliceHeight / 2); |
| 524 | |
| 525 | VideoFrame.I420Buffer frameBuffer = I420BufferImpl.allocate(width, height); |
| 526 | |
| 527 | ByteBuffer dataY = frameBuffer.getDataY(); |
| 528 | dataY.position(0); // Ensure we are in the beginning. |
| 529 | buffer.position(yPos); |
| 530 | buffer.limit(uPos); |
| 531 | dataY.put(buffer); |
| 532 | dataY.position(0); // Go back to beginning. |
| 533 | |
| 534 | ByteBuffer dataU = frameBuffer.getDataU(); |
| 535 | dataU.position(0); // Ensure we are in the beginning. |
| 536 | buffer.position(uPos); |
| 537 | buffer.limit(uEnd); |
| 538 | dataU.put(buffer); |
| 539 | if (sliceHeight % 2 != 0) { |
| 540 | buffer.position(uEnd - uvStride); // Repeat the last row. |
| 541 | dataU.put(buffer); |
| 542 | } |
| 543 | dataU.position(0); // Go back to beginning. |
| 544 | |
| 545 | ByteBuffer dataV = frameBuffer.getDataU(); |
| 546 | dataV.position(0); // Ensure we are in the beginning. |
| 547 | buffer.position(vPos); |
| 548 | buffer.limit(vEnd); |
| 549 | dataV.put(buffer); |
| 550 | if (sliceHeight % 2 != 0) { |
| 551 | buffer.position(vEnd - uvStride); // Repeat the last row. |
| 552 | dataV.put(buffer); |
| 553 | } |
| 554 | dataV.position(0); // Go back to beginning. |
| 555 | |
| 556 | codec.releaseOutputBuffer(outputBufferIndex, false); |
| 557 | |
| 558 | return frameBuffer; |
| 559 | } |
| 560 | |
| 561 | private VideoFrame.Buffer wrapI420Buffer(ByteBuffer buffer, int outputBufferIndex, int stride, |
| 562 | int sliceHeight, int width, int height) { |
| 563 | final int uvStride = stride / 2; |
| 564 | |
| 565 | final int yPos = 0; |
| 566 | final int uPos = yPos + stride * sliceHeight; |
| 567 | final int uEnd = uPos + uvStride * (sliceHeight / 2); |
| 568 | final int vPos = uPos + uvStride * sliceHeight / 2; |
| 569 | final int vEnd = vPos + uvStride * (sliceHeight / 2); |
| 570 | |
| 571 | synchronized (activeOutputBuffersLock) { |
| 572 | activeOutputBuffers++; |
| 573 | } |
| 574 | |
| 575 | Runnable releaseCallback = () -> { |
| 576 | codec.releaseOutputBuffer(outputBufferIndex, false); |
| 577 | synchronized (activeOutputBuffersLock) { |
| 578 | activeOutputBuffers--; |
| 579 | activeOutputBuffersLock.notifyAll(); |
| 580 | } |
| 581 | }; |
| 582 | |
| 583 | buffer.position(yPos); |
| 584 | buffer.limit(uPos); |
| 585 | ByteBuffer dataY = buffer.slice(); |
| 586 | |
| 587 | buffer.position(uPos); |
| 588 | buffer.limit(uEnd); |
| 589 | ByteBuffer dataU = buffer.slice(); |
| 590 | |
| 591 | buffer.position(vPos); |
| 592 | buffer.limit(vEnd); |
| 593 | ByteBuffer dataV = buffer.slice(); |
| 594 | |
| 595 | return new I420BufferImpl( |
| 596 | width, height, dataY, stride, dataU, uvStride, dataV, uvStride, releaseCallback); |
| 597 | } |
| 598 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 599 | private void reformat(MediaFormat format) { |
| 600 | outputThreadChecker.checkIsOnValidThread(); |
| 601 | Logging.d(TAG, "Decoder format changed: " + format.toString()); |
| 602 | final int newWidth; |
| 603 | final int newHeight; |
| 604 | if (format.containsKey(MEDIA_FORMAT_KEY_CROP_LEFT) |
| 605 | && format.containsKey(MEDIA_FORMAT_KEY_CROP_RIGHT) |
| 606 | && format.containsKey(MEDIA_FORMAT_KEY_CROP_BOTTOM) |
| 607 | && format.containsKey(MEDIA_FORMAT_KEY_CROP_TOP)) { |
| 608 | newWidth = 1 + format.getInteger(MEDIA_FORMAT_KEY_CROP_RIGHT) |
| 609 | - format.getInteger(MEDIA_FORMAT_KEY_CROP_LEFT); |
| 610 | newHeight = 1 + format.getInteger(MEDIA_FORMAT_KEY_CROP_BOTTOM) |
| 611 | - format.getInteger(MEDIA_FORMAT_KEY_CROP_TOP); |
| 612 | } else { |
| 613 | newWidth = format.getInteger(MediaFormat.KEY_WIDTH); |
| 614 | newHeight = format.getInteger(MediaFormat.KEY_HEIGHT); |
| 615 | } |
| 616 | // Compare to existing width, height, and save values under the dimension lock. |
| 617 | synchronized (dimensionLock) { |
| 618 | if (hasDecodedFirstFrame && (width != newWidth || height != newHeight)) { |
| 619 | stopOnOutputThread(new RuntimeException("Unexpected size change. Configured " + width + "*" |
| 620 | + height + ". New " + newWidth + "*" + newHeight)); |
| 621 | return; |
| 622 | } |
| 623 | width = newWidth; |
| 624 | height = newHeight; |
| 625 | } |
| 626 | |
Bjorn Mellem | 8fb2361 | 2017-07-18 18:33:39 | [diff] [blame] | 627 | // Note: texture mode ignores colorFormat. Hence, if the texture helper is non-null, skip |
| 628 | // color format updates. |
| 629 | if (surfaceTextureHelper == null && format.containsKey(MediaFormat.KEY_COLOR_FORMAT)) { |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 630 | colorFormat = format.getInteger(MediaFormat.KEY_COLOR_FORMAT); |
| 631 | Logging.d(TAG, "Color: 0x" + Integer.toHexString(colorFormat)); |
| 632 | if (!isSupportedColorFormat(colorFormat)) { |
| 633 | stopOnOutputThread(new IllegalStateException("Unsupported color format: " + colorFormat)); |
| 634 | return; |
| 635 | } |
| 636 | } |
| 637 | |
| 638 | // Save stride and sliceHeight under the dimension lock. |
| 639 | synchronized (dimensionLock) { |
| 640 | if (format.containsKey(MEDIA_FORMAT_KEY_STRIDE)) { |
| 641 | stride = format.getInteger(MEDIA_FORMAT_KEY_STRIDE); |
| 642 | } |
| 643 | if (format.containsKey(MEDIA_FORMAT_KEY_SLICE_HEIGHT)) { |
| 644 | sliceHeight = format.getInteger(MEDIA_FORMAT_KEY_SLICE_HEIGHT); |
| 645 | } |
| 646 | Logging.d(TAG, "Frame stride and slice height: " + stride + " x " + sliceHeight); |
| 647 | stride = Math.max(width, stride); |
| 648 | sliceHeight = Math.max(height, sliceHeight); |
| 649 | } |
| 650 | } |
| 651 | |
| 652 | private void releaseCodecOnOutputThread() { |
| 653 | outputThreadChecker.checkIsOnValidThread(); |
| 654 | Logging.d(TAG, "Releasing MediaCodec on output thread"); |
Sami Kalliomäki | 8d08a92 | 2017-06-28 09:20:50 | [diff] [blame] | 655 | waitOutputBuffersReleasedOnOutputThread(); |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 656 | try { |
| 657 | codec.stop(); |
| 658 | } catch (Exception e) { |
| 659 | Logging.e(TAG, "Media decoder stop failed", e); |
| 660 | } |
| 661 | try { |
| 662 | codec.release(); |
| 663 | } catch (Exception e) { |
| 664 | Logging.e(TAG, "Media decoder release failed", e); |
| 665 | // Propagate exceptions caught during release back to the main thread. |
| 666 | shutdownException = e; |
| 667 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 668 | Logging.d(TAG, "Release on output thread done"); |
| 669 | } |
| 670 | |
Sami Kalliomäki | 8d08a92 | 2017-06-28 09:20:50 | [diff] [blame] | 671 | private void waitOutputBuffersReleasedOnOutputThread() { |
| 672 | outputThreadChecker.checkIsOnValidThread(); |
| 673 | synchronized (activeOutputBuffersLock) { |
| 674 | while (activeOutputBuffers > 0) { |
| 675 | Logging.d(TAG, "Waiting for all frames to be released."); |
| 676 | try { |
| 677 | activeOutputBuffersLock.wait(); |
| 678 | } catch (InterruptedException e) { |
| 679 | Logging.e(TAG, "Interrupted while waiting for output buffers to be released.", e); |
| 680 | return; |
| 681 | } |
| 682 | } |
| 683 | } |
| 684 | } |
| 685 | |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 686 | private void stopOnOutputThread(Exception e) { |
| 687 | outputThreadChecker.checkIsOnValidThread(); |
| 688 | running = false; |
| 689 | shutdownException = e; |
| 690 | } |
| 691 | |
| 692 | private boolean isSupportedColorFormat(int colorFormat) { |
Bjorn Mellem | 3814524 | 2017-06-20 18:35:04 | [diff] [blame] | 693 | for (int supported : MediaCodecUtils.DECODER_COLOR_FORMATS) { |
| 694 | if (supported == colorFormat) { |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 695 | return true; |
Bjorn Mellem | 3814524 | 2017-06-20 18:35:04 | [diff] [blame] | 696 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 697 | } |
Bjorn Mellem | 3814524 | 2017-06-20 18:35:04 | [diff] [blame] | 698 | return false; |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 699 | } |
Bjorn Mellem | b080b46 | 2017-06-20 17:02:36 | [diff] [blame] | 700 | } |