diff -r 000000000000 -r a41df078684a kerneltest/f32test/filesystem/fat/t_main.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/kerneltest/f32test/filesystem/fat/t_main.cpp Mon Oct 19 15:55:17 2009 +0100 @@ -0,0 +1,638 @@ +// Copyright (c) 1997-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\server\t_main.cpp +// +// + +#define __E32TEST_EXTENSION__ + +#include +#include +#include +#include +#include "t_server.h" + +GLDEF_D RFs TheFs; +GLDEF_D TFileName gSessionPath; +GLDEF_D TInt gAllocFailOff=KAllocFailureOff; +GLDEF_D TInt gAllocFailOn=KAllocFailureOff; +GLDEF_D TInt64 gSeed=51703; + +GLDEF_D TChar gDriveToTest; +GLDEF_D TVolumeInfo gVolInfo; // volume info for current drive +GLDEF_D TFileCacheFlags gDriveCacheFlags; + +_LIT(KPrivate, "\\Private\\"); + + +//////////////////////////////////////////////////////////// +// Template functions encapsulating ControlIo magic +// +GLDEF_D template +GLDEF_C TInt controlIo(RFs &fs, TInt drv, TInt fkn, C &c) +{ + TPtr8 ptrC((TUint8 *)&c, sizeof(C), sizeof(C)); + + TInt r = fs.ControlIo(drv, fkn, ptrC); + + return r; +} + + +GLDEF_C void CreateShortName(TDes& aFileName,TInt64& aSeed) +// +// Create a random, dos legal 8.3 char name +// + { + + TInt length=Math::Rand(aSeed)%11; + if (length==0) + length=1; + else if (length==3) // don't create three letter names like 'AUX' or 'PRN' + length++; + else if (length>8) // end in '.' if no extension + length++; + + aFileName.SetLength(length); + for(TInt i=0;i0) + length=aLength; + else + { + length=Math::Rand(aSeed)%128; + length+=Math::Rand(aSeed)%128; + length+=Math::Rand(aSeed)%128; + length+=Math::Rand(aSeed)%128; + length-=256; + length=Abs(length); + if (length==0) + length=1; + if (length>220) + length=31; + } + if (length==3) // don't create three letter names like 'AUX' or 'PRN' + length++; + + aFileName.SetLength(length); + TInt spaceChar=-1; + TInt i; + for(i=0;i': + case ':': + case '"': + case '/': + case '|': + case '*': + case '?': + case '\\': + case '\0': + break; + default: + illegalChar=EFalse; + }; + } + aFileName[i]=(TText)letter; + } + + if (spaceChar!=-1) + { + i=spaceChar; + goto StartAgain; + } + } + + +GLDEF_C void CheckDisk() +// +// Do a checkdisk and report failure +// + { + test.Next(_L("Check Disk")); + TInt r=TheFs.CheckDisk(gSessionPath); + if (r!=KErrNone && r!=KErrNotSupported && r!=KErrPermissionDenied) + ReportCheckDiskFailure(r); + } + +GLDEF_C void ReportCheckDiskFailure(TInt aRet) +// +// Report the failure of checkdisk +// + { + + test.Printf(_L("CHECKDISK FAILED: ")); + switch(aRet) + { + case 1: test.Printf(_L("File cluster chain contains a bad value (<2 or >maxCluster)\n")); break; + case 2: test.Printf(_L("Two files are linked to the same cluster\n")); break; + case 3: test.Printf(_L("Unallocated cluster contains a value != 0\n")); break; + case 4: test.Printf(_L("Size of file != number of clusters in chain\n")); break; + default: test.Printf(_L("Undefined Error value %d\n"),aRet); + } + test(EFalse); + } + + +GLDEF_C void MakeFile(const TDesC& aFileName,const TUidType& aUidType,const TDesC8& aFileContents) +// +// Make a file and write uid and data +// + { + + RFile file; + TInt r=file.Replace(TheFs,aFileName,0); + if (r==KErrPathNotFound) + { + r=TheFs.MkDirAll(aFileName); + test_KErrNone(r); + r=file.Replace(TheFs,aFileName,0); + } + test_KErrNone(r); + TCheckedUid checkedUid(aUidType); + TPtrC8 uidData((TUint8*)&checkedUid,sizeof(TCheckedUid)); + r=file.Write(uidData); + test_KErrNone(r); + r=file.Write(aFileContents); + test_KErrNone(r); + file.Close(); + } + +GLDEF_C void MakeFile(const TDesC& aFileName,const TDesC8& aFileContents) +// +// Make a file and write something in it +// + { + + RFile file; + TInt r=file.Replace(TheFs,aFileName,0); + if (r==KErrPathNotFound) + { + r=TheFs.MkDirAll(aFileName); + test_KErrNone(r); + r=file.Replace(TheFs,aFileName,0); + } + test_KErrNone(r); + r=file.Write(aFileContents); + test_KErrNone(r); + file.Close(); + } + +GLDEF_C void MakeFile(const TDesC& aFileName,TInt anAttributes) +// +// Make a file and write something in it +// + { + + RFile file; + TInt r=file.Replace(TheFs,aFileName,0); + if (r==KErrPathNotFound) + { + r=TheFs.MkDirAll(aFileName); + test_KErrNone(r); + r=file.Replace(TheFs,aFileName,0); + } + test_KErrNone(r); + file.Close(); + r=TheFs.SetAtt(aFileName,anAttributes,0); + test_KErrNone(r); + } + +GLDEF_C void MakeFile(const TDesC& aFileName) +// +// Make a file +// + { + + MakeFile(aFileName,_L8("")); + } + +GLDEF_C void MakeDir(const TDesC& aDirName) +// +// Make a directory +// + { + + TInt r=TheFs.MkDirAll(aDirName); + if (r!=KErrNone && r!=KErrAlreadyExists) + { + test.Printf(_L("%c: MakeDir Error %d\n"),aDirName[0],r); + test(0); + } + } + + +GLDEF_C void DeleteTestDirectory() +// +// Delete the leaf session path directory +// + { + + TheFs.SetAtt(_L("\\F32-TST\\SCANTEST\\Left\\Dir3\\Dir4\\Hidden"), 0, KEntryAttHidden); + TheFs.SetAtt(_L("\\F32-TST\\SCANTEST\\Left\\Dir3\\Dir4\\Hidden\\HiddenFile"), 0, KEntryAttHidden); + TheFs.SetAtt(_L("\\F32-TST\\SCANTEST\\Left\\Dir3\\Dir4\\Hidden\\System"), 0, KEntryAttSystem); + test.Next(_L("Delete test directory")); + CFileMan* fMan=CFileMan::NewL(TheFs); + test(fMan!=NULL); + TInt r=TheFs.SessionPath(gSessionPath); + test_KErrNone(r); + r=TheFs.CheckDisk(gSessionPath); + if (r!=KErrNone && r!=KErrNotSupported) + ReportCheckDiskFailure(r); + r=fMan->RmDir(gSessionPath); + test_KErrNone(r); + delete fMan; + } + +GLDEF_C void CreateTestDirectory(const TDesC& aSessionPath) +// +// Create directory for test +// + { + TParsePtrC path(aSessionPath); + test(path.DrivePresent()==EFalse); + + TInt r=TheFs.SetSessionPath(aSessionPath); + test_KErrNone(r); + r=TheFs.SessionPath(gSessionPath); + test_KErrNone(r); + r=TheFs.MkDirAll(gSessionPath); + test_Value(r, r == KErrNone || r == KErrAlreadyExists); + } + +GLDEF_C TInt CurrentDrive() +// +// Return the current drive number +// + { + + TInt driveNum; + TInt r=TheFs.CharToDrive(gSessionPath[0],driveNum); + test_KErrNone(r); + return(driveNum); + } + +GLDEF_C void Format(TInt aDrive) +// +// Format current drive +// + { + + test.Next(_L("Format")); + TBuf<4> driveBuf=_L("?:\\"); + driveBuf[0]=(TText)(aDrive+'A'); + RFormat format; + TInt count; + TInt r=format.Open(TheFs,driveBuf,EQuickFormat,count); + test_KErrNone(r); + while(count) + { + TInt r=format.Next(count); + test_KErrNone(r); + } + format.Close(); + } + +LOCAL_C void PushLotsL() +// +// Expand the cleanup stack +// + { + TInt i; + for(i=0;i<1000;i++) + CleanupStack::PushL((CBase*)NULL); + CleanupStack::Pop(1000); + } + + +LOCAL_C void DoTests(TInt aDrive) +// +// Do testing on aDrive +// + { + + gSessionPath=_L("?:\\F32-TST\\"); + TChar driveLetter; + TInt r=TheFs.DriveToChar(aDrive,driveLetter); + test_KErrNone(r); + gSessionPath[0]=(TText)driveLetter; + r=TheFs.SetSessionPath(gSessionPath); + test_KErrNone(r); + + User::After(1000000); + +// Format(CurrentDrive()); + + test.Printf(_L("Creating session path")); + r=TheFs.MkDirAll(gSessionPath); + if(r == KErrCorrupt) + { + test.Printf(_L("Attempting to create directory \'%S\' failed, KErrCorrupt\n"), &gSessionPath); + test.Printf(_L("This could be caused by a previous failing test, or a test media defect\n")); + test.Printf(_L("Formatting drive, retrying MkDirall\nShould subsequent tests fail with KErrCorrupt (%d) as well, replace test medium !\n"), + r); + Format(aDrive); + r=TheFs.MkDirAll(gSessionPath); + test_KErrNone(r); + } + else if (r == KErrNotReady) + { + TDriveInfo d; + r=TheFs.Drive(d, aDrive); + test_KErrNone(r); + if (d.iType == EMediaNotPresent) + test.Printf(_L("%c: Medium not present - cannot perform test.\n"), (TUint)driveLetter); + else + test.Printf(_L("medium found (type %d) but drive %c: not ready\nPrevious test may have hung; else, check hardware.\n"), (TInt)d.iType, (TUint)driveLetter); + } + test_Value(r, r == KErrNone || r == KErrAlreadyExists); + TheFs.ResourceCountMarkStart(); + test.Printf(_L("Calling main test sequence ...\n")); + TRAP(r,CallTestsL()); + test_KErrNone(r); + test.Printf(_L("test sequence completed without error\n")); + TheFs.ResourceCountMarkEnd(); + + CheckDisk(); + } + + +void ParseCommandArguments() +// +// +// + { + TBuf<0x100> cmd; + User::CommandLine(cmd); + TLex lex(cmd); + TPtrC token=lex.NextToken(); + TFileName thisfile=RProcess().FileName(); + if (token.MatchF(thisfile)==0) + { + token.Set(lex.NextToken()); + } + test.Printf(_L("CLP=%S\n"),&token); + + if(token.Length()!=0) + { + gDriveToTest=token[0]; + gDriveToTest.UpperCase(); + } + else + gDriveToTest='C'; + } + +TFullName gExtName; +TBool gPrimaryExtensionExists = EFalse; + +GLDEF_C TInt DismountFileSystem(RFs& aFs, const TDesC& aFileSystemName,TInt aDrive) + { + //Make note of the first extension if it exists, so that we remount + //it when the file system is remounted. + TInt r = aFs.ExtensionName(gExtName, aDrive, 0); + + if (r == KErrNone) + { + gPrimaryExtensionExists = ETrue; + } + return aFs.DismountFileSystem(aFileSystemName, aDrive); + } + +GLDEF_C TInt MountFileSystem(RFs& aFs, const TDesC& aFileSystemName,TInt aDrive, TBool aIsSync) + { + TInt r; + if (gPrimaryExtensionExists) + { + r = aFs.MountFileSystem(aFileSystemName, gExtName, aDrive, aIsSync); + } + else + { + r = aFs. MountFileSystem(aFileSystemName, aDrive, aIsSync); + } + return r; + } + +GLDEF_C TInt E32Main() +// +// Test with drive nearly full +// + { + + CTrapCleanup* cleanup; + cleanup=CTrapCleanup::New(); + TRAPD(r,PushLotsL()); + __UHEAP_MARK; + + test.Title(); + test.Start(_L("Starting tests...")); + + + ParseCommandArguments(); //need this for drive letter to test + + + r=TheFs.Connect(); + test_KErrNone(r); + TheFs.SetAllocFailure(gAllocFailOn); + TTime timerC; + timerC.HomeTime(); + TFileName sessionp; + TheFs.SessionPath(sessionp); + + TBuf<30> privatedir; + privatedir = KPrivate; + + TUid thisUID = RProcess().Identity(); + privatedir.AppendFormat(_L("%08x"),thisUID.iUid); + privatedir.Append(_L("\\")); + + test(privatedir == sessionp.Mid(2,sessionp.Length()-2)); + + test.Printf(_L("sp=%S\n"),&sessionp); + sessionp[0]=(TText)gDriveToTest; + test.Printf(_L("sp1=%S\n"),&sessionp); + + TInt theDrive; + r=TheFs.CharToDrive(gDriveToTest,theDrive); + test_KErrNone(r); + + // Get the TFileCacheFlags for this drive + r = TheFs.Volume(gVolInfo, theDrive); + if (r == KErrNotReady) + { + TDriveInfo info; + TInt err = TheFs.Drive(info,theDrive); + test_KErrNone(err); + if (info.iType == EMediaNotPresent) + test.Printf(_L("%c: Medium not present - cannot perform test.\n"), (TUint)gDriveToTest); + else + test.Printf(_L("%c: medium found (type %d) but drive not ready\nPrevious test may have hung; else, check hardware.\n"), (TUint)gDriveToTest, (TInt)info.iType); + } + else if (r == KErrCorrupt) + { + test.Printf(_L("%c: Media corruption; previous test may have aborted; else, check hardware\n"), (TUint)gDriveToTest); + } + test_KErrNone(r); + gDriveCacheFlags = gVolInfo.iFileCacheFlags; + test.Printf(_L("DriveCacheFlags = %08X\n"), gDriveCacheFlags); + +#if defined(_DEBUG) || defined(_DEBUG_RELEASE) + TPckgBuf pkgOrgValues; + TIOCacheValues& orgValues=pkgOrgValues(); + r = controlIo(TheFs,theDrive, KControlIoCacheCount, orgValues); + test_KErrNone(r); + + test.Printf(_L("\n")); + test.Printf(_L("Requests on close queue at start=%d\n"),orgValues.iCloseCount); + test.Printf(_L("Requests on free queue at start=%d\n"),orgValues.iFreeCount); + test.Printf(_L("Requests dynamically allocated at start=%d\n"),orgValues.iAllocated); + test.Printf(_L("Requests in total at start=%d\n"),orgValues.iTotalCount); + + // File cache + + // flush closed files queue + r = TheFs.ControlIo(theDrive, KControlIoFlushClosedFiles); + test_KErrNone(r); + + // get number of items on File Cache + TFileCacheStats startFileCacheStats; + r = controlIo(TheFs,theDrive, KControlIoFileCacheStats, startFileCacheStats); + test_Value(r, r == KErrNone || r == KErrNotSupported); + test.Printf(_L("File cache: Cachelines (free %d, used %d), Segments(allocated %d locked %d). Closed files(%d)\n"), + startFileCacheStats.iFreeCount, + startFileCacheStats.iUsedCount, + startFileCacheStats.iAllocatedSegmentCount, + startFileCacheStats.iLockedSegmentCount, + startFileCacheStats.iFilesOnClosedQueue); +#endif + + DoTests(theDrive); + + TTime endTimeC; + endTimeC.HomeTime(); + TTimeIntervalSeconds timeTakenC; + r=endTimeC.SecondsFrom(timerC,timeTakenC); + test_KErrNone(r); + test.Printf(_L("Time taken for test = %d seconds\n"),timeTakenC.Int()); + TheFs.SetAllocFailure(gAllocFailOff); + +#if defined(_DEBUG) || defined(_DEBUG_RELEASE) + TPckgBuf pkgValues; + TIOCacheValues& values=pkgValues(); + r = controlIo(TheFs,theDrive, KControlIoCacheCount, values); + test_KErrNone(r); + + test.Printf(_L("Requests on close queue at end=%d\n"),values.iCloseCount); + test.Printf(_L("Requests on free queue at end=%d\n"),values.iFreeCount); + test.Printf(_L("Requests dynamically allocated at end=%d\n"),values.iAllocated); + test.Printf(_L("Requests in total at end=%d\n"),values.iTotalCount); + + test(orgValues.iCloseCount==values.iCloseCount); + test(orgValues.iAllocated == values.iAllocated); + // The free count can increase if the file server runs out of requests in the RequestAllocator + // free pool but this should never decrease - this implies a request leak + test(orgValues.iFreeCount <= values.iFreeCount); + + // The total number of allocated requests should be equal to : + // requests on the close queue + requests on free queue + // + 1 (because we used one request to issue KControlIoCacheCount) + // If this doesn't equate then this implies a request leak + test(values.iTotalCount == values.iCloseCount + values.iFreeCount + 1); + + // File cache + TFileCacheStats endFileCacheStats; + r = controlIo(TheFs,theDrive, KControlIoFileCacheStats, endFileCacheStats); + test_Value(r, r == KErrNone || r == KErrNotSupported); + + test.Printf(_L("File cache: Cachelines (free %d, used %d), Segments(allocated %d locked %d). Closed files(%d)\n"), + endFileCacheStats.iFreeCount, + endFileCacheStats.iUsedCount, + endFileCacheStats.iAllocatedSegmentCount, + endFileCacheStats.iLockedSegmentCount, + endFileCacheStats.iFilesOnClosedQueue); + + // flush closed files queue + test.Printf(_L("Flushing close queue...")); + r = TheFs.ControlIo(theDrive, KControlIoFlushClosedFiles); + test_KErrNone(r); + + r = controlIo(TheFs,theDrive, KControlIoFileCacheStats, endFileCacheStats); + test_Value(r, r == KErrNone || r == KErrNotSupported); + test.Printf(_L("File cache: Cachelines (free %d, used %d), Segments(allocated %d locked %d). Closed files(%d)\n"), + endFileCacheStats.iFreeCount, + endFileCacheStats.iUsedCount, + endFileCacheStats.iAllocatedSegmentCount, + endFileCacheStats.iLockedSegmentCount, + endFileCacheStats.iFilesOnClosedQueue); + + + if (r == KErrNone) + { + test(startFileCacheStats.iFreeCount == endFileCacheStats.iFreeCount); + test(startFileCacheStats.iUsedCount == endFileCacheStats.iUsedCount); + test(startFileCacheStats.iAllocatedSegmentCount == endFileCacheStats.iAllocatedSegmentCount); + test(startFileCacheStats.iLockedSegmentCount == endFileCacheStats.iLockedSegmentCount); + test(startFileCacheStats.iFileCount == endFileCacheStats.iFileCount); + } +#endif + + TheFs.Close(); + test.End(); + test.Close(); + __UHEAP_MARKEND; + delete cleanup; + return(KErrNone); + } + +