kernel/eka/euser/cbase/ub_dtim.cpp
author Adrian Taylor <adrian@macrobug.com>
Tue, 10 Nov 2009 20:10:34 +0000
changeset 1 cb3e90eb7d89
parent 0 a41df078684a
permissions -rw-r--r--
Improving comments on panics generated by CActive::SetActive. Patch slightly altered based on comments by John Imhofe.

// Copyright (c) 1996-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// e32\euser\cbase\ub_dtim.cpp
// 
//

#include "ub_std.h"

/**
Creates a new timed event queue with the specified active object priority.

@param aPriority The priority of this active object.

@return On successful return, a pointer to the queue of timed events.

@publishedAll
@released
*/
EXPORT_C CDeltaTimer* CDeltaTimer::NewL(TInt aPriority)
	{
	TTimeIntervalMicroSeconds32 tickPeriod;
	UserHal::TickPeriod(tickPeriod);

	CDeltaTimer* timer = new(ELeave) CDeltaTimer(aPriority, tickPeriod.Int());

	TInt err = timer->iTimer.CreateLocal();

	if (err)
		{
		delete timer;
		User::Leave(err);
		}

	CActiveScheduler::Add(timer);

	return timer;
	}

/**
Creates a new timed event queue with the specified active object priority, and 
the specified timer granularity.

@param aPriority    The priority of this active object.
@param aGranularity Ignored.  The resolution of the timer is the tick period.

@return On successful return, a pointer to the queue of timed events.

@publishedAll
@deprecated
*/
EXPORT_C CDeltaTimer* CDeltaTimer::NewL(TInt aPriority, TTimeIntervalMicroSeconds32 /*aGranularity*/)
	{
	return CDeltaTimer::NewL(aPriority);
	}

/**
Constructor taking an active object priority value.

The constructor sets this active object's priority value through a call to 
the base class constructor in its c'tor list.

@param aPriority    The priority of this active object.
@param aTickPeriod  The period of a tick on the system.

@internalComponent
@released
*/
CDeltaTimer::CDeltaTimer(TInt aPriority, TInt aTickPeriod)
	: CActive(aPriority), iTickPeriod(aTickPeriod)
	{
	}

/**
Adds a new timed event entry into the timed event queue.
	
@param aTimeInMicroSeconds The interval from the present time when the timed 
	                       event entry is to expire.
@param aEntry              The timed event entry encapsulating the call back that
                           is to be called when this timed event entry expires.

@publishedAll
@released
*/
EXPORT_C void CDeltaTimer::Queue(TTimeIntervalMicroSeconds32 aTimeInMicroSeconds, TDeltaTimerEntry& aEntry)
	{
	QueueLong(TTimeIntervalMicroSeconds(MAKE_TINT64(0, aTimeInMicroSeconds.Int())), aEntry);
	}

/**
Adds a new timed event entry into the timed event queue.
	
@param aTimeInMicroSeconds The interval from the present time when the timed 
	                       event entry is to expire.
@param aEntry              The timed event entry encapsulating the call back that
                           is to be called when this timed event entry expires.

@return KErrNone if sucessful, KErrOverflow if the interval is too great or negative.

@publishedAll
@released
*/
EXPORT_C TInt CDeltaTimer::QueueLong(TTimeIntervalMicroSeconds aTimeInMicroSeconds, TDeltaTimerEntry& aEntry)
	{
	const TInt64 timeInTicks = (aTimeInMicroSeconds.Int64() + iTickPeriod - 1) / iTickPeriod;

	TInt timeInTicks32 = I64LOW(timeInTicks);

	// We are using deltas on tick values, hence using maximum signed number of ticks
	if (I64HIGH(timeInTicks) || (timeInTicks32 < 0))
		{
		return KErrOverflow;
		}

	// Make sure we queue for at least one tick
	if (timeInTicks32 == 0)
		{
		timeInTicks32 = 1;
		}
	
	// Calculate tick count for new entry
	aEntry.iLink.iTickCount = Exec::TickCount() + timeInTicks32;

	// Add this entry at the right spot
	iQueue.Add(aEntry.iLink);

	// Queue the timer, re-queuing if we've added to the head of the queue
	Activate(&aEntry.iLink == iQueue.First());
	
	return KErrNone;
	}

/**
Issues a new RTimer request, if there is no outstanding request and the queue 
is not empty.

@internalComponent
@released
*/
void CDeltaTimer::Activate(TBool aRequeueTimer)
//
// Queue a request on the timer.
//
	{
	if (IsActive())
		{
		if (aRequeueTimer)
			{
			Cancel();
			}
		else
			{
			return;
			}		
		}

	if (!iQueue.IsEmpty() && !iQueueBusy)
		{
		SetActive();

		const TInt ticksToWait = iQueue.First()->iTickCount - Exec::TickCount();

		if (ticksToWait > 0)
			{
			iTimer.AfterTicks(iStatus, ticksToWait);
			}
		else
			{
			TRequestStatus* status = &iStatus;
			User::RequestComplete(status, KErrNone);
			}
		}
	}

/**
Deals with an RTimer completion event.

The function inspects the timed event entry at the head of the queue, and 
reduces its timer value by the appropriate amount. If this timed event is 
now found to have expired, the call back function is called, and the timed 
event entry removed from the queue.

If the timed event entry has not expired, it remains at the head of the queue.

The function issues a new RTimer request, using the timer granularity value 
as the time interval.

@see RTimer
@see CActive

@internalComponent
@released
*/
void CDeltaTimer::RunL()
//
// Call all zero delta callbacks
	{
	// Queue busy
	iQueueBusy = ETrue;

	// Whilst the list of expired timers is being processed, time will pass and
	// the tick count may have increased such that there are now more expired
	// timers.  Loop until we have either emptied the queue or can wait for a
	// timer exipration in the future.
	while (!iQueue.IsEmpty())
		{
		// Calculate how long till first timer expires
		const TUint tickCount = Exec::TickCount();

		// If the first timer is yet to expire, wait some more
		if (((TInt)(iQueue.First()->iTickCount - tickCount)) > 0)
			{
			break;
			}

		// Remove entry before callback to prevent re-entrancy issues
		TTickCountQueLink* entry = iQueue.RemoveFirst();

		// Iterate through the timers we know have expired based on the
		// last calculation of delta
		while (entry)
			{
			// Make callback.  This could go reentrant on Queue[Long]() or Remove().
			reinterpret_cast<TDeltaTimerEntry*>(
				PtrSub(
					entry,
					_FOFF(TDeltaTimerEntry, iLink)
				))
			->iCallBack.CallBack();

			// Remove the next expired entry, if any
			entry = iQueue.RemoveFirst(tickCount);
			}
		}

	// Queue idle
	iQueueBusy = EFalse;

	// Requeue timer
	Activate();
	}
	
/**
Implements cancellation of an oustanding RTimer request.

@internalComponent
@released
*/
void CDeltaTimer::DoCancel()
	{
	iTimer.Cancel();
	}

/**
Removes the specified timed event entry from the timer queue.

@param aEntry The timed event entry.

@publishedAll
@released
*/
EXPORT_C void CDeltaTimer::Remove(TDeltaTimerEntry& aEntry)
	{
	// Remove the specified entry from the list
	aEntry.iLink.Deque();
	}

/**
Destructor.

Frees all resources before destruction of the object. Specifically, it cancels 
any outstanding timer requests generated by the RTimer object and then deletes 
all timed event entries from the timed event queue.

@see RTimer

@publishedAll
@released
*/
CDeltaTimer::~CDeltaTimer()
	{
	Cancel();

	while (!iQueue.IsEmpty())
		{
		iQueue.First()->Deque();
		}

	iTimer.Close();
	}