Update contrib.
1 // Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of the License "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // e32\nkernsmp\arm\ncutils.cpp
25 extern SVariantInterfaceBlock* VIB;
28 /******************************************************************************
30 ******************************************************************************/
31 /** Create a spin lock
36 EXPORT_C TSpinLock::TSpinLock(TUint aOrder)
39 __NK_ASSERT_DEBUG( (aOrder==EOrderNone) || ((aOrder&0x7f)<0x20) );
40 if (aOrder>=0x80 && aOrder!=EOrderNone)
43 iLock = TUint64(aOrder)<<48; // byte 6 = 00-1F for interrupt, 20-3F for preemption
44 // byte 7 = FF if not held
48 /******************************************************************************
49 * Read/Write Spin lock
50 ******************************************************************************/
51 /** Create a spin lock
56 EXPORT_C TRWSpinLock::TRWSpinLock(TUint aOrder)
59 __NK_ASSERT_DEBUG( (aOrder==TSpinLock::EOrderNone) || ((aOrder&0x7f)<0x20) );
60 if (aOrder>=0x80 && aOrder!=TSpinLock::EOrderNone)
63 iLock = TUint64(aOrder)<<48; // byte 6 = 00-1F for interrupt, 20-3F for preemption
64 // byte 7 = FF if not held
70 void FastMutexNestAttempt()
75 void FastMutexSignalError()
81 void NKern::Init0(TAny* a)
83 __KTRACE_OPT(KBOOT,DEBUGPRINT("VIB=%08x", a));
84 VIB = (SVariantInterfaceBlock*)a;
85 __NK_ASSERT_ALWAYS(VIB && VIB->iVer==0 && VIB->iSize==sizeof(SVariantInterfaceBlock));
86 __KTRACE_OPT(KBOOT,DEBUGPRINT("iVer=%d iSize=%d", VIB->iVer, VIB->iSize));
87 __KTRACE_OPT(KBOOT,DEBUGPRINT("iMaxCpuClock=%08x %08x", I64HIGH(VIB->iMaxCpuClock), I64LOW(VIB->iMaxCpuClock)));
88 __KTRACE_OPT(KBOOT,DEBUGPRINT("iMaxTimerClock=%u", VIB->iMaxTimerClock));
89 __KTRACE_OPT(KBOOT,DEBUGPRINT("iScuAddr=%08x", VIB->iScuAddr));
90 __KTRACE_OPT(KBOOT,DEBUGPRINT("iGicDistAddr=%08x", VIB->iGicDistAddr));
91 __KTRACE_OPT(KBOOT,DEBUGPRINT("iGicCpuIfcAddr=%08x", VIB->iGicCpuIfcAddr));
92 __KTRACE_OPT(KBOOT,DEBUGPRINT("iLocalTimerAddr=%08x", VIB->iLocalTimerAddr));
94 TScheduler& s = TheScheduler;
95 s.i_ScuAddr = (TAny*)VIB->iScuAddr;
96 s.i_GicDistAddr = (TAny*)VIB->iGicDistAddr;
97 s.i_GicCpuIfcAddr = (TAny*)VIB->iGicCpuIfcAddr;
98 s.i_LocalTimerAddr = (TAny*)VIB->iLocalTimerAddr;
99 s.i_TimerMax = (TAny*)(VIB->iMaxTimerClock / 1); // use prescaler value of 1
102 for (i=0; i<KMaxCpus; ++i)
104 TSubScheduler& ss = TheSubSchedulers[i];
105 ss.i_TimerMultF = (TAny*)KMaxTUint32;
106 ss.i_TimerMultI = (TAny*)0x01000000u;
107 ss.i_CpuMult = (TAny*)KMaxTUint32;
108 ss.i_LastTimerSet = (TAny*)KMaxTInt32;
109 ss.i_TimestampError = (TAny*)0;
110 ss.i_TimerGap = (TAny*)16;
111 ss.i_MaxCorrection = (TAny*)64;
112 VIB->iTimerMult[i] = (volatile STimerMult*)&ss.i_TimerMultF;
113 VIB->iCpuMult[i] = (volatile TUint32*)&ss.i_CpuMult;
118 /** Register the global IRQ handler
119 Called by the base port at boot time to bind the top level IRQ dispatcher
120 to the ARM IRQ vector. Should not be called at any other time.
122 The handler specified will be called in mode_irq with IRQs disabled and
123 FIQs enabled. R0-R3, R12 and the return address from the interrupt will
124 be on the top of the mode_irq stack. R14_irq will point to the kernel's
125 IRQ postamble routine, which will run IDFCs and reschedule if necessary.
126 R13_irq will point to the top of the mode_irq stack and will be 8-byte aligned.
127 The handler should preserve all registers other than R0-R3, R12, R14_irq
128 and should return to the address in R14_irq.
130 @param aHandler The address of the top level IRQ dispatcher routine
132 EXPORT_C void Arm::SetIrqHandler(TLinAddr aHandler)
134 ArmInterruptInfo.iIrqHandler=aHandler;
137 /** Register the global FIQ handler
138 Called by the base port at boot time to bind the top level FIQ dispatcher
139 to the ARM FIQ vector. Should not be called at any other time.
141 The handler specified will be called in mode_fiq with both IRQs and FIQs
142 disabled. The return address from the interrupt will be on the top of the
143 mode_fiq stack. R14_fiq will point to the kernel's FIQ postamble routine,
144 which will run IDFCs and reschedule if necessary.
145 R13_fiq will point to the top of the mode_fiq stack and will be 4 modulo 8.
146 The handler should preserve all registers other than R8_fiq-R12_fiq and
147 R14_fiq and should return to the address in R14_fiq.
149 @param aHandler The address of the top level FIQ dispatcher routine
151 EXPORT_C void Arm::SetFiqHandler(TLinAddr aHandler)
153 ArmInterruptInfo.iFiqHandler=aHandler;
156 extern void initialiseState(TInt aCpu, TSubScheduler* aSS);
158 void Arm::Init1Interrupts()
160 // Initialise the interrupt and exception vector handlers.
163 __KTRACE_OPT(KBOOT,DEBUGPRINT(">Arm::Init1Interrupts()"));
165 TSubScheduler* ss = &TheSubSchedulers[0];
166 initialiseState(0, ss);
168 ArmLocalTimer& T = LOCAL_TIMER;
169 T.iWatchdogDisable = E_ArmTmrWDD_1;
170 T.iWatchdogDisable = E_ArmTmrWDD_2;
172 T.iTimerIntStatus = E_ArmTmrIntStatus_Event;
174 T.iWatchdogIntStatus = E_ArmTmrIntStatus_Event;
178 __KTRACE_OPT(KBOOT,DEBUGPRINT("<Arm::Init1Interrupts()"));
181 extern "C" void __ArmVectorReset()
186 extern "C" void __ArmVectorReserved()
192 TInt BTraceDefaultControl(BTrace::TControl /*aFunction*/, TAny* /*aArg1*/, TAny* /*aArg2*/)
194 return KErrNotSupported;
198 EXPORT_C void BTrace::SetHandlers(BTrace::THandler aNewHandler, BTrace::TControlFunction aNewControl, BTrace::THandler& aOldHandler, BTrace::TControlFunction& aOldControl)
200 BTrace::TControlFunction nc = aNewControl ? aNewControl : &BTraceDefaultControl;
201 __ACQUIRE_BTRACE_LOCK();
202 BTrace::THandler oldh = (BTrace::THandler)__e32_atomic_swp_ord_ptr(&BTraceData.iHandler, aNewHandler);
203 BTrace::TControlFunction oldc = (BTrace::TControlFunction)__e32_atomic_swp_ord_ptr(&BTraceData.iControl, nc);
204 __RELEASE_BTRACE_LOCK();
210 EXPORT_C TInt BTrace::SetFilter(TUint aCategory, TInt aValue)
212 if(!IsSupported(aCategory))
213 return KErrNotSupported;
214 TUint8* filter = BTraceData.iFilter+aCategory;
215 TUint oldValue = *filter;
216 if(TUint(aValue)<=1u)
218 oldValue = __e32_atomic_swp_ord8(filter, (TUint8)aValue);
219 BTraceContext4(BTrace::EMetaTrace, BTrace::EMetaTraceFilterChange, (TUint8)aCategory | (aValue<<8));
224 EXPORT_C SCpuIdleHandler* NKern::CpuIdleHandler()
226 return &ArmInterruptInfo.iCpuIdleHandler;
229 TUint32 NKern::IdleGenerationCount()
231 return TheScheduler.iIdleGenerationCount;
236 TScheduler& s = TheScheduler;
237 TSubScheduler& ss = SubScheduler(); // OK since idle thread locked to CPU
238 TUint32 m = ss.iCpuMask;
239 s.iIdleSpinLock.LockIrq();
240 TUint32 orig_cpus_not_idle = __e32_atomic_and_acq32(&s.iCpusNotIdle, ~m);
241 if (orig_cpus_not_idle == m)
244 if (!s.iIdleDfcs.IsEmpty())
246 __e32_atomic_ior_ord32(&s.iCpusNotIdle, m); // we aren't idle after all
247 s.iIdleGeneration ^= 1;
248 ++s.iIdleGenerationCount;
249 s.iIdleSpillCpu = (TUint8)ss.iCpuNum;
250 ss.iDfcs.MoveFrom(&s.iIdleDfcs);
251 ss.iDfcPendingFlag = 1;
252 s.iIdleSpinLock.UnlockIrq();
254 NKern::Unlock(); // process idle DFCs here
259 // postamble happens here - interrupts cannot be reenabled
260 s.iIdleSpinLock.UnlockOnly();
261 NKIdle(orig_cpus_not_idle & ~m);
263 // interrupts have not been reenabled
264 s.iIdleSpinLock.LockOnly();
265 __e32_atomic_ior_ord32(&s.iCpusNotIdle, m);
266 if (ArmInterruptInfo.iCpuIdleHandler.iPostambleRequired)
268 ArmInterruptInfo.iCpuIdleHandler.iPostambleRequired = FALSE;
271 s.iIdleSpinLock.UnlockIrq(); // reenables interrupts
275 EXPORT_C TUint32 NKern::CpuTimeMeasFreq()
277 return NKern::TimestampFrequency();
281 /** Converts a time interval in microseconds to thread timeslice ticks
283 @param aMicroseconds time interval in microseconds.
284 @return Number of thread timeslice ticks. Non-integral results are rounded up.
286 @pre aMicroseconds should be nonnegative
289 EXPORT_C TInt NKern::TimesliceTicks(TUint32 aMicroseconds)
291 TUint32 mf32 = (TUint32)TheScheduler.i_TimerMax;
293 TUint64 ticks = mf*TUint64(aMicroseconds) + UI64LIT(999999);
294 ticks /= UI64LIT(1000000);
295 if (ticks > TUint64(TInt(KMaxTInt)))
302 /** Get the frequency of counter queried by NKern::Timestamp().
307 EXPORT_C TUint32 NKern::TimestampFrequency()
309 return (TUint32)TheScheduler.i_TimerMax;