kernel/eka/nkern/dfcs.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 14 Sep 2010 23:56:21 +0300
branchRCL_3
changeset 45 9e2d4f7f5028
parent 0 a41df078684a
permissions -rw-r--r--
Revision: 201035 Kit: 201035

// Copyright (c) 1998-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\nkern\dfcs.cpp
// DFCs
// 
//

// NThreadBase member data
#define __INCLUDE_NTHREADBASE_DEFINES__

// TDfc member data
#define __INCLUDE_TDFC_DEFINES__

#include "nk_priv.h"


/** 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), iFunction(aFunction), iDfcQ(NULL)
	{
	iPriority=0xff;
	iSpare1=0;
	iOnFinalQ=FALSE;
	iQueued=FALSE;
	}


/** 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)
	: iPtr(aPtr), iFunction(aFunction), iDfcQ(NULL)
	{
	__NK_ASSERT_DEBUG((TUint)aPriority<(TUint)KNumDfcPriorities);
	iPriority=TUint8(aPriority);
	iSpare1=0;
	iOnFinalQ=FALSE;
	iQueued=FALSE;
	}


/** 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)
	: iPtr(aPtr), iFunction(aFunction), iDfcQ(aDfcQ)
	{
	__NK_ASSERT_DEBUG((TUint)aPriority<(TUint)KNumDfcPriorities);
	iPriority=TUint8(aPriority);
	iSpare1=0;
	iOnFinalQ=FALSE;
	iQueued=FALSE;
	}


/** 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)
	{}


#ifndef __DFC_MACHINE_CODED__

/** 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_WITH_MESSAGE_DEBUG(  NKern::CurrentContext()!=NKern::EThread  ||  TheScheduler.iKernCSLocked,"Do not call from thread with preemption enabled","TDfc::Add");
	__ASSERT_WITH_MESSAGE_DEBUG(  IsIDFC() || iDfcQ != NULL, "DFC queue not set", "TDfc::Add");
#ifdef __WINS__
	__NK_ASSERT_ALWAYS(Interrupt.InInterrupt() || TheScheduler.iKernCSLocked);
#endif
	return RawAdd();
	}


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

	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();

	// make sure DFC not already queued
	TBool ok = !TestAndSetQueued();
	if (ok)
		{
		TheScheduler.iDfcs.Add(this);
		TheScheduler.iDfcPendingFlag=1;
		}

	NKern::RestoreInterrupts(irq);
	return ok;
	}


/** 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.

	@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 only from IDFC or thread with preemption disabled.
	@pre Do not call from ISR or thread with preemption enabled.

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

	// Check not already queued and then mark queued to prevent ISRs touching this DFC
	TBool ok = !TestAndSetQueued();
	if (ok)
		DoEnqueFinal();
	return ok;
	}

void TDfc::DoEnqueFinal()
//
// Add a DFC to its final queue. Assumes DFC not currently queued.
// Enter and return with kernel locked.
//
	{
	iOnFinalQ=TRUE;
	iDfcQ->Add(this);
	NThreadBase* pT=iDfcQ->iThread;
	if (pT->iNState==NThreadBase::EWaitDfc)
		pT->CheckSuspendThenReady();
	}

void TDfcQue::ThreadFunction(TAny* aDfcQ)
	{
	TDfcQue& q=*(TDfcQue*)aDfcQ;
	NThreadBase* pC=TheScheduler.iCurrentThread;
	FOREVER
		{
		NKern::Lock();
		if (q.IsEmpty())
			{
			pC->iNState=NThreadBase::EWaitDfc;
			TheScheduler.Remove(pC);
			RescheduleNeeded();
			NKern::Unlock();
			}
		else
			{
			TDfc& d=*q.First();
			q.Remove(&d);
			d.iOnFinalQ=FALSE;
			d.iQueued=FALSE;
			NKern::Unlock();
			(*d.iFunction)(d.iPtr);
			}
		}
	}


/** Cancels an IDFC or DFC.

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

	@return	TRUE	if the DFC was actually dequeued by this call. In that case
					it is guaranteed that the DFC will not execute until it is
					queued again.
			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.

	@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.
 */
EXPORT_C TBool TDfc::Cancel()
	{
	CHECK_PRECONDITIONS(MASK_NOT_ISR,"TDfc::Cancel");
	NKern::Lock();
	TBool ret = iQueued;
	if (iQueued)	// ISRs can't affect this test since they can't de-queue a DFC or IDFC
		{
		if (!iOnFinalQ)	// OK to check this with interrupts enabled since interrupts can't change it
			{
			// Must disable interrupts to protect the pending queue
			TInt irq=NKern::DisableAllInterrupts();
			SDblQueLink::Deque();
			NKern::RestoreInterrupts(irq);
			}
		else
			{
			// Final queues can't be modified by interrupts
			iDfcQ->Remove(this);
			iOnFinalQ=FALSE;
			}
		iQueued=FALSE;	// must be done last
		}
	NKern::Unlock();
	return ret;
	}
#endif

/** 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.
    
	@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 either in a thread or an IDFC context.
	@pre	Do not call from an ISR.
 */
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 (IsIDFC())
		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=NKern::DisableAllInterrupts();

	// make sure DFC not already queued
	TBool ok = !TestAndSetQueued();
	if (ok)
		TheScheduler.iIdleDfcs.Add(this);

	NKern::RestoreInterrupts(irq);
	return ok;
	}


TUint32 NKern::IdleGenerationCount()
	{
	return TheScheduler.iIdleGenerationCount;
	}


void NKern::Idle()
	{
	TInt irq = NKern::DisableAllInterrupts();
#ifdef _DEBUG
	if (!TheScheduler.iIdleDfcs.IsEmpty() && TheScheduler.iDelayedQ.IsEmpty())
#else
	if (!TheScheduler.iIdleDfcs.IsEmpty())
#endif
		{
		++TheScheduler.iIdleGenerationCount;
		TheScheduler.iDfcs.MoveFrom(&TheScheduler.iIdleDfcs);
		TheScheduler.iDfcPendingFlag=1;
		NKern::RestoreInterrupts(irq);
		return;
		}
	NKern::RestoreInterrupts(irq);
	NKIdle(0);
	}


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

#ifndef __SCHEDULER_MACHINE_CODED__
void TScheduler::QueueDfcs()
//
// Enter with interrupts off and kernel locked
// Leave with interrupts off and kernel locked
//
	{
	iInIDFC = TRUE;
	BTrace0(BTrace::ECpuUsage,BTrace::EIDFCStart);
	FOREVER
		{
		// remove from pending queue with interrupts disabled
		TDfc* d=(TDfc*)iDfcs.GetFirst();
		if (!d)
			break;
		NKern::EnableAllInterrupts();
		if (d->IsIDFC())
			{
			d->iQueued=FALSE;
			(*d->iFunction)(d->iPtr);
			}
		else
			d->DoEnqueFinal();
		NKern::DisableAllInterrupts();
		}
	iDfcPendingFlag = FALSE;
	BTrace0(BTrace::ECpuUsage,BTrace::EIDFCEnd);
	iInIDFC = FALSE;
	}
#endif


/******************************************************************************
 * 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);
		}
	}