Update contrib.
1 // Copyright (c) 1998-2009 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.
17 #include "S32FILEBUFSIZE.H"
19 //#define MAP_ROM_FILES
22 //#define SIMULATE_PARTIAL_WRITE
27 _LIT(KTraceWrite,"RFile::Write [%d,%d)\n");
28 #define _TRACE_WRITE(p,l) RDebug::Print(KTraceWrite,(p),(p)+(l))
29 _LIT(KTraceRead,"RFile::Read [%d,%d)\n");
30 #define _TRACE_READ(p,l) RDebug::Print(KTraceRead,(p),(p)+(l))
31 _LIT(KTraceSize,"RFile::Size\n");
32 #define _TRACE_SIZE() RDebug::Print(KTraceSize)
33 _LIT(KTraceFlush,"RFile::Flush\n");
34 #define _TRACE_FLUSH() RDebug::Print(KTraceFlush)
35 _LIT(KTraceSetSize,"RFile::SetSize %d\n");
36 #define _TRACE_SETSIZE(s) RDebug::Print(KTraceSetSize,(s))
38 #define _TRACE_WRITE(p,l)
39 #define _TRACE_READ(p,l)
41 #define _TRACE_FLUSH()
42 #define _TRACE_SETSIZE(s)
45 EXPORT_C RFileBuf::RFileBuf()
46 : iBase(NULL),iSize(KDefaultFileBufSize),iWLim(NULL)
47 /** Constructs the object with a default intermediate buffer size.
49 The size of the intermediate buffer is the value of the constant KDefaultFileBufSize. */
52 EXPORT_C RFileBuf::RFileBuf(TInt aSize)
53 : iBase(NULL),iSize(aSize),iWLim(NULL)
54 /** Constructs the object with the specified intermediate buffer size.
56 If the intermediate buffer size is zero, then the class provides an MStreamBuf
57 interface to unbuffered file I/O.
59 @param aSize The size of the intermediate buffer. */
62 RFileBuf::RFileBuf(TCapture<RFileBuf> aCapture)
64 // Take over from the buffer wrapped inside aCapture.
67 RFileBuf& buf=aCapture.Object();
70 SetBuf(ERead,buf.Ptr(ERead),buf.End(ERead));
71 SetBuf(EWrite,buf.Ptr(EWrite),buf.End(EWrite));
72 SetLimit(EWrite,buf.Limit(EWrite));
74 buf.SetBuf(ERead|EWrite,NULL,NULL);
77 SetPos(ERead,buf.Pos(ERead));
78 SetPos(EWrite,buf.Pos(EWrite));
82 EXPORT_C void RFileBuf::Reset()
83 /** Frees the intermediate buffer.
85 If there is any read data in the intermediate buffer, then the function reverts
86 the read position within the stream.
88 The intermediate buffer must not contain any outstanding write data, otherwise
89 the function raises a STORE-File 6 panic. */
91 __ASSERT_ALWAYS(Span(EWrite)==0,Panic(EFileWriteOutstanding));
92 MovePos(ERead,Lag(ERead));
96 EXPORT_C TInt RFileBuf::Open(RFs& aFs,const TDesC& aName,TUint aFileMode)
97 /** Opens the specified file and attaches it to this stream buffer.
99 If the file cannot be opened, then it is not attached to this stream buffer.
101 @param aFs Handle to a file server session through which the file is opened.
102 @param aName The name of the file to be opened.
103 @param aFileMode The mode in which the file is to be accessed. The mode is
104 defined by the TFileMode type.
105 @return KErrNone, if successful, otherwise one of the other system wide error
112 TInt r=file.Open(aFs,aName,aFileMode);
118 EXPORT_C TInt RFileBuf::Create(RFs& aFs,const TDesC& aName,TUint aFileMode)
119 /** Creates a file with the specified name and attaches it to this stream buffer.
121 The file must not already exist.
123 If the file cannot be created and opened, then it is not attached to this
126 @param aFs Handle to a file server session through which the file is created.
127 @param aName The name of the file to be created.
128 @param aFileMode The mode in which the file is to be accessed. The mode is
129 defined by the TFileMode type.
130 @return KErrNone, if successful, otherwise one of the other system wide error
137 TInt r=file.Create(aFs,aName,aFileMode);
143 EXPORT_C TInt RFileBuf::Replace(RFs& aFs,const TDesC& aName,TUint aFileMode)
144 /** Replaces the file with the specified name and attaches it to this stream buffer.
146 If there is an existing file with the same name, then this function overwrites
147 it. If the file does not already exist, it is created.
149 If the file cannot be replaced, then it is not attached to this stream buffer.
151 @param aFs Handle to a file server session through which the file is replaced.
152 @param aName The name of the file to be replaced.
153 @param aFileMode The mode in which the file is to be accessed. The mode is
154 defined by the TFileMode type.
155 @return KErrNone, if successful, otherwise one of the other system wide error
162 TInt r=file.Replace(aFs,aName,aFileMode);
168 EXPORT_C TInt RFileBuf::Temp(RFs& aFs,const TDesC& aPath,TFileName& aName,TUint aFileMode)
169 /** Creates and opens a temporary file with a unique name and attaches it to this
172 @param aFs Handle to a file server session through which the file is created.
173 @param aPath The directory in which the file is created.
174 @param aName On return, contains the full path and file name of the file. The
175 filename is guaranteed to be unique within the specified directory.
176 @param aFileMode The mode in which the file is to be accessed. The mode is
177 defined by the TFileMode type.
183 TInt r=file.Temp(aFs,aPath,aName,aFileMode);
189 EXPORT_C void RFileBuf::Attach(RFile& aFile,TInt aPos)
190 /** Attaches the specified file to this stream buffer.
192 The function also re-sets the intermediate buffer's read and write marks to
193 the beginning of the intermediate buffer and sets the read and write stream
194 positions to the specified offset within the file.
196 @param aFile The file to be attached.
197 @param aPos The offset within the file to which the read and write stream positions
198 are set. By default this is zero.
202 __ASSERT_ALWAYS(Span(EWrite)==0,Panic(EFileWriteOutstanding));
204 SetBuf(ERead|EWrite,base,base);
207 SetPos(ERead|EWrite,aPos);
211 EXPORT_C void RFileBuf::Close()
212 /** Writes any outstanding data from the intermediate buffer before freeing the
213 intermediate buffer and closing the attached file. */
215 TInt lag=Span(EWrite);
218 _TRACE_WRITE(Pos(EWrite),lag);
219 File().Write(Pos(EWrite),TPtrC8(iBase,lag));
225 EXPORT_C void RFileBuf::SetSizeL(TInt aSize)
226 /** Changes the size of the file attached to this buffer to the specified value.
228 Writes any outstanding data from the intermediate buffer to the stream hosted
229 by the file. Any data in the intermediate buffer that would lie beyond the
230 end of the truncated file, is not written.
232 @param aSize The new size of the file. */
235 TInt pos=Pos(EWrite);
236 TInt excess=aSize-pos;
238 FileWriteL(base,Min(excess,Span(EWrite)),0);
239 MovePos(ERead,Lag(ERead));
240 SetBuf(ERead,base,base);
242 _TRACE_SETSIZE(aSize);
243 TInt r=File().SetSize(aSize);
251 SetPos(EWrite,Min(pos+Lag(EWrite), aSize));
252 SetBuf(EWrite,base,base);
256 EXPORT_C TInt RFileBuf::UnderflowL(TInt aMaxLength)
258 // Fill the buffer's read area.
261 __ASSERT_DEBUG(Avail(ERead)==0,User::Invariant());
265 #if defined(MAP_ROM_FILES)
267 return 0; // memory mapped ROM file
269 TInt err = File().Seek(ESeekAddress,pos);
271 { // memory map a ROM file into the read zone
275 SetBuf(ERead,base,base+len);
282 TInt lag=Pos(ERead)-Pos(EWrite);
283 TInt span=Span(EWrite);
284 if (lag>=0&&lag<span)
286 SetBuf(ERead,base+lag,base+span);
287 MovePos(ERead,span-lag);
291 FileWriteL(base,Span(EWrite));
292 SetBuf(EWrite,base,base);
294 // Align file position with file 'blocks' when possible
295 TInt align = (Pos(ERead) + aMaxLength + KMaxFileBufReadAhead) & (KFileBufBlockSize-1);
296 TInt readahead = KMaxFileBufReadAhead - align;
299 // if read-ahead doesn't cross block boundary do it all
300 readahead = KMaxFileBufReadAhead;
302 TInt len=FileReadL(base, Min(iSize, aMaxLength+readahead));
304 SetBuf(ERead,base,base+len);
308 EXPORT_C void RFileBuf::OverflowL()
310 // Set up the buffer's write area.
313 __ASSERT_DEBUG(Avail(EWrite)==0,User::Invariant());
317 MovePos(ERead,Lag(ERead));
318 SetBuf(ERead,base,base);
320 __ASSERT_DEBUG(Lag(EWrite)==Span(EWrite),User::Invariant());
321 FileWriteL(base,Lag(EWrite),0);
322 SetBuf(EWrite,base,base+iSize);
325 EXPORT_C void RFileBuf::DoRelease()
327 // Release all resources, losing any outstanding data.
334 EXPORT_C void RFileBuf::DoSynchL()
336 // Synchronise this buffer with its file, giving up on outstanding writes in case of failure.
340 #if defined(MAP_ROM_FILES)
341 if (base!=NULL) // do not do this for memory mapped ROM files
344 MovePos(ERead,Lag(ERead));
345 TInt span=Span(EWrite);
346 TInt rewind=span-Lag(EWrite);
347 SetBuf(ERead|EWrite,base,base);
349 FileWriteL(base,span,rewind);
351 TInt handle = File().SubSessionHandle();
355 TInt r=File().Flush();
356 if (r!=KErrNone&&r!=KErrAccessDenied)
363 EXPORT_C TInt RFileBuf::DoReadL(TAny* aPtr,TInt aMaxLength)
365 // Read direct from file if asked to transfer more than a bufferful.
368 __ASSERT_DEBUG(aMaxLength>=0,Panic(EFileReadLengthNegative));
369 __ASSERT_DEBUG(aMaxLength>0,Panic(EFileReadNoTransfer));
370 TInt avail=Avail(ERead);
371 __ASSERT_DEBUG(avail>=0&&Avail(EWrite)>=0,User::Invariant());
374 TInt len=Min(aMaxLength,avail);
375 TUint8* ptr=Ptr(ERead);
376 aPtr=Mem::Copy(aPtr,ptr,len);
377 SetPtr(ERead,ptr+len);
380 return len; // that's it
382 __ASSERT_DEBUG(Avail(ERead)==0,User::Invariant());
383 if (aMaxLength<iSize)
384 return avail+TStreamBuf::DoReadL(aPtr,aMaxLength);
387 FileWriteL(base,Span(EWrite));
388 SetBuf(ERead|EWrite,base,base);
389 return avail+FileReadL(aPtr,aMaxLength);
392 EXPORT_C TInt RFileBuf::DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
394 // Read up to aMaxLength bytes asynchronously.
397 //#pragma message( __FILE__ " : 'RFileBuf::DoReadL(TDes8&,TInt,TRequestStatus&)' not implemented" )
398 __ASSERT_DEBUG(aMaxLength<=aDes.MaxLength(),Panic(EFileReadBeyondEnd));
399 aDes.SetLength(DoReadL((TUint8*)aDes.Ptr(),aMaxLength));
400 TRequestStatus* stat=&aStatus;
401 User::RequestComplete(stat,KErrNone);
405 EXPORT_C void RFileBuf::DoWriteL(const TAny* aPtr,TInt aLength)
407 // Write direct to file if asked to transfer more than a bufferful.
410 __ASSERT_DEBUG(aLength>=0,Panic(EFileWriteLengthNegative));
411 __ASSERT_DEBUG(aLength>0,Panic(EFileWriteNoTransfer));
412 TInt avail=Avail(EWrite);
413 __ASSERT_DEBUG(Avail(ERead)>=0&&avail>=0,User::Invariant());
416 TInt len=Min(aLength,avail);
417 SetPtr(EWrite,Mem::Copy(Ptr(EWrite),aPtr,len));
422 aPtr=(TUint8*)aPtr+len;
424 __ASSERT_DEBUG(Avail(EWrite)==0,User::Invariant());
426 TStreamBuf::DoWriteL(aPtr,aLength);
429 __ASSERT_DEBUG(Lag(EWrite)==Span(EWrite),User::Invariant());
431 FileWriteL(base,Lag(EWrite),0);
432 MovePos(ERead,Lag(ERead));
433 SetBuf(ERead|EWrite,base,base);
434 FileWriteL(aPtr,aLength,0);
438 EXPORT_C TInt RFileBuf::DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
440 // Write up to aMaxLength bytes asynchronously.
443 //#pragma message( __FILE__ " : 'RFileBuf::DoWriteL(const TDesC8&,TInt,TRequestStatus&)' not implemented" )
444 __ASSERT_DEBUG(aMaxLength<=aDes.Length(),Panic(EFileWriteBeyondEnd));
445 DoWriteL(aDes.Ptr(),aMaxLength);
446 TRequestStatus* stat=&aStatus;
447 User::RequestComplete(stat,KErrNone);
451 EXPORT_C TStreamPos RFileBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
453 // Position the mark(s) indicated by aMark at anOffset from aLocation.
461 case EStreamBeginning:
464 anOffset+=Mark(aMark);
470 Panic(EFileLocationInvalid);
479 else if (anOffset>end)
485 __ASSERT_ALWAYS(!(aMark&~(ERead|EWrite)),Panic(EFileMarkInvalid));
488 TInt lag=anOffset-Pos(ERead);
489 #if defined(MAP_ROM_FILES)
490 if (base==NULL&&End(ERead)!=NULL) // memory mapped
491 SetPtr(ERead,End(ERead)+lag);
494 if (lag>=base-End(ERead)&&lag<=0)
495 SetPtr(ERead,End(ERead)+lag);
498 SetPos(ERead,anOffset);
499 SetBuf(ERead,base,base);
504 TInt lag=anOffset-Pos(EWrite);
505 TInt span=Span(EWrite);
506 if (lag>=0&&lag<=span)
508 SetLimit(EWrite,base+span);
509 SetPtr(EWrite,base+lag);
513 FileWriteL(base,span,0);
514 SetPos(EWrite,anOffset);
515 SetBuf(EWrite,base,base);
519 return TStreamPos(anOffset);
522 TUint8* RFileBuf::AllocL()
524 // Allocate space and return a pointer to this buffer's buffer space
527 __ASSERT_DEBUG(iBase==NULL && iSize>0,User::Invariant());
529 TUint8* base=(TUint8*)User::AllocL(iSize);
531 SetBuf(ERead|EWrite,base,base);
535 void RFileBuf::Free()
537 // Free this buffer's buffer space.
542 SetBuf(ERead|EWrite,NULL,NULL);
545 void RFileBuf::SetPos(TMark aMark,TInt aPos)
547 // Set the file position for the mark(s) indicated by aMark
550 __ASSERT_ALWAYS(!(aMark&~(ERead|EWrite)),Panic(EFileMarkInvalid));
557 TInt RFileBuf::FileReadL(TAny* aPtr,TInt aMaxLength)
559 // Read from the file at the current read position.
562 __ASSERT_DEBUG(aMaxLength>=0,Panic(EFileReadLengthNegative));
566 TPtr8 des((TUint8*)aPtr,aMaxLength);
568 __LEAVE_IF_ERROR(File().Read(pos,des));
569 TInt len=des.Length();
570 _TRACE_READ(pos,len);
573 iExt=pos; // end-of-file encountered
578 void RFileBuf::FileWriteL(const TAny* aPtr,TInt aLength)
580 // Write to the file at the current write position
581 // Use write buffer status to determine the rewind
584 FileWriteL(aPtr,aLength,Span(EWrite)-Lag(EWrite));
587 void RFileBuf::FileWriteL(const TAny* aPtr,TInt aLength,TInt aRewind)
589 // Write to the file at the current write position.
590 // Rewind write position after write
593 __ASSERT_DEBUG(aLength>=0,Panic(EFileWriteLengthNegative));
599 TInt pos=Pos(EWrite);
600 _TRACE_WRITE(pos,aLength);
601 #ifdef SIMULATE_PARTIAL_WRITE
602 TPtrC8 ptr((TUint8*)aPtr,aLength);
603 TInt partial = aLength >> 1;
606 __LEAVE_IF_ERROR(File().Write(pos, ptr.Left(partial)));
608 __LEAVE_IF_ERROR(File().Write(pos + partial, ptr.Mid(partial)));
611 #if defined SYSLIBS_TEST && defined _DEBUG
612 const TInt KDefaultMediaBlockSize = 512;
613 TInt startSectorAddr = pos & ~(KDefaultMediaBlockSize - 1);
614 TInt endSectorAddr = (pos + aLength - 1) & ~(KDefaultMediaBlockSize - 1);
615 if(startSectorAddr != endSectorAddr && aLength < KDefaultMediaBlockSize)
617 TInt len1 = startSectorAddr + KDefaultMediaBlockSize - pos;
618 TInt len2 = aLength - len1;
619 __LEAVE_IF_ERROR(File().Write(pos, TPtrC8((TUint8*)aPtr, len1)));
620 __LEAVE_IF_ERROR(File().Write(pos + len1, TPtrC8((TUint8*)aPtr + len1, len2)));
624 __LEAVE_IF_ERROR(File().Write(pos,TPtrC8((TUint8*)aPtr,aLength)));
627 __LEAVE_IF_ERROR(File().Write(pos,TPtrC8((TUint8*)aPtr,aLength)));
634 SetPos(EWrite,pos-aRewind);
637 TInt RFileBuf::EndL()
639 // Determine the end of the file.
646 __LEAVE_IF_ERROR(File().Size(ext));
649 return Max(ext,Reach(EWrite));
652 TInt RFileBuf::Mark(TMark aMark) const
654 // Return the position of the mark indicated by aMark.
660 __ASSERT_ALWAYS(aMark==EWrite,Panic(EFileMarkInvalid));