os/persistentdata/persistentstorage/store/UFILE/UF_DICT.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "UF_STD.H"
    17 #include <e32math.h>
    18 
    19 const TUid KUidDictionaryFile={0x100000E4};
    20 
    21 _LIT(KSystemIniFileLocationSpec,"Z:\\System\\System.ini");
    22 
    23 const TUid KSystemIniFileUid = {0x1000010C};
    24 
    25 // Thread contention resolution constants
    26 
    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;
    31 
    32 LOCAL_C TUint8 const WaitDistribution[8]={0,0,0,0,0x1f,0x3f,0x7f,0xff};	// wait-time distribution mask
    33 
    34 class RYieldThread : public RThread
    35 	{
    36 public:
    37 	inline void Yield() const {SetPriority(Priority());}
    38 	};
    39 
    40 /////////////////////////////////////////////
    41 // CDictionaryFileStore
    42 /////////////////////////////////////////////
    43 
    44 LOCAL_C void EnsurePathL(RFs& aFs,const TDesC& aName)
    45 	{
    46 	TInt r=aFs.MkDirAll(TParsePtrC(aName).DriveAndPath());
    47 	if (r!=KErrAlreadyExists)
    48 		User::LeaveIfError(r);
    49 	}
    50 
    51 EXPORT_C CDictionaryFileStore* CDictionaryFileStore::SystemL(RFs& aFs)
    52 /** Opens the system dictionary file store.
    53 
    54 @param aFs Handle to a file server session.
    55 @return A pointer to the system file based dictionary store object. */
    56 	{
    57 	CDictionaryFileStore* store=SystemLC(aFs);
    58 	CleanupStack::Pop();
    59 	return store;
    60 	}
    61 
    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.
    65 
    66 @param aFs Handle to a file server session. 
    67 @return A pointer to the system file based dictionary store object. */
    68 	{
    69 	TDriveUnit drive(static_cast<TUint>(RFs::GetSystemDrive()));	
    70 	TParse parse;
    71 	User::LeaveIfError(parse.Set(drive.Name(), &KSystemIniFileLocationSpec, NULL));
    72 	
    73 	EnsurePathL(aFs, parse.FullName());
    74 	return OpenLC(aFs, parse.FullName(), KSystemIniFileUid);
    75 	}
    76 
    77 EXPORT_C CDictionaryFileStore* CDictionaryFileStore::OpenL(RFs& aFs,const TDesC& aName,TUid aFileId)
    78 	/** Creates a file based dictionary store object.
    79 	
    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:
    83 	
    84 	it must be a valid dictionary store
    85 	
    86 	the third UID component of the file store type must match the specified UID; 
    87 	this UID serves to differentiate between dictionary stores
    88 	
    89 	otherwise the function leaves with KErrCorrupt.
    90 	
    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 
    94 	UID value.
    95 	
    96 	Note that the file is opened in exclusive access mode.
    97 	
    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. 
   102 	@see TUid
   103 	@see TUidType */
   104 	{
   105 	CDictionaryFileStore* self = CDictionaryFileStore::OpenLC(aFs,aName,aFileId);
   106 	CleanupStack::Pop();
   107 	return self;
   108 	}
   109 
   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 
   112 	cleanup stack.
   113 	
   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:
   117 	
   118 	it must be a valid dictionary store
   119 	
   120 	the third UID component of the file store type must match the specified UID; 
   121 	this UID serves to differentiate between dictionary stores
   122 	
   123 	otherwise the function leaves with KErrCorrupt.
   124 	
   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 
   128 	UID value.
   129 	
   130 	Note that the file is opened in exclusive access mode.
   131 	
   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. 
   136 	@see TUid
   137 	@see TUidType */
   138 	{
   139 	CDictionaryFileStore* self = new(ELeave) CDictionaryFileStore();
   140 	CleanupStack::PushL(self);
   141 	self->ConstructL(aFs,aName,aFileId);
   142 	return self;
   143 	}
   144 
   145 void CDictionaryFileStore::ConstructL(RFs& aFs,const TDesC& aName,TUid aFileId)
   146 //
   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
   149 //
   150 	{
   151 	RYieldThread thread;
   152 	TInt64 seed;
   153 	const TUidType type(KPermanentFileStoreLayoutUid,KUidDictionaryFile,aFileId);
   154 	for (TInt wait=KLockoutLimit;;)
   155 		{
   156 		RFile file;
   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.
   160 
   161 		TInt r=file.Open(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
   162 		switch (r)
   163 			{
   164 		case KErrNone:
   165 			{
   166 			TInt size;
   167 			if (file.Size(size)==KErrNone && size!=0)
   168 				{
   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)
   174 					{
   175 					// treat a wrong 3rd UID as corrupt
   176 					delete store;
   177 					store = NULL;
   178 					r=KErrCorrupt;
   179 					}
   180 				if (r==KErrCorrupt)
   181 					{
   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.
   186 				//   
   187 					r=file.Replace(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
   188 					if (r==KErrInUse)
   189 						break;  // try again later...
   190 					if (r==KErrNone)
   191 						{
   192 						CreateStoreL(file,type);
   193 						return;
   194 						}
   195 					}
   196 				__LEAVE_IF_ERROR(r);
   197 				__ASSERT_DEBUG(store != NULL, User::Invariant());
   198 				//coverity[use_after_free]
   199     			iStore = store;
   200 				if (store->Root()==KNullStreamId)
   201 					CDictionaryStore::ConstructL();
   202 				}
   203 			else
   204 				CreateStoreL(file,type);
   205 			return;
   206 			}
   207 		case KErrNotFound:
   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);
   212 			if (r==KErrNone)
   213 				{
   214 				CreateStoreL(file,type);
   215 				return;
   216 				}
   217 			else if (r==KErrAlreadyExists)
   218 				;	// try and open after delay
   219 			else
   220 				__LEAVE(r);
   221 			break;
   222 		case KErrInUse:
   223 			break;
   224 		default:
   225 			__LEAVE(r);
   226 			}
   227 		wait-=KFailTime;
   228 		if (wait<=0)
   229 			break;		// waited too long
   230 		if (wait>KSpinLimit)
   231 			{			// straight back to retry
   232 			thread.Yield();	// force another reschedule
   233 			continue;
   234 			}
   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));
   241 			Math::Rand(seed);
   242 			Math::Rand(seed);
   243 			}
   244 		TInt pause=Math::Rand(seed)>>11;
   245 		pause=(pause&WaitDistribution[(pause>>8)&0x7])<<10;
   246 		if (pause)
   247 			{
   248 			wait-=pause;
   249 			User::After(pause);
   250 			}
   251 		}
   252 	__LEAVE(KErrInUse);
   253 	}
   254 
   255 void CDictionaryFileStore::CreateStoreL(RFile& aFile,const TUidType& aType)
   256 	{
   257 	CFileStore* store = CPermanentFileStore::NewL(aFile);
   258 	iStore = store;
   259 	store->SetTypeL(aType);
   260 	CDictionaryStore::ConstructL();
   261 	}