os/persistentdata/persistentstorage/store/USTRM/US_BUF.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 1998-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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "US_STD.H"
    17 
    18 /** Synchronises the stream buffer with the stream, returning any error.
    19 
    20 In effect, this ensures that buffered data is delivered to the stream.
    21 
    22 This function calls SynchL() inside a TRAPD harness and returns the leave 
    23 code if a leave occurs.
    24 
    25 @return KErrNone, if successful; otherwise one of the other system wide error 
    26 codes.
    27 @see MStreamBuf::SynchL()
    28 @see MStreamBuf::DoSynchL() */
    29 EXPORT_C TInt MStreamBuf::Synch()
    30 	{
    31 	TRAPD(r,SynchL());
    32 	return r;
    33 	}
    34 
    35 /** Closes the stream buffer.
    36 
    37 This function attempts to synchronise buffered data with the stream before 
    38 freeing any resources. All errors are ignored.
    39 
    40 @see MStreamBuf::Synch()
    41 @see MStreamBuf::Release() */
    42 EXPORT_C void MStreamBuf::Close()
    43 	{
    44 	Synch();
    45 	Release();
    46 	}
    47 
    48 /** Puts a cleanup item for this object onto the cleanup stack.
    49 
    50 This allows allocated resources to be cleaned up if a subsequent leave occurs. */
    51 EXPORT_C void MStreamBuf::PushL()
    52 	{
    53 	CleanupReleasePushL(*this);
    54 	}
    55 
    56 /** Reads data, asynchronously, from the stream buffer into the specified descriptor; 
    57 request completion is guaranteed, even if request initiation fails.
    58 
    59 The function calls the virtual function DoReadL(TDes8&,TInt,TRequestStatus&) 
    60 to implement this behaviour. The maximum number of bytes to be read is the 
    61 value of the maximum length of the descriptor.
    62 
    63 @param aDes The target descriptor for the data read from the stream buffer.
    64 @param aStatus The request status that indicates the completion status of this 
    65 asynchronous request.
    66 @return The maximum number of bytes to be read, as used in this request. This 
    67 value can be different to the maximum length of the descriptor; this is dependent 
    68 on the implementation.
    69 @see MStreamBuf::DoReadL() */
    70 EXPORT_C TInt MStreamBuf::Read(TDes8& aDes,TRequestStatus& aStatus)
    71 	{
    72 	return Read(aDes,aDes.MaxLength(),aStatus);
    73 	}
    74 
    75 /** Reads data, asynchronously, from the stream buffer into the specified descriptor; 
    76 request completion is guaranteed, even if request initiation fails.
    77 
    78 The function calls the virtual function DoReadL(TDes8&,TInt,TRequestStatus&) 
    79 to implement this behaviour.
    80 
    81 @param aDes The target descriptor for the data read from the stream buffer.
    82 @param aMaxLength The maximum number of bytes to be read.
    83 @param aStatus The request status that indicates the completion status of this 
    84 asynchronous request.
    85 @return The maximum number of bytes to be read, as used in this request. This 
    86 can be different to the value supplied in aMaxLength; this is dependent on 
    87 the implementation.
    88 @see MStreamBuf::DoReadL() */
    89 EXPORT_C TInt MStreamBuf::Read(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
    90 	{
    91 	TInt len=0;
    92 	TRAPD(r,len=DoReadL(aDes,aMaxLength,aStatus));
    93 	if (r!=KErrNone)
    94 		{
    95 		TRequestStatus* stat=&aStatus;
    96 		User::RequestComplete(stat,r);
    97 		}
    98 	return len;
    99 	}
   100 
   101 /** Reads data, asynchronously, from the stream buffer into the specified descriptor.
   102 
   103 The function calls the virtual function DoReadL(TDes8&,TInt,TRequestStatus&) 
   104 to implement this behaviour. The maximum number of bytes to be read is the 
   105 maximum length of the descriptor.
   106 
   107 If the function leaves, then no read request will have been initiated.
   108 
   109 @param aDes The target descriptor for the data read from the stream buffer.
   110 @param aStatus The request status that indicates the completion status of this 
   111 asynchronous request.
   112 @return The maximum number of bytes to be read, as used in this request. This 
   113 value can be different to the maximum length of the descriptor; this is dependent 
   114 on the implementation.
   115 @see MStreamBuf::DoReadL() */
   116 EXPORT_C TInt MStreamBuf::ReadL(TDes8& aDes,TRequestStatus& aStatus)
   117 	{
   118 	return DoReadL(aDes,aDes.MaxLength(),aStatus);
   119 	}
   120 
   121 /** Reads data from the stream buffer into the specified data sink.
   122 
   123 The function uses the virtual function DoReadL(MStreamInput&,TStreamTransfer) 
   124 to implement this behaviour.
   125 
   126 @param anInput The data sink which is the target for the read operation.
   127 @param aMaxLength The maximum amount of data available to be read.
   128 @return The amount of data that was not consumed. */
   129 EXPORT_C TInt MStreamBuf::ReadL(MStreamInput& anInput,TInt aMaxLength)
   130 	{
   131 	return aMaxLength-DoReadL(anInput,TStreamTransfer(aMaxLength)).Left();
   132 	}
   133 
   134 /** Writes data, asynchronously, from the specified descriptor into the stream buffer; 
   135 request completion is guaranteed, even if request initiation fails.
   136 
   137 The function calls the virtual function DoWriteL(const TDesC8&,TInt,TRequestStatus&) 
   138 to implement this behaviour. The maximum number of bytes to be written is 
   139 the value of the maximum length of the descriptor.
   140 
   141 @param aDes The source descriptor for the data to be written into the stream 
   142 buffer.
   143 @param aStatus The request status that indicates the completion status of this 
   144 asynchronous request.
   145 @return The maximum number of bytes to be written, as used in this request. 
   146 This can be different to the value supplied in aMaxLength; this is dependent 
   147 on the implementation.
   148 @see MStreamBuf::DoWriteL() */
   149 EXPORT_C TInt MStreamBuf::Write(const TDesC8& aDes,TRequestStatus& aStatus)
   150 	{
   151 	return Write(aDes,aDes.Length(),aStatus);
   152 	}
   153 
   154 //
   155 // Write up to aMaxLength bytes with guaranteed completion.
   156 //
   157 EXPORT_C TInt MStreamBuf::Write(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
   158 	{
   159 	TInt len=0;
   160 	TRAPD(r,len=DoWriteL(aDes,aMaxLength,aStatus));
   161 	if (r!=KErrNone)
   162 		{
   163 		TRequestStatus* stat=&aStatus;
   164 		User::RequestComplete(stat,r);
   165 		}
   166 	return len;
   167 	}
   168 
   169 /** Writes data, asynchronously, from the specified descriptor into the stream buffer.
   170 
   171 The function calls the virtual function DoWriteL(const TDesC8&,TInt,TRequestStatus&) 
   172 to implement this behaviour. The maximum number of bytes to be written is 
   173 the value of the maximum length of the descriptor.
   174 
   175 If the function leaves, then no write request will have been initiated.
   176 
   177 @param aDes The source descriptor for the data to be written into the stream 
   178 buffer.
   179 @param aStatus The request status that indicates the completion status of this 
   180 asynchronous request.
   181 @return The maximum number of bytes to be written, as used in this request. 
   182 This can be different to the maximum length of the descriptor; this is dependent 
   183 on the implementation.
   184 @see MStreamBuf::DoWriteL() */
   185 EXPORT_C TInt MStreamBuf::WriteL(const TDesC8& aDes,TRequestStatus& aStatus)
   186 	{
   187 	return DoWriteL(aDes,aDes.Length(),aStatus);
   188 	}
   189 
   190 /** Writes data into the stream buffer from the specified data source.
   191 
   192 The function calls the virtual function DoWriteL(MStreamOutput&,TStreamTransfer) 
   193 to implement this behaviour.
   194 
   195 @param anOutput The data source for the write operation.
   196 @param aMaxLength The maximum amount of data available to be written.
   197 @return The amount of data that was not consumed. */
   198 EXPORT_C TInt MStreamBuf::WriteL(MStreamOutput& anOutput,TInt aMaxLength)
   199 	{
   200 	return aMaxLength-DoWriteL(anOutput,TStreamTransfer(aMaxLength)).Left();
   201 	}
   202 
   203 /** Frees resources before abandoning the stream buffer.
   204 
   205 It is called by Release().
   206 
   207 This implementation is empty, but classes derived from MStreamBuf can provide 
   208 their own implementation, if necessary.
   209 
   210 @see MStreamBuf::Release() */
   211 EXPORT_C void MStreamBuf::DoRelease()
   212 	{	
   213 	}
   214 
   215 /** Synchronises the stream buffer with the stream, leaving if any error occurs.
   216 
   217 In effect, this ensures that buffered data is delivered to the stream.
   218 
   219 It is called by SynchL().
   220 
   221 This implementation is empty, but classes derived from MStreamBuf can provide 
   222 their own implementation, if necessary.
   223 
   224 @see MStreamBuf::SynchL() */
   225 EXPORT_C void MStreamBuf::DoSynchL()
   226 	{
   227 	}
   228 
   229 #pragma BullseyeCoverage off
   230 
   231 //
   232 // Cannot read from this stream buffer.
   233 //
   234 EXPORT_C TInt MStreamBuf::DoReadL(TAny*,TInt)
   235 	{
   236 	Panic(EStreamCannotRead);
   237 	return TInt();
   238 	}
   239 
   240 //
   241 // Cannot write to this stream buffer.
   242 //
   243 EXPORT_C void MStreamBuf::DoWriteL(const TAny*,TInt)
   244 	{
   245 	Panic(EStreamCannotWrite);
   246 	}
   247 
   248 //
   249 // This stream buffer does not support seeking.
   250 //
   251 EXPORT_C TStreamPos MStreamBuf::DoSeekL(TMark,TStreamLocation,TInt)
   252 	{
   253 	Panic(EStreamCannotSeek);
   254 	TStreamPos streamPos(-1);
   255 	return streamPos;
   256 	}
   257 
   258 #pragma BullseyeCoverage on
   259 
   260 /** Reads data from the stream buffer into the specified descriptor.
   261 
   262 This function is called by ReadL(TDes8&,TInt,TRequestStatus&).
   263 
   264 This implementation deals with the request synchronously, and completes the 
   265 request with KErrNone. Other implementations may choose to deal with this 
   266 in a true asynchronous manner.
   267 
   268 In addition, the read operation itself uses the DoReadL(TAny*,TInt) variant.
   269 
   270 @param aDes The target descriptor for the data read from the stream buffer. 
   271 On return, the length of the descriptor is set to the number of bytes read 
   272 from the stream buffer.
   273 @param aMaxLength The maximum number of bytes to be read. This value must not 
   274 be greater than the maximum length of the descriptor, otherwise the function 
   275 raises a STORE-Stream 2 panic.
   276 @param aStatus The request status that indicates the completion status of this 
   277 asynchronous request.
   278 @return The maximum number of bytes to be read, as used in this request. This 
   279 implementation uses, and returns, the value supplied in aMaxLength. Other 
   280 implementations may choose to use a different value.
   281 @see MStreamBuf::ReadL() */
   282 EXPORT_C TInt MStreamBuf::DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
   283 	{
   284 	__ASSERT_DEBUG(aMaxLength<=aDes.MaxLength(),Panic(EStreamReadBeyondEnd));
   285 	aDes.SetLength(DoReadL((TUint8*)aDes.Ptr(),aMaxLength));
   286 	TRequestStatus* stat=&aStatus;
   287 	User::RequestComplete(stat,KErrNone);
   288 	return aMaxLength;
   289 	}
   290 
   291 /** Reads data from the stream into the specified data sink.
   292 
   293 It is called by ReadL(MStreamInput&,TStreamTransfer).
   294 
   295 This implementation calls the sink's ReadFromL() function, which performs 
   296 the read (transfer) operation.
   297 
   298 This implementation of DoReadL() is called for streams that do not have buffering 
   299 capabilities, and that are derived directly from this class.
   300 
   301 @param anInput The target data sink.
   302 @param aTransfer A stream transfer object defining the amount of data available 
   303 to be read.
   304 @return A stream transfer object defining the amount of data that was not consumed.
   305 @see MStreamInput::ReadFromL() */
   306 EXPORT_C TStreamTransfer MStreamBuf::DoReadL(MStreamInput& anInput,TStreamTransfer aTransfer)
   307 	{
   308 	return anInput.ReadFromL(*this,aTransfer);
   309 	}
   310 
   311 /** Writes data from the specified descriptor into this stream buffer.
   312 
   313 This function is called by WriteL(const TDesC8&,TInt,TRequestStatus&).
   314 
   315 This implementation deals with the request synchronously, and completes the 
   316 request with KErrNone. Other implementations may choose to deal with this 
   317 in a true asynchronous manner.
   318 
   319 In addition, the write operation itself uses the DoWriteL(TAny*,TInt) variant.
   320 
   321 @param aDes The source descriptor for the data to be written into the stream 
   322 buffer.
   323 @param aMaxLength The number of bytes to be written. This value must not be 
   324 greater than the maximum length of the descriptor, otherwise the function 
   325 raises a STORE-Stream 6 panic.
   326 @param aStatus The request status that indicates the completion status of this 
   327 asynchronous request.
   328 @return The maximum number of bytes to be written, as used in this request. 
   329 This implementation uses, and returns, the value supplied in aMaxLength. Other 
   330 implementations may choose to use a different value.
   331 @see MStreamBuf::WriteL() */
   332 EXPORT_C TInt MStreamBuf::DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
   333 	{
   334 	__ASSERT_DEBUG(aMaxLength<=aDes.Length(),Panic(EStreamWriteBeyondEnd));
   335 	DoWriteL(aDes.Ptr(),aMaxLength);
   336 	TRequestStatus* stat=&aStatus;
   337 	User::RequestComplete(stat,KErrNone);
   338 	return aMaxLength;
   339 	}
   340 
   341 //
   342 // Default implementation turning around to anOutput.
   343 //
   344 EXPORT_C TStreamTransfer MStreamBuf::DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer)
   345 	{
   346 	return anOutput.WriteToL(*this,aTransfer);
   347 	}
   348 
   349 /** Sets the pointers that mark out the read and write areas within the intermediate 
   350 buffer to null. */
   351 EXPORT_C TStreamBuf::TStreamBuf()
   352 	: iRPtr(NULL),iREnd(NULL),iWPtr(NULL),iWEnd(NULL)
   353 	{
   354 	}
   355 
   356 /** Sets the start and end points of the read and/or the write area within the intermediate 
   357 buffer.
   358 
   359 A start point is always within an area; an end point is always the first byte 
   360 beyond the end of an area.
   361 
   362 @param anArea The areas within the intermediate buffer for which the start 
   363 and end points are to be set. These can be the read area and/or the write 
   364 area, as indicated by the ERead and EWrite bits. Only these bits can be set, 
   365 otherwise the function raises a STORE-Stream 17 panic.
   366 @param aPtr The start point.
   367 @param anEnd The end point.
   368 @see MStreamBuf::TRead
   369 @see MStreamBuf::TWrite */
   370 EXPORT_C void TStreamBuf::SetBuf(TArea anArea,TUint8* aPtr,TUint8* anEnd)
   371 	{
   372 	__ASSERT_ALWAYS(!(anArea&~(ERead|EWrite)),Panic(EStreamAreaInvalid));
   373 	if (anArea&ERead)
   374 		SetBuf(ERead,aPtr,anEnd);
   375 	if (anArea&EWrite)
   376 		SetBuf(EWrite,aPtr,anEnd);
   377 	}
   378 
   379 /** Sets the start point of the read and/or the write area within the intermediate 
   380 buffer.
   381 
   382 A start point is always within an area.
   383 
   384 @param anArea The areas within the intermediate buffer for which the start 
   385 point is to be set. These can be the read area and/or the write area, as indicated 
   386 by the ERead and EWrite bits. Only these bits can be set, otherwise the function 
   387 raises a STORE-Stream 17 panic.
   388 @param aPtr The start point.
   389 @see MStreamBuf::TRead
   390 @see MStreamBuf::TWrite */
   391 EXPORT_C void TStreamBuf::SetPtr(TArea anArea,TUint8* aPtr)
   392 	{
   393 	__ASSERT_ALWAYS(!(anArea&~(ERead|EWrite)),Panic(EStreamAreaInvalid));
   394 	if (anArea&ERead)
   395 		SetPtr(ERead,aPtr);
   396 	if (anArea&EWrite)
   397 		SetPtr(EWrite,aPtr);
   398 	}
   399 
   400 //
   401 // Set the end pointer for the buffer area(s) indicated by anArea.
   402 //
   403 EXPORT_C void TStreamBuf::SetEnd(TArea anArea,TUint8* anEnd)
   404 	{
   405 	__ASSERT_ALWAYS(!(anArea&~(ERead|EWrite)),Panic(EStreamAreaInvalid));
   406 	if (anArea&ERead)
   407 		SetEnd(ERead,anEnd);
   408 	if (anArea&EWrite)
   409 		SetEnd(EWrite,anEnd);
   410 	}
   411 
   412 /** Gets the current start point of the read or write area within the intermediate 
   413 buffer.
   414 
   415 @param anArea The area within the intermediate buffer for which the start 
   416 point is to be fetched. This can be either the read area or the write area, 
   417 as indicated by the ERead and EWrite bits. Only one of these can be set, otherwise 
   418 the function raises a STORE-Stream 17 panic.
   419 @return The start point.
   420 @see MStreamBuf::TRead
   421 @see MStreamBuf::TWrite */
   422 EXPORT_C TUint8* TStreamBuf::Ptr(TArea anArea) const
   423 	{
   424 	if (anArea==ERead)
   425 		return Ptr(ERead);
   426 //
   427 	__ASSERT_ALWAYS(anArea==EWrite,Panic(EStreamAreaInvalid));
   428 	return Ptr(EWrite);
   429 	}
   430 
   431 /** Gets the current end point of the read or write area within the intermediate 
   432 buffer.
   433 
   434 An end point is always the first byte beyond the end of an area.
   435 
   436 @param anArea The area within the intermediate buffer for which the end point 
   437 is to be fetched. This can be either the read area or the write area, as indicated 
   438 by the ERead and EWrite bits. Only one of these can be set, otherwise the 
   439 function raises a STORE-Stream 17 panic.
   440 @return The end point. */
   441 EXPORT_C TUint8* TStreamBuf::End(TArea anArea) const
   442 	{
   443 	if (anArea==ERead)
   444 		return End(ERead);
   445 //
   446 	__ASSERT_ALWAYS(anArea==EWrite,Panic(EStreamAreaInvalid));
   447 	return End(EWrite);
   448 	}
   449 
   450 /** Gets the number of bytes available in the read or write area within the intermediate 
   451 buffer.
   452 
   453 @param anArea The area within the intermediate buffer for which the number 
   454 of available bytes is to be fetched. This can be either the read area or the 
   455 write area, as indicated by the ERead and EWrite bits. Only one of these can 
   456 be set, otherwise the function raises a STORE-Stream 17 panic.
   457 @return The number of bytes available. */
   458 EXPORT_C TInt TStreamBuf::Avail(TArea anArea) const
   459 	{
   460 	if (anArea==ERead)
   461 		return Avail(ERead);
   462 //
   463 	__ASSERT_ALWAYS(anArea==EWrite,Panic(EStreamAreaInvalid));
   464 	return Avail(EWrite);
   465 	}
   466 
   467 /** Reads data from the intermediate buffer into the specified memory location.
   468 
   469 The function calls the virtual function UnderfLowL() to give concrete implementations 
   470 the chance to refill the intermediate buffer, and satisfy the caller's requirements.
   471 
   472 This implementation overrides the one supplied by the base class MStreamBuf, 
   473 and is called by, MStreamBuf::ReadL(TAny*,TInt).
   474 
   475 @param aPtr A pointer to the target memory location for the data read from 
   476 the intermediate buffer.
   477 @param aMaxLength The maximum number of bytes to be read.
   478 @return The number of bytes read. This may be less than the amount requested.
   479 @see MStreamBuf::ReadL()
   480 @see MStreamBuf::DoReadL() */
   481 EXPORT_C TInt TStreamBuf::DoReadL(TAny* aPtr,TInt aMaxLength)
   482 	{
   483 	__ASSERT_DEBUG(aMaxLength>=0,Panic(EStreamReadLengthNegative));
   484 	__ASSERT_DEBUG(aMaxLength>0,Panic(EStreamReadNoTransfer));
   485 	__ASSERT_DEBUG(Ptr(ERead)!=NULL||End(ERead)==NULL,Panic(EStreamCannotRead));
   486 	TInt left=aMaxLength;
   487 	TInt avail=Avail(ERead);
   488 	__ASSERT_DEBUG(avail>=0,User::Invariant());
   489 	if (avail==0)
   490 		goto underflow;
   491 //
   492 	do
   493 		{
   494 		__ASSERT_DEBUG(avail==Avail(ERead),Panic(EStreamUnderflowInBreach));
   495 		__ASSERT_DEBUG(left>0&&avail>0,User::Invariant());
   496 		{
   497 		TInt len=Min(left,avail);
   498 		TUint8* ptr=Ptr(ERead);
   499 		aPtr=Mem::Copy(aPtr,ptr,len);
   500 		SetPtr(ERead,ptr+len);
   501 		left-=len;
   502 		if (left==0)
   503 			return aMaxLength; // that's it
   504 		}
   505 //
   506 	underflow:
   507 		avail=UnderflowL(left);
   508 		} while (avail>0);
   509 	__ASSERT_DEBUG(avail==0&&Avail(ERead)==0,Panic(EStreamUnderflowInBreach));
   510 	return aMaxLength-left;
   511 	}
   512 
   513 /** Reads data from the intermediate buffer and, if necessary, any remaining data 
   514 from the stream to the specified target stream input object.
   515 
   516 It is called by ReadL(MStreamInput&,TStreamTransfer).
   517 
   518 The intermediate buffer is emptied first by calling the target stream input's 
   519 PushL() function, which performs the read from intermediate buffer operation. 
   520 Any remaining data is then read from the stream by calling the target stream 
   521 object's ReadFromL() function, which performs the read from stream operation.
   522 
   523 This implementation is called for streams that have buffering capabilities 
   524 and are derived from this class.
   525 
   526 @param anInput The target stream input object. 
   527 @param aTransfer A stream transfer object defining the amount of data available 
   528 to be written.
   529 @return The amount of data that was not consumed.
   530 @see MStreamInput::ReadFromL()
   531 @see MStreamInput::PushL() */
   532 EXPORT_C TStreamTransfer TStreamBuf::DoReadL(MStreamInput& anInput,TStreamTransfer aTransfer)
   533 	{
   534 	__ASSERT_DEBUG(aTransfer>0,Panic(EStreamReadNoTransfer));
   535 	__ASSERT_DEBUG(Ptr(ERead)!=NULL||End(ERead)==NULL,Panic(EStreamCannotRead));
   536 	__ASSERT_DEBUG(Avail(ERead)>=0,User::Invariant());
   537 	TInt len=aTransfer[Avail(ERead)];
   538 	if (len>0)
   539 		{
   540 		__DEBUG(TInt avail=Avail(ERead)); // may be pushing into this streambuf
   541 		TUint8* ptr=Ptr(ERead);
   542 		len=anInput.PushL(ptr,len);
   543 		__ASSERT_DEBUG(len>=0&&len<=aTransfer[avail]&&Ptr(ERead)==ptr&&Avail(ERead)>=avail,Panic(EStreamPushInBreach));
   544 		SetPtr(ERead,ptr+len);
   545 		aTransfer-=len;
   546 		}
   547 	if (aTransfer>0)
   548 		aTransfer=anInput.ReadFromL(*this,aTransfer);
   549 	return aTransfer;
   550 	}
   551 
   552 /** Writes data from the specified memory location into the intermediate buffer.
   553 
   554 The function calls the virtual function OverfLowL() to give concrete implementations 
   555 the chance to forward the intermediate buffer content to its destination.
   556 
   557 This implementation overrides the one supplied by the base class MStreamBuf, 
   558 and is called by MStreamBuf::WriteL(const TAny*,TInt).
   559 
   560 @param aPtr A pointer to the source memory location for the data to be written 
   561 to the intermediate buffer.
   562 @param aLength The number of bytes to be written.
   563 @return The number of bytes written.
   564 @see MStreamBuf::WriteL()
   565 @see MStreamBuf::DoWriteL() */
   566 EXPORT_C void TStreamBuf::DoWriteL(const TAny* aPtr,TInt aLength)
   567 	{
   568 	__ASSERT_DEBUG(aLength>=0,Panic(EStreamWriteLengthNegative));
   569 	__ASSERT_DEBUG(aLength>0,Panic(EStreamWriteNoTransfer));
   570 	__ASSERT_DEBUG(Ptr(EWrite)!=NULL||End(EWrite)==NULL,Panic(EStreamCannotWrite));
   571 	TInt avail=Avail(EWrite);
   572 	__ASSERT_DEBUG(avail>=0,User::Invariant());
   573 	if (avail==0)
   574 		goto overflow;
   575 //
   576 	for(;;)
   577 		{
   578 		__ASSERT_DEBUG(avail>0,Panic(EStreamOverflowInBreach));
   579 		__ASSERT_DEBUG(aLength>0,User::Invariant());
   580 		{
   581 		TInt len=Min(aLength,avail);
   582 		SetPtr(EWrite,Mem::Copy(Ptr(EWrite),aPtr,len));
   583 		aLength-=len;
   584 		if (aLength==0)
   585 			return; // done
   586 //
   587 		aPtr=(TUint8*)aPtr+len;
   588 		}
   589 //
   590 	overflow:
   591 		OverflowL();
   592 		avail=Avail(EWrite);
   593 		};
   594 	}
   595 
   596 //
   597 // Default implementation filling the buffer before turning around to anOutput.
   598 //
   599 EXPORT_C TStreamTransfer TStreamBuf::DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer)
   600 	{
   601 	__ASSERT_DEBUG(aTransfer>0,Panic(EStreamWriteNoTransfer));
   602 	__ASSERT_DEBUG(Ptr(EWrite)!=NULL||End(EWrite)==NULL,Panic(EStreamCannotWrite));
   603 	__ASSERT_DEBUG(Avail(EWrite)>=0,User::Invariant());
   604 	TInt len=aTransfer[Avail(EWrite)];
   605 	if (len>0)
   606 		{
   607 		__DEBUG(TInt avail=Avail(EWrite)); // may be pulling from this streambuf
   608 		TUint8* ptr=Ptr(EWrite);
   609 		len=anOutput.PullL(ptr,len);
   610 		__ASSERT_DEBUG(len>=0&&len<=aTransfer[avail]&&Ptr(EWrite)==ptr&&Avail(EWrite)>=avail,Panic(EStreamPullInBreach));
   611 		SetPtr(EWrite,ptr+len);
   612 		aTransfer-=len;
   613 		}
   614 	if (aTransfer>0)
   615 		aTransfer=anOutput.WriteToL(*this,aTransfer);
   616 	return aTransfer;
   617 	}
   618