kerneltest/f32test/concur/t_cfssoak.cpp
changeset 0 a41df078684a
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/kerneltest/f32test/concur/t_cfssoak.cpp	Mon Oct 19 15:55:17 2009 +0100
@@ -0,0 +1,456 @@
+// Copyright (c) 2002-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:
+//
+
+//! @file f32test\concur\t_csfsoak.cpp
+
+#include <f32file.h>
+#include <e32test.h>
+#include <e32math.h>
+#include <f32dbg.h>
+
+#include "t_server.h"
+#include "t_chlffs.h"
+#include "t_tdebug.h"
+#include "t_cfssoak.h"
+
+GLDEF_D RTest test(_L("T_CFSSOAK"));
+GLDEF_D	RFs TheFs;
+
+GLREF_D TPtrC gArgV[];
+GLREF_D TInt  gArgC;
+
+LOCAL_D TFullName gFsName;
+LOCAL_D TFullName gFsName1;
+LOCAL_D TFullName gFsName2;
+LOCAL_D TFullName gOldFsName;
+LOCAL_D TFullName gNewFsName;
+LOCAL_D TBool     gNoMedia = ETrue;
+
+_LIT(KFsFile,   "CFAFSDLY");
+_LIT(KFsName,   "DelayFS");
+
+LOCAL_D const TInt32 KSecond = 1000000;
+
+LOCAL_D TChar gRemFsChr = 0;
+LOCAL_D TInt  gRemFsDrv = 0;
+
+LOCAL_D TChar gDrvChFill = 0;
+
+LOCAL_D TInt  gNumRead = 0;
+LOCAL_D TInt  gNumFill = 0;
+LOCAL_D TInt  gNumRem  = 0;
+
+GLDEF_D TExtension gPrimaryExtensions[KMaxDrives];
+
+inline void TraceError(const char* aFile, TInt aLine, const TDesC& aStr, TInt aRsp)
+//
+// 'Helper' routine to output the file and line of an error as well as a string
+// and a response value.
+//
+    {
+    TBuf<256> fbuf;
+    TPtrC8    fptr((const TUint8*)aFile);
+    fbuf.Copy(fptr);
+    RDebug::Print(_L("%S:%d  %S: r = %d"), &fbuf, aLine, &aStr, aRsp);
+    }
+
+#define TESTSTR(r,s,cond) if (!(cond)) { TraceError(__FILE__, __LINE__, (s), (r)); test(0); }
+#define TESTLIT(r,s,cond) { _LIT(KStr,s); TESTSTR(r,KStr,cond); }
+#define TESTRES(r,cond)   TESTLIT(r,"ERROR",cond)
+
+LOCAL_C TChar MountTestFileSystem(TInt aDrive)
+///
+/// Mount a new CTestFileSystem on the drive under test.
+/// @param aDrive Drive number (EDriveC etc.) to be used.
+///
+	{
+	TInt r;
+	// Check attributes
+	TBuf<64> b;
+	TChar c;
+	r=TheFs.DriveToChar(aDrive,c);
+	TESTLIT(r, "TheFs.DriveToChar", r==KErrNone);
+		
+	b.Format(_L("Mount test file system on %c:"),(TUint)c);
+	test.Next(b);
+
+	r=TheFs.AddFileSystem(KFsFile);
+	TESTLIT(r, "TheFs.AddFileSystem", r==KErrNone || r==KErrAlreadyExists);
+	
+	r=TheFs.FileSystemName(gOldFsName,aDrive);
+	TESTLIT(r, "TheFs.FileSystemName", r==KErrNone || r==KErrNotFound);
+
+	TDriveInfo drv;
+	r = TheFs.Drive(drv, gRemFsDrv);
+	TESTLIT(r, "Drive()", r == KErrNone);
+
+	gNoMedia = (drv.iType == EMediaUnknown || drv.iType == EMediaNotPresent);
+
+	if (gOldFsName.Length() > 0)
+		{
+		r = TheFs.ExtensionName(gPrimaryExtensions[aDrive].iName, aDrive, 0);
+		if (r == KErrNone)
+			gPrimaryExtensions[aDrive].iExists = ETrue;
+
+		TTest::Printf(_L("Dismount %C: %S"), (TUint)c, &gOldFsName);
+		r=TheFs.DismountFileSystem(gOldFsName,aDrive);
+		TESTLIT(r, "DismountFileSystem", r==KErrNone);
+		}
+
+	if (gPrimaryExtensions[aDrive].iExists == EFalse)
+		r=TheFs.MountFileSystem(KFsName,aDrive);
+	else
+		r=TheFs.MountFileSystem(KFsName,gPrimaryExtensions[aDrive].iName,aDrive);
+
+	TESTLIT(r, "TheFs.MountFileSystem", r==KErrNone);
+
+	r=TheFs.FileSystemName(gNewFsName,aDrive);
+	TESTLIT(r, "TheFs.FileSystemName", r==KErrNone);
+		
+	r = gNewFsName.CompareF(KFsName);
+	TESTLIT(r, "gNewFsName.Compare", r==0);
+		
+	return c;
+	}
+
+LOCAL_C void UnmountFileSystem(TInt aDrive)
+///
+/// Dismount the filesystem and remount the original one.
+/// @param aDrive Drive number (EDriveC etc.) to be unmounted.
+///
+	{
+	TChar c;
+	TInt r=TheFs.DriveToChar(aDrive,c);
+	TESTLIT(r, "TheFs.DriveToChar", r==KErrNone);
+
+	r=TheFs.DismountFileSystem(gNewFsName,aDrive);
+	TESTLIT(r, "TheFs.DismountFileSystem", r==KErrNone);
+
+	if (gNoMedia)
+		{
+		TTest::Printf(_L("No media on %C: so don't remount it"), (TUint)c);
+		}
+	else if (gOldFsName.Length() > 0)
+		{
+		TTest::Printf(_L("Mount    %C: %S"), (TUint)c, &gOldFsName);
+		if (gPrimaryExtensions[aDrive].iExists == EFalse)
+			r=TheFs.MountFileSystem(gOldFsName,aDrive);
+		else
+			r=TheFs.MountFileSystem(gOldFsName,gPrimaryExtensions[aDrive].iName,aDrive);
+		TESTLIT(r, "MountFileSystem", r==KErrNone);
+		}
+	}
+
+LOCAL_C TInt testReadOnly(TAny* /*aData*/)
+///
+/// Task to test read-only operations.
+///
+	{
+	CTrapCleanup* cleanup = CTrapCleanup::New();
+	TSoakReadOnly rdonly;
+	TInt64 seed = 178543;
+	TInt r = KErrNone;
+	rdonly.ExcludeDrive(gRemFsDrv);
+	while (r == KErrNone)
+		{
+		r = rdonly.ScanDrives(ETrue, Math::Rand(seed) % 90 + 10);
+		TTest::PrintLock();
+		TTest::Printf();
+		TTest::Printf(_L("Read-only test results:\n"));
+		TSoakStats::Print();
+		rdonly.iDrives.Print(_L("Drives:"));
+		rdonly.iDirs.Print(_L("Dirs:"));
+		rdonly.iFiles.Print(_L("Files:"));
+		rdonly.iReads.Print(_L("Reads:"));
+		TTest::Printf();
+		TTest::PrintUnlock();
+		gNumRead++;
+		}
+	delete cleanup;
+	TTest::Fail(HERE, r, _L("ScanDrives error"));
+	return r;
+	}
+
+LOCAL_C TInt testFillEmpty(TAny* /*aData*/)
+///
+/// Task to repeatedly fill and clean a drive.
+///
+	{
+	CTrapCleanup* cleanup = CTrapCleanup::New();
+	TVolumeInfo info;
+	TFullName   gFsName;
+	TChar  drvch = 0;
+	TInt   drive = 0;
+	TInt64 free = 0;
+	RFs    fs;
+	TInt   r = fs.Connect();
+	if (r != KErrNone)
+		TTest::Fail(HERE, r, _L("can't connect to fileserver\n"));
+	if (gDrvChFill)
+		{
+		// User specified the drive, so use that explicitly
+		drvch = gDrvChFill;
+		}
+	else
+		{
+		// User didn't specify the drive, so scan through them looking for
+		// likely candidates.  Don't use ROM or RAM drives, or LFFS ones, or
+		// our own 'slow' file system.  Out of what's left, look for the one
+		// with the least (but non-zero) free space.
+		for (drive = EDriveA; drive <= EDriveZ; drive++)
+			{
+			if (drive != gRemFsDrv)
+				{
+				TDriveInfo d;
+				TInt r;
+				r = fs.Drive(d, drive);
+				if (r == KErrNone && d.iDriveAtt != 0 &&
+					!(d.iDriveAtt & KDriveAttRom) &&
+					d.iMediaAtt != EMediaRam &&
+					!IsFileSystemLFFS(fs, drive))
+					{
+					r = fs.Volume(info, drive);
+					if (r == KErrNone)
+						{
+						if (free <= 0 || info.iFree < free)
+							{
+							r = fs.DriveToChar(drive, drvch);
+							if (r != KErrNone)
+								TTest::Fail(HERE, r, _L("DriveToChar(%d)"), drive);
+							free = info.iFree;
+							}
+						}
+					else
+						{
+						// not interested in drives which produce errors, they
+						// probably don't really exist.
+						}
+					}
+				else
+					{
+					// Drive doesn't exist, is a ROM drive or an LFFS drive, don't
+					// use it.
+					}
+				}
+			}
+		}
+
+	// If no drive found, can't do fill test so fail.
+	if (drvch == 0)
+		{
+		TTest::Fail(HERE, _L("No drive found for fill test!"));
+		}
+
+	r = fs.CharToDrive(drvch, drive);
+	if (r != KErrNone)
+		TTest::Fail(HERE, r, _L("CharToDrive(%C:)\n"), (TUint)drvch);
+	r = fs.FileSystemName(gFsName, drive);
+	if (r != KErrNone)
+		TTest::Fail(HERE, r, _L("FileSystemName(%C:)\n"), (TUint)drvch);
+
+	r = fs.Volume(info, drive);
+	if (r != KErrNone)
+		TTest::Fail(HERE, r, _L("Volume(%C:)\n"), (TUint)drvch);
+	
+	fs.Close();
+
+	TTest::Printf(_L("Using %C: (%S) with %d KB free for fill/clean cycle test\n"),
+		          (TUint)drvch, &gFsName, I64LOW(info.iFree / 1024));
+
+	TSoakFill fill;
+	r = fill.SetDrive(drvch);
+	if (r != KErrNone)
+		TTest::Fail(HERE, r, _L("Unable to setup drive %C"), (TUint)drvch);
+
+	r = fill.CleanDrive();
+	if (r != KErrNone && r != KErrNotFound && r != KErrPathNotFound)
+		TTest::Fail(HERE, r, _L("Error cleaning drive %C"), (TUint)drvch);
+
+	// Loop to do the actual test
+	while (r == KErrNone)
+		{
+		r = fill.FillDrive();
+		if (r != KErrNone && r != KErrDiskFull)
+			TTest::Fail(HERE, r, _L("Error filling drive %C"), (TUint)drvch);
+
+		TTest::PrintLock();
+		TTest::Printf(_L("%C: filled\n"), (TUint)drvch);
+		TTest::PrintUnlock();
+
+		r = fill.CleanDrive();
+		if (r == KErrPathNotFound)
+			r = KErrNone;
+		if (r != KErrNone)
+			TTest::Fail(HERE, r, _L("cleaning drive %C"), (TUint)drvch);
+
+		TTest::PrintLock();
+		TTest::Printf(_L("%C: cleaned\n"), (TUint)drvch);
+		TTest::PrintUnlock();
+
+		gNumFill++;
+		}
+	delete cleanup;
+	TTest::Fail(HERE, r, _L("Fill/clean error"));
+	return r;
+	}
+
+LOCAL_C TInt testRemote(TAny* /*aData*/)
+///
+/// Task to exercise the remote (special) filesystem.
+///
+	{
+	CTrapCleanup* cleanup = CTrapCleanup::New();
+	TSoakRemote rem(gRemFsChr);
+	TInt r = KErrNone;
+	for (TBool sync = EFalse; r == KErrNone; sync = !sync)
+		{
+		for (TInt i = 0; i < 1; i++)
+			{
+			rem.Remount(sync);
+			r = rem.TestSession();
+			if (r != KErrNone)
+				TTest::Fail(HERE, r, _L("TestSession failed"));
+			r = rem.TestSubSession();
+			if (r != KErrNone)
+				TTest::Fail(HERE, r, _L("TestSubSession failed"));
+			r = rem.TestMount();
+			if (r != KErrNone)
+				TTest::Fail(HERE, r, _L("Mount/dismount failed"));
+			User::After(100*1000000);
+			gNumRem++;
+			}
+		}
+	delete cleanup;
+	TTest::Fail(HERE, r, _L("Test (remote) filesystem error"));
+	return r;
+	}
+
+GLDEF_C void CallTestsL()
+///
+/// Do all tests.
+///
+	{
+	// first thing, initialise local test class to make sure it gets done.
+	TInt r = TTest::Init();
+	test(r == KErrNone);
+	
+    const TInt KMaxArgs = 4;
+    TPtrC argv[KMaxArgs];
+    TInt  argc = TTest::ParseCommandArguments(argv, KMaxArgs);
+
+	gRemFsChr = 'M'; // probably a 'safe' drive
+	gDrvChFill = 0;
+
+	if (argc > 1)
+		gRemFsChr = User::UpperCase(argv[1][0]);
+	if (argc > 2)
+		gDrvChFill = User::UpperCase(argv[2][0]);
+
+	r = TheFs.CharToDrive(gRemFsChr, gRemFsDrv);
+
+	if (r != KErrNone || gRemFsDrv == EDriveZ)
+		{
+		test.Printf(_L("Test cannnot run on drive %C:\n"), (TUint)gRemFsChr);
+		return;
+		}
+
+	// If we can do it, check whether the drives are LFFS and are OK
+// !!! Disable platform security tests until we get the new APIs
+/*	if (User::Capability() & KCapabilityRoot)
+		{
+		if (gDrvChFill != 0 && gDrvChFill != gRemFsChr)
+			CheckMountLFFS(TheFs, gDrvChFill);
+		}
+*/
+	MountTestFileSystem(gRemFsDrv);
+
+	r = TTest::Create(0, testReadOnly, _L("T_ReadOnly"));
+	test(r == KErrNone);
+
+	r = TTest::Create(1, testFillEmpty, _L("T_FillEmpty"));
+	test(r == KErrNone);
+
+	r = TTest::Create(2, testRemote, _L("T_Remote"));
+	test(r == KErrNone);
+
+	TRequestStatus kStat;
+	test.Console()->Read(kStat);
+	TInt displaycount = 0;
+	while ((r = TTest::Run(ETrue, 1*KSecond)) == KErrNone) // run until any thread exits
+		{
+		if (++displaycount >= 60 || kStat == KErrNone)
+			{
+			test.Printf(_L("Fill %-6d Rem %-6d Read %d\n"), gNumFill, gNumRem, gNumRead);
+			displaycount = 0;
+			}
+		if (kStat == KErrNone)
+			{
+			test.Printf(_L("Aborted by user!\n"));
+			break;
+			}
+		}
+	test(r == KErrNone);
+
+	// Kill all outstanding threads.
+	TTest::KillAll(KErrAbort);
+
+	UnmountFileSystem(gRemFsDrv);
+	}
+
+GLDEF_C TInt E32Main()
+//
+// Main entry point
+//
+    {
+	for (TInt i = 0; i < KMaxDrives; i++)
+		{
+		gPrimaryExtensions[i].iExists = EFalse;
+		}
+
+    TInt r;
+    CTrapCleanup* cleanup;
+    cleanup=CTrapCleanup::New();
+    __UHEAP_MARK;
+
+    test.Title();
+    test.Start(_L("Starting tests..."));
+
+    r=TheFs.Connect();
+    test(r==KErrNone);
+
+    // TheFs.SetAllocFailure(gAllocFailOn);
+    TTime timerC;
+    timerC.HomeTime();
+
+	// Do the test
+    TRAP(r,CallTestsL());
+
+    // reset the debug register
+    TheFs.SetDebugRegister(0);
+
+    TTime endTimeC;
+    endTimeC.HomeTime();
+    TTimeIntervalSeconds timeTakenC;
+    r=endTimeC.SecondsFrom(timerC,timeTakenC);
+    test(r==KErrNone);
+    test.Printf(_L("Time taken for test = %d seconds\n"),timeTakenC.Int());
+    // TheFs.SetAllocFailure(gAllocFailOff);
+    TheFs.Close();
+    test.End();
+    test.Close();
+    __UHEAP_MARKEND;
+    delete cleanup;
+    return(KErrNone);
+    }