userlibandfileserver/fileserver/sfsrv/cl_cdir.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) 1995-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_cdir.cpp
// 
//

#include "cl_std.h"
#include <collate.h>

const TUint KCDirArrayGranularity=0x200;
const TInt KPartKeyLength = 8;
const TInt KCollationLevel0 = 0;
const TInt KCollationLevelMax = 3;

#define KCollationKeyAllocFail ((HBufC8*)-1)

///////////////////////////////////////////////////////////////////////////////
/**
 * @class TEntry2
 * @description TEntry's variant with pointer to collation key buffers
 * @internalComponent
 */
NONSHARABLE_CLASS(TEntry2)
    {
public:
	TEntry2(const TEntry& aEntry);
	~TEntry2();
public:
    TBool IsDir() const {return iEntry.IsDir();}
#ifndef	SYMBIAN_ENABLE_64_BIT_FILE_SERVER_API
    TInt64 Size() {return MAKE_TINT64(0,iEntry.iSize);}
#else
    TInt64 Size() {return iEntry.FileSize();}
#endif
    TTime Modified() const {return iEntry.iModified;}
    const TUidType& Type() const {return iEntry.iType;}
    const TDesC& Name() const {return iEntry.iName;}
private:
    TEntry2(const TEntry2& aEntry);
    TEntry2& operator=(const TEntry2& aEntry);
public:
	HBufC8* iPartKey;
	HBufC8* iFullKey;
	TEntry iEntry;
    };

TEntry2::TEntry2(const TEntry& aEntry) : iPartKey(0), iFullKey(0), iEntry(aEntry)
	{
	}

TEntry2::~TEntry2()
	{
	if (iPartKey != KCollationKeyAllocFail)
        delete iPartKey;
	if (iFullKey != KCollationKeyAllocFail)
	    delete iFullKey;
	}

inline TInt Entry2Size(const TEntry2& aEntry)
    {
    return sizeof(HBufC8*) * 2 + EntrySize(aEntry.iEntry, ETrue);
    }
///////////////////////////////////////////////////////////////////////////////

NONSHARABLE_CLASS(TKeyDir) : public TKeyArrayVar
	{
public:
	TKeyDir(TUint aKey);
	virtual TInt Compare(TInt aLeft,TInt aRight) const;
private:
	TInt CompareByName(TEntry2& aLeft, TEntry2& aRight) const;
private:
	TCollationMethod iCollationMethod;
	};

TKeyDir::TKeyDir(TUint aKey)
//
// Constructor
//
	: TKeyArrayVar(0,(TKeyCmpText)(aKey&0xff),aKey&(EDirsFirst|EDirsLast|EDescending|EDirDescending))
	{
	//
	// Create our own collation method to also consider punctuation when
	// sorting filenames.
	//
	iCollationMethod = *Mem::GetDefaultMatchingTable();
	iCollationMethod.iFlags |= TCollationMethod::EIgnoreNone | TCollationMethod::EFoldCase;
	}


TInt TKeyDir::Compare(TInt aLeft,TInt aRight) const
//
// Compare two directories for sorting.
//
	{

	if (aLeft==aRight)
		return(0);
	TEntry2& left = *(TEntry2*)At(aLeft);
	TEntry2& right = *(TEntry2*)At(aRight);
	TInt ret=0;
	if ((iKeyLength&EDirsFirst)==EDirsFirst)
		{
		if (left.IsDir())
			{
			if (!right.IsDir())
				ret=(-1); // left is a dir, right is not
			}
		else if (right.IsDir())
			ret=1; // right is a dir, left is not
		}
	else if ((iKeyLength&EDirsLast)==EDirsLast)
		{
		if (left.IsDir())
			{
			if (!right.IsDir())
				ret=1; // left is a dir, right is not
			}
		else if (right.IsDir())
			ret=(-1); // right is a dir, left is not
		}

	TInt cmpType=iCmpType;
	TInt keyLength=iKeyLength;
	TBool orderDirectories=(keyLength&EDirsFirst) || (keyLength&EDirsLast);
	if (orderDirectories && left.IsDir() && right.IsDir())
		{
		cmpType=ESortByName;
		if ((keyLength&EDirDescending)!=EDirDescending)
			keyLength&=~EDescending;
		else
			keyLength|=EDescending;
		}

	if (ret==0) // Both are the same type
		{
		ret=(-1); // left before right by default
		switch (cmpType)
			{
		case ESortNone:
			ret=1;
			break;
		case ESortByDate:
			if (left.Modified()>right.Modified())
				ret=1;
			else if (left.Modified()==right.Modified())
				ret=0;
			break;
		case ESortBySize:
#ifndef SYMBIAN_ENABLE_64_BIT_FILE_SERVER_API
		if (I64LOW(left.Size()) > I64LOW(right.Size()))
			ret=1;
		else if (I64LOW(left.Size())==I64LOW(right.Size()))
			ret=0;
#else
			if (left.Size() > right.Size())
				ret=1;
			else if (left.Size()==right.Size())
				ret=0;
#endif
			break;
		case ESortByExt:
			{
			TInt i1 = KErrNotFound, i2 = KErrNotFound;
			if (left.Name() != _L(".") && left.Name() != _L(".."))
				i1 = left.Name().LocateReverse('.');
			if (right.Name() != _L(".") && right.Name() != _L(".."))
				i2 = right.Name().LocateReverse('.');
			if (i1==KErrNotFound && i2!=KErrNotFound)
				ret=(-1);
			else if (i2==KErrNotFound && i1!=KErrNotFound)
				ret=1;
			else if ((i1==KErrNotFound && i2==KErrNotFound) || (ret=left.Name().Mid(i1).CompareC(right.Name().Mid(i2)))==0)
				goto byName;
			}
			break;
		case ESortByUid:
			if (left.Type()[1]==right.Type()[1])
				{
				if (left.Type()[2]==right.Type()[2])
					ret = CompareByName(left, right);
				else if (left.Type()[2].iUid>right.Type()[2].iUid)
					ret=1;
				}
			else if (left.Type()[1].iUid==0)
				ret=1;
			else if (right.Type()[1].iUid==0)
				ret=-1;
			else if (left.Type()[1].iUid>right.Type()[1].iUid)
				ret=1;
			break;
		case ESortByName:
byName:
			// Force the maximum collation level here (i.e. 3) for sorting strings 
			ret = CompareByName(left, right);
			break;
		default: // Default is bad news
			Panic(ECDirBadSortType);
			}
		}
	if ((keyLength&EDescending)==EDescending)
		ret=(-ret); // Descending sort order
	return(ret);
	}

TInt TKeyDir::CompareByName(TEntry2& aLeft, TEntry2& aRight) const
//
// Compare using collation key of entire name
//
	{
	TInt ret = -1;
	TInt r = KErrNone;

    // Allocate partial key first and handle potential error case
    // by calling old CompareC
	// Note: only compare on the first collation level (KCollationLevel0) for partial keys, 
	//       to avoid potential inconsistency between full key and partial key comparison.
	if (!aLeft.iPartKey)
		{
		TRAP(r, aLeft.iPartKey = aLeft.Name().Left(KPartKeyLength).GetCollationKeysL(KCollationLevel0, &iCollationMethod));
		if (r != KErrNone)
		    aLeft.iPartKey = KCollationKeyAllocFail;
		}
	if (!aRight.iPartKey)
		{
		TRAP(r, aRight.iPartKey = aRight.Name().Left(KPartKeyLength).GetCollationKeysL(KCollationLevel0, &iCollationMethod));
		if (r != KErrNone)
		    aRight.iPartKey = KCollationKeyAllocFail;
		}
	if (aLeft.iPartKey == KCollationKeyAllocFail || aRight.iPartKey == KCollationKeyAllocFail)
		return aLeft.Name().CompareC(aRight.Name());

	// Compare by partial key first
    ret = aLeft.iPartKey->Compare(*aRight.iPartKey);
    if (ret != 0)
        return ret;

    // Compare by full key if partial keys are identical
	if (!aLeft.iFullKey)
		{
		TRAP(r, aLeft.iFullKey = aLeft.Name().GetCollationKeysL(KCollationLevelMax, &iCollationMethod));
		if (r != KErrNone)
		    aLeft.iFullKey = KCollationKeyAllocFail;
		}
	if (!aRight.iFullKey)
		{
		TRAP(r, aRight.iFullKey = aRight.Name().GetCollationKeysL(KCollationLevelMax, &iCollationMethod));
		if (r != KErrNone)
		    aRight.iFullKey = KCollationKeyAllocFail;
		}
	if (aLeft.iFullKey == KCollationKeyAllocFail || aRight.iFullKey == KCollationKeyAllocFail)
	    // Using old CompareC if partial key allocation failed
		return aLeft.Name().CompareC(aRight.Name());

    // Compare using collation key of full names
    ret = aLeft.iFullKey->Compare(*aRight.iFullKey);

	return ret;
	}



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




EXPORT_C CDir::~CDir()
/**
Destructor.

Frees all resources owned by the object, prior to its destruction.
*/
	{

	delete iArray;
	}




EXPORT_C CDir* CDir::NewL()
/**
Allocates and constructs a directory object.

This function is protected, which prevents objects of this class from being
directly constructed.

@return A pointer to the newly created object.
*/
	{

	CDir* pD=new(ELeave) CDir;
	pD->iArray=new CArrayPakFlat<TEntry>(KCDirArrayGranularity);
	if (pD->iArray==NULL)
		{
		delete pD;
		User::LeaveNoMemory();
		}
	return(pD);
	}




EXPORT_C TInt CDir::Count() const
/**
Gets the number of entries in the array of directory
entries.

@return The number of entries in the array.
*/
	{

	return(iArray->Count());
	}




EXPORT_C const TEntry& CDir::operator[](TInt anIndex) const
/**
Gets an entry from the array of directory
entries.

@param anIndex of the desired entry within the array.

@return A directory entry.
*/
	{

	return((*iArray)[anIndex]);
	}


/**
 * Utility class to manage dynamic array memory
 * @internalComponent
 */
NONSHARABLE_CLASS(CAutoArray) : public CBase
    {
public:
    ~CAutoArray();
public:
    CArrayVarFlat<TEntry2>* iArray;
    };

// Have to do this trick because CArrayVarFlat won't destroy element one by one
CAutoArray::~CAutoArray()
    {
    if (iArray)
        for (TInt i=0; i<iArray->Count(); ++i)
            {
            TEntry2& e = (*iArray)[i];
           	if (e.iPartKey != KCollationKeyAllocFail)
                delete e.iPartKey;
	        if (e.iFullKey != KCollationKeyAllocFail)
	            delete e.iFullKey;
            e.iPartKey = e.iFullKey = 0;
            }
    delete iArray;
    }

EXPORT_C TInt CDir::Sort(TUint aKey)
/**
Sorts the array of directory entries.

@param aKey A set of flags describing how the directory entries are to be sorted.
       The set of flags is defined by TEntryKey.

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

@see TEntryKey
*/
	{
	CAutoArray autoArray;
	#define array autoArray.iArray

	// Create TEntry2 array from iArray.
	array = new CArrayVarFlat<TEntry2>(KCDirArrayGranularity);
	if (!array)
		return KErrNoMemory;
	
	TInt arrayCount = iArray->Count();
	if (arrayCount == 0)
		return KErrNone;
	
	TEntry2* entry2 = new TEntry2((*iArray)[0]);
	
	if (!entry2)
		return KErrNoMemory;

	TInt i, r;
	for (i=0; i<arrayCount; ++i)
		{
		entry2->iEntry = (*iArray)[i];
		// Pack here
		TUint32* pSizeHighSrc = PtrAdd((TUint32*)&(entry2->iEntry), sizeof(TEntry) - 2 * sizeof(TInt));
		TUint32* pSizeHighDst = PtrAdd((TUint32*)&(entry2->iEntry), EntrySize(entry2->iEntry, EFalse));
		
		*pSizeHighDst++ = *pSizeHighSrc++;	// Pack iSizeHigh
		*pSizeHighDst   = *pSizeHighSrc;	// Pack iReserved
		entry2->iEntry.iAtt |= KEntryAttPacked;
		
		TRAP(r, array->AppendL(*entry2, Entry2Size(*entry2)));

		if (r != KErrNone)
			{
			delete entry2;	
			return r;
			}
		}

	// Sort new array
	TKeyDir key(aKey);
	r = array->Sort(key);
	if (r != KErrNone)
		{
		delete entry2;
        return r;
		}

	// Copy sorted result back to iArray
	iArray->Reset();
	
	for (i=0; i<array->Count(); ++i)
		{
		entry2->iEntry = (*array)[i].iEntry;
		// Pack here
		TUint32* pSizeHighSrc = PtrAdd((TUint32*)&(entry2->iEntry), sizeof(TEntry) - 2 * sizeof(TInt));
		TUint32* pSizeHighDst = PtrAdd((TUint32*)&(entry2->iEntry), EntrySize(entry2->iEntry, EFalse));
		
		*pSizeHighDst++ = *pSizeHighSrc++;	// Pack iSizeHigh
		*pSizeHighDst   = *pSizeHighSrc;	// Pack iReserved
		entry2->iEntry.iAtt |= KEntryAttPacked;
		
		TRAP(r, iArray->AppendL(entry2->iEntry, EntrySize(entry2->iEntry, ETrue)));
		if (r != KErrNone)
			{
			delete entry2;	
			return r;
			}
		}

	delete entry2;
	return r;
	}
	

EXPORT_C void CDir::AddL(const TEntry& aEntry)
/**
Adds the specified entry to the directory.

Note that the function can leave.

@param aEntry The directory entry to be added.
*/
	{
	if(aEntry.iAtt & KEntryAttPacked)
		{
		iArray->AppendL(aEntry,EntrySize(aEntry, ETrue));
		}
	else
		{
		TEntry entry = aEntry;
		// Pack here
		TUint32* pSizeHighSrc = PtrAdd((TUint32*)&entry, sizeof(TEntry) - 2 * sizeof(TInt));
		TUint32* pSizeHighDst = PtrAdd((TUint32*)&entry, EntrySize(entry, EFalse));
		
		*pSizeHighDst++ = *pSizeHighSrc++;	// Pack iSizeHigh
		*pSizeHighDst   = *pSizeHighSrc;		// Pack iReserved
		entry.iAtt |= KEntryAttPacked;
		iArray->AppendL(entry,EntrySize(entry, ETrue));
		}
	}




EXPORT_C void CDir::ExtractL(TBool aRemove,CDir* & aDir)
/**
Copies all directory entries from this directory array, and adds them to
a new directory array.

The directory entries in this array can be deleted.

Note that the function can leave.

@param aRemove If ETrue, the  directory entries in this array are
               to be deleted after extraction;
               if EFalse, the directory entries are not to be deleted.
                
@param aDir    On return, a pointer to a CDir object containing
               the extracted directory entries.
*/
	{

	aDir=NULL;
	aDir=CDir::NewL();
	CArrayPakFlat<TEntry>& anArray=(*iArray);
	TInt count=anArray.Count();
	
	if (count == 0)
		return;
	
	TInt i=0;
	while (i<count)
		{
		TEntry& e=anArray[i];
		if (e.IsDir())
			aDir->AddL(e);
		i++;
		}
	if (aRemove)
		{
		i=0;
		while (i<count)
			{
			if (anArray[i].IsDir())
				{
				anArray.Delete(i);
				count--;
				continue;
				}
			i++;
			}
		anArray.Compress();
		}
	aDir->Compress();
	}




EXPORT_C void CDir::Compress()
/**
Compresses the directory.

This has the effect of potentially reducing the ammount of storage 
space required on the media for that directory and the files it contains. 
Some files are already compressed and will not compress further.

A potential side effect of compression is that each file is required to 
be uncompressed prior to use, generally increasing the time and 
processing cycles required to access that file.

*/
	{

	iArray->Compress();
	}