blob: 7f7d5383b48ce67ba6d9a5e9a64d2c4d4d9eadbe [file] [log] [blame]
Bjorn Mellemb080b462017-06-20 17:02:361/*
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
11package org.webrtc;
12
13import android.annotation.TargetApi;
Bjorn Mellemb080b462017-06-20 17:02:3614import android.media.MediaCodec;
15import android.media.MediaCodecInfo.CodecCapabilities;
16import android.media.MediaFormat;
17import android.os.SystemClock;
Bjorn Mellem8fb23612017-07-18 18:33:3918import android.view.Surface;
Bjorn Mellemb080b462017-06-20 17:02:3619import java.io.IOException;
20import java.nio.ByteBuffer;
sakal6c36dcb2017-08-24 13:03:5921import java.util.concurrent.BlockingDeque;
Bjorn Mellemb080b462017-06-20 17:02:3622import java.util.concurrent.LinkedBlockingDeque;
sakale172d892017-08-31 09:37:2823import java.util.concurrent.TimeUnit;
Bjorn Mellemb080b462017-06-20 17:02:3624import 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 Mellem8fb23612017-07-18 18:33:3929class HardwareVideoDecoder
30 implements VideoDecoder, SurfaceTextureHelper.OnTextureFrameAvailableListener {
Bjorn Mellemb080b462017-06-20 17:02:3631 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 Mellemb080b462017-06-20 17:02:3641 // 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äki3aa3ea72017-06-26 12:30:4645 // 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 Mellemb080b462017-06-20 17:02:3649 // 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 Mellem852a5602017-06-23 17:01:5657
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
sakal6c36dcb2017-08-24 13:03:5968 private final BlockingDeque<FrameInfo> frameInfos;
Bjorn Mellemb080b462017-06-20 17:02:3669 private int colorFormat;
70
71 // Output thread runs a loop which polls MediaCodec for decoded output buffers. It reformats
sakal6c36dcb2017-08-24 13:03:5972 // 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 Mellemb080b462017-06-20 17:02:3674 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äki8d08a922017-06-28 09:20:5086 // 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 Mellemb080b462017-06-20 17:02:3690 // 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
sakal6c36dcb2017-08-24 13:03:5999 // after delivering the first frame. Only accessed on the output thread while the decoder is
100 // running.
Bjorn Mellemb080b462017-06-20 17:02:36101 private boolean hasDecodedFirstFrame;
sakal6c36dcb2017-08-24 13:03:59102 // Whether the decoder has seen a key frame. The first frame must be a key frame. Only accessed
103 // on the decoder thread.
Bjorn Mellemb080b462017-06-20 17:02:36104 private boolean keyFrameRequired;
105
Bjorn Mellem8fb23612017-07-18 18:33:39106 private final EglBase.Context sharedContext;
sakal6c36dcb2017-08-24 13:03:59107 // Valid and immutable while the decoder is running.
Bjorn Mellem8fb23612017-07-18 18:33:39108 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
sakal6c36dcb2017-08-24 13:03:59132 // Decoding proceeds asynchronously. This callback returns decoded frames to the caller. Valid
133 // and immutable while the decoder is running.
Bjorn Mellemb080b462017-06-20 17:02:36134 private Callback callback;
135
sakal6c36dcb2017-08-24 13:03:59136 // Valid and immutable while the decoder is running.
Bjorn Mellemb080b462017-06-20 17:02:36137 private MediaCodec codec = null;
138
Bjorn Mellem8fb23612017-07-18 18:33:39139 HardwareVideoDecoder(
140 String codecName, VideoCodecType codecType, int colorFormat, EglBase.Context sharedContext) {
Bjorn Mellemb080b462017-06-20 17:02:36141 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 Mellem8fb23612017-07-18 18:33:39147 this.sharedContext = sharedContext;
Bjorn Mellem852a5602017-06-23 17:01:56148 this.frameInfos = new LinkedBlockingDeque<>();
Bjorn Mellemb080b462017-06-20 17:02:36149 }
150
151 @Override
152 public VideoCodecStatus initDecode(Settings settings, Callback callback) {
153 this.decoderThreadChecker = new ThreadChecker();
sakal6c36dcb2017-08-24 13:03:59154
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 Mellemb080b462017-06-20 17:02:36162 }
163
sakal6c36dcb2017-08-24 13:03:59164 // Internal variant is used when restarting the codec due to reconfiguration.
165 private VideoCodecStatus initDecodeInternal(int width, int height) {
Bjorn Mellemb080b462017-06-20 17:02:36166 decoderThreadChecker.checkIsOnValidThread();
sakal6c36dcb2017-08-24 13:03:59167 Logging.d(TAG, "initDecodeInternal");
Bjorn Mellemb080b462017-06-20 17:02:36168 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 Mellemb080b462017-06-20 17:02:36175 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 Mellem38145242017-06-20 18:35:04186 Logging.e(TAG, "Cannot create media decoder " + codecName);
Bjorn Mellemb080b462017-06-20 17:02:36187 return VideoCodecStatus.ERROR;
188 }
189 try {
190 MediaFormat format = MediaFormat.createVideoFormat(codecType.mimeType(), width, height);
Bjorn Mellem8fb23612017-07-18 18:33:39191 if (sharedContext == null) {
192 format.setInteger(MediaFormat.KEY_COLOR_FORMAT, colorFormat);
Bjorn Mellem8fb23612017-07-18 18:33:39193 }
194 codec.configure(format, surface, null, 0);
Bjorn Mellemb080b462017-06-20 17:02:36195 codec.start();
196 } catch (IllegalStateException e) {
197 Logging.e(TAG, "initDecode failed", e);
198 release();
199 return VideoCodecStatus.ERROR;
200 }
Bjorn Mellemb080b462017-06-20 17:02:36201 running = true;
Bjorn Mellemb080b462017-06-20 17:02:36202 outputThread = createOutputThread();
203 outputThread.start();
204
sakal6c36dcb2017-08-24 13:03:59205 Logging.d(TAG, "initDecodeInternal done");
Bjorn Mellemb080b462017-06-20 17:02:36206 return VideoCodecStatus.OK;
207 }
208
209 @Override
210 public VideoCodecStatus decode(EncodedImage frame, DecodeInfo info) {
211 decoderThreadChecker.checkIsOnValidThread();
212 if (codec == null || callback == null) {
sakal6c36dcb2017-08-24 13:03:59213 Logging.d(TAG, "decode uninitalized, codec: " + codec + ", callback: " + callback);
Bjorn Mellemb080b462017-06-20 17:02:36214 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 Mellem852a5602017-06-23 17:01:56224 Logging.e(TAG, "decode() - input buffer empty");
225 return VideoCodecStatus.ERR_PARAMETER;
Bjorn Mellemb080b462017-06-20 17:02:36226 }
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 Mellem852a5602017-06-23 17:01:56240 return status;
Bjorn Mellemb080b462017-06-20 17:02:36241 }
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 Mellemb080b462017-06-20 17:02:36254 }
255
Bjorn Mellemb080b462017-06-20 17:02:36256 int index;
257 try {
Sami Kalliomäki3aa3ea72017-06-26 12:30:46258 index = codec.dequeueInputBuffer(DEQUEUE_INPUT_TIMEOUT_US);
Bjorn Mellemb080b462017-06-20 17:02:36259 } catch (IllegalStateException e) {
260 Logging.e(TAG, "dequeueInputBuffer failed", e);
Bjorn Mellem852a5602017-06-23 17:01:56261 return VideoCodecStatus.ERROR;
Bjorn Mellemb080b462017-06-20 17:02:36262 }
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 Mellem852a5602017-06-23 17:01:56267 return VideoCodecStatus.ERROR;
Bjorn Mellemb080b462017-06-20 17:02:36268 }
269
270 ByteBuffer buffer;
271 try {
272 buffer = codec.getInputBuffers()[index];
273 } catch (IllegalStateException e) {
274 Logging.e(TAG, "getInputBuffers failed", e);
Bjorn Mellem852a5602017-06-23 17:01:56275 return VideoCodecStatus.ERROR;
Bjorn Mellemb080b462017-06-20 17:02:36276 }
277
278 if (buffer.capacity() < size) {
279 Logging.e(TAG, "decode() - HW buffer too small");
Bjorn Mellem852a5602017-06-23 17:01:56280 return VideoCodecStatus.ERROR;
Bjorn Mellemb080b462017-06-20 17:02:36281 }
282 buffer.put(frame.buffer);
283
Bjorn Mellem852a5602017-06-23 17:01:56284 frameInfos.offer(new FrameInfo(SystemClock.elapsedRealtime(), frame.rotation));
Bjorn Mellemb080b462017-06-20 17:02:36285 try {
sakale172d892017-08-31 09:37:28286 codec.queueInputBuffer(index, 0 /* offset */, size,
287 TimeUnit.NANOSECONDS.toMicros(frame.captureTimeNs), 0 /* flags */);
Bjorn Mellemb080b462017-06-20 17:02:36288 } catch (IllegalStateException e) {
289 Logging.e(TAG, "queueInputBuffer failed", e);
Bjorn Mellem852a5602017-06-23 17:01:56290 frameInfos.pollLast();
291 return VideoCodecStatus.ERROR;
Bjorn Mellemb080b462017-06-20 17:02:36292 }
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äki93ad1f72017-06-27 08:03:49311 // 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();
sakal6c36dcb2017-08-24 13:03:59314 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() {
sakal84778c72017-08-18 10:03:29330 if (!running) {
331 Logging.d(TAG, "release: Decoder is not running.");
332 return VideoCodecStatus.OK;
333 }
Bjorn Mellemb080b462017-06-20 17:02:36334 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.
sakal84778c72017-08-18 10:03:29339 Logging.e(TAG, "Media decoder release timeout", new RuntimeException());
Bjorn Mellemb080b462017-06-20 17:02:36340 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.
sakal84778c72017-08-18 10:03:29345 Logging.e(TAG, "Media decoder release error", new RuntimeException(shutdownException));
Bjorn Mellemb080b462017-06-20 17:02:36346 shutdownException = null;
347 return VideoCodecStatus.ERROR;
348 }
349 } finally {
350 codec = null;
Bjorn Mellemb080b462017-06-20 17:02:36351 outputThread = null;
Bjorn Mellemb080b462017-06-20 17:02:36352 }
353 return VideoCodecStatus.OK;
354 }
355
356 private VideoCodecStatus reinitDecode(int newWidth, int newHeight) {
357 decoderThreadChecker.checkIsOnValidThread();
sakal6c36dcb2017-08-24 13:03:59358 VideoCodecStatus status = releaseInternal();
Bjorn Mellemb080b462017-06-20 17:02:36359 if (status != VideoCodecStatus.OK) {
360 return status;
361 }
sakal6c36dcb2017-08-24 13:03:59362 return initDecodeInternal(newWidth, newHeight);
Bjorn Mellemb080b462017-06-20 17:02:36363 }
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 Mellemb080b462017-06-20 17:02:36387 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 Mellem852a5602017-06-23 17:01:56397 FrameInfo frameInfo = frameInfos.poll();
Bjorn Mellemb080b462017-06-20 17:02:36398 Integer decodeTimeMs = null;
Bjorn Mellem852a5602017-06-23 17:01:56399 int rotation = 0;
400 if (frameInfo != null) {
401 decodeTimeMs = (int) (SystemClock.elapsedRealtime() - frameInfo.decodeStartTimeMs);
402 rotation = frameInfo.rotation;
Bjorn Mellemb080b462017-06-20 17:02:36403 }
404
405 hasDecodedFirstFrame = true;
406
Bjorn Mellem8fb23612017-07-18 18:33:39407 if (surfaceTextureHelper != null) {
408 deliverTextureFrame(result, info, rotation, decodeTimeMs);
Bjorn Mellemb080b462017-06-20 17:02:36409 } else {
Bjorn Mellem8fb23612017-07-18 18:33:39410 deliverByteFrame(result, info, rotation, decodeTimeMs);
Bjorn Mellemb080b462017-06-20 17:02:36411 }
Bjorn Mellemb080b462017-06-20 17:02:36412
Bjorn Mellemb080b462017-06-20 17:02:36413 } catch (IllegalStateException e) {
414 Logging.e(TAG, "deliverDecodedFrame failed", e);
415 }
416 }
417
Bjorn Mellem8fb23612017-07-18 18:33:39418 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(
sakal836f60c2017-07-28 14:12:23440 renderedTextureMetadata.width, renderedTextureMetadata.height,
441 RendererCommon.convertMatrixToAndroidGraphicsMatrix(transformMatrix));
Bjorn Mellem8fb23612017-07-18 18:33:39442
443 VideoFrame frame = new VideoFrame(oesBuffer, renderedTextureMetadata.rotation,
sakal836f60c2017-07-28 14:12:23444 renderedTextureMetadata.presentationTimestampUs * 1000);
Bjorn Mellem8fb23612017-07-18 18:33:39445 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);
sakal84778c72017-08-18 10:03:29476 buffer.limit(info.offset + info.size);
477 buffer = buffer.slice();
478 final VideoFrame.Buffer frameBuffer;
Bjorn Mellem8fb23612017-07-18 18:33:39479
Bjorn Mellem8fb23612017-07-18 18:33:39480 if (colorFormat == CodecCapabilities.COLOR_FormatYUV420Planar) {
481 if (sliceHeight % 2 == 0) {
sakal84778c72017-08-18 10:03:29482 frameBuffer = wrapI420Buffer(buffer, result, stride, sliceHeight, width, height);
Bjorn Mellem8fb23612017-07-18 18:33:39483 } else {
sakal84778c72017-08-18 10:03:29484 // 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 Mellem8fb23612017-07-18 18:33:39486 }
487 } else {
Bjorn Mellem8fb23612017-07-18 18:33:39488 // All other supported color formats are NV12.
sakal84778c72017-08-18 10:03:29489 frameBuffer = wrapNV12Buffer(buffer, result, stride, sliceHeight, width, height);
Bjorn Mellem8fb23612017-07-18 18:33:39490 }
491
492 long presentationTimeNs = info.presentationTimeUs * 1000;
sakal836f60c2017-07-28 14:12:23493 VideoFrame frame = new VideoFrame(frameBuffer, rotation, presentationTimeNs);
Bjorn Mellem8fb23612017-07-18 18:33:39494
495 // Note that qp is parsed on the C++ side.
496 callback.onDecodedFrame(frame, decodeTimeMs, null /* qp */);
497 frame.release();
498 }
499
sakal84778c72017-08-18 10:03:29500 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 Mellemb080b462017-06-20 17:02:36599 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 Mellem8fb23612017-07-18 18:33:39627 // 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 Mellemb080b462017-06-20 17:02:36630 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äki8d08a922017-06-28 09:20:50655 waitOutputBuffersReleasedOnOutputThread();
Bjorn Mellemb080b462017-06-20 17:02:36656 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 Mellemb080b462017-06-20 17:02:36668 Logging.d(TAG, "Release on output thread done");
669 }
670
Sami Kalliomäki8d08a922017-06-28 09:20:50671 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 Mellemb080b462017-06-20 17:02:36686 private void stopOnOutputThread(Exception e) {
687 outputThreadChecker.checkIsOnValidThread();
688 running = false;
689 shutdownException = e;
690 }
691
692 private boolean isSupportedColorFormat(int colorFormat) {
Bjorn Mellem38145242017-06-20 18:35:04693 for (int supported : MediaCodecUtils.DECODER_COLOR_FORMATS) {
694 if (supported == colorFormat) {
Bjorn Mellemb080b462017-06-20 17:02:36695 return true;
Bjorn Mellem38145242017-06-20 18:35:04696 }
Bjorn Mellemb080b462017-06-20 17:02:36697 }
Bjorn Mellem38145242017-06-20 18:35:04698 return false;
Bjorn Mellemb080b462017-06-20 17:02:36699 }
Bjorn Mellemb080b462017-06-20 17:02:36700}