os/persistentdata/persistentstorage/sqlite3api/OsLayer/FileBuf64.h
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-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
#ifndef FILEBUF64_H
sl@0
    16
#define FILEBUF64_H
sl@0
    17
sl@0
    18
#include <f32file.h>
sl@0
    19
#include <f32file64.h>
sl@0
    20
sl@0
    21
/**
sl@0
    22
The RFileBuf64 class provides buffered file read/write operations on a single RFile64 object.
sl@0
    23
RFileBuf64::Read() and RFileBuf64::Write() methods may boost the performance of the read/write file operations up to 30% 
sl@0
    24
if compared to their RFile64 equivalents. Especially this is true in the case of a set of sequential read or write
sl@0
    25
operations when the file offset of operation N+1 is the end file offset of operation N plus one byte.
sl@0
    26
sl@0
    27
The RFileBuf64 public methods declarations match the declarations of the most often used RFile64 public methods.
sl@0
    28
That makes the source code migration from RFile64 to RFileBuf64 easier (or from RFileBuf64 to RFile64).
sl@0
    29
sl@0
    30
The RFileBuf64 capabilities are similar to those of the RFileBuf class, except the fact that RFileBuf
sl@0
    31
uses 32-bit file offsets, RFileBuf64 uses 64-bit file offsets and RFileBuf64 provides optimised read-ahead operations,
sl@0
    32
crafted especially for the case when RFileBuf64 is used with a page based database management system (like SQLite).
sl@0
    33
sl@0
    34
Usage notes:
sl@0
    35
@code
sl@0
    36
	- an object of RFileBuf64 type must be defined first, specifying the max size (capacity) of the buffer as a parameter
sl@0
    37
	  of the constructor:
sl@0
    38
	  
sl@0
    39
	  	RFileBuf64 fbuf(<N>);//<N> is the buffer capacity in bytes
sl@0
    40
	  	
sl@0
    41
	- the second step is to initialize the just defined RFileBuf64 object by calling one of the "resource acquisition"
sl@0
    42
	  methods: RFileBuf64::Create(), RFileBuf64::Open() or RFileBuf64::Temp().
sl@0
    43
	  
sl@0
    44
	  In details, to create a file and access it through a RFileBuf64 object:
sl@0
    45
	  
sl@0
    46
	  	RFs fs;
sl@0
    47
	  	TInt err = fs.Connect();
sl@0
    48
        //check the error
sl@0
    49
	  	...
sl@0
    50
	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
sl@0
    51
	  	err = fbuf.Create(fs, <file name>, <file mode>);
sl@0
    52
	  	//check the error
sl@0
    53
	  	
sl@0
    54
	  To open an existing file and access it through a RFileBuf64 object:
sl@0
    55
sl@0
    56
	  	RFs fs;
sl@0
    57
        TInt err = fs.Connect();
sl@0
    58
        //check the error
sl@0
    59
	  	...
sl@0
    60
	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
sl@0
    61
	  	err = fbuf.Open(fs, <file name>, <file mode>);
sl@0
    62
	  	//check the error
sl@0
    63
	  
sl@0
    64
	  To create a temporary file and access it through a RFileBuf64 object:
sl@0
    65
sl@0
    66
	  	RFs fs;
sl@0
    67
        TInt err = fs.Connect();
sl@0
    68
        //check the error
sl@0
    69
	  	...
sl@0
    70
	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
sl@0
    71
	  	err = fbuf.Temp(fs, <path>, <file name>, <file mode>);
sl@0
    72
	  	//check the error
sl@0
    73
	  
sl@0
    74
	- if the RFileBuf64 object is initialised successfully, now the public RFileBuf64 methods can be called to perform
sl@0
    75
	  requested operations on the file:
sl@0
    76
	  
sl@0
    77
	  	err = fbuf.Write(<file pos>, <data>);
sl@0
    78
	  	//check the error
sl@0
    79
	  	....
sl@0
    80
	  	err = fbuf.Read(<file pos>, <buf>);
sl@0
    81
	  	//check the error
sl@0
    82
	  	....
sl@0
    83
	  Note: do not forget to call RFileBuf64::Flush() at the moment when you want to ensure that the file data
sl@0
    84
	  	    (possibly buffered) is really written to the file.
sl@0
    85
sl@0
    86
	- The final step is to close the RFileBuf64 object and the corresponding file thus realising the used resouces:
sl@0
    87
	
sl@0
    88
		fbuf.Close();
sl@0
    89
	  
sl@0
    90
@endcode
sl@0
    91
sl@0
    92
Implementation notes: the current RFileBuf64 implementation is optimised for use by the SQLite OS porting layer.
sl@0
    93
	After investigation of SQLite file read/write operations it was found that buffering of
sl@0
    94
	two or more logical file writes into a single physical file write has a positive impact (as expected) on the performance 
sl@0
    95
	of the database write operations. But the picture is quite different for the file read operations. The database data is
sl@0
    96
	organised in pages with fixed size. After a database is created and set of insert/update/delete operations is performed 
sl@0
    97
	on it, after a while the database pages (identified by their numbers) are not sequential in the database file and using
sl@0
    98
	a read-ahead buffer with fixed size makes no sense because for each "page read" request of N bytes, the RFileBuf64 object
sl@0
    99
	will read up to K bytes, K >= N, where K is the read-ahead value in bytes. Since the "read page" requests in general are 
sl@0
   100
	not sequential (relatively to the page numbers), obviously the read-ahead data is wasted and the "read page" performance
sl@0
   101
	is negatively impacted. This observation is true in general except two cases:
sl@0
   102
		- sequential scan of a database table;
sl@0
   103
		- reading of a BLOB column;
sl@0
   104
	In these two cases it is likely that the table/blob data occupies pages with sequential numbers located in a continuous
sl@0
   105
	file area. Then if a read-ahead buffer is used that will have a positive impact on the "read page" performance, because
sl@0
   106
	the number of the read IPC calls to the file server will be reduced.
sl@0
   107
	In order to satisfy those two orthogonal requirements, the RFileBuf64 implementation uses a "read file offset prediction"
sl@0
   108
	algorithm:
sl@0
   109
		- The file buffer object is created with 0 read-ahead value;
sl@0
   110
		- After each "file read" operation the buffer implementation "guesses" what might be the file offset of the
sl@0
   111
		  next file read operation (it is possible because the database data is organised in pages with fixed size and
sl@0
   112
		  generally all "file read" requests are for reading a database page) and stores the value in one of its data members;
sl@0
   113
		- If the file offset of the next file read operation matches the "guessed" offset, the read-ahead value is changed from
sl@0
   114
		  0 to 1024 bytes (1024 bytes is the default database page size);
sl@0
   115
		- Every next match of the "guessed" file offset value doubles the read-ahead value. But the max read-ahead value is
sl@0
   116
		  capped by the capacity of the buffer;
sl@0
   117
		- If the file offset of the next file read operation does not match the "guessed" file offset, then the read-ahead
sl@0
   118
		  value is set back to 0;
sl@0
   119
	Shortly, depending of the nature of the file read requests, the RFileBuf64 object will dynamically change the read-ahead
sl@0
   120
	value thus minimising the amount of the wasted read data and improving the "file read" performance in general.
sl@0
   121
sl@0
   122
@see RFile64
sl@0
   123
@see RFileBuf
sl@0
   124
sl@0
   125
@internalComponent
sl@0
   126
*/
sl@0
   127
class RFileBuf64
sl@0
   128
	{
sl@0
   129
	enum {KDefaultReadAheadSize = 1024};//Default size in bytes of the read-ahead buffer
sl@0
   130
	
sl@0
   131
public:
sl@0
   132
	RFileBuf64(TInt aSize);
sl@0
   133
sl@0
   134
	TInt Create(RFs& aFs, const TDesC& aFileName, TUint aFileMode);
sl@0
   135
	TInt Open(RFs& aFs, const TDesC& aFileName, TUint aFileMode);
sl@0
   136
	TInt Temp(RFs& aFs, const TDesC& aPath, TFileName& aFileName, TUint aFileMode);
sl@0
   137
	void Close();
sl@0
   138
	TInt SetReadAheadSize(TInt aBlockSize, TInt aReadRecBufSize);
sl@0
   139
sl@0
   140
	TInt Read(TInt64 aFilePos, TDes8& aDes);
sl@0
   141
	TInt Write(TInt64 aFilePos, const TDesC8& aData);
sl@0
   142
sl@0
   143
	TInt Size(TInt64& aFileSize);
sl@0
   144
	TInt SetSize(TInt64 aFileSize);
sl@0
   145
	TInt Lock(TInt64 aFilePos, TInt64 aLength) const;
sl@0
   146
	TInt UnLock(TInt64 aFilePos, TInt64 aLength) const;
sl@0
   147
	TInt Flush(TBool aResetCachedFileSize = EFalse);
sl@0
   148
sl@0
   149
	TInt Drive(TInt& aDriveNumber, TDriveInfo& aDriveInfo) const;
sl@0
   150
sl@0
   151
private:
sl@0
   152
	void Invariant() const;
sl@0
   153
    TInt DoPreInit();
sl@0
   154
    void DoPostInit(TInt aInitErr);
sl@0
   155
	void DoDiscard();
sl@0
   156
	TInt DoFileSize();
sl@0
   157
	TInt DoSetFileSize(TInt64 aFileSize);
sl@0
   158
	TInt DoFileFlush();
sl@0
   159
	TInt DoFileWrite();
sl@0
   160
	TInt DoFileWrite1(TInt64 aNewFilePos);
sl@0
   161
	TInt DoFileWrite2(TInt64 aNewFilePos = 0LL);
sl@0
   162
	void DoDiscardBufferedReadData();
sl@0
   163
	
sl@0
   164
private:
sl@0
   165
	//Buffer related
sl@0
   166
	const TInt	iCapacity;				//The buffer size. Indicates how much data can be put in.
sl@0
   167
	TUint8*		iBase;					//Pointer to the beginning of the buffer.
sl@0
   168
	TInt		iLength;				//The length of the data currently held in the buffer.
sl@0
   169
	//File related
sl@0
   170
	TInt64		iFilePos;				//The file position associated with the beginning of the buffer.
sl@0
   171
	TInt64		iFileSize;				//The file size.
sl@0
   172
	RFile64		iFile;					//The file object.
sl@0
   173
	//Read-ahead related
sl@0
   174
	TBool		iDirty;					//The buffer contains pending data to be written to the file
sl@0
   175
	TInt64		iNextReadFilePos;		//The guessed file position of the next "file read" operation
sl@0
   176
	TInt		iNextReadFilePosHits;	//How many times the guessed file position of the "file read" operation was correct
sl@0
   177
	TInt		iReadAheadSize;
sl@0
   178
sl@0
   179
	//Profiler related
sl@0
   180
#ifdef _SQLPROFILER
sl@0
   181
public:
sl@0
   182
	TInt		iFileReadCount;		//The number of the non-buffered file reads (RFile64::Read() calls).
sl@0
   183
	TInt64		iFileReadAmount;	//The amount of the data read from the file.
sl@0
   184
	TInt		iFileWriteCount;	//The number of the non-buffered file writes (RFile64::Write() calls).
sl@0
   185
	TInt64		iFileWriteAmount;	//The amount of the data written to the file.
sl@0
   186
	TInt		iFileSizeCount;		//The number of non-buffered RFile64::Size() calls.
sl@0
   187
	TInt		iFileSetSizeCount;	//The number of non-buffered RFile64::SetSize() calls.
sl@0
   188
	TInt		iFileFlushCount;	//The number of RFile64::Flush() calls.
sl@0
   189
#endif
sl@0
   190
	
sl@0
   191
	};
sl@0
   192
sl@0
   193
#endif//FILEBUF64_H