kerneltest/e32test/realtime/d_latncy.cpp
changeset 9 96e5fb8b040d
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/kerneltest/e32test/realtime/d_latncy.cpp	Thu Dec 17 09:24:54 2009 +0200
@@ -0,0 +1,569 @@
+// Copyright (c) 1999-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\realtime\d_latncy.cpp
+// 
+//
+
+#include "platform.h"
+
+#if defined(__MEIG__)
+#include <cl7211.h>
+#elif defined(__MAWD__)
+#include <windermere.h>
+#elif defined(__MISA__)
+#include <sa1100.h>
+#elif defined(__MCOT__)
+#include <cotulla.h>
+#elif defined(__MI920__) || defined(__NI1136__)
+#include <integratorap.h>
+//#define FREE_RUNNING_MODE			// runs the millisecond timer in free running mode
+#elif defined(__IS_OMAP1610__)
+#include <omap_timer.h>
+#include <omap_plat.h>
+#elif defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+#include <omap_hw.h>
+#include <shared_instrtimer.h>
+#elif defined(__EPOC32__) && defined(__CPU_X86)
+#include <x86.h>
+#include <x86pc.h>
+#elif defined(__RVEMUBOARD__)
+#include <rvemuboard.h>
+#elif defined(__NE1_TB__)
+#include <upd35001_timer.h>
+#endif
+
+#ifdef __CPU_ARM
+#include <arm.h>
+#endif
+
+#include <kernel/kern_priv.h>		//temporary
+#include "d_latncy.h"
+
+_LIT(KLddName,"Latency");
+_LIT(KThreadName,"LatencyThreadK");
+
+#if defined(__MEIG__)
+const TInt KTickPeriodMs=2;
+const TInt KTicksPerMillisecond=512;
+#elif defined(__MAWD__)
+const TInt KTickPeriodMs=1;
+const TInt KTicksPerMillisecond=512;
+#elif defined(__MISA__) || defined(__MCOT__)
+const TInt KTicksPerMillisecond=3686;
+const TInt KOstTicks=3685;	// not quite 1ms, so it goes in and out of phase with ms timer
+TUint TriggerTime;
+#elif defined(__MI920__) || defined(__NI1136__)
+const TInt KTickPeriodMs=1;
+#if defined(__MI920__) || defined(__NI1136__)
+#ifdef FREE_RUNNING_MODE	
+const TInt KTicksPerMillisecond=1500;
+#else
+const TInt KTicksPerMillisecond=24000;
+#endif
+#endif
+#elif defined(__IS_OMAP1610__)
+const TInt KTickPeriodMs=1;
+TInt KTicksPerMillisecond = TOmapPlat::GetInputClk()/32000;
+#elif defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+const TInt KTickPeriodMs=1;					// defined for compatibility but not used (ignored)
+const TInt KTicksPerMillisecond = 12000; 	// Hard coded (12Mhz)
+#elif defined(__EPOC32__) && defined(__CPU_X86)
+const TInt KTickPeriodMs=1;
+const TInt KTicksPerMillisecond=1193;
+#elif defined(__RVEMUBOARD__)
+const TInt KTickPeriodMs=1;
+const TInt KTicksPerMillisecond=1000;
+#elif defined(__NE1_TB__)
+const TInt KTickPeriodMs=1;
+const TInt KTicksPerMillisecond=66667;
+#endif
+
+#ifdef _DEBUG
+const TInt KFudgeFactor=1;
+#else
+const TInt KFudgeFactor=1;
+#endif
+
+class DDeviceLatency : public DLogicalDevice
+	{
+public:
+	DDeviceLatency();
+	virtual TInt Install();
+	virtual void GetCaps(TDes8& aDes) const;
+	virtual TInt Create(DLogicalChannelBase*& aChannel);
+	};
+
+class DLatencyPowerHandler : public DPowerHandler
+	{
+public: // from DPOwerHandler
+	void PowerUp();
+	void PowerDown(TPowerState);
+public:
+	DLatencyPowerHandler(DLatency* aChannel);
+public:
+	DLatency* iChannel;
+	};
+
+
+
+inline TUint DLatency::Ticks()
+	{
+#if defined(__MEIG__)
+	return KTicksPerMillisecond-(*(volatile TUint*)(KEigerTimer2Data16+KEigerBaseAddress)&0xffff);
+#elif defined(__MAWD__)
+	return KTicksPerMillisecond-(*(volatile TUint*)(KWindTimer2Value16+KWindBaseAddress)&0xffff);
+#elif defined(__MISA__) || defined(__MCOT__)
+	return *(volatile TUint*)KHwRwOstOscr-iTriggerTime;
+#elif defined(__MI920__) || defined(__NI1136__)
+	return KTicksPerMillisecond-(*(volatile TUint*)(KHwCounterTimer2+KHoTimerValue)&0xffff);
+#elif defined(__IS_OMAP1610__)
+	return KTicksPerMillisecond - *(volatile TUint*)(KHwBaseOSTimer1Reg+KHoOSTimer_READ_TIM);
+#elif defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+	return (*(volatile TUint*)(iTimerInfo.iAddress + KHoGpTimer_TCRR)) - iTimerLoadValue;
+#elif defined(__X86PC__)
+	return 1194 - __HwTimer();
+#elif defined(__RVEMUBOARD__)
+	return KTicksPerMillisecond-(*(volatile TUint*)(KHwCounterTimer1+KHoTimerValue)&0xffff);
+#elif defined(__NE1_TB__)
+	return NETimer::Timer(0).iTimerCount;	// counts up, reset timer + interrupt on match
+#endif
+	}
+
+#if !defined(__SMP__)
+#if !defined(__EPOC32__) || !defined(__CPU_X86)
+extern TUint IntStackPtr();
+#endif
+#endif
+
+DECLARE_STANDARD_LDD()
+	{
+	return new DDeviceLatency;
+	}
+
+DDeviceLatency::DDeviceLatency()
+//
+// Constructor
+//
+	{
+	//iParseMask=0;
+	//iUnitsMask=0;
+	iVersion=TVersion(1,0,1);
+	}
+
+TInt DDeviceLatency::Install()
+//
+// Install the device driver.
+//
+	{
+	TInt r=SetName(&KLddName);
+	return r;
+	}
+
+void DDeviceLatency::GetCaps(TDes8& aDes) const
+//
+// Return the Comm capabilities.
+//
+	{
+	}
+
+TInt DDeviceLatency::Create(DLogicalChannelBase*& aChannel)
+//
+// Create a channel on the device.
+//
+	{
+	aChannel=new DLatency;
+	return aChannel?KErrNone:KErrNoMemory;
+	}
+
+DLatency::DLatency()
+	:	iMsCallBack(MsCallBack,this),
+		iMsDfc(MsDfc,this,NULL,1)
+//
+// Constructor
+//
+	{
+#if !defined(__SMP__)
+#if !defined(__EPOC32__) || !defined(__CPU_X86)
+	iIntStackTop=(TUint*)IntStackPtr();
+#endif
+#endif
+#if defined(__MISA__) || defined(__MCOT__)
+	iTickIncrement=KOstTicks*KFudgeFactor;
+#endif
+#if defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+	iTimerInfo.iAddress = 0;
+#endif
+	}
+
+DLatency::~DLatency()
+//
+// Destructor
+//
+	{
+	iOff = (TUint8)ETrue;
+	StopTimer();
+	iMsDfc.Cancel();
+
+	if (iRtDfcQ)
+		iRtDfcQ->Destroy();
+
+	if (iPowerHandler)
+		{
+		iPowerHandler->Remove();
+		delete iPowerHandler;
+		}
+
+	Kern::SafeClose((DObject*&)iClient, NULL);
+	}
+
+TInt DLatency::DoCreate(TInt /*aUnit*/, const TDesC8* /*anInfo*/, const TVersion& aVer)
+//
+// Create the channel from the passed info.
+//
+	{
+	if (!Kern::QueryVersionSupported(TVersion(1,0,1),aVer))
+		return KErrNotSupported;
+
+	// create the power handler
+	iPowerHandler = new DLatencyPowerHandler(this);
+	if (!iPowerHandler)
+		return KErrNoMemory;
+	iPowerHandler->Add();
+
+	// Allocate a kernel thread to run the DFC 
+	TInt r = Kern::DynamicDfcQCreate(iRtDfcQ, KNumPriorities-1,KThreadName);
+
+	if (r != KErrNone)
+		return r;
+	
+#ifdef CPU_AFFINITY_ANY
+	NKern::ThreadSetCpuAffinity((NThread*)(iRtDfcQ->iThread), KCpuAffinityAny);			
+#endif
+
+	iMsDfc.SetDfcQ(iRtDfcQ);
+	iClient=&Kern::CurrentThread();
+	iClient->Open();
+	Kern::SetThreadPriority(KNumPriorities-2);
+	return KErrNone;
+	}
+
+#if defined(__MISA__) 
+// For SA1100/SA1110 use a separate timer on a FIQ interrupt (OST match 0)
+TInt DLatency::StartTimer()
+	{
+	TInt r=Interrupt::Bind(KIntIdOstMatchGeneral,MsCallBack,this);
+	if (r==KErrNone)
+		{
+		TSa1100::ModifyIntLevels(0,KHtIntsOstMatchGeneral);	// route new timer interrupt to FIQ
+		TSa1100::SetOstMatchEOI(KHwOstMatchGeneral);
+		TUint oscr=TSa1100::OstData();
+		iTriggerTime=oscr+KOstTicks*KFudgeFactor;
+		TSa1100::SetOstMatch(KHwOstMatchGeneral,iTriggerTime);
+		TSa1100::EnableOstInterrupt(KHwOstMatchGeneral);
+		Interrupt::Enable(KIntIdOstMatchGeneral);
+		}
+	return r;
+	}
+#elif defined(__MCOT__)
+// For Cotulla use a separate timer on a FIQ interrupt (OST match 0)
+TInt DLatency::StartTimer()
+	{
+	TInt r=Interrupt::Bind(KIntIdOstMatchGeneral,MsCallBack,this);
+	if (r==KErrNone)
+		{
+		TCotulla::ModifyIntLevels(0,KHtIntsOstMatchGeneral);	// route new timer interrupt to FIQ
+		TCotulla::SetOstMatchEOI(KHwOstMatchGeneral);
+		TUint oscr=TCotulla::OstData();
+		iTriggerTime=oscr+KOstTicks*KFudgeFactor;
+		TCotulla::SetOstMatch(iTriggerTime,KHwOstMatchGeneral);
+		TCotulla::EnableOstInterrupt(KHwOstMatchGeneral);
+		Interrupt::Enable(KIntIdOstMatchGeneral);
+		}
+	return r;
+	}
+#elif defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+TInt DLatency::StartTimer()
+/* 
+ *  For OMAP2420 initialise a new timer to generate an interrupt every 1ms
+ */
+	{
+	__ASSERT_ALWAYS(!iTimerInfo.iAddress, Kern::Fault("D_Latncy: timer allocated twice.",
+													 iTimerInfo.iAddress));
+
+	// Get an available Timer from the system
+    TInt r = OmapTimerMgr::GetTimer(iGPTimerId, iTimerInfo);
+    if (KErrNone != r)
+    	{
+    	return r;
+    	}
+    	
+    // Configure the timer
+    r = ConfigureTimer();
+    if (KErrNone != r)
+    	{
+    	DisableTimer();
+    	return r;
+    	}
+    
+    // Bind to timer interrupt
+    r = Interrupt::Bind(iTimerInfo.iInterruptId, MsCallBack, this);
+    if (KErrNone != r)
+        {
+        DisableTimer();
+        return r;
+        }
+              
+    // Unmask timer IT in interrupt controller
+    r = Interrupt::Enable(iTimerInfo.iInterruptId);
+    if (KErrNone != r)
+		{
+		Interrupt::Unbind(iTimerInfo.iInterruptId);
+		DisableTimer();
+    	return r;
+    	}
+    
+    // Start timer
+    TOmap::ModifyRegister32(iTimerInfo.iAddress + KHoGpTimer_TCLR, KClear32,
+                        KHtGpTimer_TCLR_St);
+        
+    return KErrNone;
+	}
+
+void DLatency::DisableTimer()
+/*
+ *	Disable the interface and functional clock and mark the timer as available 
+ */
+	{
+	  // Stop timer
+    TOmap::ModifyRegister32(iTimerInfo.iAddress + KHoGpTimer_TCLR,
+                        KHtGpTimer_TCLR_St, KClear32);
+
+#if defined(__WAKEUP_3430__)
+    // Disable Timer clocks using Timer framework instead of using TPRcm direct calls for 3430
+    TInt r = OmapTimerMgr::DisableClocks(iGPTimerId);
+    if (r != KErrNone)
+        __ASSERT_ALWAYS(r, Kern::Fault("Timer clocks disable failed", 0)) ;
+#else
+	// Disable timer interface clock in PRCM
+	TPrcm::InterfaceClkCtrl(iTimerInfo.iPrcmDeviceId, EFalse);
+	
+	// Disable timer functional clock in PRCM
+	TPrcm::FunctionalClkCtrl(iTimerInfo.iPrcmDeviceId, EFalse);
+#endif
+
+	// Release the timer
+	OmapTimerMgr::ReleaseTimer(iGPTimerId);
+	
+	iTimerInfo.iAddress = 0;
+	}
+
+
+TInt DLatency::ConfigureTimer()
+/*
+ *	This method will configure a timer to:
+ *		-	run at the system clock (12Mhz)
+ *		-	no prescaler (disable TCLR[PRE])
+ *		-   autoreload and overflow interrupt enabled (TLDR will contain a
+ *			value to generate an interrupt every 1000microsec)
+ */
+	{
+
+#if defined(__WAKEUP_3430__)
+	// Enable Timer clocks using timer framework instead of TPrcm direct calls for 3430
+    TInt r = OmapTimerMgr::EnableClocks(iGPTimerId);
+    if (r != KErrNone)
+        __ASSERT_ALWAYS(r, Kern::Fault("Timer Clocks enable failed", 0)) ;
+
+	// Select the input clock to be system clock 
+    r = OmapTimerMgr::SetTimerClkSrc(iGPTimerId, ESysClk);
+#else
+	// Enable timer interface clock in PRCM  
+	TPrcm::InterfaceClkCtrl(iTimerInfo.iPrcmDeviceId, ETrue, ETrue);
+	// Enable timer functional clock in PRCM
+	TPrcm::FunctionalClkCtrl(iTimerInfo.iPrcmDeviceId, ETrue, ETrue);
+
+	// Select the input clock to be system clock 
+    TInt r = OmapTimerMgr::SetTimerClkSrc(iGPTimerId, ESysClk);
+#endif
+
+    if (KErrNone != r)	
+    	return r;
+
+    // Timer OCP configuration: - software reset
+    TOmap::SetRegister32( iTimerInfo.iAddress + KHoGpTimerTIOCP_CFG,
+                          KHtGpTimer_TIOCP_CFG_SoftReset);
+
+    // Wait for reset to be complete
+    TUint16 timeOut = 1000;
+    while ( !(TOmap::Register32(iTimerInfo.iAddress + KHoGpTimer_TISTAT) & 
+    			KHtGpTimer_TISTAT_ResetComplete)
+    			&& --timeOut);
+    
+   // Check if the timer has been reset or we hit the timeout
+   __ASSERT_ALWAYS((TOmap::Register32(iTimerInfo.iAddress + KHoGpTimer_TISTAT) & 
+    			KHtGpTimer_TISTAT_ResetComplete), Kern::Fault("D_Latncy: failed to reset timer.",
+													 iGPTimerId));
+	
+    // Set PRE to be 0, PTV value is ignored, AutoReload is enabled
+    TOmap::SetRegister32(iTimerInfo.iAddress + KHoGpTimer_TCLR, KHtGpTimer_TCLR_AR );
+
+	//PTV argument is 0 because of TCLR[PRE] = 0 (prescaling disabled)
+	TInt timerPTV = 0;
+	
+	// Calculate clock frequence from the ticks per ms
+	TInt timerClkSrcFreq = KTicksPerMillisecond * 1000;
+    
+    iTimerLoadValue = OmapTimerMgr::TimerLoadValue(/*microsecs*/1000, timerClkSrcFreq, timerPTV);                          
+
+	// First, load value in TCRR and TLDR registers
+    TOmap::SetRegister32(iTimerInfo.iAddress + KHoGpTimer_TCRR, iTimerLoadValue);
+    TOmap::SetRegister32(iTimerInfo.iAddress + KHoGpTimer_TLDR, iTimerLoadValue);
+
+    // Enable overflow interrupt
+    TOmap::SetRegister32(iTimerInfo.iAddress + KHoGpTimer_TIER,
+                         KHtGpTimer_TIER_OverFlow);
+
+    return KErrNone;
+	}
+#else
+TInt DLatency::StartTimer()
+	{
+	iMsCallBack.OneShot(KTickPeriodMs*KFudgeFactor);
+	return KErrNone;
+	}
+#endif
+
+#if defined(__MISA__) 
+// For SA1100/SA1110 use a separate timer on a FIQ interrupt (OST match 0)
+void DLatency::StopTimer()
+	{
+	TSa1100::ModifyIntLevels(KHtIntsOstMatchGeneral,0);
+	TSa1100::DisableOstInterrupt(KHwOstMatchGeneral);
+	Interrupt::Disable(KIntIdOstMatchGeneral);
+	Interrupt::Unbind(KIntIdOstMatchGeneral);
+	TSa1100::SetOstMatchEOI(KHwOstMatchGeneral);
+	}
+#elif defined(__MCOT__)
+// For Cotulla use a separate timer on a FIQ interrupt (OST match 0)
+void DLatency::StopTimer()
+	{
+	TCotulla::ModifyIntLevels(KHtIntsOstMatchGeneral,0);
+	TCotulla::DisableOstInterrupt(KHwOstMatchGeneral);
+	Interrupt::Disable(KIntIdOstMatchGeneral);
+	Interrupt::Unbind(KIntIdOstMatchGeneral);
+	TCotulla::SetOstMatchEOI(KHwOstMatchGeneral);
+	}
+#elif defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+void DLatency::StopTimer()
+	{
+	Interrupt::Disable(iTimerInfo.iInterruptId);
+	Interrupt::Unbind(iTimerInfo.iInterruptId);
+	DisableTimer();
+	}
+#else
+void DLatency::StopTimer()
+	{
+	iMsCallBack.Cancel();
+	}
+#endif
+
+TInt DLatency::Request(TInt aFunction, TAny* a1, TAny* a2)
+//
+// Client requests
+//
+	{
+	// Kern::Printf("DLatency::Request() 0x%x)\n", aFunction);
+	TInt r=KErrNone;
+	switch (aFunction)
+		{
+		case RLatency::EControlStart:
+			iStarted = (TUint8)ETrue;
+			StartTimer();
+			break;
+		case RLatency::EControlTicksPerMs:
+			r=KTicksPerMillisecond;
+			break;
+		case RLatency::EControlGetResults:
+			iResults.iUserThreadTicks = Ticks();
+			kumemput32(a1, &iResults, sizeof(SLatencyResults));
+			break;
+		default:
+			r = KErrNotSupported;
+			break;
+		}
+	return(r);
+	}
+
+#ifdef __CAPTURE_EXTRAS
+extern void CaptureExtras(SLatencyResults&);
+#endif
+
+#if !defined(__MISA__) && !defined(__MCOT__)
+void DLatency::MsCallBack(TAny* aPtr)
+	{
+	DLatency* pL = (DLatency*)aPtr;
+#if defined(__IS_OMAP2420__) || defined(__WAKEUP_3430__)
+	pL->iResults.iIntTicks = pL->Ticks();
+	TOmap::SetRegister32(pL->iTimerInfo.iAddress + KHoGpTimer_TISR, KHtGpTimer_TISR_OverFlow);
+#else
+	pL->iResults.iIntTicks = Ticks();
+#endif
+#ifdef __CAPTURE_EXTRAS
+	CaptureExtras(pL->iResults);
+#endif
+#if defined(__EPOC32__) && defined(__CPU_X86)
+	pL->iResults.iIntRetAddr = X86::IrqReturnAddress();
+#elif defined(__CPU_ARM) && defined(__SMP__)
+	pL->iResults.iIntRetAddr = Arm::IrqReturnAddress();
+#else
+	pL->iResults.iIntRetAddr=(pL->iIntStackTop)[-1];
+#endif
+	if (!pL->iOff)
+		{
+		pL->iMsCallBack.Again(KTickPeriodMs*KFudgeFactor);
+		pL->iMsDfc.Add();
+		}
+	}
+#endif
+
+void DLatency::MsDfc(TAny* aPtr)
+	{
+	DLatency* pL = (DLatency*)aPtr;
+	pL->iResults.iKernThreadTicks=pL->Ticks();
+	NKern::ThreadRequestSignal(&pL->iClient->iNThread);
+	}
+
+DLatencyPowerHandler::DLatencyPowerHandler(DLatency* aChannel)
+	:	DPowerHandler(KLddName), 
+		iChannel(aChannel)
+	{
+	}
+
+void DLatencyPowerHandler::PowerUp()
+	{
+	iChannel->iOff = (TUint8)EFalse;
+	if (iChannel->iStarted)
+		iChannel->StartTimer();
+	PowerUpDone();
+	}
+
+void DLatencyPowerHandler::PowerDown(TPowerState)
+	{
+	iChannel->iOff = (TUint8)ETrue;
+	iChannel->StopTimer();
+	PowerDownDone();
+	}
+
+
+