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