kerneltest/f32test/server/t_rcache.cpp
changeset 9 96e5fb8b040d
child 42 a179b74831c9
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/kerneltest/f32test/server/t_rcache.cpp	Thu Dec 17 09:24:54 2009 +0200
@@ -0,0 +1,1813 @@
+// Copyright (c) 2006-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_rcache.cpp
+// 
+//
+
+/**
+ @file
+ @internalTechnology 
+*/
+#define __E32TEST_EXTENSION__
+#include <f32file.h>
+#include <e32test.h>
+#include <e32svr.h>
+#include <e32const.h>
+#include <e32const_private.h>
+#include <f32dbg.h>
+#include "t_server.h"
+#include <e32twin.h>
+#include <e32rom.h>
+#include <hal.h>
+#include <u32hal.h>
+
+const TInt KTotalCacheSize = 32 * 1024 * 1024;
+const TInt KDefaultCacheSize = (128 + 12) * 1024;
+const TInt KFilesNeededToFillCache = (KTotalCacheSize / KDefaultCacheSize) + 2;
+
+
+//----------------------------------------------------------------------------------------------
+//! @SYMTestCaseID      PBASE-T_RCACHE-0190
+//! @SYMTestType        CIT
+//! @SYMPREQ            PREQ914
+//! @SYMTestCaseDesc    This test case is exercising the Read Caching functionality added to
+//!						the File Server. There are negative and positive tests.
+//! @SYMTestActions     0   setup the environment to execute the tests
+//!						1	TestNegative creates situations where the cached content needs to be
+//!							flushed or removed  from the cache by corrupting files and verifies
+//!							the cache behaviour
+//!						2 	TestSimpleRead ensures that the cache is working in the simple cases,
+//!							with a combination of sync and async reads:
+//!								a. The file fits in the cache
+//!								b. The file doesn't fit in the cache
+//!						3 	TestRepeatedRead verifies the cache behaviour when a file is read an
+//!							arbitrary number of times, with and without other operations ongoing
+//!						4	TestReadAhead reads 3 times from a file and verifies that the read
+//!							ahead functionality acts afterwards.
+//!						5	TestConcurrent reads files concurrently and verifies how the cache
+//!							copes with it.
+//!						6 	TestFillCache fills the cache and then executes TestSimpleRead.
+//!
+//! @SYMTestExpectedResults finishes if the read cache behaves as expected, panics otherwise
+//! @SYMTestPriority        High
+//! @SYMTestStatus          Implemented
+//----------------------------------------------------------------------------------------------
+
+
+////////////////////////////////////////////////////////////
+// Template functions encapsulating ControlIo magic
+//
+GLDEF_D template <class C>
+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_D RTest test(_L("T_RCACHE"));
+
+GLDEF_D	RFs TheFs;
+GLDEF_D TInt gDrive;
+GLDEF_D TFileName gSessionPath;
+GLDEF_D TChar gDriveToTest;
+TThreadId gMainThreadId;
+
+LOCAL_D HBufC8* gBuf = NULL;
+LOCAL_D TPtr8 gBufReadPtr(NULL, 0);
+LOCAL_D HBufC8* gBufSec = NULL;
+LOCAL_D TPtr8 gBufWritePtr(NULL, 0);
+
+LOCAL_D const TInt KOneK = 1024;
+LOCAL_D const TInt KOneMeg = KOneK * 1024;
+LOCAL_D const TInt KBlockSize = KOneK;
+LOCAL_D const TInt KWaitRequestsTableSize = 256;
+
+LOCAL_D TInt gSecondFileSize = 0;
+LOCAL_D TInt gFirstFileSize = 0;
+LOCAL_D TInt gCurrentFileSize = 0;
+
+LOCAL_D TInt64 gMediaSize = 0;
+
+LOCAL_D TTimeIntervalMicroSeconds gTimeTakenBigFile(0);
+LOCAL_D TBuf16<25> gFirstFile;
+LOCAL_D TBuf16<25> gSecondFile;
+LOCAL_D TBuf16<25> gCurrentFile;
+
+LOCAL_D TInt gNextFile = 0;
+LOCAL_D TTime gTime1;
+LOCAL_D TTime gTime2;
+_LIT(KMsg1, "1st read timing: %d\n");
+_LIT(KMsg2, "2nd read timing: %d\n");
+_LIT(KMsg3, "3rd read timing: %d\n");
+
+LOCAL_D RSemaphore gSync;
+
+// Concurrent Threads
+LOCAL_D RThread gThread1;
+LOCAL_D RThread gThread2;
+LOCAL_D RSemaphore client;
+LOCAL_D const TInt KHeapSize = 0x4000;
+LOCAL_D const TInt KMaxHeapSize = 0x100000;
+LOCAL_D TBool gPagedRom = EFalse;
+
+enum TTestState
+	{
+	EThreadWait,
+	EThreadSignal,
+	ENoThreads
+	};
+
+/** Formats the drive
+
+	@param aDrive 	Drive to be formatted
+	@param aFormatMode Mode for the format operation
+*/
+LOCAL_C void Formatting(TInt aDrive, TUint aFormatMode )
+	{
+
+	test.Next(_L("Format"));
+	TBuf<4> driveBuf = _L("?:\\");
+	driveBuf[0]=(TText)(aDrive+'A');
+	RFormat format;
+	TInt count;
+	TInt r = format.Open(TheFs,driveBuf,aFormatMode,count);
+	test_KErrNone(r);
+	while(count)
+		{
+		TInt r = format.Next(count);
+		test_KErrNone(r);
+		}
+	format.Close();
+	}
+
+/** Verifies the content of a buffer (all the letters are like the first one)
+
+	@param aBuffer  Buffer to be verified
+
+	@return KErrNone if all the letters are the same, KErrCorrupt otherwise
+*/
+LOCAL_C TInt VerifyBuffer(TDes8& aBuffer)
+	{
+	TChar c = aBuffer[0];
+
+	for(TInt i = 1; i < aBuffer.Length(); i++)
+		{
+		if(i%32 != 0)
+		{
+			if(c != (TChar)(aBuffer[i] - 1))
+				return KErrCorrupt;
+		}
+		else
+		{
+			if(aBuffer[i] != aBuffer[0])
+				return KErrCorrupt;
+		}
+		c = aBuffer[i];
+		}
+
+	return KErrNone;
+	}
+
+/**  Fills a buffer with character aC, aC+1, aC+2, ..., aC+20, aC, etc
+
+	@param aBuffer  Buffer to be filled, output
+	@param aLength  Length to be filled
+	@param aC		Character to be used to fill the buffer
+*/
+LOCAL_C void FillBuffer(TDes8& aBuffer, TInt aLength, TChar aC)
+	{
+	test (aBuffer.MaxLength() >= aLength);
+	for(TInt i = 0; i < aLength; i++)
+		{
+		aBuffer.Append((i%32) + aC);
+		}
+	}
+
+/**  Returns true if fat filesystem present on aDrive
+
+	@param aFsSession 	Session on the File Server
+	@param aDrive 		Drive to be looked at
+*/
+LOCAL_C TBool IsFSFAT(RFs &aFsSession,TInt aDrive)
+	{
+	TFileName f;
+	TInt r = aFsSession.FileSystemName(f,aDrive);
+
+	if (r != KErrNone)
+		{
+		test.Printf(_L("Unable to get file system name\n"));
+		return EFalse;
+		}
+
+	return (f.CompareF(_L("Fat")) == 0);
+	}
+
+/** Generates a file name of the form FFFFF*<aPos>.TXT (aLong.3)
+
+	@param aBuffer The filename will be returned here
+	@param aLong   Defines the longitude of the file name
+	@param aPos	   Defines the number that will be attached to the filename
+*/
+GLDEF_C void FileNameGen(TDes16& aBuffer, TInt aLong, TInt aPos)
+{
+	TInt padding;
+	TInt i = 0;
+	TBuf16<10> tempbuf;
+
+	_LIT(KNumber,"%d");
+	tempbuf.Format(KNumber,aPos);
+	padding = aLong-tempbuf.Size()/2;
+	aBuffer = _L("");
+	while(i < padding)
+	{
+		aBuffer.Append('F');
+		i++;
+	}
+	aBuffer.Append(tempbuf);
+
+	_LIT(KExtension1, ".TXT");
+	aBuffer.Append(KExtension1);
+}
+
+/**  Delete content of directory
+
+	@param aDir	Target directory
+
+	@return Error returned if any, otherwise KErrNone
+*/
+LOCAL_C TInt DeleteAll(TDes16& aDir)
+{
+	TBuf16<100> dir;
+	CFileMan* fMan = CFileMan::NewL(TheFs);
+	TInt r=0;
+
+	dir = aDir;
+	dir.Append(_L("F*.*"));
+	r = fMan->Delete(dir);
+
+	delete fMan;
+	return r;
+}
+
+/**  Waits for all the TRequestStatus in status[] to complete
+
+	@param status 	Array of TRequestStatus
+	@param aLength  Length to be filled
+	@param aC		Character to be used to fill the buffer
+*/
+LOCAL_C void WaitForAll(TRequestStatus* status, TInt aSize)
+{
+	TInt i = 0;
+
+	RTest test(_L("T_RCACHE"));
+
+	while(i < aSize)
+		{
+		User::WaitForRequest(status[i]);
+		test(status[i] == KErrNone);
+		i++;
+		}
+
+	test.Close();
+}
+
+/**  Reads the parameters from the comand line
+	 and updates the appropriate variables
+*/
+LOCAL_C void parseCommandLine()
+{
+	TBuf<0x100> cmd;
+	User::CommandLine(cmd);
+	TLex lex(cmd);
+	TPtrC token = lex.NextToken();
+	TInt r=0;
+
+	if(token.Length() != 0)
+		{
+		gDriveToTest = token[0];
+		gDriveToTest.UpperCase();
+		}
+	else
+		{
+		gDriveToTest = 'C';
+		}
+
+	r = TheFs.CharToDrive(gDriveToTest,gDrive);
+	test_KErrNone(r);
+	gSessionPath = _L("?:\\F32-TST\\");
+	gSessionPath[0] = (TUint16) gDriveToTest;
+	test.Printf(_L("\nCLP=%C\n"),(TInt)gDriveToTest);
+}
+
+/**  Writes a file synchronously in blocks of aBlockSize size
+
+	@param fs			RFs object
+	@param aFile		File name
+	@param aSize		Size of the file in bytes
+	@param aBlockSize	Size of the blocks to be used in bytes
+	@param aState		Determines if the operation is being done in the main process
+						or in a thread
+	@param aReadBack	Reads the file back if ETrue. This is used for Read-ahead testing to ensure
+						what is written is also read back through the media driver's cache. If the
+						file is bigger than the cache, then subsequent streaming reads starting from
+						position zero will have to be re-read from the media.
+
+	@return Returns KErrNone if everything ok, otherwise it panics
+*/
+LOCAL_C TInt WriteFile(RFs& fs, TDes16& aFile, TInt aSize, TInt aBlockSize, TDes8& aBuf, TTestState aState, TBool aReadBack = EFalse)
+{
+	RTest test(_L("T_RCACHE"));
+
+	TInt r = 0;
+	RFile fileWrite;
+	TInt pos = 0;
+
+	test(aBlockSize>0);
+	test((aSize%aBlockSize) == 0); 	// Ensure the size of the file is a multiple of the block size
+
+	if(aState == EThreadWait)
+	{
+		gSync.Wait();
+	}
+
+	// delete file first to ensure it's contents are not in the cache (file may be be on the closed file queue)
+	r = fs.Delete(aFile);
+	test_Value(r, r == KErrNone || r == KErrNotFound);
+
+	r = fileWrite.Replace(fs,aFile,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+	test_KErrNone(r);
+
+	TInt j = 0;
+	while(j < aSize)
+		{
+			r = fileWrite.Write(pos, aBuf);
+			test_KErrNone(r);
+
+			if (aReadBack)
+				{
+				r = fileWrite.Read(pos, aBuf);
+				test_KErrNone(r);
+				}
+
+			if((j>(aSize/2))&&(aState == EThreadSignal))
+			{
+				gSync.Signal();
+				aState = ENoThreads;
+			}
+			j += aBlockSize;
+			pos += aBlockSize;
+		}
+	fileWrite.Close();
+	test.Close();
+
+	return KErrNone;
+}
+
+/**  Read File in blocks of size aBlockSize
+
+	@param fs			RFs object
+	@param aFile		File name
+	@param aBlockSize	Size of the blocks to be used in bytes
+
+	@return Returns KErrNone if everything ok, otherwise it panics
+*/
+LOCAL_C TInt ReadFile(RFs& fs, TDes16& aFile, TInt aBlockSize)
+{
+	RTest test(_L("T_RCACHE"));
+
+	TInt r = 0, size = 0;
+	RFile fileRead;
+
+	test(aBlockSize>0);				// Block size must be greater than 0
+
+	r = fileRead.Open(fs,aFile,EFileShareAny|EFileRead|EFileReadBuffered|EFileReadAheadOff);
+	test_KErrNone(r);
+
+	r = fileRead.Size(size);
+	test_KErrNone(r);
+
+	TInt j = 0;
+	while(j < size)
+	{
+		r = fileRead.Read(gBufReadPtr, aBlockSize);
+		test_KErrNone(r);
+		j += aBlockSize;
+	}
+	fileRead.Close();
+	test.Close();
+
+	return KErrNone;
+}
+
+/** Write a file asynchronously in blocks of aBlockSize size
+
+	@param fs			RFs object
+	@param aFileWrite	RFile object, needs to exist beyond the scope of this function
+	@param aFile		File name
+	@param aSize		Size of the file in bytes
+	@param aBlockSize	Size of the blocks to be used in bytes
+	@param aStatus		TRequestStatus array for all the requests
+*/
+LOCAL_C void WriteFileAsync(RFs& fs, RFile& aFileWrite, TDes16& aFile, TInt aSize, TInt aBlockSize, TRequestStatus aStatus[])
+{
+	RTest test(_L("T_RCACHE"));
+
+	TInt r = 0;
+
+	test(aBlockSize>0);
+	test((aSize%aBlockSize) == 0); 	// Ensure the size of the file is a multiple of the block size
+
+
+	// delete file first to ensure it's contents are not in the cache (file may be be on the closed file queue)
+	r = fs.Delete(aFile);
+	test(r == KErrNone || r == KErrNotFound);
+
+	r = aFileWrite.Replace(fs,aFile,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+	test_KErrNone(r);
+
+	TInt j = 0, i = 0;
+	while(j < aSize)
+	{
+		aFileWrite.Write(gBufWritePtr,aStatus[i]);
+		r = aStatus[i].Int();
+		if (r != KErrNone && r != KRequestPending)
+			{
+			test.Printf(_L("Write %d returned %d\n"), i, r);
+			test(0);
+			}
+		i++;
+
+		j += aBlockSize;
+	}
+	test.Close();
+}
+
+/**  Read a file asynchronously in blocks of aBlockSize size
+
+	@param fs			RFs object
+	@param aFileRead	RFile object, needs to exist beyond the scope of this function
+	@param aFile		File name
+	@param aSize		Size of the file in bytes
+	@param aBlockSize	Size of the blocks to be used in bytes
+	@param aStatus		TRequestStatus array for all the requests
+
+	@return KErrNone
+*/
+LOCAL_C TInt ReadFileAsync(RFs& fs,RFile& aFileRead, TDes16& aFile, TInt aBlockSize,TRequestStatus aStatus[], TInt aFileSize)
+	{
+	RTest test(_L("T_RCACHE"));
+
+	TInt r = 0;
+	TInt size = 0;
+
+	test(aBlockSize > 0);
+
+	r = aFileRead.Open(fs,aFile,EFileShareAny|EFileRead|EFileReadBuffered|EFileReadAheadOff);
+	test_KErrNone(r);
+
+	r = aFileRead.Size(size);
+	test_KErrNone(r);
+
+	test(size == aFileSize);
+
+	TInt j = 0, i = 0;
+	while(j < size)
+		{
+		aFileRead.Read(gBufReadPtr,aStatus[i]);
+		r = aStatus[i].Int();
+		if (r != KErrNone && r != KRequestPending)
+			{
+			test.Printf(_L("Read %d returned %d\n"), i, r);
+			test(0);
+			}
+		i++;
+
+		j += aBlockSize;
+		}
+
+	test.Close();
+	return KErrNone;
+	}
+
+/** Measure the time taken for this file to be written synchronously
+*/
+LOCAL_C TInt WriteTestFile(TDes16& aFile, TInt aSize, TBool aReadBack = EFalse)
+{
+	RTest test(_L("T_RCACHE"));
+
+	TTime startTime;
+	TTime endTime;
+	TInt r = 0;
+
+	startTime.HomeTime();
+
+	r = WriteFile(TheFs,aFile, aSize, KBlockSize, gBufWritePtr, ENoThreads, aReadBack);
+	test_KErrNone(r);
+
+	endTime.HomeTime();
+
+	gTimeTakenBigFile = I64LOW(endTime.MicroSecondsFrom(startTime).Int64());
+
+	test.Close();
+	return I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+}
+
+/** Measure the time taken for this file to be read synchronously
+*/
+LOCAL_C TInt ReadTestFile(TDes16& aFile)
+{
+	TTime startTime;
+	TTime endTime;
+
+	startTime.HomeTime();
+	ReadFile(TheFs,aFile, KBlockSize);
+	endTime.HomeTime();
+
+	gTimeTakenBigFile = I64LOW(endTime.MicroSecondsFrom(startTime).Int64());
+
+	return I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+}
+
+/** Read asynchronously the test file from the disc
+
+*/
+LOCAL_C TInt ReadAsyncTestFile(TDes16& aFile, TInt aSize)
+{
+	RTest test(_L("T_RCACHE"));
+
+	TTime startTime;
+	TTime endTime;
+	TRequestStatus status[KWaitRequestsTableSize];
+	RFs fs;
+	RFile file;
+
+	TInt r = fs.Connect();
+	test (r == KErrNone);
+
+	startTime.HomeTime();
+
+	ReadFileAsync(fs, file, aFile, KBlockSize, status, aSize);
+	WaitForAll(status, aSize/KBlockSize);
+
+	endTime.HomeTime();
+
+	gTimeTakenBigFile = I64LOW(endTime.MicroSecondsFrom(startTime).Int64());
+
+	file.Close();
+	fs.Close();
+	test.Close();
+	return I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+}
+
+/** Write a file for the simple case
+
+*/
+LOCAL_C TInt WriteFileT(TAny* )
+{
+	RTest test(_L("T_RCACHE"));
+	RFs fs;
+	TInt r = fs.Connect();
+	test_KErrNone(r);
+
+	r = fs.SetSessionPath(gSessionPath);
+	test_KErrNone(r);
+	gTime1.HomeTime();
+
+	r = WriteFile(fs,gCurrentFile,gCurrentFileSize, KBlockSize,gBufWritePtr, ENoThreads);
+	test_KErrNone(r);
+
+	gTime2.HomeTime();
+
+	fs.Close();
+	test.Close();
+
+	gTimeTakenBigFile = I64LOW(gTime2.MicroSecondsFrom(gTime1).Int64());
+
+	client.Signal();
+
+	return ETrue;
+}
+
+/** Write a file for the concurrent case
+
+*/
+LOCAL_C TInt WriteFileT2(TAny* )
+{
+	RTest test(_L("T_RCACHE"));
+	RFs fs;
+	TInt r = fs.Connect();
+	test_KErrNone(r);
+	RFile file;
+	TRequestStatus status[KWaitRequestsTableSize];
+
+
+	r = fs.SetSessionPath(gSessionPath);
+	test_KErrNone(r);
+
+	WriteFileAsync(fs,file,gFirstFile,gSecondFileSize, KBlockSize, status);
+	WaitForAll(status,gSecondFileSize/KBlockSize);
+
+	TInt size = 0;
+	file.Size(size);
+	test( size == gSecondFileSize );
+
+	file.Close();
+	fs.Close();
+
+	test.Close();
+
+	client.Signal();
+
+	return ETrue;
+}
+
+/**  Write a file for the simple case
+
+*/
+LOCAL_C TInt ReadFileT(TAny* )
+{
+	RTest test(_L("T_RCACHE"));
+	RFs fs;
+	TInt r = fs.Connect();
+	test_KErrNone(r);
+
+	r = fs.SetSessionPath(gSessionPath);
+	test_KErrNone(r);
+	gTime1.HomeTime();
+
+	ReadFile(fs,gCurrentFile, KBlockSize);
+
+	gTime2.HomeTime();
+
+	fs.Close();
+	test.Close();
+
+	gTimeTakenBigFile = I64LOW(gTime2.MicroSecondsFrom(gTime1).Int64());
+
+	client.Signal();
+
+	return ETrue;
+}
+
+/**  Simple case, cache effect shown
+
+*/
+LOCAL_C void TestSimpleRead()
+{
+	TInt r = 0;
+	TInt time = 0;
+	TInt time2 = 0;
+	TInt time3 = 0;
+	TInt tcreate = 0;
+
+	test.Start(_L(""));
+
+	test.Next(_L("File fits in: read sync + read sync + read async\n"));
+
+	tcreate = WriteTestFile(gSecondFile, gSecondFileSize);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gSecondFile);
+	test.Printf(KMsg2,time2);
+	time3 = ReadAsyncTestFile(gSecondFile,gSecondFileSize);
+	test.Printf(KMsg3,time3);
+#if !defined(__WINS__)
+	if (gPagedRom)
+		test.Printf(_L("Skipping timing test on paged ROM\n"));
+	else
+		test((time2 <= time) && (time3 < time));
+#endif
+
+	r = DeleteAll(gSessionPath);
+	test(r == KErrNone || r == KErrInUse);
+
+	// Simple case filling/reading the cache from different threads
+	test.Next(_L("File fits in: read sync (another thread) + read sync + read async\n"));
+	gCurrentFile = gSecondFile;
+	gCurrentFileSize = gSecondFileSize;
+
+	TBuf<20> buf = _L("Write File");
+	r = gThread1.Create(buf,WriteFileT,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test_KErrNone(r);
+
+	gThread1.Resume();
+	client.Wait();
+
+	tcreate = I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+
+	test.Printf(_L("Time to create the file from a thread: %d ms\n"),tcreate);
+
+	buf = _L("Read File");
+	r = gThread2.Create(buf,ReadFileT,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test(r == KErrNone);
+
+	gThread2.Resume();
+	client.Wait();
+
+	gThread1.Close();
+	gThread2.Close();
+
+	time = I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gSecondFile);
+	test.Printf(KMsg2,time2);
+	time3 = ReadAsyncTestFile(gSecondFile,gSecondFileSize);
+	test.Printf(KMsg3,time3);
+#if !defined(__WINS__)
+	if (gPagedRom)
+		test.Printf(_L("Skipping timing test on paged ROM\n"));
+	else
+		test((time2<=time) && (time3<time));
+#endif
+
+	r = DeleteAll(gSessionPath);
+	test(r == KErrNone || r == KErrInUse);
+
+
+	test.Next(_L("File doesn't fit in: read sync + read sync + read async\n"));
+
+	tcreate = WriteTestFile(gFirstFile,gFirstFileSize);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+
+	time = ReadTestFile(gFirstFile);
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gFirstFile);
+	test.Printf(KMsg2,time2);
+	time3 = ReadAsyncTestFile(gFirstFile,gFirstFileSize);
+	test.Printf(KMsg3,time3);
+
+	#if !defined(__WINS__)
+	// this isn't valid as the file doesn't fit in the cache, so there's no reason why
+	// the second read should be any faster than the first
+	// test((time2 <= time) && (time3 < time));
+	#endif
+
+
+	r = DeleteAll(gSessionPath);
+	test(r == KErrNone || r == KErrInUse);
+
+
+	test.Next(_L("File doesn't fit in: read sync (another thread) + read sync + read async\n"));
+	gCurrentFile = gFirstFile;
+	gCurrentFileSize = gFirstFileSize;
+
+	buf = _L("Write Big File");
+	r = gThread1.Create(buf,WriteFileT,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test_KErrNone(r);
+
+	gThread1.Resume();
+	client.Wait();
+
+	tcreate = I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+
+	test.Printf(_L("Time to create the file from a thread: %d ms\n"),tcreate);
+
+	buf = _L("Read Big File");
+	r = gThread2.Create(buf,ReadFileT,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test(r == KErrNone);
+
+	gThread2.Resume();
+	client.Wait();
+
+	gThread1.Close();
+	gThread2.Close();
+
+	time = I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gFirstFile);
+	test.Printf(KMsg2,time2);
+	time3 = ReadAsyncTestFile(gFirstFile,gFirstFileSize);
+	test.Printf(KMsg3,time3);
+
+	#if !defined(__WINS__)
+	// this isn't valid as the file doesn't fit in the cache, so there's no reason why
+	// the second read should be any faster than the first
+	// test((time2 <= time) && (time3 < time));
+	#endif
+
+	r = DeleteAll(gSessionPath);
+	test(r == KErrNone || r == KErrInUse);
+
+	test.End();
+}
+
+/**  Thread to create file and read from it
+
+*/
+LOCAL_C TInt ReadAnotherEntry(TAny* )
+{
+	RTest test(_L("T_RCACHE"));
+	RFs fs;
+	TInt r = fs.Connect();
+	RFile file;
+	HBufC8* lBufSec = NULL;
+	TPtr8 lBufReadPtr(NULL, 0);
+
+	TRAPD(res2,lBufSec = HBufC8::NewL(KBlockSize+1));
+	test(res2 == KErrNone && lBufSec != NULL);
+	lBufReadPtr.Set(lBufSec->Des());
+
+	test(r == KErrNone);
+	r = fs.SetSessionPath(gSessionPath);
+
+
+	// delete file first to ensure it's contents are not in the cache (file may be be on the closed file queue)
+	r = fs.Delete(gFirstFile);
+	test(r == KErrNone || r == KErrNotFound);
+
+	r = file.Create(fs,gFirstFile,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+
+	if(r == KErrAlreadyExists)
+	{
+		r = file.Open(fs,gFirstFile,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+		test_KErrNone(r);
+	}
+
+	r = file.Write(gBufWritePtr);
+	test_KErrNone(r);
+
+	client.Signal();
+
+	FOREVER
+		{
+			r = file.Read(gBufReadPtr, KBlockSize);
+			test_KErrNone(r);
+		}
+
+}
+
+/**  Test the cache behaviour in repeated call situations
+
+*/
+LOCAL_C void TestRepeatedRead()
+{
+	TInt r = 0;
+	TInt time = 0;
+	TInt i = 0;
+	TInt tcreate = 0;
+
+
+	test.Start(_L(""));
+	test.Next(_L("File fits in: read sync / read async \n"));
+
+	tcreate = WriteTestFile(gSecondFile, gSecondFileSize);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+	while(i < 20)
+	{
+		if(!(i % 2))
+		{
+			time = ReadTestFile(gSecondFile);
+			test.Printf(_L("%d)  Sync Read: %d\n"), i+1 , time);
+		}
+		else
+		{
+			time = ReadAsyncTestFile(gSecondFile,gSecondFileSize);
+			test.Printf(_L("%d) Async Read: %d\n"), i+1 , time);
+		}
+		i++;
+	}
+
+	r = DeleteAll(gSessionPath);
+	test_KErrNone(r);
+
+	test.Next(_L("File fits in: read sync / read async, with another thread using the drive \n"));
+
+	TBuf<20> buf = _L("Noise Thread");
+	r = gThread1.Create(buf,ReadAnotherEntry,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test_KErrNone(r);
+
+	tcreate = WriteTestFile(gSecondFile, gSecondFileSize);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+
+	gThread1.Resume();
+	client.Wait();
+
+	i = 0;
+	while(i < 20)
+	{
+		if(!(i%2))
+		{
+			time = ReadTestFile(gSecondFile);
+			test.Printf(_L("%d)  Sync Read: %d\n"), i+1 , time);
+		}
+		else
+		{
+			time = ReadAsyncTestFile(gSecondFile,gSecondFileSize);
+			test.Printf(_L("%d) Async Read: %d\n"), i+1 , time);
+		}
+		i++;
+	}
+
+	gThread1.Kill(KErrNone);
+	gThread1.Close();
+
+	r = DeleteAll(gSessionPath);
+	test_KErrNone(r);
+
+	test.End();
+}
+
+/**  Concurrent operations testing
+
+*/
+LOCAL_C void TestConcurrent()
+{
+	TInt r = 0;
+	TInt time = 0;
+	TInt time2 = 0;
+	TInt time3 = 0;
+
+	test.Start(_L("Write two files concurrently\n"));
+
+	gCurrentFile = gSecondFile;
+	gCurrentFileSize = gSecondFileSize;
+
+	TBuf<20> buf = _L("Write Two Files 1");
+	r = gThread1.Create(buf,WriteFileT,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test_KErrNone(r);
+
+	TBuf<20> buf2 = _L("Write Two Files 2");
+	r = gThread2.Create(buf2,WriteFileT2,KDefaultStackSize*2,KHeapSize,KMaxHeapSize,NULL);
+	test(r == KErrNone);
+
+	gThread1.Resume();
+	gThread2.Resume();
+	client.Wait();
+	client.Wait();
+
+	test.Next(_L("Read the two files repeatedly\n"));
+
+	test.Printf(_L("File 1:\n"));
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gSecondFile);
+	test.Printf(KMsg2,time2);
+	time3 = ReadAsyncTestFile(gSecondFile,gSecondFileSize);
+	test.Printf(KMsg3,time3);
+
+	test.Printf(_L("File 2:\n"));
+
+	time = ReadTestFile(gFirstFile);
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gFirstFile);
+	test.Printf(KMsg2,time2);
+	time3 = ReadAsyncTestFile(gFirstFile,gSecondFileSize);
+	test.Printf(KMsg3,time3);
+
+	test.End();
+
+	r = DeleteAll(gSessionPath);
+	test_KErrNone(r);
+}
+
+/**  Create file from other thread, to be cached
+
+*/
+LOCAL_C TInt CreateFile(TAny* )
+{
+	RTest test(_L("T_RCACHE"));
+	RFs fs;
+	TInt r = fs.Connect();
+	test(r == KErrNone);
+
+	r = fs.SetSessionPath(gSessionPath);
+	test(r == KErrNone);
+
+	r = WriteFile(fs, gSecondFile, gSecondFileSize, KBlockSize, gBufWritePtr, EThreadSignal);
+	test_KErrNone(r);
+
+	return KErrNone;
+}
+
+LOCAL_C TBool FindPattern(TUint8 *aBuf, TUint8 *aPattern, TInt aLong, TInt *aOffSet)
+{
+	TInt i = 0;
+	TBool found = EFalse;
+
+	while((i < (aLong-4)) && !found)
+	{
+		found = (
+			(aBuf[i] == aPattern[0])&&
+			(aBuf[i+1] == aPattern[1])&&
+			(aBuf[i+2] == aPattern[2])&&
+			(aBuf[i+3] == aPattern[3])
+			);
+	i++;
+	}
+
+	if(found)
+		*aOffSet = --i;
+
+	return found;
+}
+
+/**  Corrupts the second file with Raw access
+
+*/
+LOCAL_C void CorruptSecondFileRaw()
+{
+	RRawDisk rDisk;
+	TUint8 gBuffer[4] =
+	{
+		65,66,67,68
+	};
+	TUint8 gBufferB[4] =
+	{
+		33,33,33,33
+	};
+	TPtr8 gBufferBPtr(&gBufferB[0], 4, 4);
+
+	TUint8 gBuffer2[KBlockSize];
+	TPtr8 gBuffer2Ptr(&gBuffer2[0], KBlockSize);
+
+	TInt r = rDisk.Open(TheFs,gDrive);
+	test_KErrNone(r);
+
+	TInt64 pos = 0;
+	TBool found = EFalse;
+	TInt offset = 0;
+
+	while(!found)
+	{
+		rDisk.Read(pos, gBuffer2Ptr);
+		found = FindPattern(gBuffer2, gBuffer, KBlockSize, &offset);
+		pos += (KBlockSize);
+	}
+	pos -= (KBlockSize+1);
+	pos = pos+offset;
+
+	r = rDisk.Write(pos+4, gBufferBPtr);
+	test_KErrNone(r);
+
+	rDisk.Close();
+}
+
+/**  Modifies the second file
+
+*/
+LOCAL_C TInt CorruptSecondFile()
+{
+	TInt r = 0;
+	RFile fileWrite;
+	HBufC8* dummy = NULL;
+	TPtr8 dummyPtr(NULL, 0);
+
+	TRAPD(res,dummy = HBufC8::NewL(4));
+	test(res == KErrNone && dummy != NULL);
+
+	dummyPtr.Set(dummy->Des());
+	FillBuffer(dummyPtr, 4, '1');
+
+	r = fileWrite.Open(TheFs,gSecondFile,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+	if(r != 0)
+		return r;
+	TInt pos = 30;
+	r = fileWrite.Seek(ESeekStart,pos);
+
+	r = fileWrite.Write(dummyPtr);
+	if(r != 0)
+		return r;
+
+	fileWrite.Close();
+
+	delete dummy;
+
+	return KErrNone;
+}
+
+//
+// Read the file verifying content
+//
+LOCAL_C TInt ReadTestFileVerif(TDes16& aFile, TBool aRaw)
+{
+	TTime startTime;
+	TTime endTime;
+	TInt r = 0, size = 0;
+	RFile fileRead;
+	TInt corrupt = 0;
+	TBool isFat=IsFSFAT(TheFs,gDrive);
+
+	startTime.HomeTime();
+
+	r = fileRead.Open(TheFs,aFile,EFileShareAny|EFileRead|EFileReadBuffered|EFileReadAheadOff);
+	test_KErrNone(r);
+
+	r = fileRead.Size(size);
+	test_KErrNone(r);
+
+	TInt j = 0;
+	while(j < size)
+		{
+			r = fileRead.Read(gBufReadPtr, KBlockSize);
+			if(aRaw)
+			{
+				if(isFat)
+				{
+					test_KErrNone(r);
+				}
+				else
+				{
+				if(r == KErrCorrupt)
+					corrupt++;
+				}
+			}
+			else
+			{
+				test_KErrNone(r);
+			}
+			j += KBlockSize;
+			r = VerifyBuffer(gBufReadPtr);
+			if(r == KErrCorrupt)
+				corrupt++;
+		}
+
+	fileRead.Close();
+
+	test(corrupt>0); // Ensure the cache returns the changed content
+
+	endTime.HomeTime();
+
+	gTimeTakenBigFile = I64LOW(endTime.MicroSecondsFrom(startTime).Int64());
+
+	return I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+}
+
+
+/** Negative testing
+
+*/
+LOCAL_C void TestNegative()
+{
+	TInt r = 0;
+	TInt time, time2, time3;
+	TInt tcreate = 0;
+
+	test.Start(_L(""));
+	// Kill a thread while writing, then read content
+	TBuf<20> buf = _L("A thread to kill");
+	gCurrentFile = gSecondFile;
+	gCurrentFileSize = gSecondFileSize;
+
+	r = gThread1.Create(buf,CreateFile,KDefaultStackSize,KHeapSize,KMaxHeapSize,NULL);
+	test_KErrNone(r);
+
+	gThread1.Resume();
+	gSync.Wait();
+	gThread1.Kill(KErrGeneral);
+	gThread1.Close();
+	test.Next(_L("Read after killing the write in the middle\n"));
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg1,time);
+	time2 = ReadTestFile(gSecondFile);
+	test.Printf(KMsg2,time2);
+
+	// Read async the content
+	TInt size = 0;
+	TRequestStatus status[KWaitRequestsTableSize];
+	TTime startTime;
+	TTime endTime;
+	RFile fileRead;
+
+	startTime.HomeTime();
+
+	r = fileRead.Open(TheFs,gSecondFile,EFileShareAny|EFileRead|EFileReadBuffered|EFileReadAheadOff);
+	test_KErrNone(r);
+
+	r = fileRead.Size(size);
+	test_KErrNone(r);
+
+	TInt j = 0, i = 0;
+	while(j < size)
+		{
+			fileRead.Read(gBufReadPtr,status[i++]);
+			test_KErrNone(r);
+			j += KBlockSize;
+		}
+
+	j = i;
+	i = 0;
+	while(i < j)
+	{
+		User::WaitForRequest(status[i++]);
+	}
+	fileRead.Close();
+	endTime.HomeTime();
+	gTimeTakenBigFile = I64LOW(endTime.MicroSecondsFrom(startTime).Int64());
+ 	time3 = I64LOW(gTimeTakenBigFile.Int64()) / 1000;
+
+	test.Printf(KMsg3,time3);
+
+	// Modify file in some position
+	test.Next(_L("Overwrite partially a file\n"));
+	Formatting(gDrive, EFullFormat);
+	r = TheFs.MkDirAll(gSessionPath);
+	if (r != KErrNone && r != KErrAlreadyExists)
+		{
+		test_KErrNone(r);
+		}
+
+	tcreate = WriteTestFile(gSecondFile, gSecondFileSize);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg1,time);
+
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg2,time);
+
+	CorruptSecondFile();
+
+	time = ReadTestFileVerif(gSecondFile,EFalse);
+	test.Printf(KMsg1,time);
+
+	time = ReadTestFileVerif(gSecondFile,EFalse);
+	test.Printf(KMsg2,time);
+
+	// Modify the file in disk, raw access
+	test.Next(_L("Overwrite file with raw access\n"));
+
+	Formatting(gDrive,EFullFormat);
+
+	r = TheFs.MkDirAll(gSessionPath);
+	if (r != KErrNone && r != KErrAlreadyExists)
+		{
+		test_KErrNone(r);
+		}
+
+
+	tcreate = WriteTestFile(gSecondFile, gSecondFileSize);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg1,time);
+
+	time = ReadTestFile(gSecondFile);
+	test.Printf(KMsg2,time);
+
+	CorruptSecondFileRaw();
+
+	time = ReadTestFileVerif(gSecondFile,ETrue);
+	test.Printf(KMsg1,time);
+
+	time = ReadTestFileVerif(gSecondFile,ETrue);
+	test.Printf(KMsg2,time);
+
+
+	r = DeleteAll(gSessionPath);
+	test_KErrNone(r);
+
+	test.End();
+}
+
+/**  Creates the files to fill the read cache
+
+	@param aFiles 	 Number of files needed to fill the cache
+	@param aFileSize The file size
+*/
+LOCAL_C void CreateFiles(TInt aFiles, TInt aFileSize)
+{
+	TInt i = 0, r = 0;
+	RFile file;
+	TBuf16<50> directory;
+
+	TBuf16<50> path;
+	TBuf16<50> buffer(50);
+
+	directory = gSessionPath;
+
+	test.Printf(_L("Creating %d files for filling the cache (size %d)\n"), aFiles, aFileSize);
+
+	// create a big buffer to speed things up
+	HBufC8* bigBuf = NULL;
+	const TInt KBigBifferSize = 32 * 1024;
+	TRAPD(res,bigBuf = HBufC8::NewL(KBigBifferSize));
+	test(res == KErrNone && bigBuf != NULL);
+
+	TPtr8 bigBufWritePtr(NULL, 0);
+	bigBufWritePtr.Set(bigBuf->Des());
+	FillBuffer(bigBufWritePtr, KBigBifferSize, 'A');
+
+
+	i = 0;
+	while(i < aFiles)
+	{
+		if (i % 10 == 0)
+			test.Printf(_L("Creating file %d of %d...\r"), i, aFiles);
+		FileNameGen(buffer, 8, i+3) ;
+		path = directory;
+		path.Append(buffer);
+
+		// delete file first to ensure it's contents are not in the cache (file may be on the closed file queue)
+		r = TheFs.Delete(path);
+		test(r == KErrNone || r == KErrNotFound);
+
+		r = file.Create(TheFs,path,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+		if(r == KErrAlreadyExists)
+			r = file.Open(TheFs,path,EFileShareAny|EFileWrite|EFileReadDirectIO|EFileWriteDirectIO);
+		TInt j = 0;
+		while(j < aFileSize)
+			{
+			bigBufWritePtr.SetLength(Min(KBigBifferSize, aFileSize - j));
+			r = file.Write(bigBufWritePtr);
+			test_KErrNone(r);
+			j += bigBufWritePtr.Length();
+			}
+
+		file.Close();
+		i++;
+	}
+	test.Printf(_L("\nFiles created\n"));
+	delete bigBuf;
+}
+
+/**  Fills the read cache
+
+	@param aFile	 Array of files needed to fill the cache
+	@param aFiles 	 Number of files needed to fill the cache
+	@param aFileSize The file size
+*/
+LOCAL_C void FillCache(RFile aFile[KFilesNeededToFillCache], TInt aFiles, TInt aFileSize)
+{
+	TInt i = 0, r = 0;
+	TBuf16<50> directory;
+
+	TBuf16<50> path;
+	TBuf16<50> buffer(50);
+	HBufC8* buf = NULL;
+	TPtr8 bufPtr(NULL, 0);
+
+	TRAPD(res,buf = HBufC8::NewL(2));
+	test(res == KErrNone && buf != NULL);
+	bufPtr.Set(buf->Des());
+
+	directory = gSessionPath;
+
+	i = 0;
+	while(i < aFiles)
+	{
+		FileNameGen(buffer, 8, i+3) ;
+		path = directory;
+		path.Append(buffer);
+		r = aFile[i].Open(TheFs,path,EFileShareAny|EFileRead|EFileReadBuffered|EFileReadAheadOff);
+		test_KErrNone(r);
+
+		TInt j = 0;
+		while(j < aFileSize)
+			{
+				r = aFile[i].Read(j,bufPtr);
+				test_KErrNone(r);
+				j += 4*KOneK;
+			}
+
+		i++;
+	}
+
+	delete buf;
+	test.Printf(_L("Cache filled\n"));
+}
+
+/** Fills the default cache
+
+*/
+LOCAL_C void TestFillCache()
+{
+	TInt nFiles = KFilesNeededToFillCache;
+	TInt fSize = KDefaultCacheSize;
+	RFile file[KFilesNeededToFillCache];
+
+	if(gMediaSize> ((fSize * nFiles)+gSecondFileSize+gFirstFileSize))
+	{
+		test.Start(_L("Creating files for filling the cache\n"));
+		CreateFiles(nFiles,fSize);
+#if defined(_DEBUG) || defined(_DEBUG_RELEASE)
+		// get number of items on Page Cache
+		TFileCacheStats startPageCacheStats;
+		TInt r = controlIo(TheFs,gDrive, KControlIoFileCacheStats, startPageCacheStats);
+		test(r==KErrNone || r == KErrNotSupported);
+		test.Printf(_L("Number of page cache lines on free list at beginning=%d\n"),startPageCacheStats.iFreeCount);
+		test.Printf(_L("Number of page cache lines on used list at beginning=%d\n"),startPageCacheStats.iUsedCount);
+		test.Printf(_L("Number of files on closed queue=%d\n"),startPageCacheStats.iFilesOnClosedQueue);
+#endif
+		FillCache(file,nFiles,fSize);
+
+#if defined(_DEBUG) || defined(_DEBUG_RELEASE)
+		// get number of items on Page Cache
+		r = controlIo(TheFs,gDrive, KControlIoFileCacheStats, startPageCacheStats);
+		test(r==KErrNone || r == KErrNotSupported);
+		test.Printf(_L("Number of page cache lines on free list at end=%d\n"),startPageCacheStats.iFreeCount);
+		test.Printf(_L("Number of page cache lines on used list at end=%d\n"),startPageCacheStats.iUsedCount);
+		test.Printf(_L("Number of files on closed queue=%d\n"),startPageCacheStats.iFilesOnClosedQueue);
+#endif
+		TestSimpleRead();
+
+		TInt i = 0;
+		while( i < KFilesNeededToFillCache )
+			{
+			file[i++].Close();
+			}
+
+		test.End();
+	}
+	else
+		test.Printf(_L("Skipping the fill of the cache due to lack of space in the current drive\n"));
+}
+
+/**	Overflow-safe tick deltas
+
+*/
+static TInt64 TicksToMsec(TUint32 aInitTicks, TUint32 aFinalTicks, TInt aFastCounterFreq)
+	{
+	TUint32 timeDelta;
+	if (aFinalTicks >= aInitTicks)
+		timeDelta = aFinalTicks - aInitTicks;
+	else
+		timeDelta = aFinalTicks + (KMaxTUint32 - aInitTicks);		// must've wrapped
+
+	return TInt64(timeDelta) * TInt64(1000000) / TInt64(aFastCounterFreq);
+	}
+
+/** Read three blocks and waits for the read ahead on the File Server to do its job
+
+*/
+LOCAL_C void TestReadAhead()
+{
+	TInt r = 0,tcreate;
+	RFile fileRead;
+	HBufC8* dummy = NULL;
+	TPtr8 dummyPtr(NULL, 0);
+
+	TUint32 initTicks = 0;
+	TUint32 finalTicks = 0;
+	TTimeIntervalMicroSeconds timeTakenReadFirst(0);
+	TTimeIntervalMicroSeconds timeTakenReadSubsequent(0);
+
+	// On NAND/FAT and NOR/LFFS drives, due to the lack of DMA support, the read-ahead is likely to happen
+	// BEFORE control is returned to this test app - for NAND this could be fixed by adding
+	// "FileCacheReadAsync OFF" to the estart.txt file, but we can't do this on the integrator as it has no
+	// estart.txt file. Also, we can't set "FileCacheReadAsync OFF" for LFFS as it kills the LFFS background
+	// processing (!)
+	// So... it's only really worth testing on MMC.
+	_LIT(KFATName,"FAT");
+	TDriveInfo driveInfo;
+	test(TheFs.Drive(driveInfo, gDrive) == KErrNone);
+	TFileName fileSystem;
+	r = TheFs.FileSystemName(fileSystem, gDrive);
+	fileSystem.UpperCase();
+	test((r==KErrNone)||(r==KErrNotFound));
+	// ONLY test on MMC
+	if ((driveInfo.iType != EMediaHardDisk) || (fileSystem.Compare(KFATName) != 0))
+		{
+		test.Printf(_L("Skipping read-ahead testing (drive is not MMC)...\n"));
+		return;
+		}
+
+	//--Find out if the drive is sync/async at this point and print information
+    TPckgBuf<TBool> drvSyncBuf;
+    r = TheFs.QueryVolumeInfoExt(gDrive, EIsDriveSync, drvSyncBuf);
+    test(r == KErrNone);
+	const TBool bDrvSync = drvSyncBuf();
+    if(bDrvSync)
+		test.Printf(_L("Drive D: is synchronous\n"));
+	else
+		test.Printf(_L("Drive D: is asynchronous\n"));
+
+	// use a fast counter as this is more accurate than using TTime
+	TInt fastCounterFreq;
+	r = HAL::Get(HAL::EFastCounterFrequency, fastCounterFreq);
+	test(r == KErrNone);
+	test.Printf(_L("HAL::EFastCounterFrequency %d\n"), fastCounterFreq);
+
+	// Bind this thread to CPU 0. This is so that timer deltas don't drift from
+	// scheduling - else, it causes spurious failures.
+	if (UserSvr::HalFunction(EHalGroupKernel, EKernelHalNumLogicalCpus, 0, 0) > 1)
+		(void)UserSvr::HalFunction(EHalGroupKernel, EKernelHalLockThreadToCpu, (TAny *)0, 0);
+
+	const TInt KReadLen = 28 * KOneK;
+
+	TRAPD(res,dummy = HBufC8::NewL(KReadLen));
+	test(res == KErrNone && dummy != NULL);
+
+	dummyPtr.Set(dummy->Des());
+
+	test.Start(_L("Creating test file..."));
+
+
+	tcreate = WriteTestFile(gFirstFile, gFirstFileSize, ETrue);
+	test.Printf(_L("Time to create the file: %d ms\n"),tcreate);
+
+	r = fileRead.Open(TheFs,gFirstFile,EFileShareAny|EFileRead|EFileReadBuffered|EFileReadAheadOn);
+	test_KErrNone(r);
+
+	// Read #1
+	test.Printf(_L("Issuing read #1...\n"));
+	initTicks = User::FastCounter();
+	r = fileRead.Read(dummyPtr);
+	finalTicks = User::FastCounter();
+	test_KErrNone(r);
+
+	timeTakenReadFirst = TicksToMsec(initTicks, finalTicks, fastCounterFreq);
+	test.Printf(_L("first read time %d \n"), I64LOW(timeTakenReadFirst.Int64()));
+
+	// Read #2
+	test.Printf(_L("Issuing read #2...\n"));
+	r = fileRead.Read(dummyPtr);
+
+	// Read #3
+	test.Printf(_L("Issuing read #3......resulting in read-ahead #1\n"));
+	r = fileRead.Read(dummyPtr);
+
+	// Wait for the read ahead #1 to be done - this should be approx the same size as previous read (KReadLen)
+	test.Printf(_L("Wait for read-ahead #1...\n"));
+	User::After(I64LOW(timeTakenReadFirst.Int64()) * 3 / 2);
+
+
+	test.Printf(_L("Issuing read #4...resulting in read-ahead #2\n"));
+	initTicks = User::FastCounter();
+	r = fileRead.Read(dummyPtr);
+	finalTicks = User::FastCounter();
+	test_KErrNone(r);
+	timeTakenReadSubsequent = TicksToMsec(initTicks, finalTicks, fastCounterFreq);
+
+	test.Printf(_L("read time:  %d \n"), I64LOW(timeTakenReadSubsequent.Int64()));
+
+#if !defined(__WINS__)
+	// NB the read-ahead on LFFS occurs "synchronously" i.e. it occurs before control is returned
+	// to the caller. However it's not a good idea to mark the drive as synchronous (FileCacheReadAsync OFF)
+	// as this causes the drive thread's priority to be lowered which kills the LFFS background processing (!)
+	if (gPagedRom)
+		test.Printf(_L("Skipping timing test on paged ROM\n"));
+	else
+		test(timeTakenReadSubsequent.Int64() < timeTakenReadFirst.Int64());
+#endif
+
+	// The read ahead #2 should now be in progress - this should be approx KReadLen * 2
+	// so this read will take result in the next read taking longer than normal (about double)
+	test.Printf(_L("Issuing read #5......resulting in read-ahead #3\n"));
+	initTicks = User::FastCounter();
+	r = fileRead.Read(dummyPtr);
+	finalTicks = User::FastCounter();
+	test_KErrNone(r);
+	timeTakenReadSubsequent = TicksToMsec(initTicks, finalTicks, fastCounterFreq);
+	test.Printf(_L("read time:  %d\n"), I64LOW(timeTakenReadSubsequent.Int64()));
+
+
+	// this read should take a long time, so don't test
+//#if !defined(__WINS__)
+//	test(gTimeTakenReadBlockFile.Int64() < gTimeTakenBigFile.Int64());
+//#endif
+
+	// The third read should be very quick as the previous read-ahead should have already buffered the data
+	test.Printf(_L("Issuing read #6......resulting in read-ahead #4\n"));
+	initTicks = User::FastCounter();
+	r = fileRead.Read(dummyPtr);
+	finalTicks = User::FastCounter();
+	test_KErrNone(r);
+	timeTakenReadSubsequent = TicksToMsec(initTicks, finalTicks, fastCounterFreq);
+	test.Printf(_L("read time:  %d\n"), I64LOW(timeTakenReadSubsequent.Int64()));
+
+
+#if !defined(__WINS__)
+	if (gPagedRom)
+		test.Printf(_L("Skipping timing test on paged ROM\n"));
+	else
+		test(timeTakenReadSubsequent.Int64() < timeTakenReadFirst.Int64());
+#endif
+
+
+	fileRead.Close();
+
+	r = DeleteAll(gSessionPath);
+	test_KErrNone(r);
+
+	delete dummy;
+	test.End();
+
+}
+
+/** Main tests function
+*/
+GLDEF_C void CallTestsL()
+	{
+#if defined(_DEBUG) || defined(_DEBUG_RELEASE)
+	test.Printf(_L("Disabling Lock Fail simulation ...\n"));
+	// turn OFF lock failure mode
+	TBool simulatelockFailureMode = EFalse;
+	TInt r = controlIo(TheFs, gDrive, KControlIoSimulateLockFailureMode, simulatelockFailureMode);
+	test (r == KErrNone);
+#endif
+
+	TBuf16<45> dir;
+
+	RProcess().SetPriority(EPriorityBackground);
+
+	// FileNames/File generation
+	test.Start(_L("Preparing the environmnet\n"));
+	FileNameGen(gFirstFile, 8, gNextFile++);
+	FileNameGen(gSecondFile, 8, gNextFile++);
+	dir = gSessionPath;
+	dir.Append(gFirstFile);
+	gFirstFile = dir;
+	dir = gSessionPath;
+	dir.Append(gSecondFile);
+	gSecondFile = dir;
+
+
+	TRAPD(res,gBuf = HBufC8::NewL(KBlockSize+1));
+	test(res == KErrNone && gBuf != NULL);
+
+	gBufWritePtr.Set(gBuf->Des());
+	FillBuffer(gBufWritePtr, KBlockSize, 'A');
+
+	TRAPD(res2,gBufSec = HBufC8::NewL(KBlockSize+1));
+	test(res2 == KErrNone && gBufSec != NULL);
+	gBufReadPtr.Set(gBufSec->Des());
+
+	test.Next(_L("Negative test\n"));
+	TestNegative();
+
+	test.Next(_L("Simple cases, use of the cache from same location/different"));
+	TestSimpleRead();
+
+	test.Next(_L("Repeated reads, same file\n"));
+	TestRepeatedRead();
+
+	test.Next(_L("Read ahead testing\n"));
+	TestReadAhead();
+
+	test.Next(_L("Concurrent read cases\n"));
+	TestConcurrent();
+
+	test.Next(_L("Fill the cache, boundary testing\n"));
+	TestFillCache();
+
+	test.End();
+	delete gBuf;
+	delete gBufSec;
+
+#if defined(_DEBUG) || defined(_DEBUG_RELEASE)
+	// turn lock failure mode back ON (if enabled)
+	simulatelockFailureMode = ETrue;
+	r = controlIo(TheFs, gDrive, KControlIoSimulateLockFailureMode, simulatelockFailureMode);
+	test (r == KErrNone);
+#endif
+
+	}
+
+/** Initialises semaphores and call the tests
+*/
+LOCAL_C void DoTests()
+	{
+	TInt r = 0;
+
+	r = client.CreateLocal(0);
+ 	test_KErrNone(r);
+
+  	r = gSync.CreateLocal(0);
+ 	test_KErrNone(r);
+
+
+	r = TheFs.SetSessionPath(gSessionPath);
+	test_KErrNone(r);
+
+	r = TheFs.MkDirAll(gSessionPath);
+	if (r != KErrNone && r != KErrAlreadyExists)
+		{
+		test_KErrNone(r);
+		}
+	TheFs.ResourceCountMarkStart();
+	TRAP(r,CallTestsL());
+	if (r == KErrNone)
+		TheFs.ResourceCountMarkEnd();
+	else
+		{
+		test_KErrNone(r);
+		}
+	}
+
+/** Determines the space that can be used for the files
+
+*/
+TBool CheckForDiskSize()
+{
+	TVolumeInfo volInfo;
+	TInt r = TheFs.Volume(volInfo, gDrive);
+	test_KErrNone(r);
+	gMediaSize = volInfo.iSize;
+	gSecondFileSize = KBlockSize*92;
+	gFirstFileSize = KBlockSize*(256);
+	while(((2*gFirstFileSize)+KOneMeg) > gMediaSize )
+		{
+			gFirstFileSize -= (2*KBlockSize);
+		}
+
+	TReal32 small = (TReal32)(gSecondFileSize/KOneK);
+	TReal32 big = (TReal32)(gFirstFileSize/KOneK);
+
+	test.Printf(_L("Test File: %.2f KB\n"), small );
+	test.Printf(_L("Too big for the cache file: %.2f KB (%.2f MB)\n"), big, big / KOneK );
+
+	if(gFirstFileSize < gSecondFileSize)
+		return EFalse;
+	else
+		return ETrue;
+}
+
+/** Main function
+
+	@return KErrNone if everything was ok, panics otherwise
+*/
+GLDEF_C TInt E32Main()
+    {
+	// Determine whether this is a paged ROM -
+	// if it is we bypass some of the timimg tests as the default paging ROMs have a deliberately
+	// small pool of pages (in order to stress the system) and reading things through the file cache
+	// in this "artificial" environment can cause code to be evicted which can result in the read timings
+	// going AWOL. In a more real-world environment, file caching should turn itself off if the amount of
+	// memory falls below a threshold.
+#if !defined(__WINS__)
+	TRomHeader* romHeader = (TRomHeader*)UserSvr::RomHeaderAddress();
+	gPagedRom = romHeader->iPageableRomStart ? (TBool)ETrue : (TBool)EFalse;
+#endif
+
+	RThread t;
+	gMainThreadId = t.Id();
+
+	CTrapCleanup* cleanup;
+	cleanup = CTrapCleanup::New();
+
+	__UHEAP_MARK;
+	test.Start(_L("Starting tests... T_RCACHE"));
+	parseCommandLine();
+
+	TInt r = TheFs.Connect();
+	test_KErrNone(r);
+
+	TDriveInfo info;
+	TVolumeInfo volInfo;
+	r = TheFs.Drive(info,gDrive);
+	test_KErrNone(r);
+
+	if(info.iMediaAtt&KMediaAttVariableSize)
+		{
+		test.Printf(_L("Tests skipped in RAM drive\n"));
+		goto out;
+		}
+
+	r = TheFs.Volume(volInfo, gDrive);
+	if (r == KErrNotReady)
+		{
+		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);
+
+	if(!(volInfo.iFileCacheFlags & (EFileCacheReadEnabled | EFileCacheReadAheadEnabled)))
+		{
+		test.Printf(_L("Skipping tests, Read caching not enabled in this drive\n"));
+		goto out;
+		}
+	
+	if (((volInfo.iDrive.iMediaAtt & KMediaAttFormattable)))
+		Formatting(gDrive,ESpecialFormat);
+
+	if(CheckForDiskSize())
+		{
+		DoTests();
+		}
+	else
+		{
+		test.Printf(_L("Skipping tests due to lack of space to perform them in this drive\n"));
+		}
+out:
+	test.End();
+
+	TheFs.Close();
+	test.Close();
+
+	__UHEAP_MARKEND;
+	delete cleanup;
+	return(KErrNone);
+    }
+