Update contrib.
1 // Copyright (c) 1998-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 const TUid KUidDictionaryFile={0x100000E4};
21 _LIT(KSystemIniFileLocationSpec,"Z:\\System\\System.ini");
23 const TUid KSystemIniFileUid = {0x1000010C};
25 // Thread contention resolution constants
27 const TInt KSpinCount=40;
28 const TInt KLockoutLimit=0xc0000; // to keep the lock-out time down to ~1sec (this is microseconds)
29 const TInt KFailTime=0x1000; // ~ time to fail to open file
30 const TInt KSpinLimit=KLockoutLimit-KSpinCount*KFailTime;
32 LOCAL_C TUint8 const WaitDistribution[8]={0,0,0,0,0x1f,0x3f,0x7f,0xff}; // wait-time distribution mask
34 class RYieldThread : public RThread
37 inline void Yield() const {SetPriority(Priority());}
40 /////////////////////////////////////////////
41 // CDictionaryFileStore
42 /////////////////////////////////////////////
44 LOCAL_C void EnsurePathL(RFs& aFs,const TDesC& aName)
46 TInt r=aFs.MkDirAll(TParsePtrC(aName).DriveAndPath());
47 if (r!=KErrAlreadyExists)
48 User::LeaveIfError(r);
51 EXPORT_C CDictionaryFileStore* CDictionaryFileStore::SystemL(RFs& aFs)
52 /** Opens the system dictionary file store.
54 @param aFs Handle to a file server session.
55 @return A pointer to the system file based dictionary store object. */
57 CDictionaryFileStore* store=SystemLC(aFs);
62 EXPORT_C CDictionaryFileStore* CDictionaryFileStore::SystemLC(RFs& aFs)
63 /** Opens the system dictionary file store and puts the pointer to the file store
64 object onto the cleanup stack.
66 @param aFs Handle to a file server session.
67 @return A pointer to the system file based dictionary store object. */
69 TDriveUnit drive(static_cast<TUint>(RFs::GetSystemDrive()));
71 User::LeaveIfError(parse.Set(drive.Name(), &KSystemIniFileLocationSpec, NULL));
73 EnsurePathL(aFs, parse.FullName());
74 return OpenLC(aFs, parse.FullName(), KSystemIniFileUid);
77 EXPORT_C CDictionaryFileStore* CDictionaryFileStore::OpenL(RFs& aFs,const TDesC& aName,TUid aFileId)
78 /** Creates a file based dictionary store object.
80 If the file with the specified full path name exists, then an attempt is made
81 to open an existing file store contained within this file. Any existing file
82 store must satisfy the following conditions:
84 it must be a valid dictionary store
86 the third UID component of the file store type must match the specified UID;
87 this UID serves to differentiate between dictionary stores
89 otherwise the function leaves with KErrCorrupt.
91 If the file with the specified full path name does not exist, then an attempt
92 is made to create a new file and to create a file based dictionary within
93 it. The third UID component of the file store type is set to the specified
96 Note that the file is opened in exclusive access mode.
98 @param aFs Handle to a file server session.
99 @param aName The full path name of the file.
100 @param aUid3 The UID used to differentiate between dictionary stores.
101 @return A pointer to the file based dictionary store object.
105 CDictionaryFileStore* self = CDictionaryFileStore::OpenLC(aFs,aName,aFileId);
110 EXPORT_C CDictionaryFileStore* CDictionaryFileStore::OpenLC(RFs& aFs,const TDesC& aName,TUid aFileId)
111 /** Creates a file based dictionary store object and puts the pointer onto the
114 If the file with the specified full path name exists, then an attempt is made
115 to open an existing file store contained within this file. Any existing file
116 store must satisfy the following conditions:
118 it must be a valid dictionary store
120 the third UID component of the file store type must match the specified UID;
121 this UID serves to differentiate between dictionary stores
123 otherwise the function leaves with KErrCorrupt.
125 If the file with the specified full path name does not exist, then an attempt
126 is made to create a new file and to create a file based dictionary within
127 it. The third UID component of the file store type is set to the specified
130 Note that the file is opened in exclusive access mode.
132 @param aFs Handle to a file server session.
133 @param aName The full path name of the file.
134 @param aUid3 The UID used to differentiate between dictionary stores.
135 @return A pointer to the file based dictionary store object.
139 CDictionaryFileStore* self = new(ELeave) CDictionaryFileStore();
140 CleanupStack::PushL(self);
141 self->ConstructL(aFs,aName,aFileId);
145 void CDictionaryFileStore::ConstructL(RFs& aFs,const TDesC& aName,TUid aFileId)
147 // try to open the file - if this fails KErrNotFound try to create it
148 // if the file is in use retry after a pause
153 const TUidType type(KPermanentFileStoreLayoutUid,KUidDictionaryFile,aFileId);
154 for (TInt wait=KLockoutLimit;;)
157 // When the file server write caching is enabled, the order of file write operations is not guaranteed.
158 // This could cause data inconsistency in some circumstances,for example, when the power is lost in the
159 // middle of a database transaction.Therefore, the file write caching is disabled for this file to maintain integrity.
161 TInt r=file.Open(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
167 if (file.Size(size)==KErrNone && size!=0)
169 CFileStore* store=NULL;
170 TRAP(r,store=CPermanentFileStore::FromL(file));
171 if (r==KErrNotSupported||r==KErrEof)
172 r=KErrCorrupt; // treat a bad store file as corrupt
173 else if (r==KErrNone && store->Type()!=type)
175 // treat a wrong 3rd UID as corrupt
182 // silently replace the entire file if it is corrupt
183 // When the file server write caching is enabled, the order of file write operations is not guaranteed.
184 // This could cause data inconsistency in some circumstances,for example, when the power is lost in the
185 // middle of a database transaction.Therefore, the file write caching is disabled for this file to maintain integrity.
187 r=file.Replace(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
189 break; // try again later...
192 CreateStoreL(file,type);
197 __ASSERT_DEBUG(store != NULL, User::Invariant());
198 //coverity[use_after_free]
200 if (store->Root()==KNullStreamId)
201 CDictionaryStore::ConstructL();
204 CreateStoreL(file,type);
208 // When the file server write caching is enabled, the order of file write operations is not guaranteed.
209 // This could cause data inconsistency in some circumstances,for example, when the power is lost in the
210 // middle of a database transaction.Therefore, the file write caching is disabled for this file to maintain integrity.
211 r=file.Create(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
214 CreateStoreL(file,type);
217 else if (r==KErrAlreadyExists)
218 ; // try and open after delay
229 break; // waited too long
231 { // straight back to retry
232 thread.Yield(); // force another reschedule
235 // random wait time...
236 if (wait==KSpinLimit)
237 { // initialise random number generator
238 TThreadId id=thread.Id();
239 TUint idVal=*(const TUint*)&id;
240 seed = MAKE_TINT64(idVal^TUint(this),idVal^TUint(&id));
244 TInt pause=Math::Rand(seed)>>11;
245 pause=(pause&WaitDistribution[(pause>>8)&0x7])<<10;
255 void CDictionaryFileStore::CreateStoreL(RFile& aFile,const TUidType& aType)
257 CFileStore* store = CPermanentFileStore::NewL(aFile);
259 store->SetTypeL(aType);
260 CDictionaryStore::ConstructL();