kerneltest/f32test/ext/t_fatext.cpp
author Tom Cosgrove <tom.cosgrove@nokia.com>
Fri, 28 May 2010 16:29:07 +0100
changeset 30 8aab599e3476
parent 0 a41df078684a
permissions -rw-r--r--
Fix for bug 2283 (RVCT 4.0 support is missing from PDK 3.0.h) Have multiple extension sections in the bld.inf, one for each version of the compiler. The RVCT version building the tools will build the runtime libraries for its version, but make sure we extract all the other versions from zip archives. Also add the archive for RVCT4.

// Copyright (c) 2005-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:
// f32test\ext\fat_ext.cpp
// 
//

//! @SYMTestCaseID FSBASE-CR-JHAS-68YPX7
//! @SYMTestType CT
//! @SYMREQ CR JHAS-68YPX7
//! @SYMTestCaseDesc Test facility used by and bad disk handling test
//! @SYMTestStatus Implemented
//! @SYMTestActions Provided plug-in test extension for FAT
//! @SYMTestExpectedResults N/A
//! @SYMTestPriority Low
//! @SYMAuthor Ying Shi
//! @SYMCreationDate 20/05/2005
//! @See EFat and EFat32 components
//! @file f32test\ext\fat_ext.cpp

#include <e32math.h>
#include "t_fatext.h"

//--------------------------  CFatTestProxyDrive  --------------------------

CFatTestProxyDrive* CFatTestProxyDrive::NewL(CProxyDrive* aProxyDrive, CMountCB* aMount)
    {
    __PRINT(_L("CFatTestProxyDrive::NewL"));
    CFatTestProxyDrive* drive = new(ELeave) CFatTestProxyDrive(aProxyDrive,aMount);
    return(drive);
    }

CFatTestProxyDrive::CFatTestProxyDrive(CProxyDrive* aProxyDrive, CMountCB* aMount)
    : CTestProxyDrive(aProxyDrive,aMount)
    {
    __PRINT(_L("CFatTestProxyDrive::CFatTestProxyDrive"));
    InitL();
    }

TInt CFatTestProxyDrive::Read(TInt64 aPos,TInt aLength,const TAny* aTrg,TInt aMessageHandle,TInt aOffset,TInt aFlags)
    {
//    __PRINT(_L("CFatTestProxyDrive::Read"));
    if (CheckEvent(aPos,aLength))
        return KErrCorrupt;

    return CTestProxyDrive::Read(aPos,aLength,aTrg,aMessageHandle,aOffset,aFlags);
    }

TInt CFatTestProxyDrive::Read(TInt64 aPos,TInt aLength,const TAny* aTrg,TInt aMessageHandle,TInt aOffset)
    {
    return Read(aPos,aLength,aTrg,aMessageHandle,aOffset,0);
    }

TInt CFatTestProxyDrive::Read(TInt64 aPos,TInt aLength,TDes8& aTrg)
    {
    return Read(aPos,aLength,&aTrg,KLocalMessageHandle,0,0);
    }

TInt CFatTestProxyDrive::Write(TInt64 aPos,TInt aLength,const TAny* aSrc,TInt aMessageHandle,TInt anOffset,TInt aFlags)
    {
//    __PRINT(_L("CFatTestProxyDrive::Write"));
    if (CheckEvent(aPos,aLength))
        return KErrCorrupt;

    return CTestProxyDrive::Write(aPos,aLength,aSrc,aMessageHandle,anOffset,aFlags);
    }

TInt CFatTestProxyDrive::Write(TInt64 aPos,TInt aLength,const TAny* aSrc,TInt aMessageHandle,TInt anOffset)
    {
    return Write(aPos,aLength,aSrc,aMessageHandle,anOffset,0);
    }

TInt CFatTestProxyDrive::Write(TInt64 aPos,const TDesC8& aSrc)
    {
    return Write(aPos,aSrc.Length(),&aSrc,KLocalMessageHandle,0,0);
    }

TInt CFatTestProxyDrive::Format(TFormatInfo& anInfo)
    {
    //__PRINT(_L("CFatTestProxyDrive::Format"));
    TInt len;
    TInt64 pos = ((TInt64)anInfo.i512ByteSectorsFormatted) << KDefaultSectorLog2;
    // base function call in order to get anInfo.iMaxBytesPerFormat
    // for the first time
    if (anInfo.iMaxBytesPerFormat == 0)
        {
        TInt r = CTestProxyDrive::Format(anInfo);
        len = anInfo.iMaxBytesPerFormat;
        if (CheckEvent(pos,len))
            {
            anInfo.i512ByteSectorsFormatted = 0;
            return KErrCorrupt;
            }
        return r;
        }
    len = anInfo.iMaxBytesPerFormat;
    if (CheckEvent(pos,len))
        return KErrCorrupt;
    return CTestProxyDrive::Format(anInfo);
    }

TInt CFatTestProxyDrive::Format(TInt64 aPos,TInt aLength)
    {
    __PRINT(_L("CFatTestProxyDrive::Format"));
    if (CheckEvent(aPos,aLength))
        return KErrCorrupt;

    return CTestProxyDrive::Format(aPos, aLength);
    }

void CFatTestProxyDrive::DoInitL()
    {
    __PRINT(_L("CFatTestProxyDrive::DoInit"));
    if (!CheckMount())
        User::Leave(KErrNotReady);

   
    iTotalSectors = iBootSector.VolumeTotalSectorNumber();
    }

TInt CFatTestProxyDrive::DoControlIO(const RMessagePtr2& aMessage,TInt aCommand,TAny* aParam1,TAny* aParam2)
    {
    __PRINT(_L("CFatTestProxyDrive::DoControlIO"));

	// read boot sector & update iFatType etc
	CheckMount();

    TInt r = KErrNone;

	// Make sure that the information is up to date.
    if ((r=ReadBootSector()) != KErrNone)
        {
        __PRINT1(_L("ReadBootSector error: %d"), r);
        return EFalse;
        }

    switch(aCommand+EExtCustom)
        {
        case ESectorsPerCluster:
            r = aMessage.Write(2, TPckgBuf<TInt>(iBootSector.SectorsPerCluster()));
            break;
        case EFatType:
            r = aMessage.Write(2, TPckgBuf<TInt>(iBootSector.FatType()));
            break;
        
        case EGetDataPosition:
            {
                //-- obtain 1st data sector media position. This is actually a nasty hack;
                //-- we expect that the drive will be freshly formatted, thust the root dir is empty and the first file we create there
                //-- will occupy the certain place.
                TUint32 dataSec; 
                
                if(iBootSector.FatType() !=EFat32)
                   dataSec = iBootSector.FirstDataSector();
                else
                  {//-- for FAT32 we assume that the root dir takes exactly 1 cluster. Another dirty trick
                   dataSec = iBootSector.RootDirStartSector() + 1*iBootSector.SectorsPerCluster(); 
                  }
                __PRINT1(_L("EGetDataPosition, sec:%d"), dataSec);
                r = aMessage.Write(2, TPckgBuf<TInt>(dataSec << KDefaultSectorLog2));
            }
            break;

        default:
            r = CBaseExtProxyDrive::ControlIO(aMessage,aCommand,aParam1,aParam2);
            __PRINT2(_L("Get unknown command %d error %d"), aCommand, r);
        }
    return r;
    }

TBool CFatTestProxyDrive::DoCheckEvent(TInt64 aPos, TInt aLength)
    {
    //__PRINT2(_L("CFatTestProxyDrive::DoCheckEvent() pos:%d, len:%d"), (TUint32)aPos, aLength);

    if (aPos<0 || aLength<=0 || (aPos>>KDefaultSectorLog2)>=iTotalSectors)
        return EFalse;

    TInt begin = (TInt)(aPos >> KDefaultSectorLog2);
    TInt end = (TInt)((aPos+aLength-1) >> KDefaultSectorLog2);
    end = Min(end, iTotalSectors-1);

    if (iEventType == ENext)
        {
        Mark(begin);
        iEventType = ENone;
        iLastErrorReason = TErrorInfo::EBadSector;
        iSuccessBytes = 0;
        return ETrue;
        }

    if (iEventType == EDeterministic)
        {
        if (iCount <= end-begin+1)
            {
            iCount = 0;
            Mark(begin+iCount-1);
            iEventType = ENone;
            iLastErrorReason = TErrorInfo::EBadSector;
            iSuccessBytes = (iCount-1) << KDefaultSectorLog2;
            return ETrue;
            }
        else
            iCount -= end-begin+1;
        }

    TInt i;
    for (i=begin; i<=end; i++)
        if (IsMarked(i))
            {
            __PRINT(_L("CFatTestProxyDrive::DoCheckEvent() Sector Marked as bad!"));
            iLastErrorReason = TErrorInfo::EBadSector;
            iSuccessBytes = (i-begin) << KDefaultSectorLog2;
            return ETrue;
            }

    return EFalse;
    }

TBool CFatTestProxyDrive::CheckMount()
    {
    __PRINT(_L("CFatTestProxyDrive::CheckMount"));

    //-- read boot sector
    if (ReadBootSector() != KErrNone)
        {
        __PRINT(_L("ReadBootSector error: %d"));
        return EFalse;
        }

    //-- validate boot sector
    if(!iBootSector.IsValid())
    {
        goto BadBootSector;
    }

    if (iBootSector.FatType() == EFat32)   // fat 32
        {
        if (iBootSector.RootDirEntries() != 0   ||
            iBootSector.TotalSectors() != 0     ||
            iBootSector.HugeSectors() == 0      ||
            iBootSector.FatSectors32() == 0     ||
            iBootSector.RootClusterNum() < 2)
            {
                goto BadBootSector;
            }
        }
    else // fat16/12
        {
        if (iBootSector.RootDirEntries() == 0 ||
            (iBootSector.TotalSectors() == 0 && iBootSector.HugeSectors() == 0))
            {
                goto BadBootSector;
            }
        }

    //-- boot sector is OK
    return ETrue;

    //-- Invalid boot sector
    BadBootSector:
        
        __PRINT(_L("Boot sector is invalid! dump:"));
        iBootSector.PrintDebugInfo();
        return EFalse;



    }

TInt CFatTestProxyDrive::ReadBootSector()
    {
    __PRINT(_L("CFatTestProxyDrive::ReadBootSector"));

    const TInt KBufSz = KSizeOfFatBootSector;
    
    TBuf8<KBufSz> bootSecBuf(KBufSz);
    TInt r = CTestProxyDrive::Read(0, KBufSz, bootSecBuf);
	if (r != KErrNone)
		return r;

    //-- initialise TFatBootSector object
    iBootSector.Internalize(bootSecBuf);

    return KErrNone;
    }


// --------------------------  CFatTestProxyDriveFactory  --------------------------

/**
Factory class constructor
@internalTechnology
*/
CFatTestProxyDriveFactory::CFatTestProxyDriveFactory()
    {
    }

/**
Factory class installer
@internalTechnology
*/
TInt CFatTestProxyDriveFactory::Install()
    {
    __PRINT(_L("CFatTestProxyDriveFactory::Install"));
    _LIT(KFatTestExt,"FatTest");
    return(SetName(&KFatTestExt));
    }

/**
@internalTechnology
*/
CProxyDrive* CFatTestProxyDriveFactory::NewProxyDriveL(CProxyDrive* aProxy,CMountCB* aMount)
    {
    __PRINT(_L("CFatTestProxyDriveFactory::NewProxyDriveL"));
    return(CFatTestProxyDrive::NewL(aProxy,aMount));
    }



/**
@internalTechnology
*/
extern "C" {
EXPORT_C CProxyDriveFactory* CreateFileSystem()
    {
    __PRINT(_L("CreateFileSystem"));
    return new CFatTestProxyDriveFactory();
    }
}