| /* |
| * Copyright (c) 2012 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 "webrtc/engine_configurations.h" |
| #if defined(COCOA_RENDERING) |
| |
| #include "webrtc/common_video/libyuv/include/webrtc_libyuv.h" |
| #include "webrtc/modules/video_render/mac/video_render_nsopengl.h" |
| #include "webrtc/system_wrappers/interface/critical_section_wrapper.h" |
| #include "webrtc/system_wrappers/interface/event_wrapper.h" |
| #include "webrtc/system_wrappers/interface/thread_wrapper.h" |
| #include "webrtc/system_wrappers/interface/trace.h" |
| |
| namespace webrtc { |
| |
| VideoChannelNSOpenGL::VideoChannelNSOpenGL(NSOpenGLContext *nsglContext, int iId, VideoRenderNSOpenGL* owner) : |
| _nsglContext( nsglContext), |
| _id( iId), |
| _owner( owner), |
| _width( 0), |
| _height( 0), |
| _startWidth( 0.0f), |
| _startHeight( 0.0f), |
| _stopWidth( 0.0f), |
| _stopHeight( 0.0f), |
| _stretchedWidth( 0), |
| _stretchedHeight( 0), |
| _oldStretchedHeight( 0), |
| _oldStretchedWidth( 0), |
| _buffer( 0), |
| _bufferSize( 0), |
| _incomingBufferSize( 0), |
| _bufferIsUpdated( false), |
| _numberOfStreams( 0), |
| _pixelFormat( GL_RGBA), |
| _pixelDataType( GL_UNSIGNED_INT_8_8_8_8), |
| _texture( 0) |
| { |
| |
| } |
| |
| VideoChannelNSOpenGL::~VideoChannelNSOpenGL() |
| { |
| if (_buffer) |
| { |
| delete [] _buffer; |
| _buffer = NULL; |
| } |
| |
| if (_texture != 0) |
| { |
| [_nsglContext makeCurrentContext]; |
| glDeleteTextures(1, (const GLuint*) &_texture); |
| _texture = 0; |
| } |
| } |
| |
| int VideoChannelNSOpenGL::ChangeContext(NSOpenGLContext *nsglContext) |
| { |
| _owner->LockAGLCntx(); |
| |
| _nsglContext = nsglContext; |
| [_nsglContext makeCurrentContext]; |
| |
| _owner->UnlockAGLCntx(); |
| return 0; |
| |
| } |
| |
| int32_t VideoChannelNSOpenGL::GetChannelProperties(float& left, float& top, |
| float& right, float& bottom) |
| { |
| |
| _owner->LockAGLCntx(); |
| |
| left = _startWidth; |
| top = _startHeight; |
| right = _stopWidth; |
| bottom = _stopHeight; |
| |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int32_t VideoChannelNSOpenGL::RenderFrame( |
| const uint32_t /*streamId*/, I420VideoFrame& videoFrame) { |
| |
| _owner->LockAGLCntx(); |
| |
| if(_width != videoFrame.width() || |
| _height != videoFrame.height()) { |
| if(FrameSizeChange(videoFrame.width(), videoFrame.height(), 1) == -1) { |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| } |
| int ret = DeliverFrame(videoFrame); |
| |
| _owner->UnlockAGLCntx(); |
| return ret; |
| } |
| |
| int VideoChannelNSOpenGL::UpdateSize(int width, int height) |
| { |
| _owner->LockAGLCntx(); |
| _width = width; |
| _height = height; |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoChannelNSOpenGL::UpdateStretchSize(int stretchHeight, int stretchWidth) |
| { |
| |
| _owner->LockAGLCntx(); |
| _stretchedHeight = stretchHeight; |
| _stretchedWidth = stretchWidth; |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoChannelNSOpenGL::FrameSizeChange(int width, int height, int numberOfStreams) |
| { |
| // We got a new frame size from VideoAPI, prepare the buffer |
| |
| _owner->LockAGLCntx(); |
| |
| if (width == _width && _height == height) |
| { |
| // We already have a correct buffer size |
| _numberOfStreams = numberOfStreams; |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| _width = width; |
| _height = height; |
| |
| // Delete the old buffer, create a new one with correct size. |
| if (_buffer) |
| { |
| delete [] _buffer; |
| _bufferSize = 0; |
| } |
| |
| _incomingBufferSize = CalcBufferSize(kI420, _width, _height); |
| _bufferSize = CalcBufferSize(kARGB, _width, _height); |
| _buffer = new unsigned char [_bufferSize]; |
| memset(_buffer, 0, _bufferSize * sizeof(unsigned char)); |
| |
| [_nsglContext makeCurrentContext]; |
| |
| if(glIsTexture(_texture)) |
| { |
| glDeleteTextures(1, (const GLuint*) &_texture); |
| _texture = 0; |
| } |
| |
| // Create a new texture |
| glGenTextures(1, (GLuint *) &_texture); |
| |
| GLenum glErr = glGetError(); |
| |
| if (glErr != GL_NO_ERROR) |
| { |
| |
| } |
| |
| glBindTexture(GL_TEXTURE_RECTANGLE_EXT, _texture); |
| |
| GLint texSize; |
| glGetIntegerv(GL_MAX_TEXTURE_SIZE, &texSize); |
| |
| if (texSize < _width || texSize < _height) |
| { |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| |
| // Set up th texture type and size |
| glTexImage2D(GL_TEXTURE_RECTANGLE_EXT, // target |
| 0, // level |
| GL_RGBA, // internal format |
| _width, // width |
| _height, // height |
| 0, // border 0/1 = off/on |
| _pixelFormat, // format, GL_RGBA |
| _pixelDataType, // data type, GL_UNSIGNED_INT_8_8_8_8 |
| _buffer); // pixel data |
| |
| glErr = glGetError(); |
| if (glErr != GL_NO_ERROR) |
| { |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoChannelNSOpenGL::DeliverFrame(const I420VideoFrame& videoFrame) { |
| |
| _owner->LockAGLCntx(); |
| |
| if (_texture == 0) { |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| if (CalcBufferSize(kI420, videoFrame.width(), videoFrame.height()) != |
| _incomingBufferSize) { |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| |
| // Using the I420VideoFrame for YV12: YV12 is YVU; I420 assumes |
| // YUV. |
| // TODO(mikhal) : Use appropriate functionality. |
| // TODO(wu): See if we are using glTexSubImage2D correctly. |
| int rgbRet = ConvertFromYV12(videoFrame, kBGRA, 0, _buffer); |
| if (rgbRet < 0) { |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| |
| [_nsglContext makeCurrentContext]; |
| |
| // Make sure this texture is the active one |
| glBindTexture(GL_TEXTURE_RECTANGLE_EXT, _texture); |
| GLenum glErr = glGetError(); |
| if (glErr != GL_NO_ERROR) { |
| WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id, |
| "ERROR %d while calling glBindTexture", glErr); |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| |
| glTexSubImage2D(GL_TEXTURE_RECTANGLE_EXT, |
| 0, // Level, not use |
| 0, // start point x, (low left of pic) |
| 0, // start point y, |
| _width, // width |
| _height, // height |
| _pixelFormat, // pictue format for _buffer |
| _pixelDataType, // data type of _buffer |
| (const GLvoid*) _buffer); // the pixel data |
| |
| glErr = glGetError(); |
| if (glErr != GL_NO_ERROR) { |
| WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id, |
| "ERROR %d while calling glTexSubImage2d", glErr); |
| _owner->UnlockAGLCntx(); |
| return -1; |
| } |
| |
| _bufferIsUpdated = true; |
| |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoChannelNSOpenGL::RenderOffScreenBuffer() |
| { |
| |
| _owner->LockAGLCntx(); |
| |
| if (_texture == 0) |
| { |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| // if(_fullscreen) |
| // { |
| // NSRect mainDisplayRect = [[NSScreen mainScreen] frame]; |
| // _width = mainDisplayRect.size.width; |
| // _height = mainDisplayRect.size.height; |
| // glViewport(0, 0, mainDisplayRect.size.width, mainDisplayRect.size.height); |
| // float newX = mainDisplayRect.size.width/_width; |
| // float newY = mainDisplayRect.size.height/_height; |
| |
| // convert from 0.0 <= size <= 1.0 to |
| // open gl world -1.0 < size < 1.0 |
| GLfloat xStart = 2.0f * _startWidth - 1.0f; |
| GLfloat xStop = 2.0f * _stopWidth - 1.0f; |
| GLfloat yStart = 1.0f - 2.0f * _stopHeight; |
| GLfloat yStop = 1.0f - 2.0f * _startHeight; |
| |
| [_nsglContext makeCurrentContext]; |
| |
| glBindTexture(GL_TEXTURE_RECTANGLE_EXT, _texture); |
| _oldStretchedHeight = _stretchedHeight; |
| _oldStretchedWidth = _stretchedWidth; |
| |
| glLoadIdentity(); |
| glEnable(GL_TEXTURE_RECTANGLE_EXT); |
| glBegin(GL_POLYGON); |
| { |
| glTexCoord2f(0.0, 0.0); glVertex2f(xStart, yStop); |
| glTexCoord2f(_width, 0.0); glVertex2f(xStop, yStop); |
| glTexCoord2f(_width, _height); glVertex2f(xStop, yStart); |
| glTexCoord2f(0.0, _height); glVertex2f(xStart, yStart); |
| } |
| glEnd(); |
| |
| glDisable(GL_TEXTURE_RECTANGLE_EXT); |
| |
| _bufferIsUpdated = false; |
| |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoChannelNSOpenGL::IsUpdated(bool& isUpdated) |
| { |
| _owner->LockAGLCntx(); |
| |
| isUpdated = _bufferIsUpdated; |
| |
| _owner->UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoChannelNSOpenGL::SetStreamSettings(int /*streamId*/, float startWidth, float startHeight, float stopWidth, float stopHeight) |
| { |
| _owner->LockAGLCntx(); |
| |
| _startWidth = startWidth; |
| _stopWidth = stopWidth; |
| _startHeight = startHeight; |
| _stopHeight = stopHeight; |
| |
| int oldWidth = _width; |
| int oldHeight = _height; |
| int oldNumberOfStreams = _numberOfStreams; |
| |
| _width = 0; |
| _height = 0; |
| |
| int retVal = FrameSizeChange(oldWidth, oldHeight, oldNumberOfStreams); |
| |
| _owner->UnlockAGLCntx(); |
| return retVal; |
| } |
| |
| int VideoChannelNSOpenGL::SetStreamCropSettings(int /*streamId*/, float /*startWidth*/, float /*startHeight*/, float /*stopWidth*/, float /*stopHeight*/) |
| { |
| return -1; |
| } |
| |
| /* |
| * |
| * VideoRenderNSOpenGL |
| * |
| */ |
| |
| VideoRenderNSOpenGL::VideoRenderNSOpenGL(CocoaRenderView *windowRef, bool fullScreen, int iId) : |
| _windowRef( (CocoaRenderView*)windowRef), |
| _fullScreen( fullScreen), |
| _id( iId), |
| _nsglContextCritSec( *CriticalSectionWrapper::CreateCriticalSection()), |
| _screenUpdateThread( 0), |
| _screenUpdateEvent( 0), |
| _nsglContext( 0), |
| _nsglFullScreenContext( 0), |
| _fullScreenWindow( nil), |
| _windowRect( ), |
| _windowWidth( 0), |
| _windowHeight( 0), |
| _nsglChannels( ), |
| _zOrderToChannel( ), |
| _threadID (0), |
| _renderingIsPaused (FALSE), |
| _windowRefSuperView(NULL), |
| _windowRefSuperViewFrame(NSMakeRect(0,0,0,0)) |
| { |
| _screenUpdateThread = ThreadWrapper::CreateThread(ScreenUpdateThreadProc, this, kRealtimePriority); |
| _screenUpdateEvent = EventWrapper::Create(); |
| } |
| |
| int VideoRenderNSOpenGL::ChangeWindow(CocoaRenderView* newWindowRef) |
| { |
| |
| LockAGLCntx(); |
| |
| _windowRef = newWindowRef; |
| |
| if(CreateMixingContext() == -1) |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| int error = 0; |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.begin(); |
| while (it!= _nsglChannels.end()) |
| { |
| error |= (it->second)->ChangeContext(_nsglContext); |
| it++; |
| } |
| if(error != 0) |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| /* Check if the thread and event already exist. |
| * If so then they will simply be restarted |
| * If not then create them and continue |
| */ |
| int32_t VideoRenderNSOpenGL::StartRender() |
| { |
| |
| LockAGLCntx(); |
| |
| const unsigned int MONITOR_FREQ = 60; |
| if(TRUE == _renderingIsPaused) |
| { |
| WEBRTC_TRACE(kTraceDebug, kTraceVideoRenderer, _id, "Restarting screenUpdateThread"); |
| |
| // we already have the thread. Most likely StopRender() was called and they were paused |
| if(FALSE == _screenUpdateThread->Start(_threadID) || |
| FALSE == _screenUpdateEvent->StartTimer(true, 1000/MONITOR_FREQ)) |
| { |
| WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id, "Failed to restart screenUpdateThread or screenUpdateEvent"); |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| |
| if (!_screenUpdateThread) |
| { |
| WEBRTC_TRACE(kTraceDebug, kTraceVideoRenderer, _id, "failed start screenUpdateThread"); |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| int32_t VideoRenderNSOpenGL::StopRender() |
| { |
| |
| LockAGLCntx(); |
| |
| /* The code below is functional |
| * but it pauses for several seconds |
| */ |
| |
| // pause the update thread and the event timer |
| if(!_screenUpdateThread || !_screenUpdateEvent) |
| { |
| _renderingIsPaused = TRUE; |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| if(FALSE == _screenUpdateThread->Stop() || FALSE == _screenUpdateEvent->StopTimer()) |
| { |
| _renderingIsPaused = FALSE; |
| |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| _renderingIsPaused = TRUE; |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoRenderNSOpenGL::configureNSOpenGLView() |
| { |
| return 0; |
| |
| } |
| |
| int VideoRenderNSOpenGL::configureNSOpenGLEngine() |
| { |
| |
| LockAGLCntx(); |
| |
| // Disable not needed functionality to increase performance |
| glDisable(GL_DITHER); |
| glDisable(GL_ALPHA_TEST); |
| glDisable(GL_STENCIL_TEST); |
| glDisable(GL_FOG); |
| glDisable(GL_TEXTURE_2D); |
| glPixelZoom(1.0, 1.0); |
| glDisable(GL_BLEND); |
| glDisable(GL_DEPTH_TEST); |
| glDepthMask(GL_FALSE); |
| glDisable(GL_CULL_FACE); |
| |
| // Set texture parameters |
| glTexParameterf(GL_TEXTURE_RECTANGLE_EXT, GL_TEXTURE_PRIORITY, 1.0); |
| glTexParameteri(GL_TEXTURE_RECTANGLE_EXT, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); |
| glTexParameteri(GL_TEXTURE_RECTANGLE_EXT, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); |
| glTexParameteri(GL_TEXTURE_RECTANGLE_EXT, GL_TEXTURE_MAG_FILTER, GL_LINEAR); |
| glTexParameteri(GL_TEXTURE_RECTANGLE_EXT, GL_TEXTURE_MIN_FILTER, GL_LINEAR); |
| glTexEnvf(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_MODULATE); |
| glPixelStorei(GL_UNPACK_ALIGNMENT, 1); |
| glTexParameteri(GL_TEXTURE_RECTANGLE_EXT, GL_TEXTURE_STORAGE_HINT_APPLE, GL_STORAGE_SHARED_APPLE); |
| |
| if (GetWindowRect(_windowRect) == -1) |
| { |
| UnlockAGLCntx(); |
| return true; |
| } |
| |
| if (_windowWidth != (_windowRect.right - _windowRect.left) |
| || _windowHeight != (_windowRect.bottom - _windowRect.top)) |
| { |
| _windowWidth = _windowRect.right - _windowRect.left; |
| _windowHeight = _windowRect.bottom - _windowRect.top; |
| } |
| glViewport(0, 0, _windowWidth, _windowHeight); |
| |
| // Synchronize buffer swaps with vertical refresh rate |
| GLint swapInt = 1; |
| [_nsglContext setValues:&swapInt forParameter:NSOpenGLCPSwapInterval]; |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoRenderNSOpenGL::setRenderTargetWindow() |
| { |
| LockAGLCntx(); |
| |
| |
| GLuint attribs[] = |
| { |
| NSOpenGLPFAColorSize, 24, |
| NSOpenGLPFAAlphaSize, 8, |
| NSOpenGLPFADepthSize, 16, |
| NSOpenGLPFAAccelerated, |
| 0 |
| }; |
| |
| NSOpenGLPixelFormat* fmt = [[[NSOpenGLPixelFormat alloc] initWithAttributes: |
| (NSOpenGLPixelFormatAttribute*) attribs] autorelease]; |
| |
| if(_windowRef) |
| { |
| [_windowRef initCocoaRenderView:fmt]; |
| } |
| else |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| _nsglContext = [_windowRef nsOpenGLContext]; |
| [_nsglContext makeCurrentContext]; |
| |
| glClearColor(0.0f, 0.0f, 0.0f, 1.0f); |
| glClear(GL_COLOR_BUFFER_BIT); |
| |
| |
| DisplayBuffers(); |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoRenderNSOpenGL::setRenderTargetFullScreen() |
| { |
| LockAGLCntx(); |
| |
| |
| GLuint attribs[] = |
| { |
| NSOpenGLPFAColorSize, 24, |
| NSOpenGLPFAAlphaSize, 8, |
| NSOpenGLPFADepthSize, 16, |
| NSOpenGLPFAAccelerated, |
| 0 |
| }; |
| |
| NSOpenGLPixelFormat* fmt = [[[NSOpenGLPixelFormat alloc] initWithAttributes: |
| (NSOpenGLPixelFormatAttribute*) attribs] autorelease]; |
| |
| // Store original superview and frame for use when exiting full screens |
| _windowRefSuperViewFrame = [_windowRef frame]; |
| _windowRefSuperView = [_windowRef superview]; |
| |
| |
| // create new fullscreen window |
| NSRect screenRect = [[NSScreen mainScreen]frame]; |
| [_windowRef setFrame:screenRect]; |
| [_windowRef setBounds:screenRect]; |
| |
| |
| _fullScreenWindow = [[CocoaFullScreenWindow alloc]init]; |
| [_fullScreenWindow grabFullScreen]; |
| [[[_fullScreenWindow window] contentView] addSubview:_windowRef]; |
| |
| if(_windowRef) |
| { |
| [_windowRef initCocoaRenderViewFullScreen:fmt]; |
| } |
| else |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| _nsglContext = [_windowRef nsOpenGLContext]; |
| [_nsglContext makeCurrentContext]; |
| |
| glClearColor(0.0f, 0.0f, 0.0f, 1.0f); |
| glClear(GL_COLOR_BUFFER_BIT); |
| |
| DisplayBuffers(); |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| VideoRenderNSOpenGL::~VideoRenderNSOpenGL() |
| { |
| |
| if(_fullScreen) |
| { |
| if(_fullScreenWindow) |
| { |
| // Detach CocoaRenderView from full screen view back to |
| // it's original parent. |
| [_windowRef removeFromSuperview]; |
| if(_windowRefSuperView) |
| { |
| [_windowRefSuperView addSubview:_windowRef]; |
| [_windowRef setFrame:_windowRefSuperViewFrame]; |
| } |
| |
| WEBRTC_TRACE(kTraceDebug, kTraceVideoRenderer, 0, "%s:%d Attempting to release fullscreen window", __FUNCTION__, __LINE__); |
| [_fullScreenWindow releaseFullScreen]; |
| |
| } |
| } |
| |
| // Signal event to exit thread, then delete it |
| ThreadWrapper* tmpPtr = _screenUpdateThread; |
| _screenUpdateThread = NULL; |
| |
| if (tmpPtr) |
| { |
| tmpPtr->SetNotAlive(); |
| _screenUpdateEvent->Set(); |
| _screenUpdateEvent->StopTimer(); |
| |
| if (tmpPtr->Stop()) |
| { |
| delete tmpPtr; |
| } |
| delete _screenUpdateEvent; |
| _screenUpdateEvent = NULL; |
| } |
| |
| if (_nsglContext != 0) |
| { |
| [_nsglContext makeCurrentContext]; |
| _nsglContext = nil; |
| } |
| |
| // Delete all channels |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.begin(); |
| while (it!= _nsglChannels.end()) |
| { |
| delete it->second; |
| _nsglChannels.erase(it); |
| it = _nsglChannels.begin(); |
| } |
| _nsglChannels.clear(); |
| |
| // Clean the zOrder map |
| std::multimap<int, int>::iterator zIt = _zOrderToChannel.begin(); |
| while(zIt != _zOrderToChannel.end()) |
| { |
| _zOrderToChannel.erase(zIt); |
| zIt = _zOrderToChannel.begin(); |
| } |
| _zOrderToChannel.clear(); |
| |
| } |
| |
| /* static */ |
| int VideoRenderNSOpenGL::GetOpenGLVersion(int& /*nsglMajor*/, int& /*nsglMinor*/) |
| { |
| return -1; |
| } |
| |
| int VideoRenderNSOpenGL::Init() |
| { |
| |
| LockAGLCntx(); |
| if (!_screenUpdateThread) |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| _screenUpdateThread->Start(_threadID); |
| |
| // Start the event triggering the render process |
| unsigned int monitorFreq = 60; |
| _screenUpdateEvent->StartTimer(true, 1000/monitorFreq); |
| |
| if (CreateMixingContext() == -1) |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| VideoChannelNSOpenGL* VideoRenderNSOpenGL::CreateNSGLChannel(int channel, int zOrder, float startWidth, float startHeight, float stopWidth, float stopHeight) |
| { |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| if (HasChannel(channel)) |
| { |
| return NULL; |
| } |
| |
| if (_zOrderToChannel.find(zOrder) != _zOrderToChannel.end()) |
| { |
| |
| } |
| |
| VideoChannelNSOpenGL* newAGLChannel = new VideoChannelNSOpenGL(_nsglContext, _id, this); |
| if (newAGLChannel->SetStreamSettings(0, startWidth, startHeight, stopWidth, stopHeight) == -1) |
| { |
| if (newAGLChannel) |
| { |
| delete newAGLChannel; |
| newAGLChannel = NULL; |
| } |
| |
| return NULL; |
| } |
| |
| _nsglChannels[channel] = newAGLChannel; |
| _zOrderToChannel.insert(std::pair<int, int>(zOrder, channel)); |
| |
| WEBRTC_TRACE(kTraceInfo, kTraceVideoRenderer, _id, "%s successfully created NSGL channel number %d", __FUNCTION__, channel); |
| |
| return newAGLChannel; |
| } |
| |
| int VideoRenderNSOpenGL::DeleteAllNSGLChannels() |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| std::map<int, VideoChannelNSOpenGL*>::iterator it; |
| it = _nsglChannels.begin(); |
| |
| while (it != _nsglChannels.end()) |
| { |
| VideoChannelNSOpenGL* channel = it->second; |
| WEBRTC_TRACE(kTraceInfo, kTraceVideoRenderer, _id, "%s Deleting channel %d", __FUNCTION__, channel); |
| delete channel; |
| it++; |
| } |
| _nsglChannels.clear(); |
| return 0; |
| } |
| |
| int32_t VideoRenderNSOpenGL::DeleteNSGLChannel(const uint32_t channel) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| std::map<int, VideoChannelNSOpenGL*>::iterator it; |
| it = _nsglChannels.find(channel); |
| if (it != _nsglChannels.end()) |
| { |
| delete it->second; |
| _nsglChannels.erase(it); |
| } |
| else |
| { |
| return -1; |
| } |
| |
| std::multimap<int, int>::iterator zIt = _zOrderToChannel.begin(); |
| while( zIt != _zOrderToChannel.end()) |
| { |
| if (zIt->second == (int)channel) |
| { |
| _zOrderToChannel.erase(zIt); |
| break; |
| } |
| zIt++; |
| } |
| |
| return 0; |
| } |
| |
| int32_t VideoRenderNSOpenGL::GetChannelProperties(const uint16_t streamId, |
| uint32_t& zOrder, |
| float& left, |
| float& top, |
| float& right, |
| float& bottom) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| bool channelFound = false; |
| |
| // Loop through all channels until we find a match. |
| // From that, get zorder. |
| // From that, get T, L, R, B |
| for (std::multimap<int, int>::reverse_iterator rIt = _zOrderToChannel.rbegin(); |
| rIt != _zOrderToChannel.rend(); |
| rIt++) |
| { |
| if(streamId == rIt->second) |
| { |
| channelFound = true; |
| |
| zOrder = rIt->second; |
| |
| std::map<int, VideoChannelNSOpenGL*>::iterator rIt = _nsglChannels.find(streamId); |
| VideoChannelNSOpenGL* tempChannel = rIt->second; |
| |
| if(-1 == tempChannel->GetChannelProperties(left, top, right, bottom) ) |
| { |
| return -1; |
| } |
| break; |
| } |
| } |
| |
| if(false == channelFound) |
| { |
| |
| return -1; |
| } |
| |
| return 0; |
| } |
| |
| int VideoRenderNSOpenGL::StopThread() |
| { |
| |
| ThreadWrapper* tmpPtr = _screenUpdateThread; |
| WEBRTC_TRACE(kTraceInfo, kTraceVideoRenderer, _id, "%s Stopping thread ", __FUNCTION__, _screenUpdateThread); |
| _screenUpdateThread = NULL; |
| |
| if (tmpPtr) |
| { |
| tmpPtr->SetNotAlive(); |
| _screenUpdateEvent->Set(); |
| if (tmpPtr->Stop()) |
| { |
| delete tmpPtr; |
| } |
| } |
| |
| delete _screenUpdateEvent; |
| _screenUpdateEvent = NULL; |
| |
| return 0; |
| } |
| |
| bool VideoRenderNSOpenGL::IsFullScreen() |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| return _fullScreen; |
| } |
| |
| bool VideoRenderNSOpenGL::HasChannels() |
| { |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| if (_nsglChannels.begin() != _nsglChannels.end()) |
| { |
| return true; |
| } |
| return false; |
| } |
| |
| bool VideoRenderNSOpenGL::HasChannel(int channel) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.find(channel); |
| |
| if (it != _nsglChannels.end()) |
| { |
| return true; |
| } |
| return false; |
| } |
| |
| int VideoRenderNSOpenGL::GetChannels(std::list<int>& channelList) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.begin(); |
| |
| while (it != _nsglChannels.end()) |
| { |
| channelList.push_back(it->first); |
| it++; |
| } |
| |
| return 0; |
| } |
| |
| VideoChannelNSOpenGL* VideoRenderNSOpenGL::ConfigureNSGLChannel(int channel, int zOrder, float startWidth, float startHeight, float stopWidth, float stopHeight) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.find(channel); |
| |
| if (it != _nsglChannels.end()) |
| { |
| VideoChannelNSOpenGL* aglChannel = it->second; |
| if (aglChannel->SetStreamSettings(0, startWidth, startHeight, stopWidth, stopHeight) == -1) |
| { |
| WEBRTC_TRACE(kTraceError, kTraceVideoRenderer, _id, "%s failed to set stream settings: channel %d. channel=%d zOrder=%d startWidth=%d startHeight=%d stopWidth=%d stopHeight=%d", |
| __FUNCTION__, channel, zOrder, startWidth, startHeight, stopWidth, stopHeight); |
| return NULL; |
| } |
| WEBRTC_TRACE(kTraceInfo, kTraceVideoRenderer, _id, "%s Configuring channel %d. channel=%d zOrder=%d startWidth=%d startHeight=%d stopWidth=%d stopHeight=%d", |
| __FUNCTION__, channel, zOrder, startWidth, startHeight, stopWidth, stopHeight); |
| |
| std::multimap<int, int>::iterator it = _zOrderToChannel.begin(); |
| while(it != _zOrderToChannel.end()) |
| { |
| if (it->second == channel) |
| { |
| if (it->first != zOrder) |
| { |
| _zOrderToChannel.erase(it); |
| _zOrderToChannel.insert(std::pair<int, int>(zOrder, channel)); |
| } |
| break; |
| } |
| it++; |
| } |
| return aglChannel; |
| } |
| |
| return NULL; |
| } |
| |
| /* |
| * |
| * Rendering process |
| * |
| */ |
| |
| bool VideoRenderNSOpenGL::ScreenUpdateThreadProc(void* obj) |
| { |
| return static_cast<VideoRenderNSOpenGL*>(obj)->ScreenUpdateProcess(); |
| } |
| |
| bool VideoRenderNSOpenGL::ScreenUpdateProcess() |
| { |
| |
| _screenUpdateEvent->Wait(10); |
| LockAGLCntx(); |
| |
| if (!_screenUpdateThread) |
| { |
| WEBRTC_TRACE(kTraceWarning, kTraceVideoRenderer, _id, "%s no screen update thread", __FUNCTION__); |
| UnlockAGLCntx(); |
| return false; |
| } |
| |
| [_nsglContext makeCurrentContext]; |
| |
| if (GetWindowRect(_windowRect) == -1) |
| { |
| UnlockAGLCntx(); |
| return true; |
| } |
| |
| if (_windowWidth != (_windowRect.right - _windowRect.left) |
| || _windowHeight != (_windowRect.bottom - _windowRect.top)) |
| { |
| _windowWidth = _windowRect.right - _windowRect.left; |
| _windowHeight = _windowRect.bottom - _windowRect.top; |
| glViewport(0, 0, _windowWidth, _windowHeight); |
| } |
| |
| // Check if there are any updated buffers |
| bool updated = false; |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.begin(); |
| while (it != _nsglChannels.end()) |
| { |
| |
| VideoChannelNSOpenGL* aglChannel = it->second; |
| aglChannel->UpdateStretchSize(_windowHeight, _windowWidth); |
| aglChannel->IsUpdated(updated); |
| if (updated) |
| { |
| break; |
| } |
| it++; |
| } |
| |
| if (updated) |
| { |
| |
| // At least on buffers is updated, we need to repaint the texture |
| if (RenderOffScreenBuffers() != -1) |
| { |
| UnlockAGLCntx(); |
| return true; |
| } |
| } |
| // } |
| UnlockAGLCntx(); |
| return true; |
| } |
| |
| /* |
| * |
| * Functions for creating mixing buffers and screen settings |
| * |
| */ |
| |
| int VideoRenderNSOpenGL::CreateMixingContext() |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| if(_fullScreen) |
| { |
| if(-1 == setRenderTargetFullScreen()) |
| { |
| return -1; |
| } |
| } |
| else |
| { |
| |
| if(-1 == setRenderTargetWindow()) |
| { |
| return -1; |
| } |
| } |
| |
| configureNSOpenGLEngine(); |
| |
| DisplayBuffers(); |
| |
| GLenum glErr = glGetError(); |
| if (glErr) |
| { |
| } |
| |
| return 0; |
| } |
| |
| /* |
| * |
| * Rendering functions |
| * |
| */ |
| |
| int VideoRenderNSOpenGL::RenderOffScreenBuffers() |
| { |
| LockAGLCntx(); |
| |
| // Get the current window size, it might have changed since last render. |
| if (GetWindowRect(_windowRect) == -1) |
| { |
| UnlockAGLCntx(); |
| return -1; |
| } |
| |
| [_nsglContext makeCurrentContext]; |
| glClear(GL_COLOR_BUFFER_BIT); |
| |
| // Loop through all channels starting highest zOrder ending with lowest. |
| for (std::multimap<int, int>::reverse_iterator rIt = _zOrderToChannel.rbegin(); |
| rIt != _zOrderToChannel.rend(); |
| rIt++) |
| { |
| int channelId = rIt->second; |
| std::map<int, VideoChannelNSOpenGL*>::iterator it = _nsglChannels.find(channelId); |
| |
| VideoChannelNSOpenGL* aglChannel = it->second; |
| |
| aglChannel->RenderOffScreenBuffer(); |
| } |
| |
| DisplayBuffers(); |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| /* |
| * |
| * Help functions |
| * |
| * All help functions assumes external protections |
| * |
| */ |
| |
| int VideoRenderNSOpenGL::DisplayBuffers() |
| { |
| |
| LockAGLCntx(); |
| |
| glFinish(); |
| [_nsglContext flushBuffer]; |
| |
| WEBRTC_TRACE(kTraceDebug, kTraceVideoRenderer, _id, "%s glFinish and [_nsglContext flushBuffer]", __FUNCTION__); |
| |
| UnlockAGLCntx(); |
| return 0; |
| } |
| |
| int VideoRenderNSOpenGL::GetWindowRect(Rect& rect) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| |
| if (_windowRef) |
| { |
| if(_fullScreen) |
| { |
| NSRect mainDisplayRect = [[NSScreen mainScreen] frame]; |
| rect.bottom = 0; |
| rect.left = 0; |
| rect.right = mainDisplayRect.size.width; |
| rect.top = mainDisplayRect.size.height; |
| } |
| else |
| { |
| rect.top = [_windowRef frame].origin.y; |
| rect.left = [_windowRef frame].origin.x; |
| rect.bottom = [_windowRef frame].origin.y + [_windowRef frame].size.height; |
| rect.right = [_windowRef frame].origin.x + [_windowRef frame].size.width; |
| } |
| |
| return 0; |
| } |
| else |
| { |
| return -1; |
| } |
| } |
| |
| int32_t VideoRenderNSOpenGL::ChangeUniqueID(int32_t id) |
| { |
| |
| CriticalSectionScoped cs(&_nsglContextCritSec); |
| _id = id; |
| return 0; |
| } |
| |
| int32_t VideoRenderNSOpenGL::SetText(const uint8_t /*textId*/, |
| const uint8_t* /*text*/, |
| const int32_t /*textLength*/, |
| const uint32_t /*textColorRef*/, |
| const uint32_t /*backgroundColorRef*/, |
| const float /*left*/, |
| const float /*top*/, |
| const float /*right*/, |
| const float /*bottom*/) |
| { |
| |
| return 0; |
| |
| } |
| |
| void VideoRenderNSOpenGL::LockAGLCntx() |
| { |
| _nsglContextCritSec.Enter(); |
| } |
| void VideoRenderNSOpenGL::UnlockAGLCntx() |
| { |
| _nsglContextCritSec.Leave(); |
| } |
| |
| /* |
| |
| bool VideoRenderNSOpenGL::SetFullScreen(bool fullscreen) |
| { |
| NSRect mainDisplayRect, viewRect; |
| |
| // Create a screen-sized window on the display you want to take over |
| // Note, mainDisplayRect has a non-zero origin if the key window is on a secondary display |
| mainDisplayRect = [[NSScreen mainScreen] frame]; |
| fullScreenWindow = [[NSWindow alloc] initWithContentRect:mainDisplayRect styleMask:NSBorderlessWindowMask |
| backing:NSBackingStoreBuffered defer:YES]; |
| |
| // Set the window level to be above the menu bar |
| [fullScreenWindow setLevel:NSMainMenuWindowLevel+1]; |
| |
| // Perform any other window configuration you desire |
| [fullScreenWindow setOpaque:YES]; |
| [fullScreenWindow setHidesOnDeactivate:YES]; |
| |
| // Create a view with a double-buffered OpenGL context and attach it to the window |
| // By specifying the non-fullscreen context as the shareContext, we automatically inherit the OpenGL objects (textures, etc) it has defined |
| viewRect = NSMakeRect(0.0, 0.0, mainDisplayRect.size.width, mainDisplayRect.size.height); |
| fullScreenView = [[MyOpenGLView alloc] initWithFrame:viewRect shareContext:[openGLView openGLContext]]; |
| [fullScreenWindow setContentView:fullScreenView]; |
| |
| // Show the window |
| [fullScreenWindow makeKeyAndOrderFront:self]; |
| |
| // Set the scene with the full-screen viewport and viewing transformation |
| [scene setViewportRect:viewRect]; |
| |
| // Assign the view's MainController to self |
| [fullScreenView setMainController:self]; |
| |
| if (!isAnimating) { |
| // Mark the view as needing drawing to initalize its contents |
| [fullScreenView setNeedsDisplay:YES]; |
| } |
| else { |
| // Start playing the animation |
| [fullScreenView startAnimation]; |
| } |
| |
| } |
| |
| |
| |
| */ |
| |
| |
| } // namespace webrtc |
| |
| #endif // COCOA_RENDERING |