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