os/persistentdata/persistentstorage/store/UFILE/UF_BUF.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "UF_STD.H"
    17 #include "S32FILEBUFSIZE.H"
    18 
    19 //#define MAP_ROM_FILES
    20 #ifdef _DEBUG
    21 //#define IO_TRACING
    22 //#define SIMULATE_PARTIAL_WRITE
    23 #endif
    24 
    25 #ifdef IO_TRACING
    26 #include <e32svr.h>
    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))
    37 #else
    38 #define _TRACE_WRITE(p,l)
    39 #define _TRACE_READ(p,l)
    40 #define _TRACE_SIZE()
    41 #define _TRACE_FLUSH()
    42 #define _TRACE_SETSIZE(s)
    43 #endif
    44 
    45 EXPORT_C RFileBuf::RFileBuf()
    46 	: iBase(NULL),iSize(KDefaultFileBufSize),iWLim(NULL)
    47 /** Constructs the object with a default intermediate buffer size.
    48 
    49 The size of the intermediate buffer is the value of the constant KDefaultFileBufSize. */
    50 	{}
    51 
    52 EXPORT_C RFileBuf::RFileBuf(TInt aSize)
    53 	: iBase(NULL),iSize(aSize),iWLim(NULL)
    54 /** Constructs the object with the specified intermediate buffer size.
    55 
    56 If the intermediate buffer size is zero, then the class provides an MStreamBuf 
    57 interface to unbuffered file I/O.
    58 
    59 @param aSize The size of the intermediate buffer. */
    60 	{}
    61 
    62 RFileBuf::RFileBuf(TCapture<RFileBuf> aCapture)
    63 //
    64 // Take over from the buffer wrapped inside aCapture.
    65 //
    66 	{
    67 	RFileBuf& buf=aCapture.Object();
    68 	iBase=buf.iBase;
    69 	iSize=buf.iSize;
    70 	SetBuf(ERead,buf.Ptr(ERead),buf.End(ERead));
    71 	SetBuf(EWrite,buf.Ptr(EWrite),buf.End(EWrite));
    72 	SetLimit(EWrite,buf.Limit(EWrite));
    73 	buf.iBase=NULL;
    74 	buf.SetBuf(ERead|EWrite,NULL,NULL);
    75 	iFile=buf.File();
    76 	buf.Detach();
    77 	SetPos(ERead,buf.Pos(ERead));
    78 	SetPos(EWrite,buf.Pos(EWrite));
    79 	iExt=buf.iExt;
    80 	}
    81 
    82 EXPORT_C void RFileBuf::Reset()
    83 /** Frees the intermediate buffer.
    84 
    85 If there is any read data in the intermediate buffer, then the function reverts 
    86 the read position within the stream.
    87 
    88 The intermediate buffer must not contain any outstanding write data, otherwise 
    89 the function raises a STORE-File 6 panic. */
    90 	{
    91 	__ASSERT_ALWAYS(Span(EWrite)==0,Panic(EFileWriteOutstanding));
    92 	MovePos(ERead,Lag(ERead));
    93 	Free();
    94 	}
    95 
    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.
    98 
    99 If the file cannot be opened, then it is not attached to this stream buffer.
   100 
   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 
   106 codes.
   107 @see Attach()
   108 @see TFileMode
   109 @see RFile */
   110 	{
   111 	RFile file;
   112 	TInt r=file.Open(aFs,aName,aFileMode);
   113 	if (r==KErrNone)
   114 		Attach(file);
   115 	return r;
   116 	}
   117 
   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.
   120 
   121 The file must not already exist.
   122 
   123 If the file cannot be created and opened, then it is not attached to this 
   124 stream buffer.
   125 
   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 
   131 codes.
   132 @see Attach()
   133 @see TFileMode
   134 @see RFile */
   135 	{
   136 	RFile file;
   137 	TInt r=file.Create(aFs,aName,aFileMode);
   138 	if (r==KErrNone)
   139 		Attach(file);
   140 	return r;
   141 	}
   142 
   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.
   145 
   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.
   148 
   149 If the file cannot be replaced, then it is not attached to this stream buffer.
   150 
   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 
   156 codes.
   157 @see Attach()
   158 @see TFileMode
   159 @see RFile */
   160 	{
   161 	RFile file;
   162 	TInt r=file.Replace(aFs,aName,aFileMode);
   163 	if (r==KErrNone)
   164 		Attach(file);
   165 	return r;
   166 	}
   167 
   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 
   170 stream buffer.
   171 
   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.
   178 @see Attach()
   179 @see TFileMode
   180 @see RFile */
   181 	{
   182 	RFile file;
   183 	TInt r=file.Temp(aFs,aPath,aName,aFileMode);
   184 	if (r==KErrNone)
   185 		Attach(file);
   186 	return r;
   187 	}
   188 
   189 EXPORT_C void RFileBuf::Attach(RFile& aFile,TInt aPos)
   190 /** Attaches the specified file to this stream buffer.
   191 
   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.
   195 
   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.
   199 @see Detach()
   200 @see Reattach() */
   201 	{
   202 	__ASSERT_ALWAYS(Span(EWrite)==0,Panic(EFileWriteOutstanding));
   203 	TUint8* base=iBase;
   204 	SetBuf(ERead|EWrite,base,base);
   205 	iFile=aFile;
   206 	aFile=RFile();
   207 	SetPos(ERead|EWrite,aPos);
   208 	iExt=-1;
   209 	}
   210 
   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. */
   214 	{
   215 	TInt lag=Span(EWrite);
   216 	if (lag>0)
   217 		{
   218 		_TRACE_WRITE(Pos(EWrite),lag);
   219 		File().Write(Pos(EWrite),TPtrC8(iBase,lag));
   220 		}
   221 	Free();
   222 	File().Close();
   223 	}
   224 
   225 EXPORT_C void RFileBuf::SetSizeL(TInt aSize)
   226 /** Changes the size of the file attached to this buffer to the specified value.
   227 
   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.
   231 
   232 @param aSize The new size of the file. */
   233 	{
   234 	TUint8* base=iBase;
   235 	TInt pos=Pos(EWrite);
   236 	TInt excess=aSize-pos;
   237 	if (excess>0)
   238 		FileWriteL(base,Min(excess,Span(EWrite)),0);
   239 	MovePos(ERead,Lag(ERead));
   240 	SetBuf(ERead,base,base);
   241 //
   242 	_TRACE_SETSIZE(aSize);
   243 	TInt r=File().SetSize(aSize);
   244 	if (r!=KErrNone)
   245 		{
   246 		SetPos(EWrite,pos);
   247 		iExt=-1;
   248 		__LEAVE(r);
   249 		}
   250 //
   251 	SetPos(EWrite,Min(pos+Lag(EWrite), aSize)); 
   252 	SetBuf(EWrite,base,base);
   253 	iExt=aSize;
   254 	}
   255 
   256 EXPORT_C TInt RFileBuf::UnderflowL(TInt aMaxLength)
   257 //
   258 // Fill the buffer's read area.
   259 //
   260 	{
   261 	__ASSERT_DEBUG(Avail(ERead)==0,User::Invariant());
   262 	TUint8* base=iBase;
   263 	if (base==NULL)
   264 		{
   265 #if defined(MAP_ROM_FILES)
   266 		if (Ptr(ERead))
   267 			return 0;		// memory mapped ROM file
   268 		TInt pos=0;
   269 		TInt err = File().Seek(ESeekAddress,pos);
   270 		if (err==KErrNone)
   271 			{				// memory map a ROM file into the read zone
   272 			base=(TUint8*)pos;
   273 			TInt len=EndL();
   274 			SetPos(ERead,len);
   275 			SetBuf(ERead,base,base+len);
   276 			return len;
   277 			}
   278 #endif
   279 		base=AllocL();
   280 		}
   281 
   282 	TInt lag=Pos(ERead)-Pos(EWrite);
   283 	TInt span=Span(EWrite);
   284 	if (lag>=0&&lag<span)
   285 		{
   286 		SetBuf(ERead,base+lag,base+span);
   287 		MovePos(ERead,span-lag);
   288 		return span-lag;
   289 		}
   290 
   291 	FileWriteL(base,Span(EWrite));
   292 	SetBuf(EWrite,base,base);
   293 	
   294 	// Align file position with file 'blocks' when possible
   295 	TInt align = (Pos(ERead) + aMaxLength + KMaxFileBufReadAhead) & (KFileBufBlockSize-1);
   296 	TInt readahead = KMaxFileBufReadAhead - align;
   297 	if(readahead < 0)
   298 		{
   299 		// if read-ahead doesn't cross block boundary do it all
   300 		readahead = KMaxFileBufReadAhead;	
   301 		}
   302 	TInt len=FileReadL(base, Min(iSize, aMaxLength+readahead));
   303 	
   304 	SetBuf(ERead,base,base+len);
   305 	return len;
   306 	}
   307 
   308 EXPORT_C void RFileBuf::OverflowL()
   309 //
   310 // Set up the buffer's write area.
   311 //
   312 	{
   313 	__ASSERT_DEBUG(Avail(EWrite)==0,User::Invariant());
   314 	TUint8* base=iBase;
   315 	if (base==NULL)
   316 		base=AllocL();
   317 	MovePos(ERead,Lag(ERead));
   318 	SetBuf(ERead,base,base);
   319 //
   320 	__ASSERT_DEBUG(Lag(EWrite)==Span(EWrite),User::Invariant());
   321 	FileWriteL(base,Lag(EWrite),0);
   322 	SetBuf(EWrite,base,base+iSize);
   323 	}
   324 
   325 EXPORT_C void RFileBuf::DoRelease()
   326 //
   327 // Release all resources, losing any outstanding data.
   328 //
   329 	{
   330 	Free();
   331 	File().Close();
   332 	}
   333 
   334 EXPORT_C void RFileBuf::DoSynchL()
   335 //
   336 // Synchronise this buffer with its file, giving up on outstanding writes in case of failure.
   337 //
   338 	{
   339 	TUint8* base=iBase;
   340 #if defined(MAP_ROM_FILES)
   341 	if (base!=NULL)	// do not do this for memory mapped ROM files
   342 #endif
   343 		{
   344 		MovePos(ERead,Lag(ERead));
   345 		TInt span=Span(EWrite);
   346 		TInt rewind=span-Lag(EWrite);
   347 		SetBuf(ERead|EWrite,base,base);
   348 		iExt=-1;
   349 		FileWriteL(base,span,rewind);
   350 		}
   351 	TInt handle = File().SubSessionHandle();
   352 	if (handle!=0)
   353 		{
   354 		_TRACE_FLUSH();
   355 		TInt r=File().Flush();
   356 		if (r!=KErrNone&&r!=KErrAccessDenied)
   357 			{
   358 			__LEAVE(r);
   359 			}
   360 		}
   361 	}
   362 
   363 EXPORT_C TInt RFileBuf::DoReadL(TAny* aPtr,TInt aMaxLength)
   364 //
   365 // Read direct from file if asked to transfer more than a bufferful.
   366 //
   367 	{
   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());
   372 	if (avail>0)
   373 		{
   374 		TInt len=Min(aMaxLength,avail);
   375 		TUint8* ptr=Ptr(ERead);
   376 		aPtr=Mem::Copy(aPtr,ptr,len);
   377 		SetPtr(ERead,ptr+len);
   378 		aMaxLength-=len;
   379 		if (aMaxLength==0)
   380 			return len; // that's it
   381 		}
   382 	__ASSERT_DEBUG(Avail(ERead)==0,User::Invariant());
   383 	if (aMaxLength<iSize)
   384 		return avail+TStreamBuf::DoReadL(aPtr,aMaxLength);
   385 //
   386 	TUint8* base=iBase;
   387 	FileWriteL(base,Span(EWrite));
   388 	SetBuf(ERead|EWrite,base,base);
   389 	return avail+FileReadL(aPtr,aMaxLength);
   390 	}
   391 
   392 EXPORT_C TInt RFileBuf::DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
   393 //
   394 // Read up to aMaxLength bytes asynchronously.
   395 //
   396 	{
   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);
   402 	return aMaxLength;
   403 	}
   404 
   405 EXPORT_C void RFileBuf::DoWriteL(const TAny* aPtr,TInt aLength)
   406 //
   407 // Write direct to file if asked to transfer more than a bufferful.
   408 //
   409 	{
   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());
   414 	if (avail>0)
   415 		{
   416 		TInt len=Min(aLength,avail);
   417 		SetPtr(EWrite,Mem::Copy(Ptr(EWrite),aPtr,len));
   418 		aLength-=len;
   419 		if (aLength==0)
   420 			return; // done
   421 //
   422 		aPtr=(TUint8*)aPtr+len;
   423 		}
   424 	__ASSERT_DEBUG(Avail(EWrite)==0,User::Invariant());
   425 	if (aLength<iSize)
   426 		TStreamBuf::DoWriteL(aPtr,aLength);
   427 	else
   428 		{
   429 		__ASSERT_DEBUG(Lag(EWrite)==Span(EWrite),User::Invariant());
   430 		TUint8* base=iBase;
   431 		FileWriteL(base,Lag(EWrite),0);
   432 		MovePos(ERead,Lag(ERead));
   433 		SetBuf(ERead|EWrite,base,base);
   434 		FileWriteL(aPtr,aLength,0);
   435 		}
   436 	}
   437 
   438 EXPORT_C TInt RFileBuf::DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
   439 //
   440 // Write up to aMaxLength bytes asynchronously.
   441 //
   442 	{
   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);
   448 	return aMaxLength;
   449 	}
   450 
   451 EXPORT_C TStreamPos RFileBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
   452 //
   453 // Position the mark(s) indicated by aMark at anOffset from aLocation.
   454 //
   455 	{
   456 	TUint8* base=iBase;
   457 	TInt end=EndL();
   458 //
   459 	switch (aLocation)
   460 		{
   461 	case EStreamBeginning:
   462 		break;
   463 	case EStreamMark:
   464 		anOffset+=Mark(aMark);
   465 		break;
   466 	case EStreamEnd:
   467 		anOffset+=end;
   468 		break;
   469 	default:
   470 		Panic(EFileLocationInvalid);
   471 		break;
   472 		}
   473 	TInt r=KErrNone;
   474 	if (anOffset<0)
   475 		{
   476 		anOffset=0;
   477 		r=KErrEof;
   478 		}
   479 	else if (anOffset>end)
   480 		{
   481 		anOffset=end;
   482 		r=KErrEof;
   483 		}
   484 //
   485 	__ASSERT_ALWAYS(!(aMark&~(ERead|EWrite)),Panic(EFileMarkInvalid));
   486 	if (aMark&ERead)
   487 		{
   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);
   492 		else
   493 #endif
   494 		if (lag>=base-End(ERead)&&lag<=0)
   495 			SetPtr(ERead,End(ERead)+lag);
   496 		else
   497 			{
   498 			SetPos(ERead,anOffset);
   499 			SetBuf(ERead,base,base);
   500 			}
   501 		}
   502 	if (aMark&EWrite)
   503 		{
   504 		TInt lag=anOffset-Pos(EWrite);
   505 		TInt span=Span(EWrite);
   506 		if (lag>=0&&lag<=span)
   507 			{
   508 			SetLimit(EWrite,base+span);
   509 			SetPtr(EWrite,base+lag);
   510 			}
   511 		else
   512 			{
   513 			FileWriteL(base,span,0);
   514 			SetPos(EWrite,anOffset);
   515 			SetBuf(EWrite,base,base);
   516 			}
   517 		}
   518 	__LEAVE_IF_ERROR(r);
   519 	return TStreamPos(anOffset);
   520 	}
   521 
   522 TUint8* RFileBuf::AllocL()
   523 //
   524 // Allocate space and return a pointer to this buffer's buffer space
   525 //
   526 	{
   527 	__ASSERT_DEBUG(iBase==NULL && iSize>0,User::Invariant());
   528 //
   529 	TUint8* base=(TUint8*)User::AllocL(iSize);
   530 	iBase=base;
   531 	SetBuf(ERead|EWrite,base,base);
   532 	return base;
   533 	}
   534 
   535 void RFileBuf::Free()
   536 //
   537 // Free this buffer's buffer space.
   538 //
   539 	{
   540 	User::Free(iBase);
   541 	iBase=NULL;
   542 	SetBuf(ERead|EWrite,NULL,NULL);
   543 	}
   544 
   545 void RFileBuf::SetPos(TMark aMark,TInt aPos)
   546 //
   547 // Set the file position for the mark(s) indicated by aMark
   548 //
   549 	{
   550 	__ASSERT_ALWAYS(!(aMark&~(ERead|EWrite)),Panic(EFileMarkInvalid));
   551 	if (aMark&ERead)
   552 		SetPos(ERead,aPos);
   553 	if (aMark&EWrite)
   554 		SetPos(EWrite,aPos);
   555 	}
   556 
   557 TInt RFileBuf::FileReadL(TAny* aPtr,TInt aMaxLength)
   558 //
   559 // Read from the file at the current read position.
   560 //
   561 	{
   562 	__ASSERT_DEBUG(aMaxLength>=0,Panic(EFileReadLengthNegative));
   563 	if (aMaxLength==0)
   564 		return 0;
   565 //
   566 	TPtr8 des((TUint8*)aPtr,aMaxLength);
   567 	TInt pos=Pos(ERead);
   568 	__LEAVE_IF_ERROR(File().Read(pos,des));
   569 	TInt len=des.Length();
   570 	_TRACE_READ(pos,len);
   571 	pos+=len;
   572 	if (len<aMaxLength)
   573 		iExt=pos; // end-of-file encountered
   574 	SetPos(ERead,pos);
   575 	return len;
   576 	}
   577 
   578 void RFileBuf::FileWriteL(const TAny* aPtr,TInt aLength)
   579 //
   580 // Write to the file at the current write position
   581 // Use write buffer status to determine the rewind
   582 //
   583 	{
   584 	FileWriteL(aPtr,aLength,Span(EWrite)-Lag(EWrite));
   585 	}
   586 
   587 void RFileBuf::FileWriteL(const TAny* aPtr,TInt aLength,TInt aRewind)
   588 //
   589 // Write to the file at the current write position.
   590 // Rewind write position after write
   591 //
   592 	{
   593 	__ASSERT_DEBUG(aLength>=0,Panic(EFileWriteLengthNegative));
   594 	if (aLength==0)
   595 		return;
   596 //
   597 	TInt ext=iExt;
   598 	iExt=-1;
   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;
   604 	if (partial)
   605 		{
   606 		__LEAVE_IF_ERROR(File().Write(pos, ptr.Left(partial)));
   607 		}
   608 	__LEAVE_IF_ERROR(File().Write(pos + partial, ptr.Mid(partial)));
   609 #else
   610 
   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)
   616 		{
   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)));
   621 		}
   622 	else
   623 		{
   624 		__LEAVE_IF_ERROR(File().Write(pos,TPtrC8((TUint8*)aPtr,aLength)));
   625 		}
   626 #else //SYSLIBS_TEST		
   627 	__LEAVE_IF_ERROR(File().Write(pos,TPtrC8((TUint8*)aPtr,aLength)));
   628 #endif//SYSLIBS_TEST
   629 
   630 #endif
   631 	pos+=aLength;
   632 	if (ext>=0)
   633 		iExt=Max(pos,ext);
   634 	SetPos(EWrite,pos-aRewind);
   635 	}
   636 
   637 TInt RFileBuf::EndL()
   638 //
   639 // Determine the end of the file.
   640 //
   641 	{
   642 	TInt ext=iExt;
   643 	if (ext<0)
   644 		{
   645 		_TRACE_SIZE();
   646 		__LEAVE_IF_ERROR(File().Size(ext));
   647 		iExt=ext;
   648 		}
   649 	return Max(ext,Reach(EWrite));
   650 	}
   651 
   652 TInt RFileBuf::Mark(TMark aMark) const
   653 //
   654 // Return the position of the mark indicated by aMark.
   655 //
   656 	{
   657 	if (aMark==ERead)
   658 		return Mark(ERead);
   659 //
   660 	__ASSERT_ALWAYS(aMark==EWrite,Panic(EFileMarkInvalid));
   661 	return Mark(EWrite);
   662 	}
   663