os/ossrv/genericservices/taskscheduler/SCHSVR/SchTimer.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) 2004-2010 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
// User includes
sl@0
    17
#include "SchTimer.h"
sl@0
    18
#include "SCHMAN.H"
sl@0
    19
#include "SCHEDULE.H"
sl@0
    20
#include "schlogger.h"
sl@0
    21
sl@0
    22
// system includes
sl@0
    23
#include <e32property.h>
sl@0
    24
sl@0
    25
/**
sl@0
    26
An instance of this class is used for each different time that a
sl@0
    27
schedule needs to be run. It is created solely in the 
sl@0
    28
CScheduleCriteriaManager class.  The timer is updated via the 
sl@0
    29
SetNext method.  When the time has been reached it notifies the schedule
sl@0
    30
manager via the CScheduleCriteriaManager::DueSchedule() method.
sl@0
    31
sl@0
    32
@internalComponent
sl@0
    33
*/
sl@0
    34
_LIT(KTaskSchedulerPanic, "TaskScheduler ");
sl@0
    35
sl@0
    36
NONSHARABLE_CLASS(CScheduleTimer) : public CTimer
sl@0
    37
	{
sl@0
    38
public:
sl@0
    39
	~CScheduleTimer();
sl@0
    40
	static CScheduleTimer* NewL(TInt aSchedule, CScheduleCriteriaManager& aManager);
sl@0
    41
	
sl@0
    42
	void SetNext(const TTsTime& aNextTime);
sl@0
    43
	TInt Id();
sl@0
    44
sl@0
    45
	//list capability
sl@0
    46
	static TInt Offset();
sl@0
    47
private:
sl@0
    48
	// From CTimer
sl@0
    49
	void RunL();
sl@0
    50
sl@0
    51
	CScheduleTimer(TInt aSchedule, CScheduleCriteriaManager& aManager);
sl@0
    52
	void ConstructL();
sl@0
    53
	
sl@0
    54
private:
sl@0
    55
	TTsTime iDueTime;
sl@0
    56
	TInt iScheduleHandle;
sl@0
    57
	TSglQueLink iLink;
sl@0
    58
	CScheduleCriteriaManager& iConditonManager;
sl@0
    59
	};
sl@0
    60
sl@0
    61
CScheduleTimer* CScheduleTimer::NewL(TInt aSchedule, CScheduleCriteriaManager& aManager)
sl@0
    62
	{
sl@0
    63
	CScheduleTimer* self = new(ELeave) CScheduleTimer(aSchedule, aManager);
sl@0
    64
	CleanupStack::PushL(self);
sl@0
    65
	self->ConstructL();
sl@0
    66
	CleanupStack::Pop(self);
sl@0
    67
	return self;
sl@0
    68
	}
sl@0
    69
sl@0
    70
CScheduleTimer::CScheduleTimer(TInt aSchedule, CScheduleCriteriaManager& aManager)
sl@0
    71
:	CTimer(EPriorityStandard),
sl@0
    72
	iScheduleHandle(aSchedule),
sl@0
    73
	iConditonManager(aManager)
sl@0
    74
	{
sl@0
    75
	}
sl@0
    76
sl@0
    77
CScheduleTimer::~CScheduleTimer()
sl@0
    78
	{
sl@0
    79
	Cancel();
sl@0
    80
	}
sl@0
    81
sl@0
    82
void CScheduleTimer::ConstructL()
sl@0
    83
	{
sl@0
    84
	CTimer::ConstructL();
sl@0
    85
	CActiveScheduler::Add(this);
sl@0
    86
	}
sl@0
    87
sl@0
    88
void CScheduleTimer::SetNext(const TTsTime& aNewTime)
sl@0
    89
	{
sl@0
    90
	// Can't handle (unlikely but theoretical) situation when year is BC (not AD)
sl@0
    91
	__ASSERT_ALWAYS(aNewTime.GetUtcTime().DateTime().Year()>0,User::Invariant());
sl@0
    92
	if (IsActive())
sl@0
    93
		Cancel();
sl@0
    94
sl@0
    95
	iDueTime=aNewTime;
sl@0
    96
	TTime currentTime;
sl@0
    97
	currentTime.UniversalTime();
sl@0
    98
sl@0
    99
	if	(aNewTime.GetUtcTime()>currentTime)
sl@0
   100
		AtUTC(aNewTime.GetUtcTime());
sl@0
   101
	else
sl@0
   102
		AtUTC(currentTime);
sl@0
   103
	}
sl@0
   104
sl@0
   105
//	Respond to an task being due.  RunL is only called once!
sl@0
   106
void CScheduleTimer::RunL()
sl@0
   107
	{
sl@0
   108
	if(iStatus != KErrAbort)
sl@0
   109
		iConditonManager.DueSchedule(iScheduleHandle);
sl@0
   110
sl@0
   111
	// RunL() will also be triggered if the system time is changed, with iStatus
sl@0
   112
	// set to KErrAbort. In this case DueSchedule() should not be called.
sl@0
   113
	// If the system time has been changed, the schedule needs to be requeued. 
sl@0
   114
	// This has already been done automatically by CTaskScheduler::HandleEnvironmentChange()
sl@0
   115
	// [called by the AO CEnvironmentChangeNotifier],
sl@0
   116
	// as the active object CEnvironmentChangeNotifier has a higher priority than CScheduleTimer.
sl@0
   117
	}	
sl@0
   118
sl@0
   119
TInt CScheduleTimer::Offset()
sl@0
   120
	{
sl@0
   121
	return (_FOFF(CScheduleTimer, iLink));
sl@0
   122
	}
sl@0
   123
sl@0
   124
TInt CScheduleTimer::Id()
sl@0
   125
	{
sl@0
   126
	return iScheduleHandle;
sl@0
   127
	}
sl@0
   128
sl@0
   129
//		
sl@0
   130
// class CPropertyNotifier
sl@0
   131
// This class handles changes to P&S variables and notifies the
sl@0
   132
// CConditionManager class when a condition is satisfied.
sl@0
   133
NONSHARABLE_CLASS(CPropertyNotifier) : public CActive
sl@0
   134
	{
sl@0
   135
public:
sl@0
   136
	~CPropertyNotifier();
sl@0
   137
	static CPropertyNotifier* NewL(CConditionManager& aManager);
sl@0
   138
	
sl@0
   139
private:	
sl@0
   140
	CPropertyNotifier(CConditionManager& aManager);
sl@0
   141
	void AttachL();
sl@0
   142
	// From CActive
sl@0
   143
	void RunL();
sl@0
   144
	void DoCancel();
sl@0
   145
	TInt RunError(TInt aError);
sl@0
   146
sl@0
   147
public:
sl@0
   148
	void SetPropertyL(const TUid& aCategory, TUint aKey);
sl@0
   149
sl@0
   150
private:
sl@0
   151
	TUid iCategory;
sl@0
   152
	TUint iKey;
sl@0
   153
	RProperty iProperty;
sl@0
   154
	CConditionManager& iConditionManager;
sl@0
   155
	};
sl@0
   156
sl@0
   157
//		
sl@0
   158
// class CConditionManager
sl@0
   159
// This class manages a set of conditions for each schedule.  It is used
sl@0
   160
// solely by the CScheduleCriteriaManager class. When the set of conditions 
sl@0
   161
// is met, a the schedule manager is notified.
sl@0
   162
NONSHARABLE_CLASS(CConditionManager) : public CActive
sl@0
   163
	{
sl@0
   164
public:
sl@0
   165
	~CConditionManager();
sl@0
   166
	static CConditionManager* NewL(TInt aSchedule, CScheduleCriteriaManager& aManager);
sl@0
   167
	
sl@0
   168
private:	
sl@0
   169
	CConditionManager(TInt aSchedule, CScheduleCriteriaManager& aManager);
sl@0
   170
	TBool MatchAllConditionsL() const;
sl@0
   171
	TBool HasVariable(const TUid& aCategory, TUint aKey) const;
sl@0
   172
	void CompleteRequest();
sl@0
   173
	// From CActive
sl@0
   174
	void RunL();
sl@0
   175
	void DoCancel();
sl@0
   176
sl@0
   177
public:
sl@0
   178
	void ReplaceL(const RArray<TTaskSchedulerCondition>& aConditions);
sl@0
   179
	TInt Id();
sl@0
   180
	void VariableChangedL(const TUid& aCategory, TUint aKey);
sl@0
   181
	//list capability
sl@0
   182
	static TInt Offset();	
sl@0
   183
	
sl@0
   184
private:
sl@0
   185
	RArray<TTaskSchedulerCondition> iConditions;
sl@0
   186
	RPointerArray<CPropertyNotifier> iPropertyNotifiers;
sl@0
   187
	TInt iScheduleHandle;
sl@0
   188
	TSglQueLink iLink;
sl@0
   189
	CScheduleCriteriaManager& iManager;
sl@0
   190
	};
sl@0
   191
sl@0
   192
CConditionManager* CConditionManager::NewL(TInt aSchedule, CScheduleCriteriaManager& aManager)
sl@0
   193
	{
sl@0
   194
	CConditionManager* self = new(ELeave) CConditionManager(aSchedule, aManager);
sl@0
   195
	return self;
sl@0
   196
	}
sl@0
   197
sl@0
   198
CConditionManager::CConditionManager(TInt aSchedule, CScheduleCriteriaManager& aManager)
sl@0
   199
:	CActive(EPriorityStandard+1), //make priority higher that propertynotifier AO
sl@0
   200
	iScheduleHandle(aSchedule),
sl@0
   201
	iManager(aManager)
sl@0
   202
	{
sl@0
   203
	CActiveScheduler::Add(this);
sl@0
   204
	}
sl@0
   205
sl@0
   206
CConditionManager::~CConditionManager()
sl@0
   207
	{
sl@0
   208
	Cancel();
sl@0
   209
	iPropertyNotifiers.ResetAndDestroy();
sl@0
   210
	iConditions.Reset();		
sl@0
   211
	}
sl@0
   212
sl@0
   213
//This function evaluates (aValue <op> aState) expression, where "op" could be
sl@0
   214
//"==", "!=", ">", "<", depending on aType value, and returns the result of expression.
sl@0
   215
static TBool DoMatchCondition(TTaskSchedulerCondition::TConditionType aType,
sl@0
   216
							  TInt aValue, 
sl@0
   217
							  TInt aState)
sl@0
   218
	{
sl@0
   219
	if(aType == TTaskSchedulerCondition::EEquals)
sl@0
   220
		{
sl@0
   221
		if(aValue == aState)
sl@0
   222
			{
sl@0
   223
			return ETrue;
sl@0
   224
			}
sl@0
   225
		}
sl@0
   226
	else if(aType == TTaskSchedulerCondition::ENotEquals)
sl@0
   227
		{
sl@0
   228
		if(aValue != aState)
sl@0
   229
			{
sl@0
   230
			return ETrue;
sl@0
   231
			}
sl@0
   232
		}
sl@0
   233
	else if(aType == TTaskSchedulerCondition::EGreaterThan)
sl@0
   234
		{
sl@0
   235
		if(aValue > aState)
sl@0
   236
			{
sl@0
   237
			return ETrue;
sl@0
   238
			}
sl@0
   239
		}
sl@0
   240
	else if(aType == TTaskSchedulerCondition::ELessThan)
sl@0
   241
		{
sl@0
   242
		if(aValue < aState)
sl@0
   243
			{
sl@0
   244
			return ETrue;
sl@0
   245
			}
sl@0
   246
		}
sl@0
   247
	else
sl@0
   248
		{
sl@0
   249
		__ASSERT_ALWAYS(0, User::Invariant());
sl@0
   250
		}
sl@0
   251
	return EFalse;
sl@0
   252
	}
sl@0
   253
sl@0
   254
void CConditionManager::ReplaceL(const RArray<TTaskSchedulerCondition>& aConditions)
sl@0
   255
	{
sl@0
   256
	// Ensure any active requests are cancelled
sl@0
   257
	if(IsActive()) 
sl@0
   258
	{ 
sl@0
   259
		Cancel(); 
sl@0
   260
	}
sl@0
   261
sl@0
   262
	//destroying existing ones will cancel outstanding requests
sl@0
   263
	iPropertyNotifiers.ResetAndDestroy(); 
sl@0
   264
	iConditions.Reset();
sl@0
   265
	const TInt count = aConditions.Count();
sl@0
   266
	TInt i;
sl@0
   267
	//Check that the properties already exist
sl@0
   268
	for(i=0;i<count;++i)
sl@0
   269
		{
sl@0
   270
		TInt value;
sl@0
   271
		TInt err = RProperty::Get(aConditions[i].iCategory, aConditions[i].iKey, value);
sl@0
   272
		if(err != KErrNone)
sl@0
   273
			{
sl@0
   274
			if(err == KErrNotFound)
sl@0
   275
				{
sl@0
   276
				err = KErrArgument; //use KErrArgument error code to signify bad conditions.
sl@0
   277
				}
sl@0
   278
			User::Leave(err);	
sl@0
   279
			}
sl@0
   280
		}
sl@0
   281
	//Add the new conditions and notifiers.
sl@0
   282
	for(i=0;i<count;++i)
sl@0
   283
		{
sl@0
   284
		//Create local CPropertyNotifier object
sl@0
   285
		CPropertyNotifier* notifier = CPropertyNotifier::NewL(*this);
sl@0
   286
		CleanupStack::PushL(notifier);
sl@0
   287
		const TTaskSchedulerCondition& condition = aConditions[i];
sl@0
   288
		notifier->SetPropertyL(condition.iCategory, condition.iKey); 
sl@0
   289
		//Add condition
sl@0
   290
		User::LeaveIfError(iConditions.Append(condition));
sl@0
   291
		//Add notifier
sl@0
   292
		TInt err = iPropertyNotifiers.Append(notifier);
sl@0
   293
		if(err != KErrNone)
sl@0
   294
			{
sl@0
   295
			iConditions.Remove(iConditions.Count() - 1);//Remove the condition we've just added
sl@0
   296
			User::Leave(err);
sl@0
   297
			}
sl@0
   298
		CleanupStack::Pop(notifier);
sl@0
   299
		}
sl@0
   300
	//Check to see that conditions are not already satisfied.
sl@0
   301
	if(MatchAllConditionsL())
sl@0
   302
		{
sl@0
   303
		SetActive(); //we need to set AO active here, otherwise RunL wont be called.	
sl@0
   304
		CompleteRequest();
sl@0
   305
		}
sl@0
   306
	}
sl@0
   307
sl@0
   308
void CConditionManager::CompleteRequest()
sl@0
   309
	{
sl@0
   310
	TRequestStatus *status = &iStatus;
sl@0
   311
	User::RequestComplete(status, KErrNone); // now compete request so RunL is triggered
sl@0
   312
	}
sl@0
   313
	
sl@0
   314
//Respond to a condition changing.
sl@0
   315
//Called from CPropertyNotifier::RunL().
sl@0
   316
void CConditionManager::VariableChangedL(const TUid& aCategory, TUint aKey)
sl@0
   317
	{
sl@0
   318
	//We have been notified that the value of one of the variables has been changed.
sl@0
   319
	//It is not enough to check that the variable's value satisfies its condition!
sl@0
   320
	//We have to check that all CConditionManager::iPropertyNotifiers satisfy their conditions.
sl@0
   321
	//----------------
sl@0
   322
	//If this is a variable, which is a part of the variables, monitored by the 
sl@0
   323
	//current CConditionManager object, only then do check variables values against
sl@0
   324
	//requested conditions	
sl@0
   325
	if(HasVariable(aCategory, aKey))
sl@0
   326
		{
sl@0
   327
		if(MatchAllConditionsL())
sl@0
   328
			{
sl@0
   329
			SetActive(); //we need to set AO active here, otherwise RunL wont be called.	
sl@0
   330
			CompleteRequest();
sl@0
   331
			}
sl@0
   332
		}
sl@0
   333
	}
sl@0
   334
	
sl@0
   335
void CConditionManager::RunL()
sl@0
   336
	{
sl@0
   337
	// cancel outstanding notification requests by destroying AO's
sl@0
   338
	iPropertyNotifiers.ResetAndDestroy();
sl@0
   339
	iManager.DueSchedule(iScheduleHandle);		
sl@0
   340
	}
sl@0
   341
sl@0
   342
void CConditionManager::DoCancel()
sl@0
   343
	{
sl@0
   344
	CompleteRequest();  		
sl@0
   345
	}
sl@0
   346
sl@0
   347
//The method returns ETrue, if all monitored variables (aConditions array) 
sl@0
   348
//satisfy their conditions, EFalse otherwise.
sl@0
   349
TBool CConditionManager::MatchAllConditionsL() const
sl@0
   350
	{
sl@0
   351
	TInt satisfiedConditionsCnt = 0;
sl@0
   352
	TInt count = iConditions.Count();
sl@0
   353
	for(TInt i=0;i<count;++i)
sl@0
   354
		{
sl@0
   355
		const TTaskSchedulerCondition& condition = iConditions[i];
sl@0
   356
		TInt value;
sl@0
   357
		// errors here typically indicate that the P&S variables is not of 
sl@0
   358
		// integer type (ie its changed) or its been deleted
sl@0
   359
		User::LeaveIfError(RProperty::Get(condition.iCategory, condition.iKey, value));
sl@0
   360
		if(::DoMatchCondition(condition.iType, value, condition.iState))
sl@0
   361
			{
sl@0
   362
			++satisfiedConditionsCnt;
sl@0
   363
			}
sl@0
   364
		}
sl@0
   365
	return satisfiedConditionsCnt == count;
sl@0
   366
	}
sl@0
   367
	
sl@0
   368
//This method checks if the variable, identified by (aCategory, aKey) pair, is a part
sl@0
   369
//of CConditionManager::iPropertyNotifiers array and returns ETrue, if that's true.
sl@0
   370
//EFalse otherwise.
sl@0
   371
TBool CConditionManager::HasVariable(const TUid& aCategory, TUint aKey) const
sl@0
   372
	{
sl@0
   373
	for(TInt i=iConditions.Count()-1;i>-1;--i)
sl@0
   374
		{
sl@0
   375
		if(iConditions[i].iCategory == aCategory && iConditions[i].iKey == aKey)
sl@0
   376
			{
sl@0
   377
			return ETrue;
sl@0
   378
			}
sl@0
   379
		}
sl@0
   380
	return EFalse;
sl@0
   381
	}
sl@0
   382
sl@0
   383
TInt CConditionManager::Offset()
sl@0
   384
	{
sl@0
   385
	return (_FOFF(CConditionManager, iLink));
sl@0
   386
	}
sl@0
   387
sl@0
   388
TInt CConditionManager::Id()
sl@0
   389
	{
sl@0
   390
	return iScheduleHandle;
sl@0
   391
	}
sl@0
   392
sl@0
   393
//
sl@0
   394
sl@0
   395
CPropertyNotifier* CPropertyNotifier::NewL(CConditionManager& aManager)
sl@0
   396
	{
sl@0
   397
	CPropertyNotifier* self = new(ELeave) CPropertyNotifier(aManager);
sl@0
   398
	return self;
sl@0
   399
	}
sl@0
   400
sl@0
   401
CPropertyNotifier::CPropertyNotifier(CConditionManager& aManager)
sl@0
   402
:	CActive(EPriorityStandard),
sl@0
   403
	iConditionManager(aManager)
sl@0
   404
	{
sl@0
   405
	CActiveScheduler::Add(this);
sl@0
   406
	}
sl@0
   407
sl@0
   408
CPropertyNotifier::~CPropertyNotifier()
sl@0
   409
	{
sl@0
   410
	Cancel();
sl@0
   411
	iProperty.Close();
sl@0
   412
	}
sl@0
   413
sl@0
   414
void CPropertyNotifier::AttachL()
sl@0
   415
	{
sl@0
   416
	User::LeaveIfError(iProperty.Attach(iCategory, iKey));
sl@0
   417
	iProperty.Subscribe(iStatus);
sl@0
   418
	SetActive();
sl@0
   419
	}
sl@0
   420
sl@0
   421
void CPropertyNotifier::SetPropertyL(const TUid& aCategory, TUint aKey)
sl@0
   422
	{
sl@0
   423
	if (IsActive())
sl@0
   424
	    {
sl@0
   425
		Cancel();
sl@0
   426
		iProperty.Close();
sl@0
   427
	    }
sl@0
   428
	iCategory = aCategory;
sl@0
   429
	iKey = aKey;
sl@0
   430
	AttachL();	
sl@0
   431
	}
sl@0
   432
sl@0
   433
//	Respond to a condition changing
sl@0
   434
void CPropertyNotifier::RunL()
sl@0
   435
	{
sl@0
   436
	if (iStatus.Int() >= KErrNone)
sl@0
   437
		{
sl@0
   438
		iConditionManager.VariableChangedL(iCategory, iKey);
sl@0
   439
		AttachL();
sl@0
   440
		}
sl@0
   441
	// if status is KErrNotFound then P&S variable has been deleted!  By 
sl@0
   442
	// resubscribing we wait for it to be created.  If it never gets
sl@0
   443
	// created then TRequestStatus never completes so this condition 
sl@0
   444
	// never gets met and iConditionManager.VariableChanged which 
sl@0
   445
	// makes sense.
sl@0
   446
	else if (iStatus.Int() == KErrNotFound)
sl@0
   447
		AttachL();			
sl@0
   448
	// If status is another error we have a problem!!! Whatever the case 
sl@0
   449
	// we should just ignore this condition from now on by doing nothing.
sl@0
   450
	}
sl@0
   451
sl@0
   452
void CPropertyNotifier::DoCancel()
sl@0
   453
	{
sl@0
   454
	iProperty.Cancel();
sl@0
   455
	}
sl@0
   456
	
sl@0
   457
TInt CPropertyNotifier::RunError(TInt aError)
sl@0
   458
    {
sl@0
   459
    if (aError)
sl@0
   460
        {
sl@0
   461
        LOGSTRING("CPropertyNotifier::RunL() leaves.");
sl@0
   462
        User::Panic(KTaskSchedulerPanic, aError);
sl@0
   463
        }
sl@0
   464
    return KErrNone;
sl@0
   465
    }
sl@0
   466
	
sl@0
   467
//
sl@0
   468
sl@0
   469
CScheduleCriteriaManager* CScheduleCriteriaManager::NewL(CTaskScheduler& aOwner)
sl@0
   470
	{
sl@0
   471
	CScheduleCriteriaManager* self = new(ELeave) CScheduleCriteriaManager(aOwner);
sl@0
   472
	return self;
sl@0
   473
	}
sl@0
   474
sl@0
   475
CScheduleCriteriaManager::~CScheduleCriteriaManager()
sl@0
   476
	{
sl@0
   477
	Cancel();
sl@0
   478
	RemoveTimers();
sl@0
   479
	RemoveConditions();
sl@0
   480
	}
sl@0
   481
sl@0
   482
CScheduleCriteriaManager::CScheduleCriteriaManager(CTaskScheduler& aOwner)
sl@0
   483
:	CActive(EPriorityStandard+2), //make priority higher than condition AO
sl@0
   484
	iTaskScheduler(aOwner),
sl@0
   485
	iTimers(CScheduleTimer::Offset()),
sl@0
   486
	iConditions(CConditionManager::Offset())	
sl@0
   487
	{
sl@0
   488
	CActiveScheduler::Add(this);
sl@0
   489
	}
sl@0
   490
sl@0
   491
void CScheduleCriteriaManager::CompleteRequest()
sl@0
   492
	{
sl@0
   493
	TRequestStatus *status = &iStatus;
sl@0
   494
	User::RequestComplete(status, KErrNone); // now compete request so RunL is triggered	
sl@0
   495
	}
sl@0
   496
	
sl@0
   497
void CScheduleCriteriaManager::DueSchedule(TInt aScheduleHandle)
sl@0
   498
	{
sl@0
   499
	iDueScheduleHandle = aScheduleHandle;
sl@0
   500
	SetActive(); // need to set AO active so RunL will subsequently be called.
sl@0
   501
	CompleteRequest();
sl@0
   502
	}
sl@0
   503
sl@0
   504
void CScheduleCriteriaManager::RunL()
sl@0
   505
	{
sl@0
   506
	// remove schedule and then notify task scheduler manager	
sl@0
   507
	RemoveSchedule(iDueScheduleHandle);
sl@0
   508
	iTaskScheduler.DueTaskNotifyL(iDueScheduleHandle);
sl@0
   509
	}
sl@0
   510
sl@0
   511
void CScheduleCriteriaManager::DoCancel()
sl@0
   512
	{
sl@0
   513
	CompleteRequest();
sl@0
   514
	}
sl@0
   515
	
sl@0
   516
// If schedule timer for this ID doesnt exist then create and add new timer.  If schedule 
sl@0
   517
// timer does exist then just amend existing timer.
sl@0
   518
//When one of the schedule entries in this schedule has become due,
sl@0
   519
//this function will be called with aNotFirstTime = ETrue
sl@0
   520
//If this function is called because of environment changes then aSchChange   = EOnlyTime and only update time based schedule
sl@0
   521
void CScheduleCriteriaManager::ReplaceScheduleL(CSchedule& aSchedule, TSchChangeType aSchChange , TBool aNotFirstTime)
sl@0
   522
	{
sl@0
   523
	aSchedule.CalculateDueTime(aNotFirstTime);
sl@0
   524
sl@0
   525
	TInt  scheduleId 		=  	aSchedule.Id();
sl@0
   526
	const TTsTime nextTime 	= 	aSchedule.DueTime();
sl@0
   527
	ReplaceScheduleL(nextTime,scheduleId);
sl@0
   528
	
sl@0
   529
	//If this function is called because of environment changes then
sl@0
   530
	//leave conditions unchanged
sl@0
   531
	if(aSchChange == EOnlyTime)
sl@0
   532
		return;
sl@0
   533
	CConditionManager* condition = FindCondition(scheduleId);
sl@0
   534
	// no point in doing work for 
sl@0
   535
	if(aSchedule.Conditions().Count() > 0)
sl@0
   536
		{
sl@0
   537
		if(!condition)
sl@0
   538
			{
sl@0
   539
			condition = CConditionManager::NewL(scheduleId, *this);
sl@0
   540
			iConditions.AddLast(*condition);
sl@0
   541
			}
sl@0
   542
		condition->ReplaceL(aSchedule.Conditions());
sl@0
   543
		}
sl@0
   544
	else if(condition)
sl@0
   545
		RemoveCondition(condition);		
sl@0
   546
	}
sl@0
   547
	
sl@0
   548
// If schedule timer for this ID doesnt exist then create and add new timer.  If schedule 
sl@0
   549
// timer does exist then just amend existing timer.
sl@0
   550
void CScheduleCriteriaManager::ReplaceScheduleL(const TTsTime& aNextTime, 
sl@0
   551
								TInt aSchedule)
sl@0
   552
	{
sl@0
   553
	CScheduleTimer* timer = Find(aSchedule);
sl@0
   554
	// if time is set to MaxTTime then we don't want to set a timer
sl@0
   555
	// off as it will complete straight away.
sl@0
   556
	if((aNextTime.GetUtcTime() != Time::MaxTTime())
sl@0
   557
	&& 	(aNextTime.GetLocalTime() != Time::MaxTTime()))
sl@0
   558
		{
sl@0
   559
		if(!timer)
sl@0
   560
			{	
sl@0
   561
			timer = CScheduleTimer::NewL(aSchedule, *this);
sl@0
   562
			iTimers.AddLast(*timer);
sl@0
   563
			}
sl@0
   564
		timer->SetNext(aNextTime);
sl@0
   565
		}
sl@0
   566
	else if(timer)	
sl@0
   567
		{
sl@0
   568
		RemoveTimer(timer); // make sure we remove the old one!
sl@0
   569
		}
sl@0
   570
	}
sl@0
   571
sl@0
   572
void CScheduleCriteriaManager::RemoveSchedule(TInt aSchedule)
sl@0
   573
	{
sl@0
   574
	CScheduleTimer* timer = Find(aSchedule);
sl@0
   575
	if(timer)
sl@0
   576
		RemoveTimer(timer); // remove timer also terminates AO
sl@0
   577
sl@0
   578
	CConditionManager* condition = FindCondition(aSchedule);
sl@0
   579
	if(condition)
sl@0
   580
		RemoveCondition(condition); // remove condition also terminates AO
sl@0
   581
	
sl@0
   582
	}
sl@0
   583
sl@0
   584
//Timer methods	
sl@0
   585
void CScheduleCriteriaManager::RemoveTimers()
sl@0
   586
	{
sl@0
   587
	CScheduleTimer* timer;
sl@0
   588
	TSglQueIter<CScheduleTimer> timerIter(iTimers);
sl@0
   589
    timerIter.SetToFirst();
sl@0
   590
    while ((timer = timerIter++) != NULL)
sl@0
   591
		{
sl@0
   592
		RemoveTimer(timer);
sl@0
   593
		}
sl@0
   594
	}
sl@0
   595
sl@0
   596
void CScheduleCriteriaManager::RemoveTimer(CScheduleTimer* aTimer)
sl@0
   597
	{
sl@0
   598
	iTimers.Remove(*aTimer);
sl@0
   599
	delete aTimer;
sl@0
   600
	}
sl@0
   601
	
sl@0
   602
CScheduleTimer* CScheduleCriteriaManager::Find(TInt aSchedule)
sl@0
   603
	{
sl@0
   604
	CScheduleTimer* timer = NULL;
sl@0
   605
	TSglQueIter<CScheduleTimer> timerIter(iTimers);
sl@0
   606
    timerIter.SetToFirst();
sl@0
   607
    while ((timer = timerIter++) != NULL)
sl@0
   608
		{
sl@0
   609
		if	(timer->Id() == aSchedule)
sl@0
   610
			break;
sl@0
   611
		}	
sl@0
   612
	return timer;
sl@0
   613
	}	
sl@0
   614
sl@0
   615
// condition methods
sl@0
   616
void CScheduleCriteriaManager::RemoveConditions()
sl@0
   617
	{
sl@0
   618
	CConditionManager* condition;
sl@0
   619
	TSglQueIter<CConditionManager> conditionIter(iConditions);
sl@0
   620
    conditionIter.SetToFirst();
sl@0
   621
    while ((condition = conditionIter++) != NULL)
sl@0
   622
		{
sl@0
   623
		RemoveCondition(condition);
sl@0
   624
		}
sl@0
   625
	}
sl@0
   626
sl@0
   627
void CScheduleCriteriaManager::RemoveCondition(CConditionManager* aCondition)
sl@0
   628
	{
sl@0
   629
	iConditions.Remove(*aCondition);
sl@0
   630
	delete aCondition;
sl@0
   631
	}
sl@0
   632
	
sl@0
   633
CConditionManager* CScheduleCriteriaManager::FindCondition(TInt aSchedule)
sl@0
   634
	{
sl@0
   635
	CConditionManager* condition = NULL;
sl@0
   636
	TSglQueIter<CConditionManager> conditionIter(iConditions);
sl@0
   637
    conditionIter.SetToFirst();
sl@0
   638
    while ((condition = conditionIter++) != NULL)
sl@0
   639
		{
sl@0
   640
		if	(condition->Id() == aSchedule)
sl@0
   641
			break;
sl@0
   642
		}	
sl@0
   643
	return condition;
sl@0
   644
	}	
sl@0
   645
	
sl@0
   646
TInt CScheduleCriteriaManager::RunError(TInt aError)
sl@0
   647
    {
sl@0
   648
    if (aError)
sl@0
   649
        {
sl@0
   650
        LOGSTRING("CScheduleCriteriaManager::RunL() leaves.");
sl@0
   651
        User::Panic(KTaskSchedulerPanic, aError);
sl@0
   652
        }    
sl@0
   653
    return KErrNone;
sl@0
   654
    }
sl@0
   655
sl@0
   656