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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
18 /** Synchronises the stream buffer with the stream, returning any error.
20 In effect, this ensures that buffered data is delivered to the stream.
22 This function calls SynchL() inside a TRAPD harness and returns the leave
23 code if a leave occurs.
25 @return KErrNone, if successful; otherwise one of the other system wide error
27 @see MStreamBuf::SynchL()
28 @see MStreamBuf::DoSynchL() */
29 EXPORT_C TInt MStreamBuf::Synch()
35 /** Closes the stream buffer.
37 This function attempts to synchronise buffered data with the stream before
38 freeing any resources. All errors are ignored.
40 @see MStreamBuf::Synch()
41 @see MStreamBuf::Release() */
42 EXPORT_C void MStreamBuf::Close()
48 /** Puts a cleanup item for this object onto the cleanup stack.
50 This allows allocated resources to be cleaned up if a subsequent leave occurs. */
51 EXPORT_C void MStreamBuf::PushL()
53 CleanupReleasePushL(*this);
56 /** Reads data, asynchronously, from the stream buffer into the specified descriptor;
57 request completion is guaranteed, even if request initiation fails.
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.
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
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)
72 return Read(aDes,aDes.MaxLength(),aStatus);
75 /** Reads data, asynchronously, from the stream buffer into the specified descriptor;
76 request completion is guaranteed, even if request initiation fails.
78 The function calls the virtual function DoReadL(TDes8&,TInt,TRequestStatus&)
79 to implement this behaviour.
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
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
88 @see MStreamBuf::DoReadL() */
89 EXPORT_C TInt MStreamBuf::Read(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
92 TRAPD(r,len=DoReadL(aDes,aMaxLength,aStatus));
95 TRequestStatus* stat=&aStatus;
96 User::RequestComplete(stat,r);
101 /** Reads data, asynchronously, from the stream buffer into the specified descriptor.
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.
107 If the function leaves, then no read request will have been initiated.
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)
118 return DoReadL(aDes,aDes.MaxLength(),aStatus);
121 /** Reads data from the stream buffer into the specified data sink.
123 The function uses the virtual function DoReadL(MStreamInput&,TStreamTransfer)
124 to implement this behaviour.
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)
131 return aMaxLength-DoReadL(anInput,TStreamTransfer(aMaxLength)).Left();
134 /** Writes data, asynchronously, from the specified descriptor into the stream buffer;
135 request completion is guaranteed, even if request initiation fails.
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.
141 @param aDes The source descriptor for the data to be written into the stream
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)
151 return Write(aDes,aDes.Length(),aStatus);
155 // Write up to aMaxLength bytes with guaranteed completion.
157 EXPORT_C TInt MStreamBuf::Write(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
160 TRAPD(r,len=DoWriteL(aDes,aMaxLength,aStatus));
163 TRequestStatus* stat=&aStatus;
164 User::RequestComplete(stat,r);
169 /** Writes data, asynchronously, from the specified descriptor into the stream buffer.
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.
175 If the function leaves, then no write request will have been initiated.
177 @param aDes The source descriptor for the data to be written into the stream
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)
187 return DoWriteL(aDes,aDes.Length(),aStatus);
190 /** Writes data into the stream buffer from the specified data source.
192 The function calls the virtual function DoWriteL(MStreamOutput&,TStreamTransfer)
193 to implement this behaviour.
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)
200 return aMaxLength-DoWriteL(anOutput,TStreamTransfer(aMaxLength)).Left();
203 /** Frees resources before abandoning the stream buffer.
205 It is called by Release().
207 This implementation is empty, but classes derived from MStreamBuf can provide
208 their own implementation, if necessary.
210 @see MStreamBuf::Release() */
211 EXPORT_C void MStreamBuf::DoRelease()
215 /** Synchronises the stream buffer with the stream, leaving if any error occurs.
217 In effect, this ensures that buffered data is delivered to the stream.
219 It is called by SynchL().
221 This implementation is empty, but classes derived from MStreamBuf can provide
222 their own implementation, if necessary.
224 @see MStreamBuf::SynchL() */
225 EXPORT_C void MStreamBuf::DoSynchL()
229 #pragma BullseyeCoverage off
232 // Cannot read from this stream buffer.
234 EXPORT_C TInt MStreamBuf::DoReadL(TAny*,TInt)
236 Panic(EStreamCannotRead);
241 // Cannot write to this stream buffer.
243 EXPORT_C void MStreamBuf::DoWriteL(const TAny*,TInt)
245 Panic(EStreamCannotWrite);
249 // This stream buffer does not support seeking.
251 EXPORT_C TStreamPos MStreamBuf::DoSeekL(TMark,TStreamLocation,TInt)
253 Panic(EStreamCannotSeek);
254 TStreamPos streamPos(-1);
258 #pragma BullseyeCoverage on
260 /** Reads data from the stream buffer into the specified descriptor.
262 This function is called by ReadL(TDes8&,TInt,TRequestStatus&).
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.
268 In addition, the read operation itself uses the DoReadL(TAny*,TInt) variant.
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)
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);
291 /** Reads data from the stream into the specified data sink.
293 It is called by ReadL(MStreamInput&,TStreamTransfer).
295 This implementation calls the sink's ReadFromL() function, which performs
296 the read (transfer) operation.
298 This implementation of DoReadL() is called for streams that do not have buffering
299 capabilities, and that are derived directly from this class.
301 @param anInput The target data sink.
302 @param aTransfer A stream transfer object defining the amount of data available
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)
308 return anInput.ReadFromL(*this,aTransfer);
311 /** Writes data from the specified descriptor into this stream buffer.
313 This function is called by WriteL(const TDesC8&,TInt,TRequestStatus&).
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.
319 In addition, the write operation itself uses the DoWriteL(TAny*,TInt) variant.
321 @param aDes The source descriptor for the data to be written into the stream
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)
334 __ASSERT_DEBUG(aMaxLength<=aDes.Length(),Panic(EStreamWriteBeyondEnd));
335 DoWriteL(aDes.Ptr(),aMaxLength);
336 TRequestStatus* stat=&aStatus;
337 User::RequestComplete(stat,KErrNone);
342 // Default implementation turning around to anOutput.
344 EXPORT_C TStreamTransfer MStreamBuf::DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer)
346 return anOutput.WriteToL(*this,aTransfer);
349 /** Sets the pointers that mark out the read and write areas within the intermediate
351 EXPORT_C TStreamBuf::TStreamBuf()
352 : iRPtr(NULL),iREnd(NULL),iWPtr(NULL),iWEnd(NULL)
356 /** Sets the start and end points of the read and/or the write area within the intermediate
359 A start point is always within an area; an end point is always the first byte
360 beyond the end of an area.
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)
372 __ASSERT_ALWAYS(!(anArea&~(ERead|EWrite)),Panic(EStreamAreaInvalid));
374 SetBuf(ERead,aPtr,anEnd);
376 SetBuf(EWrite,aPtr,anEnd);
379 /** Sets the start point of the read and/or the write area within the intermediate
382 A start point is always within an area.
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)
393 __ASSERT_ALWAYS(!(anArea&~(ERead|EWrite)),Panic(EStreamAreaInvalid));
401 // Set the end pointer for the buffer area(s) indicated by anArea.
403 EXPORT_C void TStreamBuf::SetEnd(TArea anArea,TUint8* anEnd)
405 __ASSERT_ALWAYS(!(anArea&~(ERead|EWrite)),Panic(EStreamAreaInvalid));
409 SetEnd(EWrite,anEnd);
412 /** Gets the current start point of the read or write area within the intermediate
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
427 __ASSERT_ALWAYS(anArea==EWrite,Panic(EStreamAreaInvalid));
431 /** Gets the current end point of the read or write area within the intermediate
434 An end point is always the first byte beyond the end of an area.
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
446 __ASSERT_ALWAYS(anArea==EWrite,Panic(EStreamAreaInvalid));
450 /** Gets the number of bytes available in the read or write area within the intermediate
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
463 __ASSERT_ALWAYS(anArea==EWrite,Panic(EStreamAreaInvalid));
464 return Avail(EWrite);
467 /** Reads data from the intermediate buffer into the specified memory location.
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.
472 This implementation overrides the one supplied by the base class MStreamBuf,
473 and is called by, MStreamBuf::ReadL(TAny*,TInt).
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)
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());
494 __ASSERT_DEBUG(avail==Avail(ERead),Panic(EStreamUnderflowInBreach));
495 __ASSERT_DEBUG(left>0&&avail>0,User::Invariant());
497 TInt len=Min(left,avail);
498 TUint8* ptr=Ptr(ERead);
499 aPtr=Mem::Copy(aPtr,ptr,len);
500 SetPtr(ERead,ptr+len);
503 return aMaxLength; // that's it
507 avail=UnderflowL(left);
509 __ASSERT_DEBUG(avail==0&&Avail(ERead)==0,Panic(EStreamUnderflowInBreach));
510 return aMaxLength-left;
513 /** Reads data from the intermediate buffer and, if necessary, any remaining data
514 from the stream to the specified target stream input object.
516 It is called by ReadL(MStreamInput&,TStreamTransfer).
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.
523 This implementation is called for streams that have buffering capabilities
524 and are derived from this class.
526 @param anInput The target stream input object.
527 @param aTransfer A stream transfer object defining the amount of data available
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)
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)];
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);
548 aTransfer=anInput.ReadFromL(*this,aTransfer);
552 /** Writes data from the specified memory location into the intermediate buffer.
554 The function calls the virtual function OverfLowL() to give concrete implementations
555 the chance to forward the intermediate buffer content to its destination.
557 This implementation overrides the one supplied by the base class MStreamBuf,
558 and is called by MStreamBuf::WriteL(const TAny*,TInt).
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)
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());
578 __ASSERT_DEBUG(avail>0,Panic(EStreamOverflowInBreach));
579 __ASSERT_DEBUG(aLength>0,User::Invariant());
581 TInt len=Min(aLength,avail);
582 SetPtr(EWrite,Mem::Copy(Ptr(EWrite),aPtr,len));
587 aPtr=(TUint8*)aPtr+len;
597 // Default implementation filling the buffer before turning around to anOutput.
599 EXPORT_C TStreamTransfer TStreamBuf::DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer)
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)];
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);
615 aTransfer=anOutput.WriteToL(*this,aTransfer);