os/persistentdata/persistentstorage/sql/OsLayer/FileBuf64.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 #include "FileBuf64.h"
    16 
    17 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
    18 ///////////////////////////        PROFILER       ////////////////////////////////////////////////////////////////////
    19 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
    20 
    21 #ifdef _SQLPROFILER
    22 
    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).
    24 
    25 #define PROFILE_READ(pos, amount, err) \
    26 	do \
    27 		{ \
    28 		if(TheOsCallTimeDetailedProfileEnabled) \
    29 			{ \
    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); \
    32 			} \
    33 		} while(0)
    34 	
    35 #define PROFILE_WRITE(pos, amount, err) \
    36 	do \
    37 		{ \
    38 		if(TheOsCallTimeDetailedProfileEnabled) \
    39 			{ \
    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); \
    42 			} \
    43 		} while(0)
    44 
    45 #define PROFILE_SIZE(size, err) \
    46 	do \
    47 		{ \
    48 		if(TheOsCallTimeDetailedProfileEnabled) \
    49 			{ \
    50 			++iFileSizeCount; \
    51 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Size¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSizeCount, size, err); \
    52 			} \
    53 		} while(0)
    54 
    55 #define PROFILE_SETSIZE(size, err) \
    56 	do \
    57 		{ \
    58 		if(TheOsCallTimeDetailedProfileEnabled) \
    59 			{ \
    60 			++iFileSetSizeCount; \
    61 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬SetSize¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSetSizeCount, size, err); \
    62 			} \
    63 		} while(0)
    64 
    65 #define PROFILE_FLUSH(err)	\
    66 	do \
    67 		{ \
    68 		if(TheOsCallTimeDetailedProfileEnabled) \
    69 			{ \
    70 			++iFileFlushCount; \
    71 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Flush¬%d¬¬¬¬%d\r\n"), (TUint32)this, iFileFlushCount, err); \
    72 			} \
    73 		} while(0)
    74 
    75 #define PROFILE_CREATE(fname, err) \
    76 	do \
    77 		{ \
    78 		if(TheOsCallTimeDetailedProfileEnabled) \
    79 			{ \
    80 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Create¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
    81 			} \
    82 		} while(0)
    83 
    84 #define PROFILE_OPEN(fname, err) \
    85 	do \
    86 		{ \
    87 		if(TheOsCallTimeDetailedProfileEnabled) \
    88 			{ \
    89 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Open¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
    90 			} \
    91 		} while(0)
    92 
    93 #define PROFILE_TEMP(fname, err) \
    94 	do \
    95 		{ \
    96 		if(TheOsCallTimeDetailedProfileEnabled) \
    97 			{ \
    98 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Temp¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
    99 			} \
   100 		} while(0)
   101 
   102 #define PROFILE_ADOPT(fname, err) \
   103 	do \
   104 		{ \
   105 		if(TheOsCallTimeDetailedProfileEnabled) \
   106 			{ \
   107 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Adopt¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
   108 			} \
   109 		} while(0)
   110 
   111 #define PROFILE_CLOSE() \
   112 	do \
   113 		{ \
   114 		if(TheOsCallTimeDetailedProfileEnabled) \
   115 			{ \
   116 			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Close¬¬¬¬¬¬\r\n"), (TUint32)this); \
   117 			} \
   118 		} while(0)
   119 
   120 //Resets the profiler counters
   121 void RFileBuf64::ProfilerReset()
   122 	{
   123 	iFileReadCount = 0; iFileReadAmount = 0; iFileWriteCount = 0; iFileWriteAmount = 0; iFileSizeCount = 0; iFileSetSizeCount = 0; iFileFlushCount = 0;
   124 	}
   125 
   126 #else
   127 
   128 #define PROFILE_READ(pos,amount, err)	void(0)
   129 #define PROFILE_WRITE(pos,amount, err)	void(0)
   130 
   131 #define PROFILE_SIZE(size, err)			void(0)
   132 #define PROFILE_SETSIZE(size, err)		void(0)
   133 #define PROFILE_FLUSH(err)				void(0)
   134 
   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)
   140 
   141 #endif//_SQLPROFILER
   142 
   143 /**
   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)
   147 @internalComponent
   148 */
   149 static const TInt KFileSizeNotSet = -1;
   150 
   151 /**
   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.
   154 @internalComponent
   155 */
   156 static const TInt KNextReadFilePosNotSet = -1;
   157 
   158 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   159 ///////////////////////////        ASSERTS & INVARIANT      //////////////////////////////////////////////////////////
   160 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   161 
   162 #ifdef _DEBUG
   163 
   164 #define __FILEBUF64_INVARIANT() Invariant()
   165 
   166 /**
   167 String literal used in _DEBUG mode for indicating that the reported panic happened inside the RFileBuf64 implementation.
   168 
   169 @see TFileBufPanic64
   170 @internalComponent
   171 */
   172 _LIT(KPanicCategory, "FBuf64");
   173 
   174 /**
   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.
   177 
   178 @see KPanicCategory
   179 @internalComponent
   180 */
   181 enum TFileBufPanic64
   182 	{
   183 	EFBufPanicCapacity = 1,				//1
   184 	EFBufPanicNullBuf,
   185 	EFBufPanicBufLen,
   186 	EFBufPanicFilePos,
   187 	EFBufPanicFileSize,					//5
   188 	EFBufPanicFileHandle,
   189 	EFBufPanicFsHandle,
   190 	EFBufPanicMsgHandle,
   191 	EFBufPanicMsgIndex,
   192 	EFBufPanicFileNameLen,				//10
   193 	EFBufPanicNullThis,
   194 	EFBufPanicDirty,
   195 	EFBufPanicNextReadFilePos,
   196 	EFBufPanicNextReadFilePosHits,
   197 	EFBufPanicFileBlockSize,			//15
   198 	EFBufPanicRwDataLength,
   199 	};
   200 
   201 /**
   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
   206 after that.
   207 
   208 @see TFileBufPanic64
   209 @see KPanicCategory
   210 @internalComponent
   211 */
   212 static void PanicFileBuf64(TInt aLine, TFileBufPanic64 aPanicCode)
   213 	{
   214 	TTime time;
   215 	time.HomeTime();
   216 	TDateTime dt = time.DateTime();
   217 	TBuf<16> tbuf;
   218 	tbuf.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
   219 	
   220 	TBuf<64> buf;
   221 	_LIT(KFormat,"**%S:RFileBuf64 panic %d, at line(%d)");
   222 	buf.Format(KFormat, &tbuf, aPanicCode, aLine);
   223 	RDebug::Print(buf);
   224 	User::Panic(KPanicCategory, aPanicCode);
   225 	}
   226 
   227 /**
   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.
   230 
   231 @see TFileBufPanic64
   232 @see KPanicCategory
   233 @see PanicFileBuf64()
   234 @internalComponent
   235 */
   236 #define __FBUF64_ASSERT(expr, panicCode)	(void)(!(expr) ? ::PanicFileBuf64(__LINE__, panicCode) : void(0))
   237 
   238 #else //_DEBUG
   239 
   240 #define __FILEBUF64_INVARIANT() void(0)
   241 
   242 #define __FBUF64_ASSERT(expr, panicCode) 	void(0)
   243 
   244 #endif//_DEBUG
   245 
   246 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   247 ///////////////////////////        RFileBuf64    /////////////////////////////////////////////////////////////////////
   248 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   249 	
   250 /**
   251 Initializes RFileBuf64 data members with their default values.
   252 
   253 @param aMinCapacity Minimal file buffer size (capacity) in bytes.
   254 
   255 @panic FBuf64 1 In _DEBUG mode - aMinCapacity is 0 or negative.
   256 */
   257 RFileBuf64::RFileBuf64(TInt aMinCapacity) :
   258 	iCapacity(aMinCapacity),
   259 	iBase(NULL),
   260 	iReadAheadSize(RFileBuf64::KDefaultReadAheadSize),
   261 	iOptimized(EFalse)
   262 	{
   263 	__FBUF64_ASSERT(aMinCapacity > 0, EFBufPanicCapacity);
   264 	}
   265 
   266 /**
   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.
   270 
   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
   274                  the session path.
   275 @param aFileMode The mode in which the file is opened. See TFileMode for details.
   276 
   277 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   278 
   279 @see TFileMode
   280 @see RFile64::Create()
   281 
   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).
   284 */
   285 TInt RFileBuf64::Create(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
   286 	{
   287 	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
   288 	__FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
   289 	
   290 	TInt err = DoPreInit();
   291 	if(err == KErrNone)
   292 	    {
   293 	    err = iFile.Create(aFs, aFileName, aFileMode);
   294 	    }
   295 	PROFILE_CREATE(aFileName, err);
   296 	return DoPostInit(err);
   297 	}
   298 
   299 /**
   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.
   302 
   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
   306                  the session path.
   307 @param aFileMode The mode in which the file is opened. See TFileMode for details.
   308 
   309 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   310 
   311 @see TFileMode
   312 @see RFile64::Open()
   313 
   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).
   316 */
   317 TInt RFileBuf64::Open(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
   318 	{
   319 	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
   320 	__FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
   321 	
   322     TInt err = DoPreInit();
   323     if(err == KErrNone)
   324         {
   325         err = iFile.Open(aFs, aFileName, aFileMode);
   326         }
   327 	PROFILE_OPEN(aFileName, err);
   328     return DoPostInit(err);
   329 	}
   330 
   331 /**
   332 Initializes the RFileBuf64 object and creates and opens a temporary file with unique name that will be accessed through 
   333 RFileBuf64 public interface.
   334 
   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
   339                  specified by aPath.
   340 @param aFileMode The mode in which the file is opened. The access mode is
   341                  automatically set to EFileWrite. See TFileMode for details.
   342 
   343 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   344 
   345 @see TFileMode
   346 @see RFile64::Temp()
   347 
   348 @panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
   349 */
   350 TInt RFileBuf64::Temp(RFs& aFs, const TDesC& aPath, TFileName& aFileName, TUint aFileMode)
   351 	{
   352 	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
   353 	
   354     TInt err = DoPreInit();
   355     if(err == KErrNone)
   356         {
   357         err = iFile.Temp(aFs, aPath, aFileName, aFileMode);
   358         }
   359 	PROFILE_TEMP(aFileName, err);
   360     return DoPostInit(err);
   361 	}
   362 
   363 /**
   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().
   368 
   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
   374             
   375 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   376 
   377 @see TFileMode
   378 @see RFile64::AdoptFromClient()
   379 @see KMaxMessageArguments
   380 
   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)
   384 */
   385 TInt RFileBuf64::AdoptFromClient(const RMessage2& aMsg, TInt aFsIndex, TInt aFileIndex)
   386 	{
   387 	__FBUF64_ASSERT(aMsg.Handle() != 0, EFBufPanicMsgHandle);
   388 	__FBUF64_ASSERT(aFsIndex >= 0 && aFsIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
   389 	__FBUF64_ASSERT(aFileIndex >= 0 && aFileIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
   390 	
   391     TInt err = DoPreInit();
   392     if(err == KErrNone)
   393         {
   394         err = iFile.AdoptFromClient(aMsg, aFsIndex, aFileIndex);
   395         }
   396 	PROFILE_ADOPT(KNullDesC, err);
   397     return DoPostInit(err);
   398 	}
   399 
   400 /**
   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.
   405 
   406 @see RFileBuf64::Flush()
   407 */
   408 void RFileBuf64::Close()
   409 	{
   410 	if(iBase != 0 && iFile.SubSessionHandle() != 0)
   411 		{
   412 		(void)DoFileWrite2();
   413 		}
   414 	iFile.Close();
   415 	User::Free(iBase);
   416 	iBase = 0;
   417 	PROFILE_CLOSE();
   418 	}
   419 
   420 /**
   421 Calculates and sets optimal read-ahead buffer size.
   422 aBlockSize and aReadRecBufSize values are retrieved by the caller from the file system.
   423 
   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. 
   431 
   432 
   433 @param aBlockSize The size of a file block in bytes
   434 @param aReadRecBufSize The recommended buffer size for optimised reading performance
   435 
   436 @return The new read-ahead value
   437 
   438 @see TVolumeIOParamInfo
   439 */
   440 TInt RFileBuf64::SetReadAheadSize(TInt aBlockSize, TInt aReadRecBufSize)
   441 	{
   442 	__FILEBUF64_INVARIANT();
   443 	if((aReadRecBufSize & (aReadRecBufSize - 1)) == 0 && aReadRecBufSize > RFileBuf64::KDefaultReadAheadSize)
   444 		{
   445 		iReadAheadSize = aReadRecBufSize > iCapacity ? iCapacity : aReadRecBufSize;
   446 		}
   447 	else if((aBlockSize & (aBlockSize - 1)) == 0 && aBlockSize > RFileBuf64::KDefaultReadAheadSize)
   448 		{
   449 		iReadAheadSize = aBlockSize > iCapacity ? iCapacity : aBlockSize;
   450 		}
   451 	__FILEBUF64_INVARIANT();
   452 	return iReadAheadSize;
   453 	}
   454 
   455 /**
   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.
   458 
   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
   465             bytes read.
   466             
   467 @return KErrNone if successful, otherwise one of the other system-wide error  codes.
   468 
   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.
   471 
   472 @see RFileBuf64::Invariant()
   473 */
   474 TInt RFileBuf64::Read(TInt64 aFilePos, TDes8& aDes)
   475 	{
   476 	__FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
   477 	__FILEBUF64_INVARIANT();
   478 	aDes.SetLength(0);
   479 	//0. The output buffer max len is 0
   480 	if(aDes.MaxLength() == 0)
   481 		{
   482 		__FILEBUF64_INVARIANT();
   483 		return KErrNone;	
   484 		}
   485 	//1. Initialize the "iFileSize" if it is not initialized yet
   486 	TInt err = DoFileSize();
   487 	if(err != KErrNone)
   488 		{
   489 		__FILEBUF64_INVARIANT();
   490 		return err;	
   491 		}
   492 	//2. Optimize the buffer capacity
   493 	TInt len = aDes.MaxLength();
   494 	if((err = DoSetCapacity(len)) != KErrNone)
   495 		{
   496 		return err;
   497 		}
   498 	//3. Too big "read" request - read directly from the file
   499 	if(len > iCapacity)
   500 		{
   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);
   504 			}
   505 		if(err == KErrNone)
   506 			{
   507 			err = iFile.Read(aFilePos, aDes);
   508 			PROFILE_READ(aFilePos, aDes.Size(), err);
   509 			}
   510 		__FILEBUF64_INVARIANT();
   511 		return err;
   512 		}
   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)
   516 		{
   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))
   519 			{
   520 			TInt blocklen = Min(len, (iFilePos + iLength - aFilePos));
   521 			outptr = Mem::Copy(outptr, iBase + (aFilePos - iFilePos), blocklen);
   522 			len -= blocklen;
   523 			aFilePos += blocklen;
   524 			}
   525 		//2. Perform a read-ahead operation
   526 		else
   527 			{
   528 			//Write the pending data if the iDirty flag is set, otherwise preserve the buffer content.
   529 			err = DoFileWrite1(aFilePos);
   530 			if(err != KErrNone)
   531 				{
   532 				break;	
   533 				}
   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);
   540 				if(err == KErrNone)
   541 					{
   542 					iNextReadFilePos = aFilePos + len;
   543 					len -= ptr2.Length();
   544 					}
   545 				break;
   546 				}
   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)
   550 				{
   551 				++iNextReadFilePosHits;
   552 				}
   553 			TInt maxReadAhead = iReadAheadSize * (1 << iNextReadFilePosHits);
   554 			TInt align = (aFilePos + len + maxReadAhead) & (iReadAheadSize - 1);
   555 			TInt readahead = maxReadAhead - align;
   556 			if(readahead < 0)
   557 				{
   558 				// if read-ahead doesn't cross block boundary do it all
   559 				readahead = maxReadAhead;	
   560 				}
   561 			TPtr8 ptr(iBase, Min(iCapacity, (len + readahead)));
   562 			err = iFile.Read(aFilePos, ptr);
   563 			PROFILE_READ(aFilePos, ptr.Size(), err);
   564 			if(err == KErrNone)
   565 				{
   566 				iFilePos = aFilePos;
   567 				iLength = ptr.Length();	
   568 				iNextReadFilePos = iFilePos + iLength;
   569 				if(iLength == 0)
   570 					{
   571 					break;	
   572 					}
   573 				}
   574 			else
   575 				{
   576 				DoDiscard();	
   577 				}
   578 			}
   579 		}
   580 	aDes.SetLength(aDes.MaxLength() - len);
   581 	__FILEBUF64_INVARIANT();
   582 	return err;
   583 	}
   584 	
   585 /**
   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.
   588 
   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.
   593             
   594 @param aData The descriptor from which binary data is written. The function writes 
   595              the entire contents of aData to the file.
   596 
   597 @return KErrNone if successful, otherwise one of the other system-wide error  codes.
   598 
   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.
   601 
   602 @see RFileBuf64::Invariant()
   603 */
   604 TInt RFileBuf64::Write(TInt64 aFilePos, const TDesC8& aData)
   605 	{
   606 	__FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
   607 	__FILEBUF64_INVARIANT();
   608 	if(aData.Length() == 0)
   609 		{
   610 		__FILEBUF64_INVARIANT();
   611 		return KErrNone;	
   612 		}
   613 	TInt err = DoFileSize();
   614 	if(err != KErrNone)
   615 		{
   616 		__FILEBUF64_INVARIANT();
   617 		return err;	
   618 		}
   619 	if((err = DoSetCapacity(aData.Length())) != KErrNone)
   620 		{
   621 		return err;
   622 		}
   623 	DoDiscardBufferedReadData();
   624 	const TUint8* data = aData.Ptr();
   625 	for(TInt len = aData.Length(); len > 0 && err == KErrNone;)
   626 		{
   627 		//1. The new write pos is before the buffered file pos
   628 		if(aFilePos < iFilePos)
   629 			{
   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))
   633 				{
   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
   640 				iDirty = ETrue;	
   641 				}
   642 			else
   643 			//The "aFilePos" is too far before the "iFilePos". Write the buffer and associate the new pos with the buffer
   644 				{
   645 				err = DoFileWrite2(aFilePos);
   646 				}
   647 			}
   648 		//2. The new write pos is after the associated file pos + the data length.
   649 		else if(aFilePos > (iFilePos + iLength))
   650 			{
   651 			if(aFilePos > iFileSize)											//Beyond the end of the file
   652 				{
   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));
   658 					iDirty = ETrue;	
   659 					}
   660 				else									
   661 				//Beyond the end of the file and not in the buffer - write the buffer to the file.
   662 					{
   663                     err = DoFileWrite2(aFilePos);
   664 					}
   665 				}
   666 			else										
   667 			//Within the file, not in the buffer - write the buffer and associate the new file pos with the buffer
   668 				{
   669 				err = DoFileWrite2(aFilePos);
   670 				}
   671 			}
   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
   675 		else
   676 			{
   677             if (iFilePos+iCapacity == aFilePos)	//The buffer is full. The new position to write is the end of the buffer.
   678                 {
   679                 err = DoFileWrite2(aFilePos);
   680                 }
   681             if(err == KErrNone)
   682                 {
   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));
   687                 len -= amount;
   688                 data += amount;
   689                 aFilePos += amount;
   690                 iDirty = ETrue;	
   691                 }
   692            }
   693 		}
   694 	__FILEBUF64_INVARIANT();
   695 	return err;
   696 	}
   697 	
   698 /**
   699 Gets the current file size.
   700 
   701 @param aFileSize On return, the size of the file in bytes.
   702 
   703 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   704 
   705 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
   706 
   707 @see RFileBuf64::Invariant()
   708 */
   709 TInt RFileBuf64::Size(TInt64& aFileSize)
   710 	{
   711 	__FILEBUF64_INVARIANT();
   712 	TInt err = DoFileSize();
   713 	if(err == KErrNone)
   714 		{
   715 		aFileSize = iFileSize;
   716 		}
   717 	__FILEBUF64_INVARIANT();
   718 	return err;
   719 	}
   720 
   721 /**
   722 Sets the file size.
   723 
   724 If the size of the file is reduced, data may be lost from the end of the file.
   725 
   726 Note:
   727 
   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
   731    the end of file. 
   732 
   733 2. If the file was not opened for writing, an error is returned.
   734 
   735 @param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
   736 
   737 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   738 
   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.
   741 
   742 @see RFileBuf64::Invariant()
   743 */
   744 TInt RFileBuf64::SetSize(TInt64 aFileSize)
   745 	{
   746 	__FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
   747 	__FILEBUF64_INVARIANT();
   748 	return DoSetFileSize(aFileSize);
   749 	}
   750 
   751 /**
   752 Writes the pending data and then flushes the file.
   753 
   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.
   758 
   759 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   760 
   761 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
   762 
   763 @see RFileBuf64::Invariant()
   764 */
   765 TInt RFileBuf64::Flush()
   766 	{
   767 	__FILEBUF64_INVARIANT();
   768 	return DoFileFlush();
   769 	}
   770 
   771 /**
   772 Gets information about the drive on which this file resides.
   773  
   774 @param aDriveNumber On return, the drive number.
   775 
   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.
   779 
   780 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   781 
   782 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
   783 
   784 @see RFileBuf64::Invariant()
   785 */
   786 TInt RFileBuf64::Drive(TInt& aDriveNumber, TDriveInfo& aDriveInfo) const
   787 	{
   788 	__FILEBUF64_INVARIANT();
   789 	return iFile.Drive(aDriveNumber, aDriveInfo);
   790 	}
   791 
   792 /**
   793 Initializes RFileBuf64 data members with their initial values.   
   794 Allocates memory for the file buffer. 
   795  
   796 @return KErrNone if successful, 
   797         KErrNoMemory out of memory;
   798 */
   799 TInt RFileBuf64::DoPreInit()
   800     {
   801     DoDiscard();
   802     iReadAheadSize = RFileBuf64::KDefaultReadAheadSize;
   803     iBase = static_cast <TUint8*> (User::Alloc(iCapacity));
   804     return iBase ? KErrNone : KErrNoMemory;   
   805     }
   806 
   807 /**
   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. 
   811 
   812 @param aInitErr The result of the performed before the call RFileBuf64 initialization.
   813  
   814 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   815 */
   816 TInt RFileBuf64::DoPostInit(TInt aInitErr)
   817     {
   818     if(aInitErr != KErrNone)
   819         {
   820         User::Free(iBase);
   821         iBase = 0;
   822         }
   823     return aInitErr;
   824     }
   825 
   826 /**
   827 Discards the content of the RFileBuf64 object returning it to the state as if it has just been created. 
   828 */
   829 void RFileBuf64::DoDiscard()
   830 	{
   831 	iLength = 0;
   832 	iFilePos = 0;
   833 	iFileSize = KFileSizeNotSet;
   834 	iDirty = EFalse;
   835 	iNextReadFilePos = KNextReadFilePosNotSet;
   836 	iNextReadFilePosHits = 0;
   837 	}
   838 
   839 /**
   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.
   843 
   844 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   845 
   846 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
   847 
   848 @see RFileBuf64::Invariant()
   849 */
   850 TInt RFileBuf64::DoFileSize()
   851 	{
   852 	__FILEBUF64_INVARIANT();
   853 	if(iFileSize != KFileSizeNotSet)
   854 		{
   855 		__FILEBUF64_INVARIANT();
   856 		return KErrNone;
   857 		}
   858 	TInt err = iFile.Size(iFileSize);
   859 	PROFILE_SIZE(iFileSize, err);
   860 	if(err != KErrNone)
   861 		{
   862 		DoDiscard();
   863 		}
   864 	else
   865 	    {
   866         iRealFileSize = iFileSize;
   867 	    }
   868 	__FILEBUF64_INVARIANT();
   869 	return err;
   870 	}
   871 
   872 /**
   873 Sets the file size.
   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. 
   876 
   877 @param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
   878 
   879 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   880 
   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.
   883 
   884 @see RFileBuf64::Invariant()
   885 */
   886 TInt RFileBuf64::DoSetFileSize(TInt64 aFileSize)
   887 	{
   888 	__FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
   889 	__FILEBUF64_INVARIANT();
   890 	if(aFileSize < iFilePos)
   891 		{
   892 		iDirty = EFalse;
   893 		iLength = 0;	
   894 		}
   895 	//If the new file size is "in" the buffer then change the "iLength"
   896 	else if(aFileSize < (iFilePos + iLength))
   897 		{
   898 		iLength = aFileSize - iFilePos;
   899 		}
   900 	TInt err = iFile.SetSize(aFileSize);
   901 	PROFILE_SETSIZE(aFileSize, err);
   902 	if(err != KErrNone)
   903 		{
   904 		DoDiscard();
   905 		}
   906 	else
   907 		{
   908 		iFileSize = aFileSize;
   909 		iRealFileSize = aFileSize;
   910 		}
   911 	__FILEBUF64_INVARIANT();
   912 	return err;
   913 	}
   914 
   915 /**
   916 Writes the pending data and flushes the file.
   917 
   918 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   919 
   920 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
   921 
   922 @see RFileBuf64::Invariant()
   923 */
   924 TInt RFileBuf64::DoFileFlush()
   925 	{
   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.
   928 	if(err != KErrNone)
   929 		{
   930 		__FILEBUF64_INVARIANT();
   931 		return err;	
   932 		}
   933 	err = iFile.Flush();
   934 	PROFILE_FLUSH(err);
   935 	if(err != KErrNone)
   936 		{
   937 		DoDiscard();
   938 		}
   939 	iLength = 0;
   940 	__FILEBUF64_INVARIANT();
   941 	return err;
   942 	}
   943 
   944 /**
   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.
   948 
   949 @return KErrNone if successful, otherwise one of the other system-wide error codes.
   950 
   951 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
   952 
   953 @see RFileBuf64::DoFileWrite1()
   954 @see RFileBuf64::DoFileWrite2()
   955 @see RFileBuf64::Invariant()
   956 */
   957 TInt RFileBuf64::DoFileWrite()
   958 	{
   959 	__FILEBUF64_INVARIANT();
   960 	if(iLength == 0)
   961 		{
   962 		__FILEBUF64_INVARIANT();
   963 		return KErrNone;	
   964 		}
   965 	TPtrC8 data(iBase, iLength);
   966 	TInt err = KErrNone;
   967 	if(iFilePos > iRealFileSize )
   968 	    {
   969         err = DoSetFileSize(iFileSize);
   970  	    }
   971 	if(err == KErrNone)
   972 	    {
   973         err = iFile.Write(iFilePos, data);
   974 	    }
   975 	PROFILE_WRITE(iFilePos, iLength, err);
   976 	if(err == KErrNone)
   977 		{
   978 		iRealFileSize = iFileSize;
   979 		}
   980 	else
   981 		{
   982 		DoDiscard();
   983 		}
   984 	__FILEBUF64_INVARIANT();
   985 	return err;
   986 	}
   987 
   988 /**
   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.
   997 
   998 @param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
   999 				   the aNewFilePos value.
  1000 
  1001 @return KErrNone if successful, otherwise one of the other system-wide error codes.
  1002 
  1003 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
  1004 				   
  1005 @panic FBuf64  4 In _DEBUG mode - negative aNewFilePos value.
  1006 
  1007 @see RFileBuf64::Read()
  1008 @see RFileBuf64::DoFileWrite()
  1009 @see RFileBuf64::DoFileWrite2()
  1010 @see RFileBuf64::Invariant()
  1011 */
  1012 TInt RFileBuf64::DoFileWrite1(TInt64 aNewFilePos)
  1013 	{
  1014 	__FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
  1015 	__FILEBUF64_INVARIANT();
  1016 	TInt err = KErrNone;
  1017 	if(iDirty)
  1018 		{
  1019 		err = DoFileWrite();
  1020 		if(err == KErrNone)	
  1021 			{
  1022 			iFilePos = aNewFilePos;
  1023 			iLength = 0;
  1024 			}
  1025 		}
  1026 	iDirty = EFalse;
  1027 	__FILEBUF64_INVARIANT();
  1028 	return err;		
  1029 	}
  1030 
  1031 /*
  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;
  1044 
  1045 @param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
  1046 				   the aNewFilePos value.
  1047 
  1048 @return KErrNone if successful, otherwise one of the other system-wide error codes.
  1049 
  1050 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
  1051 				   
  1052 @panic FBuf64  4 In _DEBUG mode - negative aNewFilePos value.
  1053 
  1054 @see RFileBuf64::Write()
  1055 @see RFileBuf64::DoFileWrite()
  1056 @see RFileBuf64::DoFileWrite1()
  1057 @see RFileBuf64::Invariant()
  1058 */
  1059 TInt RFileBuf64::DoFileWrite2(TInt64 aNewFilePos)
  1060 	{
  1061 	__FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
  1062 	__FILEBUF64_INVARIANT();
  1063 	TInt err = KErrNone;
  1064 	if(iDirty)
  1065 		{
  1066 		err = DoFileWrite();
  1067 		}
  1068 	if(err == KErrNone)	
  1069 		{
  1070 		iFilePos = aNewFilePos;
  1071 		iLength = 0;
  1072 		}
  1073 	iDirty = EFalse;
  1074 	__FILEBUF64_INVARIANT();
  1075 	return err;
  1076 	}
  1077 
  1078 /**
  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.
  1088 
  1089 @see RFileBuf64::Write()
  1090 
  1091 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
  1092 */
  1093 void RFileBuf64::DoDiscardBufferedReadData()
  1094 	{
  1095 	__FILEBUF64_INVARIANT();
  1096 	if(!iDirty && iLength > 0)
  1097 		{
  1098 		iLength = 0;
  1099 		iFilePos = 0;
  1100 		iNextReadFilePos = KNextReadFilePosNotSet;
  1101 		iNextReadFilePosHits = 0;
  1102 		}
  1103 	__FILEBUF64_INVARIANT();
  1104 	}
  1105 
  1106 /**
  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.
  1112 
  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.
  1116 */
  1117 TInt RFileBuf64::DoSetCapacity(TInt aRwDataLength)
  1118 	{
  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. 
  1123 	
  1124 	__FBUF64_ASSERT(aRwDataLength > 0, EFBufPanicRwDataLength);
  1125 	__FILEBUF64_INVARIANT();
  1126 	if(iOptimized)
  1127 		{
  1128 		__FILEBUF64_INVARIANT();
  1129 		return KErrNone;
  1130 		}
  1131 	if((aRwDataLength & (aRwDataLength - 1)) != 0 || aRwDataLength < KDefaultPageSize)
  1132 		{
  1133 		__FILEBUF64_INVARIANT();
  1134 		return KErrNone;
  1135 		}
  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)
  1143 		{
  1144 		TUint8* newBase = static_cast <TUint8*> (User::ReAlloc(iBase, newBufCapacity));
  1145 		if(!newBase)
  1146 			{
  1147 			__FILEBUF64_INVARIANT();
  1148 			return KErrNoMemory;
  1149 			}
  1150 		iBase = newBase;
  1151 		iCapacity = newBufCapacity;
  1152 		//Adjust the initial read-ahead size to be multiple of the page size.
  1153 		if((iReadAheadSize % pageSize) != 0)
  1154 			{
  1155 			TInt q = iReadAheadSize / pageSize;
  1156 			iReadAheadSize = q != 0 ? pageSize * q : pageSize;
  1157 			}
  1158 		}
  1159 	iOptimized = ETrue;
  1160 	__FILEBUF64_INVARIANT();
  1161 	return KErrNone;
  1162 	}
  1163 
  1164 #ifdef _DEBUG
  1165 
  1166 /**
  1167 RFileBuf64 invariant. Called in _DEBUG mode at the beginning and before the end of every RFileBuf64 method
  1168 (except the init/destroy methods).
  1169 
  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.
  1180 */
  1181 void RFileBuf64::Invariant() const
  1182 	{
  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);
  1193 	}
  1194 	
  1195 #endif