graphicstest/uibench/src/tmultiptreventhandlingperf.cpp
author Faisal Memon <faisal.memon@nokia.com>
Fri, 14 May 2010 17:03:25 +0100
branchNewGraphicsArchitecture
changeset 67 9c70c0a878c0
parent 0 5d03bc08d59c
permissions -rw-r--r--
Merge 3. Improve performance by switching to less aggressive settings for RI_NUM_TESSELLATED_SEGMENTS and RI_MAX_SAMPLES. Ignored the WIN32 specific API decoration defines when doing the merge. Note the code is now optimised in riPath.cpp to change from RI_NUM_TESSELLATED_SEGMENTS to _QUAD, _CUBIC, _ARC settings which are each now set to 8. SVG Tiger now renders in 5 seconds (15 seconds quicker). The quality of the OVG icons is slightly reduced but still very useable.

// 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 "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
//

/**
 @file
 @test
 @internalComponent - Internal Symbian test code 
*/

#include "tmultiptreventhandlingperf.h"
#include <hal.h>

_LIT(KPerformanceTimingChunk, "PerformanceTimingChunk");

// Number of iterations for eventset
const TInt KNumOfIterations = 100;

// Maximum descriptor length for results
const TInt KMaxDescLength = 528;

// Static variables used for synchronisation between threads
GLDEF_D TInt NumOfEventsAdded = 0;
GLDEF_D RMutex Mutex;
GLDEF_D RCondVar ConVar;

void MultiPtrPerfPanic(TInt aPanic)
	{
	User::Panic(_L("MultiPterEventHanldingPerformancePanic"), aPanic);
	}

// This function opens the chunk and adds the failure description followed by '*'
TInt CMultiPtrPerfTestControl::Failed(TPoint3D aExp3DPoint, TPoint3D aActual3DPoint, TInt aErrorCode, TInt aExpPtrNum, TInt aActPtrNum)
	{
	TInt ret = iChunk.OpenGlobal(KPerformanceTimingChunk, ETrue);
	if (ret != KErrNone)
		{
		return ret;
		}
	TUint8* desPtr = iChunk.Base() + iChunkOffset;
	TPtr8 ptrDes(desPtr, 0, KMaxDescLength);
	
	TBuf<80> buf;
	if (aExpPtrNum != aActPtrNum)
		{
		_LIT(KFailedPointerNum, "Failed Expected Pointer Num = %d Actual Pointer Num = %d*");
		buf.AppendFormat(KFailedPointerNum, aExpPtrNum, aActPtrNum);
		ptrDes.Append(buf);
		} 
	else if (aErrorCode != KErrNone)
		{
		_LIT(KFailedErrorCode, "Failed Errorcode = %d*");
		buf.AppendFormat(KFailedErrorCode, aErrorCode);
		ptrDes.Append(buf);
		}
	else
		{
		_LIT(KFailedWrongCoord, "Failed Coordinates Expected = [%d, %d, %d] Actual = [%d, %d, %d]*");
		buf.AppendFormat(KFailedWrongCoord, aExp3DPoint.iX, aExp3DPoint.iY, aExp3DPoint.iZ, aActual3DPoint.iX, aActual3DPoint.iY, aActual3DPoint.iZ);
		ptrDes.Append(buf);
		}
	
	iChunk.Close();
	return ret;
	}

// This function opens the chunk and adds a pointer event not supported message followed by '#'
TInt CMultiPtrPerfTestControl::PointerEventsNotSupported()
    {
    iPointerEventNotSupported = ETrue;
    TInt ret = iChunk.OpenGlobal(KPerformanceTimingChunk, ETrue);
        if (ret != KErrNone)
            {
            return ret;
            }
    TUint8* desPtr = iChunk.Base() + iChunkOffset;
    TPtr8 ptrDes(desPtr, 0, KMaxDescLength);
    TBuf<80> buf;
    _LIT(KPointerEventsNotSupported, "Test Skipped As Pointer Events Are Not Supported By This Configuration#");
    buf.AppendFormat(KPointerEventsNotSupported);
    ptrDes.Append(buf);
    iChunk.Close();
    return ret;
    }

// This function calulates the average latency, opens the chunk and appends
// the same in descriptor form. Then close the child thread and calls function 
// for creating new thread for events. 
// If all the events have been tested then stops the tests
// by calling active CActiveScheduler::Stop. 
TInt CMultiPtrPerfTestControl::CalculateLatencyAndStartThread()
	{
	TInt ret = KErrNone;
	switch (iPtrAppUi->TestCaseNum())
		{
		case 1:
		case 5:
		case 9:
		case 13:	
			iAverageLatency = iLatency/(4*KNumOfIterations);
			break;
		case 2:
		case 6:
		case 10:
		case 14:
			iAverageLatency = iLatency/(8*KNumOfIterations);
			break;
		case 3:
		case 7:
		case 11:
		case 15:
			iAverageLatency = iLatency/(16*KNumOfIterations);
			break;
		case 4:
		case 8:
		case 12:
		case 16:
			iAverageLatency = iLatency/(32*KNumOfIterations);
			break;
		default:
			MultiPtrPerfPanic(EMultiPtrPerfPanicWrongTest);
		}

	ret = iChunk.OpenGlobal(KPerformanceTimingChunk, ETrue);
	if (ret != KErrNone)
		{
		return ret;
		}
	TUint8* desPtr = iChunk.Base() + iChunkOffset;
	TPtr8 ptrDes(desPtr, iLatLenInLetters, KMaxDescLength);
	
	TBuf<66> buf;
	buf.Num(iAverageLatency);
	buf.Append(',');
	ptrDes.Append(buf);
	
	// reset the iLatency for next set of test
	iLatency = 0;	
	
	// For writing the next latency in chunk memory
	iLatLenInLetters = ptrDes.Length();
	
	iChunk.Close();
	
	// Before calling the thread Function close the current thread and then call the function
	// here when we kill the child thread it releases both mutex and condvar that it holds
	iPtrAppUi->ChildThread().Kill(KErrNone);
	iPtrAppUi->ChildThread().Close();
	
	if (iPtrAppUi->TestCaseNum() == 16)
		{
		// It it is the last test case then stop the active scheduler
		CActiveScheduler::Stop();
		return ret;
		}
	else
		{
		return iPtrAppUi->CreateEventSimulatingThreads();
		}
	}

// This function is called by Cone whenever a event is outstanding
// Here we check if the sent event is same as the expected event in array.
// Then resumes the thread for next set of events. When all the events are completed
// calls CalculateLatencyAndStartThread
void CMultiPtrPerfTestControl::HandlePointerEventL(const TPointerEvent& aPointerEvent)
	{
    ASSERT(!iPointerEventNotSupported);
	// The event that is received is (0,0,0) then it is the first event that the test code is sending
	const TAdvancedPointerEvent* advancedPointerEvent = aPointerEvent.AdvancedPointerEvent();
	
	if (iPtrAppUi->TestCaseNum() == 0)
		{
		if (advancedPointerEvent->PointerNumber() == 0 && advancedPointerEvent->Position3D() == TPoint3D())
			{
			iPtrAppUi->CreateEventSimulatingThreads();
			return;
			}
		else
			{
			return;	// ignore other events before starting the tests
			}
		}

	// Get the current time.
	// Subtract from the test start time.
	TTime currentTime;
	currentTime.UniversalTime();
	TInt64 curTime64 = currentTime.Int64();
	TInt timeSinceTestStart = curTime64 - iPtrAppUi->TestStartTime();
	
	// Get the event, get the time stored in Z corodinate
	// Then subtract the calculated time from Z coordinate
	iLatency += timeSinceTestStart - advancedPointerEvent->ProximityAndPressure(); 
	
	// Get the current event from buffer and compare it with got event
	TAdvancedPointerEvent expectedPtrEvent;
	if(0 == iEventBuffer.Remove(&expectedPtrEvent))
		{
		iPtrAppUi->Failed(TPoint3D(), TPoint3D(), KErrNotFound);
		return;
		}
	
	// Here Z corodinate is not checked.
	if (expectedPtrEvent.iPosition != advancedPointerEvent->iPosition)
		{
		iPtrAppUi->Failed(expectedPtrEvent.Position3D(), advancedPointerEvent->Position3D(), KErrNone);
		return;
		}
	
	if (expectedPtrEvent.PointerNumber() != advancedPointerEvent->PointerNumber())
		{
		iPtrAppUi->Failed(TPoint3D(), TPoint3D(), KErrNone, expectedPtrEvent.PointerNumber(), advancedPointerEvent->PointerNumber());
		return;
		}
	
	iPtrAppUi->DecrementEventCount();
	NumOfEventsAdded--;
	
	// Event count is zero when all the events have been tested
	if (iPtrAppUi->EventCount() == 0)
		{
		TInt ret = CalculateLatencyAndStartThread(); 
		if (ret != KErrNone)
			{
			iPtrAppUi->Failed(TPoint3D(), TPoint3D(), ret);
			return;
			}
		}
	else if (NumOfEventsAdded == 0) 
		{
		ConVar.Signal();
		}
	}

void CMultiPtrPerfTestControl::AddExpectedEvent(TAdvancedPointerEvent& aExpEvent)
	{
	iEventBuffer.Add(&aExpEvent);
	}

CMultiPtrPerfTestControl::~CMultiPtrPerfTestControl()
	{
	iChunk.Close();
	}

void CMultiPtrPerfTestControl::ConstructL()
	{
    iPointerEventNotSupported = EFalse;
	ConstructL(TPoint(), ControlEnv()->ScreenDevice()->SizeInPixels());
	}

void CMultiPtrPerfTestControl::ConstructL(TPoint aOrigin, TSize aSize)
	{
    iEventBuffer.SetLengthL(EEventBufferSize);
    
    iPtrAppUi = static_cast<CMultiPtrPerfTestAppUi*>(ControlEnv()->AppUi());
        
    CreateWindowL();
    Window().EnableAdvancedPointers();
    EnableDragEvents();
    SetExtent(aOrigin, aSize);
    ActivateL();
    
 	// Get the cmdline argument of this process in descriptor
 	// convert the descriptor to number and store it in iChunkOffset
 	TBuf<128> buf;
 	User::CommandLine(buf);
 	TLex lex(buf);
 	User::LeaveIfError(lex.Val(iChunkOffset));
	}

// This function gets the current time and subtracts it from the time when the whole test was started.
// Simulates the event and suspends the thread if aWaitAfterEachEvent is ETrue
// If aWaitAfterEachEvent is EFalse then it suspends the thread after each set of event i,e after 4, 8, 16 and 32 events 
void SimulatePointerEvents(TInt aCount, CMultiPtrPerfTestAppUi* appUi, TBool aMultiPtrEvent = EFalse, TBool aWaitAfterEachEvent = ETrue)
	{
	TRawEvent rawEvent;
	TPoint ptrPos;
	TTime currentTime;
	TInt64 testStartTime = appUi->TestStartTime();
	TInt ptrNum = 0;
	TInt ptrMax = 0;
	// HAL as already been tested at the start of these tests. So no need to test once again
	HAL::Get(HALData::EPointerNumberOfPointers, ptrMax);
	
	appUi->SetNumberOfEvents((KNumOfIterations*aCount)-1);

	// For testing time taken for each event and for each set of events we make use of RMutex and RConvar.
	// RMutex is used not to preempt this thread until events has been added
	// RConVar is used to hold this thread until the events have been tested
	for (TInt loop = 0; loop < KNumOfIterations; loop++)
		{
		ptrNum = 0;
		ptrPos.iX = ptrPos.iY = 0;
		Mutex.Wait();
		for (TInt count = 0; count < aCount/2; count++, ptrNum++)
			{
			if (ptrNum >= ptrMax)
				{
				ptrNum = 0;
				}			
			ptrPos.iX += 2;
			ptrPos.iY += 1;
			
			TWsEvent event;
			event.InitAdvancedPointerEvent(TPointerEvent::EButton1Down, 0, TPoint3D(ptrPos.iX, ptrPos.iY, 0), (aMultiPtrEvent ? ptrNum : 0));
			appUi->AddExpectedEvent(*event.Pointer());
			currentTime.UniversalTime();
			TInt64 curTime64 = currentTime.Int64();
			TInt timeSinceTestStart = curTime64 - testStartTime;
			rawEvent.Set(TRawEvent::EButton1Down, ptrPos.iX, ptrPos.iY, timeSinceTestStart, (aMultiPtrEvent ? ptrNum : 0));
			UserSvr::AddEvent(rawEvent);
			NumOfEventsAdded++;
			if (aWaitAfterEachEvent)
				{
				while(NumOfEventsAdded)
					ConVar.Wait(Mutex);
				}
			
			event.InitAdvancedPointerEvent(TPointerEvent::EButton1Up, 0, TPoint3D(ptrPos.iX, ptrPos.iY, 0), (aMultiPtrEvent ? ptrNum : 0));
			appUi->AddExpectedEvent(*event.Pointer());
			currentTime.UniversalTime();
			curTime64 = currentTime.Int64();
			timeSinceTestStart = curTime64 - testStartTime;
			rawEvent.Set(TRawEvent::EButton1Up, ptrPos.iX, ptrPos.iY, timeSinceTestStart, (aMultiPtrEvent ? ptrNum : 0));
			UserSvr::AddEvent(rawEvent);
			NumOfEventsAdded++;
			if (aWaitAfterEachEvent)
				{
				while(NumOfEventsAdded)
					ConVar.Wait(Mutex);
				}
			}
		if (!aWaitAfterEachEvent)
			{
			while(NumOfEventsAdded)
				ConVar.Wait(Mutex);
			}
		Mutex.Signal();
		}
	}

TInt EventSimulatingThreadStartFunc(TAny* aAny)
	{
	CMultiPtrPerfTestAppUi* appUi = static_cast<CMultiPtrPerfTestAppUi*>(aAny);

	switch(appUi->TestCaseNum()++)
		{
		case 0:
			SimulatePointerEvents(4, appUi);
			break;
		case 1:
			SimulatePointerEvents(8, appUi);
			break;
		case 2:
			SimulatePointerEvents(16, appUi);
			break;
		case 3:
			SimulatePointerEvents(32, appUi);
			break;
		case 4:
			SimulatePointerEvents(4, appUi, ETrue);
			break;
		case 5:
			SimulatePointerEvents(8, appUi, ETrue);
			break;
		case 6:
			SimulatePointerEvents(16, appUi, ETrue);
			break;
		case 7:
			SimulatePointerEvents(32, appUi, ETrue);
			break;
		case 8:
			SimulatePointerEvents(4, appUi, EFalse, EFalse);
			break;
		case 9:
			SimulatePointerEvents(8, appUi, EFalse, EFalse);
			break;
		case 10:
			SimulatePointerEvents(16, appUi, EFalse, EFalse);
			break;
		case 11:
			SimulatePointerEvents(32, appUi, EFalse, EFalse);
			break;
		case 12:
			SimulatePointerEvents(4, appUi, ETrue, EFalse);
			break;
		case 13:
			SimulatePointerEvents(8, appUi, ETrue, EFalse);
			break;
		case 14:
			SimulatePointerEvents(16, appUi, ETrue, EFalse);
			break;
		case 15:
			SimulatePointerEvents(32, appUi, ETrue, EFalse);
			break;		
		default:
			break;
		}
	return KErrNone;
	}

TInt CMultiPtrPerfTestAppUi::CreateEventSimulatingThreads()
	{
	// Depending upon the iTestCase number create thread and simulate events in that thread function
	TInt ret = KErrNone;
	_LIT(KSimulateEventsThread, "Events simulating thread");
	ret = iThread.Create(KSimulateEventsThread, EventSimulatingThreadStartFunc, KDefaultStackSize, 0x4000, 0x4000, this, EOwnerThread);
	if (ret == KErrNone)
		{
		iThread.Resume();
		}
	return ret;
	}

void CMultiPtrPerfTestAppUi::AddExpectedEvent(TAdvancedPointerEvent& aExpEvent)
	{
	iControl->AddExpectedEvent(aExpEvent);
	}

void CMultiPtrPerfTestAppUi::Failed(TPoint3D aExp3DPoint, TPoint3D aActual3DPoint, TInt aErrorCode, TInt aExpPtrNum, TInt aActPtrNum)
	{
	// Write error description in the memory of shared chunk 
	// so that the main performance test just stops by saying that it failed
	// and then stop the active scheduler.
	iControl->Failed(aExp3DPoint, aActual3DPoint, aErrorCode, aExpPtrNum, aActPtrNum);
	CActiveScheduler::Stop();
	}

void CMultiPtrPerfTestAppUi::PointerEventsNotSupported()
    {
    // Write a pointer events not supported message in the memory of shared chunk 
    // so that the main performance test just skips the pointer test
    iControl->PointerEventsNotSupported();
    }

CMultiPtrPerfTestAppUi::CMultiPtrPerfTestAppUi()
	{
	}

CMultiPtrPerfTestAppUi::~CMultiPtrPerfTestAppUi()
	{
	RemoveFromStack(iControl);
	delete iControl;
	}

void CMultiPtrPerfTestAppUi::ConstructL()
	{
	CCoeAppUi::ConstructL();
	
	TTime testStartTime;
	testStartTime.UniversalTime();
	iTestStartTime = testStartTime.Int64();
	
	iControl = new (ELeave) CMultiPtrPerfTestControl();
	iControl->ConstructL();
	AddToStackL(iControl);
	}

void ConstructControlEnvironmentL(CCoeEnv* aCoe)
	{
 	aCoe->ConstructL();
 	CMultiPtrPerfTestAppUi* appUi=new(ELeave) CMultiPtrPerfTestAppUi();
 	CleanupStack::PushL(appUi);			// If it leaves after this then there is no way of deleting the appui
 	aCoe->SetAppUi(appUi);				// So pushed it on to cleanup stack
 	aCoe->WsSession().SetAutoFlush(ETrue);
 	appUi->ConstructL();
 	User::LeaveIfError(Mutex.CreateLocal(EOwnerProcess));
 	User::LeaveIfError(ConVar.CreateLocal(EOwnerProcess));
 	CleanupStack::Pop(appUi);
	}

GLDEF_C TInt E32Main()
	{
	__UHEAP_MARK;
	CCoeEnv* coe = new CCoeEnv;
	if (!coe)
		{
		return KErrNoMemory; 
		}
	TRAPD(err, ConstructControlEnvironmentL(coe));
	if (err != KErrNone)
		{
		delete coe;
		}
	else	// If KErrNone then no need to delte coe as it is taken care by ConEnv
		{
        // Check whether the configuration supports pointer events.
        // If it dosn't support pointer events then the pointer cursor area will be empty,
        // in this case skip the test and inform the test framework that the test has been skipped
        TRect pointerCursorArea = coe->WsSession().PointerCursorArea();
        if(pointerCursorArea.IsEmpty())
            {
            RDebug::Printf("PointerCursorArea is Empty");
            CMultiPtrPerfTestAppUi* appUi = static_cast<CMultiPtrPerfTestAppUi*>(coe->AppUi());
            appUi->PointerEventsNotSupported();
            }
        else
            {
            // First event which starts the test from HandlePointerEventL
            TRawEvent rawEvent;
            rawEvent.Set(TRawEvent::EButton1Down, 0, 0, 0, 0);
            UserSvr::AddEvent(rawEvent);
            coe->ExecuteD();
            }
		Mutex.Close();
		ConVar.Close();
	 	}
	
	__UHEAP_MARKEND;
	return err;
	}