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.
22 #include <graphics/suerror.h>
23 #include "rendererrelay.h"
24 #include "buflistener.h"
25 #include "buffermanager.h"
26 #include "videoframebuffer.h"
27 #include "rendererutil.h"
28 #include "renderertimer.h"
30 /** static factory contruction */
31 CRendererRelay* CRendererRelay::NewL(MVideoRendererObserver& aObserver)
33 CRendererRelay* self = new (ELeave) CRendererRelay(aObserver);
34 CleanupStack::PushL(self);
36 CleanupStack::Pop(self);
40 CRendererRelay::CRendererRelay(MVideoRendererObserver& aObserver)
41 :iObserver(aObserver),
42 iBufAvailListeners(CBufAvailListener::iOffset),
43 iAvailListenersIter(iBufAvailListeners)
47 void CRendererRelay::ConstructL()
49 User::LeaveIfError(iSurfaceUpdateSession.Connect());
50 User::LeaveIfError(HAL::Get(HAL::EFastCounterFrequency, iFastCounterFrequency));
51 User::LeaveIfError(HAL::Get(HAL::EFastCounterCountsUp, iFastCounterCountsUp));
55 CRendererRelay::~CRendererRelay()
57 delete iDisplayListener;
59 iAvailListenersIter.SetToFirst();
60 CBufAvailListener* listener = NULL;
61 while ((listener = iAvailListenersIter++) != NULL)
63 listener->iDblQueLink.Deque();
67 iSurfaceUpdateSession.Close();
70 /** Update buffer manager pointer in this class and listeners */
71 void CRendererRelay::SetBufferManager(CRendererBufferManager* aBufManager)
73 iBufManager = aBufManager;
75 // change buffer manager pointer for listeners
76 iAvailListenersIter.SetToFirst();
77 CBufAvailListener* listener = NULL;
78 while ((listener = iAvailListenersIter++) != NULL)
80 listener->SetBufferManager(aBufManager);
82 if (!aBufManager && listener->IsAdded())
86 else if (aBufManager && !listener->IsAdded())
88 CActiveScheduler::Add(listener);
93 /** Return the next unused CBufAvailListener. */
94 CBufAvailListener* CRendererRelay::BufAvailListener()
96 CBufAvailListener* listener = NULL;
97 iAvailListenersIter.SetToFirst();
98 while ((listener = iAvailListenersIter++) != NULL)
100 if (listener->IsActive() == EFalse)
102 // Move to end so that the next search is a bit faster
103 listener->iDblQueLink.Deque();
104 iBufAvailListeners.AddLast(*listener);
109 // Should not reach here as the number of listeners is same as number of buffer
110 __ASSERT_DEBUG(EFalse, User::Panic(_L("CRR::BufAvailListener"), KErrUnknown));
114 /** Set update parameter and create listeners */
115 void CRendererRelay::PrepareL(const TSurfaceId& aSurfaceId, TInt aNumBuffers, TRequestStatus* /*aRequestStatus*/)
117 iUpdateSubmitted = EFalse;
118 iSurfaceId = aSurfaceId;
120 if (!iDisplayListener)
122 iDisplayListener = CBufDisplayListener::NewL(iObserver, iSurfaceUpdateSession, *this, iFastCounterFrequency, iFastCounterCountsUp);
125 TInt numListeners = 0;
126 iAvailListenersIter.SetToFirst();
127 while (iAvailListenersIter++ != NULL)
132 CBufAvailListener* listener = NULL;
133 // create new listeners if there are more numBuffers than listeners
134 for (TInt i = numListeners; i < aNumBuffers; ++i)
136 listener = CBufAvailListener::NewL(iObserver, iSurfaceUpdateSession);
137 iBufAvailListeners.AddFirst(*listener);
141 /** Handle update buffer request in non-timed mode */
142 void CRendererRelay::UpdateBuffer(TVideoFrameBuffer* aBuffer, const TTime& aPresentationTime, TRequestStatus* /* aRequestStatus */)
144 __ASSERT_DEBUG(iBufManager != NULL, User::Panic(_L("CRR::UpdateBuffer"), KErrNotReady));
146 iBufManager->UpdateBuffer(aBuffer, aPresentationTime);
148 if (iUpdateSubmitted == EFalse)
150 // an update haven't been submitted, submit one now
155 /** Callback from display listener that a buffer has been displayed or skipped*/
156 void CRendererRelay::BufferDisplayed(TBool aDisplayed, TInt64 aDelay)
158 DEBUGPRINT3(_L("CRendererRelay::BufferDisplayed entered aDisplayed=%d, aDelay=%Ld"), aDisplayed, aDelay);
160 iUpdateSubmitted = EFalse;
164 // rendering in timed mode
167 // update delay value
174 // rendering in non timed mode
179 /** Submit the next buffer that is waiting to be submitted in non-timed mode*/
180 void CRendererRelay::SubmitBuffer()
182 DEBUGPRINT1(_L("CRendererRelay::SubmitBuffer entered"));
183 __ASSERT_DEBUG(iBufManager != NULL, User::Panic(_L("CRR::SubmitBuffer"), KErrCorrupt));
184 __ASSERT_DEBUG(iUpdateSubmitted == EFalse, User::Panic(_L("CRR::SubmitBuffer"), KErrGeneral));
187 TVideoFrameBuffer* buf = iBufManager->WaitingBuffer(ETrue, lastBuf);
188 TInt numBufferSkipped = 0;
192 TUint32 fastCounter = User::FastCounter();
195 TInt bufId = buf->BufferId();
197 DEBUGPRINT4(_L("bufId=%d presTime=%Ld, now=%Ld"),
198 bufId, buf->PresentationTime().Int64(), now.Int64());
200 // submit the buffer for update if presntation time is not in past
201 // or if the buffer is the last in queue or presentation time is zero
202 if (buf->PresentationTime() >= now || lastBuf || buf->PresentationTime().Int64() == 0)
204 DoUpdateBuffer(bufId, now, fastCounter);
208 // The buffer presentation time occurs in past if codeflow reaches here.
209 // Change the buffer status to available and notify observer about the skipped buffer
210 iBufManager->BufferAvailable(bufId);
211 iObserver.MvroBufferSkipped(bufId);
215 buf = iBufManager->WaitingBuffer(ETrue, lastBuf);
218 //notifiy observer about the available buffers
219 for (TInt i = 0; i < numBufferSkipped; ++i)
221 iObserver.MvroVideoBufferAvailable();
225 /** Set a pointer to renderer timer in timed mode */
226 void CRendererRelay::SetRendererTimer(CRendererTimer* aRendererTimer)
228 iRendererTimer = aRendererTimer;
231 /** Callback function when a renderer timer has expired */
232 void CRendererRelay::RendererTimerExpired()
237 /** Submit the next buffer in timed mode */
238 void CRendererRelay::SubmitBufferTimed()
240 DEBUGPRINT1(_L("CRendererRelay::SubmitBufferTimed entered"));
242 __ASSERT_DEBUG(iBufManager != NULL, User::Panic(_L("CRR::SubmitBufferTimed"), KErrCorrupt));
243 __ASSERT_DEBUG(iUpdateSubmitted == EFalse,
244 User::Panic(_L("CRR::SubmitBufferTimed"), KErrGeneral));
245 __ASSERT_DEBUG(iRendererTimer && iRendererTimer->IsActive() == EFalse,
246 User::Panic(_L("CRR::SubmitBufferTimed"), KErrInUse));
249 TVideoFrameBuffer* buf = iBufManager->WaitingBuffer(EFalse, lastBuf);
250 TInt numBufferSkipped = 0;
254 TUint32 fastCounter = User::FastCounter();
257 TInt bufId = buf->BufferId();
258 TTimeIntervalMicroSeconds deltaFromNow = buf->PresentationTime().MicroSecondsFrom(now);
261 if (buf->PresentationTime().Int64() != 0)
263 // presentation time is not zero, calculate wait time. Otherwise, wait time is zero.
264 waitTime = deltaFromNow.Int64() - iDelay;
267 DEBUGPRINT4(_L("bufId=%d presTime=%Ld, now=%Ld"),
268 bufId, buf->PresentationTime().Int64(), now.Int64());
270 // submit the buffer for update if presntation time is not in past
271 // or if the buffer is the last in queue
274 iRendererTimer->Start(waitTime);
277 else if (buf->PresentationTime().Int64() == 0 ||
278 deltaFromNow.Int64() + iMaxDelay >= 0 ||
281 // if presentation time is zero (waitTime is not used for checking because it may be zero from calculation)
282 // or the frame is within maximun allowed delay, i.e. (presentation time + max delay >= now)
283 // or submission time has passed but this is the last buf, submit the buffer now
285 iBufManager->BufferSubmitted(buf);
286 DoUpdateBuffer(bufId, now, fastCounter);
290 // The buffer presentation time has passed maxDelay if codeflow reaches here, skip the buffer.
291 // Change the buffer status to available and notify observer
292 iBufManager->BufferAvailable(bufId);
293 iObserver.MvroBufferSkipped(bufId);
297 buf = iBufManager->WaitingBuffer(EFalse, lastBuf);
300 //notifiy observer about the available buffers
301 for (TInt i = 0; i < numBufferSkipped; ++i)
303 iObserver.MvroVideoBufferAvailable();
308 Submits a buffer to be updated at the specified time.
311 void CRendererRelay::DoUpdateBuffer(TInt aBufferId, const TTime& aTime, TUint32 aFastCounter)
313 CBufAvailListener* availListener = BufAvailListener();
315 availListener->Start(aBufferId);
316 iDisplayListener->Start(aBufferId, aTime, aFastCounter);
318 TInt err = iSurfaceUpdateSession.SubmitUpdate(KAllScreens, iSurfaceId, aBufferId);
320 DEBUGPRINT2(_L("SubmitUpdate return %d"), err);
322 // error will also be returned from listener, so the next submit updated will be triggered by display listener
323 iUpdateSubmitted = ETrue;
327 Return ETrue if an update has been submitted and the display notification
328 haven't been received yet. i.e. Need to wait till for listener callback before
329 the next buffer can be submitted.
331 TBool CRendererRelay::UpdateSubmitted()
333 return iUpdateSubmitted;
336 /** return the delay for surface update */
337 TInt64 CRendererRelay::Delay()
342 /** Cancel all update notification when a surface is destroyed */
343 void CRendererRelay::DestroySurface(TRequestStatus* /* aRequestStatus */ )
345 iSurfaceUpdateSession.CancelAllUpdateNotifications();
348 /* This function is not used */
349 void CRendererRelay::SetRendererThread(RThread* /* aRendererThread */)
351 __ASSERT_DEBUG(EFalse, User::Panic(_L("CRR::SetRendererThread"), KErrUnknown));
354 /* This function is not used */
355 void CRendererRelay::Terminate(TRequestStatus& /* aRequestStatus */)
357 __ASSERT_DEBUG(EFalse, User::Panic(_L("CRR::Terminate"), KErrUnknown));
360 /** Set timer info for timed mode, this function is not called in non-timed mode */
361 void CRendererRelay::SetTimerInfo(TInt64 aDefaultDelay, TInt64 aMaxDelay)
363 iDelay = aDefaultDelay;
364 iMaxDelay = aMaxDelay;
367 /** Two-phased constructor. */
368 CRendererThreadRelay* CRendererThreadRelay::NewL(MVideoRendererObserver& aObserver, TThreadId aMainThreadId)
370 CRendererThreadRelay* self = new (ELeave) CRendererThreadRelay(aObserver);
371 CleanupStack::PushL(self);
372 self->ConstructL(aMainThreadId);
373 CleanupStack::Pop(self);
377 CRendererThreadRelay::CRendererThreadRelay(MVideoRendererObserver& aObserver)
378 : CActive(EPriorityStandard), iObserver(aObserver)
380 CActiveScheduler::Add(this);
383 /** Second-phase constructor */
384 void CRendererThreadRelay::ConstructL(TThreadId aMainThreadId)
386 User::LeaveIfError(iMainThread.Open(aMainThreadId));
387 iRendererRelay = CRendererRelay::NewL(*this);
388 iRendererTimer = CRendererTimer::NewL(*iRendererRelay);
389 iRendererRelay->SetRendererTimer(iRendererTimer);
394 This active object is always cancelled by main thread so no need to cancel this active object here
396 CRendererThreadRelay::~CRendererThreadRelay()
398 delete iRendererCallbackListener;
399 delete iRendererTimer;
400 delete iRendererRelay;
404 void CRendererThreadRelay::DoCancel()
406 // Don't need to do anything, will be stopped by main thread
409 /** Function for making the initial request */
410 void CRendererThreadRelay::Start()
412 __ASSERT_DEBUG(IsActive() == EFalse, User::Panic(_L("CRTR::Start"), KErrInUse));
413 iStatus = KRequestPending;
417 /** Handle requests from main thread */
418 void CRendererThreadRelay::RunL()
420 __ASSERT_DEBUG(iStatus == KErrNone, User::Panic(_L("CRTR::RunL"), KErrUnknown));
421 TInt result = KErrNone;
423 if (iFunctionCode == ESubmitUpdate)
426 RunUpdateBuffer(iBuffer, iPresentationTime);
428 else if (iFunctionCode == EDestroySurface)
433 else if (iFunctionCode == EPrepare)
436 TRAP(result, RunPrepareL());
438 else if (iFunctionCode == ESetBufferManager)
441 RunSetBufferManager();
443 else // iFunctionCode == ETermination
445 CActiveScheduler::Stop();
448 TRequestStatus *status = iCallingStatus;
449 iMainThread.RequestComplete(status, result);
452 /** Send a signal to the main thread to indicate that the thread startup was successful. */
453 void CRendererThreadRelay::SignalSetupComplete(TRequestStatus* aSetupComplete)
455 iMainThread.RequestComplete(aSetupComplete, KErrNone);
458 /** Send update buffer request from main thread to renderer thread */
459 void CRendererThreadRelay::UpdateBuffer(TVideoFrameBuffer* aBuffer, const TTime& aPresentationTime, TRequestStatus* aRequestStatus)
461 __ASSERT_DEBUG(aRequestStatus != NULL, User::Panic(_L("CRTR::UpdateBuffer"), KErrArgument));
463 // set function parameters
464 iCallingStatus = aRequestStatus;
465 iFunctionCode = ESubmitUpdate;
467 iPresentationTime = aPresentationTime;
469 // send request to renderer thread
470 TRequestStatus* rendererRequest = Status();
471 iRendererThread->RequestComplete(rendererRequest, KErrNone);
474 /** Send terminate renderer thread request from main thread to renderer thread */
475 void CRendererThreadRelay::Terminate(TRequestStatus& aRequestStatus)
477 iCallingStatus = &aRequestStatus;
478 iFunctionCode = ETermination;
480 if (iRendererCallbackListener)
482 iRendererCallbackListener->Cancel();
485 // send request to renderer thread
486 TRequestStatus* rendererRequest = Status();
487 iRendererThread->RequestComplete(rendererRequest, KErrNone);
490 /** Send destroy surface request from main thread to renderer thread */
491 void CRendererThreadRelay::DestroySurface(TRequestStatus* aRequestStatus)
493 __ASSERT_DEBUG(aRequestStatus != NULL, User::Panic(_L("CRTR::DestroySurface"), KErrArgument));
495 iCallingStatus = aRequestStatus;
496 iFunctionCode = EDestroySurface;
498 // send request to renderer thread
499 TRequestStatus* rendererRequest = Status();
500 iRendererThread->RequestComplete(rendererRequest, KErrNone);
503 /* Prepare the object after a surface is created */
504 void CRendererThreadRelay::PrepareL(const TSurfaceId& aSurfaceId, TInt aNumBuffers, TRequestStatus* aRequestStatus)
506 __ASSERT_DEBUG(aRequestStatus != NULL, User::Panic(_L("CRTR::PrepareL"), KErrArgument));
508 if(!iRendererCallbackListener)
510 // first, create callback listener in the main thread
511 iRendererCallbackListener = CRendererCallbackListener::NewL(iObserver, aNumBuffers);
512 iRendererCallbackListener->Start();
514 else if (iNumBuffers < aNumBuffers)
516 iRendererCallbackListener->Cancel();
517 iRendererCallbackListener->ExtendMsgQueueL(aNumBuffers);
518 iRendererCallbackListener->Start();
521 // set function parameters
522 iCallingStatus = aRequestStatus;
523 iFunctionCode = EPrepare;
524 iSurfaceId = aSurfaceId;
525 iNumBuffers = aNumBuffers;
527 // send request to renderer thread
528 TRequestStatus* rendererRequest = Status();
529 iRendererThread->RequestComplete(rendererRequest, KErrNone);
532 /* Prepare the object after a surface is created */
533 void CRendererThreadRelay::RunPrepareL()
535 iRendererRelay->PrepareL(iSurfaceId, iNumBuffers, NULL);
538 /** Run update buffer in renderer thread */
539 void CRendererThreadRelay::RunUpdateBuffer(TVideoFrameBuffer* aBuffer, const TTime& aPresentationTime)
541 DEBUGPRINT1(_L("CRendererThreadRelay::RunUpdateBuffer entered"));
542 __ASSERT_DEBUG(iBufManager != NULL, User::Panic(_L("CRTR::RunUpdateBuffer"), KErrCorrupt));
545 Buffer update is determined as follow:
547 If wait list is empty (imply no active timer), always add buffer to list,
548 If a preceding buffer hasn't been displayed, the new buffer will be handled after display callback.
549 otherwise, decide whether timer should be started for submit update
550 If wait list is not empty (imply either timer is active or waiting for display callback)
551 If waiting for display callback, add new buffer to list and it will be handled after display callback.
552 (note: presentation time is not check because the new buffer may be newer than the waiting buffers even though both have passed due time)
553 If timer is active, first check if presentation time is zero. If so, display right away
554 otherwise, then check if this frame can be timed (presentation time - delay <= now), if not, check max display to skip the frame or display right away
555 otherwise, then check if presentation time < head list presentation time
556 if so, add the buffer to wait list, stop the timer and start timer with new time
557 else, just need to add buffer to wait list, the next timer expiry will hander the head buffer
560 if (iBufManager->WaitingListIsEmpty())
562 iBufManager->UpdateBuffer(aBuffer, aPresentationTime);
564 if (iRendererRelay->UpdateSubmitted() == EFalse)
566 // an update haven't been submitted, prepare to submit one now
567 iRendererRelay->SubmitBufferTimed();
572 // waiting list is not empty
573 if (iRendererRelay->UpdateSubmitted())
575 // waiting for listener callback, just update waiting list
576 iBufManager->UpdateBuffer(aBuffer, aPresentationTime);
580 // the timer is waiting
582 if (aPresentationTime.Int64() == 0)
584 // presentation time is zero, display right away.
585 iRendererTimer->Cancel();
586 iBufManager->UpdateBuffer(aBuffer, aPresentationTime);
587 iRendererRelay->SubmitBufferTimed();
593 TTimeIntervalMicroSeconds deltaFromNow = aPresentationTime.MicroSecondsFrom(now);
594 TInt64 waitTime = deltaFromNow.Int64() - iRendererRelay->Delay();
598 // the wait time has passed
599 if (deltaFromNow.Int64() + iMaxDelay >= 0)
601 // the frame is within maximun allowed delay, i.e. (presentation time + max delay >= now)
602 // display right away
603 iRendererTimer->Cancel();
604 iBufManager->UpdateBuffer(aBuffer, aPresentationTime);
605 iRendererRelay->SubmitBufferTimed();
610 iBufManager->ReleaseBuffer(aBuffer);
611 MvroBufferSkipped(aBuffer->BufferId());
613 //notifiy observer about the available buffers
614 MvroVideoBufferAvailable();
619 // wait time has not passed, add to waiting list
620 if (iBufManager->UpdateBuffer(aBuffer, aPresentationTime))
622 // head of waiting list has changed, start timer with new wait time
623 iRendererTimer->Cancel();
624 iRendererTimer->Start(waitTime);
631 /** Run destroy surface in renderer thread */
632 void CRendererThreadRelay::RunDestroySurface()
634 iRendererTimer->Cancel();
635 iRendererRelay->DestroySurface(NULL);
638 /* Update buffer manager pointer */
639 void CRendererThreadRelay::SetBufferManager(CRendererBufferManager* aBufManager)
641 TRequestStatus request = KRequestPending;
642 TRequestStatus logonRequest = KRequestPending;
644 // While a function call is in progress this thread is suspended
645 // and the undertaker will not catch panics, listen for these here
646 iRendererThread->Logon(logonRequest);
648 // Set the call parameters
649 iCallingStatus = &request;
650 iFunctionCode = ESetBufferManager;
651 iBufManager = aBufManager;
653 // send request to renderer thread
654 TRequestStatus* rendererRequest = Status();
655 iRendererThread->RequestComplete(rendererRequest, KErrNone);
657 User::WaitForRequest(logonRequest, request);
659 if(logonRequest != KRequestPending)
661 // renderer thread got panic from surface update session, so panic client
662 TInt reason = iRendererThread->ExitReason();
663 TExitCategoryName category = iRendererThread->ExitCategory();
664 User::Panic(category,reason);
667 // Thread is still alive and well
668 iRendererThread->LogonCancel(logonRequest);
669 User::WaitForRequest(logonRequest); // Consume the signal
671 __ASSERT_DEBUG(request != KRequestPending, User::Panic(_L("CRTR::SetBufferManager"), KErrCorrupt));
674 void CRendererThreadRelay::RunSetBufferManager()
676 iRendererRelay->SetBufferManager(iBufManager);
679 /** Store a pointer to the renderer thread object. */
680 void CRendererThreadRelay::SetRendererThread(RThread* aRendererThread)
682 iRendererThread = aRendererThread;
685 /** Return a pointer to the function call listener's request status. */
686 TRequestStatus* CRendererThreadRelay::Status()
691 /** Set timer info for timed mode */
692 void CRendererThreadRelay::SetTimerInfo(TInt64 aDefaultDelay, TInt64 aMaxDelay)
694 iMaxDelay = aMaxDelay;
695 iRendererRelay->SetTimerInfo(aDefaultDelay, aMaxDelay);
698 void CRendererThreadRelay::MvroVideoBufferAvailable()
700 iRendererCallbackListener->SendCallback(CRendererCallbackListener::EBufferAvailable, -1, 0);
703 void CRendererThreadRelay::MvroBufferDisplayed(TInt aBufferId, const TTime& aTime)
705 iRendererCallbackListener->SendCallback(CRendererCallbackListener::EBufferDisplayed, aBufferId, aTime);
708 void CRendererThreadRelay::MvroBufferSkipped(TInt aBufferId)
710 iRendererCallbackListener->SendCallback(CRendererCallbackListener::EBufferSkipped, aBufferId, 0);
714 /** Private constructor */
715 CRendererCallbackListener::CRendererCallbackListener(MVideoRendererObserver& aObserver)
716 : CActive(EPriorityStandard),
719 CActiveScheduler::Add(this);
722 /** Two-phased constructor */
723 CRendererCallbackListener* CRendererCallbackListener::NewL(MVideoRendererObserver& aObserver, TInt aNumBuffer)
725 CRendererCallbackListener* self = new (ELeave) CRendererCallbackListener(aObserver);
726 CleanupStack::PushL(self);
727 self->ConstructL(aNumBuffer);
728 CleanupStack::Pop(self);
732 /** Second-phase constructor */
733 void CRendererCallbackListener::ConstructL(TInt aNumBuffer)
735 // There could potentially be three messages outstanding per buffer
736 // size message queue accordingly
737 TInt slot = aNumBuffer * 3;
738 if (slot > RMsgQueueBase::KMaxLength)
740 slot = RMsgQueueBase::KMaxLength;
742 User::LeaveIfError(iMsgQueue.CreateLocal(slot));
745 CRendererCallbackListener::~CRendererCallbackListener()
747 Cancel(); // Cancel any request, if outstanding
751 void CRendererCallbackListener::ExtendMsgQueueL(TInt aNumBuffer)
753 // close the message queue and construct another one
755 ConstructL(aNumBuffer);
758 void CRendererCallbackListener::DoCancel()
760 iMsgQueue.CancelDataAvailable();
763 /** Start listener */
764 void CRendererCallbackListener::Start()
768 iMsgQueue.NotifyDataAvailable(iStatus);
773 /** Set the callback function */
774 void CRendererCallbackListener::SendCallback(TFunctionCode aFunctionCode, TInt aBufferId, const TTime& aTime)
776 DEBUGPRINT2(_L("CRendererCallbackListener::SendCallback entered aFunctionCode=%d"), aFunctionCode);
779 data.iFunctionCode = aFunctionCode;
780 data.iBufferId = aBufferId;
783 iMsgQueue.Send(data);
786 /** Call the callback function within the main thread */
787 void CRendererCallbackListener::RunL()
790 TInt err = iMsgQueue.Receive(data);
793 if (data.iFunctionCode == EBufferAvailable)
795 DEBUGPRINT1(_L("CRendererCallbackListener::RunL - EBufferAvailable"));
796 iObserver.MvroVideoBufferAvailable();
798 else if (data.iFunctionCode == EBufferDisplayed)
800 DEBUGPRINT1(_L("CRendererCallbackListener::RunL - EBufferDisplayed"));
801 iObserver.MvroBufferDisplayed(data.iBufferId, data.iTime);
803 else if (data.iFunctionCode == EBufferSkipped)
805 DEBUGPRINT1(_L("CRendererCallbackListener::RunL - EBufferSkipped"));
806 iObserver.MvroBufferSkipped(data.iBufferId);
811 DEBUGPRINT2(_L("CRendererCallbackListener::RunL err=%d"), err);