1084 lines
35 KiB
C++
1084 lines
35 KiB
C++
|
/*
|
||
|
* Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
|
||
|
*
|
||
|
* Use of this source code is governed by a BSD-style license
|
||
|
* that can be found in the LICENSE file in the root of the source
|
||
|
* tree. An additional intellectual property rights grant can be found
|
||
|
* in the file PATENTS. All contributing project authors may
|
||
|
* be found in the AUTHORS file in the root of the source tree.
|
||
|
*/
|
||
|
|
||
|
#include "video_render_impl.h"
|
||
|
#include "engine_configurations.h"
|
||
|
#include "critical_section_wrapper.h"
|
||
|
#include "video_render_defines.h"
|
||
|
#include "trace.h"
|
||
|
#include "incoming_video_stream.h"
|
||
|
#include "i_video_render.h"
|
||
|
|
||
|
#include <cassert>
|
||
|
|
||
|
#ifndef WEBRTC_VIDEO_EXTERNAL_CAPTURE_AND_RENDER
|
||
|
|
||
|
#if defined (_WIN32)
|
||
|
#include "windows/video_render_windows_impl.h"
|
||
|
#define STANDARD_RENDERING kRenderWindows
|
||
|
|
||
|
#elif defined(MAC_IPHONE) // MAC_IPHONE should go before WEBRTC_MAC_INTEL because WEBRTC_MAC_INTEL gets defined if MAC_IPHONE is defined
|
||
|
#if defined(IPHONE_GLES_RENDERING)
|
||
|
#define STANDARD_RENDERING kRenderiPhone
|
||
|
#include "iPhone/video_render_iphone_impl.h"
|
||
|
#endif
|
||
|
|
||
|
#elif defined(WEBRTC_MAC) || defined(WEBRTC_MAC_INTEL)
|
||
|
#if defined(COCOA_RENDERING)
|
||
|
#define STANDARD_RENDERING kRenderCocoa
|
||
|
#include "mac/video_render_mac_cocoa_impl.h"
|
||
|
#elif defined(CARBON_RENDERING)
|
||
|
#define STANDARD_RENDERING kRenderCarbon
|
||
|
#include "mac/video_render_mac_carbon_impl.h"
|
||
|
#endif
|
||
|
|
||
|
#elif defined(ANDROID)
|
||
|
#include "Android/video_render_android_impl.h"
|
||
|
#include "Android/video_render_android_surface_view.h"
|
||
|
#include "Android/video_render_android_native_opengl2.h"
|
||
|
#define STANDARD_RENDERING kRenderAndroid
|
||
|
|
||
|
#elif defined(WEBRTC_LINUX)
|
||
|
#include "linux/video_render_linux_impl.h"
|
||
|
#define STANDARD_RENDERING kRenderX11
|
||
|
|
||
|
#else
|
||
|
//Other platforms
|
||
|
#endif
|
||
|
|
||
|
#endif // WEBRTC_VIDEO_EXTERNAL_CAPTURE_AND_RENDER
|
||
|
|
||
|
// For external rendering
|
||
|
#include "external/video_render_external_impl.h"
|
||
|
#ifndef STANDARD_RENDERING
|
||
|
#define STANDARD_RENDERING kRenderExternal
|
||
|
#endif // STANDARD_RENDERING
|
||
|
|
||
|
namespace webrtc {
|
||
|
|
||
|
VideoRender*
|
||
|
VideoRender::CreateVideoRender(const WebRtc_Word32 id,
|
||
|
void* window,
|
||
|
const bool fullscreen,
|
||
|
const VideoRenderType videoRenderType/*=kRenderDefault*/)
|
||
|
{
|
||
|
WEBRTC_TRACE(
|
||
|
kTraceModuleCall,
|
||
|
kTraceVideoRenderer,
|
||
|
id,
|
||
|
"CreateVideoRender(videoRenderType: %d, window: %x, fullscreen: %d)",
|
||
|
videoRenderType, window, fullscreen);
|
||
|
|
||
|
VideoRenderType resultVideoRenderType = videoRenderType;
|
||
|
if (videoRenderType == kRenderDefault)
|
||
|
{
|
||
|
resultVideoRenderType = STANDARD_RENDERING;
|
||
|
}
|
||
|
return new ModuleVideoRenderImpl(id, resultVideoRenderType, window,
|
||
|
fullscreen);
|
||
|
}
|
||
|
|
||
|
void VideoRender::DestroyVideoRender(
|
||
|
VideoRender* module)
|
||
|
{
|
||
|
if (module)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
static_cast<ModuleVideoRenderImpl*> (module)->Id(),
|
||
|
"DestroyVideoRender");
|
||
|
delete module;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 VideoRender::SetAndroidObjects(void *javaVM)
|
||
|
{
|
||
|
#ifdef ANDROID
|
||
|
return VideoRenderAndroid::SetAndroidEnvVariables(javaVM);
|
||
|
#else
|
||
|
return -1;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
ModuleVideoRenderImpl::ModuleVideoRenderImpl(
|
||
|
const WebRtc_Word32 id,
|
||
|
const VideoRenderType videoRenderType,
|
||
|
void* window,
|
||
|
const bool fullscreen) :
|
||
|
_id(id), _moduleCrit(*CriticalSectionWrapper::CreateCriticalSection()),
|
||
|
_ptrWindow(window), _renderType(videoRenderType),
|
||
|
_fullScreen(fullscreen), _ptrRenderer(NULL),
|
||
|
_streamRenderMap(*(new MapWrapper()))
|
||
|
{
|
||
|
|
||
|
// Create platform specific renderer
|
||
|
switch (videoRenderType)
|
||
|
{
|
||
|
#ifndef WEBRTC_VIDEO_EXTERNAL_CAPTURE_AND_RENDER
|
||
|
|
||
|
#if defined(_WIN32)
|
||
|
case kRenderWindows:
|
||
|
{
|
||
|
VideoRenderWindowsImpl* ptrRenderer;
|
||
|
ptrRenderer = new VideoRenderWindowsImpl(_id, videoRenderType, window, _fullScreen);
|
||
|
if (ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*>(ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
#elif defined(MAC_IPHONE)
|
||
|
case kRenderiPhone:
|
||
|
{
|
||
|
VideoRenderIPhoneImpl* ptrRenderer = new VideoRenderIPhoneImpl(_id, videoRenderType, window, _fullScreen);
|
||
|
if(ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*>(ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
#elif defined(WEBRTC_MAC) || defined(WEBRTC_MAC_INTEL)
|
||
|
|
||
|
#if defined(COCOA_RENDERING)
|
||
|
case kRenderCocoa:
|
||
|
{
|
||
|
VideoRenderMacCocoaImpl* ptrRenderer = new VideoRenderMacCocoaImpl(_id, videoRenderType, window, _fullScreen);
|
||
|
if(ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*>(ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
#elif defined(CARBON_RENDERING)
|
||
|
case kRenderCarbon:
|
||
|
{
|
||
|
VideoRenderMacCarbonImpl* ptrRenderer = new VideoRenderMacCarbonImpl(_id, videoRenderType, window, _fullScreen);
|
||
|
if(ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*>(ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
#endif
|
||
|
|
||
|
#elif defined(ANDROID)
|
||
|
case kRenderAndroid:
|
||
|
{
|
||
|
if(AndroidNativeOpenGl2Renderer::UseOpenGL2(window))
|
||
|
{
|
||
|
AndroidNativeOpenGl2Renderer* ptrRenderer = NULL;
|
||
|
ptrRenderer = new AndroidNativeOpenGl2Renderer(_id, videoRenderType, window, _fullScreen);
|
||
|
if (ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*> (ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
AndroidSurfaceViewRenderer* ptrRenderer = NULL;
|
||
|
ptrRenderer = new AndroidSurfaceViewRenderer(_id, videoRenderType, window, _fullScreen);
|
||
|
if (ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*> (ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
break;
|
||
|
#elif defined(WEBRTC_LINUX)
|
||
|
case kRenderX11:
|
||
|
{
|
||
|
VideoRenderLinuxImpl* ptrRenderer = NULL;
|
||
|
ptrRenderer = new VideoRenderLinuxImpl(_id, videoRenderType, window, _fullScreen);
|
||
|
if ( ptrRenderer )
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*> (ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
#else
|
||
|
// Other platforms
|
||
|
#endif
|
||
|
|
||
|
#endif // WEBRTC_VIDEO_EXTERNAL_CAPTURE_AND_RENDER
|
||
|
case kRenderExternal:
|
||
|
{
|
||
|
VideoRenderExternalImpl* ptrRenderer(NULL);
|
||
|
ptrRenderer = new VideoRenderExternalImpl(_id, videoRenderType,
|
||
|
window, _fullScreen);
|
||
|
if (ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*> (ptrRenderer);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
// Error...
|
||
|
break;
|
||
|
}
|
||
|
if (_ptrRenderer)
|
||
|
{
|
||
|
if (_ptrRenderer->Init() == -1)
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ModuleVideoRenderImpl::~ModuleVideoRenderImpl()
|
||
|
{
|
||
|
delete &_moduleCrit;
|
||
|
|
||
|
while (_streamRenderMap.Size() > 0)
|
||
|
{
|
||
|
MapItem* item = _streamRenderMap.First();
|
||
|
IncomingVideoStream* ptrIncomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
assert(ptrIncomingStream != NULL);
|
||
|
delete ptrIncomingStream;
|
||
|
_streamRenderMap.Erase(item);
|
||
|
}
|
||
|
delete &_streamRenderMap;
|
||
|
|
||
|
// Delete platform specific renderer
|
||
|
if (_ptrRenderer)
|
||
|
{
|
||
|
VideoRenderType videoRenderType = _ptrRenderer->RenderType();
|
||
|
switch (videoRenderType)
|
||
|
{
|
||
|
case kRenderExternal:
|
||
|
{
|
||
|
VideoRenderExternalImpl
|
||
|
* ptrRenderer =
|
||
|
reinterpret_cast<VideoRenderExternalImpl*> (_ptrRenderer);
|
||
|
_ptrRenderer = NULL;
|
||
|
delete ptrRenderer;
|
||
|
}
|
||
|
break;
|
||
|
#ifndef WEBRTC_VIDEO_EXTERNAL_CAPTURE_AND_RENDER
|
||
|
|
||
|
#if defined(_WIN32)
|
||
|
case kRenderWindows:
|
||
|
{
|
||
|
VideoRenderWindowsImpl* ptrRenderer = reinterpret_cast<VideoRenderWindowsImpl*>(_ptrRenderer);
|
||
|
_ptrRenderer = NULL;
|
||
|
delete ptrRenderer;
|
||
|
}
|
||
|
break;
|
||
|
#elif defined(WEBRTC_MAC) || defined(WEBRTC_MAC_INTEL)
|
||
|
|
||
|
#if defined(COCOA_RENDERING)
|
||
|
case kRenderCocoa:
|
||
|
{
|
||
|
VideoRenderMacCocoaImpl* ptrRenderer = reinterpret_cast<VideoRenderMacCocoaImpl*> (_ptrRenderer);
|
||
|
_ptrRenderer = NULL;
|
||
|
delete ptrRenderer;
|
||
|
}
|
||
|
break;
|
||
|
#elif defined(CARBON_RENDERING)
|
||
|
case kRenderCarbon:
|
||
|
{
|
||
|
VideoRenderMacCarbonImpl* ptrRenderer = reinterpret_cast<VideoRenderMacCarbonImpl*> (_ptrRenderer);
|
||
|
_ptrRenderer = NULL;
|
||
|
delete ptrRenderer;
|
||
|
}
|
||
|
break;
|
||
|
#endif
|
||
|
|
||
|
#elif defined(MAC_IPHONE)
|
||
|
case kRenderiPhone:
|
||
|
break;
|
||
|
|
||
|
#elif defined(ANDROID)
|
||
|
case kRenderAndroid:
|
||
|
{
|
||
|
VideoRenderAndroid* ptrRenderer = reinterpret_cast<VideoRenderAndroid*> (_ptrRenderer);
|
||
|
_ptrRenderer = NULL;
|
||
|
delete ptrRenderer;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
#elif defined(WEBRTC_LINUX)
|
||
|
case kRenderX11:
|
||
|
{
|
||
|
VideoRenderLinuxImpl* ptrRenderer = reinterpret_cast<VideoRenderLinuxImpl*> (_ptrRenderer);
|
||
|
_ptrRenderer = NULL;
|
||
|
delete ptrRenderer;
|
||
|
}
|
||
|
break;
|
||
|
#else
|
||
|
//other platforms
|
||
|
#endif
|
||
|
|
||
|
#endif // WEBRTC_VIDEO_EXTERNAL_CAPTURE_AND_RENDER
|
||
|
|
||
|
default:
|
||
|
// Error...
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::Version(
|
||
|
WebRtc_Word8* version,
|
||
|
WebRtc_UWord32& remainingBufferInBytes,
|
||
|
WebRtc_UWord32& position) const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
if (version == NULL)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"Version pointer is NULL");
|
||
|
return -1;
|
||
|
}
|
||
|
WebRtc_Word8 ourVersion[256] = "VideoRender 1.1.0";
|
||
|
WebRtc_Word32 ourLength = (WebRtc_Word32) strlen(ourVersion);
|
||
|
if ((WebRtc_Word32) remainingBufferInBytes < ourLength + 1)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceWarning, kTraceVideoRenderer, _id,
|
||
|
"Version buffer not long enough");
|
||
|
return -1;
|
||
|
}
|
||
|
memcpy(version, ourVersion, ourLength);
|
||
|
version[ourLength] = 0; // null terminaion
|
||
|
position += ourLength;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::ChangeUniqueId(const WebRtc_Word32 id)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"ChangeUniqueId(new id:%d)", id);
|
||
|
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
_id = id;
|
||
|
|
||
|
if (_ptrRenderer)
|
||
|
{
|
||
|
_ptrRenderer->ChangeUniqueId(_id);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::TimeUntilNextProcess()
|
||
|
{
|
||
|
// Not used
|
||
|
return 50;
|
||
|
}
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::Process()
|
||
|
{
|
||
|
// Not used
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void*
|
||
|
ModuleVideoRenderImpl::Window()
|
||
|
{
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
return _ptrWindow;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::ChangeWindow(void* window)
|
||
|
{
|
||
|
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
|
||
|
#if defined(MAC_IPHONE) // MAC_IPHONE must go before WEBRTC_MAC or WEBRTC_MAC_INTEL
|
||
|
_ptrRenderer = NULL;
|
||
|
delete _ptrRenderer;
|
||
|
|
||
|
VideoRenderIPhoneImpl* ptrRenderer;
|
||
|
ptrRenderer = new VideoRenderIPhoneImpl(_id, kRenderiPhone, window, _fullScreen);
|
||
|
if (!ptrRenderer)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*>(ptrRenderer);
|
||
|
return _ptrRenderer->ChangeWindow(window);
|
||
|
|
||
|
#elif defined(WEBRTC_MAC) | defined(WEBRTC_MAC_INTEL)
|
||
|
|
||
|
_ptrRenderer = NULL;
|
||
|
delete _ptrRenderer;
|
||
|
|
||
|
#if defined(COCOA_RENDERING)
|
||
|
VideoRenderMacCocoaImpl* ptrRenderer;
|
||
|
ptrRenderer = new VideoRenderMacCocoaImpl(_id, kRenderCocoa, window, _fullScreen);
|
||
|
#elif defined(CARBON_RENDERING)
|
||
|
VideoRenderMacCarbonImpl* ptrRenderer;
|
||
|
ptrRenderer = new VideoRenderMacCarbonImpl(_id, kRenderCarbon, window, _fullScreen);
|
||
|
#endif
|
||
|
if (!ptrRenderer)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
_ptrRenderer = reinterpret_cast<IVideoRender*>(ptrRenderer);
|
||
|
return _ptrRenderer->ChangeWindow(window);
|
||
|
|
||
|
#else
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
return _ptrRenderer->ChangeWindow(window);
|
||
|
|
||
|
#endif
|
||
|
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::Id()
|
||
|
{
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
return _id;
|
||
|
}
|
||
|
|
||
|
WebRtc_UWord32 ModuleVideoRenderImpl::GetIncomingFrameRate(
|
||
|
const WebRtc_UWord32 streamId)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, stream: %u", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
MapItem* mapItem = _streamRenderMap.Find(streamId);
|
||
|
if (mapItem == NULL)
|
||
|
{
|
||
|
// This stream doesn't exist
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return 0;
|
||
|
}
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (mapItem->GetItem());
|
||
|
if (incomingStream == NULL)
|
||
|
{
|
||
|
// This should never happen
|
||
|
assert(false);
|
||
|
_streamRenderMap.Erase(mapItem);
|
||
|
return 0;
|
||
|
}
|
||
|
return incomingStream->IncomingRate();
|
||
|
}
|
||
|
|
||
|
VideoRenderCallback*
|
||
|
ModuleVideoRenderImpl::AddIncomingRenderStream(const WebRtc_UWord32 streamId,
|
||
|
const WebRtc_UWord32 zOrder,
|
||
|
const float left,
|
||
|
const float top,
|
||
|
const float right,
|
||
|
const float bottom)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, stream: %u", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
if (_streamRenderMap.Find(streamId) != NULL)
|
||
|
{
|
||
|
// The stream already exists...
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream already exists", __FUNCTION__);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
// Create platform independant code
|
||
|
IncomingVideoStream* ptrIncomingStream = new IncomingVideoStream(_id,
|
||
|
streamId);
|
||
|
if (ptrIncomingStream == NULL)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: Can't create incoming stream", __FUNCTION__);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
VideoRenderCallback* ptrRenderCallback =
|
||
|
_ptrRenderer->AddIncomingRenderStream(streamId, zOrder, left, top,
|
||
|
right, bottom);
|
||
|
if (ptrRenderCallback == NULL)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: Can't create incoming stream in renderer",
|
||
|
__FUNCTION__);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
if (ptrIncomingStream->SetRenderCallback(ptrRenderCallback) == -1)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: Can't set render callback", __FUNCTION__);
|
||
|
delete ptrIncomingStream;
|
||
|
_ptrRenderer->DeleteIncomingRenderStream(streamId);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
VideoRenderCallback* moduleCallback =
|
||
|
ptrIncomingStream->ModuleCallback();
|
||
|
|
||
|
// Store the stream
|
||
|
_streamRenderMap.Insert(streamId, ptrIncomingStream);
|
||
|
|
||
|
return moduleCallback;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::DeleteIncomingRenderStream(
|
||
|
const WebRtc_UWord32 streamId)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, stream: %u", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
MapItem* mapItem = _streamRenderMap.Find(streamId);
|
||
|
if (!mapItem)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
IncomingVideoStream* ptrIncomingStream =
|
||
|
static_cast<IncomingVideoStream*> (mapItem->GetItem());
|
||
|
delete ptrIncomingStream;
|
||
|
ptrIncomingStream = NULL;
|
||
|
_ptrRenderer->DeleteIncomingRenderStream(streamId);
|
||
|
_streamRenderMap.Erase(mapItem);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::AddExternalRenderCallback(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
VideoRenderCallback* renderObject)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, stream: %u, callback: %x", __FUNCTION__, streamId,
|
||
|
renderObject);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
MapItem* mapItem = _streamRenderMap.Find(streamId);
|
||
|
if (!mapItem)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
IncomingVideoStream* ptrIncomingStream =
|
||
|
static_cast<IncomingVideoStream*> (mapItem->GetItem());
|
||
|
if (!ptrIncomingStream)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: could not get stream", __FUNCTION__);
|
||
|
}
|
||
|
return ptrIncomingStream->SetExternalCallback(renderObject);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::GetIncomingRenderStreamProperties(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
WebRtc_UWord32& zOrder,
|
||
|
float& left,
|
||
|
float& top,
|
||
|
float& right,
|
||
|
float& bottom) const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, stream: %u", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return _ptrRenderer->GetIncomingRenderStreamProperties(streamId, zOrder,
|
||
|
left, top, right,
|
||
|
bottom);
|
||
|
}
|
||
|
|
||
|
WebRtc_UWord32 ModuleVideoRenderImpl::GetNumIncomingRenderStreams() const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
return (WebRtc_UWord32) _streamRenderMap.Size();
|
||
|
}
|
||
|
|
||
|
bool ModuleVideoRenderImpl::HasIncomingRenderStream(
|
||
|
const WebRtc_UWord32 streamId) const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
bool hasStream = false;
|
||
|
if (_streamRenderMap.Find(streamId) != NULL)
|
||
|
{
|
||
|
hasStream = true;
|
||
|
}
|
||
|
return hasStream;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::RegisterRawFrameCallback(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
VideoRenderCallback* callbackObj)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::StartRender(const WebRtc_UWord32 streamId)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s(%u)", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
// Start the stream
|
||
|
MapItem* item = _streamRenderMap.Find(streamId);
|
||
|
if (item == NULL)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
_id, "%s: Could find render stream %d", __FUNCTION__,
|
||
|
streamId);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream->Start() == -1)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
_id, "%s: Could not start stream %d", __FUNCTION__,
|
||
|
incomingStream->StreamId());
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
// Start the HW renderer
|
||
|
if (_ptrRenderer->StartRender() == -1)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
_id, "%s: Could not start renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::StopRender(const WebRtc_UWord32 streamId)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s(%u)", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s(%d): No renderer", __FUNCTION__, streamId);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
// Stop the incoming stream
|
||
|
MapItem* item = _streamRenderMap.Find(streamId);
|
||
|
if (item == NULL)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
_id, "%s: Could find render stream %d", __FUNCTION__,
|
||
|
streamId);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream->Stop() == -1)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
_id, "%s: Could not start stream %d", __FUNCTION__,
|
||
|
incomingStream->StreamId());
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::ResetRender()
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
WebRtc_Word32 error = 0;
|
||
|
|
||
|
// Loop through all incoming streams and stop them
|
||
|
MapItem* item = _streamRenderMap.First();
|
||
|
while (item)
|
||
|
{
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream->Reset() == -1)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer,
|
||
|
_id, "%s: Could not reset stream %d", __FUNCTION__,
|
||
|
incomingStream->StreamId());
|
||
|
error = -1;
|
||
|
}
|
||
|
item = _streamRenderMap.Next(item);
|
||
|
}
|
||
|
return error;
|
||
|
}
|
||
|
|
||
|
RawVideoType ModuleVideoRenderImpl::PreferredVideoType() const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (_ptrRenderer == NULL)
|
||
|
{
|
||
|
return kVideoI420;
|
||
|
}
|
||
|
|
||
|
return _ptrRenderer->PerferedVideoType();
|
||
|
}
|
||
|
|
||
|
bool ModuleVideoRenderImpl::IsFullScreen()
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return false;
|
||
|
}
|
||
|
return _ptrRenderer->FullScreen();
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::GetScreenResolution(
|
||
|
WebRtc_UWord32& screenWidth,
|
||
|
WebRtc_UWord32& screenHeight) const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return false;
|
||
|
}
|
||
|
return _ptrRenderer->GetScreenResolution(screenWidth, screenHeight);
|
||
|
}
|
||
|
|
||
|
WebRtc_UWord32 ModuleVideoRenderImpl::RenderFrameRate(
|
||
|
const WebRtc_UWord32 streamId)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, streamId: %u", __FUNCTION__, streamId);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return false;
|
||
|
}
|
||
|
return _ptrRenderer->RenderFrameRate(streamId);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::SetStreamCropping(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
const float left,
|
||
|
const float top,
|
||
|
const float right,
|
||
|
const float bottom)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, l: %1.1f, t: %1.1f, r: %1.1f, b: %1.1f", __FUNCTION__,
|
||
|
left, top, right, bottom);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return false;
|
||
|
}
|
||
|
return _ptrRenderer->SetStreamCropping(streamId, left, top, right, bottom);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::SetTransparentBackground(const bool enable)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, enable: %d", __FUNCTION__, enable);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return false;
|
||
|
}
|
||
|
return _ptrRenderer->SetTransparentBackground(enable);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::FullScreenRender(void* window,
|
||
|
const bool enable)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::SetText(
|
||
|
const WebRtc_UWord8 textId,
|
||
|
const WebRtc_UWord8* text,
|
||
|
const WebRtc_Word32 textLength,
|
||
|
const WebRtc_UWord32 textColorRef,
|
||
|
const WebRtc_UWord32 backgroundColorRef,
|
||
|
const float left, const float top,
|
||
|
const float right,
|
||
|
const float bottom)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
return _ptrRenderer->SetText(textId, text, textLength, textColorRef,
|
||
|
backgroundColorRef, left, top, right, bottom);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::SetBitmap(const void* bitMap,
|
||
|
const WebRtc_UWord8 pictureId,
|
||
|
const void* colorKey,
|
||
|
const float left,
|
||
|
const float top,
|
||
|
const float right,
|
||
|
const float bottom)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
return _ptrRenderer->SetBitmap(bitMap, pictureId, colorKey, left, top,
|
||
|
right, bottom);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::GetLastRenderedFrame(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
VideoFrame &frame) const
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
MapItem *item = _streamRenderMap.Find(streamId);
|
||
|
if (item == NULL)
|
||
|
{
|
||
|
// This stream doesn't exist
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return 0;
|
||
|
}
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream == NULL)
|
||
|
{
|
||
|
// This should never happen
|
||
|
assert(false);
|
||
|
_streamRenderMap.Erase(item);
|
||
|
return 0;
|
||
|
}
|
||
|
return incomingStream->GetLastRenderedFrame(frame);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::ConfigureRenderer(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
const unsigned int zOrder,
|
||
|
const float left,
|
||
|
const float top,
|
||
|
const float right,
|
||
|
const float bottom)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s, l: %1.1f, t: %1.1f, r: %1.1f, b: %1.1f", __FUNCTION__,
|
||
|
left, top, right, bottom);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return false;
|
||
|
}
|
||
|
return _ptrRenderer->ConfigureRenderer(streamId, zOrder, left, top, right,
|
||
|
bottom);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::SetStartImage(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
const VideoFrame& videoFrame)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
MapItem *item = _streamRenderMap.Find(streamId);
|
||
|
if (item == NULL)
|
||
|
{
|
||
|
// This stream doesn't exist
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream == NULL)
|
||
|
{
|
||
|
// This should never happen
|
||
|
assert(false);
|
||
|
_streamRenderMap.Erase(item);
|
||
|
return 0;
|
||
|
}
|
||
|
return incomingStream->SetStartImage(videoFrame);
|
||
|
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::SetTimeoutImage(
|
||
|
const WebRtc_UWord32 streamId,
|
||
|
const VideoFrame& videoFrame,
|
||
|
const WebRtc_UWord32 timeout)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
MapItem *item = _streamRenderMap.Find(streamId);
|
||
|
if (item == NULL)
|
||
|
{
|
||
|
// This stream doesn't exist
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream == NULL)
|
||
|
{
|
||
|
// This should never happen
|
||
|
assert(false);
|
||
|
_streamRenderMap.Erase(item);
|
||
|
return 0;
|
||
|
}
|
||
|
return incomingStream->SetTimeoutImage(videoFrame, timeout);
|
||
|
}
|
||
|
|
||
|
WebRtc_Word32 ModuleVideoRenderImpl::MirrorRenderStream(const int renderId,
|
||
|
const bool enable,
|
||
|
const bool mirrorXAxis,
|
||
|
const bool mirrorYAxis)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceModuleCall, kTraceVideoRenderer, _id,
|
||
|
"%s", __FUNCTION__);
|
||
|
CriticalSectionScoped cs(_moduleCrit);
|
||
|
|
||
|
if (!_ptrRenderer)
|
||
|
{
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: No renderer", __FUNCTION__);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
MapItem *item = _streamRenderMap.Find(renderId);
|
||
|
if (item == NULL)
|
||
|
{
|
||
|
// This stream doesn't exist
|
||
|
WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id,
|
||
|
"%s: stream doesn't exist", __FUNCTION__);
|
||
|
return 0;
|
||
|
}
|
||
|
IncomingVideoStream* incomingStream =
|
||
|
static_cast<IncomingVideoStream*> (item->GetItem());
|
||
|
if (incomingStream == NULL)
|
||
|
{
|
||
|
// This should never happen
|
||
|
assert(false);
|
||
|
_streamRenderMap.Erase(item);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return incomingStream->EnableMirroring(enable, mirrorXAxis, mirrorYAxis);
|
||
|
}
|
||
|
|
||
|
} //namespace webrtc
|
||
|
|