kerneltest/e32test/benchmark/bm_ldd.cpp
author hgs
Wed, 12 May 2010 10:34:10 +0100
changeset 133 2a0ada0a1bf8
parent 0 a41df078684a
child 148 31ea0f8e3c99
permissions -rw-r--r--
201019_04

// Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of the License "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
//

#include "k32bm.h"

const TUint8 KMutexOrder = 0xf0;

class DBMLDevice : public DLogicalDevice
	{
public:
	DBMLDevice();
	virtual TInt Install();
	virtual void GetCaps(TDes8& aDes) const;
	virtual TInt Create(DLogicalChannelBase*& aChannel);
	};

class DBMLChannel : public DLogicalChannelBase, public MBMIsr, public MBMInterruptLatencyIsr
	{
public:
	DBMLChannel();
	~DBMLChannel();

	virtual TInt DoCreate(TInt aUnit, const TDesC8* anInfo, const TVersion& aVer);
	virtual TInt Request(TInt aFunction, TAny* a1, TAny* a2);

	DBMPChannel* PChannel() { return (DBMPChannel*) iPdd; }

private:
	static const TInt	KBMDfcQThreadPriority;	
	static const TInt	KBMKernelThreadPriority;	

	static void Dfc(TAny*);

	virtual void Isr(TBMTicks now);

	TInt (DBMLChannel::*iRequestInterrupt)();	// Measurement specific RBMChannel::RequestInterrupt() implmentation 
	TInt RequestInterrupt();					// Default iRequestInterrupt() implementation

	TBMTicks (DBMLChannel::*iResult)();			// Measurement specific RBMChannel::Result() implmentation
	TBMTicks Result();							// Default iResult() implementation

	TInt Start(RBMChannel::TMode);

	TInt StartInterruptLatency();
	virtual void InterruptLatencyIsr(TBMTicks latency);

	TInt StartKernelPreemptionLatency();
	static TInt KernelPreemptionLatencyThreadEntry(TAny* ptr);
	void KernelPreemptionLatencyThread();

	TInt StartUserPreemptionLatency();
	TBMTicks UserPreemptionLatencyResult();		// iResult() implementation

	TInt StartNTimerJitter();
	TInt RequestNTimerJitterInterrupt();		// iRequestInterrupt() implementation
	static void NTimerJitterCallBack(TAny*);

	TInt StartTimerStampOverhead();
	TInt RequestTimerStampOverhead();			// iRequestInterrupt() implementation

	TInt SetAbsPrioirty(TInt aThreadHandle, TInt aNewPrio, TInt* aOldPrio);

	DMutex*			iLock;	// Shall be acquired by anyone who access the object's writable state.

	TBool			iStarted;					// ETrue when a particular sequence of measurements has been started
	TBool			iPendingInterruptRequest;	// ETrue when an interrupt has been requested

	TDynamicDfcQue*	iDfcQ;
	TDfc			iDfc;

	DThread*		iKernelThread;		// the kernel thread created by some benchmarks
	DThread*		iUserThread;		// the user-side thread
	DThread*		iInterruptThread;	// the thread signaled by DFC; if non-NULL either iKernelThread or iUserThread

	NTimer			iNTimer;			// the timer used in "NTimer jitter" benchmark
	TBMTicks		iOneNTimerTick;		// number of high-resolution timer ticks in one NKern tick.
	TInt			iNTimerShotCount;	// used in "NTimer jitter" to distinguish between the first and the second shots 

	TBMTicks		iTime;
	TBMTicks		iTimerPeriod;		// period of high-resolution timer in ticks

	NFastSemaphore*	iKernelThreadExitSemaphore;

	void Lock()
		{
		NKern::ThreadEnterCS();
		Kern::MutexWait(*iLock);
		}
	void Unlock()
		{
		Kern::MutexSignal(*iLock);
		NKern::ThreadLeaveCS();
		}
	
	TBMTicks Delta(TBMTicks aT0, TBMTicks aT1)
		{
		return (aT0 <= aT1) ? (aT1 - aT0) : iTimerPeriod - (aT0 - aT1);
		}
	};

_LIT(KBMLChannelLit, "BMLChannel");

const TInt	DBMLChannel::KBMDfcQThreadPriority = KBMLDDHighPriority;
const TInt	DBMLChannel::KBMKernelThreadPriority = KBMLDDMidPriority;



DECLARE_STANDARD_LDD()
//
// Create a new device
//
	{
	__ASSERT_CRITICAL;
	return new DBMLDevice;
	}

DBMLDevice::DBMLDevice()
//
// Constructor
//
	{
	//iUnitsMask=0;
	iVersion = TVersion(1,0,1);
	iParseMask = KDeviceAllowPhysicalDevice;
	}

TInt DBMLDevice::Install()
//
// Install the device driver.
//
	{
	TInt r = SetName(&KBMLdName);
	return r;
	}

void DBMLDevice::GetCaps(TDes8&) const
//
// Return the Comm capabilities.
//
	{
	}

TInt DBMLDevice::Create(DLogicalChannelBase*& aChannel)
//
// Create a channel on the device.
//
	{
	__ASSERT_CRITICAL;
	aChannel = new DBMLChannel;
	return aChannel ? KErrNone : KErrNoMemory;
	}

DBMLChannel::DBMLChannel() : 
		iDfc(0, this, 0, 0),
		iNTimer(NULL, this)
	{
	// iDfcQueue = NULL;
	// iStarted = EFalse;
	// iPendingInterruptRequest = EFalse;
	// iKernelThread = NULL;
	}

TInt DBMLChannel::DoCreate(TInt /*aUnit*/, const TDesC8* /* aInfo*/ , const TVersion& aVer)
//
// Create the channel from the passed info.
//
	{
	__ASSERT_CRITICAL;
	if (!Kern::QueryVersionSupported(TVersion(1,0,1),aVer))
		return KErrNotSupported;
	TInt r = Kern::MutexCreate(iLock, KBMLChannelLit, KMutexOrder);
	if (r != KErrNone)
		{
		return r;
		}
	iTimerPeriod = PChannel()->TimerPeriod();
		// Calculate the number of high-resolution timer ticks in one NKern tick
		// deviding the number of high-resolution timer ticks in one second by the
		// number of NKern ticks in one second.
		//
	iOneNTimerTick = PChannel()->TimerNsToTicks(BMSecondsToNs(1))/NKern::TimerTicks(1000);
	return KErrNone;
	}

DBMLChannel::~DBMLChannel()
// Called on a channel close. Note that if the PDD channel create failed 
// then the DoCreate() call will not have been made so don't assume anything 
// about non-ctor initialisation of members.
	{
	if (iLock) 
		iLock->Close(0);

	if (iPendingInterruptRequest)
		{
		PChannel()->CancelInterrupt();
		iDfc.Cancel();
		}

	if (iDfcQ)
		{
		iDfcQ->Destroy();
		}

	if (iKernelThread)
		{
		NFastSemaphore exitSemaphore;
		exitSemaphore.iOwningThread = NKern::CurrentThread();
		iKernelThreadExitSemaphore = &exitSemaphore;
		NKern::ThreadRequestSignal(&iKernelThread->iNThread);
		NKern::FSWait(&exitSemaphore);
		}
	}

void DBMLChannel::Dfc(TAny* ptr)
	{	
	DBMLChannel* lCh = (DBMLChannel*) ptr;
	BM_ASSERT(lCh->iPendingInterruptRequest);
	BM_ASSERT(lCh->iInterruptThread);
	NKern::ThreadRequestSignal(&lCh->iInterruptThread->iNThread);
	lCh->iPendingInterruptRequest = EFalse;
	}

//
// Default DBMLChannel::iRequestInterrupt implementation
//
TInt DBMLChannel::RequestInterrupt()
	{
	if (!iStarted)
		{
		return KErrNotReady;
		}
	if (iPendingInterruptRequest) 
		{
		return KErrInUse;
		}
	iPendingInterruptRequest = ETrue;
	PChannel()->RequestInterrupt();
	return KErrNone;
	}

//
// Default DBMLChannel::iResult implementation
//
TBMTicks DBMLChannel::Result()
	{
	return iTime;
	}

void DBMLChannel::Isr(TBMTicks aNow)
	{
	//
	// Store the ISR entry time and queue a DFC.
	//
	iTime = aNow;
	iDfc.Add();
	}

//
// "INTERRUPT LATENCY"
// 
// SCENARIO:
//
//		A user thread requests an interrupt (RBMChannel::RequestInterrupt()) and waits at User::WaitForAnyRequest()
//		(RBMChannel::Result()).
//		When the interrupt occurs DBMLChannel::InterruptLatencyIsr() stores the interrupt latency 
//		provided by LDD, in DBMLChannel::iTime and queues a DFC (DBMLChannel::iDfc, DBMLChannel::Dfc())
//		which in its turn signals the user thread.
//

TInt DBMLChannel::StartInterruptLatency()
	{
	if (iStarted)
		{
		return KErrInUse;
		}
	TInt r = PChannel()->BindInterrupt((MBMInterruptLatencyIsr*) this);
	if (r != KErrNone)
		{
		return r;
		}
		// Use the default iRequestInterrupt() implmentation
	iRequestInterrupt = &DBMLChannel::RequestInterrupt;
		// Use the default iResult() implmentation
	iResult = &DBMLChannel::Result;
	iInterruptThread = &Kern::CurrentThread();
	iStarted = ETrue;
	return KErrNone;
	}

void DBMLChannel::InterruptLatencyIsr(TBMTicks aLatency)
	{
	iTime = aLatency;
	iDfc.Add();
	}

//
// "KERNEL THREAD PREEMPTION LATENCY"
// 
// SCENARIO:
//
//		A new kernel thread is created at the beginning of a sequence of measurements 
//		(DBMLChannel::StartKernelPreemptionLatency()). The kernel thread waits at Kern::WaitForAnyRequest()
//		(DBMLChannel::KernelPreemptionLatencyThread()).
//		The user thread requests an interrupt (RBMChannel::RequestInterrupt()) and waits at User::WaitForAnyRequest()
//		(RBMChannel::Result()).
//		When the interrupt occurs DBMLChannel::Isr() stores the ISR entry time, provided by LDD 
//		in DBMLChannel::iTime and queues a DFC (DBMLChannel::iDfc, DBMLChannel::Dfc()) which, in its turn,
//		signals the kernel thread.
//		The kernel thread, when awaken, calculates the latency as the difference between the ISR entry time 
//		and the current time and finally signals the user thread.
//

TInt DBMLChannel::StartKernelPreemptionLatency()
	{
	if (iStarted)
		{
		return KErrInUse;
		}
	TInt r = PChannel()->BindInterrupt((MBMIsr*) this);
	if (r != KErrNone)
		{
		return r;
		}
	{
	SThreadCreateInfo info;
	info.iType = EThreadSupervisor;
	info.iFunction = DBMLChannel::KernelPreemptionLatencyThreadEntry;
	info.iPtr = this;
	info.iSupervisorStack = NULL;
	info.iSupervisorStackSize = 0;
	info.iInitialThreadPriority = KBMKernelThreadPriority;
	info.iName.Set(KBMLChannelLit);
	info.iTotalSize = sizeof(info);
	r = Kern::ThreadCreate(info);
	if (r != KErrNone)
		{
		return r;
		}
	iKernelThread = (DThread*) info.iHandle;
	}

	iUserThread = &Kern::CurrentThread();
		// Use the default iRequestInterrupt() implmentation
	iRequestInterrupt = &DBMLChannel::RequestInterrupt;
		// Use the default iResult() implmentation
	iResult = &DBMLChannel::Result;
	iInterruptThread = iKernelThread;
	iStarted = ETrue;

	Kern::ThreadResume(*iKernelThread);

	return KErrNone;
	}

TInt DBMLChannel::KernelPreemptionLatencyThreadEntry(TAny* ptr)
	{
	DBMLChannel* lCh = (DBMLChannel*) ptr;
	lCh->KernelPreemptionLatencyThread();
	BM_ASSERT(0);
	return 0;
	}

void DBMLChannel::KernelPreemptionLatencyThread()
	{
	for(;;)
		{
		NKern::WaitForAnyRequest();

		if(iKernelThreadExitSemaphore)
			break;

		TBMTicks now = PChannel()->TimerStamp();
		iTime = Delta(iTime, now);
		BM_ASSERT(iUserThread);
		NKern::ThreadRequestSignal(&iUserThread->iNThread);
		}

	NKern::FSSignal(iKernelThreadExitSemaphore);
	Kern::Exit(0); 
	}


//
// "USER THREAD PREEMPTION LATENCY"
// 
// SCENARIO:
//
//		A user thread requests an interrupt (RBMChannel::RequestInterrupt()) and waits at User::WaitForAnyRequest()
//		(RBMChannel::Result()).
//		When the interrupt occurs DBMLChannel::Isr() stores the ISR entry time provided by LDD, 
//		in DBMLChannel::iTime and queues a DFC (DBMLChannel::iDfc, DBMLChannel::Dfc()) which in its turn
//		signals the user thread.
//		The user thread, when awaken, immediately re-enters in the LDD, and calculates the latency as 
//		the difference between the ISR entry time and the current time.
//

TInt DBMLChannel::StartUserPreemptionLatency()
	{
	if (iStarted)
		{
		return KErrInUse;
		}
	TInt r = PChannel()->BindInterrupt((MBMIsr*) this);
	if (r != KErrNone)
		{
		return r;
		}
		// Default iRequestInterrupt() implmentation
	iRequestInterrupt = &DBMLChannel::RequestInterrupt;
	iResult = &DBMLChannel::UserPreemptionLatencyResult;
	iInterruptThread = &Kern::CurrentThread();
	iStarted = ETrue;
	return KErrNone;
	}

TBMTicks DBMLChannel::UserPreemptionLatencyResult()
	{
	TBMTicks now = PChannel()->TimerStamp();
	return Delta(iTime, now);
	}

//
// "NTimer PERIOD JITTER"
// 
// SCENARIO:
//
//		One measuremnt is done by two consecutive NTimer callbacks. 
//		The first callback stores the current time and the second one calculate the actual period as 
//		the difference between its own current time and the time stored by the first callback.
//		The difference between this actual period and the theoretical period is considered as the jitter.
//

TInt DBMLChannel::StartNTimerJitter()
	{
	if (iStarted)
		{
		return KErrInUse;
		}
	new (&iNTimer) NTimer(&NTimerJitterCallBack, this);
	iRequestInterrupt = &DBMLChannel::RequestNTimerJitterInterrupt;
		// Use the default iResult() implmentation
	iResult = &DBMLChannel::Result;
	iInterruptThread = &Kern::CurrentThread();
	iStarted = ETrue;
	return KErrNone;
	}

TInt DBMLChannel::RequestNTimerJitterInterrupt()
	{
	if (!iStarted)
		{
		return KErrNotReady;
		}
	if (iPendingInterruptRequest) 
		{
		return KErrInUse;
		}
	iPendingInterruptRequest = ETrue;
	iNTimerShotCount = 0;
	iNTimer.OneShot(1);
	return KErrNone;
	}


void DBMLChannel::NTimerJitterCallBack(TAny* ptr)
	{
	DBMLChannel* lCh = (DBMLChannel*) ptr;
	TBMTicks now = lCh->PChannel()->TimerStamp();
	if (lCh->iNTimerShotCount++ == 0)
		{
		//
		// This is the first callback: store the time and request another one.
		//
		lCh->iTime = now;
		lCh->iNTimer.Again(1);
		}
	else 
		{
		//
		// This is the second callback: measure the jitter and schedule a DFC
		// which in its turn will signal the user thread.
		//
		lCh->iTime = lCh->Delta(lCh->iTime, now);
		lCh->iDfc.Add();
		}
	}

//
// "TIMER OVERHEAD"
// 
// SCENARIO:
//		To measure the overhead of the high-precision timer read operation we get
//		two consecutive timestamps through DBMPChannel::TimerStamp() interface.
//		The difference beween this two values is considered as the measured overhead.
//

TInt DBMLChannel::StartTimerStampOverhead()
	{
	if (iStarted)
		{
		return KErrInUse;
		}
	iRequestInterrupt = &DBMLChannel::RequestTimerStampOverhead;
		// Use the default iResult() implmentation
	iResult = &DBMLChannel::Result;
	iInterruptThread = &Kern::CurrentThread();
	iStarted = ETrue;
	return KErrNone;
	}

TInt DBMLChannel::RequestTimerStampOverhead()
	{
	TBMTicks t1 = PChannel()->TimerStamp();
	TBMTicks t2 = PChannel()->TimerStamp();
	iTime = Delta(t1, t2);
	NKern::ThreadRequestSignal(&iInterruptThread->iNThread);
	return KErrNone;
	}
//
// END OF "GETTING TIMER OVERHEAD"
//

//
// The implmentation of RBMDriver::SetAbsPrioirty() call.
//
TInt DBMLChannel::SetAbsPrioirty(TInt aThreadHandle, TInt aNewPrio, TInt* aOldPrio)
	{
	NKern::LockSystem();
	//
	// Under the system lock find the DThread object and increment its ref-count (i.e Open()) 
	//
	DThread* thr = (DThread*) Kern::ObjectFromHandle(&Kern::CurrentThread(), aThreadHandle, EThread);
	TInt r;
	if (!thr)
		{
		r = EBadHandle;
		}
	else
		{
		r = thr->Open();
		}
	//
	// Now it's safe to release the system lock and to work with the object.
	//
	NKern::ThreadEnterCS();
	NKern::UnlockSystem();
	if (r != KErrNone)
		{
		NKern::ThreadLeaveCS();
		return r;
		}
	*aOldPrio = thr->iDefaultPriority;
	Kern::SetThreadPriority(aNewPrio, thr);
	//
	// Work is done - close the object.
	//	
	thr->Close(NULL);
	NKern::ThreadLeaveCS();
	return KErrNone;
	}

_LIT(KBmDfcQName, "BmDfcQ");

//
// Starts a new sequence of measurements.
//
// Only one sequence can be started for any particular DBMLChannel object during its life. 
// If more than one sequence is required a new DBMLChannel object must be created.
//
TInt DBMLChannel::Start(RBMChannel::TMode aMode)
	{
	TInt r;
	if (iDfcQ == NULL)
		{
		r = Kern::DynamicDfcQCreate(iDfcQ, KBMDfcQThreadPriority, KBmDfcQName);
		if (r != KErrNone)
			return r;

		iDfc.SetDfcQ(iDfcQ);
		iDfc.SetFunction(Dfc);
		}

	switch (aMode)
		{
	case RBMChannel::EInterruptLatency:
		r = StartInterruptLatency();
		break;
	case RBMChannel::EKernelPreemptionLatency:
		r = StartKernelPreemptionLatency();
		break;
	case RBMChannel::EUserPreemptionLatency:
		r = StartUserPreemptionLatency();
		break;
	case RBMChannel::ENTimerJitter:
		r = StartNTimerJitter();
		break;
	case RBMChannel::ETimerStampOverhead:
		r = StartTimerStampOverhead();
		break;
	default:
		r = KErrNotSupported;
		break;
		}

	return r;
	}

//
// Client requests.
//
TInt DBMLChannel::Request(TInt aFunction, TAny* a1, TAny* a2)
	{
	TInt r = KErrNone;
	switch (aFunction)
		{
		case RBMChannel::EStart:
			{
			RBMChannel::TMode mode = (RBMChannel::TMode) (TInt) a1;
			Lock();
			r = Start(mode);
			Unlock();
			break;
			}
		case RBMChannel::ERequestInterrupt:
			{
			Lock();
			r = (this->*iRequestInterrupt)();
			Unlock();
			break;
			}
		case RBMChannel::EResult:
			{
			//
			// We don't acquire the lock because:
			//	(1) iResult() typically reads iTime which was written BEFORE to signal the current thread 
			//      and therefore BEFORE the current thread comes here. 
			//  (2) we really want if possible (i.e. correct!) to avoid the lock acquisition because it can
			//		increase the measurement overhead in the case when we are in a measured path (e.g. user
			//		preemption latency benchmark).
			//
			TBMTicks ticks = (this->*iResult)();
			umemput(a1, &ticks, sizeof(ticks));
			break;
			}
		//
		// All below requests do not access writable DBMChannel state and therefore do not require the lock
		//
		case RBMChannel::ETimerStamp:
			{
			TBMTicks ticks = PChannel()->TimerStamp();
			umemput(a1, &ticks, sizeof(ticks));
			break;
			}
		case RBMChannel::ETimerPeriod:
			{
			TBMTicks ticks = iTimerPeriod;
			umemput(a1, &ticks, sizeof(ticks));
			break;
			}
		case RBMChannel::ETimerTicksToNs:
			{
			TBMTicks ticks;
			umemget(&ticks, a1, sizeof(ticks));
			TBMNs ns = PChannel()->TimerTicksToNs(ticks);
			umemput(a2, &ns, sizeof(ns));
			break;
			}
		case RBMChannel::ETimerNsToTicks:
			{
			TBMNs ns;
			umemget(&ns, a1, sizeof(ns));
			TBMTicks ticks = PChannel()->TimerNsToTicks(ns);
			umemput(a2, &ticks, sizeof(ticks));
			break;
			}
		case RBMChannel::ESetAbsPriority:
			{
			TInt newPrio;
			TInt oldPrio;
			umemget(&newPrio, a2, sizeof(newPrio));
			r = SetAbsPrioirty((TInt) a1, newPrio, &oldPrio);
			umemput(a2, &oldPrio, sizeof(oldPrio));
			break;
			}
		default:
			r = KErrNotSupported;
			break;
		}
	return r;
	}