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.
sl@0
     1
// Copyright (c) 2008-2010 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
#include "FileBuf64.h"
sl@0
    16
sl@0
    17
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    18
///////////////////////////        PROFILER       ////////////////////////////////////////////////////////////////////
sl@0
    19
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
    20
sl@0
    21
#ifdef _SQLPROFILER
sl@0
    22
sl@0
    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).
sl@0
    24
sl@0
    25
#define PROFILE_READ(pos, amount, err) \
sl@0
    26
	do \
sl@0
    27
		{ \
sl@0
    28
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    29
			{ \
sl@0
    30
			++iFileReadCount; iFileReadAmount += (amount); \
sl@0
    31
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Read¬%d¬%ld¬%d¬%ld¬%d\r\n"), (TUint32)this, iFileReadCount, pos, amount, iFileReadAmount, err); \
sl@0
    32
			} \
sl@0
    33
		} while(0)
sl@0
    34
	
sl@0
    35
#define PROFILE_WRITE(pos, amount, err) \
sl@0
    36
	do \
sl@0
    37
		{ \
sl@0
    38
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    39
			{ \
sl@0
    40
			++iFileWriteCount; iFileWriteAmount += (amount); \
sl@0
    41
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Write¬%d¬%ld¬%d¬%ld¬%d\r\n"), (TUint32)this, iFileWriteCount, pos, amount, iFileWriteAmount, err); \
sl@0
    42
			} \
sl@0
    43
		} while(0)
sl@0
    44
sl@0
    45
#define PROFILE_SIZE(size, err) \
sl@0
    46
	do \
sl@0
    47
		{ \
sl@0
    48
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    49
			{ \
sl@0
    50
			++iFileSizeCount; \
sl@0
    51
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Size¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSizeCount, size, err); \
sl@0
    52
			} \
sl@0
    53
		} while(0)
sl@0
    54
sl@0
    55
#define PROFILE_SETSIZE(size, err) \
sl@0
    56
	do \
sl@0
    57
		{ \
sl@0
    58
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    59
			{ \
sl@0
    60
			++iFileSetSizeCount; \
sl@0
    61
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬SetSize¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSetSizeCount, size, err); \
sl@0
    62
			} \
sl@0
    63
		} while(0)
sl@0
    64
sl@0
    65
#define PROFILE_FLUSH(err)	\
sl@0
    66
	do \
sl@0
    67
		{ \
sl@0
    68
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    69
			{ \
sl@0
    70
			++iFileFlushCount; \
sl@0
    71
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Flush¬%d¬¬¬¬%d\r\n"), (TUint32)this, iFileFlushCount, err); \
sl@0
    72
			} \
sl@0
    73
		} while(0)
sl@0
    74
sl@0
    75
#define PROFILE_CREATE(fname, err) \
sl@0
    76
	do \
sl@0
    77
		{ \
sl@0
    78
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    79
			{ \
sl@0
    80
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Create¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
sl@0
    81
			} \
sl@0
    82
		} while(0)
sl@0
    83
sl@0
    84
#define PROFILE_OPEN(fname, err) \
sl@0
    85
	do \
sl@0
    86
		{ \
sl@0
    87
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    88
			{ \
sl@0
    89
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Open¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
sl@0
    90
			} \
sl@0
    91
		} while(0)
sl@0
    92
sl@0
    93
#define PROFILE_TEMP(fname, err) \
sl@0
    94
	do \
sl@0
    95
		{ \
sl@0
    96
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
    97
			{ \
sl@0
    98
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Temp¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
sl@0
    99
			} \
sl@0
   100
		} while(0)
sl@0
   101
sl@0
   102
#define PROFILE_ADOPT(fname, err) \
sl@0
   103
	do \
sl@0
   104
		{ \
sl@0
   105
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
   106
			{ \
sl@0
   107
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Adopt¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
sl@0
   108
			} \
sl@0
   109
		} while(0)
sl@0
   110
sl@0
   111
#define PROFILE_CLOSE() \
sl@0
   112
	do \
sl@0
   113
		{ \
sl@0
   114
		if(TheOsCallTimeDetailedProfileEnabled) \
sl@0
   115
			{ \
sl@0
   116
			RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Close¬¬¬¬¬¬\r\n"), (TUint32)this); \
sl@0
   117
			} \
sl@0
   118
		} while(0)
sl@0
   119
sl@0
   120
//Resets the profiler counters
sl@0
   121
void RFileBuf64::ProfilerReset()
sl@0
   122
	{
sl@0
   123
	iFileReadCount = 0; iFileReadAmount = 0; iFileWriteCount = 0; iFileWriteAmount = 0; iFileSizeCount = 0; iFileSetSizeCount = 0; iFileFlushCount = 0;
sl@0
   124
	}
sl@0
   125
sl@0
   126
#else
sl@0
   127
sl@0
   128
#define PROFILE_READ(pos,amount, err)	void(0)
sl@0
   129
#define PROFILE_WRITE(pos,amount, err)	void(0)
sl@0
   130
sl@0
   131
#define PROFILE_SIZE(size, err)			void(0)
sl@0
   132
#define PROFILE_SETSIZE(size, err)		void(0)
sl@0
   133
#define PROFILE_FLUSH(err)				void(0)
sl@0
   134
sl@0
   135
#define PROFILE_CREATE(fname, err)		void(0)
sl@0
   136
#define PROFILE_OPEN(fname, err)		void(0)
sl@0
   137
#define PROFILE_TEMP(fname, err)		void(0)
sl@0
   138
#define PROFILE_ADOPT(fname, err)		void(0)
sl@0
   139
#define PROFILE_CLOSE()					void(0)
sl@0
   140
sl@0
   141
#endif//_SQLPROFILER
sl@0
   142
sl@0
   143
/**
sl@0
   144
This constant is used for initializing the RFileBuf64::iFileSize data member and means that
sl@0
   145
the iFileSize is not yet initialized with the real file size value. 
sl@0
   146
(RFileBuf64::iFileSize caches the file size value)
sl@0
   147
@internalComponent
sl@0
   148
*/
sl@0
   149
static const TInt KFileSizeNotSet = -1;
sl@0
   150
sl@0
   151
/**
sl@0
   152
This constant is used as a default initializer for the RFileBuf64::iNextReadFilePos data member,
sl@0
   153
indicating that the "guessed" file read offset is invalid and should not be used.
sl@0
   154
@internalComponent
sl@0
   155
*/
sl@0
   156
static const TInt KNextReadFilePosNotSet = -1;
sl@0
   157
sl@0
   158
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   159
///////////////////////////        ASSERTS & INVARIANT      //////////////////////////////////////////////////////////
sl@0
   160
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   161
sl@0
   162
#ifdef _DEBUG
sl@0
   163
sl@0
   164
#define __FILEBUF64_INVARIANT() Invariant()
sl@0
   165
sl@0
   166
/**
sl@0
   167
String literal used in _DEBUG mode for indicating that the reported panic happened inside the RFileBuf64 implementation.
sl@0
   168
sl@0
   169
@see TFileBufPanic64
sl@0
   170
@internalComponent
sl@0
   171
*/
sl@0
   172
_LIT(KPanicCategory, "FBuf64");
sl@0
   173
sl@0
   174
/**
sl@0
   175
Set of numeric constants used together with the KPanicCategory string literal in _DEBUG mode for providing more detailed
sl@0
   176
information about the reason of the panic.
sl@0
   177
sl@0
   178
@see KPanicCategory
sl@0
   179
@internalComponent
sl@0
   180
*/
sl@0
   181
enum TFileBufPanic64
sl@0
   182
	{
sl@0
   183
	EFBufPanicCapacity = 1,				//1
sl@0
   184
	EFBufPanicNullBuf,
sl@0
   185
	EFBufPanicBufLen,
sl@0
   186
	EFBufPanicFilePos,
sl@0
   187
	EFBufPanicFileSize,					//5
sl@0
   188
	EFBufPanicFileHandle,
sl@0
   189
	EFBufPanicFsHandle,
sl@0
   190
	EFBufPanicMsgHandle,
sl@0
   191
	EFBufPanicMsgIndex,
sl@0
   192
	EFBufPanicFileNameLen,				//10
sl@0
   193
	EFBufPanicNullThis,
sl@0
   194
	EFBufPanicDirty,
sl@0
   195
	EFBufPanicNextReadFilePos,
sl@0
   196
	EFBufPanicNextReadFilePosHits,
sl@0
   197
	EFBufPanicFileBlockSize,			//15
sl@0
   198
	EFBufPanicRwDataLength,
sl@0
   199
	};
sl@0
   200
sl@0
   201
/**
sl@0
   202
Helper function used in the implementation of the __FBUF64_ASSERT() macro.
sl@0
   203
In case if the expression in __FBUF64_ASSERT() macro evaluates to false, 
sl@0
   204
PanicFileBuf64() will use the supplied aLine and aPanicCode arguments together with the KPanicCategory string literal
sl@0
   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
sl@0
   206
after that.
sl@0
   207
sl@0
   208
@see TFileBufPanic64
sl@0
   209
@see KPanicCategory
sl@0
   210
@internalComponent
sl@0
   211
*/
sl@0
   212
static void PanicFileBuf64(TInt aLine, TFileBufPanic64 aPanicCode)
sl@0
   213
	{
sl@0
   214
	TTime time;
sl@0
   215
	time.HomeTime();
sl@0
   216
	TDateTime dt = time.DateTime();
sl@0
   217
	TBuf<16> tbuf;
sl@0
   218
	tbuf.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
sl@0
   219
	
sl@0
   220
	TBuf<64> buf;
sl@0
   221
	_LIT(KFormat,"**%S:RFileBuf64 panic %d, at line(%d)");
sl@0
   222
	buf.Format(KFormat, &tbuf, aPanicCode, aLine);
sl@0
   223
	RDebug::Print(buf);
sl@0
   224
	User::Panic(KPanicCategory, aPanicCode);
sl@0
   225
	}
sl@0
   226
sl@0
   227
/**
sl@0
   228
This macro should be used when there is a need to panic the client/server if "expr" condition is not satisfied.
sl@0
   229
Works in only in debug mode. In release mode evaluates to nothing.
sl@0
   230
sl@0
   231
@see TFileBufPanic64
sl@0
   232
@see KPanicCategory
sl@0
   233
@see PanicFileBuf64()
sl@0
   234
@internalComponent
sl@0
   235
*/
sl@0
   236
#define __FBUF64_ASSERT(expr, panicCode)	(void)(!(expr) ? ::PanicFileBuf64(__LINE__, panicCode) : void(0))
sl@0
   237
sl@0
   238
#else //_DEBUG
sl@0
   239
sl@0
   240
#define __FILEBUF64_INVARIANT() void(0)
sl@0
   241
sl@0
   242
#define __FBUF64_ASSERT(expr, panicCode) 	void(0)
sl@0
   243
sl@0
   244
#endif//_DEBUG
sl@0
   245
sl@0
   246
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   247
///////////////////////////        RFileBuf64    /////////////////////////////////////////////////////////////////////
sl@0
   248
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
sl@0
   249
	
sl@0
   250
/**
sl@0
   251
Initializes RFileBuf64 data members with their default values.
sl@0
   252
sl@0
   253
@param aMinCapacity Minimal file buffer size (capacity) in bytes.
sl@0
   254
sl@0
   255
@panic FBuf64 1 In _DEBUG mode - aMinCapacity is 0 or negative.
sl@0
   256
*/
sl@0
   257
RFileBuf64::RFileBuf64(TInt aMinCapacity) :
sl@0
   258
	iCapacity(aMinCapacity),
sl@0
   259
	iBase(NULL),
sl@0
   260
	iReadAheadSize(RFileBuf64::KDefaultReadAheadSize),
sl@0
   261
	iOptimized(EFalse)
sl@0
   262
	{
sl@0
   263
	__FBUF64_ASSERT(aMinCapacity > 0, EFBufPanicCapacity);
sl@0
   264
	}
sl@0
   265
sl@0
   266
/**
sl@0
   267
Initializes the RFileBuf64 object and creates and opens a new file that will be accessed through RFileBuf64 public interface.
sl@0
   268
If the file already exists, an error is returned.
sl@0
   269
If the resulting path does not exist, then the operation cannot proceed and the function returns an error code.
sl@0
   270
sl@0
   271
@param aFs       The file server session.
sl@0
   272
@param aFileName The name of the file. Any path components (i.e. drive letter
sl@0
   273
                 or directory), which are not specified, are taken from
sl@0
   274
                 the session path.
sl@0
   275
@param aFileMode The mode in which the file is opened. See TFileMode for details.
sl@0
   276
sl@0
   277
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   278
sl@0
   279
@see TFileMode
sl@0
   280
@see RFile64::Create()
sl@0
   281
sl@0
   282
@panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
sl@0
   283
@panic FBuf64 10 In _DEBUG mode - Invalid file name length (zero file name length).
sl@0
   284
*/
sl@0
   285
TInt RFileBuf64::Create(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
sl@0
   286
	{
sl@0
   287
	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
sl@0
   288
	__FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
sl@0
   289
	
sl@0
   290
	TInt err = DoPreInit();
sl@0
   291
	if(err == KErrNone)
sl@0
   292
	    {
sl@0
   293
	    err = iFile.Create(aFs, aFileName, aFileMode);
sl@0
   294
	    }
sl@0
   295
	PROFILE_CREATE(aFileName, err);
sl@0
   296
	return DoPostInit(err);
sl@0
   297
	}
sl@0
   298
sl@0
   299
/**
sl@0
   300
Initializes the RFileBuf64 object and opens an existing file that will be accessed through RFileBuf64 public interface.
sl@0
   301
If the file does not already exist, an error is returned.
sl@0
   302
sl@0
   303
@param aFs       The file server session.
sl@0
   304
@param aFileName The name of the file. Any path components (i.e. drive letter
sl@0
   305
                 or directory), which are not specified, are taken from
sl@0
   306
                 the session path.
sl@0
   307
@param aFileMode The mode in which the file is opened. See TFileMode for details.
sl@0
   308
sl@0
   309
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   310
sl@0
   311
@see TFileMode
sl@0
   312
@see RFile64::Open()
sl@0
   313
sl@0
   314
@panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
sl@0
   315
@panic FBuf64 10 In _DEBUG mode - Invalid file name length (zero file name length).
sl@0
   316
*/
sl@0
   317
TInt RFileBuf64::Open(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
sl@0
   318
	{
sl@0
   319
	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
sl@0
   320
	__FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
sl@0
   321
	
sl@0
   322
    TInt err = DoPreInit();
sl@0
   323
    if(err == KErrNone)
sl@0
   324
        {
sl@0
   325
        err = iFile.Open(aFs, aFileName, aFileMode);
sl@0
   326
        }
sl@0
   327
	PROFILE_OPEN(aFileName, err);
sl@0
   328
    return DoPostInit(err);
sl@0
   329
	}
sl@0
   330
sl@0
   331
/**
sl@0
   332
Initializes the RFileBuf64 object and creates and opens a temporary file with unique name that will be accessed through 
sl@0
   333
RFileBuf64 public interface.
sl@0
   334
sl@0
   335
@param aFs       The file server session.
sl@0
   336
@param aPath     The directory in which the file is created.
sl@0
   337
@param aFileName On return, contains the full path and file name of the file.
sl@0
   338
                 The filename is guaranteed to be unique within the directory
sl@0
   339
                 specified by aPath.
sl@0
   340
@param aFileMode The mode in which the file is opened. The access mode is
sl@0
   341
                 automatically set to EFileWrite. See TFileMode for details.
sl@0
   342
sl@0
   343
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   344
sl@0
   345
@see TFileMode
sl@0
   346
@see RFile64::Temp()
sl@0
   347
sl@0
   348
@panic FBuf64  7 In _DEBUG mode - Invalid aFs object (null file session handle).
sl@0
   349
*/
sl@0
   350
TInt RFileBuf64::Temp(RFs& aFs, const TDesC& aPath, TFileName& aFileName, TUint aFileMode)
sl@0
   351
	{
sl@0
   352
	__FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
sl@0
   353
	
sl@0
   354
    TInt err = DoPreInit();
sl@0
   355
    if(err == KErrNone)
sl@0
   356
        {
sl@0
   357
        err = iFile.Temp(aFs, aPath, aFileName, aFileMode);
sl@0
   358
        }
sl@0
   359
	PROFILE_TEMP(aFileName, err);
sl@0
   360
    return DoPostInit(err);
sl@0
   361
	}
sl@0
   362
sl@0
   363
/**
sl@0
   364
Initializes the RFileBuf64 object and creates and adopts an already open file from a client that will be accessed through 
sl@0
   365
RFileBuf64 public interface.
sl@0
   366
The client's RFs and RFile or RFile64 handles are contained in message slots within aMsg.
sl@0
   367
Assumes that the client's RFs and RFile or RFile64 handles have been sent to the server using TransferToServer().
sl@0
   368
sl@0
   369
@param	aMsg	  The message received from the client
sl@0
   370
@param	aFsIndex  The index that identifies the message slot 
sl@0
   371
				  of a file server session (RFs) handle
sl@0
   372
@param aFileIndex The index that identifies the message slot 
sl@0
   373
				  of the sub-session (RFile or RFile64) handle of the already opened file
sl@0
   374
            
sl@0
   375
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   376
sl@0
   377
@see TFileMode
sl@0
   378
@see RFile64::AdoptFromClient()
sl@0
   379
@see KMaxMessageArguments
sl@0
   380
sl@0
   381
@panic FBuf64  8 In _DEBUG mode - Invalid aMsg object (null message handle).
sl@0
   382
@panic FBuf64  9 In _DEBUG mode - Invalid file session handle message slot index or invalid file handle message slot index.
sl@0
   383
                 (Probably negative index or index bigger or equal to KMaxMessageArguments)
sl@0
   384
*/
sl@0
   385
TInt RFileBuf64::AdoptFromClient(const RMessage2& aMsg, TInt aFsIndex, TInt aFileIndex)
sl@0
   386
	{
sl@0
   387
	__FBUF64_ASSERT(aMsg.Handle() != 0, EFBufPanicMsgHandle);
sl@0
   388
	__FBUF64_ASSERT(aFsIndex >= 0 && aFsIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
sl@0
   389
	__FBUF64_ASSERT(aFileIndex >= 0 && aFileIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
sl@0
   390
	
sl@0
   391
    TInt err = DoPreInit();
sl@0
   392
    if(err == KErrNone)
sl@0
   393
        {
sl@0
   394
        err = iFile.AdoptFromClient(aMsg, aFsIndex, aFileIndex);
sl@0
   395
        }
sl@0
   396
	PROFILE_ADOPT(KNullDesC, err);
sl@0
   397
    return DoPostInit(err);
sl@0
   398
	}
sl@0
   399
sl@0
   400
/**
sl@0
   401
Writes to the file the pending data (if the buffer contains pending data), closes the file and releases
sl@0
   402
the RFileBuf64 resources. 
sl@0
   403
RFileBuf64::Flush() should be called before RFileBuf64::Close() to ensure that if there are pending data, they will
sl@0
   404
be written to the file and if the operation fails, the caller will be notified with an appropriate return error.
sl@0
   405
sl@0
   406
@see RFileBuf64::Flush()
sl@0
   407
*/
sl@0
   408
void RFileBuf64::Close()
sl@0
   409
	{
sl@0
   410
	if(iBase != 0 && iFile.SubSessionHandle() != 0)
sl@0
   411
		{
sl@0
   412
		(void)DoFileWrite2();
sl@0
   413
		}
sl@0
   414
	iFile.Close();
sl@0
   415
	User::Free(iBase);
sl@0
   416
	iBase = 0;
sl@0
   417
	PROFILE_CLOSE();
sl@0
   418
	}
sl@0
   419
sl@0
   420
/**
sl@0
   421
Calculates and sets optimal read-ahead buffer size.
sl@0
   422
aBlockSize and aReadRecBufSize values are retrieved by the caller from the file system.
sl@0
   423
sl@0
   424
Initialization rules:
sl@0
   425
Rule 1: If aReadRecBufSize is positive, bigger than the default read-ahead and 
sl@0
   426
        a power of two then the read-ahead value will be
sl@0
   427
        initialized with aReadRecBufSize (if aReadRecBufSize is less than the buffer capacity otherwise
sl@0
   428
        the buffer capacity will be used as a read-ahead value). 
sl@0
   429
Rule 2: If rule#1 is not applicable then the same checks, as in rule#1, are performed this time for aBlockSize.
sl@0
   430
        If aBlockSize passes the checks then it will be used as a read-ahead value. 
sl@0
   431
sl@0
   432
sl@0
   433
@param aBlockSize The size of a file block in bytes
sl@0
   434
@param aReadRecBufSize The recommended buffer size for optimised reading performance
sl@0
   435
sl@0
   436
@return The new read-ahead value
sl@0
   437
sl@0
   438
@see TVolumeIOParamInfo
sl@0
   439
*/
sl@0
   440
TInt RFileBuf64::SetReadAheadSize(TInt aBlockSize, TInt aReadRecBufSize)
sl@0
   441
	{
sl@0
   442
	__FILEBUF64_INVARIANT();
sl@0
   443
	if((aReadRecBufSize & (aReadRecBufSize - 1)) == 0 && aReadRecBufSize > RFileBuf64::KDefaultReadAheadSize)
sl@0
   444
		{
sl@0
   445
		iReadAheadSize = aReadRecBufSize > iCapacity ? iCapacity : aReadRecBufSize;
sl@0
   446
		}
sl@0
   447
	else if((aBlockSize & (aBlockSize - 1)) == 0 && aBlockSize > RFileBuf64::KDefaultReadAheadSize)
sl@0
   448
		{
sl@0
   449
		iReadAheadSize = aBlockSize > iCapacity ? iCapacity : aBlockSize;
sl@0
   450
		}
sl@0
   451
	__FILEBUF64_INVARIANT();
sl@0
   452
	return iReadAheadSize;
sl@0
   453
	}
sl@0
   454
sl@0
   455
/**
sl@0
   456
Reads from the file at the specified position (aFilePos).
sl@0
   457
If the data to be read is in the buffer, then the data will be taken from the buffer.
sl@0
   458
sl@0
   459
@param aFilePos Position of first byte to be read.  This is an offset from
sl@0
   460
            the start of the file. 
sl@0
   461
            If aPos is beyond the end of the file, the function returns
sl@0
   462
            a zero length descriptor.
sl@0
   463
@param aDes Descriptor into which binary data is read. Any existing contents 
sl@0
   464
            are overwritten. On return, its length is set to the number of
sl@0
   465
            bytes read.
sl@0
   466
            
sl@0
   467
@return KErrNone if successful, otherwise one of the other system-wide error  codes.
sl@0
   468
sl@0
   469
@panic FBuf64  4 In _DEBUG mode - negative aFilePos value.
sl@0
   470
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
   471
sl@0
   472
@see RFileBuf64::Invariant()
sl@0
   473
*/
sl@0
   474
TInt RFileBuf64::Read(TInt64 aFilePos, TDes8& aDes)
sl@0
   475
	{
sl@0
   476
	__FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
sl@0
   477
	__FILEBUF64_INVARIANT();
sl@0
   478
	aDes.SetLength(0);
sl@0
   479
	//0. The output buffer max len is 0
sl@0
   480
	if(aDes.MaxLength() == 0)
sl@0
   481
		{
sl@0
   482
		__FILEBUF64_INVARIANT();
sl@0
   483
		return KErrNone;	
sl@0
   484
		}
sl@0
   485
	//1. Initialize the "iFileSize" if it is not initialized yet
sl@0
   486
	TInt err = DoFileSize();
sl@0
   487
	if(err != KErrNone)
sl@0
   488
		{
sl@0
   489
		__FILEBUF64_INVARIANT();
sl@0
   490
		return err;	
sl@0
   491
		}
sl@0
   492
	//2. Optimize the buffer capacity
sl@0
   493
	TInt len = aDes.MaxLength();
sl@0
   494
	if((err = DoSetCapacity(len)) != KErrNone)
sl@0
   495
		{
sl@0
   496
		return err;
sl@0
   497
		}
sl@0
   498
	//3. Too big "read" request - read directly from the file
sl@0
   499
	if(len > iCapacity)
sl@0
   500
		{
sl@0
   501
		if((aFilePos + len) > iFilePos && aFilePos < (iFilePos + iLength))
sl@0
   502
			{//Write the pending data if the iDirty flag is set, otherwise preserve the buffer content.
sl@0
   503
			err = DoFileWrite1(aFilePos);
sl@0
   504
			}
sl@0
   505
		if(err == KErrNone)
sl@0
   506
			{
sl@0
   507
			err = iFile.Read(aFilePos, aDes);
sl@0
   508
			PROFILE_READ(aFilePos, aDes.Size(), err);
sl@0
   509
			}
sl@0
   510
		__FILEBUF64_INVARIANT();
sl@0
   511
		return err;
sl@0
   512
		}
sl@0
   513
	//4. The requested data size is smaller than the buffer capacity
sl@0
   514
	TUint8* outptr = const_cast <TUint8*> (aDes.Ptr());
sl@0
   515
	while(len > 0 && err == KErrNone && aFilePos < iFileSize)
sl@0
   516
		{
sl@0
   517
		//1. If part or all of the data is in the buffer - copy the data to the target location
sl@0
   518
		if(aFilePos >= iFilePos && aFilePos < (iFilePos + iLength))
sl@0
   519
			{
sl@0
   520
			TInt blocklen = Min(len, (iFilePos + iLength - aFilePos));
sl@0
   521
			outptr = Mem::Copy(outptr, iBase + (aFilePos - iFilePos), blocklen);
sl@0
   522
			len -= blocklen;
sl@0
   523
			aFilePos += blocklen;
sl@0
   524
			}
sl@0
   525
		//2. Perform a read-ahead operation
sl@0
   526
		else
sl@0
   527
			{
sl@0
   528
			//Write the pending data if the iDirty flag is set, otherwise preserve the buffer content.
sl@0
   529
			err = DoFileWrite1(aFilePos);
sl@0
   530
			if(err != KErrNone)
sl@0
   531
				{
sl@0
   532
				break;	
sl@0
   533
				}
sl@0
   534
			if(iNextReadFilePos != aFilePos)
sl@0
   535
				{//Guessed read ahead was wrong. Direct "file read" operation
sl@0
   536
				iNextReadFilePosHits = 0;
sl@0
   537
				TPtr8 ptr2(outptr, len);
sl@0
   538
				err = iFile.Read(aFilePos, ptr2);
sl@0
   539
				PROFILE_READ(aFilePos, ptr2.Size(), err);
sl@0
   540
				if(err == KErrNone)
sl@0
   541
					{
sl@0
   542
					iNextReadFilePos = aFilePos + len;
sl@0
   543
					len -= ptr2.Length();
sl@0
   544
					}
sl@0
   545
				break;
sl@0
   546
				}
sl@0
   547
			//The guessed from the previous "file read" operation file pos is correct. Start reading-ahead.
sl@0
   548
			const TInt KMaxReadFilePosHits = 4;//The max read-ahead buffer size can be up to 2^4 times the iReadAheadSize
sl@0
   549
			if(iNextReadFilePosHits < KMaxReadFilePosHits)
sl@0
   550
				{
sl@0
   551
				++iNextReadFilePosHits;
sl@0
   552
				}
sl@0
   553
			TInt maxReadAhead = iReadAheadSize * (1 << iNextReadFilePosHits);
sl@0
   554
			TInt align = (aFilePos + len + maxReadAhead) & (iReadAheadSize - 1);
sl@0
   555
			TInt readahead = maxReadAhead - align;
sl@0
   556
			if(readahead < 0)
sl@0
   557
				{
sl@0
   558
				// if read-ahead doesn't cross block boundary do it all
sl@0
   559
				readahead = maxReadAhead;	
sl@0
   560
				}
sl@0
   561
			TPtr8 ptr(iBase, Min(iCapacity, (len + readahead)));
sl@0
   562
			err = iFile.Read(aFilePos, ptr);
sl@0
   563
			PROFILE_READ(aFilePos, ptr.Size(), err);
sl@0
   564
			if(err == KErrNone)
sl@0
   565
				{
sl@0
   566
				iFilePos = aFilePos;
sl@0
   567
				iLength = ptr.Length();	
sl@0
   568
				iNextReadFilePos = iFilePos + iLength;
sl@0
   569
				if(iLength == 0)
sl@0
   570
					{
sl@0
   571
					break;	
sl@0
   572
					}
sl@0
   573
				}
sl@0
   574
			else
sl@0
   575
				{
sl@0
   576
				DoDiscard();	
sl@0
   577
				}
sl@0
   578
			}
sl@0
   579
		}
sl@0
   580
	aDes.SetLength(aDes.MaxLength() - len);
sl@0
   581
	__FILEBUF64_INVARIANT();
sl@0
   582
	return err;
sl@0
   583
	}
sl@0
   584
	
sl@0
   585
/**
sl@0
   586
Writes to the file at the specified offset (aFilePos) within the file.
sl@0
   587
If certain conditions are met, the data will be stored in the buffer - no call to the file server.
sl@0
   588
sl@0
   589
@param aFilePos The offset from the start of the file at which the first byte is written. 
sl@0
   590
                If a position beyond the end of the file is specified, then
sl@0
   591
                the write operation begins at the end of the file.
sl@0
   592
                If the position has been locked, then the write fails.
sl@0
   593
            
sl@0
   594
@param aData The descriptor from which binary data is written. The function writes 
sl@0
   595
             the entire contents of aData to the file.
sl@0
   596
sl@0
   597
@return KErrNone if successful, otherwise one of the other system-wide error  codes.
sl@0
   598
sl@0
   599
@panic FBuf64  4 In _DEBUG mode - negative aFilePos value.
sl@0
   600
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
   601
sl@0
   602
@see RFileBuf64::Invariant()
sl@0
   603
*/
sl@0
   604
TInt RFileBuf64::Write(TInt64 aFilePos, const TDesC8& aData)
sl@0
   605
	{
sl@0
   606
	__FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
sl@0
   607
	__FILEBUF64_INVARIANT();
sl@0
   608
	if(aData.Length() == 0)
sl@0
   609
		{
sl@0
   610
		__FILEBUF64_INVARIANT();
sl@0
   611
		return KErrNone;	
sl@0
   612
		}
sl@0
   613
	TInt err = DoFileSize();
sl@0
   614
	if(err != KErrNone)
sl@0
   615
		{
sl@0
   616
		__FILEBUF64_INVARIANT();
sl@0
   617
		return err;	
sl@0
   618
		}
sl@0
   619
	if((err = DoSetCapacity(aData.Length())) != KErrNone)
sl@0
   620
		{
sl@0
   621
		return err;
sl@0
   622
		}
sl@0
   623
	DoDiscardBufferedReadData();
sl@0
   624
	const TUint8* data = aData.Ptr();
sl@0
   625
	for(TInt len = aData.Length(); len > 0 && err == KErrNone;)
sl@0
   626
		{
sl@0
   627
		//1. The new write pos is before the buffered file pos
sl@0
   628
		if(aFilePos < iFilePos)
sl@0
   629
			{
sl@0
   630
			//If the new data sticks to/overlapps the old data and there is room in the buffer to move the old data 
sl@0
   631
			//toward the end, then the new data can be copied at the beginning of the buffer.
sl@0
   632
			if((aFilePos + len) >= iFilePos && (iFilePos - aFilePos) <= (iCapacity - iLength))
sl@0
   633
				{
sl@0
   634
				(void)Mem::Copy(iBase + (iFilePos - aFilePos), iBase, iLength);	//Make room - move the existing data toward the end
sl@0
   635
				(void)Mem::Copy(iBase, data, len);								//of the buffer. Stick the new data to the old data
sl@0
   636
				iLength += (iFilePos - aFilePos);
sl@0
   637
				iFilePos = aFilePos;										//The new file pos is associated with the buffer
sl@0
   638
				iFileSize = Max(iFileSize, (iFilePos + iLength));
sl@0
   639
				len = 0;													//No more new data
sl@0
   640
				iDirty = ETrue;	
sl@0
   641
				}
sl@0
   642
			else
sl@0
   643
			//The "aFilePos" is too far before the "iFilePos". Write the buffer and associate the new pos with the buffer
sl@0
   644
				{
sl@0
   645
				err = DoFileWrite2(aFilePos);
sl@0
   646
				}
sl@0
   647
			}
sl@0
   648
		//2. The new write pos is after the associated file pos + the data length.
sl@0
   649
		else if(aFilePos > (iFilePos + iLength))
sl@0
   650
			{
sl@0
   651
			if(aFilePos > iFileSize)											//Beyond the end of the file
sl@0
   652
				{
sl@0
   653
				if((iFilePos + iLength) == iFileSize && (aFilePos - iFilePos) < iCapacity)	
sl@0
   654
					{															//but within the buffer => extend the file with zeros.
sl@0
   655
					Mem::FillZ(iBase + iLength, aFilePos - iFilePos - iLength);
sl@0
   656
					iLength = aFilePos - iFilePos;
sl@0
   657
					iFileSize = Max(iFileSize, (iFilePos + iLength));
sl@0
   658
					iDirty = ETrue;	
sl@0
   659
					}
sl@0
   660
				else									
sl@0
   661
				//Beyond the end of the file and not in the buffer - write the buffer to the file.
sl@0
   662
					{
sl@0
   663
                    err = DoFileWrite2(aFilePos);
sl@0
   664
					}
sl@0
   665
				}
sl@0
   666
			else										
sl@0
   667
			//Within the file, not in the buffer - write the buffer and associate the new file pos with the buffer
sl@0
   668
				{
sl@0
   669
				err = DoFileWrite2(aFilePos);
sl@0
   670
				}
sl@0
   671
			}
sl@0
   672
		//3. The new write pos is in the buffer, but the data length is too big
sl@0
   673
		//   (For SQLite is OK, otherwise the whole block must be written to the file)
sl@0
   674
		//4. The new write pos is in the buffer, the data entirely fits in the buffer
sl@0
   675
		else
sl@0
   676
			{
sl@0
   677
            if (iFilePos+iCapacity == aFilePos)	//The buffer is full. The new position to write is the end of the buffer.
sl@0
   678
                {
sl@0
   679
                err = DoFileWrite2(aFilePos);
sl@0
   680
                }
sl@0
   681
            if(err == KErrNone)
sl@0
   682
                {
sl@0
   683
                TInt amount = Min(len, (iCapacity - (aFilePos - iFilePos)));
sl@0
   684
                const TUint8* end = Mem::Copy(iBase + (aFilePos - iFilePos), data, amount);
sl@0
   685
                iLength = Max(iLength, (end - iBase));
sl@0
   686
                iFileSize = Max(iFileSize, (iFilePos + iLength));
sl@0
   687
                len -= amount;
sl@0
   688
                data += amount;
sl@0
   689
                aFilePos += amount;
sl@0
   690
                iDirty = ETrue;	
sl@0
   691
                }
sl@0
   692
           }
sl@0
   693
		}
sl@0
   694
	__FILEBUF64_INVARIANT();
sl@0
   695
	return err;
sl@0
   696
	}
sl@0
   697
	
sl@0
   698
/**
sl@0
   699
Gets the current file size.
sl@0
   700
sl@0
   701
@param aFileSize On return, the size of the file in bytes.
sl@0
   702
sl@0
   703
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   704
sl@0
   705
See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
sl@0
   706
sl@0
   707
@see RFileBuf64::Invariant()
sl@0
   708
*/
sl@0
   709
TInt RFileBuf64::Size(TInt64& aFileSize)
sl@0
   710
	{
sl@0
   711
	__FILEBUF64_INVARIANT();
sl@0
   712
	TInt err = DoFileSize();
sl@0
   713
	if(err == KErrNone)
sl@0
   714
		{
sl@0
   715
		aFileSize = iFileSize;
sl@0
   716
		}
sl@0
   717
	__FILEBUF64_INVARIANT();
sl@0
   718
	return err;
sl@0
   719
	}
sl@0
   720
sl@0
   721
/**
sl@0
   722
Sets the file size.
sl@0
   723
sl@0
   724
If the size of the file is reduced, data may be lost from the end of the file.
sl@0
   725
sl@0
   726
Note:
sl@0
   727
sl@0
   728
1. The current file position remains unchanged unless SetSize() reduces the size 
sl@0
   729
   of the file in such a way that the current file position is now beyond
sl@0
   730
   the end of the file. In this case, the current file position is set to
sl@0
   731
   the end of file. 
sl@0
   732
sl@0
   733
2. If the file was not opened for writing, an error is returned.
sl@0
   734
sl@0
   735
@param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
sl@0
   736
sl@0
   737
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   738
sl@0
   739
@panic FBuf64  5 In _DEBUG mode - negative aFileSize value.
sl@0
   740
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
   741
sl@0
   742
@see RFileBuf64::Invariant()
sl@0
   743
*/
sl@0
   744
TInt RFileBuf64::SetSize(TInt64 aFileSize)
sl@0
   745
	{
sl@0
   746
	__FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
sl@0
   747
	__FILEBUF64_INVARIANT();
sl@0
   748
	return DoSetFileSize(aFileSize);
sl@0
   749
	}
sl@0
   750
sl@0
   751
/**
sl@0
   752
Writes the pending data and then flushes the file.
sl@0
   753
sl@0
   754
Although RFileBuf64::Close() also flushes internal buffers, it is better
sl@0
   755
to call RFileBuf64::Flush() before the file is closed. This is because Close() returns no 
sl@0
   756
error information, so there is no way of telling whether the final data was 
sl@0
   757
written to the file successfully or not.
sl@0
   758
sl@0
   759
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   760
sl@0
   761
See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
sl@0
   762
sl@0
   763
@see RFileBuf64::Invariant()
sl@0
   764
*/
sl@0
   765
TInt RFileBuf64::Flush()
sl@0
   766
	{
sl@0
   767
	__FILEBUF64_INVARIANT();
sl@0
   768
	return DoFileFlush();
sl@0
   769
	}
sl@0
   770
sl@0
   771
/**
sl@0
   772
Gets information about the drive on which this file resides.
sl@0
   773
 
sl@0
   774
@param aDriveNumber On return, the drive number.
sl@0
   775
sl@0
   776
@param aDriveInfo   On return, contains information describing the drive
sl@0
   777
                    and the medium mounted on it. The value of TDriveInfo::iType
sl@0
   778
                    shows whether the drive contains media.
sl@0
   779
sl@0
   780
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   781
sl@0
   782
See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
sl@0
   783
sl@0
   784
@see RFileBuf64::Invariant()
sl@0
   785
*/
sl@0
   786
TInt RFileBuf64::Drive(TInt& aDriveNumber, TDriveInfo& aDriveInfo) const
sl@0
   787
	{
sl@0
   788
	__FILEBUF64_INVARIANT();
sl@0
   789
	return iFile.Drive(aDriveNumber, aDriveInfo);
sl@0
   790
	}
sl@0
   791
sl@0
   792
/**
sl@0
   793
Initializes RFileBuf64 data members with their initial values.   
sl@0
   794
Allocates memory for the file buffer. 
sl@0
   795
 
sl@0
   796
@return KErrNone if successful, 
sl@0
   797
        KErrNoMemory out of memory;
sl@0
   798
*/
sl@0
   799
TInt RFileBuf64::DoPreInit()
sl@0
   800
    {
sl@0
   801
    DoDiscard();
sl@0
   802
    iReadAheadSize = RFileBuf64::KDefaultReadAheadSize;
sl@0
   803
    iBase = static_cast <TUint8*> (User::Alloc(iCapacity));
sl@0
   804
    return iBase ? KErrNone : KErrNoMemory;   
sl@0
   805
    }
sl@0
   806
sl@0
   807
/**
sl@0
   808
Performs post-initialization of the RFileBuf64 object.   
sl@0
   809
If aInitErr is not KErrNone, then the buffer memory will be released.
sl@0
   810
The function returns the aInitErr value to the caller. 
sl@0
   811
sl@0
   812
@param aInitErr The result of the performed before the call RFileBuf64 initialization.
sl@0
   813
 
sl@0
   814
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   815
*/
sl@0
   816
TInt RFileBuf64::DoPostInit(TInt aInitErr)
sl@0
   817
    {
sl@0
   818
    if(aInitErr != KErrNone)
sl@0
   819
        {
sl@0
   820
        User::Free(iBase);
sl@0
   821
        iBase = 0;
sl@0
   822
        }
sl@0
   823
    return aInitErr;
sl@0
   824
    }
sl@0
   825
sl@0
   826
/**
sl@0
   827
Discards the content of the RFileBuf64 object returning it to the state as if it has just been created. 
sl@0
   828
*/
sl@0
   829
void RFileBuf64::DoDiscard()
sl@0
   830
	{
sl@0
   831
	iLength = 0;
sl@0
   832
	iFilePos = 0;
sl@0
   833
	iFileSize = KFileSizeNotSet;
sl@0
   834
	iDirty = EFalse;
sl@0
   835
	iNextReadFilePos = KNextReadFilePosNotSet;
sl@0
   836
	iNextReadFilePosHits = 0;
sl@0
   837
	}
sl@0
   838
sl@0
   839
/**
sl@0
   840
Gets the current file size. 
sl@0
   841
If iFileSize value is valid, then no call to the file server will be made.
sl@0
   842
Otherwise the file server will be called and the file size - stored (cached) in iFileSize data member for later use.
sl@0
   843
sl@0
   844
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   845
sl@0
   846
See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
sl@0
   847
sl@0
   848
@see RFileBuf64::Invariant()
sl@0
   849
*/
sl@0
   850
TInt RFileBuf64::DoFileSize()
sl@0
   851
	{
sl@0
   852
	__FILEBUF64_INVARIANT();
sl@0
   853
	if(iFileSize != KFileSizeNotSet)
sl@0
   854
		{
sl@0
   855
		__FILEBUF64_INVARIANT();
sl@0
   856
		return KErrNone;
sl@0
   857
		}
sl@0
   858
	TInt err = iFile.Size(iFileSize);
sl@0
   859
	PROFILE_SIZE(iFileSize, err);
sl@0
   860
	if(err != KErrNone)
sl@0
   861
		{
sl@0
   862
		DoDiscard();
sl@0
   863
		}
sl@0
   864
	else
sl@0
   865
	    {
sl@0
   866
        iRealFileSize = iFileSize;
sl@0
   867
	    }
sl@0
   868
	__FILEBUF64_INVARIANT();
sl@0
   869
	return err;
sl@0
   870
	}
sl@0
   871
sl@0
   872
/**
sl@0
   873
Sets the file size.
sl@0
   874
If the buffer contains pending data, the data will be written to the file 
sl@0
   875
before the "set file size" operation, if certain conditions are met. 
sl@0
   876
sl@0
   877
@param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
sl@0
   878
sl@0
   879
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   880
sl@0
   881
@panic FBuf64  5 In _DEBUG mode - negative aFileSize value.
sl@0
   882
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
   883
sl@0
   884
@see RFileBuf64::Invariant()
sl@0
   885
*/
sl@0
   886
TInt RFileBuf64::DoSetFileSize(TInt64 aFileSize)
sl@0
   887
	{
sl@0
   888
	__FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
sl@0
   889
	__FILEBUF64_INVARIANT();
sl@0
   890
	if(aFileSize < iFilePos)
sl@0
   891
		{
sl@0
   892
		iDirty = EFalse;
sl@0
   893
		iLength = 0;	
sl@0
   894
		}
sl@0
   895
	//If the new file size is "in" the buffer then change the "iLength"
sl@0
   896
	else if(aFileSize < (iFilePos + iLength))
sl@0
   897
		{
sl@0
   898
		iLength = aFileSize - iFilePos;
sl@0
   899
		}
sl@0
   900
	TInt err = iFile.SetSize(aFileSize);
sl@0
   901
	PROFILE_SETSIZE(aFileSize, err);
sl@0
   902
	if(err != KErrNone)
sl@0
   903
		{
sl@0
   904
		DoDiscard();
sl@0
   905
		}
sl@0
   906
	else
sl@0
   907
		{
sl@0
   908
		iFileSize = aFileSize;
sl@0
   909
		iRealFileSize = aFileSize;
sl@0
   910
		}
sl@0
   911
	__FILEBUF64_INVARIANT();
sl@0
   912
	return err;
sl@0
   913
	}
sl@0
   914
sl@0
   915
/**
sl@0
   916
Writes the pending data and flushes the file.
sl@0
   917
sl@0
   918
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   919
sl@0
   920
See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
sl@0
   921
sl@0
   922
@see RFileBuf64::Invariant()
sl@0
   923
*/
sl@0
   924
TInt RFileBuf64::DoFileFlush()
sl@0
   925
	{
sl@0
   926
	__FILEBUF64_INVARIANT();
sl@0
   927
	TInt err = DoFileWrite2();//Write the buffer if the iDirty flag is set. Do not preserve the buffer content and file pos.
sl@0
   928
	if(err != KErrNone)
sl@0
   929
		{
sl@0
   930
		__FILEBUF64_INVARIANT();
sl@0
   931
		return err;	
sl@0
   932
		}
sl@0
   933
	err = iFile.Flush();
sl@0
   934
	PROFILE_FLUSH(err);
sl@0
   935
	if(err != KErrNone)
sl@0
   936
		{
sl@0
   937
		DoDiscard();
sl@0
   938
		}
sl@0
   939
	iLength = 0;
sl@0
   940
	__FILEBUF64_INVARIANT();
sl@0
   941
	return err;
sl@0
   942
	}
sl@0
   943
sl@0
   944
/**
sl@0
   945
Writes the buffered data to the file if the iLength value is > 0.
sl@0
   946
If the file write operation extends the file, the iFileSize data member will be initialized with the new file size.
sl@0
   947
No changes occur in the other data member values.
sl@0
   948
sl@0
   949
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
   950
sl@0
   951
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
   952
sl@0
   953
@see RFileBuf64::DoFileWrite1()
sl@0
   954
@see RFileBuf64::DoFileWrite2()
sl@0
   955
@see RFileBuf64::Invariant()
sl@0
   956
*/
sl@0
   957
TInt RFileBuf64::DoFileWrite()
sl@0
   958
	{
sl@0
   959
	__FILEBUF64_INVARIANT();
sl@0
   960
	if(iLength == 0)
sl@0
   961
		{
sl@0
   962
		__FILEBUF64_INVARIANT();
sl@0
   963
		return KErrNone;	
sl@0
   964
		}
sl@0
   965
	TPtrC8 data(iBase, iLength);
sl@0
   966
	TInt err = KErrNone;
sl@0
   967
	if(iFilePos > iRealFileSize )
sl@0
   968
	    {
sl@0
   969
        err = DoSetFileSize(iFileSize);
sl@0
   970
 	    }
sl@0
   971
	if(err == KErrNone)
sl@0
   972
	    {
sl@0
   973
        err = iFile.Write(iFilePos, data);
sl@0
   974
	    }
sl@0
   975
	PROFILE_WRITE(iFilePos, iLength, err);
sl@0
   976
	if(err == KErrNone)
sl@0
   977
		{
sl@0
   978
		iRealFileSize = iFileSize;
sl@0
   979
		}
sl@0
   980
	else
sl@0
   981
		{
sl@0
   982
		DoDiscard();
sl@0
   983
		}
sl@0
   984
	__FILEBUF64_INVARIANT();
sl@0
   985
	return err;
sl@0
   986
	}
sl@0
   987
sl@0
   988
/**
sl@0
   989
Writes the buffered data to the file if the iDirty flag is set.
sl@0
   990
If the iDirty flag is set and the file write operation was successful, the iFilePos will be initialized with
sl@0
   991
the aNewFilePos value, the iLength will be set to 0.
sl@0
   992
This method is called from RFileBuf64::Read(), where:
sl@0
   993
 - if the buffer contains cached writes (iDirty flag is set), the buffer has to be flushed and iFilePos initialized
sl@0
   994
   with aNewFilePos - the offset in the file where the next file read operation should start from;
sl@0
   995
 - if the buffer contains cached reads, then nothing happens, the buffer content will be kept;
sl@0
   996
The function resets the iDirty flag.
sl@0
   997
sl@0
   998
@param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
sl@0
   999
				   the aNewFilePos value.
sl@0
  1000
sl@0
  1001
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
  1002
sl@0
  1003
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
  1004
				   
sl@0
  1005
@panic FBuf64  4 In _DEBUG mode - negative aNewFilePos value.
sl@0
  1006
sl@0
  1007
@see RFileBuf64::Read()
sl@0
  1008
@see RFileBuf64::DoFileWrite()
sl@0
  1009
@see RFileBuf64::DoFileWrite2()
sl@0
  1010
@see RFileBuf64::Invariant()
sl@0
  1011
*/
sl@0
  1012
TInt RFileBuf64::DoFileWrite1(TInt64 aNewFilePos)
sl@0
  1013
	{
sl@0
  1014
	__FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
sl@0
  1015
	__FILEBUF64_INVARIANT();
sl@0
  1016
	TInt err = KErrNone;
sl@0
  1017
	if(iDirty)
sl@0
  1018
		{
sl@0
  1019
		err = DoFileWrite();
sl@0
  1020
		if(err == KErrNone)	
sl@0
  1021
			{
sl@0
  1022
			iFilePos = aNewFilePos;
sl@0
  1023
			iLength = 0;
sl@0
  1024
			}
sl@0
  1025
		}
sl@0
  1026
	iDirty = EFalse;
sl@0
  1027
	__FILEBUF64_INVARIANT();
sl@0
  1028
	return err;		
sl@0
  1029
	}
sl@0
  1030
sl@0
  1031
/*
sl@0
  1032
Writes the buffered data to the file if the iDirty flag is set.
sl@0
  1033
If the file write operation was successful or if the iDirty flag was not set, the iFilePos will be initialized with
sl@0
  1034
the aNewFilePos value, the iLength will be set to 0.
sl@0
  1035
This method is called from RFileBuf64::Write() an other RFileBuf64 methods (but not from RFileBuf64::Read()), where:
sl@0
  1036
 - if the buffer contains cached writes (iDirty flag is set), the buffer has to be flushed and iFilePos initialized
sl@0
  1037
   with aNewFilePos - the offset in the file for which the write data will be cached in the buffer;
sl@0
  1038
 - if the buffer contains cached reads, then the buffer content will be destroyed, iFilePos initialized with aNewFilePos
sl@0
  1039
   and iLength set to 0;
sl@0
  1040
The function resets the iDirty flag.
sl@0
  1041
The difference between RFileBuf64::DoFileWrite1() and RFileBuf64::DoFileWrite2() is:
sl@0
  1042
 - RFileBuf64::DoFileWrite1() perserves the buffer content if iDirty is not set;
sl@0
  1043
 - RFileBuf64::DoFileWrite2() always destroys the buffer content and initializes iFilePos;
sl@0
  1044
sl@0
  1045
@param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
sl@0
  1046
				   the aNewFilePos value.
sl@0
  1047
sl@0
  1048
@return KErrNone if successful, otherwise one of the other system-wide error codes.
sl@0
  1049
sl@0
  1050
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
  1051
				   
sl@0
  1052
@panic FBuf64  4 In _DEBUG mode - negative aNewFilePos value.
sl@0
  1053
sl@0
  1054
@see RFileBuf64::Write()
sl@0
  1055
@see RFileBuf64::DoFileWrite()
sl@0
  1056
@see RFileBuf64::DoFileWrite1()
sl@0
  1057
@see RFileBuf64::Invariant()
sl@0
  1058
*/
sl@0
  1059
TInt RFileBuf64::DoFileWrite2(TInt64 aNewFilePos)
sl@0
  1060
	{
sl@0
  1061
	__FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
sl@0
  1062
	__FILEBUF64_INVARIANT();
sl@0
  1063
	TInt err = KErrNone;
sl@0
  1064
	if(iDirty)
sl@0
  1065
		{
sl@0
  1066
		err = DoFileWrite();
sl@0
  1067
		}
sl@0
  1068
	if(err == KErrNone)	
sl@0
  1069
		{
sl@0
  1070
		iFilePos = aNewFilePos;
sl@0
  1071
		iLength = 0;
sl@0
  1072
		}
sl@0
  1073
	iDirty = EFalse;
sl@0
  1074
	__FILEBUF64_INVARIANT();
sl@0
  1075
	return err;
sl@0
  1076
	}
sl@0
  1077
sl@0
  1078
/**
sl@0
  1079
This function discards the buffer content if the buffer contains cached read data.
sl@0
  1080
The function is called from RFileBuf64::Write(), because if the buffer contains cached read data,
sl@0
  1081
they cannot be mixed with the cached write data.
sl@0
  1082
Reason: for example the buffer contains 8Kb cached read data from file offset 0.
sl@0
  1083
        The data write request is 10 bytes at offset 4000. The write data will be cached,
sl@0
  1084
        because the buffer contains data from from this file area: [0..8192].
sl@0
  1085
        The iDirty flag will be set. Later when RFileBuf64::Flush() is called, the whole
sl@0
  1086
        8Kb buffer will be written. There is nothing wrong with that, the file content will be consistent.
sl@0
  1087
        But from performance point of view: 8Kb written vs. 10 bytes written - that may badly impact the performance.
sl@0
  1088
sl@0
  1089
@see RFileBuf64::Write()
sl@0
  1090
sl@0
  1091
See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
sl@0
  1092
*/
sl@0
  1093
void RFileBuf64::DoDiscardBufferedReadData()
sl@0
  1094
	{
sl@0
  1095
	__FILEBUF64_INVARIANT();
sl@0
  1096
	if(!iDirty && iLength > 0)
sl@0
  1097
		{
sl@0
  1098
		iLength = 0;
sl@0
  1099
		iFilePos = 0;
sl@0
  1100
		iNextReadFilePos = KNextReadFilePosNotSet;
sl@0
  1101
		iNextReadFilePosHits = 0;
sl@0
  1102
		}
sl@0
  1103
	__FILEBUF64_INVARIANT();
sl@0
  1104
	}
sl@0
  1105
sl@0
  1106
/**
sl@0
  1107
Sets the most appropriate buffer capacity based on the database page size.
sl@0
  1108
The function does a lazy evaluation. The first time the function is called and 
sl@0
  1109
aRwDataLength parameter is recognized to be a database or journal page size, the new (optimal)
sl@0
  1110
buffer capacity is calculated and set. All next DoSetCapacity() calls will detect that the new
sl@0
  1111
capacity is already set and will return KErrNone.
sl@0
  1112
sl@0
  1113
@param  aRwDataLength The length of the data being read or written.
sl@0
  1114
@return KErrNone The new capacity was set successfully,
sl@0
  1115
        KErrNoMemory Out of memory.
sl@0
  1116
*/
sl@0
  1117
TInt RFileBuf64::DoSetCapacity(TInt aRwDataLength)
sl@0
  1118
	{
sl@0
  1119
	const TInt KMinPageCount = 4;//the buffer capacity should be at least (KMinPageCount * page size) 
sl@0
  1120
	                             //but not less than the original capacity.
sl@0
  1121
	const TInt KDefaultPageSize = 1024;//The journal header size is equal to 512 bytes, so it is not easy
sl@0
  1122
                                       //to detect the 512 bytes page size. 
sl@0
  1123
	
sl@0
  1124
	__FBUF64_ASSERT(aRwDataLength > 0, EFBufPanicRwDataLength);
sl@0
  1125
	__FILEBUF64_INVARIANT();
sl@0
  1126
	if(iOptimized)
sl@0
  1127
		{
sl@0
  1128
		__FILEBUF64_INVARIANT();
sl@0
  1129
		return KErrNone;
sl@0
  1130
		}
sl@0
  1131
	if((aRwDataLength & (aRwDataLength - 1)) != 0 || aRwDataLength < KDefaultPageSize)
sl@0
  1132
		{
sl@0
  1133
		__FILEBUF64_INVARIANT();
sl@0
  1134
		return KErrNone;
sl@0
  1135
		}
sl@0
  1136
	//Here: aRwDataLength is power of 2 and is bigger than the default db page size.
sl@0
  1137
	//aRwDataLength is the size of the db page.
sl@0
  1138
	const TInt pageSize = aRwDataLength;
sl@0
  1139
	TInt cnt = iCapacity / pageSize;//how many pages can fit in the buffer now
sl@0
  1140
	TInt pageCount = Max(cnt, KMinPageCount);//the number of pages that should fit in the new buffer
sl@0
  1141
	TInt newBufCapacity = pageCount * pageSize;
sl@0
  1142
	if(newBufCapacity != iCapacity)
sl@0
  1143
		{
sl@0
  1144
		TUint8* newBase = static_cast <TUint8*> (User::ReAlloc(iBase, newBufCapacity));
sl@0
  1145
		if(!newBase)
sl@0
  1146
			{
sl@0
  1147
			__FILEBUF64_INVARIANT();
sl@0
  1148
			return KErrNoMemory;
sl@0
  1149
			}
sl@0
  1150
		iBase = newBase;
sl@0
  1151
		iCapacity = newBufCapacity;
sl@0
  1152
		//Adjust the initial read-ahead size to be multiple of the page size.
sl@0
  1153
		if((iReadAheadSize % pageSize) != 0)
sl@0
  1154
			{
sl@0
  1155
			TInt q = iReadAheadSize / pageSize;
sl@0
  1156
			iReadAheadSize = q != 0 ? pageSize * q : pageSize;
sl@0
  1157
			}
sl@0
  1158
		}
sl@0
  1159
	iOptimized = ETrue;
sl@0
  1160
	__FILEBUF64_INVARIANT();
sl@0
  1161
	return KErrNone;
sl@0
  1162
	}
sl@0
  1163
sl@0
  1164
#ifdef _DEBUG
sl@0
  1165
sl@0
  1166
/**
sl@0
  1167
RFileBuf64 invariant. Called in _DEBUG mode at the beginning and before the end of every RFileBuf64 method
sl@0
  1168
(except the init/destroy methods).
sl@0
  1169
sl@0
  1170
@panic FBuf64  11 In _DEBUG mode - null "this" pointer.
sl@0
  1171
@panic FBuf64   1 In _DEBUG mode - negative iCapacity value.
sl@0
  1172
@panic FBuf64   2 In _DEBUG mode - the buffer pointer is null (possible the buffer is not allocated or already destroyed).
sl@0
  1173
@panic FBuf64   3 In _DEBUG mode - invalid iLength value (negative or bigger than iCapacity).
sl@0
  1174
@panic FBuf64   4 In _DEBUG mode - negative iFilePos value.
sl@0
  1175
@panic FBuf64   5 In _DEBUG mode - set but negative iFileSize value.
sl@0
  1176
@panic FBuf64   6 In _DEBUG mode - null file handle (the RFile64 object is not created or already destroyed).
sl@0
  1177
@panic FBuf64  13 In _DEBUG mode - set but negative iNextReadFilePos value.
sl@0
  1178
@panic FBuf64  14 In _DEBUG mode - negative iNextReadFilePosHits value.
sl@0
  1179
@panic FBuf64  15 In _DEBUG mode - iReadAheadSize is negative or is bigger than iCapacity.
sl@0
  1180
*/
sl@0
  1181
void RFileBuf64::Invariant() const
sl@0
  1182
	{
sl@0
  1183
	__FBUF64_ASSERT(this != 0, EFBufPanicNullThis);
sl@0
  1184
	__FBUF64_ASSERT(iCapacity > 0, EFBufPanicCapacity);
sl@0
  1185
	__FBUF64_ASSERT(iBase != 0, EFBufPanicNullBuf);
sl@0
  1186
	__FBUF64_ASSERT(iLength >= 0 && iLength <= iCapacity, EFBufPanicBufLen);
sl@0
  1187
	__FBUF64_ASSERT(iFilePos >= 0, EFBufPanicFilePos);
sl@0
  1188
	__FBUF64_ASSERT(iFileSize == KFileSizeNotSet || iFileSize >= 0, EFBufPanicFileSize);
sl@0
  1189
	__FBUF64_ASSERT(iFile.SubSessionHandle() != 0, EFBufPanicFileHandle);
sl@0
  1190
	__FBUF64_ASSERT(iNextReadFilePos == KNextReadFilePosNotSet || iNextReadFilePos >= 0, EFBufPanicNextReadFilePos);
sl@0
  1191
	__FBUF64_ASSERT(iNextReadFilePosHits >= 0, EFBufPanicNextReadFilePosHits);
sl@0
  1192
	__FBUF64_ASSERT(iReadAheadSize > 0, EFBufPanicFileBlockSize);
sl@0
  1193
	}
sl@0
  1194
	
sl@0
  1195
#endif