kerneltest/e32test/mmu/t_mmustress.cpp
changeset 0 a41df078684a
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/kerneltest/e32test/mmu/t_mmustress.cpp	Mon Oct 19 15:55:17 2009 +0100
@@ -0,0 +1,669 @@
+// Copyright (c) 2008-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:
+// e32test\mmu\t_mmustress.cpp
+// Stress test for memory management services performed by the kernel's memory model.
+// 
+//
+
+/**
+ @file
+*/
+
+#define __E32TEST_EXTENSION__
+#include <e32test.h>
+#include "u32std.h"
+#include <u32hal.h>
+#include <e32svr.h>
+#include <dptest.h>
+#include <e32def.h>
+#include <e32def_private.h>
+#include "d_memorytest.h"
+#include "..\defrag\d_pagemove.h"
+
+TBool TRACE = 0;
+
+LOCAL_D RTest test(_L("T_MMUSTRESS"));
+
+TUint32 MemModelAttributes;
+TUint32 MemModel;
+TInt PageSize;
+TInt PageMask;
+
+#if !defined(__WINS__) && !defined(__X86__)
+const TPtrC KMoveLddFileName=_L("D_PAGEMOVE.LDD");
+RPageMove MoveLdd;
+#endif
+
+RMemoryTestLdd Ldd;
+
+const TUint KNumTestChunks = 6;
+RChunk Chunks[KNumTestChunks];
+TInt Committed[KNumTestChunks] = {0}; // for each chunk, is the 'owned' region uncommited(0), commited(1) or mixed(-1)
+class TNicePtr8 : public TPtr8 { public: TNicePtr8() : TPtr8(0,0) {} } ChunkPtr[KNumTestChunks];
+
+const TUint KNumSlaveProcesses = 4;
+RProcess Slaves[KNumSlaveProcesses];
+TRequestStatus SlaveLogons[KNumSlaveProcesses];
+TRequestStatus SlaveRendezvous[KNumSlaveProcesses];
+
+TInt SlaveNumber = -1; // master process is slave -1
+
+const TInt KLocalIpcBufferSize = 0x10000;
+TUint8* LocalIpcBuffer = 0;
+
+RSemaphore StartSemaphore;
+
+//
+// Random number generation
+//
+
+TUint32 RandomSeed;
+
+TUint32 Random()
+	{
+	RandomSeed = RandomSeed*69069+1;
+	return RandomSeed;
+	}
+
+TUint32 Random(TUint32 aRange)
+	{
+	return (TUint32)((TUint64(Random())*TUint64(aRange))>>32);
+	}
+
+void RandomInit(TUint32 aSeed)
+	{
+	RandomSeed = aSeed+(aSeed<<8)+(aSeed<<16)+(aSeed<<24);
+	Random();
+	Random();
+	}
+
+
+
+//
+// Chunk utils
+//
+
+TBuf<KMaxKernelName> ChunkName(TInt aChunkNumber)
+	{
+	TBuf<KMaxKernelName> name;
+	name.Format(_L("T_MMUSTRESS-Chunk%d"),aChunkNumber);
+	return name;
+	}
+
+#ifdef __WINS__
+TInt KChunkShift = 16;
+#elif defined(__X86__)
+TInt KChunkShift = 22;
+#else
+TInt KChunkShift = 20;
+#endif
+
+TInt ChunkSize(TInt aChunkNumber)
+	{
+	// biggest chunk (number 0) is big enough for each slave to own a region which is
+	// 2 page tables ('chunks') in size...
+	return (2*KNumSlaveProcesses)<<(KChunkShift-aChunkNumber);
+	}
+
+// check smallest chunk is less than 'chunk' size...
+__ASSERT_COMPILE((2*KNumSlaveProcesses>>(KNumTestChunks-1))==0);
+
+
+/* Memory region 'owned' by this slave process */
+void ChunkOwnedRegion(TInt aChunkNumber,TInt& aOffset,TInt& aSize)
+	{
+	TInt size = ChunkSize(aChunkNumber)/KNumSlaveProcesses;
+	aSize = size;
+	aOffset = SlaveNumber*size;
+	test_Equal(0,size&PageMask);
+	}
+
+void ChunkMarkRegion(TInt aChunkNumber,TInt aOffset,TInt aSize)
+	{
+	TInt pageSize = PageSize;
+	TUint32 mark = aOffset|aChunkNumber|(SlaveNumber<<4);
+	TUint8* ptr = Chunks[aChunkNumber].Base()+aOffset;
+	TUint8* ptrEnd = ptr+aSize;
+	while(ptr<ptrEnd)
+		{
+		((TUint32*)ptr)[0] = mark;
+		((TUint32*)ptr)[1] = ~mark;
+		mark += pageSize;
+		ptr += pageSize;
+		}
+	}
+
+void ChunkCheckRegion(TInt aChunkNumber,TInt aOffset,TInt aSize)
+	{
+	TInt pageSize = PageSize;
+	TUint32 mark = aOffset|aChunkNumber|(SlaveNumber<<4);
+	TUint8* ptr = Chunks[aChunkNumber].Base()+aOffset;
+	TUint8* ptrEnd = ptr+aSize;
+	while(ptr<ptrEnd)
+		{
+		test_Equal(mark,((TUint32*)ptr)[0]);
+		test_Equal(~mark,((TUint32*)ptr)[1]);
+		mark += pageSize;
+		ptr += pageSize;
+		}
+	}
+
+TInt ChunkOpen(TInt aChunkNumber)
+	{
+	RChunk& chunk = Chunks[aChunkNumber];
+	if(chunk.Handle()!=0)
+		return KErrNone;
+
+	if(TRACE) RDebug::Printf("%d %d Open",SlaveNumber,aChunkNumber);
+	TInt r = chunk.OpenGlobal(ChunkName(aChunkNumber),false);
+	if(r!=KErrNoMemory)
+		test_KErrNone(r);
+	return r;
+	}
+
+//
+// Server utils
+//
+
+TBuf<KMaxKernelName> ServerName(TInt aSlaveNumber)
+	{
+	TBuf<KMaxKernelName> name;
+	name.Format(_L("T_MMUSTRESS-Server%d"),aSlaveNumber);
+	return name;
+	}
+
+RServer2 Server;
+RMessage2 ServerMessage;
+TRequestStatus ServerStatus;
+
+class RTestSession : public RSessionBase
+	{
+public:
+	TInt Connect(TInt aServerNumber)
+		{
+		return CreateSession(ServerName(aServerNumber),TVersion(),1,EIpcSession_Unsharable,0,&iStatus);
+		}
+	TInt Send(TInt aChunkNumber)
+		{
+		return RSessionBase::Send(0,TIpcArgs(SlaveNumber,aChunkNumber,&ChunkPtr[aChunkNumber]));
+		}
+	TRequestStatus iStatus;
+	};
+RTestSession Sessions[KNumSlaveProcesses];
+
+
+//
+//
+//
+
+void SlaveInit()
+	{
+	RDebug::Printf("Slave %d initialising",SlaveNumber);
+
+	TBuf<KMaxKernelName> name;
+	name.Format(_L("T_MMUSTRESS-Slave%d"),SlaveNumber);
+	User::RenameThread(name);
+
+	test_KErrNone(StartSemaphore.Open(2));
+	TInt r;
+#if !defined(__WINS__) && !defined(__X86__)
+	// Move ldd may not be in the ROM so needs to be loaded.
+	r=User::LoadLogicalDevice(KMoveLddFileName);
+	test_Value(r, r==KErrNone || r==KErrAlreadyExists);
+	test_KErrNone(MoveLdd.Open());
+#endif
+
+	test_KErrNone(Ldd.Open());
+	test_KErrNone(Ldd.CreateVirtualPinObject());
+
+	LocalIpcBuffer = (TUint8*)User::Alloc(KLocalIpcBufferSize);
+	test(LocalIpcBuffer!=0);
+
+	test_KErrNone(Server.CreateGlobal(ServerName(SlaveNumber)));
+
+	TUint i;
+
+	// create sessions with other slaves...
+	for(i=0; i<KNumSlaveProcesses; i++)
+		{
+		for(;;)
+			{
+			r = Sessions[i].Connect(i);
+//			RDebug::Printf("%d Session %d = %d,%d",SlaveNumber,i,r,Sessions[i].iStatus.Int());
+			if(r==KErrNotFound)
+				{
+				// give other slaves time to create their servers...
+				User::After(10000);
+				continue;
+				}
+			test_KErrNone(r);
+			break;
+			}
+		}
+
+	// process session connect messages...
+	for(i=0; i<KNumSlaveProcesses; i++)
+		{
+		RMessage2 m;
+//		RDebug::Printf("%d Server waiting for connect message",SlaveNumber);
+		Server.Receive(m);
+		test_Equal(RMessage2::EConnect,m.Function())
+		m.Complete(KErrNone);
+		}
+
+	// wait for our session connections...
+	for(i=0; i<KNumSlaveProcesses; i++)
+		{
+//		RDebug::Printf("%d Session wait %d",SlaveNumber,i);
+		User::WaitForRequest(Sessions[i].iStatus);
+		}
+
+	// prime server for receiving mesages...
+	Server.Receive(ServerMessage,ServerStatus);
+
+	// synchronise with other processes...
+	RDebug::Printf("Slave %d waiting for trigger",SlaveNumber);
+	RProcess::Rendezvous(KErrNone);
+	StartSemaphore.Wait();
+	RDebug::Printf("Slave %d started",SlaveNumber);
+	}
+
+
+
+//
+// Test by random operations...
+//
+
+void DoTest()
+	{
+	RandomInit(SlaveNumber);
+	TInt r;
+	for(;;)
+		{
+		// select random chunk...
+		TInt chunkNumber = Random(KNumTestChunks);
+		RChunk& chunk = Chunks[chunkNumber];
+
+		// get the region of this chunk which this process 'owns'...
+		TInt offset;
+		TInt size;
+		ChunkOwnedRegion(chunkNumber,offset,size);
+
+		// calculate a random region in the owned part...
+		TInt randomOffset = offset+(Random(size)&~PageMask);
+		TInt randomSize = (Random(size-(randomOffset-offset))+PageMask)&~PageMask;
+		if(!randomSize)
+			continue; // try again
+
+		// pick a random slave...
+		TInt randomSlave = Random(KNumSlaveProcesses);
+
+		// open chunk if it isn't already...
+		r = ChunkOpen(chunkNumber);
+		if(r==KErrNoMemory)
+			continue; // can't do anything with chunk if we can't open it
+
+		// check our contents of chunk...
+		if(Committed[chunkNumber]==1)
+			{
+			if(TRACE) RDebug::Printf("%d %d Check %08x+%08x",SlaveNumber,chunkNumber,offset,size);
+			ChunkCheckRegion(chunkNumber,offset,size);
+			}
+
+		// perform random operation...
+		switch(Random(12))
+			{
+		case 0:
+		case 1:
+			// close chunk...
+			if(TRACE) RDebug::Printf("%d %d Close",SlaveNumber,chunkNumber);
+			chunk.Close();
+			break;
+
+		case 2:
+			// commit all...
+			if(TRACE) RDebug::Printf("%d %d Commit all %08x+%08x",SlaveNumber,chunkNumber,offset,size);
+			if(Committed[chunkNumber]!=0)
+				{
+				r = chunk.Decommit(offset,size);
+				test_KErrNone(r);
+				Committed[chunkNumber] = 0;
+				}
+			r = chunk.Commit(offset,size);
+			if(r!=KErrNoMemory)
+				{
+				test_KErrNone(r);
+				Committed[chunkNumber] = 1;
+				ChunkMarkRegion(chunkNumber,offset,size);
+				}
+			break;
+
+		case 3:
+			// decommit all...
+			if(TRACE) RDebug::Printf("%d %d Decommit all %08x+%08x",SlaveNumber,chunkNumber,offset,size);
+			r = chunk.Decommit(offset,size);
+			test_KErrNone(r);
+			Committed[chunkNumber] = 0;
+			break;
+
+		case 4:
+		case 5:
+			// commit random...
+			if(TRACE) RDebug::Printf("%d %d Commit %08x+%08x",SlaveNumber,chunkNumber,randomOffset,randomSize);
+			r = chunk.Commit(randomOffset,randomSize);
+			if(r!=KErrNoMemory)
+				{
+				if(Committed[chunkNumber]==0)
+					{
+					test_KErrNone(r);
+					Committed[chunkNumber] = -1;
+					}
+				else if(Committed[chunkNumber]==1)
+					{
+					test_Equal(KErrAlreadyExists,r);
+					}
+				else
+					{
+					if(r!=KErrAlreadyExists)
+						test_KErrNone(r);
+					}
+				}
+			break;
+
+		case 6:
+		case 7:
+			// decommit random...
+			if(TRACE) RDebug::Printf("%d %d Decommit %08x+%08x",SlaveNumber,chunkNumber,randomOffset,randomSize);
+			r = chunk.Decommit(randomOffset,randomSize);
+			test_KErrNone(r);
+			if(Committed[chunkNumber]==1)
+				Committed[chunkNumber] = -1;
+			break;
+
+		case 8:
+			if(TRACE) RDebug::Printf("%d %d IPC Send->%d",SlaveNumber,chunkNumber,randomSlave);
+//			ChunkPtr[chunkNumber].Set(chunk.Base(),ChunkSize(chunkNumber),ChunkSize(chunkNumber));
+			ChunkPtr[chunkNumber].Set(chunk.Base()+offset,size,size);
+			Sessions[randomSlave].Send(chunkNumber);
+			break;
+
+		case 9:
+			// process IPC messages...
+			if(ServerStatus.Int()==KRequestPending)
+				continue;
+			User::WaitForRequest(ServerStatus);
+
+			{
+			TInt sourceSlave = ServerMessage.Int0();
+			chunkNumber = ServerMessage.Int1();
+			if(TRACE) RDebug::Printf("%d %d IPC Receive<-%d",SlaveNumber,chunkNumber,sourceSlave);
+			test_Equal(0,ServerMessage.Function());
+
+			// get local descriptor for owned region in chunk...
+			size = ServerMessage.GetDesMaxLength(2);
+			test_NotNegative(size);
+			if(size>KLocalIpcBufferSize)
+				size = KLocalIpcBufferSize;
+			TPtr8 local(LocalIpcBuffer,size,size);
+
+//			if(Random(2))
+				{
+				// IPC read from other slave...
+				if(TRACE) RDebug::Printf("%d %d IPC Read<-%d",SlaveNumber,chunkNumber,sourceSlave);
+				TInt panicTrace = Ldd.SetPanicTrace(EFalse);
+				r = ServerMessage.Read(2,local);
+				Ldd.SetPanicTrace(panicTrace);
+				if(r!=KErrBadDescriptor)
+					test_KErrNone(r);
+				}
+//			else
+//				{
+//				// IPC write to other slave...
+//				if(TRACE) RDebug::Printf("%d %d IPC Write->%d",SlaveNumber,chunkNumber,sourceSlave);
+//				r = ServerMessage.Write(2,local,offset);
+//				if(r!=KErrBadDescriptor)
+//					test_KErrNone(r);
+//				if(Committed[chunkNumber]==1)
+//					ChunkMarkRegion(chunkNumber,offset,size);
+//				}
+			}
+
+			ServerMessage.Complete(KErrNone);
+			Server.Receive(ServerMessage,ServerStatus);
+			break;
+
+		case 10:
+		case 11:
+			// pin memory...
+			{
+			test_KErrNone(Ldd.UnpinVirtualMemory());
+			for(TInt tries=10; tries>0; --tries)
+				{
+				TInt chunkSize = ChunkSize(chunkNumber);
+				offset = Random(chunkSize);
+				TInt maxSize = chunkSize-offset;
+				if(maxSize>0x1000)
+					maxSize = 0x1000;
+				size = Random(maxSize);
+				r = Ldd.PinVirtualMemory((TLinAddr)chunk.Base()+offset, size);
+				if(r!=KErrNotFound && r!=KErrNoMemory)
+					{
+					test_KErrNone(r);
+					break;
+					}
+				}
+			}
+			break;
+		case 12:
+		case 13:
+			// Move any page in the chunk, not just the owned region.
+			{
+#if !defined(__WINS__) && !defined(__X86__)
+			for(TInt tries=10; tries>0; --tries)
+				{
+				TInt chunkSize = ChunkSize(chunkNumber);
+				offset = Random(chunkSize);
+				MoveLdd.TryMovingUserPage((TAny*)(chunk.Base()+offset), ETrue);
+				// Allow the move to fail for any reason as the page of the chunk
+				// may or may not be currently committed, pinned, or accessed.
+				}
+#endif
+			}
+			break;
+		default:
+			test(false); // can't happen
+			break;
+			}
+		}
+	}
+
+
+
+TInt E32Main()
+	{
+	// get system info...
+	MemModelAttributes = UserSvr::HalFunction(EHalGroupKernel, EKernelHalMemModelInfo, NULL, NULL);
+	MemModel = MemModelAttributes&EMemModelTypeMask;
+	UserHal::PageSizeInBytes(PageSize);
+	PageMask = PageSize-1;
+
+	// see if we are a slave process...
+	if(User::GetTIntParameter(1,SlaveNumber)==KErrNone)
+		{
+		// do testing...
+		SlaveInit();
+		DoTest();
+		return KErrGeneral; // shouldn't have returned from testing
+		}
+
+	// master process...
+	TBool pass = true; // final test result
+	test.Title();
+	if((MemModelAttributes&EMemModelAttrVA)==false)
+		{
+		test.Start(_L("TESTS NOT RUN - Not relevent for the memory model"));
+		test.End();
+		return KErrNone;
+		}
+
+	// get time to run tests for...
+	TInt timeout = 10; // time in seconds
+	TInt cmdLineLen = User::CommandLineLength();
+	if(cmdLineLen)
+		{
+		// get timeout value from command line
+		RBuf cmdLine;
+		test_KErrNone(cmdLine.Create(cmdLineLen));
+		User::CommandLine(cmdLine);
+		test_KErrNone(TLex(cmdLine).Val(timeout));
+		if(timeout==0)
+			timeout = KMaxTInt;
+		}
+	TTimeIntervalMicroSeconds32 tickTime;
+	test_KErrNone(UserHal::TickPeriod(tickTime));
+	TInt ticksPerSecond = 1000000/tickTime.Int();
+	TInt timeoutTicks;
+	if(timeout<KMaxTInt/ticksPerSecond)
+		timeoutTicks = timeout*ticksPerSecond;
+	else
+		{
+		timeoutTicks = KMaxTInt;
+		timeout = timeoutTicks/ticksPerSecond;
+		}
+
+	// master process runs at higher priority than slaves so it can timeout and kill them...
+	RThread().SetPriority(EPriorityMore);
+
+	test.Start(_L("Creating test chunks"));
+	TUint i;
+	for(i=0; i<KNumTestChunks; i++)
+		{
+		test.Printf(_L("Size %dkB\r\n"),ChunkSize(i)>>10);
+		test_KErrNone(Chunks[i].CreateDisconnectedGlobal(ChunkName(i),0,0,ChunkSize(i)));
+		}
+
+	test.Next(_L("Spawning slave processes"));
+	test_KErrNone(StartSemaphore.CreateGlobal(KNullDesC,0));
+	TFileName processFile(RProcess().FileName());
+	for(i=0; i<KNumSlaveProcesses; i++)
+		{
+		test.Printf(_L("Slave %d\r\n"),i);
+		RProcess& slave = Slaves[i];
+		test_KErrNone(slave.Create(processFile,KNullDesC));
+		test_KErrNone(slave.SetParameter(1,i));
+		test_KErrNone(slave.SetParameter(2,StartSemaphore));
+		slave.Logon(SlaveLogons[i]);
+		test_Equal(KRequestPending,SlaveLogons[i].Int());
+		slave.Rendezvous(SlaveRendezvous[i]);
+		test_Equal(KRequestPending,SlaveRendezvous[i].Int());
+		}
+
+	test.Next(_L("Create timer"));
+	RTimer timer;
+	test_KErrNone(timer.CreateLocal());
+
+	test.Next(_L("Resuming slave processes"));
+	for(i=0; i<KNumSlaveProcesses; i++)
+		Slaves[i].Resume();
+
+	// this test must now take care not to die (e.g. panic due to assert fail)
+	// until it has killed the slave processes
+
+	test.Next(_L("Change paging cache size"));
+	TUint cacheOriginalMin = 0;
+	TUint cacheOriginalMax = 0;
+	TUint cacheCurrentSize = 0;
+	DPTest::CacheSize(cacheOriginalMin, cacheOriginalMax, cacheCurrentSize);
+	DPTest::SetCacheSize(1, 2*ChunkSize(0)); // big enough for all the test chunks
+
+	test.Next(_L("Wait for slaves to initialise"));
+	TRequestStatus timeoutStatus;
+	timer.After(timeoutStatus,10*1000000); // allow short time for slaves to initialise
+	for(i=0; i<KNumSlaveProcesses; i++)
+		{
+		User::WaitForAnyRequest(); // wait for a rendexvous
+		if(timeoutStatus.Int()!=KRequestPending)
+			{
+			test.Printf(_L("Timeout waiting for slaves to initialise\r\n"));
+			pass = false;
+			break;
+			}
+		}
+
+	test.Next(_L("Restore paging cache size"));
+	DPTest::SetCacheSize(cacheOriginalMin, cacheOriginalMax);
+
+	if(pass)
+		{
+		timer.Cancel();
+		User::WaitForAnyRequest(); // swallow timer signal
+
+		test.Next(_L("Check slaves are ready"));
+		for(i=0; i<KNumSlaveProcesses; i++)
+			{
+			if(SlaveRendezvous[i].Int()!=KErrNone || Slaves[i].ExitType()!=EExitPending)
+				{
+				test.Printf(_L("Slaves not ready or died!\r\n"));
+				pass = false;
+				break;
+				}
+			}
+		}
+
+	if(pass)
+		{
+		test.Next(_L("Setup simulated kernel heap failure"));
+		__KHEAP_SETFAIL(RAllocator::EDeterministic,100);
+
+		TBuf<80> text;
+		text.Format(_L("Stressing for %d seconds..."),timeout);
+		test.Next(text);
+		timer.AfterTicks(timeoutStatus,timeoutTicks);
+		StartSemaphore.Signal(KNumSlaveProcesses); // release slaves to start testing
+		User::WaitForAnyRequest(); // wait for timeout or slave death via logon completion
+
+		pass = timeoutStatus.Int()==KErrNone; // timeout means slaves are still running OK
+
+		test.Next(_L("Check slaves still running"));
+		for(i=0; i<KNumSlaveProcesses; i++)
+			if(Slaves[i].ExitType()!=EExitPending)
+				pass = false;
+
+		test.Next(_L("Clear kernel heap failure"));
+		TUint kheapFails = __KHEAP_CHECKFAILURE;
+		__KHEAP_RESET;
+		test.Printf(_L("Number of simulated memory failures = %d\r\n"),kheapFails);
+		}
+
+	test.Next(_L("Killing slave processes"));
+	for(i=0; i<KNumSlaveProcesses; i++)
+		Slaves[i].Kill(0);
+
+	test.Next(_L("Assert test passed"));
+	test(pass);
+
+	test.End();
+
+	for(i=0; i<KNumSlaveProcesses; i++)
+		Slaves[i].Close();
+	for(i=0; i<KNumTestChunks; i++)
+		Chunks[i].Close();
+	timer.Close();
+	for(i=0; i<KNumSlaveProcesses; i++)
+		User::WaitForRequest(SlaveLogons[i]);
+
+	UserSvr::HalFunction(EHalGroupKernel, EKernelHalSupervisorBarrier, (TAny*)5000, 0);
+
+	return KErrNone;
+	}