os/mm/mmswadaptation/videorenderer/src/videorenderer.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "videorenderer.h"
    17 #include "rendererrelay.h"
    18 #include "buffermanager.h"
    19 #include "rendererutil.h"
    20 #include "renderertimer.h"
    21 #include "resourcefilereader.h"
    22 #include "videoframebuffer.h"
    23 
    24 
    25 _LIT(KVideoRendererThreadName, "VideoRendererThread");
    26 _LIT(KResourceFileName, "Z:\\resource\\videorenderer\\videorenderer.rsc");
    27 
    28 const TThreadPriority KSubThreadPriority = EPriorityRealTime;
    29 const TInt KSubThreadStackSize = KDefaultStackSize;	// 0x2000 = 8k
    30 
    31 
    32 
    33 /**
    34 Factory method. Creates a new video renderer instance.
    35 
    36 @param aObserver Video renderer observer
    37 @param aTimed ETrue if CVideoRenderer should handle timing of buffer updates. 
    38               If ETrue CVideoRenderer launches its own high-priority thread to handle buffer update timing. If EFalse CVideoRenderer requires a CActiveScheduler to be present in calling thread.
    39 @return  A pointer to the newly created CVideoRenderer object.
    40 */
    41 EXPORT_C CVideoRenderer* CVideoRenderer::NewL(MVideoRendererObserver& aObserver, TBool aTimed)
    42 	{
    43 	CVideoRenderer* self = new (ELeave) CVideoRenderer(aObserver, aTimed);
    44 	CleanupStack::PushL(self);
    45 	self->ConstructL();
    46 	CleanupStack::Pop(self);
    47 	return self;
    48 	}
    49 
    50 /**
    51 Private constructor
    52 @internalComponent
    53 */
    54 CVideoRenderer::CVideoRenderer(MVideoRendererObserver& aObserver, TBool aTimed)
    55 :iObserver(aObserver), iTimed(aTimed)
    56 	{
    57 	}
    58 
    59 /**
    60 Second phase constructor
    61 @internalComponent
    62 */
    63 void CVideoRenderer::ConstructL()
    64 	{
    65 	User::LeaveIfError(iSurfaceManager.Open());
    66 
    67 	User::LeaveIfError(iWsSession.Connect());
    68 	
    69 	CResourceFileReader* reader = CResourceFileReader::NewLC(KResourceFileName);
    70 	reader->ReadSupportedFormatL(iSupportedFormat);
    71 
    72 	if (iTimed)
    73 		{
    74 		// Create a high priority thread for timed mode
    75 
    76 		// get timer info for timed mode
    77 		TInt64 defaultDelay;
    78 		TInt64 maxDelay;
    79 		reader->ReadTimerInfoL(defaultDelay, maxDelay);
    80 
    81 		//Get a reference to this thread's heap
    82 		RHeap& thisHeap = User::Heap();
    83 
    84 		//Parameters to send to the sub thread
    85 		TThreadRelayParam param;
    86 		param.iObserver = &iObserver;
    87 		param.iThreadRelay = &iRendererRelay; // return pointer to allow direct calls
    88 
    89 		//Get the id of this thread
    90 		RThread thisThread;
    91 		TThreadId thisThreadId = thisThread.Id();
    92 		param.iMainThreadId = thisThreadId;
    93 
    94 		//Get a request to signal for setup completion
    95 		TRequestStatus setupComplete = KRequestPending;
    96 		param.iSetupComplete = &setupComplete;
    97 		
    98 		//current time and the "this" pointer for a unique key
    99 		_LIT(KFormatString,"%S.%020Lu.%08X");
   100 	   	TName threadName;
   101 		TTime now;
   102 		now.UniversalTime();
   103 		threadName.Format(KFormatString, &KVideoRendererThreadName, now.Int64(), reinterpret_cast<TUint>(this));
   104 
   105 		//Create a new thread using the same heap as this thread
   106 		TInt result = iRendererThread.Create(threadName,
   107 										ThreadCreateFunction,
   108 										KSubThreadStackSize,
   109 										&thisHeap,
   110 										&param);
   111 		User::LeaveIfError(result);
   112 
   113 		//Run the thread under high priority
   114 		iRendererThread.SetPriority(KSubThreadPriority); 
   115 
   116 		//Wait for thread startup to complete
   117 		TRequestStatus threadStatus = KRequestPending;
   118 		iRendererThread.Logon(threadStatus);
   119 
   120 		//Start the thread
   121 		iRendererThread.Resume();
   122 		User::WaitForRequest(threadStatus, setupComplete);
   123 		if(threadStatus != KRequestPending)
   124 			{
   125 			//Thread creation failed
   126 			TInt reason = iRendererThread.ExitReason();
   127 			DEBUGPRINT3(_L("Renderer thread died with type=%d, reason=%d"), iRendererThread.ExitType(), reason);
   128 			User::Leave(reason);
   129 			}
   130 
   131 		// Thread creation was successfull
   132 		TInt error = iRendererThread.LogonCancel(threadStatus);
   133 		User::LeaveIfError(error); // There is no outstanding request
   134 		User::WaitForRequest(threadStatus); // Consume the signal
   135 
   136 		__ASSERT_DEBUG(iRendererRelay != NULL, User::Panic(_L("CVR::ConstructL"), KErrCorrupt));
   137 		iRendererRelay->SetRendererThread(&iRendererThread);
   138 		iRendererRelay->SetTimerInfo(defaultDelay, maxDelay);
   139 		
   140 		iThreadCreated = ETrue;
   141 		User::LeaveIfError(setupComplete.Int());
   142 
   143 		//Create a listener that will monitor the thread
   144 		iRendererThreadUndertaker = CThreadUndertaker::NewL(iRendererThread);
   145 		}
   146 	else
   147 		{
   148 		iRendererRelay = CRendererRelay::NewL(iObserver);
   149 		}
   150 
   151 	CleanupStack::PopAndDestroy(reader);
   152 	}
   153 
   154 /**
   155 Main thread entry point for the video renderer sub-thread.
   156 Create a cleanup stack for the thread and process the codec
   157 inside a trap for cleanup behaviour.
   158 @internalComponent
   159 
   160 @param aPtr Parameters to be used for creating the thread.
   161 @return The error code for thread termination.
   162 */
   163 TInt CVideoRenderer::ThreadCreateFunction(TAny* aPtr)
   164 	{
   165 	TInt error = KErrNone;
   166 
   167 	// Create a cleanup stack for the thread
   168 	CTrapCleanup* cleanupStack = CTrapCleanup::New();
   169 	if (cleanupStack)
   170 		{
   171 		if(error == KErrNone)
   172 			{
   173 			TRAP(error, ThreadTrapFunctionL(aPtr));
   174 			}
   175 		}
   176 	else
   177 		{
   178 		error = KErrNoMemory;
   179 		}
   180 
   181 	delete cleanupStack;
   182 	return error;
   183 	}
   184 
   185 /**
   186 Function for thread execution. Create an active scheduler for the thread
   187 and start the function call listener. If the thread is successfully created signal
   188 the main thread that the thread creation was successfull.
   189 @internalComponent
   190 
   191 @param aPtr A pointer to a TThreadRelayParam object containing the startup parameters
   192 			for the thread.
   193 */
   194 void CVideoRenderer::ThreadTrapFunctionL(TAny* aPtr)
   195 	{
   196 	//Create an active scheduler for the thread
   197 	CActiveScheduler* scheduler = new (ELeave) CActiveScheduler;
   198 	CleanupStack::PushL(scheduler);
   199 	CActiveScheduler::Install(scheduler);
   200 
   201 	//Call a Factory function to create a CSubThreadRelay derived object
   202 	TThreadRelayParam* relayParam;
   203 	relayParam = static_cast<TThreadRelayParam*>(aPtr);
   204 
   205 	CRendererThreadRelay* threadRelay = CRendererThreadRelay::NewL(*relayParam->iObserver, relayParam->iMainThreadId);
   206 	CleanupStack::PushL(threadRelay);
   207 	
   208 	//Send a pointer to the sub thread relay back to the main thread
   209 	*relayParam->iThreadRelay = threadRelay;
   210 
   211 	threadRelay->Start();
   212 	threadRelay->SignalSetupComplete(relayParam->iSetupComplete);
   213 
   214 	CActiveScheduler::Start();	
   215 
   216 	CleanupStack::PopAndDestroy(2,scheduler); //threadRelay, scheduler
   217 	}
   218 
   219 /**
   220 Destructor
   221 */
   222 EXPORT_C CVideoRenderer::~CVideoRenderer()
   223 	{
   224 	iWsSession.Close();
   225 	iSupportedFormat.Close();
   226 	
   227 	// first, stop all active objects that may use any buffers
   228 	if(iThreadCreated)
   229 		{
   230 		delete iRendererThreadUndertaker;
   231 
   232 		// this also delete CRendererThreadRelay
   233     	TRequestStatus terminateRequest = KRequestPending;
   234 	    iRendererRelay->Terminate(terminateRequest);
   235     	User::WaitForRequest(terminateRequest);
   236         iThreadCreated = EFalse;
   237 
   238         iRendererThread.Close();
   239 		}
   240 	else if (iTimed == EFalse)
   241 		{
   242 		delete iRendererRelay;
   243 		}
   244 	
   245 	// next, close surface if still open
   246 	if (iBufferManager)
   247 		{
   248 		delete iBufferManager;
   249 		iSurfaceManager.CloseSurface(iSurfaceId);
   250 		}
   251 	
   252 	// close the rest
   253 	iSurfaceManager.Close();
   254 	}
   255 
   256 /**
   257 Retrieves the list of supported surface formats from the renderer.
   258 
   259 @param aArray Supported video formats. The renderer will populate the array with supported formats.
   260 @leave KErrNoMemory Out of memory
   261 */
   262 EXPORT_C void CVideoRenderer::GetSupportedFormatsL(RArray<TUncompressedVideoFormat>& aArray)
   263 	{
   264 	aArray.Reset(); // first clear the old data
   265 	
   266 	TInt count = iSupportedFormat.Count();
   267 	for (TInt i = 0; i < count; ++i)
   268 		{
   269 		aArray.AppendL(iSupportedFormat[i]);
   270 		}
   271 	}
   272 
   273 /**
   274 Creates a new surface for video rendering and registers it with the windows server for all displays.
   275 
   276 @param aSize Surface size in pixels
   277 @param aNumBuffers The minimum number of buffers required. The renderer can create a surface with more buffers.
   278 @param aFormat Surface data format
   279 @param aSurface Output: Surface ID for the new surface
   280 
   281 @leave KErrNoMemory Out of memory
   282 @leave KErrNotSupported The requested parameters are not supported
   283 @leave KErrInUse Too many video surfaces are already in use
   284 */
   285 EXPORT_C void CVideoRenderer::CreateSurfaceL(const TSize& aSize, TInt aNumBuffers, const TUncompressedVideoFormat& aFormat, TSurfaceId& aSurface)
   286 	{
   287 	if (iBufferManager != NULL)
   288 		{
   289 		User::Leave(KErrInUse);
   290 		}
   291 	else if (aNumBuffers <= 0)
   292 		{
   293 		User::Leave(KErrNotSupported);
   294 		}
   295 	
   296 	RSurfaceManager::TSurfaceCreationAttributesBuf attribBuf;
   297 	RSurfaceManager::TSurfaceCreationAttributes& attrib = attribBuf();
   298 	attrib.iSize = aSize;
   299 	attrib.iBuffers = aNumBuffers;
   300 	attrib.iPixelFormat = VideoRendererUtil::ConvertUncompressedVideoFormatToUidPixelFormatL(aFormat);
   301 	attrib.iStride = aSize.iWidth * VideoRendererUtil::BytesPerPixelL(attrib.iPixelFormat);
   302 	attrib.iOffsetToFirstBuffer = 0;
   303 	attrib.iAlignment = 4;
   304 	attrib.iContiguous = ETrue;
   305 	attrib.iMappable = ETrue;
   306 
   307 	User::LeaveIfError(iSurfaceManager.CreateSurface(attribBuf, iSurfaceId));
   308 	
   309 	// Push surfaceId to cleanup stack in case buffer manager constructor leave.
   310 	// SurfaceId needs to be pushed because the renderer state is inconsistent 
   311 	// if buffer manager creation leave
   312 	CleanupReleasePushL(*this);
   313 	
   314 	RChunk chunk;
   315 	User::LeaveIfError(iSurfaceManager.MapSurface(iSurfaceId, chunk));
   316 	CleanupClosePushL(chunk);
   317 	
   318 	RSurfaceManager::TInfoBuf infoBuf;
   319 	User::LeaveIfError(iSurfaceManager.SurfaceInfo(iSurfaceId, infoBuf));
   320 
   321 	// Register must take place after 'this' pushed onto cleanupstack just in case 
   322 	// function leaves after registration
   323 	RegisterSurfaceL();
   324 	
   325 	// prepare the renderer relay
   326 	RSurfaceManager::TSurfaceInfoV01 info = infoBuf();
   327 	if (iTimed)
   328 		{
   329 		TRequestStatus updateRequest = KRequestPending;
   330 		TRequestStatus logonRequest = KRequestPending;
   331 
   332 		// While a function call is in progress this thread is suspended
   333 		// and the undertaker will not catch panics, listen for these here
   334 		iRendererThread.Logon(logonRequest);
   335 
   336 		// Send request to renderer thread
   337 		iRendererRelay->PrepareL(iSurfaceId, info.iBuffers, &updateRequest);
   338 		User::WaitForRequest(logonRequest, updateRequest);
   339 
   340 		if(logonRequest != KRequestPending)
   341 			{
   342 			// renderer thread got panic from surface update session, so panic client
   343 			TInt reason = iRendererThread.ExitReason();
   344 			TExitCategoryName category = iRendererThread.ExitCategory();
   345 			User::Panic(category,reason);
   346 			}
   347 
   348 		// Thread is still alive and well
   349 		iRendererThread.LogonCancel(logonRequest);
   350 		User::WaitForRequest(logonRequest); // Consume the signal
   351 		
   352 		// leave if memory allocation failed in renderer thread
   353 		User::LeaveIfError(updateRequest.Int());
   354 
   355 		__ASSERT_DEBUG(updateRequest != KRequestPending, User::Panic(_L("CVR::UpdateBuffer"), KErrCorrupt));
   356 		}
   357 	else
   358 		{
   359 		iRendererRelay->PrepareL(iSurfaceId, info.iBuffers, NULL);
   360 		}
   361 	
   362 	// find the buffer offsets
   363 	RArray<TInt> offsets;
   364 	CleanupClosePushL(offsets);
   365 	
   366 	TInt offsetInChunk = 0;
   367 	for (TInt i = 0; i < info.iBuffers; ++i)
   368 		{
   369 		User::LeaveIfError(iSurfaceManager.GetBufferOffset(iSurfaceId, i, offsetInChunk));
   370 		offsets.AppendL(offsetInChunk);
   371 		}
   372 	
   373 	// finally, create buffer manager
   374 	iBufferManager = CRendererBufferManager::NewL(info, offsets, chunk, iTimed);
   375 
   376 	CleanupStack::PopAndDestroy(&offsets);
   377 	CleanupStack::Pop(&chunk);
   378 	CleanupStack::Pop(this); // surfaceId
   379 	
   380 	aSurface = iSurfaceId;
   381 	iRendererRelay->SetBufferManager(iBufferManager);
   382 	
   383 	for (TInt i = 0; i < info.iBuffers; ++i)
   384 		{
   385 		// notify observer once for each buffer created
   386 		iObserver.MvroVideoBufferAvailable();
   387 		}
   388 	}
   389 
   390 /** 
   391  Helper function to unregister and release surfaceId when CreateSurfaceL leave
   392  @internalComponent
   393  */
   394 void CVideoRenderer::Release()
   395 	{
   396 	UnregisterSurface();
   397 	iSurfaceManager.CloseSurface(iSurfaceId);
   398 	}
   399 
   400 /** 
   401  Helper function to register surfaces for all displays
   402  @internalComponent
   403  */
   404 void CVideoRenderer::RegisterSurfaceL()
   405 	{
   406 	TInt screens = iWsSession.NumberOfScreens();
   407 	TInt error;
   408 	
   409 	for(TInt i=0; i<screens; i++)
   410 		{
   411 		error = iWsSession.RegisterSurface(i, iSurfaceId);
   412 		User::LeaveIfError(error);
   413 		}
   414 	}
   415 
   416 /** 
   417  Helper function to unregister surafces for all displays
   418  @internalComponent
   419  */
   420 void CVideoRenderer::UnregisterSurface()
   421 	{
   422 	TInt screens = iWsSession.NumberOfScreens();
   423 	
   424 	for(TInt i=0; i<screens; i++)
   425 		{
   426 		iWsSession.UnregisterSurface(i, iSurfaceId);
   427 		}
   428 	}
   429 	
   430 /**
   431 Destroys a surface previously created with CreateSurfaceL() and unregisters it with the windows
   432 server for all displays. The client must stop using the surface before calling this method. All
   433 buffers retrieved with NextBuffer() will become invalid and can no longer be used.
   434 
   435 @param aSurface The surface to delete
   436 */
   437 EXPORT_C void CVideoRenderer::DestroySurface(const TSurfaceId& aSurface)
   438 	{
   439 	if (aSurface != iSurfaceId || iBufferManager == NULL)
   440 		{
   441 		// surface id is not expected or surface has not been created
   442 		return;
   443 		}
   444 	
   445 	// cancel the active objects that use the surface
   446 	if (iTimed)
   447 		{
   448 		TRequestStatus request = KRequestPending;
   449 		TRequestStatus logonRequest = KRequestPending;
   450 
   451 		// While a function call is in progress this thread is suspended
   452 		// and the undertaker will not catch panics, listen for these here
   453 		iRendererThread.Logon(logonRequest);
   454 
   455 		// Send request to renderer thread
   456 		iRendererRelay->DestroySurface(&request);
   457 		User::WaitForRequest(logonRequest, request);
   458 
   459 		if(logonRequest != KRequestPending)
   460 			{
   461 			// renderer thread got panic from surface update session, so panic client
   462 			TInt reason = iRendererThread.ExitReason();
   463 			TExitCategoryName category = iRendererThread.ExitCategory();
   464 			User::Panic(category,reason);
   465 			}
   466 
   467 		// Thread is still alive and well
   468 		iRendererThread.LogonCancel(logonRequest);
   469 		User::WaitForRequest(logonRequest); // Consume the signal
   470 
   471 		__ASSERT_DEBUG(request != KRequestPending, User::Panic(_L("CVR::DestroySurface"), KErrCorrupt));
   472 		}
   473 	else
   474 		{
   475 		iRendererRelay->DestroySurface(NULL);
   476 		}
   477 
   478 	UnregisterSurface();
   479 	
   480 	delete iBufferManager;
   481 	iBufferManager = NULL;
   482 	iRendererRelay->SetBufferManager(NULL);
   483 	
   484 	TInt err = iSurfaceManager.CloseSurface(iSurfaceId);
   485 	DEBUGPRINT2(_L("RSurfaceManager::CloseSurface returned with %d"), err);
   486 	}
   487 
   488 /**
   489 Retrieves the next free buffer from the renderer. The client can write data to 
   490 the buffer and update it to the screen using UpdateBuffer(). The buffer will 
   491 remain accessible to the client until it returns it to the renderer with 
   492 UpdateBuffer() or ReleaseBuffer(), the surface is destroyed (DestroySurface()), 
   493 or the renderer object is deleted.
   494 
   495 If no free buffers are available the client should wait for a 
   496 MvroVideoBufferAvailable() callback instead of polling for free buffers continuously.
   497 
   498 @return A pointer to the free buffer, or NULL if no free buffers are available.
   499 */
   500 EXPORT_C TVideoFrameBuffer* CVideoRenderer::NextBuffer()
   501 	{
   502 	if (iBufferManager)
   503 		{
   504 		return iBufferManager->NextBuffer();
   505 		}
   506 	
   507 	// surface had not been created, return null
   508 	return NULL;
   509 	}
   510 
   511 /**
   512 Updates a surface buffer on the display. The buffer must have previously been 
   513 retrieved with NextBuffer() and must contain a valid video picture. The 
   514 renderer uses statistics from previous video frame rendering times to render 
   515 the picture as close to the requested presentation time as possible.
   516 
   517 @param aBuffer The buffer to update
   518 @param aPresentationTime The system clock time when the buffer should be 
   519 			visible on the display. If the time is zero the renderer will 
   520 			update the buffer as soon as possible without further synchronization.
   521 */
   522 EXPORT_C void CVideoRenderer::UpdateBuffer(TVideoFrameBuffer* aBuffer, const TTime& aPresentationTime)
   523 	{
   524 	__ASSERT_DEBUG(aBuffer != NULL, User::Panic(_L("CVR::UpdateBuffer"), KErrArgument));
   525 	DEBUGPRINT3(_L("CVideoRenderer::UpdateBuffer entered with bufId=%d, aPresentationTime=%Ld"), 
   526 				aBuffer->BufferId() , aPresentationTime.Int64());
   527 
   528 	// ignore request if a surface has not been created or has been destroyed
   529 	if (!iBufferManager)
   530 		{
   531 		return;
   532 		}
   533 
   534 	if (iTimed)
   535 		{
   536 		TRequestStatus updateRequest = KRequestPending;
   537 		TRequestStatus logonRequest = KRequestPending;
   538 
   539 		// While a function call is in progress this thread is suspended
   540 		// and the undertaker will not catch panics, listen for these here
   541 		iRendererThread.Logon(logonRequest);
   542 
   543 		// Send request to renderer thread
   544 		iRendererRelay->UpdateBuffer(aBuffer, aPresentationTime, &updateRequest);
   545 		User::WaitForRequest(logonRequest, updateRequest);
   546 
   547 		if(logonRequest != KRequestPending)
   548 			{
   549 			// renderer thread got panic from surface update session, so panic client
   550 			TInt reason = iRendererThread.ExitReason();
   551 			TExitCategoryName category = iRendererThread.ExitCategory();
   552 			User::Panic(category,reason);
   553 			}
   554 
   555 		// Thread is still alive and well
   556 		iRendererThread.LogonCancel(logonRequest);
   557 		User::WaitForRequest(logonRequest); // Consume the signal
   558 
   559 		__ASSERT_DEBUG(updateRequest != KRequestPending, User::Panic(_L("CVR::UpdateBuffer"), KErrCorrupt));
   560 		}
   561 	else
   562 		{
   563 		iRendererRelay->UpdateBuffer(aBuffer, aPresentationTime, NULL);
   564 		}
   565 	}
   566 
   567 /**
   568 Releases a buffer without updating it to the display.
   569 The buffer must have previously been retrieved with NextBuffer() and 
   570 UpdateBuffer() is not called yet, otherwise the release request is ignored.
   571 
   572 @param aBuffer The buffer to release
   573 */
   574 EXPORT_C void CVideoRenderer::ReleaseBuffer(TVideoFrameBuffer* aBuffer)
   575 	{
   576 	// ignore if if a surface has not been created or has beed destroyed
   577 	if (!iBufferManager || !aBuffer)
   578 		{
   579 		return;
   580 		}
   581 
   582 	if (iBufferManager->ReleaseBuffer(aBuffer))
   583 		{
   584 		// a buffer is releaseed, notify observer
   585 		iObserver.MvroVideoBufferAvailable();
   586 		}
   587 	}