sl@0: // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies). sl@0: // All rights reserved. sl@0: // This component and the accompanying materials are made available sl@0: // under the terms of "Eclipse Public License v1.0" sl@0: // which accompanies this distribution, and is available sl@0: // at the URL "http://www.eclipse.org/legal/epl-v10.html". sl@0: // sl@0: // Initial Contributors: sl@0: // Nokia Corporation - initial contribution. sl@0: // sl@0: // Contributors: sl@0: // sl@0: // Description: sl@0: // Implements installed file notification sl@0: // sl@0: // sl@0: sl@0: /** sl@0: @internalComponent sl@0: */ sl@0: sl@0: #include "install.h" sl@0: #include "srvrepos_noc.h" sl@0: #include "srvres.h" sl@0: #include "srvparams.h" sl@0: #include "cachemgr.h" sl@0: sl@0: CCentRepSWIWatcher* CCentRepSWIWatcher::NewL(RFs& aFs) sl@0: { sl@0: CCentRepSWIWatcher* self = new(ELeave) CCentRepSWIWatcher(aFs); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: CleanupStack::Pop(self); sl@0: return self; sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::ConstructL() sl@0: { sl@0: // Attach to SWI property sl@0: User::LeaveIfError(iSWIKey.Attach( KUidSystemCategory, KSAUidSoftwareInstallKeyValue)); sl@0: sl@0: // Initialise SWI operation and status sl@0: TInt swiProperty; sl@0: const TInt error = iSWIKey.Get(KUidSystemCategory, KSAUidSoftwareInstallKeyValue, swiProperty); sl@0: sl@0: if (error == KErrNone) sl@0: { sl@0: iSWIOperation=swiProperty & KSASwisOperationMask; sl@0: iSWIStatus= swiProperty & KSASwisOperationStatusMask; sl@0: } sl@0: else if (error != KErrNotFound) sl@0: { sl@0: User::LeaveIfError(error); sl@0: } sl@0: sl@0: // Get last saved contents of install directory sl@0: GetInstallDirL(); sl@0: sl@0: // Do any actions required by pre server start-up SWI activity sl@0: TRAP_IGNORE(FindChangedEntriesL(ETrue)); sl@0: } sl@0: sl@0: CCentRepSWIWatcher::CCentRepSWIWatcher(RFs& aFs) : sl@0: CActive(EPriorityStandard), sl@0: iInstallDir(*TServerResources::iInstallDirectory), sl@0: iFs(aFs), sl@0: iSWIOperation( ESASwisNone), sl@0: iSWIStatus( ESASwisStatusSuccess) sl@0: { sl@0: CActiveScheduler::Add(this); sl@0: } sl@0: sl@0: CCentRepSWIWatcher::~CCentRepSWIWatcher() sl@0: { sl@0: Cancel(); sl@0: if(iSWIKey.Handle() != KNullHandle) sl@0: { sl@0: iSWIKey.Cancel(); sl@0: iSWIKey.Close(); sl@0: } sl@0: iInstallEntryArray.ResetAndDestroy(); sl@0: iCurrentInstallDirEntries.ResetAndDestroy(); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::Start() sl@0: { sl@0: if(IsActive()) sl@0: return; sl@0: sl@0: NotifyChange(); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::NotifyChange() sl@0: { sl@0: sl@0: // Register for P&S of SWI flag sl@0: iSWIKey.Subscribe(iStatus); sl@0: sl@0: SetActive(); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::RunL() sl@0: { sl@0: NotifyChange(); sl@0: sl@0: // Get SWI Key sl@0: TInt swiProperty; sl@0: User::LeaveIfError(iSWIKey.Get(KUidSystemCategory, KSAUidSoftwareInstallKeyValue, swiProperty)); sl@0: sl@0: HandleSWIEventL(swiProperty); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::HandleSWIEventL(TInt aSWIProperty) sl@0: { sl@0: iSWIOperation=aSWIProperty & KSASwisOperationMask; sl@0: iSWIStatus= aSWIProperty & KSASwisOperationStatusMask; sl@0: sl@0: // Need to handle successful and aborted install/uninstall and successful restore sl@0: // Can't handle aborted restore sl@0: switch(iSWIOperation) sl@0: { sl@0: case ESASwisNone: sl@0: break; sl@0: case ESASwisInstall: sl@0: case ESASwisUninstall: sl@0: if(iSWIStatus==ESASwisStatusSuccess) sl@0: { sl@0: // Handle SWI events sl@0: FindChangedEntriesL(); sl@0: } sl@0: else if(iSWIStatus==ESASwisStatusAborted) sl@0: { sl@0: // Update directory to reset timestamps sl@0: ReadInstallDirL(iInstallEntryArray); sl@0: SaveInstallDirL(); sl@0: } sl@0: break; sl@0: case ESASwisRestore: sl@0: break; sl@0: default: sl@0: break; sl@0: } sl@0: } sl@0: sl@0: // Catch leaves so they don't stop the server sl@0: TInt CCentRepSWIWatcher::RunError( TInt aError) sl@0: { sl@0: static_cast(aError); sl@0: sl@0: RDebug::Print(_L("Run error %d"), aError); sl@0: sl@0: // Reinitialise directory list sl@0: iInstallEntryArray.ResetAndDestroy(); sl@0: sl@0: if(!IsActive()) sl@0: { sl@0: NotifyChange(); sl@0: } sl@0: sl@0: // Renable cache activity in case of errors during SWI events sl@0: TServerResources::iCacheManager->EnableCache(); sl@0: sl@0: return KErrNone; sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::DoCancel() sl@0: { sl@0: // Cancel subscription to SW P&S flag sl@0: iSWIKey.Cancel(); sl@0: iSWIKey.Close(); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::ReadInstallDirL(RPointerArray &aEntryArray) sl@0: { sl@0: RDir dir; sl@0: CleanupClosePushL(dir); sl@0: sl@0: //Empty contents of directory sl@0: aEntryArray.ResetAndDestroy(); sl@0: sl@0: // Read contents of install directory sl@0: User::LeaveIfError(dir.Open(iFs, iInstallDir, KEntryAttNormal)); sl@0: sl@0: TEntryArray dirEntries; sl@0: TInt readError = KErrNone; sl@0: sl@0: while (readError != KErrEof) sl@0: { sl@0: readError = dir.Read(dirEntries); sl@0: sl@0: if(readError != KErrNone && readError != KErrEof) sl@0: { sl@0: User::Leave(readError); sl@0: } sl@0: else sl@0: { sl@0: const TInt dirCount = dirEntries.Count(); sl@0: for (TInt i=0; iSetL(const_cast(dirEntries[i])); sl@0: if (installEntry->FileExt()==EUnknown) sl@0: { sl@0: CleanupStack::PopAndDestroy(); sl@0: } sl@0: else sl@0: { sl@0: User::LeaveIfError(aEntryArray.Append(installEntry)); sl@0: CleanupStack::Pop(installEntry); sl@0: } sl@0: } sl@0: } sl@0: } sl@0: sl@0: CleanupStack::PopAndDestroy(&dir); sl@0: } sl@0: sl@0: TBool CCentRepSWIWatcher::MatchEntries( const CInstallEntry &aSource, const CInstallEntry &aTarget) sl@0: { sl@0: return((aSource.Uid()==aTarget.Uid()) && sl@0: (aSource.FileExt()==aTarget.FileExt())); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::FindChangedEntriesL(TBool aStartup) sl@0: { sl@0: // Find added or updated entries sl@0: ReadInstallDirL(iCurrentInstallDirEntries); sl@0: sl@0: // We don't want cache activity during SWI operations sl@0: TServerResources::iCacheManager->DisableCache(); sl@0: sl@0: TRAPD(err, HandleFileChangesL(aStartup)); sl@0: sl@0: // SWI operations finished, enable cache sl@0: TServerResources::iCacheManager->EnableCache(); sl@0: User::LeaveIfError(err); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::HandleFileChangesL(TBool aStartup) sl@0: { sl@0: TInt newCount=iCurrentInstallDirEntries.Count(); sl@0: TInt currentCount=iInstallEntryArray.Count(); sl@0: TInt i; sl@0: TInt r; sl@0: TInt operation; sl@0: sl@0: // If both counts are 0, we shouldn't have been notified, just return sl@0: if( (newCount==0) && (currentCount==0)) sl@0: return; sl@0: sl@0: if(aStartup) sl@0: { sl@0: operation=ESASwisNone; sl@0: } sl@0: else sl@0: { sl@0: operation=iSWIOperation; sl@0: } sl@0: sl@0: if( newCount==0) // currentCount > 0, newCount = 0 sl@0: { // All installed files have been deleted sl@0: // Handle deletes of all files sl@0: for(i=0;iHandleFileDeleteL(operation); sl@0: } sl@0: // Free memory of elements sl@0: iInstallEntryArray.ResetAndDestroy(); sl@0: } sl@0: else sl@0: { sl@0: if( currentCount==0) // currentCount = 0, newCount > 0 sl@0: { // All new files need to be handled sl@0: for(i=0;iHandleFileCreateL(operation); sl@0: } sl@0: } sl@0: else // currentCount > 0, newCount > 0 sl@0: { sl@0: // Find added and modified entries by going through new entries and sl@0: // looking for them in current array sl@0: for(i=0;i=KErrNone) sl@0: { sl@0: CInstallEntry* currentEntry=iInstallEntryArray[r]; sl@0: if( newEntry->Modified() > currentEntry->Modified()) sl@0: { sl@0: // Deal with newly installed file, note use newEntry sl@0: // so we use new timestamp sl@0: newEntry->HandleFileUpdateL(operation); sl@0: } sl@0: } sl@0: else if(r==KErrNotFound) // File has been added sl@0: { sl@0: // Handle add sl@0: newEntry->HandleFileCreateL(operation); sl@0: // Don't leave on KErrNotFound sl@0: r=KErrNone; sl@0: } sl@0: User::LeaveIfError(r); sl@0: } sl@0: sl@0: // Find deleted entries by going through current entries and looking for them sl@0: // in new array sl@0: for(i=0;iHandleFileDeleteL(operation); sl@0: // Don't leave on KErrNotFound sl@0: r=KErrNone; sl@0: } sl@0: User::LeaveIfError(r); sl@0: } sl@0: } sl@0: sl@0: // Clear out old list sl@0: iInstallEntryArray.ResetAndDestroy(); sl@0: sl@0: // Re-read directory - if any files were corrupt they have been deleted sl@0: // during the merge, so we need to re-read in case this has occurred sl@0: ReadInstallDirL(iInstallEntryArray); sl@0: } sl@0: sl@0: SaveInstallDirL(); sl@0: iCurrentInstallDirEntries.ResetAndDestroy(); sl@0: } sl@0: sl@0: CInstallEntry::CInstallEntry() : sl@0: iUid(KNullUid), sl@0: iModified(0), sl@0: iFileExt(EUnknown) sl@0: { sl@0: } sl@0: sl@0: CInstallEntry::~CInstallEntry() sl@0: { sl@0: if( iRepository) sl@0: { sl@0: iRepository->Close(); sl@0: delete iRepository; sl@0: } sl@0: delete iNotifier; sl@0: } sl@0: sl@0: void CInstallEntry::SetL(TEntry& aEntry) sl@0: { sl@0: // Get uid from file name sl@0: const TInt KUidLen = 8; sl@0: TPtrC uidPtr=aEntry.iName.Des().LeftTPtr(KUidLen); sl@0: TLex lex=uidPtr; sl@0: sl@0: TUint32 uid; sl@0: User::LeaveIfError(lex.Val(uid,EHex)); sl@0: iUid.iUid=static_cast(uid); sl@0: sl@0: // save extension type sl@0: _LIT(KIniFileExtension, ".txt"); sl@0: _LIT(KExternalizedFileExt, ".cre"); sl@0: sl@0: const TInt KExtLen = 4; sl@0: TPtrC extPtr=aEntry.iName.Des().RightTPtr(KExtLen); sl@0: sl@0: if(extPtr.Compare(KIniFileExtension)==0) sl@0: { sl@0: iFileExt=EIni; sl@0: } sl@0: else if(extPtr.Compare(KExternalizedFileExt)==0) sl@0: { sl@0: iFileExt=ECre; sl@0: } sl@0: else sl@0: { sl@0: iFileExt=EUnknown; sl@0: } sl@0: sl@0: iModified=aEntry.iModified; sl@0: } sl@0: sl@0: void CInstallEntry::ConstructL() sl@0: { sl@0: // Create repository object sl@0: iRepository = new(ELeave) CServerRepository; sl@0: // Notifier needed to open repositories. sl@0: iNotifier = new(ELeave)CSessionNotifier ; sl@0: } sl@0: sl@0: CInstallEntry* CInstallEntry::NewL() sl@0: { sl@0: CInstallEntry* self=new(ELeave)CInstallEntry(); sl@0: CleanupStack::PushL(self); sl@0: self->ConstructL(); sl@0: CleanupStack::Pop(self); sl@0: return self; sl@0: } sl@0: sl@0: TUid CInstallEntry::Uid() const sl@0: { sl@0: return iUid; sl@0: } sl@0: sl@0: TTime CInstallEntry::Modified() const sl@0: { sl@0: return iModified; sl@0: } sl@0: sl@0: TCentRepFileType CInstallEntry::FileExt() const sl@0: { sl@0: return iFileExt; sl@0: } sl@0: sl@0: void CInstallEntry::ExternalizeL(RWriteStream& aStream) const sl@0: { sl@0: aStream << iUid.iUid; sl@0: aStream << iModified.Int64(); sl@0: TUint32 fileExt=iFileExt; sl@0: aStream << fileExt; sl@0: } sl@0: sl@0: void CInstallEntry::InternalizeL(RReadStream& aStream) sl@0: { sl@0: aStream >> iUid.iUid; sl@0: TInt64 time; sl@0: aStream >> time; sl@0: iModified=time; sl@0: TUint32 fileExt; sl@0: aStream >> fileExt; sl@0: iFileExt = static_cast(fileExt); sl@0: } sl@0: sl@0: sl@0: void CCentRepSWIWatcher::ReadAndInternalizeInstallDirL(const TDesC& aInstallDirFilePath) sl@0: { sl@0: RFile file; sl@0: User::LeaveIfError(file.Open(TServerResources::iFs,aInstallDirFilePath, EFileRead|EFileShareReadersOnly)); sl@0: CleanupClosePushL(file); sl@0: sl@0: CDirectFileStore* store = CDirectFileStore::FromLC (file); sl@0: if(store->Type()[0] != KDirectFileStoreLayoutUid) sl@0: { sl@0: User::Leave(KErrCorrupt); sl@0: } sl@0: sl@0: iInstallEntryArray.ResetAndDestroy(); sl@0: sl@0: // Get the root stream and attempt to read the index from it sl@0: TStreamId rootStreamId = store->Root() ; sl@0: RStoreReadStream rootStream ; sl@0: rootStream.OpenLC(*store, rootStreamId); sl@0: sl@0: // Internalize the repository sl@0: TUint32 count; sl@0: rootStream >> count; sl@0: for(TUint i=0; i> *installEntry; sl@0: User::LeaveIfError(iInstallEntryArray.Append(installEntry)); sl@0: CleanupStack::Pop(installEntry); sl@0: } sl@0: sl@0: CleanupStack::PopAndDestroy(&rootStream); sl@0: CleanupStack::PopAndDestroy(store); sl@0: CleanupStack::PopAndDestroy(); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::GetInstallDirL() sl@0: { sl@0: _LIT(KInstallDirFile, "installdir.bin"); sl@0: HBufC* filePath = HBufC::NewLC(TServerResources::iDataDirectory->Length() + KInstallDirFile().Length()); sl@0: TPtr installDirFilePath(filePath->Des()); sl@0: installDirFilePath.Append(*TServerResources::iDataDirectory); sl@0: installDirFilePath.Append(KInstallDirFile); sl@0: sl@0: TRAPD(err, ReadAndInternalizeInstallDirL(installDirFilePath)); // try to open installdir file and internalize its contents sl@0: if (err != KErrNone) sl@0: { sl@0: TInt fileDeleteErr = TServerResources::iFs.Delete(installDirFilePath); sl@0: // If a debug build - record error sl@0: #ifdef _DEBUG sl@0: if (fileDeleteErr != KErrNone && err != KErrNotFound) sl@0: { sl@0: RDebug::Print(_L("CCentRepSWIWatcher::GetInstallDirL - Failed to delete file. Error = %d"), fileDeleteErr); sl@0: } sl@0: #else sl@0: (void)fileDeleteErr; sl@0: #endif sl@0: sl@0: // No file saved - read initial contents of directory sl@0: ReadInstallDirL(iInstallEntryArray); sl@0: SaveInstallDirL(); sl@0: } sl@0: CleanupStack::PopAndDestroy(filePath); sl@0: } sl@0: sl@0: void CCentRepSWIWatcher::SaveInstallDirL() sl@0: { sl@0: _LIT(KInstallDirFile, "installdir.bin"); sl@0: _LIT(KInstallDirTmpFile, "installdir.tmp"); sl@0: sl@0: TBuf installDirTrnsFilePath; sl@0: installDirTrnsFilePath.Append(*TServerResources::iDataDirectory); sl@0: installDirTrnsFilePath.Append(KInstallDirTmpFile); sl@0: sl@0: // Create file store sl@0: CDirectFileStore* store = CDirectFileStore::ReplaceLC(TServerResources::iFs, installDirTrnsFilePath, sl@0: (EFileWrite | EFileShareExclusive)); sl@0: const TUid uid2 = KNullUid ; sl@0: store->SetTypeL(TUidType(KDirectFileStoreLayoutUid, uid2, KServerUid3)) ; sl@0: sl@0: // Write the stream index/dictionary as root stream within the store sl@0: // so we can access it when we do a restore later on sl@0: RStoreWriteStream rootStream ; sl@0: TStreamId rootStreamId = rootStream.CreateLC(*store) ; sl@0: sl@0: TUint32 count=iInstallEntryArray.Count(); sl@0: rootStream << count; sl@0: for(TUint i=0; iSetRootL(rootStreamId); sl@0: store->CommitL(); sl@0: CleanupStack::PopAndDestroy(store) ; sl@0: sl@0: TBuf installDirFilePath; sl@0: installDirFilePath.Append(*TServerResources::iDataDirectory); sl@0: installDirFilePath.Append(KInstallDirFile); sl@0: sl@0: User::LeaveIfError(TServerResources::iFs.Replace(installDirTrnsFilePath, installDirFilePath)); sl@0: } sl@0: sl@0: sl@0: void CInstallEntry::HandleFileDeleteL(TInt aOperation) sl@0: { sl@0: // File should only have been deleted if operation was uninstall sl@0: // or in startup case sl@0: if((aOperation!=ESASwisNone) && (aOperation!=ESASwisUninstall)) sl@0: User::Leave(KErrAbort); sl@0: sl@0: iRepository->HandleSWIDeleteL(Uid(), *iNotifier); sl@0: } sl@0: sl@0: void CInstallEntry::HandleFileCreateL(TInt aOperation) sl@0: { sl@0: // File should only have been created if operation was install sl@0: // or in startup case sl@0: if((aOperation!=ESASwisNone) && (aOperation!=ESASwisInstall)) sl@0: User::Leave(KErrAbort); sl@0: sl@0: iRepository->HandleSWIUpdateL(Uid(), Modified(), *iNotifier); sl@0: } sl@0: sl@0: void CInstallEntry::HandleFileUpdateL(TInt aOperation) sl@0: { sl@0: // File should only have been modified if operation was install sl@0: // or in startup case sl@0: if((aOperation!=ESASwisNone) && (aOperation!=ESASwisInstall)) sl@0: User::Leave(KErrAbort); sl@0: sl@0: iRepository->HandleSWIUpdateL(Uid(), Modified(), *iNotifier); sl@0: }