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