kerneltest/f32test/server/t_fsrv.cpp
changeset 0 a41df078684a
child 33 0173bcd7697c
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/kerneltest/f32test/server/t_fsrv.cpp	Mon Oct 19 15:55:17 2009 +0100
@@ -0,0 +1,1922 @@
+// 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:
+// f32test\server\t_fsrv.cpp
+// 
+//
+#define __E32TEST_EXTENSION__
+#include <f32file.h>
+#include <e32test.h>
+#include "t_server.h"
+#include "t_chlffs.h"
+
+#include "f32_test_utils.h"
+
+using namespace F32_Test_Utils;
+
+
+RTest test(_L("T_FSRV"));
+
+                 
+
+const TInt KMaxParses=7;
+const TInt KHeapSize=0x2000;
+TFileName tPath;
+TBool gFirstTime=ETrue;
+const TInt KMaxVolumeName=11;
+
+TInt substDrive=0;
+
+TFileName gTestSessionPath;
+
+TInt LargeFileSize;
+TInt gDrive=-1;
+
+struct SParse
+	{
+	const TText* src;
+	const TText* rel;
+	const TText* def;
+	const TText* fullName;
+	const TText* drive;
+	const TText* path;
+	const TText* name;
+	const TText* ext;
+	};
+
+struct SParseServer
+	{
+	const TText* src;
+	const TText* rel;
+	const TText* fullName;
+	const TText* drive;
+	const TText* path;
+	const TText* name;
+	const TText* ext;
+	};
+
+LOCAL_D SParse parse[KMaxParses] =
+	{
+	{_S("A:\\PATH\\NAME.EXT"),NULL,NULL,_S("A:\\PATH\\NAME.EXT"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S(".EXT")},
+	{_S("A:\\PATH\\NAME"),NULL,NULL,_S("A:\\PATH\\NAME"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S("")},
+	{_S("A:\\PATH\\"),NULL,NULL,_S("A:\\PATH\\"),_S("A:"),_S("\\PATH\\"),_S(""),_S("")},
+	{_S("A:"),NULL,NULL,_S("A:"),_S("A:"),_S(""),_S(""),_S("")},
+	{_S(""),NULL,NULL,_S(""),_S(""),_S(""),_S(""),_S("")},
+	{_S("A:\\PATH\\NAME"),_S("A:\\ZZZZ\\YYYY.XXX"),NULL,_S("A:\\PATH\\NAME.XXX"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S(".XXX")},
+	{_S("NAME"),_S(".YYY"),_S("A:\\PATH\\"),_S("A:\\PATH\\NAME.YYY"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S(".YYY")}
+	};
+
+LOCAL_D SParseServer parseServer[KMaxParses] =
+	{
+	{_S("A:\\PATH\\NAME.EXT"),NULL,_S("A:\\PATH\\NAME.EXT"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S(".EXT")},
+	{_S("A:\\PATH\\NAME"),NULL,_S("A:\\PATH\\NAME"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S("")},
+	{_S("A:\\PATH\\"),NULL,_S("A:\\PATH\\"),_S("A:"),_S("\\PATH\\"),_S(""),_S("")},
+	{_S("A:"),NULL,_S("A:\\ABCDEF\\"),_S("A:"),_S("\\ABCDEF\\"),_S(""),_S("")},
+	{_S(""),NULL,_S("C:\\ABCDEF\\"),_S("C:"),_S("\\ABCDEF\\"),_S(""),_S("")},
+	{_S("A:\\PATH\\NAME"),_S("A:\\ZZZZ\\YYYY.XXX"),_S("A:\\PATH\\NAME.XXX"),_S("A:"),_S("\\PATH\\"),_S("NAME"),_S(".XXX")},
+	{_S("NAME"),_S(".YYY"),_S("C:\\ABCDEF\\NAME.YYY"),_S("C:"),_S("\\ABCDEF\\"),_S("NAME"),_S(".YYY")}
+	};
+
+LOCAL_C TInt pathTestThread(TAny*)
+//
+// The entry point for the producer thread.
+//
+	{
+
+	RTest test(_L("Second thread path handling"));
+	test.Title();
+
+	test.Start(_L("Path test thread"));
+	RFs f;
+	TInt r=f.Connect();
+	test(r==KErrNone);
+	r=f.SessionPath(tPath);
+	test(r==KErrNone);
+	f.Close();
+
+	return(KErrNone);
+	}
+
+LOCAL_C void printDriveAtt(TInt aDrive,TUint anAtt)
+//
+// Print a drive attribute.
+//
+	{
+
+	test.Printf(_L("%c: "),aDrive+'A');
+	if (anAtt&KDriveAttLocal)
+		test.Printf(_L("LOCAL "));
+	if (anAtt&KDriveAttRom)
+		test.Printf(_L("ROM "));
+	if (anAtt&KDriveAttRedirected)
+		test.Printf(_L("REDIR "));
+	if (anAtt&KDriveAttSubsted)
+		test.Printf(_L("SUBST "));
+	if (anAtt&KDriveAttInternal)
+		test.Printf(_L("INTERNAL "));
+	if (anAtt&KDriveAttHidden)
+		test.Printf(_L("HIDDEN "));
+	if ((anAtt&KDriveAttRemovable) && !(anAtt&KDriveAttLogicallyRemovable))
+		test.Printf(_L("PHYSICALLYREMOVABLE "));
+	if ((anAtt&KDriveAttRemovable) && (anAtt&KDriveAttLogicallyRemovable))
+		test.Printf(_L("LOGICALLYREMOVABLE "));
+	
+	test.Printf(_L("\n"));
+	}
+
+LOCAL_C void printDriveInfo(TInt aDrive,TDriveInfo& anInfo)
+//
+// Print a drive info.
+//
+	{
+
+	printDriveAtt(aDrive,anInfo.iDriveAtt);
+	test.Printf(_L("   MEDIA-ATT="));
+	if (anInfo.iMediaAtt==0)
+		test.Printf(_L("<none>"));
+	if (anInfo.iMediaAtt&KMediaAttVariableSize)
+		test.Printf(_L("VariableSize "));
+	if (anInfo.iMediaAtt&KMediaAttDualDensity)
+		test.Printf(_L("DualDensity "));
+	if (anInfo.iMediaAtt&KMediaAttFormattable)
+		test.Printf(_L("Formattable "));
+	if (anInfo.iMediaAtt&KMediaAttWriteProtected)
+		test.Printf(_L("WProtected "));
+	test.Printf(_L("\n   BATTERY="));
+	switch(anInfo.iBattery)
+		{
+	case EBatNotSupported: test.Printf(_L("Not supported\n")); break;
+	case EBatGood: test.Printf(_L("Good\n")); break;
+	case EBatLow: test.Printf(_L("Low\n")); break;
+	default:
+		test.Printf(_L("Unknown value\n"));
+		}
+	test.Printf(_L("   MEDIA="));
+	switch(anInfo.iType)
+		{
+	case EMediaNotPresent: test.Printf(_L("Not present\n")); break;
+	case EMediaUnknown: test.Printf(_L("Unknown\n")); break;
+	case EMediaFloppy: test.Printf(_L("Floppy\n")); break;
+	case EMediaHardDisk: test.Printf(_L("Hard disk\n")); break;
+	case EMediaCdRom: test.Printf(_L("CD Rom\n")); break;
+	case EMediaRam: test.Printf(_L("Ram\n")); break;
+	case EMediaFlash: test.Printf(_L("Flash\n")); break;
+	case EMediaRom: test.Printf(_L("Rom\n")); break;
+	case EMediaRemote: test.Printf(_L("Remote\n")); break;
+	case EMediaNANDFlash:  test.Printf(_L("NANDFlash\n")); break;
+	default:
+		test.Printf(_L("Unknown value\n"));
+		}
+	}
+	
+	
+			
+	
+LOCAL_C void MountRemoteFilesystem()
+	{
+	test.Next(_L("Mount Remote Drive simulator on Q:"));
+
+	TInt r=TheFs.AddFileSystem(_L("CFAFSDLY"));
+	test.Printf(_L("Add remote file system"));
+	test.Printf(_L("AddFileSystem returned %d\n"),r);
+	test_Value(r, r==KErrNone || r==KErrAlreadyExists);
+
+	r=TheFs.MountFileSystem(_L("DELAYFS"),EDriveQ);
+
+	
+	test.Printf(_L("Mount remote file system"));
+	test.Printf(_L("MountFileSystem returned %d\n"),r);
+	test_Value(r, r==KErrNone || r==KErrCorrupt || r==KErrNotReady || r==KErrAlreadyExists);
+
+	}	
+
+
+LOCAL_C void DisMountRemoteFilesystem()	
+	{
+
+	test.Printf(_L("Dismounting the remote Drives \n"));
+ 	
+ 	TInt r=TheFs.DismountFileSystem(_L("DELAYFS"),EDriveQ);
+ 	 
+ 	test.Printf(_L("Dismounting the Remote Drive returned %d\n"),r);
+ 	
+ 	test(r==KErrNone );
+	}
+
+
+LOCAL_C void CreateSubstDrive()
+	{
+     test.Printf(_L("Create Substitute Drive \n"));
+    
+    TDriveList driveList;   
+    TInt i ;
+      
+    TInt r=TheFs.SessionPath(gTestSessionPath);
+	test(r==KErrNone);  
+ 	
+ 	r=TheFs.DriveList(driveList, KDriveAttExclude|KDriveAttLocal);
+   	test( r==KErrNone );
+   
+
+   	for ( i = EDriveO; i < KMaxDrives; i++) 
+      	{
+    
+      	if ( driveList[i] == 0) 
+          	{
+          	if (i == EDriveQ) continue;  // Q reserved to mount a virtual Remote Drive, as part of the test.
+        	substDrive = i;
+          	break;          
+          	}
+      	}
+           
+   
+   	if (substDrive)
+   		{
+ 		TDriveInfo driveInfo;
+		r=TheFs.Drive(driveInfo,substDrive);
+		test(r==KErrNone);
+	
+		if (driveInfo.iDriveAtt==KDriveAttLocal)
+			{	
+			return;	//	Subst local drives fails
+			}
+	
+		TFileName n;
+		r=TheFs.Subst(n,substDrive);
+		test(r==KErrNone);
+		test(n.Length()==0);	
+		r=TheFs.SetSubst(gTestSessionPath,substDrive);
+		test(r==KErrNone);
+		r=TheFs.Subst(n,substDrive);
+		test(r==KErrNone);
+		test(n==gTestSessionPath);   		
+   		}
+
+	
+	}
+
+		
+LOCAL_C void RemoveSubstDrive()
+	{
+	 	if( substDrive)
+	 		{
+	 		test.Printf(_L("Removing Substitute Drive \n"));	 	 
+	 		TInt r =TheFs.SetSubst(_L(""),substDrive);	
+			test(r ==KErrNone);	 		
+	 		}
+
+	}
+
+
+
+LOCAL_C void testDriveInfo(TInt aDrive,TDriveInfo& anInfo)
+//
+// Test the drive info is reasonable
+//
+	{
+
+	test(anInfo.iBattery==EBatLow || anInfo.iBattery==EBatGood || anInfo.iBattery==EBatNotSupported);
+	
+	if (aDrive==EDriveZ)
+		{
+		if (anInfo.iType==EMediaNotPresent)
+			return;
+		
+		test(anInfo.iMediaAtt==KMediaAttWriteProtected);
+		test(anInfo.iDriveAtt==(KDriveAttRom|KDriveAttInternal));
+		test(anInfo.iType==EMediaRom);
+		}
+
+	else if (GetDriveLFFS()==aDrive)
+		{
+        if (anInfo.iType==EMediaNotPresent)
+            return;
+
+		test(anInfo.iDriveAtt&(KDriveAttLocal|KDriveAttInternal)==KDriveAttLocal|KDriveAttInternal);	// LFFS sets KDriveAttTransaction as well
+        test(anInfo.iType==EMediaFlash);
+        test(anInfo.iMediaAtt==KMediaAttFormattable);
+		}
+/*
+Why assume certain drive letters can only refer to certain drive types?
+	else if (aDrive==EDriveC || aDrive==EDriveY)
+		{
+		if (anInfo.iType==EMediaNotPresent)
+			return;
+
+//		test(anInfo.iDriveAtt==(KDriveAttLocal|KDriveAttInternal));
+
+		test(anInfo.iDriveAtt&(KDriveAttLocal|KDriveAttInternal)==KDriveAttLocal|KDriveAttInternal);	// LFFS sets KDriveAttTransaction as well
+		test(anInfo.iType==EMediaRam || anInfo.iType==EMediaFlash);
+		if(anInfo.iType==EMediaRam)	test(anInfo.iMediaAtt==(KMediaAttVariableSize|KMediaAttFormattable));
+		else if(anInfo.iType==EMediaFlash) test(anInfo.iMediaAtt==KMediaAttFormattable);
+		}
+	
+	else if (aDrive==EDriveD || aDrive==EDriveX)
+		{
+		if (anInfo.iType==EMediaNotPresent)
+			return;
+
+		test(anInfo.iDriveAtt==(KDriveAttLocal|KDriveAttRemovable));
+		test(anInfo.iType==EMediaHardDisk);
+		test(anInfo.iMediaAtt & KMediaAttFormattable);
+		}
+*/
+	}
+
+
+/*
+*  The following test has the requirement that the only remote drive is the one we mount 
+*  during the test and which doesn't have any other attributes set. If this is not the
+* case then the test conditions must be changed, in order for the test to stop failing.
+*/
+
+
+LOCAL_C void testDriveList()
+//
+// Test the drive list.
+//
+	{
+
+	test.Start(_L("The drive list\n"));
+
+
+    TInt err;
+    TInt drivecount = 0;
+    TDriveList driveList;
+    TDriveInfo info;
+    TUint flags;
+    TInt removableDriveCount =0;
+	TInt nonHiddenRemovableDriveCount =0;
+	TInt logicallyRemovableDriveCount =0;
+    TInt substDriveCount =0;
+    TInt exclusiveSubstDriveCount =0;
+	TInt hiddenDriveCount = 0;
+	TInt hiddenOrRemoteDriveCount = 0;
+    
+    TInt i ;
+
+  
+ 	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0544
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909, CR1086
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList) does not return remote or hidden drives.
+	//! @SYMTestActions			Call DriveList to get the drive list.
+	//! @SYMTestExpectedResults	Check that no remote or hidden drive was returned. Count the number of drives we got 
+	//!							and seperately the number of none hidden removable ones.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+ 
+    
+    test.Printf(_L("Test existing DriveList \n"));
+    
+    err = TheFs.DriveList(driveList);
+    test( err == KErrNone );
+    
+    for ( i = 0; i < KMaxDrives; i++) 
+        {
+        if (driveList[i]) 
+            {
+            err = TheFs.Drive(info, i);
+            test( err == KErrNone );
+            test( info.iType  !=  EMediaRemote  );
+            test( !(info.iDriveAtt & KDriveAttRemote ) ); 
+			test( !(info.iDriveAtt & KDriveAttHidden ) );
+            drivecount++; 
+            
+            if( info.iDriveAtt  & KDriveAttRemovable) 
+           	nonHiddenRemovableDriveCount++;
+
+           	printDriveAtt(i,info.iDriveAtt);
+            
+            }
+    
+        }
+        
+    
+    TInt nonHiddenNonRemovables = drivecount -  nonHiddenRemovableDriveCount;
+	test.Printf(_L("Found %d non hidden non removable drives\n"), nonHiddenNonRemovables);
+	test.Printf(_L("Found %d non hidden removable drives \n"),nonHiddenRemovableDriveCount);
+
+
+  
+ 	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0545
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909, CR1086
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can be used to return all
+	//! 						available drives, including the remote and the hidden ones.
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of KDriveAttAll.
+	//! @SYMTestExpectedResults	Check that the number of drives is increased by the number of hidden and remote drives and also that the number of remote drives is one (one remote drive we mounted). 
+	//!							Also count the number of substitute, exclusively substitute, removable, physically removable and logically removable drives.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+ 
+
+   test.Printf(_L("Return all available drives\n"));
+      
+   TInt allDrivecount = 0;
+   flags = KDriveAttAll;
+   err = TheFs.DriveList(driveList, flags);
+
+   test( err == KErrNone );
+   for ( i = 0; i < KMaxDrives; i++) 
+       {
+       if (driveList[i]) 
+           {
+           err = TheFs.Drive(info,i);
+           test( err == KErrNone );
+           allDrivecount++;
+           
+           if( info.iDriveAtt  & KDriveAttSubsted ) 
+           		substDriveCount++;
+            
+           if( info.iDriveAtt  == KDriveAttSubsted) 
+           		exclusiveSubstDriveCount++;
+
+		   if( info.iDriveAtt  & KDriveAttHidden) 
+           		hiddenDriveCount++;
+
+		   if( info.iDriveAtt  & (KDriveAttHidden|KDriveAttRemote))
+           		hiddenOrRemoteDriveCount++;
+
+		   if( info.iDriveAtt  & KDriveAttRemovable) 
+           		removableDriveCount++;
+
+			if( info.iDriveAtt  & KDriveAttLogicallyRemovable) 
+           		logicallyRemovableDriveCount++;
+				           	
+           
+            printDriveAtt(i,info.iDriveAtt);
+           }
+        }  
+
+ 	test(allDrivecount == drivecount + hiddenOrRemoteDriveCount);
+	test(hiddenOrRemoteDriveCount - hiddenDriveCount == 1);
+
+	test.Printf(_L("Found %d substitute drives\n"), substDriveCount);
+	test.Printf(_L("Found %d exclusively substitute  drives \n"),exclusiveSubstDriveCount);
+	test.Printf(_L("Found %d hidden drives\n"), hiddenDriveCount);
+
+	TInt nonRemovables = drivecount -  removableDriveCount;
+	TInt physicallyRemovable = removableDriveCount - logicallyRemovableDriveCount;
+	test.Printf(_L("Found %d non removables drives\n"), nonRemovables);
+	test.Printf(_L("Found %d physically removable drives \n"),physicallyRemovable);
+	test.Printf(_L("Found %d logically removable drives \n"),logicallyRemovableDriveCount);
+  
+  
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0546
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can be used to specify
+	//! 						certain attributes that drives to be returned should have.
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of KDriveAttRemovable.
+	//! @SYMTestExpectedResults	Check that the number of drives returned is the same, since no removable drive was added.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+
+ 
+    test.Printf(_L("Return only Removable \n"));
+	
+	drivecount = 0;
+	
+    flags = KDriveAttRemovable;
+    err = TheFs.DriveList(driveList, flags);
+    test( err == KErrNone );
+    for ( i = 0; i < KMaxDrives; i++) 
+        {
+        if (driveList[i]) 
+            {
+            err = TheFs.Drive(info, i);
+            test( err == KErrNone );
+            test( info.iDriveAtt & KDriveAttRemovable );
+            drivecount++; 
+      
+            printDriveAtt(i,info.iDriveAtt);
+            }
+
+        }
+
+    test( drivecount == removableDriveCount ); // no removable drive was added
+
+
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0547
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can be used to specify
+	//! 						certain attributes(as a combination)that drives to be returned should have.
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of KDriveAttRemovable | KDriveAttRemote
+	//! @SYMTestExpectedResults	Check that the number of drives is increased by one, since we also allow remote drives.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+
+
+ 	test.Printf(_L("Return only Removable and Remote \n"));
+ 	
+    drivecount = 0;
+    flags = KDriveAttRemovable | KDriveAttRemote;
+    err = TheFs.DriveList(driveList, flags);
+    test( err == KErrNone );
+    for ( i = 0; i < KMaxDrives; i++) 
+        {
+        if (driveList[i]) 
+            {
+            err = TheFs.Drive(info, i);
+            test( err == KErrNone );
+            test( (info.iDriveAtt & KDriveAttRemovable ) || (info.iDriveAtt & KDriveAttRemote)); 
+            drivecount++; 
+           
+           	printDriveAtt(i,info.iDriveAtt);
+            
+            }
+
+        }
+    test( drivecount == removableDriveCount + 1 );  //contains the remote drive we mounted
+    
+
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0548
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can be used to exclusively 
+	//!							return drives with certain attributes.
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of KDriveAttExclusive | KDriveAttRemote.
+	//! @SYMTestExpectedResults	Check that only the remote drive that was mounted is returned.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+
+ 	
+ 	test.Printf(_L("Return Exclusively Remote drives \n"));
+ 	
+ 	drivecount = 0;
+    flags = KDriveAttExclusive | KDriveAttRemote;
+    TUint match = KDriveAttRemote;
+    err = TheFs.DriveList(driveList, flags);
+    test( err == KErrNone );
+    for ( i = 0; i < KMaxDrives; i++) 
+        {
+        if (driveList[i]) 
+            {
+            err = TheFs.Drive(info, i);
+            test( err == KErrNone );
+            test( (info.iDriveAtt == match)); 
+            drivecount++;
+            
+            printDriveAtt(i,info.iDriveAtt);
+            }
+
+        }
+   	test( drivecount == 1 ); //The remote drive we mounted.
+
+
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0549
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can be used to exclude drives 
+	//!							to be returned with certain attributes.
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of KDriveAttExclude | KDriveAttRemovable.
+	//! @SYMTestExpectedResults	Check that the remote drive we mounted is included as a non removable one.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+
+	
+   test.Printf(_L("Exclude Removable\n"));
+
+   drivecount = 0; 	
+   flags = KDriveAttExclude | KDriveAttRemovable;
+   err = TheFs.DriveList(driveList, flags);
+   test( err == KErrNone );
+   for (i = 0; i < KMaxDrives; i++) 
+       {
+       if (driveList[i]) 
+           {
+           err = TheFs.Drive(info, i);
+           test( err == KErrNone );
+           test( (!(info.iDriveAtt & KDriveAttRemovable ) )); 
+           drivecount++;
+           
+           printDriveAtt(i,info.iDriveAtt);
+           }
+
+       }  
+     test ( drivecount == allDrivecount - removableDriveCount); 
+	 test ( drivecount == nonRemovables + hiddenDriveCount + 1) ;   //The remote drive we added is non removable  
+
+
+
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0550
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can be used to exclude 
+	//! 					    drives to be returned with certain attributes(as a combination).
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of 
+	//!							KDriveAttExclude | KDriveAttRemovable | KDriveAttRemote. 						
+	//! @SYMTestExpectedResults	Test that only the remote drive that was mounted is returned.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+  
+ 
+ 
+   test.Printf(_L("Exclude Removable and Remote\n"));
+   
+   drivecount = 0;
+   flags = KDriveAttExclude | KDriveAttRemovable | KDriveAttRemote;
+   err = TheFs.DriveList(driveList, flags);
+   
+   test( err == KErrNone );
+   
+   for ( i = 0; i < KMaxDrives; i++) 
+       {
+       if (driveList[i]) 
+           {
+           err = TheFs.Drive(info,i);
+           test( err == KErrNone );
+           test( (!(info.iDriveAtt & KDriveAttRemovable ) && (!(info.iDriveAtt & KDriveAttRemote ))));
+           drivecount++;
+           
+           printDriveAtt(i,info.iDriveAtt);
+           }
+       }
+   test(drivecount == (allDrivecount - removableDriveCount - 1)  ); // also excluding the removables and the remote drive   
+  
+  
+
+
+
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0551
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test RFs::DriveList(TDriveList& aList, TUint aFlags) with substitute drives.  	
+	//! @SYMTestActions			Call DriveList by passing to aFlags combinations of attributes that exclude, exclusive or
+	//!							not substitute drives.
+	//! @SYMTestExpectedResults	Test that only the appropriate drives are returned.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+
+
+
+	if (substDrive)
+		{
+
+
+   		test.Printf(_L("Exclude Remote and Substed\n"));
+   
+   		drivecount = 0;
+   		flags = KDriveAttExclude | KDriveAttRemote | KDriveAttSubsted;
+   		err = TheFs.DriveList(driveList, flags);
+   
+   		test( err == KErrNone );
+   
+   		for ( i = 0; i < KMaxDrives; i++) 
+       		{
+       		if (driveList[i]) 
+           		{
+           		err = TheFs.Drive(info,i);
+           		test( err == KErrNone );
+           		test( (!(info.iDriveAtt & KDriveAttRemote )  && (!(info.iDriveAtt & KDriveAttSubsted ))));
+           		drivecount++;
+           
+           		printDriveAtt(i,info.iDriveAtt);
+           		}
+       		}
+    	test(drivecount == (allDrivecount - substDriveCount- 1)  );    
+
+
+		
+   		test.Printf(_L("Exclusively Exclude Substed drives\n"));
+   
+   		drivecount = 0;
+   		flags = KDriveAttExclusive | KDriveAttExclude | KDriveAttSubsted;
+   		err = TheFs.DriveList(driveList, flags);
+   
+   		test( err == KErrNone );
+   
+   		for ( i = 0; i < KMaxDrives; i++) 
+       		{
+       		if (driveList[i]) 
+           		{
+           		err = TheFs.Drive(info,i);
+           		test( err == KErrNone );
+           		test( info.iDriveAtt != KDriveAttSubsted );
+           		drivecount++;
+           
+           		printDriveAtt(i,info.iDriveAtt);
+           		}		
+   		    
+   		    }
+    
+		test(drivecount == (allDrivecount - exclusiveSubstDriveCount)  );        
+		
+		}
+
+
+  	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0552
+	//! @SYMTestType			UT 
+	//! @SYMREQ					CR909
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) returns the correct value for every
+	//!							combination of matching mask.
+	//! @SYMTestActions			Call DriveList for every combination of mask and check that the correct value is returned.   
+	//!							A structure is used to store the expected value for each combination.
+	//! @SYMTestExpectedResults	Test for every combination that only drives with the correct attributes are retruned.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented 
+	//--------------------------------------------- 
+
+   
+  	test.Printf(_L("Test All Combinations \n"));
+
+	struct TCombinations
+		{
+		TUint iMatchMask;			  // The Match Mask to be combined with drive attributes
+		TInt  iExpectedResultNoAtts;	  // Expected result when flag used on it's own
+		TInt  iExpectedResultWithAtts;  // Expected result when flag used in combination with drive flags
+		};
+
+	TCombinations testCombinations[] = {
+		{ 0,														KErrNone,     KErrNone},
+		{ KDriveAttAll,												KErrNone,     KErrArgument },
+		{ KDriveAttExclude,											KErrArgument, KErrNone },
+		{ KDriveAttExclusive,										KErrArgument, KErrNone },
+		{ KDriveAttExclude | KDriveAttExclusive,					KErrArgument, KErrNone },
+		{ KDriveAttAll	   | KDriveAttExclude,						KErrArgument, KErrArgument },
+		{ KDriveAttAll     | KDriveAttExclusive,					KErrArgument, KErrArgument},
+		{ KDriveAttAll     | KDriveAttExclude | KDriveAttExclusive, KErrArgument, KErrArgument}};
+
+	TDriveList fullDriveList;
+	err = TheFs.DriveList(fullDriveList, KDriveAttAll);
+	test( err == KErrNone );
+
+	for ( i = 0; i < KMaxDrives; i++) 
+		{
+		if (driveList[i]) 
+			{
+			err = TheFs.Drive(info,i);
+			test( err == KErrNone );
+			printDriveAtt(i,info.iDriveAtt);
+			}
+		}
+
+	for(TUint matchIdx = 0; matchIdx < sizeof(testCombinations) / sizeof(TCombinations); matchIdx++)
+		{
+		test.Printf(_L("\nMatch Flags: KDriveAttAll[%c] KDriveAttExclude[%c] KDriveAttExclusive[%c]\n"), testCombinations[matchIdx].iMatchMask & KDriveAttAll       ? 'X' : ' ',
+																										 testCombinations[matchIdx].iMatchMask & KDriveAttExclude   ? 'X' : ' ',
+																										 testCombinations[matchIdx].iMatchMask & KDriveAttExclusive ? 'X' : ' ');
+
+		for(TUint testAtt = 0; testAtt <= KMaxTUint8; testAtt++)
+			{
+			TDriveList newDriveList;
+			err = TheFs.DriveList(newDriveList, testCombinations[matchIdx].iMatchMask | testAtt);
+	 		
+			//test.Printf(_L("            ATT : 0x%08x \n"), testAtt);
+			//test.Printf(_L("Expected Result : %d     \n"), testAtt == 0 ? testCombinations[matchIdx].iExpectedResultNoAtts : testCombinations[matchIdx].iExpectedResultWithAtts);
+			//test.Printf(_L("  Actual Result : 0x%08x \n"), err);
+
+			test( err == (testAtt == 0 ? testCombinations[matchIdx].iExpectedResultNoAtts : testCombinations[matchIdx].iExpectedResultWithAtts) );
+
+			if(err == KErrNone)
+				{
+				//printDriveAtt(0, testAtt);  //Prints attributes   
+				for ( i = 0; i < KMaxDrives; i++) 
+					{
+					TBool expectMatch = EFalse;                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             
+
+					switch(testCombinations[matchIdx].iMatchMask)
+						{
+						case 0:
+							expectMatch = (fullDriveList[i] & testAtt) != 0;                                                                                                                                                           
+							break;
+
+						case KDriveAttAll:
+							expectMatch = ETrue;
+							break;
+
+						case KDriveAttExclude :
+							expectMatch = (fullDriveList[i] & testAtt) == 0;
+							break;
+
+						case KDriveAttExclusive :
+							expectMatch = (fullDriveList[i] == testAtt);
+							break;
+
+						case KDriveAttExclude | KDriveAttExclusive:
+							expectMatch = (fullDriveList[i] != testAtt);
+							break;
+	
+						case KDriveAttAll | KDriveAttExclude :
+							test(0);	// Invalid - should never get here as this returns KErrArgument for all cases
+							break;
+
+						case KDriveAttAll | KDriveAttExclusive :
+							test(0);	// Invalid - should never get here as this returns KErrArgument for all cases
+							break;
+
+						case KDriveAttAll | KDriveAttExclude | KDriveAttExclusive :
+							test(0);	// Invalid - should never get here as this returns KErrArgument for all cases
+							break;
+
+						default:
+							test.Printf(_L("Unexpected Match Mask %08x"), testCombinations[matchIdx].iMatchMask);
+							test(0);
+							break;
+						}
+
+					if(expectMatch) 
+						{
+						//test.Printf(_L(" %c MATCHED OK "), 'A' + i);
+						test(newDriveList[i] == fullDriveList[i]);
+						}
+					else
+						{
+						/*if(fullDriveList[i] == 0)
+							{
+							test.Printf(_L(" %c NOT PRESENT "), 'A' + i);
+							}
+						else
+							{
+							test.Printf(_L(" %c NOT MATCHED "), 'A' + i);
+							}
+						*/
+						test(newDriveList[i] == 0);
+						}
+					}
+				}
+			}
+		}
+
+
+
+	//--------------------------------------------- 
+	//! @SYMTestCaseID			PBASE-T_FSRV-0605
+	//! @SYMTestType			UT
+	//! @SYMREQ					CR1086
+	//! @SYMTestCaseDesc		Test that RFs::DriveList(TDriveList& aList, TUint aFlags) can identify the 
+	//!							logically removable drives.
+	//! @SYMTestActions			Call DriveList by passing to aFlags a value of KDriveAttLogicallyRemovable.
+	//! @SYMTestExpectedResults	Check that only the logically removable drives specified in estart.txt are returned.
+	//! @SYMTestPriority		High
+	//! @SYMTestStatus			Implemented
+	//--------------------------------------------- 
+
+	test.Printf(_L("Return only Logically Removable drives \n"));
+ 	
+ 	drivecount = 0;
+    flags = KDriveAttLogicallyRemovable;
+    err = TheFs.DriveList(driveList, flags);
+    test( err == KErrNone );
+    for ( i = 0; i < KMaxDrives; i++) 
+        {
+        if (driveList[i]) 
+            {
+            err = TheFs.Drive(info, i);
+            test( err == KErrNone );
+            test( info.iDriveAtt & KDriveAttLogicallyRemovable );
+            drivecount++; 
+      
+            printDriveAtt(i,info.iDriveAtt);
+            }
+
+        }
+
+    test( drivecount == logicallyRemovableDriveCount ); // no logically removable drive was added
+ 
+
+	test.End();
+	}
+	
+
+
+
+
+
+
+LOCAL_C void testDriveInfo()
+//
+// Test the drive info.
+//
+	{
+
+	test.Start(_L("The drive info"));
+	TDriveList list;
+	TInt r=TheFs.DriveList(list);
+	test(r==KErrNone);
+	for (TInt i=0;i<KMaxDrives;i++)
+		{
+		TInt att=list[i];
+		if (att)
+			{
+			TDriveInfo d;
+			r=TheFs.Drive(d,i);
+			test(r==KErrNone);
+			printDriveInfo(i,d);
+			test.Printf(_L("\n"));
+			testDriveInfo(i,d);
+			}
+		}
+
+	test.End();
+	}
+
+LOCAL_C void testVolumeInfo()
+//
+// Test volume info.
+//
+	{
+
+	test.Start(_L("The volume info"));
+	TDriveList list;
+	TInt r=TheFs.DriveList(list);
+	test(r==KErrNone);
+	for (TInt i=0;i<KMaxDrives;i++)
+		{
+		TVolumeInfo v;
+		TDriveInfo d;
+		switch (r=TheFs.Volume(v,i))
+			{
+			case KErrNone:
+				printDriveInfo(i,v.iDrive);
+				test.Printf(_L("   VOL=\"%S\" ID=%08x\n"),&v.iName,v.iUniqueID);
+				test.Printf(_L("   SIZE=%ldK FREE=%ldK\n"),v.iSize/1024,v.iFree/1024);
+				break;
+			case KErrNotReady:
+				r=TheFs.Drive(d, i);
+				test_KErrNone(r);
+				if (d.iType == EMediaNotPresent)
+					test.Printf(_L("%c: Medium not present - cannot perform test.\n"), i + 'A');
+				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, i + 'A');
+				break;
+			case KErrPathNotFound:
+				test.Printf(_L("%c: Not Found\n"), i + 'A');
+				break;
+			case KErrCorrupt:
+				test.Printf(_L("%c: Media corruption; previous test may have aborted; else, check hardware\n"), i + 'A');
+			default:
+				test_KErrNone(r);
+			}
+		test.Printf(_L("\n"));
+		}
+
+	test.End();
+	}
+
+LOCAL_C void testClientParse()
+//
+// Test the client side parse.
+//
+	{
+
+	test.Start(_L("Test client side parsing"));
+
+	for (TInt i=0;i<KMaxParses;i++)
+		{
+		TInt r;
+		TParse f;
+		SParse& p=parse[i];
+		TPtrC name(p.src);
+		if (p.rel)
+			{
+			if (p.def)
+                {
+                TPtrC rel(p.rel);
+                TPtrC def(p.def);
+				r=f.Set(name,&rel,&def);
+                }
+			else
+                {
+                TPtrC rel(p.rel);
+				r=f.Set(name,&rel,NULL);
+                }
+			}
+		else
+			r=f.Set(name,NULL,NULL);
+		test(r==KErrNone);
+		test(TPtrC(p.fullName)==f.FullName());
+		test(TPtrC(p.drive)==f.Drive());
+		test(TPtrC(p.path)==f.Path());
+		test(TPtrC(p.name)==f.Name());
+		test(TPtrC(p.ext)==f.Ext());
+		}
+
+	test.End();
+	}
+
+LOCAL_C void testPath()
+//
+// Test the path handling.
+//
+	{
+
+	test.Start(_L("Test path handling"));
+	TFileName p;
+	TInt r=TheFs.SessionPath(p);
+	test(r==KErrNone);
+	test.Printf(_L("SESSION=\"%S\"\n"),&p);
+	r=TheFs.SetSessionPath(_L("A:\\TEST\\"));
+	test(r==KErrNone);
+	r=TheFs.SessionPath(p);
+	test(r==KErrNone);
+	test(p==_L("A:\\TEST\\"));
+	r=TheFs.SetSessionPath(gSessionPath);
+	test(r==KErrNone);
+
+
+	TheFs.SetAllocFailure(gAllocFailOff);
+
+	RThread t;
+	r=t.Create(_L("PathTest"),pathTestThread,KDefaultStackSize,KHeapSize,KHeapSize,NULL);
+	test(r==KErrNone);
+	TRequestStatus tStat;
+	t.Logon(tStat);
+	t.Resume();
+	User::WaitForRequest(tStat);
+	test(tStat==KErrNone);
+	test(r==KErrNone);
+	t.Close();
+
+	TheFs.SetAllocFailure(gAllocFailOn);
+
+	test.End();
+	}
+
+LOCAL_C void testServerParse()
+//
+// Test the client side parse.
+//
+	{
+
+	test.Start(_L("Test server side parsing"));
+
+	TFileName old;
+	TInt r=TheFs.SessionPath(old);
+	test(r==KErrNone);
+	r=TheFs.SetSessionPath(_L("C:\\ABCDEF\\"));
+	test(r==KErrNone);
+	for (TInt i=0;i<KMaxParses;i++)
+		{
+		TInt r;
+		TParse f;
+		SParseServer& p=parseServer[i];
+		TPtrC name(p.src);
+		if (p.rel)
+			r=TheFs.Parse(name,TPtrC(p.rel),f);
+		else
+			r=TheFs.Parse(name,f);
+		test(r==KErrNone);
+		test(TPtrC(p.fullName)==f.FullName());
+		test(TPtrC(p.drive)==f.Drive());
+		test(TPtrC(p.path)==f.Path());
+		test(TPtrC(p.name)==f.Name());
+		test(TPtrC(p.ext)==f.Ext());
+		}
+	r=TheFs.SetSessionPath(old);
+	test(r==KErrNone);
+
+	test.End();
+	}
+
+LOCAL_C void testSubst()
+//
+// Test the substitute functions.
+//
+	{
+
+	test.Printf(_L("Test subst"));
+	TVolumeInfo v;
+	TInt r=TheFs.Volume(v);
+	test(r==KErrNone);
+	TDriveInfo origDI;
+	r=TheFs.Drive(origDI);
+	test(r==KErrNone);
+	
+	TDriveInfo driveInfo;
+	r=TheFs.Drive(driveInfo,EDriveO);
+	test(r==KErrNone);
+	
+	if (driveInfo.iDriveAtt==KDriveAttLocal)
+		{	
+		return;	//	Subst local drives fails
+		}
+	
+	TFileName n;
+	r=TheFs.Subst(n,EDriveO);
+	test(r==KErrNone);
+	test(n.Length()==0);
+	r=TheFs.SetSubst(gSessionPath,EDriveO);
+	test(r==KErrNone);
+	r=TheFs.Subst(n,EDriveO);
+	test(r==KErrNone);
+	test(n==gSessionPath);
+	TVolumeInfo w;
+	r=TheFs.Volume(w,EDriveO);
+	test(r==KErrNone);
+	test(w.iDrive.iType==v.iDrive.iType);
+	test(w.iDrive.iBattery==v.iDrive.iBattery);
+	test(w.iDrive.iDriveAtt==KDriveAttSubsted);
+	test(w.iDrive.iMediaAtt==v.iDrive.iMediaAtt);
+	test(w.iUniqueID==v.iUniqueID);
+	test(w.iSize==v.iSize);
+	test(w.iFree==v.iFree);
+	test(w.iName==v.iName);
+	TDriveList driveList;
+	r=TheFs.DriveList(driveList);
+	test(r==KErrNone);
+	test(driveList[EDriveO]==KDriveAttSubsted);
+	TDriveInfo d;
+	r=TheFs.Drive(d,EDriveO);
+	test(r==KErrNone);
+	test(d.iDriveAtt==KDriveAttSubsted);
+	test(d.iMediaAtt==origDI.iMediaAtt);
+	test(d.iType==origDI.iType);
+	test(d.iBattery==origDI.iBattery);
+
+
+	test.Next(_L("Test real name"));
+	r=TheFs.RealName(_L("O:\\FILE.XXX"),n);
+	test(r==KErrNone);
+	TFileName substedPath=gSessionPath;
+	substedPath.Append(_L("FILE.XXX"));
+	test(n.CompareF(substedPath)==KErrNone);
+//
+	test.Next(_L("Test MkDir, Rename and RmDir on Substed drive"));
+	_LIT(KTurgid,"turgid\\");
+	TFileName dir=gSessionPath;
+	dir+=KTurgid;
+	r=TheFs.MkDirAll(dir);
+	test(r==KErrNone);
+	dir+=_L("subdir\\");
+	r=TheFs.MkDir(dir);
+	test(r==KErrNone);
+	r=TheFs.RmDir(_L("O:\\turgid\\subdir\\"));
+	test(r==KErrNone);
+	r=TheFs.Rename(_L("O:\\turgid"), _L("O:\\facile"));
+	test(r==KErrNone);
+	r=TheFs.MkDir(_L("O:\\insipid\\"));
+	test(r==KErrNone);
+	r=TheFs.Rename(_L("O:\\insipid"), _L("O:\\glib"));
+	test(r==KErrNone);
+	r=TheFs.RmDir(_L("O:\\facile\\"));
+	test(r==KErrNone);
+	_LIT(KGlib,"glib\\");
+	dir=gSessionPath;
+	dir+=KGlib;
+	r=TheFs.RmDir(dir);
+	test(r==KErrNone);
+//	
+	test.Next(_L("Test file operations on Substed drive"));
+	_LIT(File1,"File1.txt");
+	_LIT(File2,"File2.txt");
+	_LIT(SubstRoot,"O:\\");
+	_LIT(Subdir,"subdir\\");
+	TFileName name1,name2;
+	name1=gSessionPath;
+	name1+=File1;
+	RFile f1;
+	r=f1.Replace(TheFs,name1,EFileShareExclusive|EFileWrite);
+	test(r==KErrNone);
+	name2=SubstRoot;
+	name2+=File2;
+	TBool isValid=TheFs.IsValidName(name2);
+	test(isValid);
+	r=f1.Rename(name2);
+	test(r==KErrNone);
+	f1.Close();
+	r=f1.Create(TheFs,name1,EFileShareExclusive|EFileWrite);
+	test(r==KErrNone);
+	f1.Close();
+	r=TheFs.Replace(name2,name1);
+	test(r==KErrNone);
+	r=TheFs.Delete(name1);
+	test(r==KErrNone);
+	test.Next(_L("Test notifications on Substed drive"));
+	name1=gSessionPath;
+	name1+=Subdir;
+	name2=SubstRoot;
+	name2+=Subdir;
+	// set up some extended notifications
+	TRequestStatus status1;
+	TRequestStatus status2;
+	TRequestStatus status3;
+	TheFs.NotifyChange(ENotifyDir,status1,name1);
+	test(status1==KRequestPending);
+	TheFs.NotifyChange(ENotifyDir,status2,name2);
+	test(status2==KRequestPending);
+	r=TheFs.MkDirAll(name1);
+	test(r==KErrNone);
+	User::WaitForRequest(status1);
+	User::WaitForRequest(status2);
+	test(status1==KErrNone && status2==KErrNone);
+	TheFs.NotifyChange(ENotifyDir,status1,name1);
+	test(status1==KRequestPending);
+	TheFs.NotifyChange(ENotifyDir,status2,name2);
+	test(status2==KRequestPending);
+	TheFs.NotifyChange(ENotifyAll,status3,name2);
+	test(status3==KRequestPending);
+	r=f1.Temp(TheFs,name2,n,EFileShareAny|EFileWrite);
+	test(r==KErrNone);
+	User::WaitForRequest(status3);
+	test(status3==KErrNone && status1==KRequestPending && status2==KRequestPending);
+	f1.Close();
+	TheFs.NotifyChangeCancel();
+	test(status1==KErrCancel && status2==KErrCancel);
+	r=TheFs.Delete(n);
+	test(r==KErrNone);
+	r=TheFs.RmDir(name1);
+	test(r==KErrNone);
+//
+	test.Next(_L("Test file systems on Substed drive"));
+	// test cannot mount file system on substituted drive
+	TInt sessionDrv;
+	r=TheFs.CharToDrive(gSessionPath[0],sessionDrv);
+	test(r==KErrNone);
+	r=TheFs.FileSystemName(n,sessionDrv);
+	test(r==KErrNone || r==KErrNotFound);
+	r=TheFs.MountFileSystem(n,EDriveO);
+	test(r==KErrAccessDenied);
+	// test file system name on substitued drive is null
+	r=TheFs.FileSystemName(n,EDriveO);
+	test(r==KErrNotFound && n==KNullDesC);
+	// test cannot format a substitued drive
+	RFormat format;
+	TInt count;
+	r=format.Open(TheFs,SubstRoot,EHighDensity,count);
+	test(r==KErrAccessDenied);
+	
+	r=TheFs.SetSubst(_L(""),EDriveO);
+	test(r==KErrNone);
+	r=TheFs.Subst(n,EDriveO);
+	test(r==KErrNone);
+	test(n==_L(""));
+	r=TheFs.Drive(d,EDriveO);
+	test(r==KErrNone);
+	test(d.iDriveAtt==0);
+	}
+
+LOCAL_C void testSetVolume()
+//
+// Test setting the volume info.
+//
+	{
+
+	test.Start(_L("Test setting the volume label"));
+
+	const TInt driveNum=CurrentDrive();
+
+	TVolumeInfo v;
+	TInt r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+	TFileName n=v.iName;
+	test.Printf(_L("VOL=\"%S\"\n"),&n);
+
+	test.Next(_L("Set volume label to nothing"));
+	r=TheFs.SetVolumeLabel(_L(""),driveNum);
+
+	if(Is_Win32(TheFs, gDrive) && (r==KErrGeneral || r==KErrAccessDenied || r==KErrNotSupported))
+		{
+		test.Printf(_L("Error %d: Set volume label not testing on WINS\n"),r);
+		test.End();
+		return;
+		}
+
+	test(r==KErrNone);
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone );
+	test(v.iName==_L(""));
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+	test.Next(_L("Set volume label to ABCDEFGHIJK"));
+	r=TheFs.SetVolumeLabel(_L("ABCDEFGHIJK"),driveNum);
+	test(r==KErrNone);
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+	test(v.iName==_L("ABCDEFGHIJK"));
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+	test.Next(_L("Set volume label to ABCDE"));
+	r=TheFs.SetVolumeLabel(_L("ABCDE"),driveNum);
+	test(r==KErrNone);
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+	test(v.iName==_L("ABCDE"));
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+
+	test.Next(_L("Test replacement of non-ascii chars"));
+	TBuf<KMaxVolumeName> uBuf(KMaxVolumeName);
+	uBuf.FillZ();
+	uBuf[0]='a';
+	uBuf[1]=0x100;
+	uBuf[2]='b';
+	uBuf[3]=0x101;
+	uBuf[4]='c';
+	uBuf[5]=0x102;
+	uBuf[6]='d';
+	uBuf[7]=0x103;
+	uBuf[8]='e';
+	uBuf[9]=0x104;
+	uBuf[10]='f';
+	r=TheFs.SetVolumeLabel(uBuf,driveNum);
+	test(r==KErrNone);
+	TFileName drive=_L("?:");
+	drive[0]=gSessionPath[0];
+
+// ??? this needs to be replaced
+//	UserSvr::ForceRemountMedia(ERemovableMedia0);
+	User::After(1000000);
+
+	TFileName sess;
+	r=TheFs.SessionPath(sess);
+	test(r==KErrNone);
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+
+	if(Is_Fat(TheFs, gDrive)) //-- FAT doesn't support normal UNICODE in volume labels
+		test(v.iName==_L("a_b_c_d_e_f"));
+	else
+		test(v.iName == uBuf);
+
+
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+
+	test.Next(_L("Set volume label back to nothing"));
+	r=TheFs.SetVolumeLabel(_L(""),driveNum);
+	test(r==KErrNone);
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+	test(v.iName==_L(""));
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+	test.Next(_L("Attempt to set volume label containing illegal characters"));
+	r=TheFs.SetVolumeLabel(_L("abc>def"),driveNum);
+	test(r==KErrBadName);
+	r=TheFs.SetVolumeLabel(_L("ghi*jkl"),driveNum);
+	test(r==KErrBadName);
+	r=TheFs.SetVolumeLabel(_L("mno?pqr"),driveNum);
+	test(r==KErrBadName);
+	r=TheFs.SetVolumeLabel(_L("stu|vwx"),driveNum);
+	test(r==KErrBadName);
+	r=TheFs.SetVolumeLabel(_L("yz<abc"),driveNum);
+	test(r==KErrBadName);
+	r=TheFs.SetVolumeLabel(_L("def//ghi"),driveNum);
+	test(r==KErrBadName);
+
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+	test(v.iName==_L(""));
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+	// test volume label after remount (for removable media only)
+	test.Next(_L("Test volume label after remount"));
+
+	TDriveInfo info;
+	test(TheFs.Drive(info, driveNum) == KErrNone);
+
+	if((info.iDriveAtt & KDriveAttRemovable) != 0)
+		{
+		// 1. set volume label
+		test(TheFs.SetVolumeLabel(_L("XXX"), driveNum) == KErrNone);
+
+		// 2. change bootsector volume label
+		const TInt	offset = IsFileSystemFAT32(TheFs, driveNum)? 
+			71 /*KFat32VolumeLabelPos*/ 
+			: 
+			43 /*KFat16VolumeLabelPos*/;	// both from sfat32\inc\sl_bpb.h
+
+		RRawDisk	rdisk;
+		TPtrC8		label(_S8("Z"), 1);
+
+		test(rdisk.Open(TheFs, driveNum) == KErrNone);
+		test(rdisk.Write(offset, label) == KErrNone);
+		rdisk.Close();
+
+		// 3. remount the drive
+		test(TheFs.RemountDrive(driveNum) == KErrNone);
+
+		// 4. check volume label
+		test(TheFs.Volume(v, driveNum) == KErrNone);
+		test(v.iName == _L("XXX"));
+		test.Printf(_L("- Passed.\n"));
+		}
+	else
+		test.Printf(_L("- Skipped (drive is not removable).\n"));
+
+	// clean up
+	test.Next(_L("Set volume label to original"));
+	r=TheFs.SetVolumeLabel(n,driveNum);
+	test(r==KErrNone);
+	r=TheFs.Volume(v,driveNum);
+	test(r==KErrNone);
+	test(v.iName==n);
+	test.Printf(_L("VOL=\"%S\"\n"),&v.iName);
+
+	test.End();
+	}
+
+LOCAL_C void testModified()
+//
+// Test the Modified/SetModified functions.
+//
+	{
+	
+	test.Start(_L("Test modified/SetModified functions"));
+	TTime savedTime;
+	TInt r=TheFs.Modified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),savedTime);
+	test(r==KErrNone);
+	TDateTime dateTime=savedTime.DateTime();
+	test.Printf(_L("T_FSRV.CPP last modified %d/%d/%d %d:%d:%d.%-06d\n"),dateTime.Day()+1,dateTime.Month()+1,dateTime.Year(),dateTime.Hour(),dateTime.Minute(),dateTime.Second(),dateTime.MicroSecond());
+	test.Next(_L("Set modified"));
+	dateTime.Set(1993,EAugust,23,1,13,54,123456);
+	TTime newTime(dateTime);
+	r=TheFs.SetModified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),newTime);
+	test(r==KErrNone);
+	TTime checkTime;
+	r=TheFs.Modified(_L("\\XXXX\\YYYY\\ZZZZ.CPP"),checkTime);
+	test(r==KErrPathNotFound);
+	r=TheFs.Modified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),checkTime);
+	test(r==KErrNone);
+	dateTime=checkTime.DateTime();	
+	test(dateTime.Year()==1993);
+	test(dateTime.Month()==EAugust);
+	test(dateTime.Day()==23);
+	test(dateTime.Hour()==1);
+	test(dateTime.Minute()==13);
+	test(dateTime.Second()==54);
+	test.Printf(_L("T_FSRV.CPP last modified %d/%d/%d %d:%d:%d.%-06d\n"),dateTime.Day()+1,dateTime.Month()+1,dateTime.Year(),dateTime.Hour(),dateTime.Minute(),dateTime.Second(),dateTime.MicroSecond());
+//		test(dateTime.MicroSecond()==123456); // dos is not accurate enough
+	r=TheFs.SetModified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),savedTime);
+	test(r==KErrNone);
+	r=TheFs.Modified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),checkTime);
+	test(r==KErrNone);
+	test(checkTime==savedTime);
+
+	RFile f;
+	r=f.Open(TheFs,_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),EFileWrite);
+	test(r==KErrNone);
+	dateTime.Set(1997,EJanuary,1,2,55,51,999999);
+	newTime=dateTime;
+	r=f.SetModified(newTime);
+	test(r==KErrNone);
+	r=TheFs.Modified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),checkTime);
+	test(r==KErrNone);
+
+	dateTime=checkTime.DateTime();	
+	test(dateTime.Year()==1997);
+	test(dateTime.Month()==EJanuary);
+	test(dateTime.Day()==1);
+	test(dateTime.Hour()==2);
+	test(dateTime.Minute()==55);
+	test(dateTime.Second()>=50 && dateTime.Second()<=51); // Dos stores seconds %2
+
+	r=f.Modified(checkTime);
+	test(r==KErrNone);
+
+	dateTime=checkTime.DateTime();	
+	test(dateTime.Year()==1997);
+	test(dateTime.Month()==EJanuary);
+	test(dateTime.Day()==1);
+	test(dateTime.Hour()==2);
+	test(dateTime.Minute()==55);
+	test(dateTime.Second()>=50 && dateTime.Second()<=51); // Dos stores seconds %2
+	f.Close();
+
+	r=TheFs.Modified(_L("\\F32-TST\\TFSRV\\T_FSRV.CPP"),checkTime);
+	test(r==KErrNone);
+
+	dateTime=checkTime.DateTime();	
+	test(dateTime.Year()==1997);
+	test(dateTime.Month()==EJanuary);
+	test(dateTime.Day()==1);
+	test(dateTime.Hour()==2);
+	test(dateTime.Minute()==55);
+	test(dateTime.Second()>=50 && dateTime.Second()<=51); // Dos stores seconds %2
+	test.End();
+	}
+
+LOCAL_C void testName()
+//
+// Test the Modified/SetModified functions.
+//
+	{
+	test.Start(_L("Test file name functions"));
+
+
+	_LIT(KFileNameAndPath, "\\F32-TST\\TFSRV\\T_FSRV.CPP");
+	_LIT(KFileName, "T_FSRV.CPP");
+	
+	RFile file;
+	
+	TInt r=file.Open(TheFs, KFileName, 0 );
+	if (r!=KErrNone)
+		{
+		test.Printf(_L("Error %d opening file %S\n"), r, &KFileName);
+		test(0);
+		}
+	
+	TFileName fileName;
+
+	// Check RFile::Name just retuns the file name, without path and drive
+	r=file.Name(fileName);
+	test(r==KErrNone);
+	test(fileName==KFileName());
+
+	// Check RFile::FullName returns the complete file name and path
+	r=file.FullName(fileName);
+	test(r==KErrNone);
+	test(fileName.Mid(2)==KFileNameAndPath); // chop off drive letter + ':'
+	
+	file.Close();
+	
+	test.End();
+	}
+	
+LOCAL_C TInt CreateFileX(const TDesC& aBaseName,TInt aX)
+//
+// Create a large file. Return KErrEof or KErrNone
+//
+	{
+
+	TBuf<128> fileName=aBaseName;
+	fileName.AppendNum(aX);
+	RFile file;
+
+	TInt r=file.Replace(TheFs,fileName,EFileWrite);
+	if (r==KErrDiskFull)
+		return(r);
+	if (r!=KErrNone)
+		{
+		test.Printf(_L("ERROR:: Replace returned %d\n"),r);
+		test(0);
+		//test.Getch();
+		return(KErrDiskFull);
+		}
+
+	if (!IsTestingLFFS())
+		r=file.SetSize(LargeFileSize);
+	else
+		{ // ??? Whats wrong with setsize 
+    	TBuf8<1024> testdata(1024);
+    	TInt count=(LargeFileSize/testdata.Length());
+    	r=KErrNone;
+    	while (count-- && r==KErrNone) 
+        	r=file.Write(testdata);
+		}
+	if (r==KErrDiskFull)
+		{
+		file.Close();
+		return(r);
+		}
+	if (r!=KErrNone)
+		{
+		test.Printf(_L("ERROR:: SetSize/Write returned %d\n"),r);
+		test(0);
+		//test.Getch();
+		file.Close();
+		return(KErrDiskFull);
+		}
+
+	file.Close();
+//	r=TheFs.CheckDisk(fileName);
+//	if (r!=KErrNone && r!=KErrNotSupported)
+//		{
+//		test.Printf(_L("ERROR:: CheckDisk returned %d\n"),r);
+//		test.Getch();
+//		return(KErrDiskFull);
+//		}
+	test.Printf(_L("Created file %d size %d\n"),aX,LargeFileSize);
+	return(KErrNone);
+	}
+
+LOCAL_C TInt DeleteFileX(TBuf<128>& aBaseName,TInt aX)
+//
+// Delete a file.
+//
+	{
+
+	TBuf<128> fileName=aBaseName;
+	fileName.AppendNum(aX);
+	TInt r=TheFs.Delete(fileName);
+	test(r==KErrNone);
+//	r=TheFs.CheckDisk(fileName);
+//	if (r!=KErrNone && r!=KErrNotSupported)
+//		{
+//		test.Printf(_L("ERROR:: CheckDisk returned %d\n"),r);
+//		test(r==KErrNone);
+//		}
+	test.Printf(_L("Deleted File %d\n"),aX);
+	return(KErrNone);
+	}
+
+LOCAL_C void MakeAndDeleteFiles()
+//
+// Create and delete large files in a randomish order
+//
+	{
+
+	test.Start(_L("Create and delete large files"));
+	TInt r=TheFs.MkDirAll(_L("\\F32-TST\\SMALLDIRECTORY\\"));
+	test(r==KErrNone || r==KErrAlreadyExists);
+	TBuf<128> fileName=_L("\\F32-TST\\SMALLDIRECTORY\\FILE");
+	r=CreateFileX(fileName,0);
+	test(r==KErrNone);
+	r=CreateFileX(fileName,1);
+	test(r==KErrNone);
+	r=DeleteFileX(fileName,0);	
+	test(r==KErrNone);
+	r=CreateFileX(fileName,2);
+	test(r==KErrNone);
+	r=CreateFileX(fileName,1);
+	test(r==KErrNone);
+	r=CreateFileX(fileName,3);
+	test(r==KErrNone);
+	r=DeleteFileX(fileName,1);	
+	test(r==KErrNone);
+	r=CreateFileX(fileName,4);
+	test(r==KErrNone);
+	r=DeleteFileX(fileName,2);	
+	test(r==KErrNone);
+	r=DeleteFileX(fileName,3);	
+	test(r==KErrNone);
+	r=DeleteFileX(fileName,4);	
+	test(r==KErrNone);
+	r=CreateFileX(fileName,1);
+	test(r==KErrNone);
+	r=DeleteFileX(fileName,1);	
+	test(r==KErrNone);
+
+	r=TheFs.CheckDisk(fileName);
+	if (r!=KErrNone && r!=KErrNotSupported)
+		{
+		test.Printf(_L("ERROR:: CheckDisk returned %d\n"),r);
+		test(0);
+		//test.Getch();
+		}
+	
+	test.End();
+	}
+
+LOCAL_C void FillUpDisk()
+//
+// Test that a full disk is ok
+//
+	{
+
+	test.Start(_L("Fill disk to capacity"));
+	TInt r=TheFs.MkDirAll(_L("\\F32-TST\\BIGDIRECTORY\\"));
+	test(r==KErrNone || r==KErrAlreadyExists);
+	TInt count=0;
+	TFileName sessionPath;
+	r=TheFs.SessionPath(sessionPath);
+	test(r==KErrNone);
+	TBuf<128> fileName=_L("\\F32-TST\\BIGDIRECTORY\\FILE");
+	FOREVER
+		{
+		TInt r=CreateFileX(fileName,count);
+		if (r==KErrDiskFull)
+			break;
+		test(r==KErrNone);
+		count++;
+#if defined(__WINS__)
+		if (count==32 && sessionPath[0]=='C')
+			break;
+#endif
+		}
+
+	r=TheFs.CheckDisk(fileName);
+	if (r!=KErrNone && r!=KErrNotSupported)
+		{
+		test.Printf(_L("ERROR:: CheckDisk returned %d\n"),r);
+		test(0);
+		//test.Getch();
+		}
+
+	while(count--)
+		DeleteFileX(fileName,count);
+
+	r=TheFs.CheckDisk(fileName);
+	if (r!=KErrNone && r!=KErrNotSupported)
+		{
+		test.Printf(_L("ERROR:: CheckDisk returned %d\n"),r);
+		test(0);
+		//test.Getch();
+		}
+
+	test.End();
+	}
+
+LOCAL_C void CopyFileToTestDirectory()
+//
+// Make a copy of the file in ram
+//
+	{
+
+	TFileName fn = _L("Z:\\TEST\\T_FSRV.CPP");
+	fn[0] = gExeFileName[0];
+	TParse f;
+	TInt r=TheFs.Parse(fn,f);
+	test(r==KErrNone);
+	test.Next(_L("Copying file to test directory"));
+	TParse fCopy;
+	r=TheFs.Parse(f.NameAndExt(),fCopy);
+	test(r==KErrNone);
+
+	RFile f1;
+	r=f1.Open(TheFs,f.FullName(),EFileStreamText|EFileShareReadersOnly);
+	test(r==KErrNone);
+	RFile f2;
+	r=f2.Replace(TheFs,fCopy.FullName(),EFileWrite);
+	test(r==KErrNone);
+	TBuf8<512> copyBuf;
+	TInt rem;
+	r=f1.Size(rem);
+	test(r==KErrNone);
+	TInt pos=0;
+	while (rem)
+		{
+		TInt s=Min(rem,copyBuf.MaxSize());
+		r=f1.Read(pos,copyBuf,s);
+		test(r==KErrNone);
+		test(copyBuf.Length()==s);
+		r=f2.Write(pos,copyBuf,s);
+		test(r==KErrNone);
+		pos+=s;
+		rem-=s;
+		}
+	f1.Close();
+	f2.Close();
+	}
+
+
+//---------------------------------------------------------------------------
+
+/**
+    test RFs::SetErrorCondition() aspects
+*/
+void TestSetErrorCondition()
+{
+#ifdef _DEBUG
+
+    test.Next(_L("TestSetErrorCondition\n"));
+
+    _LIT(KFileName, "\\A.swf");
+
+    TInt        nRes;
+    RFile       file;    
+    RFile       file1;
+    
+    const TInt KMyError = -756; //-- specific error code we will simulate
+    
+    //==========  just create a file
+    nRes = TheFs.SetErrorCondition(KMyError,0); //-- set up FS error simulation
+    test(nRes == KErrNone);
+
+    //-- this shall fail immediately 
+    nRes = file.Replace(TheFs, KFileName, EFileWrite);
+    test(nRes == KMyError);
+
+    nRes = TheFs.SetErrorCondition(KErrNone); //-- disable FS error simulation
+    file.Close();
+
+    //========== create file & duplicate a handle #1
+    nRes = TheFs.SetErrorCondition(KMyError,1); //-- set up FS error simulation
+    test(nRes == KErrNone);
+
+    //-- this shall succeed
+    nRes = file.Replace(TheFs, KFileName, EFileWrite); //-- err cnt -> 0
+    test(nRes == KErrNone);
+
+    //-- this shall fail inside RFile::Duplicate() half way through in the RFile::DuplicateHandle()
+    nRes = file1.Duplicate(file);
+    test(nRes == KMyError);
+    file1.Close();
+    
+    nRes = TheFs.SetErrorCondition(KErrNone); //-- disable FS error simulation
+    file.Close();
+
+    //-- check that the file isn't locked
+    nRes = TheFs.Delete(KFileName);
+    test(nRes == KErrNone);
+
+    //========== create file & duplicate a handle #2
+    nRes = TheFs.SetErrorCondition(KMyError,2); //-- set up FS error simulation
+    test(nRes == KErrNone);
+
+    //-- this shall succeed
+    nRes = file.Replace(TheFs, KFileName, EFileWrite); //-- err cnt -> 1
+    test(nRes == KErrNone);
+
+    //-- this must not fail, because EFsFileAdopt is excluded from the erros simulation
+    nRes = file1.Duplicate(file);
+    test(nRes == KErrNone);
+    file1.Close();
+    
+    nRes = TheFs.SetErrorCondition(KErrNone); //-- disable FS error simulation
+    file.Close();
+
+    //-- check that the file isn't locked
+    nRes = TheFs.Delete(KFileName);
+    test(nRes == KErrNone);
+
+    //========== crazy loop, for DEF103757
+
+    for(TInt i=0; i<6; ++i)
+    {
+        nRes = TheFs.SetErrorCondition(KMyError,i); //-- set up FS error simulation
+        
+        nRes = file.Replace(TheFs, KFileName, EFileWrite); //-- err cnt -> 1
+        if(nRes != KErrNone)
+            continue;
+
+        nRes = file1.Duplicate(file);
+
+        file1.Close();
+        file.Close();
+        
+        nRes = TheFs.SetErrorCondition(KErrNone); //-- disable FS error simulation
+    }
+
+    //-- check that the file isn't locked
+    nRes = TheFs.Delete(KFileName);
+    test(nRes == KErrNone);
+
+
+#endif
+}
+
+//---------------------------------------------------------------------------
+
+GLDEF_C void CallTestsL()
+//
+// Test the file server.
+//
+    {
+
+    //-- set up console output 
+    F32_Test_Utils::SetConsole(test.Console()); 
+    
+    TInt nRes=TheFs.CharToDrive(gDriveToTest, gDrive);
+    test(nRes==KErrNone);
+    
+    PrintDrvInfo(TheFs, gDrive);
+
+
+	TVolumeInfo v;
+	TInt r=TheFs.Volume(v, CurrentDrive());
+	test(r==KErrNone);
+	LargeFileSize=Max((TUint32)I64LOW(v.iFree >> 7), (TUint32)65536u);
+
+    if (gFirstTime)
+		{
+		MountRemoteFilesystem();
+		CreateSubstDrive();
+		
+		testDriveList();
+		
+		DisMountRemoteFilesystem();
+		RemoveSubstDrive();
+		
+		testDriveInfo();
+		testVolumeInfo();
+		testClientParse();
+		testPath();
+		testServerParse();
+		gFirstTime=EFalse;
+		}
+
+	CreateTestDirectory(_L("\\F32-TST\\TFSRV\\"));
+	testSubst();
+	testSetVolume();
+	CopyFileToTestDirectory();
+	testModified();
+	testName();
+	MakeAndDeleteFiles();
+	FillUpDisk();
+	DeleteTestDirectory();
+
+    TestSetErrorCondition();
+
+    }
+