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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
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"
25 _LIT(KVideoRendererThreadName, "VideoRendererThread");
26 _LIT(KResourceFileName, "Z:\\resource\\videorenderer\\videorenderer.rsc");
28 const TThreadPriority KSubThreadPriority = EPriorityRealTime;
29 const TInt KSubThreadStackSize = KDefaultStackSize; // 0x2000 = 8k
34 Factory method. Creates a new video renderer instance.
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.
41 EXPORT_C CVideoRenderer* CVideoRenderer::NewL(MVideoRendererObserver& aObserver, TBool aTimed)
43 CVideoRenderer* self = new (ELeave) CVideoRenderer(aObserver, aTimed);
44 CleanupStack::PushL(self);
46 CleanupStack::Pop(self);
54 CVideoRenderer::CVideoRenderer(MVideoRendererObserver& aObserver, TBool aTimed)
55 :iObserver(aObserver), iTimed(aTimed)
60 Second phase constructor
63 void CVideoRenderer::ConstructL()
65 User::LeaveIfError(iSurfaceManager.Open());
67 User::LeaveIfError(iWsSession.Connect());
69 CResourceFileReader* reader = CResourceFileReader::NewLC(KResourceFileName);
70 reader->ReadSupportedFormatL(iSupportedFormat);
74 // Create a high priority thread for timed mode
76 // get timer info for timed mode
79 reader->ReadTimerInfoL(defaultDelay, maxDelay);
81 //Get a reference to this thread's heap
82 RHeap& thisHeap = User::Heap();
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
89 //Get the id of this thread
91 TThreadId thisThreadId = thisThread.Id();
92 param.iMainThreadId = thisThreadId;
94 //Get a request to signal for setup completion
95 TRequestStatus setupComplete = KRequestPending;
96 param.iSetupComplete = &setupComplete;
98 //current time and the "this" pointer for a unique key
99 _LIT(KFormatString,"%S.%020Lu.%08X");
103 threadName.Format(KFormatString, &KVideoRendererThreadName, now.Int64(), reinterpret_cast<TUint>(this));
105 //Create a new thread using the same heap as this thread
106 TInt result = iRendererThread.Create(threadName,
107 ThreadCreateFunction,
111 User::LeaveIfError(result);
113 //Run the thread under high priority
114 iRendererThread.SetPriority(KSubThreadPriority);
116 //Wait for thread startup to complete
117 TRequestStatus threadStatus = KRequestPending;
118 iRendererThread.Logon(threadStatus);
121 iRendererThread.Resume();
122 User::WaitForRequest(threadStatus, setupComplete);
123 if(threadStatus != KRequestPending)
125 //Thread creation failed
126 TInt reason = iRendererThread.ExitReason();
127 DEBUGPRINT3(_L("Renderer thread died with type=%d, reason=%d"), iRendererThread.ExitType(), reason);
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
136 __ASSERT_DEBUG(iRendererRelay != NULL, User::Panic(_L("CVR::ConstructL"), KErrCorrupt));
137 iRendererRelay->SetRendererThread(&iRendererThread);
138 iRendererRelay->SetTimerInfo(defaultDelay, maxDelay);
140 iThreadCreated = ETrue;
141 User::LeaveIfError(setupComplete.Int());
143 //Create a listener that will monitor the thread
144 iRendererThreadUndertaker = CThreadUndertaker::NewL(iRendererThread);
148 iRendererRelay = CRendererRelay::NewL(iObserver);
151 CleanupStack::PopAndDestroy(reader);
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.
160 @param aPtr Parameters to be used for creating the thread.
161 @return The error code for thread termination.
163 TInt CVideoRenderer::ThreadCreateFunction(TAny* aPtr)
165 TInt error = KErrNone;
167 // Create a cleanup stack for the thread
168 CTrapCleanup* cleanupStack = CTrapCleanup::New();
171 if(error == KErrNone)
173 TRAP(error, ThreadTrapFunctionL(aPtr));
178 error = KErrNoMemory;
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.
191 @param aPtr A pointer to a TThreadRelayParam object containing the startup parameters
194 void CVideoRenderer::ThreadTrapFunctionL(TAny* aPtr)
196 //Create an active scheduler for the thread
197 CActiveScheduler* scheduler = new (ELeave) CActiveScheduler;
198 CleanupStack::PushL(scheduler);
199 CActiveScheduler::Install(scheduler);
201 //Call a Factory function to create a CSubThreadRelay derived object
202 TThreadRelayParam* relayParam;
203 relayParam = static_cast<TThreadRelayParam*>(aPtr);
205 CRendererThreadRelay* threadRelay = CRendererThreadRelay::NewL(*relayParam->iObserver, relayParam->iMainThreadId);
206 CleanupStack::PushL(threadRelay);
208 //Send a pointer to the sub thread relay back to the main thread
209 *relayParam->iThreadRelay = threadRelay;
211 threadRelay->Start();
212 threadRelay->SignalSetupComplete(relayParam->iSetupComplete);
214 CActiveScheduler::Start();
216 CleanupStack::PopAndDestroy(2,scheduler); //threadRelay, scheduler
222 EXPORT_C CVideoRenderer::~CVideoRenderer()
225 iSupportedFormat.Close();
227 // first, stop all active objects that may use any buffers
230 delete iRendererThreadUndertaker;
232 // this also delete CRendererThreadRelay
233 TRequestStatus terminateRequest = KRequestPending;
234 iRendererRelay->Terminate(terminateRequest);
235 User::WaitForRequest(terminateRequest);
236 iThreadCreated = EFalse;
238 iRendererThread.Close();
240 else if (iTimed == EFalse)
242 delete iRendererRelay;
245 // next, close surface if still open
248 delete iBufferManager;
249 iSurfaceManager.CloseSurface(iSurfaceId);
253 iSurfaceManager.Close();
257 Retrieves the list of supported surface formats from the renderer.
259 @param aArray Supported video formats. The renderer will populate the array with supported formats.
260 @leave KErrNoMemory Out of memory
262 EXPORT_C void CVideoRenderer::GetSupportedFormatsL(RArray<TUncompressedVideoFormat>& aArray)
264 aArray.Reset(); // first clear the old data
266 TInt count = iSupportedFormat.Count();
267 for (TInt i = 0; i < count; ++i)
269 aArray.AppendL(iSupportedFormat[i]);
274 Creates a new surface for video rendering and registers it with the windows server for all displays.
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
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
285 EXPORT_C void CVideoRenderer::CreateSurfaceL(const TSize& aSize, TInt aNumBuffers, const TUncompressedVideoFormat& aFormat, TSurfaceId& aSurface)
287 if (iBufferManager != NULL)
289 User::Leave(KErrInUse);
291 else if (aNumBuffers <= 0)
293 User::Leave(KErrNotSupported);
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;
307 User::LeaveIfError(iSurfaceManager.CreateSurface(attribBuf, iSurfaceId));
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);
315 User::LeaveIfError(iSurfaceManager.MapSurface(iSurfaceId, chunk));
316 CleanupClosePushL(chunk);
318 RSurfaceManager::TInfoBuf infoBuf;
319 User::LeaveIfError(iSurfaceManager.SurfaceInfo(iSurfaceId, infoBuf));
321 // Register must take place after 'this' pushed onto cleanupstack just in case
322 // function leaves after registration
325 // prepare the renderer relay
326 RSurfaceManager::TSurfaceInfoV01 info = infoBuf();
329 TRequestStatus updateRequest = KRequestPending;
330 TRequestStatus logonRequest = KRequestPending;
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);
336 // Send request to renderer thread
337 iRendererRelay->PrepareL(iSurfaceId, info.iBuffers, &updateRequest);
338 User::WaitForRequest(logonRequest, updateRequest);
340 if(logonRequest != KRequestPending)
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);
348 // Thread is still alive and well
349 iRendererThread.LogonCancel(logonRequest);
350 User::WaitForRequest(logonRequest); // Consume the signal
352 // leave if memory allocation failed in renderer thread
353 User::LeaveIfError(updateRequest.Int());
355 __ASSERT_DEBUG(updateRequest != KRequestPending, User::Panic(_L("CVR::UpdateBuffer"), KErrCorrupt));
359 iRendererRelay->PrepareL(iSurfaceId, info.iBuffers, NULL);
362 // find the buffer offsets
363 RArray<TInt> offsets;
364 CleanupClosePushL(offsets);
366 TInt offsetInChunk = 0;
367 for (TInt i = 0; i < info.iBuffers; ++i)
369 User::LeaveIfError(iSurfaceManager.GetBufferOffset(iSurfaceId, i, offsetInChunk));
370 offsets.AppendL(offsetInChunk);
373 // finally, create buffer manager
374 iBufferManager = CRendererBufferManager::NewL(info, offsets, chunk, iTimed);
376 CleanupStack::PopAndDestroy(&offsets);
377 CleanupStack::Pop(&chunk);
378 CleanupStack::Pop(this); // surfaceId
380 aSurface = iSurfaceId;
381 iRendererRelay->SetBufferManager(iBufferManager);
383 for (TInt i = 0; i < info.iBuffers; ++i)
385 // notify observer once for each buffer created
386 iObserver.MvroVideoBufferAvailable();
391 Helper function to unregister and release surfaceId when CreateSurfaceL leave
394 void CVideoRenderer::Release()
397 iSurfaceManager.CloseSurface(iSurfaceId);
401 Helper function to register surfaces for all displays
404 void CVideoRenderer::RegisterSurfaceL()
406 TInt screens = iWsSession.NumberOfScreens();
409 for(TInt i=0; i<screens; i++)
411 error = iWsSession.RegisterSurface(i, iSurfaceId);
412 User::LeaveIfError(error);
417 Helper function to unregister surafces for all displays
420 void CVideoRenderer::UnregisterSurface()
422 TInt screens = iWsSession.NumberOfScreens();
424 for(TInt i=0; i<screens; i++)
426 iWsSession.UnregisterSurface(i, iSurfaceId);
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.
435 @param aSurface The surface to delete
437 EXPORT_C void CVideoRenderer::DestroySurface(const TSurfaceId& aSurface)
439 if (aSurface != iSurfaceId || iBufferManager == NULL)
441 // surface id is not expected or surface has not been created
445 // cancel the active objects that use the surface
448 TRequestStatus request = KRequestPending;
449 TRequestStatus logonRequest = KRequestPending;
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);
455 // Send request to renderer thread
456 iRendererRelay->DestroySurface(&request);
457 User::WaitForRequest(logonRequest, request);
459 if(logonRequest != KRequestPending)
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);
467 // Thread is still alive and well
468 iRendererThread.LogonCancel(logonRequest);
469 User::WaitForRequest(logonRequest); // Consume the signal
471 __ASSERT_DEBUG(request != KRequestPending, User::Panic(_L("CVR::DestroySurface"), KErrCorrupt));
475 iRendererRelay->DestroySurface(NULL);
480 delete iBufferManager;
481 iBufferManager = NULL;
482 iRendererRelay->SetBufferManager(NULL);
484 TInt err = iSurfaceManager.CloseSurface(iSurfaceId);
485 DEBUGPRINT2(_L("RSurfaceManager::CloseSurface returned with %d"), err);
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.
495 If no free buffers are available the client should wait for a
496 MvroVideoBufferAvailable() callback instead of polling for free buffers continuously.
498 @return A pointer to the free buffer, or NULL if no free buffers are available.
500 EXPORT_C TVideoFrameBuffer* CVideoRenderer::NextBuffer()
504 return iBufferManager->NextBuffer();
507 // surface had not been created, return null
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.
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.
522 EXPORT_C void CVideoRenderer::UpdateBuffer(TVideoFrameBuffer* aBuffer, const TTime& aPresentationTime)
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());
528 // ignore request if a surface has not been created or has been destroyed
536 TRequestStatus updateRequest = KRequestPending;
537 TRequestStatus logonRequest = KRequestPending;
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);
543 // Send request to renderer thread
544 iRendererRelay->UpdateBuffer(aBuffer, aPresentationTime, &updateRequest);
545 User::WaitForRequest(logonRequest, updateRequest);
547 if(logonRequest != KRequestPending)
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);
555 // Thread is still alive and well
556 iRendererThread.LogonCancel(logonRequest);
557 User::WaitForRequest(logonRequest); // Consume the signal
559 __ASSERT_DEBUG(updateRequest != KRequestPending, User::Panic(_L("CVR::UpdateBuffer"), KErrCorrupt));
563 iRendererRelay->UpdateBuffer(aBuffer, aPresentationTime, NULL);
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.
572 @param aBuffer The buffer to release
574 EXPORT_C void CVideoRenderer::ReleaseBuffer(TVideoFrameBuffer* aBuffer)
576 // ignore if if a surface has not been created or has beed destroyed
577 if (!iBufferManager || !aBuffer)
582 if (iBufferManager->ReleaseBuffer(aBuffer))
584 // a buffer is releaseed, notify observer
585 iObserver.MvroVideoBufferAvailable();