telephonyserverplugins/simtsy/src/CSimReduceTimers.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 31 Aug 2010 16:23:08 +0300
branchRCL_3
changeset 19 630d2f34d719
parent 0 3553901f7fa8
child 20 07a122eea281
permissions -rw-r--r--
Revision: 201035 Kit: 201035

// Copyright (c) 2004-2010 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "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:
// A subject for oberving requests to reduce the timers currently
// running in the simulator.
// 
//

/**
 @file
*/




#include "OstTraceDefinitions.h"
#ifdef OST_TRACE_COMPILER_IN_USE
#include "CSimReduceTimersTraces.h"
#endif

#include "CSimReduceTimers.h"


/**
Factory function for creating a CSimReduceTimers object.

@return A pointer to the object created.
*/
CSimReduceTimers* CSimReduceTimers::NewL()
	{
	CSimReduceTimers* self = new(ELeave) CSimReduceTimers;
	CleanupStack::PushL(self);
	self->ConstructL();
	CleanupStack::Pop(self);
	return self;
	}

/**
Second phase constructor.  Ensures that the required Publish &
Subscribe property is defined and starts listening to any changes (or
Set() calls) made to this property.
*/
void CSimReduceTimers::ConstructL()
	{
	TInt dummy;
	TInt ret = RProperty::Get(KUidPSSimTsyCategory, KPSSimTsyTimersReduceTime, dummy);

	if (ret == KErrNone)
		{
		iPSProperty = CSimPubSub::TPubSubProperty(KUidPSSimTsyCategory,KPSSimTsyTimersReduceTime,
												KPSSimTsyTimersReduceTimeSignalKeyType);
		iSimPubSub = CSimPubSub::NewL(this,iPSProperty);
		iSimPubSub->Start();
		}
	else
		{
		User::Leave(ret);
		}
	}

/**
Default, first-phase, constructor.
*/
CSimReduceTimers::CSimReduceTimers()
	{
	// NOP
	}

/**
Default destrucor for the reduce timers subject.
*/
CSimReduceTimers::~CSimReduceTimers()
	{
	delete iSimPubSub;
	iObservers.Close();
	}

void CSimReduceTimers::SimPSEvent(const CSimPubSub::TPubSubProperty aProperty, TInt aStatus)
	{
	if (aProperty == iPSProperty && // Condition for correct property
		(aStatus == ETimerIdAllTimers)) // and correct value for signal
		{
		OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_SIMPSEVENT_1, ">>CSimReduceTimers.cpp: Reduce Timer event fired for all timers");
		Notify();
		}
	else if(aProperty == iPSProperty && aStatus > 0)
		{
		OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_SIMPSEVENT_2, ">>CSimReduceTimers.cpp: Reduce Timer event fired for event ID %d", aStatus);
		Notify(aStatus);
		}
	}

void CSimReduceTimers::AttachL(MSimTimerUpdateObserver* aObserver)
	{
	OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_ATTACHL_1, "CSimReduceTimers::AttachL");
	User::LeaveIfError(iObservers.Append(aObserver));
	}

void CSimReduceTimers::DetachL(MSimTimerUpdateObserver* aObserver)
	{
	TInt pos = iObservers.Find(aObserver);
	OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_DETACHL_1, "CSimReduceTimers::DetachL, from position %d", pos);
	User::LeaveIfError(pos);
	iObservers.Remove(pos);
	}

void CSimReduceTimers::Notify()
	{
	TInt totalItems = iObservers.Count();
	OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY_1, "CSimReduceTimers::Notify, %d observers registered", totalItems);
	if (totalItems <= 0)
		{
		OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY_2, "CSimReduceTimers::Notify, No timers to reduce.");
		return;
		}

	TInt minTimeReduce = KMaxTInt;
	TInt num;
	for (num = 0; num < totalItems; num++)
		{
		TInt temp = iObservers[num]->GetRemainingSeconds();
		OstTraceDefExt2(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY_3, "CSimReduceTimers::Notify, iObservers[%d] remaining time = %d", num, temp);

		// Ensure no error is returned and check if the current itteration gives a lower time remaining.
		if (temp > 0 && temp < minTimeReduce)
			{
			minTimeReduce = temp;
			}
		}

	if (minTimeReduce == KMaxTInt)
		{
		OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY_4, "CSimReduceTimers::Notify, No running timers.");
		return;
		}

	OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY_5, "CSimReduceTimers::Notify, Min time to reduce all timers: %d", minTimeReduce);
	// Ensure that reducing timers by minTimeReduce does not reduce any timer to less than KTimerDelayOnReduceTimeSignal.
	minTimeReduce -= KTimerDelayOnReduceTimeSignal;
	if (minTimeReduce <= 0)
		{
		OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY_6, "CSimReduceTimers::Notify, Min time to reduce <= KTimerDelayOnReduceTimeSignal, no change to timers.");
		return;
		}

	for (num = 0; num < totalItems; num++)
		{
		iObservers[num]->Update(minTimeReduce);
		}
	}
	
void CSimReduceTimers::Notify(TInt aTimerEventId)
	{
	TInt totalItems = iObservers.Count();
	OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY1_1, "CSimReduceTimers::Notify, %d observers registered", totalItems);
	if (totalItems <= 0)
		{
		OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY1_2, "CSimReduceTimers::Notify, No timers to reduce.");
		return;
		}
	
	//Get the index of the observer with with event id aTimerEventId
	//and the lowest remaining timer 
	TInt lowestTime = KMaxTInt;
	TInt indexOfLowest = KErrNotFound;
	
	TInt index;
	for(index = 0; index < totalItems; index++)
		{
		if((iObservers[index]->GetTimerEventInfo() == aTimerEventId) &&
			(iObservers[index]->GetRemainingSeconds() > 0) &&
			(iObservers[index]->GetRemainingSeconds() < lowestTime))
			{
			indexOfLowest = index;
			lowestTime = iObservers[index]->GetRemainingSeconds();
			}
		}
		
	if(indexOfLowest == KErrNotFound)
		{
		OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY1_3, ">>CSimReduceTimers::Notify, No observers with event ID %d found", aTimerEventId);
		return;
		}

	//Check that reducing the timer of the selected observer is acceptable
	TInt reduceBy = lowestTime - KTimerDelayOnReduceTimeSignal;
	if(reduceBy >= lowestTime)
		{
		OstTraceDefExt2(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY1_4, "CSimReduceTimers::Notify, Lowest timer for event ID %d already less than %d sec. No timer reduced.",KTimerDelayOnReduceTimeSignal, aTimerEventId);
		}
	else
		{
		OstTraceDefExt2(OST_TRACE_CATEGORY_DEBUG, TRACE_INTERNALS, CSIMREDUCETIMERS_NOTIFY1_5, ">>CSimReduceTimers::Notify, Timer for observer with event ID %d, reduced by %d sec", aTimerEventId, reduceBy);
		iObservers[indexOfLowest]->Update(reduceBy);
		}
	}