os/persistentdata/persistentstorage/sql/OsLayer/FileBuf64.h
changeset 0 bde4ae8d615e
     1.1 --- /dev/null	Thu Jan 01 00:00:00 1970 +0000
     1.2 +++ b/os/persistentdata/persistentstorage/sql/OsLayer/FileBuf64.h	Fri Jun 15 03:10:57 2012 +0200
     1.3 @@ -0,0 +1,210 @@
     1.4 +// Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
     1.5 +// All rights reserved.
     1.6 +// This component and the accompanying materials are made available
     1.7 +// under the terms of "Eclipse Public License v1.0"
     1.8 +// which accompanies this distribution, and is available
     1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
    1.10 +//
    1.11 +// Initial Contributors:
    1.12 +// Nokia Corporation - initial contribution.
    1.13 +//
    1.14 +// Contributors:
    1.15 +//
    1.16 +// Description:
    1.17 +//
    1.18 +#ifndef FILEBUF64_H
    1.19 +#define FILEBUF64_H
    1.20 +
    1.21 +#include <f32file.h>
    1.22 +#include <f32file64.h>
    1.23 +
    1.24 +/**
    1.25 +The RFileBuf64 class provides buffered file read/write operations on a single RFile64 object.
    1.26 +RFileBuf64::Read() and RFileBuf64::Write() methods may boost the performance of the read/write file operations up to 30% 
    1.27 +if compared to their RFile64 equivalents. Especially this is true in the case of a set of sequential read or write
    1.28 +operations when the file offset of operation N+1 is the end file offset of operation N plus one byte.
    1.29 +
    1.30 +The RFileBuf64 public methods declarations match the declarations of the most often used RFile64 public methods.
    1.31 +That makes the source code migration from RFile64 to RFileBuf64 easier (or from RFileBuf64 to RFile64).
    1.32 +
    1.33 +The RFileBuf64 capabilities are similar to those of the RFileBuf class, except the fact that RFileBuf
    1.34 +uses 32-bit file offsets, RFileBuf64 uses 64-bit file offsets and RFileBuf64 provides optimised read-ahead operations,
    1.35 +crafted especially for the case when RFileBuf64 is used with a page based database management system (like SQLite).
    1.36 +
    1.37 +Usage notes:
    1.38 +@code
    1.39 +	- an object of RFileBuf64 type must be defined first, specifying the max size (capacity) of the buffer as a parameter
    1.40 +	  of the constructor:
    1.41 +	  
    1.42 +	  	RFileBuf64 fbuf(<N>);//<N> is the minimal buffer capacity in bytes
    1.43 +	  	
    1.44 +	- the second step is to initialize the just defined RFileBuf64 object by calling one of the "resource acquisition"
    1.45 +	  methods: RFileBuf64::Create(), RFileBuf64::Open(), RFileBuf64::Temp() or RFileBuf64::AdoptFromClient().
    1.46 +	  
    1.47 +	  In details, to create a file and access it through a RFileBuf64 object:
    1.48 +	  
    1.49 +	  	RFs fs;
    1.50 +        TInt err = fs.Connect();
    1.51 +        //check the error
    1.52 +	  	...
    1.53 +	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
    1.54 +	  	err = fbuf.Create(fs, <file name>, <file mode>);
    1.55 +	  	//check the error
    1.56 +	  	
    1.57 +	  To open an existing file and access it through a RFileBuf64 object:
    1.58 +
    1.59 +	  	RFs fs;
    1.60 +        TInt err = fs.Connect();
    1.61 +        //check the error
    1.62 +	  	...
    1.63 +	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
    1.64 +	  	err = fbuf.Open(fs, <file name>, <file mode>);
    1.65 +	  	//check the error
    1.66 +	  
    1.67 +	  To create a temporary file and access it through a RFileBuf64 object:
    1.68 +
    1.69 +	  	RFs fs;
    1.70 +        TInt err = fs.Connect();
    1.71 +        //check the error
    1.72 +	  	...
    1.73 +	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
    1.74 +	  	err = fbuf.Temp(fs, <path>, <file name>, <file mode>);
    1.75 +	  	//check the error
    1.76 +	  
    1.77 +	  To open a file from handle and access it through a RFileBuf64 object:
    1.78 +
    1.79 +	  	RFs fs;
    1.80 +        TInt err = fs.Connect();
    1.81 +        //check the error
    1.82 +	  	...
    1.83 +	  	RFileBuf64 fbuf(<N>);									//<N> is the buffer capacity in bytes
    1.84 +	  	err = fbuf.AdoptFromClient(<msg>, <fs handle index>, <file handle index>);
    1.85 +	  	//check the error
    1.86 +
    1.87 +	- if the RFileBuf64 object is initialised successfully, now the public RFileBuf64 methods can be called to perform
    1.88 +	  requested operations on the file:
    1.89 +	  
    1.90 +	  	err = fbuf.Write(<file pos>, <data>);
    1.91 +	  	//check the error
    1.92 +	  	....
    1.93 +	  	err = fbuf.Read(<file pos>, <buf>);
    1.94 +	  	//check the error
    1.95 +	  	....
    1.96 +	  Note: do not forget to call RFileBuf64::Flush() at the moment when you want to ensure that the file data
    1.97 +	  	    (possibly buffered) is really written to the file.
    1.98 +
    1.99 +	- The final step is to close the RFileBuf64 object and the corresponding file thus realising the used resouces:
   1.100 +	
   1.101 +		fbuf.Close();
   1.102 +	  
   1.103 +@endcode
   1.104 +
   1.105 +Implementation notes: the current RFileBuf64 implementation is optimised for use by the SQLite OS porting layer.
   1.106 +	After investigation of SQLite file read/write operations it was found that buffering of
   1.107 +	two or more logical file writes into a single physical file write has a positive impact (as expected) on the performance 
   1.108 +	of the database write operations. But the picture is quite different for the file read operations. The database data is
   1.109 +	organised in pages with fixed size. After a database is created and set of insert/update/delete operations is performed 
   1.110 +	on it, after a while the database pages (identified by their numbers) are not sequential in the database file and using
   1.111 +	a read-ahead buffer with fixed size makes no sense because for each "page read" request of N bytes, the RFileBuf64 object
   1.112 +	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 
   1.113 +	not sequential (relatively to the page numbers), obviously the read-ahead data is wasted and the "read page" performance
   1.114 +	is negatively impacted. This observation is true in general except two cases:
   1.115 +		- sequential scan of a database table;
   1.116 +		- reading of a BLOB column;
   1.117 +	In these two cases it is likely that the table/blob data occupies pages with sequential numbers located in a continuous
   1.118 +	file area. Then if a read-ahead buffer is used that will have a positive impact on the "read page" performance, because
   1.119 +	the number of the read IPC calls to the file server will be reduced.
   1.120 +	In order to satisfy those two orthogonal requirements, the RFileBuf64 implementation uses a "read file offset prediction"
   1.121 +	algorithm:
   1.122 +		- The file buffer object is created with 0 read-ahead value;
   1.123 +		- After each "file read" operation the buffer implementation "guesses" what might be the file offset of the
   1.124 +		  next file read operation (it is possible because the database data is organised in pages with fixed size and
   1.125 +		  generally all "file read" requests are for reading a database page) and stores the value in one of its data members;
   1.126 +		- If the file offset of the next file read operation matches the "guessed" offset, the read-ahead value is changed from
   1.127 +		  0 to 1024 bytes (1024 bytes is the default database page size);
   1.128 +		- Every next match of the "guessed" file offset value doubles the read-ahead value. But the max read-ahead value is
   1.129 +		  capped by the capacity of the buffer;
   1.130 +		- If the file offset of the next file read operation does not match the "guessed" file offset, then the read-ahead
   1.131 +		  value is set back to 0;
   1.132 +	Shortly, depending of the nature of the file read requests, the RFileBuf64 object will dynamically change the read-ahead
   1.133 +	value thus minimising the amount of the wasted read data and improving the "file read" performance in general.
   1.134 +
   1.135 +@see RFile64
   1.136 +@see RFileBuf
   1.137 +
   1.138 +@internalComponent
   1.139 +*/
   1.140 +class RFileBuf64
   1.141 +	{
   1.142 +	friend void SetReadAheadSizeTest();
   1.143 +	
   1.144 +	enum {KDefaultReadAheadSize = 1024};//Default size in bytes of the read-ahead buffer
   1.145 +	
   1.146 +public:
   1.147 +	RFileBuf64(TInt aMinCapacity);
   1.148 +
   1.149 +	TInt Create(RFs& aFs, const TDesC& aFileName, TUint aFileMode);
   1.150 +	TInt Open(RFs& aFs, const TDesC& aFileName, TUint aFileMode);
   1.151 +	TInt Temp(RFs& aFs, const TDesC& aPath, TFileName& aFileName, TUint aFileMode);
   1.152 +	TInt AdoptFromClient(const RMessage2& aMsg, TInt aFsIndex, TInt aFileIndex);
   1.153 +	void Close();
   1.154 +	TInt SetReadAheadSize(TInt aBlockSize, TInt aReadRecBufSize);
   1.155 +
   1.156 +	TInt Read(TInt64 aFilePos, TDes8& aDes);
   1.157 +	TInt Write(TInt64 aFilePos, const TDesC8& aData);
   1.158 +
   1.159 +	TInt Size(TInt64& aFileSize);
   1.160 +	TInt SetSize(TInt64 aFileSize);
   1.161 +	TInt Flush();
   1.162 +
   1.163 +	TInt Drive(TInt& aDriveNumber, TDriveInfo& aDriveInfo) const;
   1.164 +
   1.165 +private:
   1.166 +	void Invariant() const;
   1.167 +	TInt DoPreInit();
   1.168 +	TInt DoPostInit(TInt aInitErr);
   1.169 +	void DoDiscard();
   1.170 +	TInt DoFileSize();
   1.171 +	TInt DoSetFileSize(TInt64 aFileSize);
   1.172 +	TInt DoFileFlush();
   1.173 +	TInt DoFileWrite();
   1.174 +	TInt DoFileWrite1(TInt64 aNewFilePos);
   1.175 +	TInt DoFileWrite2(TInt64 aNewFilePos = 0LL);
   1.176 +	void DoDiscardBufferedReadData();
   1.177 +	TInt DoSetCapacity(TInt aRwDataLength);
   1.178 +	
   1.179 +private:
   1.180 +	//Buffer related
   1.181 +	TInt		iCapacity;				//The buffer size. Indicates how much data can be put in.
   1.182 +	TUint8*		iBase;					//Pointer to the beginning of the buffer.
   1.183 +	TInt		iLength;				//The length of the data currently held in the buffer.
   1.184 +	//File related
   1.185 +	TInt64		iFilePos;				//The file position associated with the beginning of the buffer.
   1.186 +	TInt64		iFileSize;				//The file size which is the nominal file length including the buffer contents.
   1.187 +    TInt64      iRealFileSize;          //The real file size in disk.
   1.188 +	RFile64		iFile;					//The file object.
   1.189 +	//Read-ahead related
   1.190 +	TBool		iDirty;					//The buffer contains pending data to be written to the file
   1.191 +	TInt64		iNextReadFilePos;		//The guessed file position of the next "file read" operation
   1.192 +	TInt		iNextReadFilePosHits;	//How many times the guessed file position of the "file read" operation was correct
   1.193 +	TInt		iReadAheadSize;
   1.194 +	//
   1.195 +	TBool		iOptimized;				//True if the file buffer capacity has been optimized already
   1.196 +
   1.197 +	//Profiler related
   1.198 +#ifdef _SQLPROFILER
   1.199 +public:
   1.200 +    void		ProfilerReset();
   1.201 +    
   1.202 +	TInt		iFileReadCount;		//The number of the non-buffered file reads (RFile64::Read() calls).
   1.203 +	TInt64		iFileReadAmount;	//The amount of the data read from the file.
   1.204 +	TInt		iFileWriteCount;	//The number of the non-buffered file writes (RFile64::Write() calls).
   1.205 +	TInt64		iFileWriteAmount;	//The amount of the data written to the file.
   1.206 +	TInt		iFileSizeCount;		//The number of non-buffered RFile64::Size() calls.
   1.207 +	TInt		iFileSetSizeCount;	//The number of non-buffered RFile64::SetSize() calls.
   1.208 +	TInt		iFileFlushCount;	//The number of RFile64::Flush() calls.
   1.209 +#endif
   1.210 +	
   1.211 +	};
   1.212 +
   1.213 +#endif//FILEBUF64_H