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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
20 #include "schlogger.h"
23 #include <e32property.h>
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.
34 _LIT(KTaskSchedulerPanic, "TaskScheduler ");
36 NONSHARABLE_CLASS(CScheduleTimer) : public CTimer
40 static CScheduleTimer* NewL(TInt aSchedule, CScheduleCriteriaManager& aManager);
42 void SetNext(const TTsTime& aNextTime);
51 CScheduleTimer(TInt aSchedule, CScheduleCriteriaManager& aManager);
58 CScheduleCriteriaManager& iConditonManager;
61 CScheduleTimer* CScheduleTimer::NewL(TInt aSchedule, CScheduleCriteriaManager& aManager)
63 CScheduleTimer* self = new(ELeave) CScheduleTimer(aSchedule, aManager);
64 CleanupStack::PushL(self);
66 CleanupStack::Pop(self);
70 CScheduleTimer::CScheduleTimer(TInt aSchedule, CScheduleCriteriaManager& aManager)
71 : CTimer(EPriorityStandard),
72 iScheduleHandle(aSchedule),
73 iConditonManager(aManager)
77 CScheduleTimer::~CScheduleTimer()
82 void CScheduleTimer::ConstructL()
85 CActiveScheduler::Add(this);
88 void CScheduleTimer::SetNext(const TTsTime& aNewTime)
90 // Can't handle (unlikely but theoretical) situation when year is BC (not AD)
91 __ASSERT_ALWAYS(aNewTime.GetUtcTime().DateTime().Year()>0,User::Invariant());
97 currentTime.UniversalTime();
99 if (aNewTime.GetUtcTime()>currentTime)
100 AtUTC(aNewTime.GetUtcTime());
105 // Respond to an task being due. RunL is only called once!
106 void CScheduleTimer::RunL()
108 if(iStatus != KErrAbort)
109 iConditonManager.DueSchedule(iScheduleHandle);
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.
119 TInt CScheduleTimer::Offset()
121 return (_FOFF(CScheduleTimer, iLink));
124 TInt CScheduleTimer::Id()
126 return iScheduleHandle;
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
136 ~CPropertyNotifier();
137 static CPropertyNotifier* NewL(CConditionManager& aManager);
140 CPropertyNotifier(CConditionManager& aManager);
145 TInt RunError(TInt aError);
148 void SetPropertyL(const TUid& aCategory, TUint aKey);
154 CConditionManager& iConditionManager;
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
165 ~CConditionManager();
166 static CConditionManager* NewL(TInt aSchedule, CScheduleCriteriaManager& aManager);
169 CConditionManager(TInt aSchedule, CScheduleCriteriaManager& aManager);
170 TBool MatchAllConditionsL() const;
171 TBool HasVariable(const TUid& aCategory, TUint aKey) const;
172 void CompleteRequest();
178 void ReplaceL(const RArray<TTaskSchedulerCondition>& aConditions);
180 void VariableChangedL(const TUid& aCategory, TUint aKey);
182 static TInt Offset();
185 RArray<TTaskSchedulerCondition> iConditions;
186 RPointerArray<CPropertyNotifier> iPropertyNotifiers;
187 TInt iScheduleHandle;
189 CScheduleCriteriaManager& iManager;
192 CConditionManager* CConditionManager::NewL(TInt aSchedule, CScheduleCriteriaManager& aManager)
194 CConditionManager* self = new(ELeave) CConditionManager(aSchedule, aManager);
198 CConditionManager::CConditionManager(TInt aSchedule, CScheduleCriteriaManager& aManager)
199 : CActive(EPriorityStandard+1), //make priority higher that propertynotifier AO
200 iScheduleHandle(aSchedule),
203 CActiveScheduler::Add(this);
206 CConditionManager::~CConditionManager()
209 iPropertyNotifiers.ResetAndDestroy();
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,
219 if(aType == TTaskSchedulerCondition::EEquals)
226 else if(aType == TTaskSchedulerCondition::ENotEquals)
233 else if(aType == TTaskSchedulerCondition::EGreaterThan)
240 else if(aType == TTaskSchedulerCondition::ELessThan)
249 __ASSERT_ALWAYS(0, User::Invariant());
254 void CConditionManager::ReplaceL(const RArray<TTaskSchedulerCondition>& aConditions)
256 // Ensure any active requests are cancelled
262 //destroying existing ones will cancel outstanding requests
263 iPropertyNotifiers.ResetAndDestroy();
265 const TInt count = aConditions.Count();
267 //Check that the properties already exist
271 TInt err = RProperty::Get(aConditions[i].iCategory, aConditions[i].iKey, value);
274 if(err == KErrNotFound)
276 err = KErrArgument; //use KErrArgument error code to signify bad conditions.
281 //Add the new conditions and notifiers.
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);
290 User::LeaveIfError(iConditions.Append(condition));
292 TInt err = iPropertyNotifiers.Append(notifier);
295 iConditions.Remove(iConditions.Count() - 1);//Remove the condition we've just added
298 CleanupStack::Pop(notifier);
300 //Check to see that conditions are not already satisfied.
301 if(MatchAllConditionsL())
303 SetActive(); //we need to set AO active here, otherwise RunL wont be called.
308 void CConditionManager::CompleteRequest()
310 TRequestStatus *status = &iStatus;
311 User::RequestComplete(status, KErrNone); // now compete request so RunL is triggered
314 //Respond to a condition changing.
315 //Called from CPropertyNotifier::RunL().
316 void CConditionManager::VariableChangedL(const TUid& aCategory, TUint aKey)
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.
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))
327 if(MatchAllConditionsL())
329 SetActive(); //we need to set AO active here, otherwise RunL wont be called.
335 void CConditionManager::RunL()
337 // cancel outstanding notification requests by destroying AO's
338 iPropertyNotifiers.ResetAndDestroy();
339 iManager.DueSchedule(iScheduleHandle);
342 void CConditionManager::DoCancel()
347 //The method returns ETrue, if all monitored variables (aConditions array)
348 //satisfy their conditions, EFalse otherwise.
349 TBool CConditionManager::MatchAllConditionsL() const
351 TInt satisfiedConditionsCnt = 0;
352 TInt count = iConditions.Count();
353 for(TInt i=0;i<count;++i)
355 const TTaskSchedulerCondition& condition = iConditions[i];
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))
362 ++satisfiedConditionsCnt;
365 return satisfiedConditionsCnt == count;
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.
371 TBool CConditionManager::HasVariable(const TUid& aCategory, TUint aKey) const
373 for(TInt i=iConditions.Count()-1;i>-1;--i)
375 if(iConditions[i].iCategory == aCategory && iConditions[i].iKey == aKey)
383 TInt CConditionManager::Offset()
385 return (_FOFF(CConditionManager, iLink));
388 TInt CConditionManager::Id()
390 return iScheduleHandle;
395 CPropertyNotifier* CPropertyNotifier::NewL(CConditionManager& aManager)
397 CPropertyNotifier* self = new(ELeave) CPropertyNotifier(aManager);
401 CPropertyNotifier::CPropertyNotifier(CConditionManager& aManager)
402 : CActive(EPriorityStandard),
403 iConditionManager(aManager)
405 CActiveScheduler::Add(this);
408 CPropertyNotifier::~CPropertyNotifier()
414 void CPropertyNotifier::AttachL()
416 User::LeaveIfError(iProperty.Attach(iCategory, iKey));
417 iProperty.Subscribe(iStatus);
421 void CPropertyNotifier::SetPropertyL(const TUid& aCategory, TUint aKey)
428 iCategory = aCategory;
433 // Respond to a condition changing
434 void CPropertyNotifier::RunL()
436 if (iStatus.Int() >= KErrNone)
438 iConditionManager.VariableChangedL(iCategory, iKey);
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
446 else if (iStatus.Int() == KErrNotFound)
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.
452 void CPropertyNotifier::DoCancel()
457 TInt CPropertyNotifier::RunError(TInt aError)
461 LOGSTRING("CPropertyNotifier::RunL() leaves.");
462 User::Panic(KTaskSchedulerPanic, aError);
469 CScheduleCriteriaManager* CScheduleCriteriaManager::NewL(CTaskScheduler& aOwner)
471 CScheduleCriteriaManager* self = new(ELeave) CScheduleCriteriaManager(aOwner);
475 CScheduleCriteriaManager::~CScheduleCriteriaManager()
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())
488 CActiveScheduler::Add(this);
491 void CScheduleCriteriaManager::CompleteRequest()
493 TRequestStatus *status = &iStatus;
494 User::RequestComplete(status, KErrNone); // now compete request so RunL is triggered
497 void CScheduleCriteriaManager::DueSchedule(TInt aScheduleHandle)
499 iDueScheduleHandle = aScheduleHandle;
500 SetActive(); // need to set AO active so RunL will subsequently be called.
504 void CScheduleCriteriaManager::RunL()
506 // remove schedule and then notify task scheduler manager
507 RemoveSchedule(iDueScheduleHandle);
508 iTaskScheduler.DueTaskNotifyL(iDueScheduleHandle);
511 void CScheduleCriteriaManager::DoCancel()
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)
523 aSchedule.CalculateDueTime(aNotFirstTime);
525 TInt scheduleId = aSchedule.Id();
526 const TTsTime nextTime = aSchedule.DueTime();
527 ReplaceScheduleL(nextTime,scheduleId);
529 //If this function is called because of environment changes then
530 //leave conditions unchanged
531 if(aSchChange == EOnlyTime)
533 CConditionManager* condition = FindCondition(scheduleId);
534 // no point in doing work for
535 if(aSchedule.Conditions().Count() > 0)
539 condition = CConditionManager::NewL(scheduleId, *this);
540 iConditions.AddLast(*condition);
542 condition->ReplaceL(aSchedule.Conditions());
545 RemoveCondition(condition);
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,
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()))
561 timer = CScheduleTimer::NewL(aSchedule, *this);
562 iTimers.AddLast(*timer);
564 timer->SetNext(aNextTime);
568 RemoveTimer(timer); // make sure we remove the old one!
572 void CScheduleCriteriaManager::RemoveSchedule(TInt aSchedule)
574 CScheduleTimer* timer = Find(aSchedule);
576 RemoveTimer(timer); // remove timer also terminates AO
578 CConditionManager* condition = FindCondition(aSchedule);
580 RemoveCondition(condition); // remove condition also terminates AO
585 void CScheduleCriteriaManager::RemoveTimers()
587 CScheduleTimer* timer;
588 TSglQueIter<CScheduleTimer> timerIter(iTimers);
589 timerIter.SetToFirst();
590 while ((timer = timerIter++) != NULL)
596 void CScheduleCriteriaManager::RemoveTimer(CScheduleTimer* aTimer)
598 iTimers.Remove(*aTimer);
602 CScheduleTimer* CScheduleCriteriaManager::Find(TInt aSchedule)
604 CScheduleTimer* timer = NULL;
605 TSglQueIter<CScheduleTimer> timerIter(iTimers);
606 timerIter.SetToFirst();
607 while ((timer = timerIter++) != NULL)
609 if (timer->Id() == aSchedule)
616 void CScheduleCriteriaManager::RemoveConditions()
618 CConditionManager* condition;
619 TSglQueIter<CConditionManager> conditionIter(iConditions);
620 conditionIter.SetToFirst();
621 while ((condition = conditionIter++) != NULL)
623 RemoveCondition(condition);
627 void CScheduleCriteriaManager::RemoveCondition(CConditionManager* aCondition)
629 iConditions.Remove(*aCondition);
633 CConditionManager* CScheduleCriteriaManager::FindCondition(TInt aSchedule)
635 CConditionManager* condition = NULL;
636 TSglQueIter<CConditionManager> conditionIter(iConditions);
637 conditionIter.SetToFirst();
638 while ((condition = conditionIter++) != NULL)
640 if (condition->Id() == aSchedule)
646 TInt CScheduleCriteriaManager::RunError(TInt aError)
650 LOGSTRING("CScheduleCriteriaManager::RunL() leaves.");
651 User::Panic(KTaskSchedulerPanic, aError);