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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
15 #include "FileBuf64.h"
17 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
18 /////////////////////////// PROFILER ////////////////////////////////////////////////////////////////////
19 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
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).
25 #define PROFILE_READ(pos, amount, err) \
28 if(TheOsCallTimeDetailedProfileEnabled) \
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); \
35 #define PROFILE_WRITE(pos, amount, err) \
38 if(TheOsCallTimeDetailedProfileEnabled) \
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); \
45 #define PROFILE_SIZE(size, err) \
48 if(TheOsCallTimeDetailedProfileEnabled) \
51 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Size¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSizeCount, size, err); \
55 #define PROFILE_SETSIZE(size, err) \
58 if(TheOsCallTimeDetailedProfileEnabled) \
60 ++iFileSetSizeCount; \
61 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬SetSize¬%d¬%ld¬¬¬%d\r\n"), (TUint32)this, iFileSetSizeCount, size, err); \
65 #define PROFILE_FLUSH(err) \
68 if(TheOsCallTimeDetailedProfileEnabled) \
71 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Flush¬%d¬¬¬¬%d\r\n"), (TUint32)this, iFileFlushCount, err); \
75 #define PROFILE_CREATE(fname, err) \
78 if(TheOsCallTimeDetailedProfileEnabled) \
80 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Create¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
84 #define PROFILE_OPEN(fname, err) \
87 if(TheOsCallTimeDetailedProfileEnabled) \
89 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Open¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
93 #define PROFILE_TEMP(fname, err) \
96 if(TheOsCallTimeDetailedProfileEnabled) \
98 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Temp¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
102 #define PROFILE_ADOPT(fname, err) \
105 if(TheOsCallTimeDetailedProfileEnabled) \
107 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Adopt¬¬¬¬¬%d¬%S\r\n"), (TUint32)this, err, &fname); \
111 #define PROFILE_CLOSE() \
114 if(TheOsCallTimeDetailedProfileEnabled) \
116 RDebug::Print(_L("[SQL-FBUF]¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬¬\"%X\"¬Close¬¬¬¬¬¬\r\n"), (TUint32)this); \
120 //Resets the profiler counters
121 void RFileBuf64::ProfilerReset()
123 iFileReadCount = 0; iFileReadAmount = 0; iFileWriteCount = 0; iFileWriteAmount = 0; iFileSizeCount = 0; iFileSetSizeCount = 0; iFileFlushCount = 0;
128 #define PROFILE_READ(pos,amount, err) void(0)
129 #define PROFILE_WRITE(pos,amount, err) void(0)
131 #define PROFILE_SIZE(size, err) void(0)
132 #define PROFILE_SETSIZE(size, err) void(0)
133 #define PROFILE_FLUSH(err) void(0)
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)
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)
149 static const TInt KFileSizeNotSet = -1;
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.
156 static const TInt KNextReadFilePosNotSet = -1;
158 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
159 /////////////////////////// ASSERTS & INVARIANT //////////////////////////////////////////////////////////
160 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
164 #define __FILEBUF64_INVARIANT() Invariant()
167 String literal used in _DEBUG mode for indicating that the reported panic happened inside the RFileBuf64 implementation.
172 _LIT(KPanicCategory, "FBuf64");
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.
183 EFBufPanicCapacity = 1, //1
187 EFBufPanicFileSize, //5
188 EFBufPanicFileHandle,
192 EFBufPanicFileNameLen, //10
195 EFBufPanicNextReadFilePos,
196 EFBufPanicNextReadFilePosHits,
197 EFBufPanicFileBlockSize, //15
198 EFBufPanicRwDataLength,
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
212 static void PanicFileBuf64(TInt aLine, TFileBufPanic64 aPanicCode)
216 TDateTime dt = time.DateTime();
218 tbuf.Format(_L("%02d:%02d:%02d.%06d"), dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond());
221 _LIT(KFormat,"**%S:RFileBuf64 panic %d, at line(%d)");
222 buf.Format(KFormat, &tbuf, aPanicCode, aLine);
224 User::Panic(KPanicCategory, aPanicCode);
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.
233 @see PanicFileBuf64()
236 #define __FBUF64_ASSERT(expr, panicCode) (void)(!(expr) ? ::PanicFileBuf64(__LINE__, panicCode) : void(0))
240 #define __FILEBUF64_INVARIANT() void(0)
242 #define __FBUF64_ASSERT(expr, panicCode) void(0)
246 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
247 /////////////////////////// RFileBuf64 /////////////////////////////////////////////////////////////////////
248 //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
251 Initializes RFileBuf64 data members with their default values.
253 @param aMinCapacity Minimal file buffer size (capacity) in bytes.
255 @panic FBuf64 1 In _DEBUG mode - aMinCapacity is 0 or negative.
257 RFileBuf64::RFileBuf64(TInt aMinCapacity) :
258 iCapacity(aMinCapacity),
260 iReadAheadSize(RFileBuf64::KDefaultReadAheadSize),
263 __FBUF64_ASSERT(aMinCapacity > 0, EFBufPanicCapacity);
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.
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
275 @param aFileMode The mode in which the file is opened. See TFileMode for details.
277 @return KErrNone if successful, otherwise one of the other system-wide error codes.
280 @see RFile64::Create()
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).
285 TInt RFileBuf64::Create(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
287 __FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
288 __FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
290 TInt err = DoPreInit();
293 err = iFile.Create(aFs, aFileName, aFileMode);
295 PROFILE_CREATE(aFileName, err);
296 return DoPostInit(err);
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.
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
307 @param aFileMode The mode in which the file is opened. See TFileMode for details.
309 @return KErrNone if successful, otherwise one of the other system-wide error codes.
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).
317 TInt RFileBuf64::Open(RFs& aFs, const TDesC& aFileName, TUint aFileMode)
319 __FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
320 __FBUF64_ASSERT(aFileName.Length() > 0, EFBufPanicFileNameLen);
322 TInt err = DoPreInit();
325 err = iFile.Open(aFs, aFileName, aFileMode);
327 PROFILE_OPEN(aFileName, err);
328 return DoPostInit(err);
332 Initializes the RFileBuf64 object and creates and opens a temporary file with unique name that will be accessed through
333 RFileBuf64 public interface.
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
340 @param aFileMode The mode in which the file is opened. The access mode is
341 automatically set to EFileWrite. See TFileMode for details.
343 @return KErrNone if successful, otherwise one of the other system-wide error codes.
348 @panic FBuf64 7 In _DEBUG mode - Invalid aFs object (null file session handle).
350 TInt RFileBuf64::Temp(RFs& aFs, const TDesC& aPath, TFileName& aFileName, TUint aFileMode)
352 __FBUF64_ASSERT(aFs.Handle() != 0, EFBufPanicFsHandle);
354 TInt err = DoPreInit();
357 err = iFile.Temp(aFs, aPath, aFileName, aFileMode);
359 PROFILE_TEMP(aFileName, err);
360 return DoPostInit(err);
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().
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
375 @return KErrNone if successful, otherwise one of the other system-wide error codes.
378 @see RFile64::AdoptFromClient()
379 @see KMaxMessageArguments
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)
385 TInt RFileBuf64::AdoptFromClient(const RMessage2& aMsg, TInt aFsIndex, TInt aFileIndex)
387 __FBUF64_ASSERT(aMsg.Handle() != 0, EFBufPanicMsgHandle);
388 __FBUF64_ASSERT(aFsIndex >= 0 && aFsIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
389 __FBUF64_ASSERT(aFileIndex >= 0 && aFileIndex < KMaxMessageArguments, EFBufPanicMsgIndex);
391 TInt err = DoPreInit();
394 err = iFile.AdoptFromClient(aMsg, aFsIndex, aFileIndex);
396 PROFILE_ADOPT(KNullDesC, err);
397 return DoPostInit(err);
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.
406 @see RFileBuf64::Flush()
408 void RFileBuf64::Close()
410 if(iBase != 0 && iFile.SubSessionHandle() != 0)
412 (void)DoFileWrite2();
421 Calculates and sets optimal read-ahead buffer size.
422 aBlockSize and aReadRecBufSize values are retrieved by the caller from the file system.
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.
433 @param aBlockSize The size of a file block in bytes
434 @param aReadRecBufSize The recommended buffer size for optimised reading performance
436 @return The new read-ahead value
438 @see TVolumeIOParamInfo
440 TInt RFileBuf64::SetReadAheadSize(TInt aBlockSize, TInt aReadRecBufSize)
442 __FILEBUF64_INVARIANT();
443 if((aReadRecBufSize & (aReadRecBufSize - 1)) == 0 && aReadRecBufSize > RFileBuf64::KDefaultReadAheadSize)
445 iReadAheadSize = aReadRecBufSize > iCapacity ? iCapacity : aReadRecBufSize;
447 else if((aBlockSize & (aBlockSize - 1)) == 0 && aBlockSize > RFileBuf64::KDefaultReadAheadSize)
449 iReadAheadSize = aBlockSize > iCapacity ? iCapacity : aBlockSize;
451 __FILEBUF64_INVARIANT();
452 return iReadAheadSize;
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.
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
467 @return KErrNone if successful, otherwise one of the other system-wide error codes.
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.
472 @see RFileBuf64::Invariant()
474 TInt RFileBuf64::Read(TInt64 aFilePos, TDes8& aDes)
476 __FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
477 __FILEBUF64_INVARIANT();
479 //0. The output buffer max len is 0
480 if(aDes.MaxLength() == 0)
482 __FILEBUF64_INVARIANT();
485 //1. Initialize the "iFileSize" if it is not initialized yet
486 TInt err = DoFileSize();
489 __FILEBUF64_INVARIANT();
492 //2. Optimize the buffer capacity
493 TInt len = aDes.MaxLength();
494 if((err = DoSetCapacity(len)) != KErrNone)
498 //3. Too big "read" request - read directly from the file
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);
507 err = iFile.Read(aFilePos, aDes);
508 PROFILE_READ(aFilePos, aDes.Size(), err);
510 __FILEBUF64_INVARIANT();
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)
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))
520 TInt blocklen = Min(len, (iFilePos + iLength - aFilePos));
521 outptr = Mem::Copy(outptr, iBase + (aFilePos - iFilePos), blocklen);
523 aFilePos += blocklen;
525 //2. Perform a read-ahead operation
528 //Write the pending data if the iDirty flag is set, otherwise preserve the buffer content.
529 err = DoFileWrite1(aFilePos);
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);
542 iNextReadFilePos = aFilePos + len;
543 len -= ptr2.Length();
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)
551 ++iNextReadFilePosHits;
553 TInt maxReadAhead = iReadAheadSize * (1 << iNextReadFilePosHits);
554 TInt align = (aFilePos + len + maxReadAhead) & (iReadAheadSize - 1);
555 TInt readahead = maxReadAhead - align;
558 // if read-ahead doesn't cross block boundary do it all
559 readahead = maxReadAhead;
561 TPtr8 ptr(iBase, Min(iCapacity, (len + readahead)));
562 err = iFile.Read(aFilePos, ptr);
563 PROFILE_READ(aFilePos, ptr.Size(), err);
567 iLength = ptr.Length();
568 iNextReadFilePos = iFilePos + iLength;
580 aDes.SetLength(aDes.MaxLength() - len);
581 __FILEBUF64_INVARIANT();
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.
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.
594 @param aData The descriptor from which binary data is written. The function writes
595 the entire contents of aData to the file.
597 @return KErrNone if successful, otherwise one of the other system-wide error codes.
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.
602 @see RFileBuf64::Invariant()
604 TInt RFileBuf64::Write(TInt64 aFilePos, const TDesC8& aData)
606 __FBUF64_ASSERT(aFilePos >= 0, EFBufPanicFilePos);
607 __FILEBUF64_INVARIANT();
608 if(aData.Length() == 0)
610 __FILEBUF64_INVARIANT();
613 TInt err = DoFileSize();
616 __FILEBUF64_INVARIANT();
619 if((err = DoSetCapacity(aData.Length())) != KErrNone)
623 DoDiscardBufferedReadData();
624 const TUint8* data = aData.Ptr();
625 for(TInt len = aData.Length(); len > 0 && err == KErrNone;)
627 //1. The new write pos is before the buffered file pos
628 if(aFilePos < iFilePos)
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))
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
643 //The "aFilePos" is too far before the "iFilePos". Write the buffer and associate the new pos with the buffer
645 err = DoFileWrite2(aFilePos);
648 //2. The new write pos is after the associated file pos + the data length.
649 else if(aFilePos > (iFilePos + iLength))
651 if(aFilePos > iFileSize) //Beyond the end of the file
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));
661 //Beyond the end of the file and not in the buffer - write the buffer to the file.
663 err = DoFileWrite2(aFilePos);
667 //Within the file, not in the buffer - write the buffer and associate the new file pos with the buffer
669 err = DoFileWrite2(aFilePos);
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
677 if (iFilePos+iCapacity == aFilePos) //The buffer is full. The new position to write is the end of the buffer.
679 err = DoFileWrite2(aFilePos);
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));
694 __FILEBUF64_INVARIANT();
699 Gets the current file size.
701 @param aFileSize On return, the size of the file in bytes.
703 @return KErrNone if successful, otherwise one of the other system-wide error codes.
705 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
707 @see RFileBuf64::Invariant()
709 TInt RFileBuf64::Size(TInt64& aFileSize)
711 __FILEBUF64_INVARIANT();
712 TInt err = DoFileSize();
715 aFileSize = iFileSize;
717 __FILEBUF64_INVARIANT();
724 If the size of the file is reduced, data may be lost from the end of the file.
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
733 2. If the file was not opened for writing, an error is returned.
735 @param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
737 @return KErrNone if successful, otherwise one of the other system-wide error codes.
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.
742 @see RFileBuf64::Invariant()
744 TInt RFileBuf64::SetSize(TInt64 aFileSize)
746 __FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
747 __FILEBUF64_INVARIANT();
748 return DoSetFileSize(aFileSize);
752 Writes the pending data and then flushes the file.
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.
759 @return KErrNone if successful, otherwise one of the other system-wide error codes.
761 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
763 @see RFileBuf64::Invariant()
765 TInt RFileBuf64::Flush()
767 __FILEBUF64_INVARIANT();
768 return DoFileFlush();
772 Gets information about the drive on which this file resides.
774 @param aDriveNumber On return, the drive number.
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.
780 @return KErrNone if successful, otherwise one of the other system-wide error codes.
782 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
784 @see RFileBuf64::Invariant()
786 TInt RFileBuf64::Drive(TInt& aDriveNumber, TDriveInfo& aDriveInfo) const
788 __FILEBUF64_INVARIANT();
789 return iFile.Drive(aDriveNumber, aDriveInfo);
793 Initializes RFileBuf64 data members with their initial values.
794 Allocates memory for the file buffer.
796 @return KErrNone if successful,
797 KErrNoMemory out of memory;
799 TInt RFileBuf64::DoPreInit()
802 iReadAheadSize = RFileBuf64::KDefaultReadAheadSize;
803 iBase = static_cast <TUint8*> (User::Alloc(iCapacity));
804 return iBase ? KErrNone : KErrNoMemory;
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.
812 @param aInitErr The result of the performed before the call RFileBuf64 initialization.
814 @return KErrNone if successful, otherwise one of the other system-wide error codes.
816 TInt RFileBuf64::DoPostInit(TInt aInitErr)
818 if(aInitErr != KErrNone)
827 Discards the content of the RFileBuf64 object returning it to the state as if it has just been created.
829 void RFileBuf64::DoDiscard()
833 iFileSize = KFileSizeNotSet;
835 iNextReadFilePos = KNextReadFilePosNotSet;
836 iNextReadFilePosHits = 0;
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.
844 @return KErrNone if successful, otherwise one of the other system-wide error codes.
846 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
848 @see RFileBuf64::Invariant()
850 TInt RFileBuf64::DoFileSize()
852 __FILEBUF64_INVARIANT();
853 if(iFileSize != KFileSizeNotSet)
855 __FILEBUF64_INVARIANT();
858 TInt err = iFile.Size(iFileSize);
859 PROFILE_SIZE(iFileSize, err);
866 iRealFileSize = iFileSize;
868 __FILEBUF64_INVARIANT();
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.
877 @param aFileSize The new size of the file, in bytes. This value must not be negative, otherwise the function raises a panic.
879 @return KErrNone if successful, otherwise one of the other system-wide error codes.
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.
884 @see RFileBuf64::Invariant()
886 TInt RFileBuf64::DoSetFileSize(TInt64 aFileSize)
888 __FBUF64_ASSERT(aFileSize >= 0, EFBufPanicFileSize);
889 __FILEBUF64_INVARIANT();
890 if(aFileSize < iFilePos)
895 //If the new file size is "in" the buffer then change the "iLength"
896 else if(aFileSize < (iFilePos + iLength))
898 iLength = aFileSize - iFilePos;
900 TInt err = iFile.SetSize(aFileSize);
901 PROFILE_SETSIZE(aFileSize, err);
908 iFileSize = aFileSize;
909 iRealFileSize = aFileSize;
911 __FILEBUF64_INVARIANT();
916 Writes the pending data and flushes the file.
918 @return KErrNone if successful, otherwise one of the other system-wide error codes.
920 See RFileBuf64::Invariant() for possible panics that may occur when this method is called.
922 @see RFileBuf64::Invariant()
924 TInt RFileBuf64::DoFileFlush()
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.
930 __FILEBUF64_INVARIANT();
940 __FILEBUF64_INVARIANT();
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.
949 @return KErrNone if successful, otherwise one of the other system-wide error codes.
951 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
953 @see RFileBuf64::DoFileWrite1()
954 @see RFileBuf64::DoFileWrite2()
955 @see RFileBuf64::Invariant()
957 TInt RFileBuf64::DoFileWrite()
959 __FILEBUF64_INVARIANT();
962 __FILEBUF64_INVARIANT();
965 TPtrC8 data(iBase, iLength);
967 if(iFilePos > iRealFileSize )
969 err = DoSetFileSize(iFileSize);
973 err = iFile.Write(iFilePos, data);
975 PROFILE_WRITE(iFilePos, iLength, err);
978 iRealFileSize = iFileSize;
984 __FILEBUF64_INVARIANT();
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.
998 @param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
999 the aNewFilePos value.
1001 @return KErrNone if successful, otherwise one of the other system-wide error codes.
1003 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
1005 @panic FBuf64 4 In _DEBUG mode - negative aNewFilePos value.
1007 @see RFileBuf64::Read()
1008 @see RFileBuf64::DoFileWrite()
1009 @see RFileBuf64::DoFileWrite2()
1010 @see RFileBuf64::Invariant()
1012 TInt RFileBuf64::DoFileWrite1(TInt64 aNewFilePos)
1014 __FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
1015 __FILEBUF64_INVARIANT();
1016 TInt err = KErrNone;
1019 err = DoFileWrite();
1022 iFilePos = aNewFilePos;
1027 __FILEBUF64_INVARIANT();
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;
1045 @param aNewFilePos If the buffer is successfully written to the file the iFilePos data member will be initialized with
1046 the aNewFilePos value.
1048 @return KErrNone if successful, otherwise one of the other system-wide error codes.
1050 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
1052 @panic FBuf64 4 In _DEBUG mode - negative aNewFilePos value.
1054 @see RFileBuf64::Write()
1055 @see RFileBuf64::DoFileWrite()
1056 @see RFileBuf64::DoFileWrite1()
1057 @see RFileBuf64::Invariant()
1059 TInt RFileBuf64::DoFileWrite2(TInt64 aNewFilePos)
1061 __FBUF64_ASSERT(aNewFilePos >= 0, EFBufPanicFilePos);
1062 __FILEBUF64_INVARIANT();
1063 TInt err = KErrNone;
1066 err = DoFileWrite();
1070 iFilePos = aNewFilePos;
1074 __FILEBUF64_INVARIANT();
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.
1089 @see RFileBuf64::Write()
1091 See RFileBuf64::Invariant() for other possible panics that may occur when this method is called.
1093 void RFileBuf64::DoDiscardBufferedReadData()
1095 __FILEBUF64_INVARIANT();
1096 if(!iDirty && iLength > 0)
1100 iNextReadFilePos = KNextReadFilePosNotSet;
1101 iNextReadFilePosHits = 0;
1103 __FILEBUF64_INVARIANT();
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.
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.
1117 TInt RFileBuf64::DoSetCapacity(TInt aRwDataLength)
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.
1124 __FBUF64_ASSERT(aRwDataLength > 0, EFBufPanicRwDataLength);
1125 __FILEBUF64_INVARIANT();
1128 __FILEBUF64_INVARIANT();
1131 if((aRwDataLength & (aRwDataLength - 1)) != 0 || aRwDataLength < KDefaultPageSize)
1133 __FILEBUF64_INVARIANT();
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)
1144 TUint8* newBase = static_cast <TUint8*> (User::ReAlloc(iBase, newBufCapacity));
1147 __FILEBUF64_INVARIANT();
1148 return KErrNoMemory;
1151 iCapacity = newBufCapacity;
1152 //Adjust the initial read-ahead size to be multiple of the page size.
1153 if((iReadAheadSize % pageSize) != 0)
1155 TInt q = iReadAheadSize / pageSize;
1156 iReadAheadSize = q != 0 ? pageSize * q : pageSize;
1160 __FILEBUF64_INVARIANT();
1167 RFileBuf64 invariant. Called in _DEBUG mode at the beginning and before the end of every RFileBuf64 method
1168 (except the init/destroy methods).
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.
1181 void RFileBuf64::Invariant() const
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);