Update contrib.
1 // Copyright (c) 1997-2009 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.
19 #include <bafl/backup_std.h>
21 #include <bsul/clientmessage.h>
22 #include "patchdata.h"
24 extern const BSUL::TClientMessageServerData KServerData;
26 const TInt KBakServMaxOperationTimerLoops = 3; // Number of iterations for base operation timer
28 _LIT(KPanic,"BackupServer");
30 // RMessage::Panic() also completes the message. This is:
31 // (a) important for efficient cleanup within the kernel
32 // (b) a problem if the message is completed a second time
34 void PanicClient(const RMessagePtr2& aMessage, TInt aPanic)
36 aMessage.Panic(KPanic,aPanic);
41 // class CShutdownServer
44 NONSHARABLE_CLASS(CShutdownServer) : public CTimer
46 enum {KMyShutdownDelay=0x2000000}; // approx 2s
48 inline CShutdownServer();
49 inline void ConstructL();
55 inline CShutdownServer::CShutdownServer()
58 CActiveScheduler::Add(this);
61 inline void CShutdownServer::ConstructL()
65 inline void CShutdownServer::Start()
67 After(KMyShutdownDelay);
71 // Initiate server exit when the timer expires
73 void CShutdownServer::RunL()
75 CActiveScheduler::Stop();
80 // class CBaServBackupScheduler
83 EXPORT_C CBaServBackupScheduler::CBaServBackupScheduler()
87 EXPORT_C CBaServBackupScheduler::~CBaServBackupScheduler()
93 * Set the error handler to aErrorHandler to the current scheduler.
95 * @param "CBaServBackupSession* aErrorHandler"
96 * The handler session.
98 EXPORT_C void CBaServBackupScheduler::SetErrorHandler(CBaServBackupSession* aErrorHandler)
100 iErrorHandler=aErrorHandler;
105 * Handles the error aError.
107 * @param "TInt aError"
110 EXPORT_C void CBaServBackupScheduler::Error(TInt aError) const
114 iErrorHandler->HandleError(aError);
116 else if (aError!=KLeaveWithoutAlert)
118 // Handling the in this way implies that ServiceL should not leave.
119 CActiveScheduler::Error(aError);
124 Class CBaServBackupSession::CReRegistrationTimer
126 NONSHARABLE_CLASS(CBaServBackupSession::CReRegistrationTimer) : public CPeriodic
129 static CReRegistrationTimer* NewL(TInt aPriority);
130 static TInt ReRegistrationTimerCallBack(TAny* aPtr);
133 TInt RunError(TInt aError);
136 CReRegistrationTimer(TInt aPriority);
137 void HandleReRegistrationTimerCallBack();
140 CBaBackupServer* iBackupServer;
144 Class CBaBackupServer::CBaServCloseAllOperationTimer
147 NONSHARABLE_CLASS(CBaBackupServer::CBaServCloseAllOperationTimer): public CPeriodic
150 static CBaServCloseAllOperationTimer* NewL(CBaBackupServer* aBackupServer);
151 void ConstructL(CBaBackupServer* aBackupServer);
152 ~CBaServCloseAllOperationTimer();
153 void SetMessage(const RMessagePtr2& aPtr);
154 RMessagePtr2 Message();
155 TInt GetOperationCount();
156 void SetOperationCount(TInt aCount);
157 static TInt OperationTimerCallBack(TAny* aPtr);
159 TInt RunError(TInt aError);
161 void HandleOperationTimerCallBack();
162 CBaServCloseAllOperationTimer();
164 RMessagePtr2 iCloseAllFilesMessage;
165 TInt16 iOperationCount;
166 CBaBackupServer* iBackupServer;
170 // class CBaBackupServer
175 * Returns a pointer to <code>CBaBackupServer</code> object.
177 * @return "CBaBackupServer"
178 * A newly-constructed backup server object.
180 EXPORT_C CBaBackupServer* CBaBackupServer::NewL()
182 CBaBackupServer* self=new(ELeave) CBaBackupServer(CActive::EPriorityStandard);
190 * @param "TInt aPriority"
191 * The active object priority.
193 EXPORT_C CBaBackupServer::CBaBackupServer(TInt aPriority)
194 : CServer2(aPriority)
201 EXPORT_C CBaBackupServer::~CBaBackupServer()
203 TDblQueIter<CSession2> iter(iSessionIter);
206 for (CSession2* session=iter++; session!=NULL; session=iter++)
212 delete iCloseAllFilesOperationTimer;
217 * Completes the server construction by adding the server to the active scheduler and creating
218 * a <code>CShutdownServer</code> object.
221 EXPORT_C void CBaBackupServer::ConstructL()
223 StartL(__BACKUP_SERVER_NAME_V2);
224 iExtension = new(ELeave) CBaBackupServerExt();
225 iShutdown = new (ELeave) CShutdownServer;
226 iShutdown->ConstructL();
227 // ensure that the server still exits even if the 1st client fails to connect
229 iRegisteredFilesCount = 0;
230 iCloseAllFilesOperationTimer = CBaBackupServer::CBaServCloseAllOperationTimer::NewL(this);
231 iCloseAllOperationRunning = EFalse;
233 //initialise the client message framework
234 BSUL::CClientMessage::InitialiseFrameworkL(KServerData);
239 * Sets the server to be busy with the aUniqueClientId client.
241 * @param "TUint32 aUniqueClientId"
242 * A unique client identifier.
244 EXPORT_C void CBaBackupServer::SetBusy(TUint32 aUniqueClientId)
246 iExtension->iUniqueBusyClientId=aUniqueClientId;
251 * Returns <code>ETrue</code> if the client using the server is not the one identified by aUniqueClientId
252 * oterwise <code>EFalse</code>
254 * @param "TUint32 aUniqueClientId"
255 * A unique client identifier.
257 EXPORT_C TBool CBaBackupServer::IsOtherClientBusy(TUint32 aUniqueClientId) const
259 return (iExtension->iUniqueBusyClientId!=0 && iExtension->iUniqueBusyClientId!=aUniqueClientId);
264 * Returns <code>ETrue</code> if the client using the server corresponds has aUniqueClientId as unique id
265 * oterwise <code>EFalse</code>
267 * @param "TUint32 aUniqueClientId"
268 * A unique client identifier id.
270 EXPORT_C TBool CBaBackupServer::IsClientBusy(TUint32 aUniqueClientId) const
272 return ((iExtension->iUniqueBusyClientId==aUniqueClientId) ||
273 ((iExtension->iCachedBusyClientId != 0) && (iExtension->iCachedBusyClientId == aUniqueClientId)));
278 * Handles the closing of all files by signaling the new file lock flag, aFlag, to all client.
280 * @param "MBackupObserver::TFileLockFlags aFlag"
283 EXPORT_C void CBaBackupServer::CloseAllFilesL(MBackupObserver::TFileLockFlags aFlag)
285 // Set the close all operation flag and the reregistration counter to zero
286 iCloseAllOperationRunning = ETrue;
287 iSessionLockReRegistrationCount = 0;
289 // cleanup that calls RestartAll if required
290 TDblQueIter<CSession2> iter(iSessionIter);
293 for (CSession2* session=iter++; session!=NULL; session=iter++)
295 static_cast<CBaServBackupSession*>(session)->SignalReleaseAllFileLocksL(aFlag);
301 * Signals to all clients of getting the lock of their respective files.
304 EXPORT_C void CBaBackupServer::RestartAll()
306 TDblQueIter<CSession2> iter(iSessionIter);
309 for (CSession2* session=iter++; session!=NULL; session=iter++)
311 static_cast<CBaServBackupSession*>(session)->SignalRetakeAllFileLocks();
317 * Allows any subclass of <code>CBaBackupServer</code> of completing the closing of files.
318 * The server will lose the ownership of aClosedFiles object, which implies it is up to this virtual
319 * function to deal with it in order to avoid a memory leak.
321 * @param "CArrayFix<CBaServBackupSession::TClosedFile>* aClosedFiles"
322 * An array of closed files.
324 EXPORT_C void CBaBackupServer::CompleteClosingFiles(CArrayFix<CBaServBackupSession::TClosedFile>* aClosedFiles)
331 * Allows application framework backup code to check if all registerd files have beem updated
334 EXPORT_C TBool CBaBackupServer::HaveAllCloseAllFilesClientsReRegistered()
336 return((iRegisteredFilesCount == iSessionLockReRegistrationCount) ? ETrue : EFalse);
339 void CBaBackupServer::CloseFileL(MBackupObserver::TFileLockFlags aFlag,const TDesC& aFileName)
341 TDblQueIter<CSession2> iter(iSessionIter);
344 for (CSession2* session=iter++; session!=NULL; session=iter++)
346 static_cast<CBaServBackupSession*>(session)->SignalReleaseFileLockL(aFlag,aFileName);
350 void CBaBackupServer::RestartFile(const TDesC& aFileName)
352 TDblQueIter<CSession2> iter(iSessionIter);
355 for (CSession2* session=iter++; session!=NULL; session=iter++)
357 static_cast<CBaServBackupSession*>(session)->SignalRetakeFileLocks(aFileName);
363 * Creates a server-side client <code>CBaServBackupSession</code> session object by checking first if
364 * the version of the server is compatible with the client.
366 * @param "const TVersion &aVersion"
367 * Version information supplied by the client.
369 EXPORT_C CSession2* CBaBackupServer::NewSessionL(const TVersion &aVersion, const RMessage2& /*aMessage*/) const
371 TVersion ver(KBakServMajorVN,KBakServMinorVN,KBakServBuildVN);
372 if (!User::QueryVersionSupported(ver,aVersion))
373 User::Leave(KErrNotSupported);
375 return CBaServBackupSession::NewL();
380 * Signals to all clients the aBackupOperationAttributes.
382 * @param "const TBackupOperationAttributes& aBackupOperationAttributes"
383 * Backup operation attributes.
385 EXPORT_C void CBaBackupServer::SignalBackupOperation(const TBackupOperationAttributes& aBackupOperationAttributes)
387 TDblQueIter<CSession2> iter(iSessionIter);
390 for (CSession2* session=iter++; session!=NULL; session=iter++)
392 static_cast<CBaServBackupSession*>(session)->SignalBackupOperation(aBackupOperationAttributes);
394 // Cache session id if starting backup
395 if (aBackupOperationAttributes.iOperation == MBackupOperationObserver::EStart)
396 iExtension->iCachedBusyClientId = iExtension->iUniqueBusyClientId;
399 void CBaBackupServer::AddSession()
405 void CBaBackupServer::DropSession()
407 if (--iSessionCount==0)
414 * Check to see if a CloseAll opertation is in progress
417 TBool CBaBackupServer::IsCloseAllOperationRunning()
419 return iCloseAllOperationRunning;
423 * Set value of IsCloseAllOperationRunning
426 EXPORT_C void CBaBackupServer::SetCloseAllOperationRunningState(TBool aRunning)
428 iCloseAllOperationRunning = aRunning;
433 * This timer will only be created in cases of no derived backup server
436 void CBaBackupServer::StartCloseAllFilesOperationTimer(const RMessagePtr2& aPtr)
438 // Store the CloseAll message in case we need to Complete on a timeout
439 iCloseAllFilesOperationTimer->SetMessage(aPtr);
441 // If hardware use patchable constant if not default to 3 seconds
442 iCloseAllFilesOperationTimer->Start(KBaBackupCloseallFilesTimeout, KBaBackupCloseallFilesTimeout, TCallBack(CBaBackupServer::CBaServCloseAllOperationTimer::OperationTimerCallBack, iCloseAllFilesOperationTimer));
446 * Handle an error from CBaServBackupSession::ServiceL()
447 * A bad descriptor error implies a badly programmed client, so panic it;
448 * otherwise report the error to the client
450 * @param "TInt aError"
453 EXPORT_C TInt CBaBackupServer::RunError(TInt aError)
455 if (aError==KErrBadDescriptor)
457 PanicClient(Message(), EBufferOverflow);
461 Message().Complete(aError);
464 // The leave will result in an early return from CServer::RunL(), skipping
465 // the call to request another message. So do that now in order to keep the
468 return KErrNone; // handled the error fully
472 // class CBaBackupServer::CBaServCloseAllOperationTimer
478 CBaBackupServer::CBaServCloseAllOperationTimer* CBaBackupServer::CBaServCloseAllOperationTimer::NewL(CBaBackupServer* aBackupServer)
480 CBaBackupServer::CBaServCloseAllOperationTimer* self=new(ELeave) CBaBackupServer::CBaServCloseAllOperationTimer();
481 CleanupStack::PushL(self);
482 self->ConstructL(aBackupServer);
483 CleanupStack::Pop(); // self
484 CActiveScheduler::Add(self);
492 void CBaBackupServer::CBaServCloseAllOperationTimer::ConstructL(CBaBackupServer* aBackupServer)
494 CTimer::ConstructL();
495 iBackupServer = aBackupServer;
502 TInt CBaBackupServer::CBaServCloseAllOperationTimer::GetOperationCount()
504 return iOperationCount;
511 void CBaBackupServer::CBaServCloseAllOperationTimer::SetOperationCount(TInt aCount)
513 iOperationCount = aCount;
520 TInt CBaBackupServer::CBaServCloseAllOperationTimer::RunError(TInt aError)
522 if (aError==KLeaveWithoutAlert)
532 CBaBackupServer::CBaServCloseAllOperationTimer::CBaServCloseAllOperationTimer()
533 : CPeriodic(CActive::EPriorityStandard)
539 CBaBackupServer::CBaServCloseAllOperationTimer::~CBaServCloseAllOperationTimer()
548 void CBaBackupServer::CBaServCloseAllOperationTimer::SetMessage(const RMessagePtr2& aPtr)
550 iCloseAllFilesMessage = aPtr;
556 RMessagePtr2 CBaBackupServer::CBaServCloseAllOperationTimer::Message()
558 return iCloseAllFilesMessage;
564 TInt CBaBackupServer::CBaServCloseAllOperationTimer::OperationTimerCallBack(TAny* aPtr)
566 TRAP_IGNORE(REINTERPRET_CAST(CBaBackupServer::CBaServCloseAllOperationTimer*,aPtr)->HandleOperationTimerCallBack());
573 void CBaBackupServer::CBaServCloseAllOperationTimer::HandleOperationTimerCallBack()
575 TBool finished = iBackupServer->HaveAllCloseAllFilesClientsReRegistered();
577 if (finished || (iOperationCount == KBakServMaxOperationTimerLoops))
585 retCode = KErrLocked;
588 // One way or another CloseAll is finished at this point
589 iBackupServer->SetCloseAllOperationRunningState(EFalse);
591 iCloseAllFilesMessage.Complete(retCode);
601 // If hardware use patchable constant if not default to 3 seconds
602 Start(KBaBackupCloseallFilesTimeout, KBaBackupCloseallFilesTimeout, TCallBack(CBaBackupServer::CBaServCloseAllOperationTimer::OperationTimerCallBack,this));
608 // class CBaServBackupMessageQueue
611 inline CBaServBackupMessageQueue::TQueueItem::TQueueItem(HBufC* aFileName,TInt aOperation)
612 : iFileName(aFileName), iOperation(aOperation)
615 CBaServBackupMessageQueue* CBaServBackupMessageQueue::NewL()
617 CBaServBackupMessageQueue* self=new(ELeave) CBaServBackupMessageQueue();
621 CBaServBackupMessageQueue::~CBaServBackupMessageQueue()
623 const TInt count=iQueue.Count();
624 for (TInt ii=0;ii<count;ii++)
626 delete iQueue[ii].iFileName;
632 void CBaServBackupMessageQueue::AddItemL(const TDesC& aFileName,MBackupObserver::TFileLockFlags aFlag)
634 const TInt count=iQueue.Count();
635 for (TInt ii=0;ii<count;ii++)
637 TQueueItem& item=iQueue[ii];
638 if(aFileName.MatchF(*item.iFileName) == KErrNone)
640 if (item.iOperation==MBackupObserver::EReleaseLockReadOnly && aFlag==MBackupObserver::EReleaseLockNoAccess)
642 item.iOperation=MBackupObserver::EReleaseLockNoAccess;
644 else if (item.iOperation!=MBackupObserver::EReleaseLockNoAccess)
646 HBufC* file=aFileName.AllocLC();
647 TQueueItem item(file,aFlag);
648 User::LeaveIfError(iQueue.Insert(item,ii));
649 CleanupStack::Pop(); // file
654 HBufC* file=aFileName.AllocLC();
655 HBufC* copy=aFileName.AllocLC();
656 TQueueItem item(file,aFlag);
657 User::LeaveIfError(iQueue.Append(item));
658 item=TQueueItem(copy,-1);
659 const TInt err=iQueue.Append(item);
662 iQueue.Remove(iQueue.Count()-1);
665 CleanupStack::Pop(2); // copy, file
668 void CBaServBackupMessageQueue::AddItem(const TDesC& aFileName)
670 const TInt count=iQueue.Count();
671 for (TInt ii=0;ii<count;ii++)
673 TQueueItem& item=iQueue[ii];
674 if (aFileName.MatchF(*item.iFileName) == KErrNone && item.iOperation==-1)
682 TBool CBaServBackupMessageQueue::IsEmpty() const
684 return HeadIndex()==KErrNotFound;
687 void CBaServBackupMessageQueue::RemoveHead()
689 const TInt index=HeadIndex();
690 if (index!=KErrNotFound)
692 delete iQueue[index].iFileName;
693 iQueue.Remove(index);
698 void CBaServBackupMessageQueue::RemoveItem(const TDesC& aFileName)
700 const TInt count=iQueue.Count();
701 for (TInt ii=count-1;ii>=0;ii--)
703 const TQueueItem& item=iQueue[ii];
704 if (aFileName.MatchF(*item.iFileName) == KErrNone)
706 delete item.iFileName;
713 void CBaServBackupMessageQueue::GetHead(TDes& aFileName,MBackupObserver::TFileLockFlags& aFlag) const
716 const TInt index=HeadIndex();
717 if (index!=KErrNotFound)
719 const TQueueItem& item=iQueue[index];
720 aFileName=item.iFileName->Des();
721 aFlag=(MBackupObserver::TFileLockFlags)item.iOperation;
725 TInt CBaServBackupMessageQueue::HeadIndex() const
727 TInt index=KErrNotFound;
728 const TInt count=iQueue.Count();
729 for (TInt ii=0;ii<count;ii++)
731 const TQueueItem& item=iQueue[ii];
732 if (item.iOperation!=ENoOp)
742 // class CBaServBackupSession
745 CBaServBackupSession* CBaServBackupSession::NewL()
747 CBaServBackupSession* self=new(ELeave) CBaServBackupSession();
748 CleanupStack::PushL(self);
750 CleanupStack::Pop(); // self
758 * @param "RThread aClient"
759 * The client thread for which this server-side client session is being constructed.
761 EXPORT_C CBaServBackupSession::CBaServBackupSession()
764 iUniqueClientId = (TUint32)this;
772 EXPORT_C CBaServBackupSession::~CBaServBackupSession()
774 CBaBackupServer* server=BackupServer();
775 if (server->IsClientBusy(iUniqueClientId))
778 if (server->IsBackupOperationRunning())
780 TBackupOperationAttributes backupOpAtt(MBackupObserver::ETakeLock, MBackupOperationObserver::EAbort);
781 server->SignalBackupOperation(backupOpAtt);
782 server->SetBackupOperationRunning(EFalse);
787 if (iFileLockObservers)
789 server->DecrementRegisteredFilesCount(iFileLockObservers->Count());
793 delete iFileLockObservers;
794 delete iReleasedFiles;
795 delete iMessageQueue;
796 delete iReRegistrationTimer;
797 iReRegistrationTimer = 0;
798 server->DropSession();
803 * Completes the session construction by creating a <code>CBaServBackupMessageQueue</code> object.
806 EXPORT_C void CBaServBackupSession::ConstructL()
808 iMessageQueue=CBaServBackupMessageQueue::NewL();
810 // Cannot set iBackupServer in CReRegistrationTimer at this point but must be done before call to Start
811 iReRegistrationTimer=CBaServBackupSession::CReRegistrationTimer::NewL(CActive::EPriorityStandard);
816 * Completes the construction of this <code>CBaServBackupSession</code> session by just doing a base call class.
817 * It also notify the server that a new session has been created.
819 * @param "const CServer& aServer"
820 * The server active object which is responsible for this session
822 EXPORT_C void CBaServBackupSession::CreateL()
824 BackupServer()->AddSession();
827 void CBaServBackupSession::SignalReleaseAllFileLocksL(MBackupObserver::TFileLockFlags aFlag)
829 CBaBackupServer* server=BackupServer();
830 if (iFileLockObservers)
832 const TInt count=iFileLockObservers->Count();
833 for (TInt ii=0;ii<count;ii++)
835 TFileName name=(*iFileLockObservers)[ii];
836 server->CloseFileL(aFlag,name);
841 void CBaServBackupSession::SignalRetakeAllFileLocks()
843 // CBaBackupServer* server=BackupServer();
846 const TInt count=iReleasedFiles->Count();
847 for (TInt ii=count-1;ii>=0;ii--)
849 TFileName name=(*iReleasedFiles)[ii];
850 // server->RestartFile(name);
851 SignalRetakeFileLocks(name);
856 LOCAL_C void CleanupCloseFail(TAny* aPtr)
858 CDesCArray* array=REINTERPRET_CAST(CDesCArray*,aPtr);
859 array->Delete(array->Count()-1);
862 void CBaServBackupSession::SignalReleaseFileLockL(MBackupObserver::TFileLockFlags aFlag,const TDesC& aFileName)
865 if (iFileLockObservers && iFileLockObservers->Find(aFileName,pos)==KErrNone)
867 if (iReleasedFiles==NULL)
869 iReleasedFiles=new(ELeave) CDesCArraySeg(1);
871 const TBool addedReleasedFile=iReleasedFiles->Find(aFileName,pos)!=KErrNone;
872 if (addedReleasedFile)
874 iReleasedFiles->AppendL(aFileName);
875 CleanupStack::PushL(TCleanupItem(CleanupCloseFail,iReleasedFiles));
877 iMessageQueue->AddItemL(aFileName,aFlag);
878 if (!iNotificationPullMsg.IsNull())
880 iNotificationPullMsg.Complete(KErrNone);
882 // Okay - if this is part of a CloseAll operation we need to kick off a ReRegistration timer
883 if (BackupServer()->IsCloseAllOperationRunning())
885 // Set backup server for the timer
886 iReRegistrationTimer->iBackupServer = BackupServer();
888 // If hardware use patchable constant if not default to 3 seconds
889 iReRegistrationTimer->Start(KBaBackupFileLockReRegistrationTimeout, KBaBackupFileLockReRegistrationTimeout, TCallBack(CBaServBackupSession::CReRegistrationTimer::ReRegistrationTimerCallBack, iReRegistrationTimer));
893 if (addedReleasedFile)
895 CleanupStack::Pop(); // CleanupCloseFail
900 void CBaServBackupSession::SignalRetakeFileLocks(const TDesC& aFileName)
904 const TInt count=iReleasedFiles->Count();
905 for (TInt ii=count-1;ii>=0;ii--)
907 TFileName name=(*iReleasedFiles)[ii];
908 if (name.MatchF(aFileName)==0)
910 iMessageQueue->AddItem(aFileName);
911 if (!iNotificationPullMsg.IsNull())
913 iNotificationPullMsg.Complete(KErrNone);
915 iReleasedFiles->Delete(ii);
918 if (iReleasedFiles->Count()==0)
920 delete iReleasedFiles;
922 if (iClosedFiles==NULL && BackupServer()->IsClientBusy(iUniqueClientId))
924 BackupServer()->SetBusy(0);
932 * Allows the session to handle the error.
934 * @param "TInt aError"
937 EXPORT_C void CBaServBackupSession::HandleError(TInt aError)
939 if (aError==KLeaveWithoutAlert)
941 CBaServBackupScheduler::Current()->SetErrorHandler(NULL);
945 void CBaServBackupSession::DoServiceL(TCompletionType& aCompleteType)
947 switch (iClientMessage->Function())
949 case EBakOpCodeEventReady:
952 aCompleteType = ECompleteAsync;
955 case EBakOpCodeStopNotifications:
960 case EBakOpCodeGetEvent:
963 case EBakOpCodeCloseAllFiles:
965 aCompleteType = CloseAllFilesL(iClientMessage->Message());
968 case EBakOpCodeRestartAll:
971 case EBakOpCodeCloseFile:
974 case EBakOpCodeRestartFile:
977 case EBakOpCodeNotifyLockChange:
980 case EBakOpCodeNotifyLockChangeCancel:
981 NotifyLockChangeCancelL();
983 case EBakOpCodeNotifyBackupOperation:
984 NotifyBackupOperationL();
986 case EBakOpCodeCancelOutstandingBackupOperationEvent:
988 iBackupOperationObserverPresent = EFalse;
989 if (!iBackupOperationMessagePtr.IsNull())
991 GetBackupOperationEventL(iBackupOperationMessagePtr);
992 iBackupOperationMessagePtr.Complete(KErrCancel);
996 case EBakOpCodeGetBackupOperationState:
997 GetBackupOperationStateL();
999 case EBakOpCodeBackupOperationEventReady:
1000 BackupOperationEventReadyL();
1001 aCompleteType = ECompleteAsync;
1003 case EBakOpCodeGetBackupOperationEvent:
1004 GetBackupOperationEventL(iClientMessage->Message());
1006 case EBakOpCodeSetBackupOperationObserverIsPresent:
1008 iBackupOperationObserverPresent = iClientMessage->GetIntL(0);
1012 User::Leave(KErrNotSupported);
1020 * Handles the servicing of client requests passed to the backup server.
1022 * @param "const RMessage& aMessage"
1023 * The message containing the client request.
1025 EXPORT_C void CBaServBackupSession::ServiceL(const RMessage2& aMessage)
1028 TCompletionType completionType = ECompleteSync;
1030 BSUL::CClientMessage* clientMessage = 0;
1031 clientMessage = BSUL::CClientMessage::NewL(aMessage);
1033 //Push iClientMessage onto the cleanupstack. Although an instance variable,
1034 //the lifetime of the object is contained to this function so it needs to
1035 //be pushed and popped here as it is not deleted in the destructor
1036 CleanupStack::PushL(clientMessage);
1038 //Validate the message
1039 TRAPD(error, clientMessage->ValidateL());
1041 iClientMessage = clientMessage;
1043 if(error == KErrNone)
1045 TRAP(error, DoServiceL(completionType));
1048 if(completionType == ECompleteSync)
1050 if (error==KLeaveWithoutAlert)
1052 CBaServBackupScheduler::Current()->SetErrorHandler(NULL);
1056 iClientMessage->CompleteRequestL(error);
1060 //Pop and destroy message
1061 CleanupStack::PopAndDestroy(clientMessage);
1062 clientMessage = NULL;
1063 iClientMessage = NULL;
1066 void CBaServBackupSession::HandleEventReadyL()
1068 if (iReRegistrationTimer->IsActive())
1069 { // If timer is still active (ie not timed out) we need to increment the counter and cancel the timer
1070 BackupServer()->IncrementFilesReRegistrationCount();
1071 iReRegistrationTimer->Cancel();
1073 // else timer is already inactive because it's expired
1075 if (iMessageQueue->IsEmpty())
1077 iNotificationPullMsg = iClientMessage->Message();
1081 iClientMessage->CompleteRequestL(KErrNone);
1085 const TInt KEventFileNameOffset=1;
1087 void CBaServBackupSession::GetEventL()
1090 MBackupObserver::TFileLockFlags fileFlag;
1091 if(!iMessageQueue->IsEmpty())
1093 iMessageQueue->GetHead(fileName,fileFlag);
1094 TBuf<KEventFileNameOffset> num;
1095 num.Num((TInt)fileFlag);
1097 iClientMessage->WriteL(0,num,0);
1099 iClientMessage->WriteL(0,fileName,KEventFileNameOffset);
1101 iMessageQueue->RemoveHead();
1105 iClientMessage->PanicClient(KPanic,ENoEventToFetch);
1106 User::Leave(KLeaveWithoutAlert);
1110 void CBaServBackupSession::CleanupCloseAllFiles(TAny* aPtr)
1112 CBaServBackupSession* self=REINTERPRET_CAST(CBaServBackupSession*,aPtr);
1113 delete self->iClosedFiles;
1114 self->iClosedFiles=NULL;
1115 self->BackupServer()->RestartAll();
1119 Asks the server to close all files. This function may leave in case the server is busy.
1120 If the requesting client is the current busy client and it has already requested CloseAll,
1121 this request will be ignored.
1123 @param aMessage The reference to the message containing the client request: file lock.
1124 @leave KErrServerBusy if the requesting client is not the busy client. Plus other system-wide
1127 EXPORT_C TCompletionType CBaServBackupSession::CloseAllFilesL(const RMessage2& aMessage)
1129 // Raise file lock notifications
1130 TRAPD(err,DoCloseAllFilesL(aMessage));
1132 if (err == KErrNone)
1134 // Start timer to check all file locks re-registered before completing message
1135 BackupServer()->StartCloseAllFilesOperationTimer(aMessage);
1139 // If the error is that the same client has already requested CloseAll, ignore this request
1140 // If not this error, recover the CloseAllOperationRunningState flag and leave with the error.
1141 if (err != KErrAlreadyExists)
1143 BackupServer()->SetCloseAllOperationRunningState(EFalse);
1148 return ECompleteAsync;
1151 void CBaBackupServer::IncrementRegisteredFilesCount()
1153 iRegisteredFilesCount++;
1156 void CBaBackupServer::DecrementRegisteredFilesCount(TInt aNumberFiles)
1158 iRegisteredFilesCount = iRegisteredFilesCount - aNumberFiles;
1161 void CBaBackupServer::IncrementFilesReRegistrationCount()
1163 iSessionLockReRegistrationCount++;
1167 Function called by both base and derived backup sessions in order to start asynchronous
1168 file lock notifications.
1169 The requesting client is set to the current busy client.
1171 @leave KErrServerBusy if the requesting client is not the current busy client or the server
1172 is under CloseAll operation. KErrAlreadyExists if the same client has sent CloseAll request.
1173 Plus other system-wide errors.
1176 EXPORT_C TCompletionType CBaServBackupSession::DoCloseAllFilesL(const RMessage2& /*aMessage*/)
1178 CBaBackupServer* server=BackupServer();
1179 if (server->IsOtherClientBusy(iUniqueClientId))
1181 User::Leave(KErrServerBusy);
1184 if (BackupServer()->IsCloseAllOperationRunning())
1186 User::Leave(KErrAlreadyExists);
1189 CleanupStack::PushL(TCleanupItem(CleanupCloseAllFiles,this));
1192 delete iClosedFiles;
1195 server->SetBusy(iUniqueClientId);
1197 MBackupObserver::TFileLockFlags fileFlag=
1198 (MBackupObserver::TFileLockFlags)iClientMessage->GetIntL(0);
1199 server->CloseAllFilesL(fileFlag);
1201 iClosedFiles=new(ELeave) CArrayFixSeg<CBaServBackupSession::TClosedFile>(1);
1202 CBaServBackupScheduler::Current()->SetErrorHandler(this);
1203 CleanupStack::Pop(); // CleanupCloseAllFiles
1204 return ECompleteAsync;
1209 * Asks the server to signal all clients of getting the lock of their respective files.
1212 EXPORT_C void CBaServBackupSession::RestartAll()
1214 CBaBackupServer* server=BackupServer();
1215 if (server->IsClientBusy(iUniqueClientId))
1221 void CBaServBackupSession::DoRestartAll()
1223 CBaBackupServer* server=BackupServer();
1224 server->RestartAll();
1225 delete iReleasedFiles;
1226 iReleasedFiles=NULL;
1229 CArrayFix<CBaServBackupSession::TClosedFile>* closedFiles=iClosedFiles;
1231 server->CompleteClosingFiles(closedFiles); // takes ownership of closedFiles immediately
1236 Handle the client's request to close a file.
1237 The request will be ignored if the requesting client the current busy client and the server is
1238 under CloseAll operation.
1240 @leave KErrServerBusy if the server is busy with the other client, plus other system-wide errors.
1242 void CBaServBackupSession::CloseFileL()
1244 CBaBackupServer* server=BackupServer();
1245 if (server->IsOtherClientBusy(iUniqueClientId))
1247 User::Leave(KErrServerBusy);
1250 if (! server->IsCloseAllOperationRunning())
1252 MBackupObserver::TFileLockFlags flag=
1253 static_cast<MBackupObserver::TFileLockFlags>(iClientMessage->GetIntL(2));
1255 GetFileNameL(fileName);
1256 server->SetBusy(iUniqueClientId);
1257 server->CloseFileL(flag,fileName);
1261 void CBaServBackupSession::RestartFileL()
1264 GetFileNameL(fileName);
1265 BackupServer()->RestartFile(fileName);
1269 Handles the client's request of notification of a file lock change.
1271 @leave KErrServerBusy if the server is under CloseAll operation, KLeaveWithoutAlert if the requested
1272 file has been registered. Plus other system-wide errors.
1274 void CBaServBackupSession::NotifyLockChangeL()
1276 if(BackupServer()->IsCloseAllOperationRunning())
1278 User::Leave(KErrServerBusy);
1282 GetFileNameL(fileName);
1283 if (iFileLockObservers==NULL)
1285 iFileLockObservers=new(ELeave) CDesCArraySeg(1);
1290 if(iFileLockObservers->Find(fileName,pos)== KErrNone)
1292 iClientMessage->PanicClient(KPanic,EReqAlreadyOutstanding);
1293 User::Leave(KLeaveWithoutAlert);
1296 iFileLockObservers->AppendL(fileName);
1297 BackupServer()->IncrementRegisteredFilesCount();
1300 LOCAL_C void RemoveFileName(CDesCArray& aArray,const TDesC& aFileName)
1303 if (aArray.Find(aFileName,pos)==KErrNone)
1309 void CBaServBackupSession::NotifyLockChangeCancelL()
1312 GetFileNameL(fileName);
1313 if (iFileLockObservers)
1315 RemoveFileName(*iFileLockObservers,fileName);
1316 BackupServer()->DecrementRegisteredFilesCount();
1320 RemoveFileName(*iReleasedFiles,fileName);
1322 iMessageQueue->RemoveItem(fileName);
1325 void CBaServBackupSession::GetFileNameL(TDes& aFileName)
1327 //The verification of this parameter has already been handled
1328 //by the CClientMessage class so we can safely read the value
1329 iClientMessage->ReadL(1,aFileName);
1333 void CBaServBackupSession::GetBackupOperationEventL(const RMessagePtr2& aPtr)
1335 TPckg<TBackupOperationAttributes> backupOpAttPkg(iBackupOperationAttributes);
1337 aPtr.WriteL(0, backupOpAttPkg);
1340 void CBaServBackupSession::BackupOperationEventReadyL()
1342 if (iBackupOperationObserverPresent)
1344 iBackupOperationMessagePtr = iClientMessage->Message();
1348 iClientMessage->CompleteRequestL(KErrNone);
1352 void CBaServBackupSession::NotifyBackupOperationL()
1354 CBaBackupServer* server=BackupServer();
1355 if (server->CBaBackupServer::IsOtherClientBusy(iUniqueClientId))
1357 User::Leave(KErrServerBusy);
1359 TPckg<TBackupOperationAttributes> backupOpAttPkg(iBackupOperationAttributes);
1361 iClientMessage->ReadL(0,backupOpAttPkg);
1363 const TBool backupOperationRunning = ((iBackupOperationAttributes.iOperation==MBackupOperationObserver::EStart) ? ETrue : EFalse);
1364 server->SetBackupOperationRunning(backupOperationRunning);
1365 server->SetBusy(backupOperationRunning ? iUniqueClientId : 0);
1366 server->SignalBackupOperation(iBackupOperationAttributes);
1367 if (!iBackupOperationMessagePtr.IsNull())
1369 GetBackupOperationEventL(iBackupOperationMessagePtr);
1370 iBackupOperationMessagePtr.Complete(KErrCancel);
1374 void CBaServBackupSession::GetBackupOperationStateL()
1376 const TBool isRunning = BackupServer()->IsBackupOperationRunning();
1377 TPckgC<TBool> pkgObs(isRunning);
1379 iClientMessage->WriteL(0, pkgObs);
1382 void CBaServBackupSession::SignalBackupOperation(const TBackupOperationAttributes& aBackupOperationAttributes)
1384 iBackupOperationAttributes = aBackupOperationAttributes;
1385 if (!iBackupOperationMessagePtr.IsNull())
1387 TRAPD(err,GetBackupOperationEventL(iBackupOperationMessagePtr));
1388 iBackupOperationMessagePtr.Complete(err);
1392 void CBaServBackupSession::StopNotifications()
1394 if(!iNotificationPullMsg.IsNull())
1395 {// complete the registration message
1396 iNotificationPullMsg.Complete(KErrNone);
1403 CBaServBackupSession::CReRegistrationTimer* CBaServBackupSession::CReRegistrationTimer::NewL(TInt aPriority)
1405 CBaServBackupSession::CReRegistrationTimer* self=new(ELeave) CBaServBackupSession::CReRegistrationTimer(aPriority);
1406 CleanupStack::PushL(self);
1408 CleanupStack::Pop(); // self
1409 CActiveScheduler::Add(self);
1416 TInt CBaServBackupSession::CReRegistrationTimer::RunError(TInt aError)
1418 if (aError==KLeaveWithoutAlert)
1428 CBaServBackupSession::CReRegistrationTimer::CReRegistrationTimer(TInt aPriority)
1429 : CPeriodic(aPriority)
1435 TInt CBaServBackupSession::CReRegistrationTimer::ReRegistrationTimerCallBack(TAny* aPtr)
1437 TRAP_IGNORE(REINTERPRET_CAST(CBaServBackupSession::CReRegistrationTimer*,aPtr)->HandleReRegistrationTimerCallBack());
1444 void CBaServBackupSession::CReRegistrationTimer::HandleReRegistrationTimerCallBack()
1446 iBackupServer->IncrementFilesReRegistrationCount();