os/persistentdata/persistentstorage/store/UFILE/UF_DICT.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 1998-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
#include "UF_STD.H"
sl@0
    17
#include <e32math.h>
sl@0
    18
sl@0
    19
const TUid KUidDictionaryFile={0x100000E4};
sl@0
    20
sl@0
    21
_LIT(KSystemIniFileLocationSpec,"Z:\\System\\System.ini");
sl@0
    22
sl@0
    23
const TUid KSystemIniFileUid = {0x1000010C};
sl@0
    24
sl@0
    25
// Thread contention resolution constants
sl@0
    26
sl@0
    27
const TInt KSpinCount=40;
sl@0
    28
const TInt KLockoutLimit=0xc0000;	// to keep the lock-out time down to ~1sec (this is microseconds)
sl@0
    29
const TInt KFailTime=0x1000;		// ~ time to fail to open file
sl@0
    30
const TInt KSpinLimit=KLockoutLimit-KSpinCount*KFailTime;
sl@0
    31
sl@0
    32
LOCAL_C TUint8 const WaitDistribution[8]={0,0,0,0,0x1f,0x3f,0x7f,0xff};	// wait-time distribution mask
sl@0
    33
sl@0
    34
class RYieldThread : public RThread
sl@0
    35
	{
sl@0
    36
public:
sl@0
    37
	inline void Yield() const {SetPriority(Priority());}
sl@0
    38
	};
sl@0
    39
sl@0
    40
/////////////////////////////////////////////
sl@0
    41
// CDictionaryFileStore
sl@0
    42
/////////////////////////////////////////////
sl@0
    43
sl@0
    44
LOCAL_C void EnsurePathL(RFs& aFs,const TDesC& aName)
sl@0
    45
	{
sl@0
    46
	TInt r=aFs.MkDirAll(TParsePtrC(aName).DriveAndPath());
sl@0
    47
	if (r!=KErrAlreadyExists)
sl@0
    48
		User::LeaveIfError(r);
sl@0
    49
	}
sl@0
    50
sl@0
    51
EXPORT_C CDictionaryFileStore* CDictionaryFileStore::SystemL(RFs& aFs)
sl@0
    52
/** Opens the system dictionary file store.
sl@0
    53
sl@0
    54
@param aFs Handle to a file server session.
sl@0
    55
@return A pointer to the system file based dictionary store object. */
sl@0
    56
	{
sl@0
    57
	CDictionaryFileStore* store=SystemLC(aFs);
sl@0
    58
	CleanupStack::Pop();
sl@0
    59
	return store;
sl@0
    60
	}
sl@0
    61
sl@0
    62
EXPORT_C CDictionaryFileStore* CDictionaryFileStore::SystemLC(RFs& aFs)
sl@0
    63
/** Opens the system dictionary file store and puts the pointer to the file store 
sl@0
    64
object onto the cleanup stack.
sl@0
    65
sl@0
    66
@param aFs Handle to a file server session. 
sl@0
    67
@return A pointer to the system file based dictionary store object. */
sl@0
    68
	{
sl@0
    69
	TDriveUnit drive(static_cast<TUint>(RFs::GetSystemDrive()));	
sl@0
    70
	TParse parse;
sl@0
    71
	User::LeaveIfError(parse.Set(drive.Name(), &KSystemIniFileLocationSpec, NULL));
sl@0
    72
	
sl@0
    73
	EnsurePathL(aFs, parse.FullName());
sl@0
    74
	return OpenLC(aFs, parse.FullName(), KSystemIniFileUid);
sl@0
    75
	}
sl@0
    76
sl@0
    77
EXPORT_C CDictionaryFileStore* CDictionaryFileStore::OpenL(RFs& aFs,const TDesC& aName,TUid aFileId)
sl@0
    78
	/** Creates a file based dictionary store object.
sl@0
    79
	
sl@0
    80
	If the file with the specified full path name exists, then an attempt is made 
sl@0
    81
	to open an existing file store contained within this file. Any existing file 
sl@0
    82
	store must satisfy the following conditions:
sl@0
    83
	
sl@0
    84
	it must be a valid dictionary store
sl@0
    85
	
sl@0
    86
	the third UID component of the file store type must match the specified UID; 
sl@0
    87
	this UID serves to differentiate between dictionary stores
sl@0
    88
	
sl@0
    89
	otherwise the function leaves with KErrCorrupt.
sl@0
    90
	
sl@0
    91
	If the file with the specified full path name does not exist, then an attempt 
sl@0
    92
	is made to create a new file and to create a file based dictionary within 
sl@0
    93
	it. The third UID component of the file store type is set to the specified 
sl@0
    94
	UID value.
sl@0
    95
	
sl@0
    96
	Note that the file is opened in exclusive access mode.
sl@0
    97
	
sl@0
    98
	@param aFs Handle to a file server session. 
sl@0
    99
	@param aName The full path name of the file. 
sl@0
   100
	@param aUid3 The UID used to differentiate between dictionary stores.
sl@0
   101
	@return A pointer to the file based dictionary store object. 
sl@0
   102
	@see TUid
sl@0
   103
	@see TUidType */
sl@0
   104
	{
sl@0
   105
	CDictionaryFileStore* self = CDictionaryFileStore::OpenLC(aFs,aName,aFileId);
sl@0
   106
	CleanupStack::Pop();
sl@0
   107
	return self;
sl@0
   108
	}
sl@0
   109
sl@0
   110
EXPORT_C CDictionaryFileStore* CDictionaryFileStore::OpenLC(RFs& aFs,const TDesC& aName,TUid aFileId)
sl@0
   111
	/** Creates a file based dictionary store object and puts the pointer onto the 
sl@0
   112
	cleanup stack.
sl@0
   113
	
sl@0
   114
	If the file with the specified full path name exists, then an attempt is made 
sl@0
   115
	to open an existing file store contained within this file. Any existing file 
sl@0
   116
	store must satisfy the following conditions:
sl@0
   117
	
sl@0
   118
	it must be a valid dictionary store
sl@0
   119
	
sl@0
   120
	the third UID component of the file store type must match the specified UID; 
sl@0
   121
	this UID serves to differentiate between dictionary stores
sl@0
   122
	
sl@0
   123
	otherwise the function leaves with KErrCorrupt.
sl@0
   124
	
sl@0
   125
	If the file with the specified full path name does not exist, then an attempt 
sl@0
   126
	is made to create a new file and to create a file based dictionary within 
sl@0
   127
	it. The third UID component of the file store type is set to the specified 
sl@0
   128
	UID value.
sl@0
   129
	
sl@0
   130
	Note that the file is opened in exclusive access mode.
sl@0
   131
	
sl@0
   132
	@param aFs Handle to a file server session. 
sl@0
   133
	@param aName The full path name of the file. 
sl@0
   134
	@param aUid3 The UID used to differentiate between dictionary stores.
sl@0
   135
	@return A pointer to the file based dictionary store object. 
sl@0
   136
	@see TUid
sl@0
   137
	@see TUidType */
sl@0
   138
	{
sl@0
   139
	CDictionaryFileStore* self = new(ELeave) CDictionaryFileStore();
sl@0
   140
	CleanupStack::PushL(self);
sl@0
   141
	self->ConstructL(aFs,aName,aFileId);
sl@0
   142
	return self;
sl@0
   143
	}
sl@0
   144
sl@0
   145
void CDictionaryFileStore::ConstructL(RFs& aFs,const TDesC& aName,TUid aFileId)
sl@0
   146
//
sl@0
   147
// try to open the file - if this fails KErrNotFound try to create it
sl@0
   148
// if the file is in use retry after a pause
sl@0
   149
//
sl@0
   150
	{
sl@0
   151
	RYieldThread thread;
sl@0
   152
	TInt64 seed;
sl@0
   153
	const TUidType type(KPermanentFileStoreLayoutUid,KUidDictionaryFile,aFileId);
sl@0
   154
	for (TInt wait=KLockoutLimit;;)
sl@0
   155
		{
sl@0
   156
		RFile file;
sl@0
   157
        //  When the file server write caching is enabled, the order of file write operations is not guaranteed.
sl@0
   158
        // This could cause data inconsistency in some circumstances,for example, when the power is lost in the 
sl@0
   159
        // middle of a database transaction.Therefore, the file write caching is disabled for this file to maintain integrity.
sl@0
   160
sl@0
   161
		TInt r=file.Open(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
sl@0
   162
		switch (r)
sl@0
   163
			{
sl@0
   164
		case KErrNone:
sl@0
   165
			{
sl@0
   166
			TInt size;
sl@0
   167
			if (file.Size(size)==KErrNone && size!=0)
sl@0
   168
				{
sl@0
   169
				CFileStore* store=NULL;
sl@0
   170
				TRAP(r,store=CPermanentFileStore::FromL(file));
sl@0
   171
			    if (r==KErrNotSupported||r==KErrEof)
sl@0
   172
					r=KErrCorrupt; // treat a bad store file as corrupt
sl@0
   173
			    else if (r==KErrNone && store->Type()!=type)
sl@0
   174
					{
sl@0
   175
					// treat a wrong 3rd UID as corrupt
sl@0
   176
					delete store;
sl@0
   177
					store = NULL;
sl@0
   178
					r=KErrCorrupt;
sl@0
   179
					}
sl@0
   180
				if (r==KErrCorrupt)
sl@0
   181
					{
sl@0
   182
				// silently replace the entire file if it is corrupt
sl@0
   183
				//  When the file server write caching is enabled, the order of file write operations is not guaranteed.
sl@0
   184
				// This could cause data inconsistency in some circumstances,for example, when the power is lost in the 
sl@0
   185
				// middle of a database transaction.Therefore, the file write caching is disabled for this file to maintain integrity.
sl@0
   186
				//   
sl@0
   187
					r=file.Replace(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
sl@0
   188
					if (r==KErrInUse)
sl@0
   189
						break;  // try again later...
sl@0
   190
					if (r==KErrNone)
sl@0
   191
						{
sl@0
   192
						CreateStoreL(file,type);
sl@0
   193
						return;
sl@0
   194
						}
sl@0
   195
					}
sl@0
   196
				__LEAVE_IF_ERROR(r);
sl@0
   197
				__ASSERT_DEBUG(store != NULL, User::Invariant());
sl@0
   198
				//coverity[use_after_free]
sl@0
   199
    			iStore = store;
sl@0
   200
				if (store->Root()==KNullStreamId)
sl@0
   201
					CDictionaryStore::ConstructL();
sl@0
   202
				}
sl@0
   203
			else
sl@0
   204
				CreateStoreL(file,type);
sl@0
   205
			return;
sl@0
   206
			}
sl@0
   207
		case KErrNotFound:
sl@0
   208
            //  When the file server write caching is enabled, the order of file write operations is not guaranteed.
sl@0
   209
            // This could cause data inconsistency in some circumstances,for example, when the power is lost in the 
sl@0
   210
            // middle of a database transaction.Therefore, the file write caching is disabled for this file to maintain integrity.
sl@0
   211
			r=file.Create(aFs,aName,EFileShareExclusive|EFileWrite|EFileWriteDirectIO);
sl@0
   212
			if (r==KErrNone)
sl@0
   213
				{
sl@0
   214
				CreateStoreL(file,type);
sl@0
   215
				return;
sl@0
   216
				}
sl@0
   217
			else if (r==KErrAlreadyExists)
sl@0
   218
				;	// try and open after delay
sl@0
   219
			else
sl@0
   220
				__LEAVE(r);
sl@0
   221
			break;
sl@0
   222
		case KErrInUse:
sl@0
   223
			break;
sl@0
   224
		default:
sl@0
   225
			__LEAVE(r);
sl@0
   226
			}
sl@0
   227
		wait-=KFailTime;
sl@0
   228
		if (wait<=0)
sl@0
   229
			break;		// waited too long
sl@0
   230
		if (wait>KSpinLimit)
sl@0
   231
			{			// straight back to retry
sl@0
   232
			thread.Yield();	// force another reschedule
sl@0
   233
			continue;
sl@0
   234
			}
sl@0
   235
		// random wait time...
sl@0
   236
		if (wait==KSpinLimit)
sl@0
   237
			{	// initialise random number generator
sl@0
   238
			TThreadId id=thread.Id();
sl@0
   239
			TUint idVal=*(const TUint*)&id;
sl@0
   240
			seed = MAKE_TINT64(idVal^TUint(this),idVal^TUint(&id));
sl@0
   241
			Math::Rand(seed);
sl@0
   242
			Math::Rand(seed);
sl@0
   243
			}
sl@0
   244
		TInt pause=Math::Rand(seed)>>11;
sl@0
   245
		pause=(pause&WaitDistribution[(pause>>8)&0x7])<<10;
sl@0
   246
		if (pause)
sl@0
   247
			{
sl@0
   248
			wait-=pause;
sl@0
   249
			User::After(pause);
sl@0
   250
			}
sl@0
   251
		}
sl@0
   252
	__LEAVE(KErrInUse);
sl@0
   253
	}
sl@0
   254
sl@0
   255
void CDictionaryFileStore::CreateStoreL(RFile& aFile,const TUidType& aType)
sl@0
   256
	{
sl@0
   257
	CFileStore* store = CPermanentFileStore::NewL(aFile);
sl@0
   258
	iStore = store;
sl@0
   259
	store->SetTypeL(aType);
sl@0
   260
	CDictionaryStore::ConstructL();
sl@0
   261
	}