Update contrib.
1 // Copyright (c) 1998-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\nkern\nk_timer.cpp
15 // Fast Millisecond Timer Implementation
16 // This file is just a template - you'd be mad not to machine code this
22 #define i_NTimer_iState i8888.iHState1
23 #define i_NTimer_iCompleteInDfc i8888.iHState2
25 const TInt KTimerQDfcPriority=6;
27 GLDEF_D NTimerQ TheTimerQ;
29 extern "C" void send_irq_ipi(TSubScheduler*);
31 #ifndef __MSTIM_MACHINE_CODED__
33 #define __DEBUG_CALLBACK(n) {if (iDebugFn) (*iDebugFn)(iDebugPtr,n);}
35 #define __DEBUG_CALLBACK(n)
38 /** Construct a nanokernel timer tied to a specified thread or group
41 @param aTied Pointer to the thread/group to which the timer should be tied
42 @param aFunction Pointer to the function to call on timer expiry
43 @param aPtr Parameter to pass to the expiry handler
50 EXPORT_C NTimer::NTimer(NSchedulable* aTied, NTimerFn aFunction, TAny* aPtr)
54 iHType = EEventHandlerNTimer;
55 // i8888.iHState1 = EIdle; done by NEventHandler constructor
63 /** Construct a nanokernel timer which mutates into and runs as a DFC on expiry
64 The DFC queue is not specified at object construction time, but must be set
65 using NTimer::SetDfcQ() before the timer is used.
67 @param aFunction Pointer to the function to call on timer expiry
68 @param aPtr Parameter to pass to the expiry handler
69 @param aPriority Priority of DFC within the queue (0 to 7, where 7 is highest)
76 EXPORT_C NTimer::NTimer(TDfcFn aFunction, TAny* aPtr, TInt aPriority)
81 iHType = (TUint8)aPriority;
82 // i8888.iHState0 = 0; done by NEventHandler constructor
83 // i8888.iHState1 = EIdle; done by NEventHandler constructor
84 // i8888.iHState2 = 0; done by NEventHandler constructor
88 /** Construct a nanokernel timer which mutates into and runs as a DFC on expiry
90 @param aFunction Pointer to the function to call on timer expiry
91 @param aPtr Parameter to pass to the expiry handler
92 @param aDfcQ Pointer to DFC queue which this timer should use
93 @param aPriority Priority of DFC within the queue (0 to 7, where 7 is highest)
100 EXPORT_C NTimer::NTimer(TDfcFn aFunction, TAny* aPtr, TDfcQue* aDfcQ, TInt aPriority)
105 iHType = (TUint8)aPriority;
106 // i8888.iHState0 = 0; done by NEventHandler constructor
107 // i8888.iHState1 = EIdle; done by NEventHandler constructor
108 // i8888.iHState2 = 0; done by NEventHandler constructor
112 /** Set the DFC queue to be used by an NTimer constructed using a TDfcFn
114 @param aDfcQ Pointer to DFC queue which this timer should use
116 @pre Timer cannot be in use
122 EXPORT_C void NTimer::SetDfcQ(TDfcQue* aDfcQ)
124 __NK_ASSERT_ALWAYS(aDfcQ!=0);
125 __NK_ASSERT_ALWAYS(iHType < KNumDfcPriorities);
126 __NK_ASSERT_ALWAYS(i8816.iHState16==EIdle);
131 /** Tie a nanokernel timer to a thread or group
133 @param aTied = pointer to thread or group to which IDFC should be tied
134 @return KErrNone if successful
135 @return KErrDied if thread has exited or group has been destroyed.
137 @pre Call in thread context, interrupts enabled
138 @pre Timer must not be queued or running
139 @pre Timer must not already be tied
140 @pre Must not be a mutating timer (constructed with TDfcFn)
145 EXPORT_C TInt NTimer::SetTied(NSchedulable* aTied)
147 __NK_ASSERT_ALWAYS(!IsMutating());
148 __NK_ASSERT_ALWAYS(i8888.iHState1 == EIdle);
149 __NK_ASSERT_ALWAYS(aTied && !iTied);
151 TInt r = aTied->AddTiedEvent(this);
152 __NK_ASSERT_ALWAYS(r==KErrNone || r==KErrDied);
158 /** Destroy a nanokernel timer
160 @pre Call in thread context, interrupts enabled, preemption enabled
161 @pre Calling thread in critical section
162 @pre No fast mutex held
167 EXPORT_C NTimer::~NTimer()
169 if (!IsMutating() && iTied)
172 // remove from tied thread/group
173 NEventHandler::TiedLock.LockOnly();
174 NSchedulable* tied = iTied;
175 DoCancel(ECancelDestroy);
176 if (tied) // might have been dequeued by thread/group termination
187 NEventHandler::TiedLock.UnlockOnly();
190 else if (IsMutating() && iDfcQ)
191 DoCancelMutating(ECancelDestroy);
193 DoCancel(ECancelDestroy);
197 /** Starts a nanokernel timer in one-shot mode with ISR callback.
199 Queues the timer to expire in the specified number of nanokernel ticks. The
200 actual wait time will be at least that much and may be up to one tick more.
201 The expiry handler will be called in ISR context.
203 Note that NKern::TimerTicks() can be used to convert milliseconds to ticks.
205 @param aTime Timeout in nanokernel ticks
207 @return KErrNone if no error; KErrInUse if timer is already active.
211 @see NKern::TimerTicks()
213 EXPORT_C TInt NTimer::OneShot(TInt aTime)
215 return OneShot(aTime,FALSE);
219 /** Starts a nanokernel timer in one-shot mode with ISR or DFC callback.
221 Queues the timer to expire in the specified number of nanokernel ticks. The
222 actual wait time will be at least that much and may be up to one tick more.
223 For normal timers (constructed with NTimerFn) the expiry handler will be
224 called in either ISR context or in the context of the nanokernel timer
225 thread (DfcThread1). For mutating timers (constructed with TDfcFn) the
226 expiry handler is called in the context of the thread running the relevant
229 Note that NKern::TimerTicks() can be used to convert milliseconds to ticks.
231 @param aTime Timeout in nanokernel ticks
232 @param aDfc TRUE if DFC callback required, FALSE if ISR callback required.
233 Note that this parameter is ignored for mutating timers.
235 @return KErrNone if no error
236 @return KErrInUse if timer is already active.
237 @return KErrDied if tied thread/group has exited
241 @see NKern::TimerTicks()
243 EXPORT_C TInt NTimer::OneShot(TInt aTime, TBool aDfc)
245 __NK_ASSERT_DEBUG(aTime>=0);
246 /** iFn could be set to NULL after NTimer::OneShot(TInt, TDfc&) call.
247 Call-back mechanism cannot be changed in the life time of a timer. */
248 __NK_ASSERT_DEBUG(iFn!=NULL);
250 TInt irq = TheTimerQ.iTimerSpinLock.LockIrqSave();
253 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
256 TUint16 state = i8816.iHState16;
260 aDfc = FALSE; // mutating timers start as ISR completion
263 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
266 mb(); // ensure that if we observe an idle state all accesses to the NTimer have also been observed
267 i_NTimer_iCompleteInDfc=TUint8(aDfc?1:0);
268 iTriggerTime=TheTimerQ.iMsCount+(TUint32)aTime;
270 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
275 /** Starts a nanokernel timer in one-shot mode with callback in dfc thread that provided DFC belongs to.
277 Queues the timer to expire in the specified number of nanokernel ticks. The
278 actual wait time will be at least that much and may be up to one tick more.
279 On expiry aDfc will be queued in ISR context.
281 Note that NKern::TimerTicks() can be used to convert milliseconds to ticks.
283 @param aTime Timeout in nanokernel ticks
284 @param aDfc - Dfc to be queued when the timer expires.
286 @return KErrNone if no error
287 @return KErrInUse if timer is already active.
288 @return KErrDied if tied thread/group has exited
291 @pre Must not be a mutating timer (constructed with TDfcFn)
293 @see NKern::TimerTicks()
295 EXPORT_C TInt NTimer::OneShot(TInt aTime, TDfc& aDfc)
297 __NK_ASSERT_DEBUG(!IsMutating());
298 __NK_ASSERT_DEBUG(aTime>=0);
299 TInt irq = TheTimerQ.iTimerSpinLock.LockIrqSave();
300 if (iHType != EEventHandlerNTimer)
302 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
305 if (i_NTimer_iState!=EIdle)
307 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
310 mb(); // ensure that if we observe an idle state all accesses to the NTimer have also been observed
311 i_NTimer_iCompleteInDfc = 0;
313 iPtr = (TAny*) &aDfc;
314 iTriggerTime=TheTimerQ.iMsCount+(TUint32)aTime;
316 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
321 /** Starts a nanokernel timer in zero-drift periodic mode with ISR or DFC callback.
323 Queues the timer to expire in the specified number of nanokernel ticks,
324 measured from the time at which it last expired. This allows exact periodic
325 timers to be implemented with no drift caused by delays in requeueing the
328 The expiry handler will be called in the same context as the previous timer
329 expiry. Generally the way this is used is that NTimer::OneShot() is used to start
330 the first time interval and this specifies whether the callback is in ISR context
331 or in the context of the nanokernel timer thread (DfcThread1) or other Dfc thread.
332 The expiry handler then uses NTimer::Again() to requeue the timer.
334 @param aTime Timeout in nanokernel ticks
336 @return KErrNone if no error
337 @return KErrInUse if timer is already active;
338 @return KErrArgument if the requested expiry time is in the past.
339 @return KErrDied if tied thread/group has exited
343 EXPORT_C TInt NTimer::Again(TInt aTime)
345 // Wait aTime from last trigger time - used for periodic timers
348 __NK_ASSERT_DEBUG(aTime>0);
349 TInt irq = TheTimerQ.iTimerSpinLock.LockIrqSave();
352 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
355 TUint16 state = i8816.iHState16;
360 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
363 mb(); // ensure that if we observe an idle state all accesses to the NTimer have also been observed
364 TUint32 nextTick=TheTimerQ.iMsCount;
365 TUint32 trigger=iTriggerTime+(TUint32)aTime;
366 TUint32 d=trigger-nextTick;
369 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
370 return KErrArgument; // requested time is in the past
372 iTriggerTime=trigger;
374 TheTimerQ.iTimerSpinLock.UnlockIrqRestore(irq);
379 /** Cancels a nanokernel timer.
381 Removes this timer from the nanokernel timer queue. Does nothing if the
382 timer is inactive or has already expired.
383 Note that if the timer was queued and DFC callback requested it is possible
384 for the expiry handler to run even after Cancel() has been called. This will
385 occur in the case where DfcThread1 is preempted just before calling the
386 expiry handler for this timer and the preempting thread/ISR/IDFC calls
387 Cancel() on the timer.
389 @pre Any context for a non-mutating NTimer (constructed with NTimerFn)
390 @pre For mutating NTimer (constructed with TDfcFn), IDFC or thread context only.
391 @return TRUE if timer was actually cancelled
392 @return FALSE if timer was not cancelled - this could be because it was not
393 active or because its expiry handler was already running on
394 another CPU or in the timer DFC.
396 EXPORT_C TBool NTimer::Cancel()
398 if (IsMutating() && iDfcQ)
399 return DoCancelMutating(0);
400 return DoCancel(0)!=EIdle;
403 void NTimer::DoCancel0(TUint aState)
405 if (aState>ETransferring && aState<=EFinal) // idle or transferring timers are not on a queue
409 case ETransferring: // signal DFC to abort this iteration
410 TheTimerQ.iTransferringCancelled=TRUE;
412 case ECritical: // signal DFC to abort this iteration
413 TheTimerQ.iCriticalCancelled=TRUE;
417 // Need to clear bit in iPresent if both final queues now empty
418 // NOTE: Timer might actually be on the completed queue rather than the final queue
419 // but the check is harmless in any case.
420 TInt i=iTriggerTime & NTimerQ::ETimerQMask;
421 NTimerQ::STimerQ& q=TheTimerQ.iTickQ[i];
422 if (q.iIntQ.IsEmpty() && q.iDfcQ.IsEmpty())
423 TheTimerQ.iPresent &= ~(1<<i);
426 case EIdle: // nothing to do
427 case EHolding: // just deque
428 case EOrdered: // just deque
431 __NK_ASSERT_ALWAYS(0);
435 TUint NTimer::DoCancel(TUint aFlags)
437 NSchedulable* tied = 0;
438 TInt irq = NKern::DisableAllInterrupts();
439 TheTimerQ.iTimerSpinLock.LockOnly();
440 TUint state = i_NTimer_iState;
442 if (IsNormal() && state>=EEventQ)
444 // It's on a CPU's event handler queue
445 TInt cpu = state - EEventQ;
446 if (cpu < TheScheduler.iNumCpus)
448 TSubScheduler* ss = TheSubSchedulers + cpu;
449 ss->iEventHandlerLock.LockOnly();
450 state = i_NTimer_iState;
453 Deque(); // we got to it first
455 i_NTimer_iState = EIdle;
457 ss->iEventHandlerLock.UnlockOnly();
465 i_NTimer_iState=EIdle;
467 if (aFlags & ECancelDestroy)
468 iHType = EEventHandlerDummy;
469 TheTimerQ.iTimerSpinLock.UnlockOnly();
471 tied->EndTiedEvent(); // FIXME - Could be called in thread context
472 NKern::RestoreInterrupts(irq);
476 TBool NTimer::DoCancelMutating(TUint aFlags)
478 CHECK_PRECONDITIONS(MASK_NOT_ISR,"NTimer::Cancel (mutating NTimer)");
479 TSubScheduler& ss0 = SubScheduler();
483 TDfc* d = (TDfc*)this;
486 NThreadBase* t = q->iThread;
488 TheTimerQ.iTimerSpinLock.LockIrq();
490 // 0000->0000, XX00->ZZ00, xxYY->zzYY
491 TUint state = d->CancelInitialStateChange();
494 // someone else cancelling at the same time - just wait for them to finish
495 // they can only be waiting for the cancel IPI
500 if (state == 0) // timer was not active
505 if (state>=ETransferring && state<=EFinal)
508 // cancel is complete
513 // on DFC final queue
514 q->Remove((TPriListLink*)this);
517 // must be on IDFC queue - need to send cancel IPI
518 __NK_ASSERT_ALWAYS((state>>5)==4);
520 if (TUint(cpu) == ss0.iCpuNum)
522 // it's on this CPU's IDFC queue so just dequeue it and finish
526 d->ResetState(); // release semantics
529 if (aFlags & ECancelDestroy)
530 iHType = EEventHandlerDummy;
531 TheTimerQ.iTimerSpinLock.UnlockIrq();
537 ipi.WaitCompletion();
542 TUint n = 0x01000000;
543 while ((i8816.iHState16>>8) & ss0.iCpuMask)
556 /** Obtains the address of the nanokernel timer queue object.
558 Not intended for general use. Intended only for base ports in order to get
559 the address used to call NTimerQ::Tick() with.
561 @return The address of the nanokernel timer queue object
564 EXPORT_C TAny* NTimerQ::TimerAddress()
570 : iDfc(NTimerQ::DfcFn,this,NULL,KTimerQDfcPriority),
571 iDfcCompleteCount(1),
572 iTimerSpinLock(TSpinLock::EOrderNTimerQ)
574 // NOTE: All other members are initialised to zero since the single instance
575 // of NTimerQ resides in .bss
578 void NTimerQ::Init1(TInt aTickPeriod)
580 TheTimerQ.iTickPeriod=aTickPeriod;
581 __KTRACE_OPT(KBOOT,DEBUGPRINT("NTimerQ::Init1 - period %d us",aTickPeriod));
582 __KTRACE_OPT(KMEMTRACE, DEBUGPRINT("MT:P %d",aTickPeriod));
585 void NTimerQ::Init3(TDfcQue* aDfcQ)
587 __KTRACE_OPT(KBOOT,DEBUGPRINT("NTimerQ::Init3 DFCQ at %08x",aDfcQ));
588 TheTimerQ.iDfc.SetDfcQ(aDfcQ);
591 #ifndef __MSTIM_MACHINE_CODED__
592 void NTimerQ::Add(NTimer* aTimer)
594 // Internal function to add a timer to the queue.
595 // Enter and return with timer queue spin lock held.
598 TInt t=TInt(aTimer->iTriggerTime-iMsCount);
599 if (t<ENumTimerQueues)
603 // >=32ms to expiry, so put on holding queue
604 aTimer->i_NTimer_iState=NTimer::EHolding;
605 iHoldingQ.Add(aTimer);
609 void NTimerQ::AddFinal(NTimer* aTimer)
611 // Internal function to add a timer to the corresponding final queue.
612 // Enter and return with timer queue spin lock held.
615 TInt i=aTimer->iTriggerTime & ETimerQMask;
617 if (aTimer->i_NTimer_iCompleteInDfc)
622 aTimer->i_NTimer_iState=NTimer::EFinal;
626 void NTimerQ::DfcFn(TAny* aPtr)
628 ((NTimerQ*)aPtr)->Dfc();
633 // Do deferred timer queue processing and/or DFC completions
636 // First transfer entries on the Ordered queue to the Final queues
639 iTimerSpinLock.LockIrq();
640 if (iOrderedQ.IsEmpty())
642 NTimer* pC=(NTimer*)iOrderedQ.First();
643 TInt remain=pC->iTriggerTime-iMsCount;
644 if (remain>=ENumTimerQueues)
647 // If remaining time <32 ticks, add it to final queue;
648 // also if remain < 0 we've 'missed it' so add to final queue.
651 iTimerSpinLock.UnlockIrq();
654 iTimerSpinLock.UnlockIrq();
657 // Next transfer entries on the Holding queue to the Ordered queue or final queue
660 iTimerSpinLock.LockIrq();
661 if (iHoldingQ.IsEmpty())
663 NTimer* pC=(NTimer*)iHoldingQ.First();
665 pC->i_NTimer_iState=NTimer::ETransferring;
666 iTransferringCancelled=FALSE;
667 TUint32 trigger=pC->iTriggerTime;
668 if (TInt(trigger-iMsCount)<ENumTimerQueues)
670 // <32ms remaining so put it on final queue
677 iTimerSpinLock.UnlockIrq();
680 // we now need to walk ordered queue to find correct position for pC
681 SDblQueLink* anchor=&iOrderedQ.iA;
682 iCriticalCancelled=FALSE;
683 iTimerSpinLock.LockIrq();
684 NTimer* pN=(NTimer*)iOrderedQ.First();
685 while (pN!=anchor && !iTransferringCancelled)
687 if ((pN->iTriggerTime-trigger)<0x80000000u)
688 break; // insert before pN
689 pN->i_NTimer_iState=NTimer::ECritical;
690 iTimerSpinLock.UnlockIrq();
692 iTimerSpinLock.LockIrq();
693 if (iCriticalCancelled)
695 pN->i_NTimer_iState=NTimer::EOrdered;
696 pN=(NTimer*)pN->iNext;
699 if (iTransferringCancelled)
700 break; // this one has been cancelled, go on to next one
701 if (!iCriticalCancelled)
703 pC->InsertBefore(pN);
704 pC->i_NTimer_iState=NTimer::EOrdered;
705 break; // done this one
709 iTimerSpinLock.UnlockIrq();
712 iTimerSpinLock.UnlockIrq();
715 // Finally do call backs for timers which requested DFC callback
718 iTimerSpinLock.LockIrq();
719 if (iCompletedQ.IsEmpty())
721 NTimer* pC=(NTimer*)iCompletedQ.First();
723 pC->i_NTimer_iState=NTimer::EIdle;
726 iTimerSpinLock.UnlockIrq();
730 iTimerSpinLock.UnlockIrq();
731 __e32_atomic_add_rel32(&iDfcCompleteCount, 2);
735 /** Tick over the nanokernel timer queue.
736 This function should be called by the base port in the system tick timer ISR.
737 It should not be called at any other time.
738 The value of 'this' to pass is the value returned by NTimerQ::TimerAddress().
740 @see NTimerQ::TimerAddress()
742 EXPORT_C void NTimerQ::Tick()
744 TInt irq = iTimerSpinLock.LockIrqSave();
745 TInt i=iMsCount & ETimerQMask;
747 STimerQ* pQ=iTickQ+i;
750 if (!pQ->iDfcQ.IsEmpty())
752 // transfer DFC completions to completed queue and queue DFC
753 iCompletedQ.MoveFrom(&pQ->iDfcQ);
756 if ((i&(ETimerQMask>>1))==0)
758 // Every 16 ticks we check if a DFC is required.
759 // This allows a DFC latency of up to 16 ticks before timers are missed.
760 if (!iHoldingQ.IsEmpty())
762 doDfc=TRUE; // if holding queue nonempty, queue DFC to sort
764 else if (!iOrderedQ.IsEmpty())
766 // if first ordered queue entry expires in <32ms, queue the DFC to transfer
767 NTimer* pC=(NTimer*)iOrderedQ.First();
768 TUint x = pC->iTriggerTime - iMsCount;
769 if (x < (TUint)ENumTimerQueues)
775 if (!pQ->iIntQ.IsEmpty())
777 // transfer ISR completions to a temporary queue
778 // careful here - other CPUs could dequeue timers!
779 SDblQue q(&pQ->iIntQ,0);
780 for (; !q.IsEmpty(); iTimerSpinLock.LockIrqSave())
782 NTimer* pC=(NTimer*)q.First();
784 if (pC->IsMutating())
786 pC->AddAsDFC(); //mutate NTimer into TDfc and Add() it
787 iTimerSpinLock.UnlockIrqRestore(irq);
792 pC->i_NTimer_iState=NTimer::EIdle;
793 iTimerSpinLock.UnlockIrqRestore(irq);
794 ((TDfc*)(pC->iPtr))->Add();
797 NSchedulable* tied = pC->iTied;
800 TInt cpu = tied->BeginTiedEvent();
801 if (cpu != NKern::CurrentCpu())
803 pC->i_NTimer_iState = TUint8(NTimer::EEventQ + cpu);
804 TSubScheduler* ss = TheSubSchedulers + cpu;
805 TBool kick = ss->QueueEvent(pC);
806 iTimerSpinLock.UnlockIrqRestore(irq);
812 pC->i_NTimer_iState=NTimer::EIdle;
814 NTimerFn f = pC->iFn;
815 iTimerSpinLock.UnlockIrqRestore(irq);
818 tied->EndTiedEvent();
821 iTimerSpinLock.UnlockIrqRestore(irq);
827 /** Mutate an NTimer into a DFC and Add() it
829 If NTimer state is EFinal, change to DFC state 008n and add to endogenous IDFC
832 Enter and return with IRQs disabled and timer spin lock held
833 No need to worry about Cancel()s since timer spin lock is held
838 void NTimer::AddAsDFC()
840 TSubScheduler& ss = SubScheduler();
841 i8816.iHState16 = (TUint16)(0x80|ss.iCpuNum);
843 ss.iDfcPendingFlag = 1;
847 /** Check if a nanokernel timer is pending or not
849 @return TRUE if the timer is pending (OneShot() etc. would return KErrInUse)
850 @return FALSE if the timer is idle (OneShot() etc. would succeed)
856 EXPORT_C TBool NTimer::IsPending()
858 TUint16 state = i8816.iHState16;
859 return state != EIdle;
863 /** Return the number of ticks before the next nanokernel timer expiry.
864 May on occasion return a pessimistic estimate (i.e. too low).
865 Used by base port to disable the system tick interrupt when the system
868 @return The number of ticks before the next nanokernel timer expiry.
870 @pre Interrupts must be disabled.
872 @post Interrupts are disabled.
874 EXPORT_C TInt NTimerQ::IdleTime()
876 CHECK_PRECONDITIONS(MASK_INTERRUPTS_DISABLED,"NTimerQ::IdleTime");
877 NTimerQ& m=TheTimerQ;
878 TUint32 next=m.iMsCount; // number of next tick
879 TUint32 p=m.iPresent;
883 // Final queues nonempty
884 TInt nx=next&0x1f; // number of next tick modulo 32
885 p=(p>>nx)|(p<<(32-nx)); // rotate p right by nx (so lsb corresponds to next tick)
886 r=__e32_find_ls1_32(p); // find number of zeros before LS 1
888 if (!m.iHoldingQ.IsEmpty())
890 // Sort operation required - need to process next tick divisible by 16
891 TInt nx=next&0x0f; // number of next tick modulo 16
892 TInt r2=nx?(16-nx):0; // number of ticks before next divisible by 16
896 if (!m.iOrderedQ.IsEmpty())
898 // Timers present on ordered queue
899 NTimer* pC=(NTimer*)m.iOrderedQ.First();
900 TUint32 tt=pC->iTriggerTime;
901 tt=(tt&~0x0f)-16; // time at which transfer to final queue would occur
902 TInt r3=(TInt)(tt-next);
911 /** Advance the nanokernel timer queue by the specified number of ticks.
912 It is assumed that no timers expire as a result of this.
913 Used by base port when system comes out of idle mode after disabling the
914 system tick interrupt to bring the timer queue up to date.
916 @param aTicks Number of ticks skipped due to tick suppression
918 @pre Interrupts must be disabled.
920 @post Interrupts are disabled.
922 EXPORT_C void NTimerQ::Advance(TInt aTicks)
924 CHECK_PRECONDITIONS(MASK_INTERRUPTS_DISABLED,"NTimerQ::Advance");
925 TheTimerQ.iMsCount+=(TUint32)aTicks;
929 /** Returns the period of the nanokernel timer.
930 @return Period in microseconds
934 EXPORT_C TInt NKern::TickPeriod()
936 return TheTimerQ.iTickPeriod;
940 /** Converts a time interval to timer ticks.
942 @param aMilliseconds time interval in milliseconds.
943 @return Number of nanokernel timer ticks. Non-integral results are rounded up.
945 @pre aMilliseconds should be <=2147483 to avoid integer overflow.
948 EXPORT_C TInt NKern::TimerTicks(TInt aMilliseconds)
950 __ASSERT_WITH_MESSAGE_DEBUG(aMilliseconds<=2147483,"aMilliseconds should be <=2147483","NKern::TimerTicks");
951 TUint32 msp=TheTimerQ.iTickPeriod;
952 if (msp==1000) // will be true except on pathological hardware
953 return aMilliseconds;
954 TUint32 us=(TUint32)aMilliseconds*1000;
955 return (us+msp-1)/msp;