commsfwsupport/commselements/nodemessages/inc/nm_node.h
author Fionntina Carville <fionntinac@symbian.org>
Wed, 17 Nov 2010 16:18:58 +0000
branchRCL_3
changeset 88 077156ad1d4e
parent 29 9644881fedd0
permissions -rw-r--r--
Bug 2675. Take default commdb from ipconnmgmt instead.

// 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 "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:
//

/**
 @file
 @publishedPartner
 @released
*/

#ifndef SYMBIAN_NM_NODE_H
#define SYMBIAN_NM_NODE_H

#include <elements/nm_address_internal.h>
#include <elements/nm_log.h>
#include <elements/interfacetable.h>


#ifdef _DEBUG
// Panic category for "absolutely impossible!" vanilla ASSERT()-type panics from this module
// (if it could happen through user error then you should give it an explicit, documented, category + code)
_LIT(KSpecAssert_ElemNodeMessNodH, "ElemNodeMessNodH");
#endif


namespace Messages
{

//-=========================================================
//
//MNodeId
//
//-=========================================================
class MNodeId
	{
public:
	virtual const TNodeId& NodeId() const = 0;
	};

class TSignatureBase;
class ANode : public MNodeId
	{
public:
    //if Received returns != KErrNone than the dispatcher will send a TError to the
    //originator of the message
	virtual void ReceivedL(const TRuntimeCtxId& aSender, const TNodeId& aRecipient, TSignatureBase& aMessage) = 0;

	IMPORT_C virtual NetInterfaces::TInterfaceControl* FetchNodeInterfaceControlL(TInt aInterfaceId);
	IMPORT_C TAny* FetchNodeInterfaceL(TInt aInterfaceId);
	};

//-=========================================================
//
//ANodeId
//
//-=========================================================
class ANodeId
	{
public:
	const TNodeId& Id() const
		{
		return iNodeId;
		}

protected:
	explicit ANodeId(ANode& aNode) :
		iNodeId(aNode)
		{
		}

	explicit ANodeId(TAny* aNode) :
		iNodeId(aNode)
		{
		}

	void UnregisterSelf()
		{
		iNodeId.UnregisterSelf();
		}

private:
    TNodeOid iNodeId;
	};


//-=========================================================
//
//ASimpleNodeIdBase - aggregation base class for simple nodes able to receive messages
//
//-=========================================================
class ASimpleNodeIdBase : protected ANode, public ANodeId
	{
protected:
    explicit ASimpleNodeIdBase()
    :	ANodeId(*(static_cast<ANode*>(this)))
        {
        }

    IMPORT_C virtual const TNodeId& NodeId() const;
	};

//-=========================================================
//
//ANodeBase
//
//-=========================================================
class ANodeBase : protected NetInterfaces::TInterfaceControl,
                  public ITFHIERARCHY_1(ANodeBase, ANodeBase)
	{
public:
	typedef ITFHIERARCHY_1(ANodeBase, ANodeBase) TIfStaticFetcherNearestInHierarchy;

public:
	static const TInt KInterfaceId = 10005; //Normally a UID - 10005 for compatibility reasons
	IMPORT_C void ReturnInterfacePtrL(ANodeBase*& aInterface);

public:
    IMPORT_C virtual RNodeInterface* NewClientInterfaceL(const TClientType& aClientType, TAny* aClientInfo = NULL);
    IMPORT_C virtual RNodeInterface* AddClientL(const TNodeId& aClientId, const TClientType& aClientType, TAny* aClientInfo = NULL);
    IMPORT_C void RemoveClient(RNodeInterface& aClient);
    IMPORT_C void RemoveClient(TInt aIndex);
    inline const TNodeId& Id() const
    	{
    	return iSelfInterface.RecipientId();
    	}

    template<class TTYPEMATCHPOLICY>
    inline TClientIter<TTYPEMATCHPOLICY> GetClientIter(const TClientType& aIncClientType, const TClientType& aExcClientType = TClientType::NullType()) const
    	{
    	return TClientIter<TTYPEMATCHPOLICY>(iClients, aIncClientType, aExcClientType);
    	}

	template<class TTYPEMATCHPOLICY>
    inline RNodeInterface* GetFirstClient(const TClientType& aIncClientType, const TClientType& aExcClientType = TClientType::NullType()) const
        {
        return GetClientIter<TTYPEMATCHPOLICY>(aIncClientType, aExcClientType)[0];
        }

	template<class TTYPEMATCHPOLICY>
    inline TInt PostToClients(const TRuntimeCtxId& aPostFrom, const TSignalBase& aMessage, const TClientType& aIncClientType, const TClientType& aExcClientType = TClientType::NullType(), TUint32 aFlagsToSet = 0, TUint32 aFlagsToClear = 0) const
        {
        TClientIter<TTYPEMATCHPOLICY> iter = GetClientIter<TTYPEMATCHPOLICY>(aIncClientType, aExcClientType);
        return PostToClients(iter, aPostFrom, aMessage, aFlagsToSet, aFlagsToClear);
        }

	template<class TTYPEMATCHPOLICY>
    inline TUint CountClients(const TClientType& aIncClientType, const TClientType& aExcClientType = TClientType::NullType()) const
        {
        TClientIter<TTYPEMATCHPOLICY> iter = GetClientIter<TTYPEMATCHPOLICY>(aIncClientType, aExcClientType);
        return CountClients(iter);
        }

    IMPORT_C RNodeInterface* FindClient(const TRuntimeCtxId& aId) const;
    IMPORT_C RNodeInterface* FindClientL(const TRuntimeCtxId& aId) const; //Must not return NULL
    RNodeInterface& SelfInterface()
    	{
    	__ASSERT_DEBUG(iSelfInterface.IsOpen(), User::Panic(KSpecAssert_ElemNodeMessNodH, 1)); //TODO - add panic code
    	return iSelfInterface;
    	}

protected:
	IMPORT_C explicit ANodeBase(const TNodeId& aNodeId);
	IMPORT_C ~ANodeBase();
	IMPORT_C TInt PostToClients(TClientIterBase& aClientIterator, const TRuntimeCtxId& aPostFrom, const TSignalBase& aMessage, TUint32 aFlagsToSet, TUint32 aFlagsToClear) const;
	IMPORT_C TUint CountClients(TClientIterBase& aClientIterator) const;
	IMPORT_C RNodeInterface* DoFindClient(const TRuntimeCtxId& aId, TInt& aFoundAtIndex) const;

protected:
    RPointerArray<RNodeInterface> iClients;
//TODO[PROD] - could it be combined with the Id() somehow to optimise for size?
   	RNodeInterface iSelfInterface; //For posting messages to ourselves + necessary for requests origiated from the same node at the MM level
	};

//-=========================================================
//
//ANodeIdBase - aggregation base class for nodes able to receive messages
//
//-=========================================================
class ANodeIdBase : protected ANode, public ANodeId, public ANodeBase
	{
protected:
    explicit ANodeIdBase()
    :   ANodeId(static_cast<ANode&>(*this)),
    	ANodeBase(ANodeId::Id())
        {
        }

    IMPORT_C virtual const TNodeId& NodeId() const;
	};

//-=========================================================
//
//mnode_cast
//
//-=========================================================
template<class NODE>
NODE* mnode_cast(Messages::ANode* aNode)
    {
   	return static_cast<NODE*>(aNode);
    }

template <class NODE>
NODE& mnode_cast(Messages::ANode& aNode)
    {
    return static_cast<NODE&>(aNode);
    }


} //namespace Messages


#endif //SYMBIAN_NM_NODE_H