sl@0
|
1 |
// Copyright (c) 2004-2009 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 |
// System includes
|
sl@0
|
17 |
#include <bacntf.h>
|
sl@0
|
18 |
|
sl@0
|
19 |
// User includes
|
sl@0
|
20 |
#include "SCHMAN.H"
|
sl@0
|
21 |
#include "SchTimer.h"
|
sl@0
|
22 |
#include "SchLogger.h"
|
sl@0
|
23 |
#include "SCHEDULE.H"
|
sl@0
|
24 |
#include "SCHCLI.H"
|
sl@0
|
25 |
#include "SCHSTORE.H"
|
sl@0
|
26 |
#include <schtask.h>
|
sl@0
|
27 |
#include "SCHLOG.h"
|
sl@0
|
28 |
|
sl@0
|
29 |
// Constants
|
sl@0
|
30 |
const TInt KMinScheduleId = 0;
|
sl@0
|
31 |
const TInt KMaxSchedules = 25000;
|
sl@0
|
32 |
|
sl@0
|
33 |
//Command line argument
|
sl@0
|
34 |
_LIT(KCommandLine, "SYSSTARTSCHEXE");
|
sl@0
|
35 |
|
sl@0
|
36 |
//SID of SysStart
|
sl@0
|
37 |
const TInt KSysStartSID = 0x10205C44;
|
sl@0
|
38 |
|
sl@0
|
39 |
|
sl@0
|
40 |
#define UNUSED_VAR(a) a = a
|
sl@0
|
41 |
|
sl@0
|
42 |
//
|
sl@0
|
43 |
// Construction/Destruction functions
|
sl@0
|
44 |
//
|
sl@0
|
45 |
|
sl@0
|
46 |
CTaskScheduler::CTaskScheduler()
|
sl@0
|
47 |
: iSchedules(CSchedule::Offset()),
|
sl@0
|
48 |
iClients(CClientProxy::Offset()),
|
sl@0
|
49 |
iStartupStatePassNonCritical(EFalse)
|
sl@0
|
50 |
{
|
sl@0
|
51 |
}
|
sl@0
|
52 |
|
sl@0
|
53 |
CTaskScheduler::~CTaskScheduler()
|
sl@0
|
54 |
{
|
sl@0
|
55 |
if (iBackupNotification)
|
sl@0
|
56 |
{
|
sl@0
|
57 |
iBackupNotification->DeRegisterBackupOperationObserver(*this);
|
sl@0
|
58 |
}
|
sl@0
|
59 |
delete iBackupNotification;
|
sl@0
|
60 |
delete iBackupManager;
|
sl@0
|
61 |
delete iNotifier;
|
sl@0
|
62 |
delete iScheduleCriteriaManager;
|
sl@0
|
63 |
delete iSchLogManager;
|
sl@0
|
64 |
|
sl@0
|
65 |
//remove clients and schedules as well!!
|
sl@0
|
66 |
TDblQueIter<CClientProxy> clientIter(iClients);
|
sl@0
|
67 |
|
sl@0
|
68 |
clientIter.SetToFirst();
|
sl@0
|
69 |
CClientProxy* client=NULL;
|
sl@0
|
70 |
while ((client=clientIter++)!=NULL)
|
sl@0
|
71 |
{
|
sl@0
|
72 |
client->Remove();
|
sl@0
|
73 |
delete client;
|
sl@0
|
74 |
}
|
sl@0
|
75 |
|
sl@0
|
76 |
TSglQueIter<CSchedule> schedIter(iSchedules);
|
sl@0
|
77 |
|
sl@0
|
78 |
schedIter.SetToFirst();
|
sl@0
|
79 |
CSchedule* schedule=NULL;
|
sl@0
|
80 |
while ((schedule=schedIter++)!=NULL)
|
sl@0
|
81 |
{
|
sl@0
|
82 |
iSchedules.Remove(*schedule);
|
sl@0
|
83 |
delete schedule;
|
sl@0
|
84 |
}
|
sl@0
|
85 |
}
|
sl@0
|
86 |
|
sl@0
|
87 |
void CTaskScheduler::ConstructL()
|
sl@0
|
88 |
{
|
sl@0
|
89 |
LOGSTRING("CTaskScheduler::ConstructL - Creating new schedule server log entry");
|
sl@0
|
90 |
|
sl@0
|
91 |
User::LeaveIfError(iFsSession.Connect());
|
sl@0
|
92 |
iBackupManager = new(ELeave) CSchBackupManager(iFsSession);
|
sl@0
|
93 |
iBackupManager->ConstructL();
|
sl@0
|
94 |
|
sl@0
|
95 |
iNotifier = CEnvironmentChangeNotifier::NewL(CActive::EPriorityHigh, TCallBack(EnvironmentChanged, this));
|
sl@0
|
96 |
iNotifier->Start();
|
sl@0
|
97 |
|
sl@0
|
98 |
iScheduleCriteriaManager = CScheduleCriteriaManager::NewL(*this);
|
sl@0
|
99 |
|
sl@0
|
100 |
iSchLogManager = CSchLogManager::NewL(iFsSession);
|
sl@0
|
101 |
|
sl@0
|
102 |
LOGSTRING("CTaskScheduler::ConstructL - Restoring clients and schedules");
|
sl@0
|
103 |
TRAPD(err, iBackupManager->RestoreL(iClients, iSchedules,*iSchLogManager));
|
sl@0
|
104 |
if (err != KErrNone) // the file's corrupt or something...
|
sl@0
|
105 |
{
|
sl@0
|
106 |
LOGSTRING2("CTaskScheduler::ConstructL - had to create new store because of error: %d", err);
|
sl@0
|
107 |
iBackupManager->CreateEmptyBackupL();
|
sl@0
|
108 |
}
|
sl@0
|
109 |
|
sl@0
|
110 |
//checking the SID of the process which started the Task scheduler
|
sl@0
|
111 |
if (User::CreatorSecureId() == KSysStartSID)
|
sl@0
|
112 |
{
|
sl@0
|
113 |
TInt argLen = User::CommandLineLength();
|
sl@0
|
114 |
if (argLen)
|
sl@0
|
115 |
{
|
sl@0
|
116 |
HBufC* arg = HBufC::NewLC(argLen);
|
sl@0
|
117 |
TPtr argPtr = arg->Des();
|
sl@0
|
118 |
User::CommandLine(argPtr);
|
sl@0
|
119 |
argPtr.UpperCase();
|
sl@0
|
120 |
|
sl@0
|
121 |
//Checking Comman dLine arg passed to it is same as in SSCForStartupMode0.rss
|
sl@0
|
122 |
//and checking for persisted schedules
|
sl@0
|
123 |
if((argPtr.Compare(KCommandLine) == 0) && iSchedules.IsEmpty())
|
sl@0
|
124 |
{
|
sl@0
|
125 |
//if no schedule leave
|
sl@0
|
126 |
User::Leave(KErrNone);
|
sl@0
|
127 |
}
|
sl@0
|
128 |
CleanupStack::PopAndDestroy(arg);
|
sl@0
|
129 |
}
|
sl@0
|
130 |
}
|
sl@0
|
131 |
|
sl@0
|
132 |
// Each client now contains a list of associated tasks. We need
|
sl@0
|
133 |
// to now associate those tasks with specific schedules
|
sl@0
|
134 |
CClientProxy* client;
|
sl@0
|
135 |
TDblQueIter<CClientProxy> clientIter(iClients);
|
sl@0
|
136 |
clientIter.SetToFirst();
|
sl@0
|
137 |
while ((client = clientIter++) != NULL)
|
sl@0
|
138 |
{
|
sl@0
|
139 |
// Fetch an iterator for each task owned by this client
|
sl@0
|
140 |
CScheduledTask* task;
|
sl@0
|
141 |
TDblQueIter<CScheduledTask> taskIterator = client->TaskIterator();
|
sl@0
|
142 |
taskIterator.SetToFirst();
|
sl@0
|
143 |
|
sl@0
|
144 |
// Iterate through all the tasks owned by this client, trying to find
|
sl@0
|
145 |
// the corresponding schedules.
|
sl@0
|
146 |
while ((task = taskIterator++) != NULL)
|
sl@0
|
147 |
{
|
sl@0
|
148 |
CSchedule* schedule = NULL;
|
sl@0
|
149 |
schedule = Find(task->ScheduleId());
|
sl@0
|
150 |
if (schedule)
|
sl@0
|
151 |
{
|
sl@0
|
152 |
TScheduledTask* taskRef = new(ELeave) TScheduledTask(*task,*client);
|
sl@0
|
153 |
schedule->AddTask(*taskRef);
|
sl@0
|
154 |
}
|
sl@0
|
155 |
}
|
sl@0
|
156 |
}
|
sl@0
|
157 |
iBackupNotification = CBaBackupSessionWrapper::NewL();
|
sl@0
|
158 |
iBackupNotification->RegisterBackupOperationObserverL(*this);
|
sl@0
|
159 |
}
|
sl@0
|
160 |
|
sl@0
|
161 |
CTaskScheduler* CTaskScheduler::NewL()
|
sl@0
|
162 |
{
|
sl@0
|
163 |
CTaskScheduler* self = CTaskScheduler::NewLC();
|
sl@0
|
164 |
CleanupStack::Pop();
|
sl@0
|
165 |
return self;
|
sl@0
|
166 |
}
|
sl@0
|
167 |
|
sl@0
|
168 |
CTaskScheduler* CTaskScheduler::NewLC()
|
sl@0
|
169 |
{
|
sl@0
|
170 |
CTaskScheduler* self = new(ELeave) CTaskScheduler();
|
sl@0
|
171 |
CleanupStack::PushL(self);
|
sl@0
|
172 |
self->ConstructL();
|
sl@0
|
173 |
return self;
|
sl@0
|
174 |
}
|
sl@0
|
175 |
|
sl@0
|
176 |
//
|
sl@0
|
177 |
// Client, Schedule and Task functions
|
sl@0
|
178 |
//
|
sl@0
|
179 |
|
sl@0
|
180 |
CClientProxy* CTaskScheduler::AddClientL(const TDesC& aFilename, TInt aPriority)
|
sl@0
|
181 |
{
|
sl@0
|
182 |
//check we don't already have a client that will do...
|
sl@0
|
183 |
TDblQueIter<CClientProxy> clientIter(iClients);
|
sl@0
|
184 |
clientIter.SetToFirst();
|
sl@0
|
185 |
CClientProxy* client=NULL;
|
sl@0
|
186 |
while ((client=clientIter++)!=NULL)
|
sl@0
|
187 |
{
|
sl@0
|
188 |
if (client->IsEqual(aFilename, aPriority))
|
sl@0
|
189 |
return client;
|
sl@0
|
190 |
}
|
sl@0
|
191 |
client = CClientProxy::NewL(iFsSession, aFilename, aPriority,*iSchLogManager);
|
sl@0
|
192 |
iClients.Add(*client);
|
sl@0
|
193 |
return client;
|
sl@0
|
194 |
}
|
sl@0
|
195 |
|
sl@0
|
196 |
void CTaskScheduler::AddScheduleL(CSchedule& aSchedule)
|
sl@0
|
197 |
{
|
sl@0
|
198 |
LOGSTRING3("CTaskScheduler::AddScheduleL - schedule: %S, %d", &aSchedule.Name(), aSchedule.Id());
|
sl@0
|
199 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationAdd, aSchedule);
|
sl@0
|
200 |
iSchedules.AddLast(aSchedule);
|
sl@0
|
201 |
LOGSTRING("CTaskScheduler::AddScheduleL - schedule added");
|
sl@0
|
202 |
}
|
sl@0
|
203 |
|
sl@0
|
204 |
void CTaskScheduler::EditScheduleL(TInt aScheduleHandle, CArrayFixFlat<TScheduleEntryInfo2>& aEntryList)
|
sl@0
|
205 |
{
|
sl@0
|
206 |
CSchedule* schedule = FindL(aScheduleHandle);
|
sl@0
|
207 |
|
sl@0
|
208 |
// remove schedule from condition manager before replacing entries to ensure
|
sl@0
|
209 |
// its deleted properly.
|
sl@0
|
210 |
iScheduleCriteriaManager->RemoveSchedule(schedule->Id());
|
sl@0
|
211 |
schedule->ReplaceEntriesL(aEntryList);
|
sl@0
|
212 |
|
sl@0
|
213 |
TRAPD(err, iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, *schedule));
|
sl@0
|
214 |
if(err)
|
sl@0
|
215 |
{
|
sl@0
|
216 |
schedule->RemoveEntries();
|
sl@0
|
217 |
User::Leave(err);
|
sl@0
|
218 |
}
|
sl@0
|
219 |
// recalculate due time only if schedule is enabled and has tasks to run
|
sl@0
|
220 |
if (IsScheduleReadyForUpdate(*schedule))
|
sl@0
|
221 |
{
|
sl@0
|
222 |
iScheduleCriteriaManager->ReplaceScheduleL(*schedule);
|
sl@0
|
223 |
}
|
sl@0
|
224 |
}
|
sl@0
|
225 |
|
sl@0
|
226 |
void CTaskScheduler::DoEditScheduleL(CSchedule& aSchedule,
|
sl@0
|
227 |
CArrayFixFlat<TTaskSchedulerCondition>& aConditionList,
|
sl@0
|
228 |
const TTsTime& aDefaultTime)
|
sl@0
|
229 |
{
|
sl@0
|
230 |
aSchedule.ReplaceConditionsL(aConditionList);
|
sl@0
|
231 |
|
sl@0
|
232 |
//Default Time is represented by a single entry class
|
sl@0
|
233 |
CArrayFixFlat<TScheduleEntryInfo2>* entries
|
sl@0
|
234 |
= new(ELeave) CArrayFixFlat<TScheduleEntryInfo2>(1);
|
sl@0
|
235 |
CleanupStack::PushL(entries);
|
sl@0
|
236 |
TScheduleEntryInfo2 info;
|
sl@0
|
237 |
info.SetStartTime(aDefaultTime);
|
sl@0
|
238 |
info.SetInterval(1);
|
sl@0
|
239 |
info.SetIntervalType(EDaily);
|
sl@0
|
240 |
//validityperiod of 24 hours will ensure task is always run
|
sl@0
|
241 |
info.SetValidityPeriod(60*24);
|
sl@0
|
242 |
entries->AppendL(info);
|
sl@0
|
243 |
aSchedule.ReplaceEntriesL(*entries);
|
sl@0
|
244 |
CleanupStack::Pop(entries);
|
sl@0
|
245 |
|
sl@0
|
246 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, aSchedule);
|
sl@0
|
247 |
}
|
sl@0
|
248 |
|
sl@0
|
249 |
void CTaskScheduler::EditScheduleL(TInt aScheduleHandle,
|
sl@0
|
250 |
CArrayFixFlat<TTaskSchedulerCondition>& aConditionList,
|
sl@0
|
251 |
const TTsTime& aDefaultTime)
|
sl@0
|
252 |
{
|
sl@0
|
253 |
CSchedule* schedule = FindL(aScheduleHandle);
|
sl@0
|
254 |
|
sl@0
|
255 |
// remove schedule from condition manager before replacing entries to ensure
|
sl@0
|
256 |
// its deleted properly.
|
sl@0
|
257 |
iScheduleCriteriaManager->RemoveSchedule(schedule->Id());
|
sl@0
|
258 |
|
sl@0
|
259 |
TRAPD(err, DoEditScheduleL(*schedule, aConditionList, aDefaultTime));
|
sl@0
|
260 |
if(err)
|
sl@0
|
261 |
{
|
sl@0
|
262 |
schedule->RemoveEntries();
|
sl@0
|
263 |
schedule->RemoveConditions();
|
sl@0
|
264 |
User::Leave(err);
|
sl@0
|
265 |
}
|
sl@0
|
266 |
|
sl@0
|
267 |
// recalculate due time only if schedule is enabled and has tasks to run
|
sl@0
|
268 |
if (IsScheduleReadyForUpdate(*schedule))
|
sl@0
|
269 |
{
|
sl@0
|
270 |
iScheduleCriteriaManager->ReplaceScheduleL(*schedule);
|
sl@0
|
271 |
}
|
sl@0
|
272 |
}
|
sl@0
|
273 |
|
sl@0
|
274 |
void CTaskScheduler::RemoveScheduleL(TInt aHandle)
|
sl@0
|
275 |
{
|
sl@0
|
276 |
CSchedule* schedule = FindL(aHandle);
|
sl@0
|
277 |
LOGSTRING3("CTaskScheduler::RemoveScheduleL - schedule: %S, %d", &schedule->Name(), schedule->Id());
|
sl@0
|
278 |
if (!schedule->HasTasks())
|
sl@0
|
279 |
{
|
sl@0
|
280 |
LOGSTRING("CTaskScheduler::RemoveScheduleL - schedule doesn't have any tasks, removing");
|
sl@0
|
281 |
//remove schedule from timer
|
sl@0
|
282 |
iScheduleCriteriaManager->RemoveSchedule(schedule->Id());
|
sl@0
|
283 |
DoRemoveL(schedule);
|
sl@0
|
284 |
}
|
sl@0
|
285 |
else
|
sl@0
|
286 |
{
|
sl@0
|
287 |
// Can't delete a schedule which has tasks
|
sl@0
|
288 |
LOGSTRING("CTaskScheduler::RemoveScheduleL - schedule has tasks, can't delete");
|
sl@0
|
289 |
User::Leave(KErrArgument);
|
sl@0
|
290 |
}
|
sl@0
|
291 |
}
|
sl@0
|
292 |
|
sl@0
|
293 |
void CTaskScheduler::DisableScheduleL(TInt aHandle)
|
sl@0
|
294 |
{
|
sl@0
|
295 |
CSchedule* schedule = FindL(aHandle);
|
sl@0
|
296 |
schedule->SetEnabled(EFalse);
|
sl@0
|
297 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, *schedule);
|
sl@0
|
298 |
//remove schedule from timer as its disabled
|
sl@0
|
299 |
iScheduleCriteriaManager->RemoveSchedule(schedule->Id());
|
sl@0
|
300 |
}
|
sl@0
|
301 |
|
sl@0
|
302 |
void CTaskScheduler::EnableScheduleL(TInt aHandle)
|
sl@0
|
303 |
{
|
sl@0
|
304 |
CSchedule* schedule = FindL(aHandle);
|
sl@0
|
305 |
schedule->SetEnabled(ETrue);
|
sl@0
|
306 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, *schedule);
|
sl@0
|
307 |
|
sl@0
|
308 |
// recalculate due time only if schedule has tasks to run
|
sl@0
|
309 |
if ( IsScheduleReadyForUpdate(*schedule))
|
sl@0
|
310 |
{
|
sl@0
|
311 |
iScheduleCriteriaManager->ReplaceScheduleL(*schedule);
|
sl@0
|
312 |
}
|
sl@0
|
313 |
}
|
sl@0
|
314 |
|
sl@0
|
315 |
void CTaskScheduler::ScheduleTaskL(CSchedule& aSchedule, CClientProxy& aClient)
|
sl@0
|
316 |
{
|
sl@0
|
317 |
// Backup the task
|
sl@0
|
318 |
if (aSchedule.Persists())
|
sl@0
|
319 |
{
|
sl@0
|
320 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, aClient);
|
sl@0
|
321 |
}
|
sl@0
|
322 |
// if schedule is enabled then add schedule to timer.
|
sl@0
|
323 |
if (aSchedule.Enabled() && IsStartupStateNonCritical())
|
sl@0
|
324 |
{
|
sl@0
|
325 |
iScheduleCriteriaManager->ReplaceScheduleL(aSchedule);
|
sl@0
|
326 |
}
|
sl@0
|
327 |
}
|
sl@0
|
328 |
|
sl@0
|
329 |
void CTaskScheduler::DeleteTaskL(TInt aScheduleHandle, TInt aTaskHandle)
|
sl@0
|
330 |
{
|
sl@0
|
331 |
CSchedule* schedule = FindL(aScheduleHandle);
|
sl@0
|
332 |
|
sl@0
|
333 |
TScheduledTask* task = schedule->Task(aTaskHandle);
|
sl@0
|
334 |
if (!task)
|
sl@0
|
335 |
{
|
sl@0
|
336 |
LOGSTRING("CTaskScheduler::DeleteTaskL - task wasn't found");
|
sl@0
|
337 |
User::Leave(KErrNotFound);
|
sl@0
|
338 |
}
|
sl@0
|
339 |
|
sl@0
|
340 |
const CClientProxy& clientForTask = task->Client();
|
sl@0
|
341 |
|
sl@0
|
342 |
// This deletes the task and removes the CScheduledTask
|
sl@0
|
343 |
// from the CClientProxy's queue of tasks
|
sl@0
|
344 |
task->RemoveInfo();
|
sl@0
|
345 |
|
sl@0
|
346 |
// This deletes the TScheduledTask defined above and removes it
|
sl@0
|
347 |
// from CSchedule's queue of TScheduledTask's.
|
sl@0
|
348 |
schedule->RemoveTask(task);
|
sl@0
|
349 |
if (!schedule->HasTasks()) //i.e. it was the last task
|
sl@0
|
350 |
{
|
sl@0
|
351 |
LOGSTRING("CTaskScheduler::DeleteTaskL - schedule doesn't have any more tasks left");
|
sl@0
|
352 |
//remove scheule from timer as there are no more tasks
|
sl@0
|
353 |
iScheduleCriteriaManager->RemoveSchedule(schedule->Id());
|
sl@0
|
354 |
// If the schedule isn't persistent then we delete it (transient schedules only
|
sl@0
|
355 |
// have one task).
|
sl@0
|
356 |
if (!schedule->Persists())
|
sl@0
|
357 |
DoRemoveL(schedule);
|
sl@0
|
358 |
else
|
sl@0
|
359 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, clientForTask);
|
sl@0
|
360 |
}
|
sl@0
|
361 |
else
|
sl@0
|
362 |
{
|
sl@0
|
363 |
// Backup the changes to the tasks. Although we are deleting a task, we are not actually
|
sl@0
|
364 |
// deleting the client, so this is actually an edit operation.
|
sl@0
|
365 |
if (schedule->Persists())
|
sl@0
|
366 |
iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, clientForTask);
|
sl@0
|
367 |
}
|
sl@0
|
368 |
}
|
sl@0
|
369 |
|
sl@0
|
370 |
//
|
sl@0
|
371 |
// Utility Functions
|
sl@0
|
372 |
//
|
sl@0
|
373 |
|
sl@0
|
374 |
TInt CTaskScheduler::GenerateId()
|
sl@0
|
375 |
{
|
sl@0
|
376 |
TInt id = KMinScheduleId;
|
sl@0
|
377 |
CSchedule* schedule = Find(id);
|
sl@0
|
378 |
while (schedule!=NULL)
|
sl@0
|
379 |
{
|
sl@0
|
380 |
id+=KScheduleIdDifferential;//=10 000
|
sl@0
|
381 |
if ((id/KScheduleIdDifferential) > KMaxSchedules)
|
sl@0
|
382 |
return KErrOverflow;
|
sl@0
|
383 |
schedule = Find(id);
|
sl@0
|
384 |
}
|
sl@0
|
385 |
return id;
|
sl@0
|
386 |
}
|
sl@0
|
387 |
|
sl@0
|
388 |
void CTaskScheduler::DoRemoveL(CSchedule* aSchedule)
|
sl@0
|
389 |
{
|
sl@0
|
390 |
TRAPD(err, iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationDelete, *aSchedule));
|
sl@0
|
391 |
if (err < KErrNone && err != KErrNotFound)
|
sl@0
|
392 |
User::Leave(err);
|
sl@0
|
393 |
iSchedules.Remove(*aSchedule);
|
sl@0
|
394 |
delete aSchedule;
|
sl@0
|
395 |
aSchedule = NULL;
|
sl@0
|
396 |
}
|
sl@0
|
397 |
|
sl@0
|
398 |
CSchedule* CTaskScheduler::FindL(TInt aHandle)
|
sl@0
|
399 |
{
|
sl@0
|
400 |
CSchedule* schedule = Find(aHandle);
|
sl@0
|
401 |
if (!schedule)
|
sl@0
|
402 |
User::Leave(KErrNotFound);
|
sl@0
|
403 |
return schedule;
|
sl@0
|
404 |
}
|
sl@0
|
405 |
|
sl@0
|
406 |
CSchedule* CTaskScheduler::Find(TInt aHandle)
|
sl@0
|
407 |
{
|
sl@0
|
408 |
TSglQueIter<CSchedule> scheduleIter(iSchedules);
|
sl@0
|
409 |
scheduleIter.SetToFirst();
|
sl@0
|
410 |
CSchedule* schedule;
|
sl@0
|
411 |
while ((schedule = scheduleIter++)!=NULL)
|
sl@0
|
412 |
{
|
sl@0
|
413 |
if (schedule->Id() == aHandle)
|
sl@0
|
414 |
return schedule;
|
sl@0
|
415 |
}
|
sl@0
|
416 |
return NULL;
|
sl@0
|
417 |
}
|
sl@0
|
418 |
|
sl@0
|
419 |
// If aRefArray is NULL then only count it returned.
|
sl@0
|
420 |
TInt CTaskScheduler::GetScheduleRefsL(CArrayFixFlat<TSchedulerItemRef>* aRefArray,
|
sl@0
|
421 |
TScheduleFilter aFilter,
|
sl@0
|
422 |
const RMessagePtr2& aMessage)
|
sl@0
|
423 |
{
|
sl@0
|
424 |
TInt count = 0;
|
sl@0
|
425 |
TSglQueIter<CSchedule> iter(iSchedules);
|
sl@0
|
426 |
iter.SetToFirst();
|
sl@0
|
427 |
CSchedule* schedule = NULL;
|
sl@0
|
428 |
while ((schedule = iter++) != NULL)
|
sl@0
|
429 |
{
|
sl@0
|
430 |
if(aFilter == EAllSchedules || (schedule->Enabled() && schedule->HasTasks()))
|
sl@0
|
431 |
{
|
sl@0
|
432 |
//only add information for schedules that the client has permission to alter
|
sl@0
|
433 |
if(schedule->IsAccessAllowed(aMessage))
|
sl@0
|
434 |
{
|
sl@0
|
435 |
if(aRefArray)
|
sl@0
|
436 |
{
|
sl@0
|
437 |
TSchedulerItemRef ref;
|
sl@0
|
438 |
ref.iHandle = schedule->Id();
|
sl@0
|
439 |
ref.iName = schedule->Name();
|
sl@0
|
440 |
aRefArray->AppendL(ref);
|
sl@0
|
441 |
}
|
sl@0
|
442 |
count++;
|
sl@0
|
443 |
}
|
sl@0
|
444 |
}
|
sl@0
|
445 |
}
|
sl@0
|
446 |
return count;
|
sl@0
|
447 |
}
|
sl@0
|
448 |
|
sl@0
|
449 |
// If aRefArray is NULL then only count it returned.
|
sl@0
|
450 |
TInt CTaskScheduler::GetTaskRefsL(CArrayFixFlat<TSchedulerItemRef>* aRefArray,
|
sl@0
|
451 |
TScheduleFilter aScheduleFilter,
|
sl@0
|
452 |
TTaskFilter aTaskFilter,
|
sl@0
|
453 |
CClientProxy* aClient,
|
sl@0
|
454 |
const RMessagePtr2& aMessage)
|
sl@0
|
455 |
{
|
sl@0
|
456 |
TInt count = 0;
|
sl@0
|
457 |
TSglQueIter<CSchedule> iter(iSchedules);
|
sl@0
|
458 |
iter.SetToFirst();
|
sl@0
|
459 |
CSchedule* schedule = NULL;
|
sl@0
|
460 |
while ((schedule = iter++) != NULL)
|
sl@0
|
461 |
{
|
sl@0
|
462 |
if(aScheduleFilter == EAllSchedules || (schedule->Enabled() && schedule->HasTasks()))
|
sl@0
|
463 |
{
|
sl@0
|
464 |
//only add information for schedules that the client has permission to alter
|
sl@0
|
465 |
if(schedule->IsAccessAllowed(aMessage))
|
sl@0
|
466 |
{
|
sl@0
|
467 |
TSglQueIter<TScheduledTask> taskIter(*(schedule->Tasks()));
|
sl@0
|
468 |
taskIter.SetToFirst();
|
sl@0
|
469 |
TScheduledTask* task;
|
sl@0
|
470 |
while ((task=taskIter++)!=NULL)
|
sl@0
|
471 |
{
|
sl@0
|
472 |
if (aTaskFilter==EAllTasks||&task->Client() == aClient) // This pointer comparison is a bit rubbish. Change?
|
sl@0
|
473 |
{
|
sl@0
|
474 |
if(aRefArray)
|
sl@0
|
475 |
{
|
sl@0
|
476 |
TTaskInfo info = task->Info();
|
sl@0
|
477 |
TSchedulerItemRef ref;
|
sl@0
|
478 |
ref.iHandle = info.iTaskId;
|
sl@0
|
479 |
ref.iName = info.iName;
|
sl@0
|
480 |
aRefArray->AppendL(ref);
|
sl@0
|
481 |
}
|
sl@0
|
482 |
count++;
|
sl@0
|
483 |
}
|
sl@0
|
484 |
}
|
sl@0
|
485 |
}
|
sl@0
|
486 |
}
|
sl@0
|
487 |
}
|
sl@0
|
488 |
return count;
|
sl@0
|
489 |
}
|
sl@0
|
490 |
|
sl@0
|
491 |
|
sl@0
|
492 |
//
|
sl@0
|
493 |
// Schedule Execution functions
|
sl@0
|
494 |
//
|
sl@0
|
495 |
|
sl@0
|
496 |
// A schedule is ready to be run
|
sl@0
|
497 |
void CTaskScheduler::DueTaskNotifyL(TInt aScheduleHandle)
|
sl@0
|
498 |
{
|
sl@0
|
499 |
CSchedule* schedule = FindL(aScheduleHandle);
|
sl@0
|
500 |
//NotifyTasks() also removes tasks from the schedule if there are no
|
sl@0
|
501 |
//repeats left.
|
sl@0
|
502 |
schedule->NotifyTasks();
|
sl@0
|
503 |
|
sl@0
|
504 |
if (!schedule->HasTasks())
|
sl@0
|
505 |
{
|
sl@0
|
506 |
// remove schedule.
|
sl@0
|
507 |
TRAPD(ignore, DoRemoveL(schedule));
|
sl@0
|
508 |
//??error only occurs in relation to persistence!! Do something.
|
sl@0
|
509 |
UNUSED_VAR(ignore);
|
sl@0
|
510 |
}
|
sl@0
|
511 |
else
|
sl@0
|
512 |
{
|
sl@0
|
513 |
__ASSERT_ALWAYS(IsStartupStateNonCritical(), User::Invariant());
|
sl@0
|
514 |
iScheduleCriteriaManager->ReplaceScheduleL(*schedule,EConditionAndTime,ETrue);
|
sl@0
|
515 |
}
|
sl@0
|
516 |
|
sl@0
|
517 |
// Execute all clients. This method doesn't leave as all errors are either
|
sl@0
|
518 |
// logged in the log engine or handled elsewhere.
|
sl@0
|
519 |
ExecuteClients();
|
sl@0
|
520 |
}
|
sl@0
|
521 |
|
sl@0
|
522 |
// Go through all clients, executing their tasks
|
sl@0
|
523 |
void CTaskScheduler::ExecuteClients(TBool aUpdateClient)
|
sl@0
|
524 |
{
|
sl@0
|
525 |
if ((BUROperationInProgress() == EBUROperationNoActivity) || !aUpdateClient)
|
sl@0
|
526 |
{
|
sl@0
|
527 |
TDblQueIter<CClientProxy> clientIter(iClients);
|
sl@0
|
528 |
clientIter.SetToFirst();
|
sl@0
|
529 |
CClientProxy* client;
|
sl@0
|
530 |
while( (client = clientIter++) != NULL)
|
sl@0
|
531 |
{
|
sl@0
|
532 |
// Does this client have anything ready to run?
|
sl@0
|
533 |
if (client->IsReadyToExecute())
|
sl@0
|
534 |
{
|
sl@0
|
535 |
client->ExecuteTasks();
|
sl@0
|
536 |
// Clears the 'IsReadyToExecute' flag...
|
sl@0
|
537 |
client->RemoveDueTasks();
|
sl@0
|
538 |
}
|
sl@0
|
539 |
}
|
sl@0
|
540 |
if (aUpdateClient)
|
sl@0
|
541 |
{
|
sl@0
|
542 |
// Update the store file now
|
sl@0
|
543 |
UpdateClients();
|
sl@0
|
544 |
}
|
sl@0
|
545 |
}
|
sl@0
|
546 |
else
|
sl@0
|
547 |
{
|
sl@0
|
548 |
// Sets the flag to trigger delayed store operation when BUR ends
|
sl@0
|
549 |
iTaskExecutedDuringBUR = ETrue;
|
sl@0
|
550 |
}
|
sl@0
|
551 |
}
|
sl@0
|
552 |
|
sl@0
|
553 |
// Go through all clients, update the store file with modified client info
|
sl@0
|
554 |
void CTaskScheduler::UpdateClients()
|
sl@0
|
555 |
{
|
sl@0
|
556 |
// iterate the client list to perform delayed update of the store file
|
sl@0
|
557 |
TDblQueIter<CClientProxy> clientIter(iClients);
|
sl@0
|
558 |
clientIter.SetToFirst();
|
sl@0
|
559 |
CClientProxy* client;
|
sl@0
|
560 |
while( (client = clientIter++) != NULL)
|
sl@0
|
561 |
{
|
sl@0
|
562 |
if (!client->Users())
|
sl@0
|
563 |
{
|
sl@0
|
564 |
// Remove from store
|
sl@0
|
565 |
TRAPD(ignore, iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationDelete, *client));
|
sl@0
|
566 |
//?? if ignore is not KErrNone then there is a problem with the store
|
sl@0
|
567 |
UNUSED_VAR(ignore);
|
sl@0
|
568 |
// Remove client & delete it
|
sl@0
|
569 |
client->Remove();
|
sl@0
|
570 |
delete client;
|
sl@0
|
571 |
}
|
sl@0
|
572 |
else
|
sl@0
|
573 |
{
|
sl@0
|
574 |
// Update this clients data in the store...
|
sl@0
|
575 |
TRAPD(ignore, iBackupManager->PerformStoreOperationL(CSchBackupManager::ESchBackupOperationEdit, *client));
|
sl@0
|
576 |
|
sl@0
|
577 |
//?? if ignore is not KErrNone then there is a problem with the store
|
sl@0
|
578 |
UNUSED_VAR(ignore);
|
sl@0
|
579 |
}
|
sl@0
|
580 |
}
|
sl@0
|
581 |
}
|
sl@0
|
582 |
|
sl@0
|
583 |
//
|
sl@0
|
584 |
// Environment change functions
|
sl@0
|
585 |
//
|
sl@0
|
586 |
|
sl@0
|
587 |
TInt CTaskScheduler::EnvironmentChanged(TAny* aScheduler)
|
sl@0
|
588 |
{
|
sl@0
|
589 |
CTaskScheduler* self = reinterpret_cast<CTaskScheduler*>(aScheduler);
|
sl@0
|
590 |
self->HandleEnvironmentChange();
|
sl@0
|
591 |
return KErrNone;
|
sl@0
|
592 |
}
|
sl@0
|
593 |
|
sl@0
|
594 |
void CTaskScheduler::HandleEnvironmentChange()
|
sl@0
|
595 |
{
|
sl@0
|
596 |
// If staged startup still in critical region, can safely
|
sl@0
|
597 |
// ignore system time change.
|
sl@0
|
598 |
if (!IsStartupStateNonCritical())
|
sl@0
|
599 |
{
|
sl@0
|
600 |
return;
|
sl@0
|
601 |
}
|
sl@0
|
602 |
|
sl@0
|
603 |
TInt changes=iNotifier->Change();
|
sl@0
|
604 |
if (changes & EChangesSystemTime)
|
sl@0
|
605 |
{
|
sl@0
|
606 |
#ifdef __SCHLOGGING__
|
sl@0
|
607 |
{
|
sl@0
|
608 |
TTime time;
|
sl@0
|
609 |
time.HomeTime();
|
sl@0
|
610 |
TDateTime due(time.DateTime());
|
sl@0
|
611 |
LOGSTRING7("CTaskScheduler::HandleEnvironmentChangeL - system time is now: [%02d/%02d/%d] @ %02d:%02d:%02d", due.Day(), (TInt) due.Month() + 1, due.Year(), due.Hour(), due.Minute(), due.Second());
|
sl@0
|
612 |
}
|
sl@0
|
613 |
#endif
|
sl@0
|
614 |
|
sl@0
|
615 |
|
sl@0
|
616 |
// Cannot use AddSchedulesToTimerL() because this method
|
sl@0
|
617 |
// uses the non-condition version of ReplaceScheduleL.
|
sl@0
|
618 |
TSglQueIter<CSchedule> scheduleIter(iSchedules);
|
sl@0
|
619 |
scheduleIter.SetToFirst();
|
sl@0
|
620 |
CSchedule* schedule = NULL;
|
sl@0
|
621 |
while ((schedule=scheduleIter++)!=NULL)
|
sl@0
|
622 |
{
|
sl@0
|
623 |
if(IsScheduleReadyForUpdate(*schedule))
|
sl@0
|
624 |
{
|
sl@0
|
625 |
TRAPD(err, iScheduleCriteriaManager->ReplaceScheduleL(*schedule, EOnlyTime));
|
sl@0
|
626 |
UNUSED_VAR(err);
|
sl@0
|
627 |
}
|
sl@0
|
628 |
}
|
sl@0
|
629 |
}
|
sl@0
|
630 |
}
|
sl@0
|
631 |
|
sl@0
|
632 |
void CTaskScheduler::AddSchedulesToTimerL()
|
sl@0
|
633 |
{
|
sl@0
|
634 |
if (!IsStartupStateNonCritical())
|
sl@0
|
635 |
{
|
sl@0
|
636 |
return; // not ready
|
sl@0
|
637 |
}
|
sl@0
|
638 |
|
sl@0
|
639 |
TInt ret = KErrNone;
|
sl@0
|
640 |
|
sl@0
|
641 |
TSglQueIter<CSchedule> scheduleIter(iSchedules);
|
sl@0
|
642 |
scheduleIter.SetToFirst();
|
sl@0
|
643 |
CSchedule* schedule = NULL;
|
sl@0
|
644 |
while ((schedule=scheduleIter++)!=NULL)
|
sl@0
|
645 |
{
|
sl@0
|
646 |
if(IsScheduleReadyForUpdate(*schedule))
|
sl@0
|
647 |
{
|
sl@0
|
648 |
TRAPD(err, iScheduleCriteriaManager->ReplaceScheduleL(*schedule));
|
sl@0
|
649 |
if (err != KErrNone)
|
sl@0
|
650 |
{
|
sl@0
|
651 |
ret = err;
|
sl@0
|
652 |
}
|
sl@0
|
653 |
}
|
sl@0
|
654 |
}
|
sl@0
|
655 |
User::LeaveIfError(ret);
|
sl@0
|
656 |
}
|
sl@0
|
657 |
|
sl@0
|
658 |
void CTaskScheduler::CleanupScheduledTasksL()
|
sl@0
|
659 |
{
|
sl@0
|
660 |
RFs fs;
|
sl@0
|
661 |
_LIT(KTempFilePath, "_:\\private\\10005399\\*.tmp");
|
sl@0
|
662 |
TBuf<32> filePath(KTempFilePath);
|
sl@0
|
663 |
|
sl@0
|
664 |
filePath[0] = RFs::GetSystemDriveChar();
|
sl@0
|
665 |
|
sl@0
|
666 |
fs.Connect();
|
sl@0
|
667 |
CleanupClosePushL(fs);
|
sl@0
|
668 |
|
sl@0
|
669 |
CFileMan* fileMan = CFileMan::NewL(fs);
|
sl@0
|
670 |
CleanupStack::PushL(fileMan);
|
sl@0
|
671 |
|
sl@0
|
672 |
//Delete all temporary files in the private folder
|
sl@0
|
673 |
fileMan->Delete(filePath,0);
|
sl@0
|
674 |
|
sl@0
|
675 |
//Pop and destroy fs and fileMan
|
sl@0
|
676 |
//This will call fs.Close() so no need to call it explicitly
|
sl@0
|
677 |
CleanupStack::PopAndDestroy(2);
|
sl@0
|
678 |
}
|
sl@0
|
679 |
|
sl@0
|
680 |
/**
|
sl@0
|
681 |
CSchStartupStateMgr calls this to notify startup state changes of
|
sl@0
|
682 |
interest.
|
sl@0
|
683 |
|
sl@0
|
684 |
@internalComponent
|
sl@0
|
685 |
*/
|
sl@0
|
686 |
void CTaskScheduler::ProcessSSAEventL(TStartupStateIdentifier aKnownState)
|
sl@0
|
687 |
{
|
sl@0
|
688 |
LOGSTRING2("ProcessSSAEventL receive SS 0x%x", aKnownState);
|
sl@0
|
689 |
|
sl@0
|
690 |
if (! IsStartupStateNonCritical() &&
|
sl@0
|
691 |
(aKnownState >= KSchFinalStartupState))
|
sl@0
|
692 |
{
|
sl@0
|
693 |
iStartupStatePassNonCritical = ETrue;
|
sl@0
|
694 |
CleanupScheduledTasksL();
|
sl@0
|
695 |
AddSchedulesToTimerL();
|
sl@0
|
696 |
}
|
sl@0
|
697 |
}
|
sl@0
|
698 |
|
sl@0
|
699 |
/**
|
sl@0
|
700 |
Returns ETrue if Start-up State is NonCritical
|
sl@0
|
701 |
*/
|
sl@0
|
702 |
TBool CTaskScheduler::IsStartupStateNonCritical()
|
sl@0
|
703 |
{
|
sl@0
|
704 |
return iStartupStatePassNonCritical;
|
sl@0
|
705 |
}
|
sl@0
|
706 |
|
sl@0
|
707 |
/**
|
sl@0
|
708 |
Check schedule is valid
|
sl@0
|
709 |
*/
|
sl@0
|
710 |
TBool CTaskScheduler::IsScheduleReadyForUpdate(CSchedule& aSchedule)
|
sl@0
|
711 |
{
|
sl@0
|
712 |
if(aSchedule.IsUpdatable() && IsStartupStateNonCritical())
|
sl@0
|
713 |
return ETrue;
|
sl@0
|
714 |
else
|
sl@0
|
715 |
return EFalse;
|
sl@0
|
716 |
}
|
sl@0
|
717 |
|
sl@0
|
718 |
/**
|
sl@0
|
719 |
babackup server calls this to notify backup operations. The attributes are read and translated to determine
|
sl@0
|
720 |
which operation is actually in progress.
|
sl@0
|
721 |
@internalComponent
|
sl@0
|
722 |
*/
|
sl@0
|
723 |
void CTaskScheduler::HandleBackupOperationEventL(const TBackupOperationAttributes& aBackupOperationAttributes)
|
sl@0
|
724 |
{
|
sl@0
|
725 |
TBUROperation type;
|
sl@0
|
726 |
|
sl@0
|
727 |
// determine the operation type (backup or restore)
|
sl@0
|
728 |
switch(aBackupOperationAttributes.iFileFlag)
|
sl@0
|
729 |
{
|
sl@0
|
730 |
case MBackupObserver::EReleaseLockReadOnly:
|
sl@0
|
731 |
type = EBUROperationBackup;
|
sl@0
|
732 |
break;
|
sl@0
|
733 |
case MBackupObserver::EReleaseLockNoAccess:
|
sl@0
|
734 |
type = EBUROperationRestore;
|
sl@0
|
735 |
break;
|
sl@0
|
736 |
case MBackupObserver::ETakeLock:
|
sl@0
|
737 |
// No information is passed from babackup server, so we need to depend on our own memory
|
sl@0
|
738 |
type = iBUROperationInProgress;
|
sl@0
|
739 |
break;
|
sl@0
|
740 |
default:
|
sl@0
|
741 |
type = EBUROperationNoActivity;
|
sl@0
|
742 |
break;
|
sl@0
|
743 |
}
|
sl@0
|
744 |
|
sl@0
|
745 |
// determine the operation status (e.g. starting, ending)
|
sl@0
|
746 |
switch(aBackupOperationAttributes.iOperation)
|
sl@0
|
747 |
{
|
sl@0
|
748 |
case EStart:
|
sl@0
|
749 |
BURBeginningL(type);
|
sl@0
|
750 |
break;
|
sl@0
|
751 |
case EEnd:
|
sl@0
|
752 |
BURCompleteL(type, EBUROperationSuccess);
|
sl@0
|
753 |
break;
|
sl@0
|
754 |
case EAbort:
|
sl@0
|
755 |
BURCompleteL(type, EBUROperationAbort);
|
sl@0
|
756 |
break;
|
sl@0
|
757 |
default:
|
sl@0
|
758 |
break;
|
sl@0
|
759 |
}
|
sl@0
|
760 |
}
|
sl@0
|
761 |
|
sl@0
|
762 |
|
sl@0
|
763 |
/**
|
sl@0
|
764 |
This function is called to notify when a Backup or Restore operation is commencing.
|
sl@0
|
765 |
|
sl@0
|
766 |
@internalComponent
|
sl@0
|
767 |
*/
|
sl@0
|
768 |
void CTaskScheduler::BURBeginningL(TBUROperation aOperationType)
|
sl@0
|
769 |
{
|
sl@0
|
770 |
// This will stop the API calls that directly modify the store file
|
sl@0
|
771 |
iBUROperationInProgress = aOperationType;
|
sl@0
|
772 |
|
sl@0
|
773 |
//cancel background compaction of store during backup/restore
|
sl@0
|
774 |
iBackupManager->Cancel();
|
sl@0
|
775 |
}
|
sl@0
|
776 |
|
sl@0
|
777 |
|
sl@0
|
778 |
/**
|
sl@0
|
779 |
This function is called to notify when a Backup or Restore operation is finished.
|
sl@0
|
780 |
|
sl@0
|
781 |
@internalComponent
|
sl@0
|
782 |
*/
|
sl@0
|
783 |
void CTaskScheduler::BURCompleteL(TBUROperation aOperationType, TBUROperationResult aBURResult)
|
sl@0
|
784 |
{
|
sl@0
|
785 |
// If there is a successful restore, this means that we have a different store file then we were using
|
sl@0
|
786 |
// so we have to internalize and use that file. In any other case, we can proceed with the delayed
|
sl@0
|
787 |
// updates to the old file
|
sl@0
|
788 |
if ((aOperationType == EBUROperationRestore)&&(aBURResult == EBUROperationSuccess))
|
sl@0
|
789 |
{
|
sl@0
|
790 |
LOGSTRING("CTaskScheduler::BURCompleteL - Restoring clients and schedules after successful restore");
|
sl@0
|
791 |
|
sl@0
|
792 |
//First check whether any task expires during the restore process now that this is completed
|
sl@0
|
793 |
if (iTaskExecutedDuringBUR)
|
sl@0
|
794 |
{
|
sl@0
|
795 |
// performed the delayed task execution but with no externalizing as we dont want to modify
|
sl@0
|
796 |
// the just restored file as after the delayed execution, all persistent schedule will be removed
|
sl@0
|
797 |
ExecuteClients(EFalse);
|
sl@0
|
798 |
iTaskExecutedDuringBUR=EFalse;
|
sl@0
|
799 |
}
|
sl@0
|
800 |
|
sl@0
|
801 |
// Now remove existing persistent schedules, tasks and clients
|
sl@0
|
802 |
TSglQueIter<CSchedule> scheduleIter(iSchedules);
|
sl@0
|
803 |
scheduleIter.SetToFirst();
|
sl@0
|
804 |
CSchedule* schedule;
|
sl@0
|
805 |
while ((schedule = scheduleIter++)!=NULL)
|
sl@0
|
806 |
{
|
sl@0
|
807 |
if (schedule->Persists())
|
sl@0
|
808 |
{
|
sl@0
|
809 |
iSchedules.Remove(*schedule);
|
sl@0
|
810 |
schedule->RemoveTasks(ETrue);
|
sl@0
|
811 |
delete schedule;
|
sl@0
|
812 |
}
|
sl@0
|
813 |
}
|
sl@0
|
814 |
|
sl@0
|
815 |
CClientProxy* client;
|
sl@0
|
816 |
TDblQueIter<CClientProxy> clientIter(iClients);
|
sl@0
|
817 |
|
sl@0
|
818 |
// remove clients which don't have any associated tasks left (tasks in persistent schedules are
|
sl@0
|
819 |
// already removed, but the client might have transient schedules as well)
|
sl@0
|
820 |
clientIter.SetToFirst();
|
sl@0
|
821 |
while ((client = clientIter++) != NULL)
|
sl@0
|
822 |
{
|
sl@0
|
823 |
TDblQueIter<CScheduledTask> taskIter = client->TaskIterator();
|
sl@0
|
824 |
taskIter.SetToFirst();
|
sl@0
|
825 |
|
sl@0
|
826 |
// remove client if no more tasks
|
sl@0
|
827 |
if (taskIter++ == NULL)
|
sl@0
|
828 |
{
|
sl@0
|
829 |
client->Remove();
|
sl@0
|
830 |
delete client; // removes associated tasks
|
sl@0
|
831 |
}
|
sl@0
|
832 |
else
|
sl@0
|
833 |
{
|
sl@0
|
834 |
//remove any persisted task and if the client only has persisted task, remove client as well
|
sl@0
|
835 |
taskIter.SetToFirst();
|
sl@0
|
836 |
CScheduledTask* task;
|
sl@0
|
837 |
TInt taskCount=0;
|
sl@0
|
838 |
TInt persistCount=0;
|
sl@0
|
839 |
while ((task = taskIter++) != NULL)
|
sl@0
|
840 |
{
|
sl@0
|
841 |
taskCount++;
|
sl@0
|
842 |
if (task->Persists())
|
sl@0
|
843 |
{
|
sl@0
|
844 |
persistCount++;
|
sl@0
|
845 |
client->RemoveTask(task);
|
sl@0
|
846 |
}
|
sl@0
|
847 |
}
|
sl@0
|
848 |
//if after removing the persist tasks, there are no more other tasks, we can remove the client too
|
sl@0
|
849 |
if (taskCount==persistCount)
|
sl@0
|
850 |
{
|
sl@0
|
851 |
client->Remove();
|
sl@0
|
852 |
delete client;
|
sl@0
|
853 |
}
|
sl@0
|
854 |
}
|
sl@0
|
855 |
}
|
sl@0
|
856 |
|
sl@0
|
857 |
// now re-read the clients and schedules from the restored store file
|
sl@0
|
858 |
TRAPD(err, iBackupManager->RestoreL(iClients, iSchedules,*iSchLogManager,ETrue));
|
sl@0
|
859 |
|
sl@0
|
860 |
if (err != KErrNone) // the file's corrupt or something...
|
sl@0
|
861 |
{
|
sl@0
|
862 |
LOGSTRING2("CTaskScheduler::BURCompleteL - had to create new store because of error: %d", err);
|
sl@0
|
863 |
iBackupManager->CreateEmptyBackupL();
|
sl@0
|
864 |
}
|
sl@0
|
865 |
|
sl@0
|
866 |
|
sl@0
|
867 |
// Each client now contains a list of associated tasks. We need
|
sl@0
|
868 |
// to now associate those tasks with specific schedules
|
sl@0
|
869 |
clientIter.SetToFirst();
|
sl@0
|
870 |
while ((client = clientIter++) != NULL)
|
sl@0
|
871 |
{
|
sl@0
|
872 |
// Fetch an iterator for each task owned by this client
|
sl@0
|
873 |
CScheduledTask* task;
|
sl@0
|
874 |
TDblQueIter<CScheduledTask> taskIterator = client->TaskIterator();
|
sl@0
|
875 |
taskIterator.SetToFirst();
|
sl@0
|
876 |
|
sl@0
|
877 |
// Iterate through all the tasks owned by this client, trying to find
|
sl@0
|
878 |
// the corresponding schedules.
|
sl@0
|
879 |
while ((task = taskIterator++) != NULL)
|
sl@0
|
880 |
{
|
sl@0
|
881 |
TSglQueIter<CSchedule> persScheduleIter(iSchedules);
|
sl@0
|
882 |
persScheduleIter.SetToFirst();
|
sl@0
|
883 |
CSchedule* persSchedule;
|
sl@0
|
884 |
while ((persSchedule = persScheduleIter++)!=NULL)
|
sl@0
|
885 |
{
|
sl@0
|
886 |
if ((persSchedule->Persists())&&(persSchedule->Id() == task->ScheduleId()))
|
sl@0
|
887 |
{
|
sl@0
|
888 |
TScheduledTask* taskRef = new(ELeave) TScheduledTask(*task,*client);
|
sl@0
|
889 |
persSchedule->AddTask(*taskRef);
|
sl@0
|
890 |
}
|
sl@0
|
891 |
}
|
sl@0
|
892 |
}
|
sl@0
|
893 |
}
|
sl@0
|
894 |
|
sl@0
|
895 |
// Activate the scheduler with the new schedules
|
sl@0
|
896 |
AddSchedulesToTimerL();
|
sl@0
|
897 |
}
|
sl@0
|
898 |
else
|
sl@0
|
899 |
{
|
sl@0
|
900 |
if (iTaskExecutedDuringBUR)
|
sl@0
|
901 |
{
|
sl@0
|
902 |
iBUROperationInProgress = EBUROperationNoActivity;
|
sl@0
|
903 |
iTaskExecutedDuringBUR = EFalse;
|
sl@0
|
904 |
// performed the delayed task execution
|
sl@0
|
905 |
ExecuteClients();
|
sl@0
|
906 |
}
|
sl@0
|
907 |
}
|
sl@0
|
908 |
|
sl@0
|
909 |
// BUR operation is completed
|
sl@0
|
910 |
iBUROperationInProgress = EBUROperationNoActivity;
|
sl@0
|
911 |
iTaskExecutedDuringBUR = EFalse;
|
sl@0
|
912 |
}
|