kernel/eka/nkernsmp/dfcs.cpp
author William Roberts <williamr@symbian.org>
Tue, 19 Jan 2010 13:48:03 +0000
changeset 7 f497542af8e4
parent 0 a41df078684a
child 43 c1f20ce4abcf
permissions -rw-r--r--
Merge improved comments (now included as part of Revision: 201001)

// Copyright (c) 2006-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:
// e32\nkernsmp\dfcs.cpp
// DFCs
// 
//

// NThreadBase member data
#define __INCLUDE_NTHREADBASE_DEFINES__

// TDfc member data
#define __INCLUDE_TDFC_DEFINES__

#include "nk_priv.h"

extern "C" void send_self_resched_ipi();

/** Construct an IDFC

	@param aFunction = function to call
	@param aPtr = parameter to be passed to function
 */
EXPORT_C TDfc::TDfc(TDfcFn aFunction, TAny* aPtr)
	{
	iPtr = aPtr;
	iFn = aFunction;
	iTied = 0;
	iHType = EEventHandlerIDFC;
	i8888.iHState0 = 0;
	i8888.iHState1 = 0;
	i8888.iHState2 = 0;
	iTiedLink.iNext = 0;
	}


/** Construct an IDFC tied to a thread or group

	@param aTied = pointer to thread or group to which IDFC should be tied
	@param aFunction = function to call
	@param aPtr = parameter to be passed to function

	@pre Call in thread context, interrupts enabled
 */
EXPORT_C TDfc::TDfc(NSchedulable* aTied, TDfcFn aFunction, TAny* aPtr)
	{
	iPtr = aPtr;
	iFn = aFunction;
	iTied = 0;
	iHType = EEventHandlerIDFC;
	i8888.iHState0 = 0;
	i8888.iHState1 = 0;
	i8888.iHState2 = 0;
	iTiedLink.iNext = 0;
	if (aTied)
		{
		SetTied(aTied);
		}
	}


/** Construct a DFC without specifying a DFC queue.
	The DFC queue must be set before the DFC may be queued.

	@param aFunction = function to call
	@param aPtr = parameter to be passed to function
	@param aPriority = priority of DFC within the queue (0 to 7, where 7 is highest)
 */
EXPORT_C TDfc::TDfc(TDfcFn aFunction, TAny* aPtr, TInt aPriority)
	{
	__NK_ASSERT_DEBUG((TUint)aPriority<(TUint)KNumDfcPriorities);
	iPtr = aPtr;
	iFn = aFunction;
	iTied = 0;
	iHType = TUint8(aPriority);
	i8888.iHState0 = 0;
	i8888.iHState1 = 0;
	i8888.iHState2 = 0;
	iTiedLink.iNext = 0;
	}


/** Construct a DFC specifying a DFC queue.

	@param aFunction = function to call
	@param aPtr = parameter to be passed to function
	@param aDfcQ = pointer to DFC queue which this DFC should use
	@param aPriority = priority of DFC within the queue (0-7)
 */
EXPORT_C TDfc::TDfc(TDfcFn aFunction, TAny* aPtr, TDfcQue* aDfcQ, TInt aPriority)
	{
	__NK_ASSERT_DEBUG((TUint)aPriority<(TUint)KNumDfcPriorities);
	iPtr = aPtr;
	iFn = aFunction;
	iDfcQ = aDfcQ;
	iHType = TUint8(aPriority);
	i8888.iHState0 = 0;
	i8888.iHState1 = 0;
	i8888.iHState2 = 0;
	iTiedLink.iNext = 0;
	}


/** Tie an IDFC to a thread or group

	@param	aTied = pointer to thread or group to which IDFC should be tied
	@return	KErrNone if successful
	@return	KErrDied if thread has exited or group has been destroyed.

	@pre Call in thread context, interrupts enabled
	@pre Must be IDFC not DFC
	@pre IDFC must not be queued or running
	@pre IDFC must not already be tied
 */
EXPORT_C TInt TDfc::SetTied(NSchedulable* aTied)
	{
	__NK_ASSERT_ALWAYS(IsIDFC() && i8816.iHState16==0);
	__NK_ASSERT_ALWAYS(aTied && !iTied);
	NKern::Lock();
	TInt r = aTied->AddTiedEvent(this);
	__NK_ASSERT_ALWAYS(r==KErrNone || r==KErrDied);
	NKern::Unlock();
	return r;
	}


/** Destroy a DFC or IDFC

	@pre Call from thread context with interrupts and preemption enabled
	@pre Calling thread holds no fast mutex
	@pre Calling thread in critical section
 */
EXPORT_C TDfc::~TDfc()
	{
	CHECK_PRECONDITIONS(MASK_THREAD_CRITICAL,"TDfc::~TDfc");
	NKern::Lock();
	NEventHandler::TiedLock.LockOnly();
	NSchedulable* tied = iTied;
	if (IsDFC() || (IsIDFC() && !tied))
		{
		Cancel();
		iHType = (TUint8)EEventHandlerDummy;
		}
	if (IsIDFC())
		{
		__NK_ASSERT_ALWAYS(tied!=0);
		tied->AcqSLock();
		if (iTiedLink.iNext)
			{
			iTiedLink.Deque();
			iTiedLink.iNext = 0;
			}
		tied->RelSLock();
		Cancel();
		iHType = (TUint8)EEventHandlerDummy;
		iTied = 0;
		}
	NEventHandler::TiedLock.UnlockOnly();
	NKern::Unlock();
	}


/** Construct a DFC queue
	Kern::DfcQInit() should be called on the new DFC queue before it can be used.
 */
EXPORT_C TDfcQue::TDfcQue()
	: iThread(NULL)
	{}



/** Queue an IDFC or a DFC from an ISR

	This function is the only way to queue an IDFC and is the only way to queue
	a DFC from an ISR. To queue a DFC from an IDFC or a thread either Enque()
	or DoEnque() should be used.

	This function does nothing if the IDFC/DFC is already queued.

	@pre Call only from ISR, IDFC or thread with preemption disabled.
	@pre Do not call from thread with preemption enabled.
	@return	TRUE if DFC was actually queued by this call
			FALSE if DFC was already queued on entry so this call did nothing

	@see TDfc::DoEnque()
	@see TDfc::Enque()
 */
EXPORT_C TBool TDfc::Add()
	{
	__ASSERT_DEBUG(NKern::CurrentContext()!=NKern::EThread || NKern::KernelLocked(), *(int*)0xdfcadd01=0);
	__ASSERT_DEBUG(IsIDFC() || (IsDFC() && iDfcQ), *(int*)0xdfcadd03=0);
//	__ASSERT_WITH_MESSAGE_DEBUG(  NKern::CurrentContext()!=NKern::EThread  ||  NKern::KernelLocked(),"Do not call from thread with preemption enabled","TDfc::Add");
//	__ASSERT_WITH_MESSAGE_DEBUG(  IsIDFC() || (IsDFC() && iDfcQ), "DFC queue not set", "TDfc::Add");
#ifdef __WINS__
	__NK_ASSERT_ALWAYS(Interrupt.InInterrupt() || NKern::KernelLocked());
#endif
	TInt irq = NKern::DisableAllInterrupts();
	TSubScheduler& ss = SubScheduler();
	TUint32 orig = 0xFF00;

	// Transition the state to 'on normal IDFC queue'
	// 0000->008n
	// 00Cn->00En
	// All other states unchanged
	// Return original state
	if (IsValid())	// don't add if tied and tied thread/group is being/has been destroyed
		orig = AddStateChange();
	if (orig==0)
		{
		// wasn't already queued
		i8888.iHState0 = 0;	// BeginTiedEvent() not done
		ss.iDfcs.Add(this);
		ss.iDfcPendingFlag = 1;
#ifdef _DEBUG
		TUint32 st8 = DFC_STATE(this) & 0xFF;
		if (st8 != (0x80|ss.iCpuNum))
			__crash();
#endif
		}
	NKern::RestoreInterrupts(irq);
	return (orig==0 || (orig&0xFFE0)==0x00C0);
	}


/** Queue an IDFC or a DFC from any context

	This function is identical to TDfc::Add() but no checks are performed for correct usage,
	and it contains no instrumentation code.

	@return	TRUE if DFC was actually queued by this call
			FALSE if DFC was already queued on entry so this call did nothing

	@see TDfc::DoEnque()
	@see TDfc::Enque()
	@see TDfc::Add()
 */
EXPORT_C TBool TDfc::RawAdd()
	{
	TInt irq = NKern::DisableAllInterrupts();
	TSubScheduler& ss = SubScheduler();
	TUint32 orig = 0xFF00;
	if (IsValid())	// don't add if tied and tied thread/group is being/has been destroyed
		orig = AddStateChange();
	if (orig==0)
		{
		// wasn't already queued
		i8888.iHState0 = 0;	// BeginTiedEvent() not done
		ss.iDfcs.Add(this);
		ss.iDfcPendingFlag = 1;
		send_self_resched_ipi();	// ensure current CPU runs the DFC
#ifdef _DEBUG
		TUint32 st8 = DFC_STATE(this) & 0xFF;
		if (st8 != (0x80|ss.iCpuNum))
			__crash();
#endif
		// FIXME: Need to wait to ensure IRQ is active before reenabling interrupts
		}
	NKern::RestoreInterrupts(irq);
	return (orig==0 || (orig&0xFFE0)==0x00C0);
	}


/** Queue a DFC (not an IDFC) from an IDFC or thread with preemption disabled.

	This function is the preferred way to queue a DFC from an IDFC. It should not
	be used to queue an IDFC - use TDfc::Add() for this.

	This function does nothing if the DFC is already queued.

	@pre Call only from IDFC or thread with preemption disabled.
	@pre Do not call from ISR or thread with preemption enabled.
	@return	TRUE if DFC was actually queued by this call
			FALSE if DFC was already queued on entry so this call did nothing

	@see TDfc::Add()
	@see TDfc::Enque()
 */
EXPORT_C TBool TDfc::DoEnque()
	{
	__ASSERT_WITH_MESSAGE_DEBUG(  (NKern::CurrentContext()==NKern::EIDFC )||( NKern::CurrentContext()==NKern::EThread && NKern::KernelLocked()),"Do not call from ISR or thread with preemption enabled","TDfc::DoEnque");
	__NK_ASSERT_DEBUG(IsDFC());
	__ASSERT_WITH_MESSAGE_DEBUG(iDfcQ, "DFC queue not set", "TDfc::DoEnque");

	// Check not already queued and then mark queued to prevent ISRs touching this DFC
	TDfcQue* q = iDfcQ;
	NThreadBase* t = q->iThread;
	t->AcqSLock();	// also protects DFC queue
	TUint16 expect = 0;
	TBool ok = __e32_atomic_cas_acq16(&iDfcState, &expect, 1);
	if (ok)
		{
		// wasn't already queued, now marked as on final queue, which means
		// attempts to cancel will block on the thread spin lock
		TUint present = q->iPresent[0];
		q->Add((TPriListLink*)this);
		if (!present)
			t->iWaitState.UnBlockT(NThreadBase::EWaitDfc, q, KErrNone);
		}
	t->RelSLock();	// also protects DFC queue
	return ok;
	}

void TDfcQue::ThreadFunction(TAny* aDfcQ)
	{
	TDfcQue& q = *(TDfcQue*)aDfcQ;
	NThreadBase* t = NKern::CurrentThread();
	FOREVER
		{
		NKern::Lock();
		t->AcqSLock();	// also protects DFC queue
		if (q.IsEmpty())
			{
			t->iWaitState.SetUpWait(NThreadBase::EWaitDfc, 0, &q);
			RescheduleNeeded();
			t->RelSLock();	// also protects DFC queue
			NKern::Unlock();
			}
		else
			{
			TDfc* d = q.First();
			q.Remove((TPriListLink*)d);
			TDfcFn f = d->iFn;
			TAny* p = d->iPtr;
			d->ResetState();
			t->RelSLock();	// also protects DFC queue
			NKern::Unlock();
			(*f)(p);
			}
		}
	}



void TCancelIPI::Send(TDfc* aDfc, TInt aCpu)
	{
	iDfc = aDfc;
	Queue(&Isr, 1u<<aCpu);
	}

void TCancelIPI::Isr(TGenericIPI* aIPI)
	{
	TCancelIPI* p = (TCancelIPI*)aIPI;
	TDfc* d = p->iDfc;
	if (d->iNext)
		{
		// QueueDfcs() hasn't dequeued it yet
		// just dequeue it here and reset the state - QueueDfcs() will never see it
		// Note that this means we have to release the tied thread/group if necessary
		// BeginTiedEvent() has occurred if iHState0 is set and it's actually an IDFC not an NTimer
		NSchedulable* tied = (d->iHType==NEventHandler::EEventHandlerIDFC && d->i8888.iHState0) ? d->iTied : 0;
		d->Deque();
		d->ResetState();
		if (tied)
			tied->EndTiedEvent();
		}
	else
		{
		// QueueDfcs() has already dequeued it
		// state transition:
		//		XXYY->XX00
		//		XX00->0000
		// QueueDfcs() will take care of the tied thread/group
		d->CancelFinalStateChange();
		}
	}


/** Cancels an IDFC or DFC.

	This function does nothing if the IDFC or DFC is not queued.

	For any DFC or IDFC the following identity holds:
			Number of times Add() is called and returns TRUE
		+	Number of times DoEnque() is called and returns TRUE
		+	Number of times Enque() is called and returns TRUE
		+	Number of times QueueOnIdle() is called and returns TRUE
		=	Number of times Cancel() is called and returns TRUE
		+	Number of times the DFC/IDFC function executes

	@pre IDFC or thread context. Do not call from ISRs.

	@pre If the DFC function accesses the DFC object itself, the user must ensure that
	     Cancel() cannot be called while the DFC function is running.

	@return	TRUE	if the DFC was actually dequeued by this call - i.e. an
					instance of the DFC's execution has been prevented. It
					is still possible that a previous execution is still in
					progress.
			FALSE	if the DFC was not queued on entry to the call, or was in
					the process of being executed or cancelled. In this case
					it is possible that the DFC executes after this call
					returns.

	@post	However in either case it is safe to delete the DFC object on
			return from this call provided only that the DFC function does not
			refer to the DFC object itself.
 */
EXPORT_C TBool TDfc::Cancel()
	{
	enum TAction { EDeque=1, EReset=2, EIdleUnlock=4, ESendIPI=8, EWait=16 };

	CHECK_PRECONDITIONS(MASK_NOT_ISR|MASK_INTERRUPTS_ENABLED,"TDfc::Cancel");
	if (!iDfcState)
		return FALSE;
	TUint action = EIdleUnlock;
	TBool ret = FALSE;
	TInt cpu = -1;
	NSchedulable* tied = 0;
	TDfcQue* q = 0;
	NThreadBase* t = 0;
	NKern::Lock();
	TSubScheduler& ss0 = SubScheduler();
	if (IsDFC())
		q = iDfcQ, t = q->iThread, t->AcqSLock();
	TInt irq = NKern::DisableAllInterrupts();
	TheScheduler.iIdleSpinLock.LockOnly();

	// 0000->0000, XX00->ZZ00, xxYY->zzYY
	TUint state = CancelInitialStateChange();
	TUint stt = state >> 5;
	if (state & 0xFF00)
		{
		// someone else cancelling at the same time - just wait for them to finish
		action = EWait|EIdleUnlock;
		goto end;
		}
	if (state == 0)	// DFC not active
		goto end;

	// possible states here are 0001, 002g, 006m, 008m, 00Am, 00Cm, 00Em
	ret = (stt!=6);	// if running but not pending, Cancel() will not have prevented an execution
	if (state == TUint(TheScheduler.iIdleGeneration | 0x20))
		{
		// was on idle queue, BeginTiedEvent() isn't called until QueueDfcs() runs
		action = EDeque|EReset|EIdleUnlock;
		goto end;
		}
	if (state == 1)
		{
		// was on final queue, must be DFC not IDFC
		q->Remove((TPriListLink*)this);
		action = EReset|EIdleUnlock;
		goto end;
		}

	// possible states here are 002g (spilled), 006m, 008m, 00Am, 00Cm, 00Em
	// i.e. either on IDFC queue, ExIDFC queue or running
	// For IDFCs, tied thread/group is now in play.
	cpu = state & 0x1f;	// CPU it's on for states 006m, 008m, 00Am, 00Cm, 00Em
	if (stt==3 || stt==6 || stt==7)
		{
		// It's actually running - must be IDFC. A re-queue may also be pending.
		TheScheduler.iIdleSpinLock.UnlockOnly();
		TSubScheduler* ss = TheSubSchedulers + cpu;
		TDfc* expect = this;
		TBool done = __e32_atomic_cas_acq_ptr(&ss->iCurrentIDFC, &expect, 0);
		if (done)
			{
			// We cleared iCurrentIDFC so QueueDfcs() won't touch this again - we reset the state and finish up
			// We must also release the tied thread/group
			tied = iTied;
			action = EReset;
			goto end;
			}
		// QueueDfcs() got to iCurrentIDFC before we did, so we interlock with it
		// and we can leave the EndTiedEvent to it as well
		// State transition:
		//		XXAm->XX00, wait
		//		XX00->0000, don't wait
		TUint32 orig = CancelFinalStateChange() & 0xFF;
		__NK_ASSERT_ALWAYS(orig==0 || orig==state);
		action = orig ? EWait : 0;
		goto end;
		}

	// possible states here 002g (propagated), 008m, 00Am so it's either on the endogenous or exogenous IDFC queue
	if (stt==5)
		{
		// it's on the exogenous IDFC queue
		TheScheduler.iIdleSpinLock.UnlockOnly();
		TSubScheduler* ss = TheSubSchedulers + cpu;
		ss->iExIDfcLock.LockOnly();
		if (iNext)
			{
			// we got to it before QueueDfcs() on the other CPU so we can finish up here
			// QueueDfcs() will never see it again so we must release tied thread/group
			Deque();
			tied = iTied;
			ss->iExIDfcLock.UnlockOnly();
			action = EReset;
			goto end;
			}
		// QueueDfcs() on other CPU has already dequeued it - we must now interlock with RunIDFCStateChange()
		ss->iExIDfcLock.UnlockOnly();
		// State transition:
		//		XXAm->XX00, wait
		//		XX00->0000, don't wait
		// QueueDfcs() will take care of tied thread/group
		TUint32 orig = CancelFinalStateChange() & 0xFF;
		__NK_ASSERT_ALWAYS(orig==0 || orig==state);
		action = orig ? EWait : 0;
		goto end;
		}

	// possible states here 002g (propagated idle) or 008m (IDFC or DFC on endogenous DFC queue)
	if (stt==1)	// propagated idle
		cpu = TheScheduler.iIdleSpillCpu;

	// if it's on this CPU's IDFC queue we can just remove it and reset the state here
	// otherwise we send a cancel IPI to the CPU it's on
	// We are guaranteed to dequeue the DFC before it executes since the
	// QueueDfcs() on the target CPU will notice that a cancel is in progress and
	// so will not run the DFC even if it dequeues it.
	// QueueDfcs() takes care of the tied thread/group if it sees the DFC/IDFC again, otherwise
	// we must do it here.
	if (TUint(cpu) == ss0.iCpuNum)
		{
		if (IsIDFC())
			tied = iTied;
		action = EDeque|EReset|EIdleUnlock;
		}
	else
		action = EIdleUnlock|ESendIPI|EWait;

end:
	// Common exit point
	if (action & EDeque)
		Deque();
	if (action & EReset)
		{
		ResetState();
		}
	if (action & EIdleUnlock)
		TheScheduler.iIdleSpinLock.UnlockOnly();
	NKern::RestoreInterrupts(irq);
	if (t)
		t->RelSLock();

	// on another CPU's IDFC queue so send IPI to remove it
	if (action & ESendIPI)
		{
		TCancelIPI ipi;
		ipi.Send(this, cpu);
		ipi.WaitCompletion();
		tied = 0;
		}

	// wait for cancel to complete
	if (action & EWait)
		{
		TUint n = 0x01000000;
		while ((iDfcState>>8) & ss0.iCpuMask)
			{
			__chill();
			if (!--n)
				__crash();
			}
		}

	// release tied thread/group if waiting for IDFC to complete
	if (tied)
		tied->EndTiedEvent();
	NKern::Unlock();
	return ret;
	}


/** Queues a DFC (not an IDFC) from a thread.

	Does nothing if DFC is already queued.

    NOTE: Although this can be called in an IDFC context, it is more efficient to call
    DoEnque() in this case.
    
    @pre    Call either in a thread or an IDFC context.
	@pre	Do not call from an ISR.
	@return	TRUE if DFC was actually queued by this call
			FALSE if DFC was already queued on entry so this call did nothing
 */
EXPORT_C TBool TDfc::Enque()
	{
	CHECK_PRECONDITIONS(MASK_NOT_ISR,"TDfc::Enque()");		
	NKern::Lock();
	TBool ret = DoEnque();
	NKern::Unlock();
	return ret;
	}


/** Queue a DFC (not an IDFC) from a thread and also signals a fast mutex.

	The DFC is unaffected if it is already queued.

	The fast mutex is signalled before preemption is reenabled to avoid potential
	scheduler thrashing.

	@param	aMutex =	pointer to fast mutex to be signalled;
						NULL means system lock mutex.
	@return	TRUE if DFC was actually queued by this call
			FALSE if DFC was already queued on entry so this call did nothing
	@pre	Call in a thread context.
	@pre	Kernel must be unlocked.
	@pre	Do not call from an ISR.
	@pre    Do not call from an IDFC.
 */
EXPORT_C TBool TDfc::Enque(NFastMutex* aMutex)
	{
	CHECK_PRECONDITIONS(MASK_KERNEL_UNLOCKED|MASK_NOT_ISR|MASK_NOT_IDFC,"TDfc::Enque(NFastMutex* aMutex)");		
	if (!aMutex)
		aMutex=&TheScheduler.iLock;
	NKern::Lock();
	TBool ret = DoEnque();
	aMutex->Signal();
	NKern::Unlock();
	return ret;
	}


/** Returns a pointer to the thread on which a DFC runs

	@return	If this is a DFC and the DFC queue has been set, a pointer to the
			thread which will run the DFC.
			NULL if this is an IDFC or the DFC queue has not been set.
 */
EXPORT_C NThreadBase* TDfc::Thread()
	{
	if (!IsDFC())
		return 0;
	return iDfcQ ? iDfcQ->iThread : 0;
	}


/******************************************************************************
 * Idle notification
 ******************************************************************************/

/** Register an IDFC or a DFC to be called when the system goes idle

	This function does nothing if the IDFC/DFC is already queued.

	@return	TRUE if DFC was actually queued by this call
			FALSE if DFC was already queued on entry so this call did nothing
 */
EXPORT_C TBool TDfc::QueueOnIdle()
	{
	TInt irq = TheScheduler.iIdleSpinLock.LockIrqSave();
	TUint32 orig = 0xFF00;

	// Transition the state to 'on normal idle queue'
	// 0000->002g
	// 00Cn->006n
	// All other states unchanged
	// Return original state
	if (IsValid())	// don't add if tied and tied thread/group is being/has been destroyed
		orig = QueueOnIdleStateChange();
	if (orig==0)
		{
		i8888.iHState0 = 0;	// BeginTiedEvent() not done
		TheScheduler.iIdleDfcs.Add(this);
		}

	TheScheduler.iIdleSpinLock.UnlockIrqRestore(irq);
	return (orig==0 || (orig&0xFFE0)==0x00C0);
	}


/******************************************************************************
 * Scheduler IDFC/DFC Processing
 ******************************************************************************/

void TSubScheduler::QueueDfcs()
//
// Enter with interrupts off and kernel locked
// Leave with interrupts off and kernel locked
//
// In state descriptions:
//		XX=8 bits not all zero (bitmask representing cancelling CPUs)
//		xx=8 bits (bitmask representing cancelling CPUs)
//		YY=8 bits not all zero
//		ZZ=XX with an additional bit set corresponding to the current CPU
//		zz=xx with an additional bit set corresponding to the current CPU
//		n = current CPU number
//		m = another CPU number
//		g = idle generation number
	{
	__KTRACE_OPT(KSCHED2,DEBUGPRINT("^"));
	iInIDFC = TRUE;
	BTrace0(BTrace::ECpuUsage, BTrace::EIDFCStart);
	TDfc* d = 0;
	NSchedulable* tied = 0;
	FOREVER
		{
		NKern::DisableAllInterrupts();
		// remove from pending queue with interrupts disabled
		d = (TDfc*)iDfcs.GetFirst();
		if (d)
			{
			d->iNext = 0;
#ifdef _DEBUG
			TUint32 st8 = DFC_STATE(d) & 0xFF;
			if (st8 != TUint(0x80|iCpuNum) && st8 != TUint(0x21^TheScheduler.iIdleGeneration))
				__crash();
#endif
			if (d->IsDFC())	// also true for mutating NTimer
				{
				NKern::EnableAllInterrupts();
				TDfcQue* q = d->iDfcQ;
				NThreadBase* t = q->iThread;
				t->AcqSLock();	// also protects DFC queue

				// transition to 'final queue' state
				// 002g->0001, ok=TRUE
				// 008n->0001, ok=TRUE
				// XXYY->XX00, ok=FALSE
				// XX00->0000, ok=FALSE
				// other starting states invalid
				TUint32 orig = d->MoveToFinalQStateChange() >> 5;
				if (orig==1 || orig==4)
					{
					// wasn't being cancelled, now marked as on final queue, which means
					// attempts to cancel will block on the thread spin lock
					TUint present = q->iPresent[0];
					q->Add((TPriListLink*)d);
					if (!present)
						t->iWaitState.UnBlockT(NThreadBase::EWaitDfc, q, KErrNone);
					}
				t->RelSLock();	// also protects DFC queue
				continue;
				}
			// endogenous IDFC - could be tied in which case may need to be punted over to another CPU
			// can't be mutating NTimer since that would have gone into IsDFC() path
			tied = d->iTied;
			if (tied && !d->i8888.iHState0)	// if tied and BeginTiedEvent() not already done
				{
				d->i8888.iHState0 = 1;		// flag that BeginTiedEvent() done
				TInt cpu = tied->BeginTiedEvent();
				if (TUint(cpu) != iCpuNum)
					{
					// punt over to other CPU
					TBool kick = FALSE;
					TSubScheduler* ss = TheSubSchedulers + cpu;
					ss->iExIDfcLock.LockOnly();
					// transition state here to handle cancel
					// XXYY->XX00, ok=FALSE
					// XX00->0000, ok=FALSE
					// 008n->00Am, ok=TRUE
					// 002g->00Am, ok=TRUE
					// other starting states invalid
					TUint32 orig = d->TransferIDFCStateChange(cpu) >> 5;
					if (orig==1 || orig==4)
						{
						kick = !ss->iExIDfcPendingFlag;
						ss->iExIDfcPendingFlag = TRUE;
						ss->iExIDfcs.Add(d);
						}
					ss->iExIDfcLock.UnlockOnly();
					if (kick)
						send_resched_ipi(cpu);
					NKern::EnableAllInterrupts();	// let interrupts in
					if (orig >= 8)
						tied->EndTiedEvent();		// IDFC cancelled so release tied thread/group
					continue;
					}
				}
			}
		else
			{
			if (!iExIDfcPendingFlag)
				break;
			iExIDfcLock.LockOnly();
			d = (TDfc*)iExIDfcs.GetFirst();
			if (!d)
				{
				iExIDfcPendingFlag = 0;
				iExIDfcLock.UnlockOnly();
				break;
				}
			d->iNext = 0;
			tied = d->iTied;
			__NK_ASSERT_ALWAYS(d->IsIDFC() && tied);	// only tied IDFCs should get here
#ifdef _DEBUG
			TUint32 st8 = DFC_STATE(d) & 0xFF;
			if (st8 != (0xA0|iCpuNum))
				__crash();
#endif
			iExIDfcLock.UnlockOnly();
			}

		// endogenous or exogenous IDFC
		// if tied, we are on correct CPU
		TDfcFn f = d->iFn;
		TAny* p = d->iPtr;

		// If Cancel() finds the IDFC in the running state (00Cn or 00En) it will do the following
		// atomic { if (iCurrentIDFC==d) iCurrentIDFC=0; }
		// We must guarantee that the following access is observed before the state change in RunIDFCStateChange()
		// We assume the latter has full barrier semantics to guarantee this.
		iCurrentIDFC = d;

		// transition to running state
		// 002g->00Cn, ok=TRUE
		// 008n->00Cn, ok=TRUE
		// 00An->00Cn, ok=TRUE
		// XXYY->XX00, ok=FALSE
		// XX00->0000, ok=FALSE
		// other starting states invalid
		TUint32 orig = d->RunIDFCStateChange() >> 5;
		NKern::EnableAllInterrupts();
		if (orig==1 || orig==4 || orig==5)
			{
			(*f)(p);

			// transition to idle state or rerun if necessary
			// first swap iCurrentIDFC with 0 - if original value != d, don't touch d again, return 0xFFFFFFFF
			// 00Cn->0000
			// 00En->008n
			// 006n->006n
			// XXCn->XX00
			// XXEn->XX00
			// XX6n->XX00
			// other starting states invalid
			// return original state
			NKern::DisableAllInterrupts();
			TUint32 orig = d->EndIDFCStateChange(this);
			if ((orig>>5)==7)
				{
				iDfcs.Add(d);
#ifdef _DEBUG
				TUint32 st8 = DFC_STATE(d) & 0xFF;
				if (st8 != (0x80|iCpuNum))
					__crash();
#endif
				continue;
				}
			else if ((orig>>5)==3)
				{
				TheScheduler.iIdleSpinLock.LockOnly();
				// 006n->002g
				// XX6n->XX00
				orig = d->EndIDFCStateChange2();
				if ((orig>>5)==3)
					TheScheduler.iIdleDfcs.Add(d);
				TheScheduler.iIdleSpinLock.UnlockOnly();
				}
			NKern::EnableAllInterrupts();
			if (tied && orig<0x10000)
				tied->EndTiedEvent(); // if we set iCurrentIDFC back to 0, we release the tied thread/group
			}
		else
			{
			iCurrentIDFC = 0;
			if (tied)
				tied->EndTiedEvent();		// IDFC cancelled so release tied thread/group
			}
		}
	iDfcPendingFlag = 0;
	BTrace0(BTrace::ECpuUsage, BTrace::EIDFCEnd);
	iInIDFC = 0;
	__KTRACE_OPT(KSCHED2,DEBUGPRINT("~"));
	}


/******************************************************************************
 * Kernel-side asynchronous request DFCs
 ******************************************************************************/

EXPORT_C TAsyncRequest::TAsyncRequest(TDfcFn aFunction, TDfcQue* aDfcQ, TInt aPriority)
	: TDfc(aFunction, this, aDfcQ, aPriority), iCompletionObject(0), iCancel(0), iResult(0)
	{
	}


EXPORT_C void TAsyncRequest::Send(TDfc* aCompletionDfc)
	{
	__NK_ASSERT_DEBUG(!iCompletionObject);
	iCancel = EFalse;
	iCompletionObject = (TAny*)((TLinAddr)aCompletionDfc|1);
	TDfc::Enque();
	}


EXPORT_C void TAsyncRequest::Send(NFastSemaphore* aCompletionSemaphore)
	{
	__NK_ASSERT_DEBUG(!iCompletionObject);
	iCancel = EFalse;
	iCompletionObject = aCompletionSemaphore;
	TDfc::Enque();
	}


EXPORT_C TInt TAsyncRequest::SendReceive()
	{
	NFastSemaphore signal;
	NKern::FSSetOwner(&signal, 0);
	Send(&signal);
	NKern::FSWait(&signal);
	return iResult;
	}


EXPORT_C void TAsyncRequest::Cancel()
	{
	iCancel = ETrue;
	if(TDfc::Cancel())
		Complete(KErrCancel);
	}


EXPORT_C void TAsyncRequest::Complete(TInt aResult)
	{
	TLinAddr signal = (TLinAddr)__e32_atomic_swp_ord_ptr(&iCompletionObject, 0);
	if(signal)
		{
		iResult = aResult;
		if(signal&1)
			((TDfc*)(signal&~1))->Enque();
		else
			NKern::FSSignal((NFastSemaphore*)signal);
		}
	}