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