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