userlibandfileserver/fileserver/sfsrv/cl_ftext.cpp
author William Roberts <williamr@symbian.org>
Mon, 21 Dec 2009 16:15:43 +0000
changeset 3 9947e075979d
parent 0 a41df078684a
permissions -rw-r--r--
Merge improved comments

// Copyright (c) 1996-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
// f32\sfsrv\cl_ftext.cpp
// 
//

#include "cl_std.h"




EXPORT_C TFileText::TFileText()
/**
Default constructor.
*/
	{}




EXPORT_C void TFileText::Set(RFile& aFile)
/**
Sets the Unicode file to be read from, or written to.

This function must be called before 
Read(), Write() or Seek() can be used.

@param aFile The file to be used. Must be open.

@see TFileText::Read
@see TFileText::Write
@see TFileText::Seek
*/
	{
#ifndef SYMBIAN_ENABLE_64_BIT_FILE_SERVER_API
	iFile=aFile;
#else
	iFile=(RFile64&)aFile;
#endif
	iReadBuf.Zero();
	iNext=(TText*)iReadBuf.Ptr();
	iEnd=iNext;
#ifndef SYMBIAN_ENABLE_64_BIT_FILE_SERVER_API
	TInt pos = 0;
#else
	TInt64 pos = 0;
#endif
	iFile.Seek(ESeekCurrent,pos);
 	if (pos == 0)
 		iState = EStartOfFile;
 	else
 		iState = ENormal;
	}




EXPORT_C TInt TFileText::Read(TDes& aDes)
/**
Reads single line text record from a Unicode file into the specified descriptor.

The read operation begins at the current file position, and ends when
a line delimiter character is read.

If the maximum length of the descriptor is insufficient to hold the record, 
the function returns KErrTooBig and the descriptor is filled to its maximum 
length.

If Read() is called when the current position is the end of the file (that 
is, after the last line delimiter in the file), KErrEof is returned, and the 
length of the buffer is set to zero.

@param aDes On return, contains the single record read from the file. Any 
            previous contents are overwritten.

@return KErrNone if successful, otherwise one of the other system-wide error 
        codes.
*/
	{

	TText* pD=(TText*)aDes.Ptr();
	TInt len=aDes.MaxLength();
	TInt newLen=0;
	while (newLen<len)
		{
		if (iNext>=iEnd)
			{
			TInt r=FillBuffer();
			if (r!=KErrNone && r!=KErrEof)
				return(r);
			if (r==KErrEof)
				{
				aDes.SetLength(newLen);
				return(newLen ? KErrNone : KErrEof);
				}
			continue;
			}
		TBool terminate=newLen;
		TInt r=CheckForTerminator(terminate);
		if (r!=KErrNone || terminate)
			{
			aDes.SetLength(newLen);
			return(r);
			}
		*pD++=(*iNext++);
		newLen++;
		}
	aDes.SetLength(newLen);
	TBool terminate=newLen;
	TInt r=CheckForTerminator(terminate);
	if (r!=KErrNone || terminate)
		return(r);
	NextRecord();
	return(KErrTooBig);
	}

void TFileText::NextRecord()
//
// Move to the start of the next record
//
	{

	FOREVER
		{
		TBool terminate=EFalse;
		TInt r=CheckForTerminator(terminate);
		if (r!=KErrNone || terminate)
			return;
		iNext++;
		}
	}

static void SwapWords(TText* aStart,TInt aCount)
 	{
 	TUint8* p = (TUint8*)aStart;
 	while (aCount-- > 0)
 		{
 		TUint8 temp = *p;
 		*p = p[1];
 		p[1] = temp;
 		p += 2;
   		}
   	}

TInt TFileText::FillBuffer()
//
// Read the new data from the file
//
	{
	
	TInt r=iFile.Read(iReadBuf);
	if (r!=KErrNone)
		return(r);
	if (iReadBuf.Length()==0)
		return(KErrEof);
	iNext=(const TText*)iReadBuf.Ptr();
	iEnd=iNext+iReadBuf.Length()/sizeof(TText);
	 
 	// Use any leading byte order marker to determine endianness.
 	if (iState == EStartOfFile)
 		{
 		iState = ENormal;

 		// Ignore an ordinary byte order marker.
 		if (*iNext == 0xFEFF)
 			iNext++;

 		// Set the endianness state to 'reverse' if a reversed byte order marker is found.
 		else if (*iNext == 0xFFFE)
 			{
 			iNext++;
 			iState = EReverse;
 			}
 
 		if (iNext == iEnd)
 			return KErrEof;
 		}
 
 	if (iState == EReverse)
		SwapWords((TText*)iNext,(iEnd - iNext));

	return(KErrNone);
	}

TInt TFileText::CheckForTerminator(TBool& anAnswer)
//
// Return ETrue if the next char is a record terminator: PARAGRAPH SEPARATOR (U+2029), LINE SEPARATOR (U+2028),
// CR-LF (U+000D, U+000A), or LF (U+000A)
//
	{

	if (iNext>=iEnd)
		{
		TInt r=FillBuffer();
		if (r!=KErrNone)
			{
			if (r==KErrEof && anAnswer)
				return(KErrNone);
			return(r);
			}
		}

	anAnswer=EFalse;
	const TText* oldNext=iNext;
	TInt oldBufferLength=iReadBuf.Length();
	TText c=(*iNext);
	TBool peek=EFalse;

	// Check for unambiguous paragraph or line separator.
 	if (c == 0x2029 || c == 0x2028)
 		{
 		iNext++;
 		anAnswer = ETrue;
		return KErrNone;
 		}
 
 	// Check for CR-LF or LF.
 	if (c == 0x000D)
		{
		iNext++;
		if (iNext<iEnd)
			c=(*iNext);
		else
			{
			peek=ETrue;
			TInt r=FillBuffer();
			if (r!=KErrNone && r!=KErrEof)
				return(r);
			if (r==KErrNone)
				c=(*iNext);
			}
		}

	if (c == 0x000A)
		{
		iNext++;
		anAnswer=ETrue;
		return(KErrNone);
		}

	iNext=oldNext;
	if (!peek)
		return(KErrNone);

#ifndef SYMBIAN_ENABLE_64_BIT_FILE_SERVER_API
	TInt pos=(-1)*(oldBufferLength+iReadBuf.Length());
#else
	TInt64 pos=(-1)*(oldBufferLength+iReadBuf.Length());
#endif
	TInt r=iFile.Seek(ESeekCurrent,pos);
	if (r==KErrNone)
		r=FillBuffer();
	if (r!=KErrNone)
		return(r);
	iNext=oldNext;
	return(KErrNone);
	}




EXPORT_C TInt TFileText::Write(const TDesC& aDes)
/**
Writes the contents of a descriptor to the end of a Unicode file.

A line delimiter is appended to the descriptor, and the current file position
is set to the new end of file.

If the descriptor contains one or more paragraph delimiters, Read() will treat 
the contents of the descriptor as more than one record.

@param aDes The descriptor content to be appended to the file.

@return KErrNone if successful, otherwise one of the other system-wide error 
        codes.
        
@see TFileText::Read
*/
	{

	TInt r=Seek(ESeekEnd);
	if (r!=KErrNone)
		return(r);
	TPtrC8 writeBuf((const TUint8*)aDes.Ptr(),aDes.Size());
	r=iFile.Write(writeBuf);
	if (r!=KErrNone)
		return(r);
 	TText lf = 0x000A;
 	TPtrC8 lf8((const TUint8*)&lf,sizeof(TText));
 	r=iFile.Write(lf8);
	return(r);
	}




EXPORT_C TInt TFileText::Seek(TSeek aMode)
/**
Seeks to start or end of file.

It is only necessary to call this function before 
using Read() because Write() always seeks to the end of the file
before writing.

@param aMode ESeekStart to seek to the start of the file;
             ESeekEnd to seek to the end.
             
@return KErrNone if successful, otherwise one of the other system-wide error 
        codes.

@panic FSCLIENT 5 if aMode is neither ESeekStart nor ESeekEnd.

@see TFileText::Read
@see TFileText::Write
*/
	{

	__ASSERT_ALWAYS(aMode==ESeekStart || aMode==ESeekEnd,Panic(EFTextIllegalSeekMode));
#ifndef	SYMBIAN_ENABLE_64_BIT_FILE_SERVER_API
	TInt pos=0;
#else
	TInt64 pos = 0;
#endif
	TInt ret=iFile.Seek(aMode,pos);
 	if (ret == 0 && aMode == ESeekStart)
 		iState = EStartOfFile;
	iNext = (TText*)iReadBuf.Ptr();
 	iEnd = iNext;
	return ret;
	}