Move rendering code in SurfaceViewRenderer to a separate class.

The new SurfaceEglRenderer helper class extends EglRenderer and
implements rendering on a SurfaceView.

Bug: webrtc:8242
Change-Id: Ic532fe487755d3b54c6bd03f239d714e1ecb10ad
Reviewed-on: https://webrtc-review.googlesource.com/2940
Commit-Queue: Sami Kalliomäki <sakal@webrtc.org>
Reviewed-by: Sami Kalliomäki <sakal@webrtc.org>
Cr-Commit-Position: refs/heads/master@{#20562}
This commit is contained in:
Xiaolei Yu
2017-11-03 07:55:01 +08:00
committed by Commit Bot
parent ed2b1c954c
commit 149533abd4
5 changed files with 267 additions and 149 deletions

View File

@ -53,6 +53,7 @@ Vladimir Beloborodov <VladimirTechMan@gmail.com>
Vicken Simonian <vsimon@gmail.com> Vicken Simonian <vsimon@gmail.com>
Victor Costan <costan@gmail.com> Victor Costan <costan@gmail.com>
Xiaohong Xu <freemine@yeah.net> Xiaohong Xu <freemine@yeah.net>
Xiaolei Yu <dreifachstein@gmail.com>
Hans Knoechel <hans@hans-knoechel.de> Hans Knoechel <hans@hans-knoechel.de>
Korniltsev Anatoly <korniltsev.anatoly@gmail.com> Korniltsev Anatoly <korniltsev.anatoly@gmail.com>

View File

@ -438,6 +438,7 @@ android_library("libjingle_peerconnection_java") {
"api/org/webrtc/StatsObserver.java", "api/org/webrtc/StatsObserver.java",
"api/org/webrtc/StatsReport.java", "api/org/webrtc/StatsReport.java",
"api/org/webrtc/SurfaceTextureHelper.java", "api/org/webrtc/SurfaceTextureHelper.java",
"api/org/webrtc/SurfaceEglRenderer.java",
"api/org/webrtc/SurfaceViewRenderer.java", "api/org/webrtc/SurfaceViewRenderer.java",
"api/org/webrtc/TurnCustomizer.java", "api/org/webrtc/TurnCustomizer.java",
"api/org/webrtc/VideoCapturer.java", "api/org/webrtc/VideoCapturer.java",

View File

@ -76,7 +76,7 @@ public class EglRenderer implements VideoRenderer.Callbacks, VideoSink {
} }
} }
private final String name; protected final String name;
// |renderThreadHandler| is a handler for communicating with |renderThread|, and is synchronized // |renderThreadHandler| is a handler for communicating with |renderThread|, and is synchronized
// on |handlerLock|. // on |handlerLock|.

View File

@ -0,0 +1,194 @@
/*
* Copyright 2017 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.view.SurfaceHolder;
import java.util.concurrent.CountDownLatch;
/**
* Display the video stream on a Surface.
* renderFrame() is asynchronous to avoid blocking the calling thread.
* This class is thread safe and handles access from potentially three different threads:
* Interaction from the main app in init, release and setMirror.
* Interaction from C++ rtc::VideoSinkInterface in renderFrame.
* Interaction from SurfaceHolder lifecycle in surfaceCreated, surfaceChanged, and surfaceDestroyed.
*/
public class SurfaceEglRenderer extends EglRenderer implements SurfaceHolder.Callback {
private static final String TAG = "SurfaceEglRenderer";
// Callback for reporting renderer events. Read-only after initilization so no lock required.
private RendererCommon.RendererEvents rendererEvents;
private final Object layoutLock = new Object();
private boolean isRenderingPaused = false;
private boolean isFirstFrameRendered;
private int rotatedFrameWidth;
private int rotatedFrameHeight;
private int frameRotation;
/**
* In order to render something, you must first call init().
*/
public SurfaceEglRenderer(String name) {
super(name);
}
/**
* Initialize this class, sharing resources with |sharedContext|. The custom |drawer| will be used
* for drawing frames on the EGLSurface. This class is responsible for calling release() on
* |drawer|. It is allowed to call init() to reinitialize the renderer after a previous
* init()/release() cycle.
*/
public void init(final EglBase.Context sharedContext,
RendererCommon.RendererEvents rendererEvents, final int[] configAttributes,
RendererCommon.GlDrawer drawer) {
ThreadUtils.checkIsOnMainThread();
this.rendererEvents = rendererEvents;
synchronized (layoutLock) {
isFirstFrameRendered = false;
rotatedFrameWidth = 0;
rotatedFrameHeight = 0;
frameRotation = 0;
}
super.init(sharedContext, configAttributes, drawer);
}
@Override
public void init(final EglBase.Context sharedContext, final int[] configAttributes,
RendererCommon.GlDrawer drawer) {
init(sharedContext, null /* rendererEvents */, configAttributes, drawer);
}
/**
* Limit render framerate.
*
* @param fps Limit render framerate to this value, or use Float.POSITIVE_INFINITY to disable fps
* reduction.
*/
@Override
public void setFpsReduction(float fps) {
synchronized (layoutLock) {
isRenderingPaused = fps == 0f;
}
super.setFpsReduction(fps);
}
@Override
public void disableFpsReduction() {
synchronized (layoutLock) {
isRenderingPaused = false;
}
super.disableFpsReduction();
}
@Override
public void pauseVideo() {
synchronized (layoutLock) {
isRenderingPaused = true;
}
super.pauseVideo();
}
// VideoRenderer.Callbacks interface.
@Override
public void renderFrame(VideoRenderer.I420Frame frame) {
updateFrameDimensionsAndReportEvents(frame);
super.renderFrame(frame);
}
// VideoSink interface.
@Override
public void onFrame(VideoFrame frame) {
updateFrameDimensionsAndReportEvents(frame);
super.onFrame(frame);
}
// SurfaceHolder.Callback interface.
@Override
public void surfaceCreated(final SurfaceHolder holder) {
ThreadUtils.checkIsOnMainThread();
createEglSurface(holder.getSurface());
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
ThreadUtils.checkIsOnMainThread();
final CountDownLatch completionLatch = new CountDownLatch(1);
releaseEglSurface(completionLatch::countDown);
ThreadUtils.awaitUninterruptibly(completionLatch);
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
ThreadUtils.checkIsOnMainThread();
logD("surfaceChanged: format: " + format + " size: " + width + "x" + height);
}
// Update frame dimensions and report any changes to |rendererEvents|.
private void updateFrameDimensionsAndReportEvents(VideoRenderer.I420Frame frame) {
synchronized (layoutLock) {
if (isRenderingPaused) {
return;
}
if (!isFirstFrameRendered) {
isFirstFrameRendered = true;
logD("Reporting first rendered frame.");
if (rendererEvents != null) {
rendererEvents.onFirstFrameRendered();
}
}
if (rotatedFrameWidth != frame.rotatedWidth() || rotatedFrameHeight != frame.rotatedHeight()
|| frameRotation != frame.rotationDegree) {
logD("Reporting frame resolution changed to " + frame.width + "x" + frame.height
+ " with rotation " + frame.rotationDegree);
if (rendererEvents != null) {
rendererEvents.onFrameResolutionChanged(frame.width, frame.height, frame.rotationDegree);
}
rotatedFrameWidth = frame.rotatedWidth();
rotatedFrameHeight = frame.rotatedHeight();
frameRotation = frame.rotationDegree;
}
}
}
// Update frame dimensions and report any changes to |rendererEvents|.
private void updateFrameDimensionsAndReportEvents(VideoFrame frame) {
synchronized (layoutLock) {
if (isRenderingPaused) {
return;
}
if (!isFirstFrameRendered) {
isFirstFrameRendered = true;
logD("Reporting first rendered frame.");
if (rendererEvents != null) {
rendererEvents.onFirstFrameRendered();
}
}
if (rotatedFrameWidth != frame.getRotatedWidth()
|| rotatedFrameHeight != frame.getRotatedHeight()
|| frameRotation != frame.getRotation()) {
logD("Reporting frame resolution changed to " + frame.getBuffer().getWidth() + "x"
+ frame.getBuffer().getHeight() + " with rotation " + frame.getRotation());
if (rendererEvents != null) {
rendererEvents.onFrameResolutionChanged(
frame.getBuffer().getWidth(), frame.getBuffer().getHeight(), frame.getRotation());
}
rotatedFrameWidth = frame.getRotatedWidth();
rotatedFrameHeight = frame.getRotatedHeight();
frameRotation = frame.getRotation();
}
}
}
private void logD(String string) {
Logging.d(TAG, name + ": " + string);
}
}

View File

@ -13,41 +13,31 @@ package org.webrtc;
import android.content.Context; import android.content.Context;
import android.content.res.Resources.NotFoundException; import android.content.res.Resources.NotFoundException;
import android.graphics.Point; import android.graphics.Point;
import android.os.Looper;
import android.util.AttributeSet; import android.util.AttributeSet;
import android.view.SurfaceHolder; import android.view.SurfaceHolder;
import android.view.SurfaceView; import android.view.SurfaceView;
import java.util.concurrent.CountDownLatch;
/** /**
* Implements org.webrtc.VideoRenderer.Callbacks by displaying the video stream on a SurfaceView. * Display the video stream on a SurfaceView.
* renderFrame() is asynchronous to avoid blocking the calling thread.
* This class is thread safe and handles access from potentially four different threads:
* Interaction from the main app in init, release, setMirror, and setScalingtype.
* Interaction from C++ rtc::VideoSinkInterface in renderFrame.
* Interaction from the Activity lifecycle in surfaceCreated, surfaceChanged, and surfaceDestroyed.
* Interaction with the layout framework in onMeasure and onSizeChanged.
*/ */
public class SurfaceViewRenderer public class SurfaceViewRenderer extends SurfaceView implements SurfaceHolder.Callback,
extends SurfaceView implements SurfaceHolder.Callback, VideoRenderer.Callbacks, VideoSink { VideoRenderer.Callbacks, VideoSink,
RendererCommon.RendererEvents {
private static final String TAG = "SurfaceViewRenderer"; private static final String TAG = "SurfaceViewRenderer";
// Cached resource name. // Cached resource name.
private final String resourceName; private final String resourceName;
private final RendererCommon.VideoLayoutMeasure videoLayoutMeasure = private final RendererCommon.VideoLayoutMeasure videoLayoutMeasure =
new RendererCommon.VideoLayoutMeasure(); new RendererCommon.VideoLayoutMeasure();
private final EglRenderer eglRenderer; private final SurfaceEglRenderer eglRenderer;
// Callback for reporting renderer events. Read-only after initilization so no lock required. // Callback for reporting renderer events. Read-only after initilization so no lock required.
private RendererCommon.RendererEvents rendererEvents; private RendererCommon.RendererEvents rendererEvents;
private final Object layoutLock = new Object(); // Accessed only on the main thread.
private boolean isRenderingPaused = false;
private boolean isFirstFrameRendered;
private int rotatedFrameWidth; private int rotatedFrameWidth;
private int rotatedFrameHeight; private int rotatedFrameHeight;
private int frameRotation;
// Accessed only on the main thread.
private boolean enableFixedSize; private boolean enableFixedSize;
private int surfaceWidth; private int surfaceWidth;
private int surfaceHeight; private int surfaceHeight;
@ -58,8 +48,9 @@ public class SurfaceViewRenderer
public SurfaceViewRenderer(Context context) { public SurfaceViewRenderer(Context context) {
super(context); super(context);
this.resourceName = getResourceName(); this.resourceName = getResourceName();
eglRenderer = new EglRenderer(resourceName); eglRenderer = new SurfaceEglRenderer(resourceName);
getHolder().addCallback(this); getHolder().addCallback(this);
getHolder().addCallback(eglRenderer);
} }
/** /**
@ -68,8 +59,9 @@ public class SurfaceViewRenderer
public SurfaceViewRenderer(Context context, AttributeSet attrs) { public SurfaceViewRenderer(Context context, AttributeSet attrs) {
super(context, attrs); super(context, attrs);
this.resourceName = getResourceName(); this.resourceName = getResourceName();
eglRenderer = new EglRenderer(resourceName); eglRenderer = new SurfaceEglRenderer(resourceName);
getHolder().addCallback(this); getHolder().addCallback(this);
getHolder().addCallback(eglRenderer);
} }
/** /**
@ -91,13 +83,9 @@ public class SurfaceViewRenderer
RendererCommon.GlDrawer drawer) { RendererCommon.GlDrawer drawer) {
ThreadUtils.checkIsOnMainThread(); ThreadUtils.checkIsOnMainThread();
this.rendererEvents = rendererEvents; this.rendererEvents = rendererEvents;
synchronized (layoutLock) {
isFirstFrameRendered = false;
rotatedFrameWidth = 0; rotatedFrameWidth = 0;
rotatedFrameHeight = 0; rotatedFrameHeight = 0;
frameRotation = 0; eglRenderer.init(sharedContext, this /* rendererEvents */, configAttributes, drawer);
}
eglRenderer.init(sharedContext, configAttributes, drawer);
} }
/** /**
@ -181,37 +169,26 @@ public class SurfaceViewRenderer
* reduction. * reduction.
*/ */
public void setFpsReduction(float fps) { public void setFpsReduction(float fps) {
synchronized (layoutLock) {
isRenderingPaused = fps == 0f;
}
eglRenderer.setFpsReduction(fps); eglRenderer.setFpsReduction(fps);
} }
public void disableFpsReduction() { public void disableFpsReduction() {
synchronized (layoutLock) {
isRenderingPaused = false;
}
eglRenderer.disableFpsReduction(); eglRenderer.disableFpsReduction();
} }
public void pauseVideo() { public void pauseVideo() {
synchronized (layoutLock) {
isRenderingPaused = true;
}
eglRenderer.pauseVideo(); eglRenderer.pauseVideo();
} }
// VideoRenderer.Callbacks interface. // VideoRenderer.Callbacks interface.
@Override @Override
public void renderFrame(VideoRenderer.I420Frame frame) { public void renderFrame(VideoRenderer.I420Frame frame) {
updateFrameDimensionsAndReportEvents(frame);
eglRenderer.renderFrame(frame); eglRenderer.renderFrame(frame);
} }
// VideoSink interface. // VideoSink interface.
@Override @Override
public void onFrame(VideoFrame frame) { public void onFrame(VideoFrame frame) {
updateFrameDimensionsAndReportEvents(frame);
eglRenderer.onFrame(frame); eglRenderer.onFrame(frame);
} }
@ -219,11 +196,8 @@ public class SurfaceViewRenderer
@Override @Override
protected void onMeasure(int widthSpec, int heightSpec) { protected void onMeasure(int widthSpec, int heightSpec) {
ThreadUtils.checkIsOnMainThread(); ThreadUtils.checkIsOnMainThread();
final Point size; Point size =
synchronized (layoutLock) {
size =
videoLayoutMeasure.measure(widthSpec, heightSpec, rotatedFrameWidth, rotatedFrameHeight); videoLayoutMeasure.measure(widthSpec, heightSpec, rotatedFrameWidth, rotatedFrameHeight);
}
setMeasuredDimension(size.x, size.y); setMeasuredDimension(size.x, size.y);
logD("onMeasure(). New size: " + size.x + "x" + size.y); logD("onMeasure(). New size: " + size.x + "x" + size.y);
} }
@ -237,7 +211,6 @@ public class SurfaceViewRenderer
private void updateSurfaceSize() { private void updateSurfaceSize() {
ThreadUtils.checkIsOnMainThread(); ThreadUtils.checkIsOnMainThread();
synchronized (layoutLock) {
if (enableFixedSize && rotatedFrameWidth != 0 && rotatedFrameHeight != 0 && getWidth() != 0 if (enableFixedSize && rotatedFrameWidth != 0 && rotatedFrameHeight != 0 && getWidth() != 0
&& getHeight() != 0) { && getHeight() != 0) {
final float layoutAspectRatio = getWidth() / (float) getHeight(); final float layoutAspectRatio = getWidth() / (float) getHeight();
@ -267,39 +240,24 @@ public class SurfaceViewRenderer
getHolder().setSizeFromLayout(); getHolder().setSizeFromLayout();
} }
} }
}
// SurfaceHolder.Callback interface. // SurfaceHolder.Callback interface.
@Override @Override
public void surfaceCreated(final SurfaceHolder holder) { public void surfaceCreated(final SurfaceHolder holder) {
ThreadUtils.checkIsOnMainThread(); ThreadUtils.checkIsOnMainThread();
eglRenderer.createEglSurface(holder.getSurface());
surfaceWidth = surfaceHeight = 0; surfaceWidth = surfaceHeight = 0;
updateSurfaceSize(); updateSurfaceSize();
} }
@Override @Override
public void surfaceDestroyed(SurfaceHolder holder) { public void surfaceDestroyed(SurfaceHolder holder) {}
ThreadUtils.checkIsOnMainThread();
final CountDownLatch completionLatch = new CountDownLatch(1);
eglRenderer.releaseEglSurface(new Runnable() {
@Override
public void run() {
completionLatch.countDown();
}
});
ThreadUtils.awaitUninterruptibly(completionLatch);
}
@Override @Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {}
ThreadUtils.checkIsOnMainThread();
logD("surfaceChanged: format: " + format + " size: " + width + "x" + height);
}
private String getResourceName() { private String getResourceName() {
try { try {
return getResources().getResourceEntryName(getId()) + ": "; return getResources().getResourceEntryName(getId());
} catch (NotFoundException e) { } catch (NotFoundException e) {
return ""; return "";
} }
@ -312,74 +270,38 @@ public class SurfaceViewRenderer
eglRenderer.clearImage(); eglRenderer.clearImage();
} }
// Update frame dimensions and report any changes to |rendererEvents|. @Override
private void updateFrameDimensionsAndReportEvents(VideoRenderer.I420Frame frame) { public void onFirstFrameRendered() {
synchronized (layoutLock) {
if (isRenderingPaused) {
return;
}
if (!isFirstFrameRendered) {
isFirstFrameRendered = true;
logD("Reporting first rendered frame.");
if (rendererEvents != null) { if (rendererEvents != null) {
rendererEvents.onFirstFrameRendered(); rendererEvents.onFirstFrameRendered();
} }
} }
if (rotatedFrameWidth != frame.rotatedWidth() || rotatedFrameHeight != frame.rotatedHeight()
|| frameRotation != frame.rotationDegree) {
logD("Reporting frame resolution changed to " + frame.width + "x" + frame.height
+ " with rotation " + frame.rotationDegree);
if (rendererEvents != null) {
rendererEvents.onFrameResolutionChanged(frame.width, frame.height, frame.rotationDegree);
}
rotatedFrameWidth = frame.rotatedWidth();
rotatedFrameHeight = frame.rotatedHeight();
frameRotation = frame.rotationDegree;
post(new Runnable() {
@Override
public void run() {
updateSurfaceSize();
requestLayout();
}
});
}
}
}
// Update frame dimensions and report any changes to |rendererEvents|. @Override
private void updateFrameDimensionsAndReportEvents(VideoFrame frame) { public void onFrameResolutionChanged(int videoWidth, int videoHeight, int rotation) {
synchronized (layoutLock) {
if (isRenderingPaused) {
return;
}
if (!isFirstFrameRendered) {
isFirstFrameRendered = true;
logD("Reporting first rendered frame.");
if (rendererEvents != null) { if (rendererEvents != null) {
rendererEvents.onFirstFrameRendered(); rendererEvents.onFrameResolutionChanged(videoWidth, videoHeight, rotation);
} }
} int rotatedWidth = rotation == 0 || rotation == 180 ? videoWidth : videoHeight;
if (rotatedFrameWidth != frame.getRotatedWidth() int rotatedHeight = rotation == 0 || rotation == 180 ? videoHeight : videoWidth;
|| rotatedFrameHeight != frame.getRotatedHeight() // run immediately if possible for ui thread tests
|| frameRotation != frame.getRotation()) { postOrRun(() -> {
logD("Reporting frame resolution changed to " + frame.getBuffer().getWidth() + "x" rotatedFrameWidth = rotatedWidth;
+ frame.getBuffer().getHeight() + " with rotation " + frame.getRotation()); rotatedFrameHeight = rotatedHeight;
if (rendererEvents != null) {
rendererEvents.onFrameResolutionChanged(
frame.getBuffer().getWidth(), frame.getBuffer().getHeight(), frame.getRotation());
}
rotatedFrameWidth = frame.getRotatedWidth();
rotatedFrameHeight = frame.getRotatedHeight();
frameRotation = frame.getRotation();
post(() -> {
updateSurfaceSize(); updateSurfaceSize();
requestLayout(); requestLayout();
}); });
} }
private void postOrRun(Runnable r) {
if (Thread.currentThread() == Looper.getMainLooper().getThread()) {
r.run();
} else {
post(r);
} }
} }
private void logD(String string) { private void logD(String string) {
Logging.d(TAG, resourceName + string); Logging.d(TAG, resourceName + ": " + string);
} }
} }