Update contrib.
     1 // Copyright (c) 2008-2010 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.
 
    15 #include "FileBuf64.h"
 
    17 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 
    18 ///////////////////////////        PROFILER       ////////////////////////////////////////////////////////////////////
 
    19 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 
    23 extern TBool TheOsCallTimeDetailedProfileEnabled;//If true, the OS porting layer call details are enabled and for each call an entry will be added to the log file (epocwind.out).
 
    25 #define PROFILE_READ(pos, amount, err) \
 
    28 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    30 			++iFileReadCount; iFileReadAmount += (amount); \
 
    31 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Read¬%d¬%ld¬%d¬%ld¬%d\r\n"), (TUint32)this, iFileReadCount, pos, amount, iFileReadAmount, err); \
 
    35 #define PROFILE_WRITE(pos, amount, err) \
 
    38 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    40 			++iFileWriteCount; iFileWriteAmount += (amount); \
 
    41 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Write¬%d¬%ld¬%d¬%ld¬%d\r\n"), (TUint32)this, iFileWriteCount, pos, amount, iFileWriteAmount, err); \
 
    45 #define PROFILE_SIZE(size, err) \
 
    48 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    51 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Size¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSizeCount, size, err); \
 
    55 #define PROFILE_SETSIZE(size, err) \
 
    58 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    60 			++iFileSetSizeCount; \
 
    61 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬SetSize¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSetSizeCount, size, err); \
 
    65 #define PROFILE_FLUSH(err)	\
 
    68 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    71 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Flush¬%d¬¬¬¬%d\r\n"), (TUint32)this, iFileFlushCount, err); \
 
    75 #define PROFILE_CREATE(fname, err) \
 
    78 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    80 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Create¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
 
    84 #define PROFILE_OPEN(fname, err) \
 
    87 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    89 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Open¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
 
    93 #define PROFILE_TEMP(fname, err) \
 
    96 		if(TheOsCallTimeDetailedProfileEnabled) \
 
    98 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Temp¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
 
   102 #define PROFILE_ADOPT(fname, err) \
 
   105 		if(TheOsCallTimeDetailedProfileEnabled) \
 
   107 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Adopt¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
 
   111 #define PROFILE_CLOSE() \
 
   114 		if(TheOsCallTimeDetailedProfileEnabled) \
 
   116 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Close¬¬¬¬¬¬\r\n"), (TUint32)this); \
 
   120 //Resets the profiler counters
 
   121 void RFileBuf64::ProfilerReset()
 
   123 	iFileReadCount = 0; iFileReadAmount = 0; iFileWriteCount = 0; iFileWriteAmount = 0; iFileSizeCount = 0; iFileSetSizeCount = 0; iFileFlushCount = 0;
 
   128 #define PROFILE_READ(pos,amount, err)	void(0)
 
   129 #define PROFILE_WRITE(pos,amount, err)	void(0)
 
   131 #define PROFILE_SIZE(size, err)			void(0)
 
   132 #define PROFILE_SETSIZE(size, err)		void(0)
 
   133 #define PROFILE_FLUSH(err)				void(0)
 
   135 #define PROFILE_CREATE(fname, err)		void(0)
 
   136 #define PROFILE_OPEN(fname, err)		void(0)
 
   137 #define PROFILE_TEMP(fname, err)		void(0)
 
   138 #define PROFILE_ADOPT(fname, err)		void(0)
 
   139 #define PROFILE_CLOSE()					void(0)
 
   144 This constant is used for initializing the RFileBuf64::iFileSize data member and means that
 
   145 the iFileSize is not yet initialized with the real file size value. 
 
   146 (RFileBuf64::iFileSize caches the file size value)
 
   149 static const TInt KFileSizeNotSet = -1;
 
   152 This constant is used as a default initializer for the RFileBuf64::iNextReadFilePos data member,
 
   153 indicating that the "guessed" file read offset is invalid and should not be used.
 
   156 static const TInt KNextReadFilePosNotSet = -1;
 
   158 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 
   159 ///////////////////////////        ASSERTS & INVARIANT      //////////////////////////////////////////////////////////
 
   160 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 
   164 #define __FILEBUF64_INVARIANT() Invariant()
 
   167 String literal used in _DEBUG mode for indicating that the reported panic happened inside the RFileBuf64 implementation.
 
   172 _LIT(KPanicCategory, "FBuf64");
 
   175 Set of numeric constants used together with the KPanicCategory string literal in _DEBUG mode for providing more detailed
 
   176 information about the reason of the panic.
 
   183 	EFBufPanicCapacity = 1,				//1
 
   187 	EFBufPanicFileSize,					//5
 
   188 	EFBufPanicFileHandle,
 
   192 	EFBufPanicFileNameLen,				//10
 
   195 	EFBufPanicNextReadFilePos,
 
   196 	EFBufPanicNextReadFilePosHits,
 
   197 	EFBufPanicFileBlockSize,			//15
 
   198 	EFBufPanicRwDataLength,
 
   202 Helper function used in the implementation of the __FBUF64_ASSERT() macro.
 
   203 In case if the expression in __FBUF64_ASSERT() macro evaluates to false, 
 
   204 PanicFileBuf64() will use the supplied aLine and aPanicCode arguments together with the KPanicCategory string literal
 
   205 to prepare and print out a line (including the time of the panic) to the default log. The calling thread will be panic'ed
 
   212 static void PanicFileBuf64(TInt aLine, TFileBufPanic64 aPanicCode)
 
   216 	TDateTime dt = time.DateTime();
 
   218 	tbuf.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
 
   221 	_LIT(KFormat,"**%S:RFileBuf64 panic %d, at line(%d)");
 
   222 	buf.Format(KFormat, &tbuf, aPanicCode, aLine);
 
   224 	User::Panic(KPanicCategory, aPanicCode);
 
   228 This macro should be used when there is a need to panic the client/server if "expr" condition is not satisfied.
 
   229 Works in only in debug mode. In release mode evaluates to nothing.
 
   233 @see PanicFileBuf64()
 
   236 #define __FBUF64_ASSERT(expr, panicCode)	(void)(!(expr) ? ::PanicFileBuf64(__LINE__, panicCode) : void(0))
 
   240 #define __FILEBUF64_INVARIANT() void(0)
 
   242 #define __FBUF64_ASSERT(expr, panicCode) 	void(0)
 
   246 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 
   247 ///////////////////////////        RFileBuf64    /////////////////////////////////////////////////////////////////////
 
   248 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
 
   251 Initializes RFileBuf64 data members with their default values.
 
   253 @param aMinCapacity Minimal file buffer size (capacity) in bytes.
 
   255 @panic FBuf64 1 In _DEBUG mode - aMinCapacity is 0 or negative.
 
   257 RFileBuf64::RFileBuf64(TInt aMinCapacity) :
 
   258 	iCapacity(aMinCapacity),
 
   260 	iReadAheadSize(RFileBuf64::KDefaultReadAheadSize),
 
   263 	__FBUF64_ASSERT(aMinCapacity > 0, EFBufPanicCapacity);
 
   267 Initializes the RFileBuf64 object and creates and opens a new file that will be accessed through RFileBuf64 public interface.
 
   268 If the file already exists, an error is returned.
 
   269 If the resulting path does not exist, then the operation cannot proceed and the function returns an error code.
 
   271 @param aFs       The file server session.
 
   272 @param aFileName The name of the file. Any path components (i.e. drive letter
 
   273                  or directory), which are not specified, are taken from
 
   275 @param aFileMode The mode in which the file is opened. See TFileMode for details.
 
   277 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   280 @see RFile64::Create()
 
   282 @panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
 
   283 @panic FBuf64 10 In _DEBUG mode - Invalid file name length (zero file name length).
 
   285 TInt RFileBuf64::Create(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
 
   287 	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
 
   288 	__FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
 
   290 	TInt err = DoPreInit();
 
   293 	    err = iFile.Create(aFs, aFileName, aFileMode);
 
   295 	PROFILE_CREATE(aFileName, err);
 
   296 	return DoPostInit(err);
 
   300 Initializes the RFileBuf64 object and opens an existing file that will be accessed through RFileBuf64 public interface.
 
   301 If the file does not already exist, an error is returned.
 
   303 @param aFs       The file server session.
 
   304 @param aFileName The name of the file. Any path components (i.e. drive letter
 
   305                  or directory), which are not specified, are taken from
 
   307 @param aFileMode The mode in which the file is opened. See TFileMode for details.
 
   309 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   314 @panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
 
   315 @panic FBuf64 10 In _DEBUG mode - Invalid file name length (zero file name length).
 
   317 TInt RFileBuf64::Open(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
 
   319 	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
 
   320 	__FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
 
   322     TInt err = DoPreInit();
 
   325         err = iFile.Open(aFs, aFileName, aFileMode);
 
   327 	PROFILE_OPEN(aFileName, err);
 
   328     return DoPostInit(err);
 
   332 Initializes the RFileBuf64 object and creates and opens a temporary file with unique name that will be accessed through 
 
   333 RFileBuf64 public interface.
 
   335 @param aFs       The file server session.
 
   336 @param aPath     The directory in which the file is created.
 
   337 @param aFileName On return, contains the full path and file name of the file.
 
   338                  The filename is guaranteed to be unique within the directory
 
   340 @param aFileMode The mode in which the file is opened. The access mode is
 
   341                  automatically set to EFileWrite. See TFileMode for details.
 
   343 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   348 @panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
 
   350 TInt RFileBuf64::Temp(RFs& aFs, const TDesC& aPath, TFileName& aFileName, TUint aFileMode)
 
   352 	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
 
   354     TInt err = DoPreInit();
 
   357         err = iFile.Temp(aFs, aPath, aFileName, aFileMode);
 
   359 	PROFILE_TEMP(aFileName, err);
 
   360     return DoPostInit(err);
 
   364 Initializes the RFileBuf64 object and creates and adopts an already open file from a client that will be accessed through 
 
   365 RFileBuf64 public interface.
 
   366 The client's RFs and RFile or RFile64 handles are contained in message slots within aMsg.
 
   367 Assumes that the client's RFs and RFile or RFile64 handles have been sent to the server using TransferToServer().
 
   369 @param	aMsg	  The message received from the client
 
   370 @param	aFsIndex  The index that identifies the message slot 
 
   371 				  of a file server session (RFs) handle
 
   372 @param aFileIndex The index that identifies the message slot 
 
   373 				  of the sub-session (RFile or RFile64) handle of the already opened file
 
   375 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   378 @see RFile64::AdoptFromClient()
 
   379 @see KMaxMessageArguments
 
   381 @panic FBuf64  8 In _DEBUG mode - Invalid aMsg object (null message handle).
 
   382 @panic FBuf64  9 In _DEBUG mode - Invalid file session handle message slot index or invalid file handle message slot index.
 
   383                  (Probably negative index or index bigger or equal to KMaxMessageArguments)
 
   385 TInt RFileBuf64::AdoptFromClient(const RMessage2& aMsg, TInt aFsIndex, TInt aFileIndex)
 
   387 	__FBUF64_ASSERT(aMsg.Handle() != 0, EFBufPanicMsgHandle);
 
   388 	__FBUF64_ASSERT(aFsIndex >= 0 && aFsIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
 
   389 	__FBUF64_ASSERT(aFileIndex >= 0 && aFileIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
 
   391     TInt err = DoPreInit();
 
   394         err = iFile.AdoptFromClient(aMsg, aFsIndex, aFileIndex);
 
   396 	PROFILE_ADOPT(KNullDesC, err);
 
   397     return DoPostInit(err);
 
   401 Writes to the file the pending data (if the buffer contains pending data), closes the file and releases
 
   402 the RFileBuf64 resources. 
 
   403 RFileBuf64::Flush() should be called before RFileBuf64::Close() to ensure that if there are pending data, they will
 
   404 be written to the file and if the operation fails, the caller will be notified with an appropriate return error.
 
   406 @see RFileBuf64::Flush()
 
   408 void RFileBuf64::Close()
 
   410 	if(iBase != 0 && iFile.SubSessionHandle() != 0)
 
   412 		(void)DoFileWrite2();
 
   421 Calculates and sets optimal read-ahead buffer size.
 
   422 aBlockSize and aReadRecBufSize values are retrieved by the caller from the file system.
 
   424 Initialization rules:
 
   425 Rule 1: If aReadRecBufSize is positive, bigger than the default read-ahead and 
 
   426         a power of two then the read-ahead value will be
 
   427         initialized with aReadRecBufSize (if aReadRecBufSize is less than the buffer capacity otherwise
 
   428         the buffer capacity will be used as a read-ahead value). 
 
   429 Rule 2: If rule#1 is not applicable then the same checks, as in rule#1, are performed this time for aBlockSize.
 
   430         If aBlockSize passes the checks then it will be used as a read-ahead value. 
 
   433 @param aBlockSize The size of a file block in bytes
 
   434 @param aReadRecBufSize The recommended buffer size for optimised reading performance
 
   436 @return The new read-ahead value
 
   438 @see TVolumeIOParamInfo
 
   440 TInt RFileBuf64::SetReadAheadSize(TInt aBlockSize, TInt aReadRecBufSize)
 
   442 	__FILEBUF64_INVARIANT();
 
   443 	if((aReadRecBufSize & (aReadRecBufSize - 1)) == 0 && aReadRecBufSize > RFileBuf64::KDefaultReadAheadSize)
 
   445 		iReadAheadSize = aReadRecBufSize > iCapacity ? iCapacity : aReadRecBufSize;
 
   447 	else if((aBlockSize & (aBlockSize - 1)) == 0 && aBlockSize > RFileBuf64::KDefaultReadAheadSize)
 
   449 		iReadAheadSize = aBlockSize > iCapacity ? iCapacity : aBlockSize;
 
   451 	__FILEBUF64_INVARIANT();
 
   452 	return iReadAheadSize;
 
   456 Reads from the file at the specified position (aFilePos).
 
   457 If the data to be read is in the buffer, then the data will be taken from the buffer.
 
   459 @param aFilePos Position of first byte to be read.  This is an offset from
 
   460             the start of the file. 
 
   461             If aPos is beyond the end of the file, the function returns
 
   462             a zero length descriptor.
 
   463 @param aDes Descriptor into which binary data is read. Any existing contents 
 
   464             are overwritten. On return, its length is set to the number of
 
   467 @return KErrNone if successful, otherwise one of the other system-wide error  codes.
 
   469 @panic FBuf64  4 In _DEBUG mode - negative aFilePos value.
 
   470 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
   472 @see RFileBuf64::Invariant()
 
   474 TInt RFileBuf64::Read(TInt64 aFilePos, TDes8& aDes)
 
   476 	__FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
 
   477 	__FILEBUF64_INVARIANT();
 
   479 	//0. The output buffer max len is 0
 
   480 	if(aDes.MaxLength() == 0)
 
   482 		__FILEBUF64_INVARIANT();
 
   485 	//1. Initialize the "iFileSize" if it is not initialized yet
 
   486 	TInt err = DoFileSize();
 
   489 		__FILEBUF64_INVARIANT();
 
   492 	//2. Optimize the buffer capacity
 
   493 	TInt len = aDes.MaxLength();
 
   494 	if((err = DoSetCapacity(len)) != KErrNone)
 
   498 	//3. Too big "read" request - read directly from the file
 
   501 		if((aFilePos + len) > iFilePos && aFilePos < (iFilePos + iLength))
 
   502 			{//Write the pending data if the iDirty flag is set, otherwise preserve the buffer content.
 
   503 			err = DoFileWrite1(aFilePos);
 
   507 			err = iFile.Read(aFilePos, aDes);
 
   508 			PROFILE_READ(aFilePos, aDes.Size(), err);
 
   510 		__FILEBUF64_INVARIANT();
 
   513 	//4. The requested data size is smaller than the buffer capacity
 
   514 	TUint8* outptr = const_cast <TUint8*> (aDes.Ptr());
 
   515 	while(len > 0 && err == KErrNone && aFilePos < iFileSize)
 
   517 		//1. If part or all of the data is in the buffer - copy the data to the target location
 
   518 		if(aFilePos >= iFilePos && aFilePos < (iFilePos + iLength))
 
   520 			TInt blocklen = Min(len, (iFilePos + iLength - aFilePos));
 
   521 			outptr = Mem::Copy(outptr, iBase + (aFilePos - iFilePos), blocklen);
 
   523 			aFilePos += blocklen;
 
   525 		//2. Perform a read-ahead operation
 
   528 			//Write the pending data if the iDirty flag is set, otherwise preserve the buffer content.
 
   529 			err = DoFileWrite1(aFilePos);
 
   534 			if(iNextReadFilePos != aFilePos)
 
   535 				{//Guessed read ahead was wrong. Direct "file read" operation
 
   536 				iNextReadFilePosHits = 0;
 
   537 				TPtr8 ptr2(outptr, len);
 
   538 				err = iFile.Read(aFilePos, ptr2);
 
   539 				PROFILE_READ(aFilePos, ptr2.Size(), err);
 
   542 					iNextReadFilePos = aFilePos + len;
 
   543 					len -= ptr2.Length();
 
   547 			//The guessed from the previous "file read" operation file pos is correct. Start reading-ahead.
 
   548 			const TInt KMaxReadFilePosHits = 4;//The max read-ahead buffer size can be up to 2^4 times the iReadAheadSize
 
   549 			if(iNextReadFilePosHits < KMaxReadFilePosHits)
 
   551 				++iNextReadFilePosHits;
 
   553 			TInt maxReadAhead = iReadAheadSize * (1 << iNextReadFilePosHits);
 
   554 			TInt align = (aFilePos + len + maxReadAhead) & (iReadAheadSize - 1);
 
   555 			TInt readahead = maxReadAhead - align;
 
   558 				// if read-ahead doesn't cross block boundary do it all
 
   559 				readahead = maxReadAhead;	
 
   561 			TPtr8 ptr(iBase, Min(iCapacity, (len + readahead)));
 
   562 			err = iFile.Read(aFilePos, ptr);
 
   563 			PROFILE_READ(aFilePos, ptr.Size(), err);
 
   567 				iLength = ptr.Length();	
 
   568 				iNextReadFilePos = iFilePos + iLength;
 
   580 	aDes.SetLength(aDes.MaxLength() - len);
 
   581 	__FILEBUF64_INVARIANT();
 
   586 Writes to the file at the specified offset (aFilePos) within the file.
 
   587 If certain conditions are met, the data will be stored in the buffer - no call to the file server.
 
   589 @param aFilePos The offset from the start of the file at which the first byte is written. 
 
   590                 If a position beyond the end of the file is specified, then
 
   591                 the write operation begins at the end of the file.
 
   592                 If the position has been locked, then the write fails.
 
   594 @param aData The descriptor from which binary data is written. The function writes 
 
   595              the entire contents of aData to the file.
 
   597 @return KErrNone if successful, otherwise one of the other system-wide error  codes.
 
   599 @panic FBuf64  4 In _DEBUG mode - negative aFilePos value.
 
   600 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
   602 @see RFileBuf64::Invariant()
 
   604 TInt RFileBuf64::Write(TInt64 aFilePos, const TDesC8& aData)
 
   606 	__FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
 
   607 	__FILEBUF64_INVARIANT();
 
   608 	if(aData.Length() == 0)
 
   610 		__FILEBUF64_INVARIANT();
 
   613 	TInt err = DoFileSize();
 
   616 		__FILEBUF64_INVARIANT();
 
   619 	if((err = DoSetCapacity(aData.Length())) != KErrNone)
 
   623 	DoDiscardBufferedReadData();
 
   624 	const TUint8* data = aData.Ptr();
 
   625 	for(TInt len = aData.Length(); len > 0 && err == KErrNone;)
 
   627 		//1. The new write pos is before the buffered file pos
 
   628 		if(aFilePos < iFilePos)
 
   630 			//If the new data sticks to/overlapps the old data and there is room in the buffer to move the old data 
 
   631 			//toward the end, then the new data can be copied at the beginning of the buffer.
 
   632 			if((aFilePos + len) >= iFilePos && (iFilePos - aFilePos) <= (iCapacity - iLength))
 
   634 				(void)Mem::Copy(iBase + (iFilePos - aFilePos), iBase, iLength);	//Make room - move the existing data toward the end
 
   635 				(void)Mem::Copy(iBase, data, len);								//of the buffer. Stick the new data to the old data
 
   636 				iLength += (iFilePos - aFilePos);
 
   637 				iFilePos = aFilePos;										//The new file pos is associated with the buffer
 
   638 				iFileSize = Max(iFileSize, (iFilePos + iLength));
 
   639 				len = 0;													//No more new data
 
   643 			//The "aFilePos" is too far before the "iFilePos". Write the buffer and associate the new pos with the buffer
 
   645 				err = DoFileWrite2(aFilePos);
 
   648 		//2. The new write pos is after the associated file pos + the data length.
 
   649 		else if(aFilePos > (iFilePos + iLength))
 
   651 			if(aFilePos > iFileSize)											//Beyond the end of the file
 
   653 				if((iFilePos + iLength) == iFileSize && (aFilePos - iFilePos) < iCapacity)	
 
   654 					{															//but within the buffer => extend the file with zeros.
 
   655 					Mem::FillZ(iBase + iLength, aFilePos - iFilePos - iLength);
 
   656 					iLength = aFilePos - iFilePos;
 
   657 					iFileSize = Max(iFileSize, (iFilePos + iLength));
 
   661 				//Beyond the end of the file and not in the buffer - write the buffer to the file.
 
   663                     err = DoFileWrite2(aFilePos);
 
   667 			//Within the file, not in the buffer - write the buffer and associate the new file pos with the buffer
 
   669 				err = DoFileWrite2(aFilePos);
 
   672 		//3. The new write pos is in the buffer, but the data length is too big
 
   673 		//   (For SQLite is OK, otherwise the whole block must be written to the file)
 
   674 		//4. The new write pos is in the buffer, the data entirely fits in the buffer
 
   677             if (iFilePos+iCapacity == aFilePos)	//The buffer is full. The new position to write is the end of the buffer.
 
   679                 err = DoFileWrite2(aFilePos);
 
   683                 TInt amount = Min(len, (iCapacity - (aFilePos - iFilePos)));
 
   684                 const TUint8* end = Mem::Copy(iBase + (aFilePos - iFilePos), data, amount);
 
   685                 iLength = Max(iLength, (end - iBase));
 
   686                 iFileSize = Max(iFileSize, (iFilePos + iLength));
 
   694 	__FILEBUF64_INVARIANT();
 
   699 Gets the current file size.
 
   701 @param aFileSize On return, the size of the file in bytes.
 
   703 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   705 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
 
   707 @see RFileBuf64::Invariant()
 
   709 TInt RFileBuf64::Size(TInt64& aFileSize)
 
   711 	__FILEBUF64_INVARIANT();
 
   712 	TInt err = DoFileSize();
 
   715 		aFileSize = iFileSize;
 
   717 	__FILEBUF64_INVARIANT();
 
   724 If the size of the file is reduced, data may be lost from the end of the file.
 
   728 1. The current file position remains unchanged unless SetSize() reduces the size 
 
   729    of the file in such a way that the current file position is now beyond
 
   730    the end of the file. In this case, the current file position is set to
 
   733 2. If the file was not opened for writing, an error is returned.
 
   735 @param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
 
   737 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   739 @panic FBuf64  5 In _DEBUG mode - negative aFileSize value.
 
   740 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
   742 @see RFileBuf64::Invariant()
 
   744 TInt RFileBuf64::SetSize(TInt64 aFileSize)
 
   746 	__FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
 
   747 	__FILEBUF64_INVARIANT();
 
   748 	return DoSetFileSize(aFileSize);
 
   752 Writes the pending data and then flushes the file.
 
   754 Although RFileBuf64::Close() also flushes internal buffers, it is better
 
   755 to call RFileBuf64::Flush() before the file is closed. This is because Close() returns no 
 
   756 error information, so there is no way of telling whether the final data was 
 
   757 written to the file successfully or not.
 
   759 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   761 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
 
   763 @see RFileBuf64::Invariant()
 
   765 TInt RFileBuf64::Flush()
 
   767 	__FILEBUF64_INVARIANT();
 
   768 	return DoFileFlush();
 
   772 Gets information about the drive on which this file resides.
 
   774 @param aDriveNumber On return, the drive number.
 
   776 @param aDriveInfo   On return, contains information describing the drive
 
   777                     and the medium mounted on it. The value of TDriveInfo::iType
 
   778                     shows whether the drive contains media.
 
   780 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   782 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
 
   784 @see RFileBuf64::Invariant()
 
   786 TInt RFileBuf64::Drive(TInt& aDriveNumber, TDriveInfo& aDriveInfo) const
 
   788 	__FILEBUF64_INVARIANT();
 
   789 	return iFile.Drive(aDriveNumber, aDriveInfo);
 
   793 Initializes RFileBuf64 data members with their initial values.   
 
   794 Allocates memory for the file buffer. 
 
   796 @return KErrNone if successful, 
 
   797         KErrNoMemory out of memory;
 
   799 TInt RFileBuf64::DoPreInit()
 
   802     iReadAheadSize = RFileBuf64::KDefaultReadAheadSize;
 
   803     iBase = static_cast <TUint8*> (User::Alloc(iCapacity));
 
   804     return iBase ? KErrNone : KErrNoMemory;   
 
   808 Performs post-initialization of the RFileBuf64 object.   
 
   809 If aInitErr is not KErrNone, then the buffer memory will be released.
 
   810 The function returns the aInitErr value to the caller. 
 
   812 @param aInitErr The result of the performed before the call RFileBuf64 initialization.
 
   814 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   816 TInt RFileBuf64::DoPostInit(TInt aInitErr)
 
   818     if(aInitErr != KErrNone)
 
   827 Discards the content of the RFileBuf64 object returning it to the state as if it has just been created. 
 
   829 void RFileBuf64::DoDiscard()
 
   833 	iFileSize = KFileSizeNotSet;
 
   835 	iNextReadFilePos = KNextReadFilePosNotSet;
 
   836 	iNextReadFilePosHits = 0;
 
   840 Gets the current file size. 
 
   841 If iFileSize value is valid, then no call to the file server will be made.
 
   842 Otherwise the file server will be called and the file size - stored (cached) in iFileSize data member for later use.
 
   844 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   846 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
 
   848 @see RFileBuf64::Invariant()
 
   850 TInt RFileBuf64::DoFileSize()
 
   852 	__FILEBUF64_INVARIANT();
 
   853 	if(iFileSize != KFileSizeNotSet)
 
   855 		__FILEBUF64_INVARIANT();
 
   858 	TInt err = iFile.Size(iFileSize);
 
   859 	PROFILE_SIZE(iFileSize, err);
 
   866         iRealFileSize = iFileSize;
 
   868 	__FILEBUF64_INVARIANT();
 
   874 If the buffer contains pending data, the data will be written to the file 
 
   875 before the "set file size" operation, if certain conditions are met. 
 
   877 @param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
 
   879 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   881 @panic FBuf64  5 In _DEBUG mode - negative aFileSize value.
 
   882 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
   884 @see RFileBuf64::Invariant()
 
   886 TInt RFileBuf64::DoSetFileSize(TInt64 aFileSize)
 
   888 	__FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
 
   889 	__FILEBUF64_INVARIANT();
 
   890 	if(aFileSize < iFilePos)
 
   895 	//If the new file size is "in" the buffer then change the "iLength"
 
   896 	else if(aFileSize < (iFilePos + iLength))
 
   898 		iLength = aFileSize - iFilePos;
 
   900 	TInt err = iFile.SetSize(aFileSize);
 
   901 	PROFILE_SETSIZE(aFileSize, err);
 
   908 		iFileSize = aFileSize;
 
   909 		iRealFileSize = aFileSize;
 
   911 	__FILEBUF64_INVARIANT();
 
   916 Writes the pending data and flushes the file.
 
   918 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   920 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
 
   922 @see RFileBuf64::Invariant()
 
   924 TInt RFileBuf64::DoFileFlush()
 
   926 	__FILEBUF64_INVARIANT();
 
   927 	TInt err = DoFileWrite2();//Write the buffer if the iDirty flag is set. Do not preserve the buffer content and file pos.
 
   930 		__FILEBUF64_INVARIANT();
 
   940 	__FILEBUF64_INVARIANT();
 
   945 Writes the buffered data to the file if the iLength value is > 0.
 
   946 If the file write operation extends the file, the iFileSize data member will be initialized with the new file size.
 
   947 No changes occur in the other data member values.
 
   949 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
   951 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
   953 @see RFileBuf64::DoFileWrite1()
 
   954 @see RFileBuf64::DoFileWrite2()
 
   955 @see RFileBuf64::Invariant()
 
   957 TInt RFileBuf64::DoFileWrite()
 
   959 	__FILEBUF64_INVARIANT();
 
   962 		__FILEBUF64_INVARIANT();
 
   965 	TPtrC8 data(iBase, iLength);
 
   967 	if(iFilePos > iRealFileSize )
 
   969         err = DoSetFileSize(iFileSize);
 
   973         err = iFile.Write(iFilePos, data);
 
   975 	PROFILE_WRITE(iFilePos, iLength, err);
 
   978 		iRealFileSize = iFileSize;
 
   984 	__FILEBUF64_INVARIANT();
 
   989 Writes the buffered data to the file if the iDirty flag is set.
 
   990 If the iDirty flag is set and the file write operation was successful, the iFilePos will be initialized with
 
   991 the aNewFilePos value, the iLength will be set to 0.
 
   992 This method is called from RFileBuf64::Read(), where:
 
   993  - if the buffer contains cached writes (iDirty flag is set), the buffer has to be flushed and iFilePos initialized
 
   994    with aNewFilePos - the offset in the file where the next file read operation should start from;
 
   995  - if the buffer contains cached reads, then nothing happens, the buffer content will be kept;
 
   996 The function resets the iDirty flag.
 
   998 @param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
 
   999 				   the aNewFilePos value.
 
  1001 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
  1003 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
  1005 @panic FBuf64  4 In _DEBUG mode - negative aNewFilePos value.
 
  1007 @see RFileBuf64::Read()
 
  1008 @see RFileBuf64::DoFileWrite()
 
  1009 @see RFileBuf64::DoFileWrite2()
 
  1010 @see RFileBuf64::Invariant()
 
  1012 TInt RFileBuf64::DoFileWrite1(TInt64 aNewFilePos)
 
  1014 	__FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
 
  1015 	__FILEBUF64_INVARIANT();
 
  1016 	TInt err = KErrNone;
 
  1019 		err = DoFileWrite();
 
  1022 			iFilePos = aNewFilePos;
 
  1027 	__FILEBUF64_INVARIANT();
 
  1032 Writes the buffered data to the file if the iDirty flag is set.
 
  1033 If the file write operation was successful or if the iDirty flag was not set, the iFilePos will be initialized with
 
  1034 the aNewFilePos value, the iLength will be set to 0.
 
  1035 This method is called from RFileBuf64::Write() an other RFileBuf64 methods (but not from RFileBuf64::Read()), where:
 
  1036  - if the buffer contains cached writes (iDirty flag is set), the buffer has to be flushed and iFilePos initialized
 
  1037    with aNewFilePos - the offset in the file for which the write data will be cached in the buffer;
 
  1038  - if the buffer contains cached reads, then the buffer content will be destroyed, iFilePos initialized with aNewFilePos
 
  1039    and iLength set to 0;
 
  1040 The function resets the iDirty flag.
 
  1041 The difference between RFileBuf64::DoFileWrite1() and RFileBuf64::DoFileWrite2() is:
 
  1042  - RFileBuf64::DoFileWrite1() perserves the buffer content if iDirty is not set;
 
  1043  - RFileBuf64::DoFileWrite2() always destroys the buffer content and initializes iFilePos;
 
  1045 @param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
 
  1046 				   the aNewFilePos value.
 
  1048 @return KErrNone if successful, otherwise one of the other system-wide error codes.
 
  1050 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
  1052 @panic FBuf64  4 In _DEBUG mode - negative aNewFilePos value.
 
  1054 @see RFileBuf64::Write()
 
  1055 @see RFileBuf64::DoFileWrite()
 
  1056 @see RFileBuf64::DoFileWrite1()
 
  1057 @see RFileBuf64::Invariant()
 
  1059 TInt RFileBuf64::DoFileWrite2(TInt64 aNewFilePos)
 
  1061 	__FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
 
  1062 	__FILEBUF64_INVARIANT();
 
  1063 	TInt err = KErrNone;
 
  1066 		err = DoFileWrite();
 
  1070 		iFilePos = aNewFilePos;
 
  1074 	__FILEBUF64_INVARIANT();
 
  1079 This function discards the buffer content if the buffer contains cached read data.
 
  1080 The function is called from RFileBuf64::Write(), because if the buffer contains cached read data,
 
  1081 they cannot be mixed with the cached write data.
 
  1082 Reason: for example the buffer contains 8Kb cached read data from file offset 0.
 
  1083         The data write request is 10 bytes at offset 4000. The write data will be cached,
 
  1084         because the buffer contains data from from this file area: [0..8192].
 
  1085         The iDirty flag will be set. Later when RFileBuf64::Flush() is called, the whole
 
  1086         8Kb buffer will be written. There is nothing wrong with that, the file content will be consistent.
 
  1087         But from performance point of view: 8Kb written vs. 10 bytes written - that may badly impact the performance.
 
  1089 @see RFileBuf64::Write()
 
  1091 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
 
  1093 void RFileBuf64::DoDiscardBufferedReadData()
 
  1095 	__FILEBUF64_INVARIANT();
 
  1096 	if(!iDirty && iLength > 0)
 
  1100 		iNextReadFilePos = KNextReadFilePosNotSet;
 
  1101 		iNextReadFilePosHits = 0;
 
  1103 	__FILEBUF64_INVARIANT();
 
  1107 Sets the most appropriate buffer capacity based on the database page size.
 
  1108 The function does a lazy evaluation. The first time the function is called and 
 
  1109 aRwDataLength parameter is recognized to be a database or journal page size, the new (optimal)
 
  1110 buffer capacity is calculated and set. All next DoSetCapacity() calls will detect that the new
 
  1111 capacity is already set and will return KErrNone.
 
  1113 @param  aRwDataLength The length of the data being read or written.
 
  1114 @return KErrNone The new capacity was set successfully,
 
  1115         KErrNoMemory Out of memory.
 
  1117 TInt RFileBuf64::DoSetCapacity(TInt aRwDataLength)
 
  1119 	const TInt KMinPageCount = 4;//the buffer capacity should be at least (KMinPageCount * page size) 
 
  1120 	                             //but not less than the original capacity.
 
  1121 	const TInt KDefaultPageSize = 1024;//The journal header size is equal to 512 bytes, so it is not easy
 
  1122                                        //to detect the 512 bytes page size. 
 
  1124 	__FBUF64_ASSERT(aRwDataLength > 0, EFBufPanicRwDataLength);
 
  1125 	__FILEBUF64_INVARIANT();
 
  1128 		__FILEBUF64_INVARIANT();
 
  1131 	if((aRwDataLength & (aRwDataLength - 1)) != 0 || aRwDataLength < KDefaultPageSize)
 
  1133 		__FILEBUF64_INVARIANT();
 
  1136 	//Here: aRwDataLength is power of 2 and is bigger than the default db page size.
 
  1137 	//aRwDataLength is the size of the db page.
 
  1138 	const TInt pageSize = aRwDataLength;
 
  1139 	TInt cnt = iCapacity / pageSize;//how many pages can fit in the buffer now
 
  1140 	TInt pageCount = Max(cnt, KMinPageCount);//the number of pages that should fit in the new buffer
 
  1141 	TInt newBufCapacity = pageCount * pageSize;
 
  1142 	if(newBufCapacity != iCapacity)
 
  1144 		TUint8* newBase = static_cast <TUint8*> (User::ReAlloc(iBase, newBufCapacity));
 
  1147 			__FILEBUF64_INVARIANT();
 
  1148 			return KErrNoMemory;
 
  1151 		iCapacity = newBufCapacity;
 
  1152 		//Adjust the initial read-ahead size to be multiple of the page size.
 
  1153 		if((iReadAheadSize % pageSize) != 0)
 
  1155 			TInt q = iReadAheadSize / pageSize;
 
  1156 			iReadAheadSize = q != 0 ? pageSize * q : pageSize;
 
  1160 	__FILEBUF64_INVARIANT();
 
  1167 RFileBuf64 invariant. Called in _DEBUG mode at the beginning and before the end of every RFileBuf64 method
 
  1168 (except the init/destroy methods).
 
  1170 @panic FBuf64  11 In _DEBUG mode - null "this" pointer.
 
  1171 @panic FBuf64   1 In _DEBUG mode - negative iCapacity value.
 
  1172 @panic FBuf64   2 In _DEBUG mode - the buffer pointer is null (possible the buffer is not allocated or already destroyed).
 
  1173 @panic FBuf64   3 In _DEBUG mode - invalid iLength value (negative or bigger than iCapacity).
 
  1174 @panic FBuf64   4 In _DEBUG mode - negative iFilePos value.
 
  1175 @panic FBuf64   5 In _DEBUG mode - set but negative iFileSize value.
 
  1176 @panic FBuf64   6 In _DEBUG mode - null file handle (the RFile64 object is not created or already destroyed).
 
  1177 @panic FBuf64  13 In _DEBUG mode - set but negative iNextReadFilePos value.
 
  1178 @panic FBuf64  14 In _DEBUG mode - negative iNextReadFilePosHits value.
 
  1179 @panic FBuf64  15 In _DEBUG mode - iReadAheadSize is negative or is bigger than iCapacity.
 
  1181 void RFileBuf64::Invariant() const
 
  1183 	__FBUF64_ASSERT(this != 0, EFBufPanicNullThis);
 
  1184 	__FBUF64_ASSERT(iCapacity > 0, EFBufPanicCapacity);
 
  1185 	__FBUF64_ASSERT(iBase != 0, EFBufPanicNullBuf);
 
  1186 	__FBUF64_ASSERT(iLength >= 0 && iLength <= iCapacity, EFBufPanicBufLen);
 
  1187 	__FBUF64_ASSERT(iFilePos >= 0, EFBufPanicFilePos);
 
  1188 	__FBUF64_ASSERT(iFileSize == KFileSizeNotSet || iFileSize >= 0, EFBufPanicFileSize);
 
  1189 	__FBUF64_ASSERT(iFile.SubSessionHandle() != 0, EFBufPanicFileHandle);
 
  1190 	__FBUF64_ASSERT(iNextReadFilePos == KNextReadFilePosNotSet || iNextReadFilePos >= 0, EFBufPanicNextReadFilePos);
 
  1191 	__FBUF64_ASSERT(iNextReadFilePosHits >= 0, EFBufPanicNextReadFilePosHits);
 
  1192 	__FBUF64_ASSERT(iReadAheadSize > 0, EFBufPanicFileBlockSize);