networkcontrol/ipnetworklayer/src/IPProtoTierManager.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 26 Jan 2010 15:23:49 +0200
changeset 0 af10295192d8
permissions -rw-r--r--
Revision: 201004

// 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:
// IPProto Tier Manager
// 
//

/**
 @file
 @internalComponent
*/

#include <comms-infras/ss_log.h>
#include <comms-infras/ss_coreprstates.h>
#include <comms-infras/ss_corepractivities.h>
#include <comms-infras/coretiermanagerstates.h>
#include <comms-infras/coretiermanageractivities.h>
#include <comms-infras/coretiernotificationactivity.h>
#include <comms-infras/coretiernotificationstates.h>
#include "IPProtoTierManager.h"
#include "IPProtoTierManagerSelector.h"
#include <comms-infras/ss_metaconnprov.h>
#include <in_sock.h>

#ifdef SYMBIAN_TRACE_ENABLE
#define KIPProtoTierMgrTag KESockMetaConnectionTag
// _LIT8(KIPProtoTierMgrSubTag, "ipprototiermgr");
#endif // SYMBIAN_TRACE_ENABLE

using namespace Messages;
using namespace MeshMachine;
using namespace ESock;
using namespace NetStateMachine;
using namespace TMActivities;
using namespace CorePanics;

//-=========================================================
// Panics
#ifdef _DEBUG
_LIT (KIpProtoTMPanic,"IpProtoTMPanic");
#endif


//This transition creates a link between legacy NetMCpr & legacy CLinkPrefsSelector
DEFINE_SMELEMENT(IpProtoTMStates::TSelectProvider, NetStateMachine::MStateTransition, IpProtoTMStates::TContext)
void IpProtoTMStates::TSelectProvider::DoL()
	{
	__ASSERT_DEBUG(iContext.iNodeActivity, User::Panic(KIpProtoTMPanic, KPanicNoActivity));
	CSelectProviderActivity* act = static_cast<CSelectProviderActivity*>(iContext.iNodeActivity);
	
	//Create selector
	act->iSelector = iContext.Node().DoCreateProviderSelectorL(message_cast<TCFSelector::TSimpleSelect>(iContext.iMessage).iSelectionPrefs);
	
	TBool isItLegacyMcpr = EFalse;
	
	//First assumption: At this layer (IpProto) all SelectProvider requests are always comming from MCprs.
#if !defined(__GCCXML__)
	CMetaConnectionProviderBase& mcpr = mnode_cast<CMetaConnectionProviderBase>(address_cast<TNodeId>(iContext.iSender).Node());
#else
	CMetaConnectionProviderBase& mcpr =
		reinterpret_cast<CMetaConnectionProviderBase&>(address_cast<TNodeId>(iContext.iSender).Node());
#endif

	//Second assumption: All MCprs at an upper layer (above IpProto) use the Instance() to indicate that they are legacy.
	if (mcpr.ProviderInfo().Instance() !=0) 
		{ //We have a legacy provider above us
		ASSERT(mcpr.ProviderInfo().TierId() == TUid::Uid(KAfInet)); //Here we can partially test the second assumption

		ASimpleSelectorBase& baseselector = static_cast<ASimpleSelectorBase&>(*act->iSelector);
		if (baseselector.TypeId() == CLinkPrefsSelector::KUid)
			{
			CLinkPrefsSelector& selector = static_cast<CLinkPrefsSelector&>(*act->iSelector);
			selector.iLegacyMCpr = &mcpr;
			isItLegacyMcpr = ETrue;
			}
		}
	
	if (!isItLegacyMcpr)
		{
		//A 399 selection is ongoing. Note: this code is intentionally not generic and needs to be reworking.
		AIpProtoSelectorBase& currentSelector = static_cast<AIpProtoSelectorBase&>(*act->iSelector);
		currentSelector.iNetworkMCpr = &mcpr;
		
		if (iContext.ActivityId() == ESock::ECFActivityConnectionStartRecovery)
			{
			currentSelector.iReselection = ETrue;
			}
		}
	
	ISelectionNotify selectionNotify(act,act->InterfaceVTable());
	act->iSelector->SelectL(selectionNotify);
	//Do not do any further processing here as selectors may be returning synchronously (via ISelectionNotify).
	}

DEFINE_SMELEMENT(IpProtoTMStates::TSelectProviderConnPrefList, NetStateMachine::MStateTransition, IpProtoTMStates::TContext)
void IpProtoTMStates::TSelectProviderConnPrefList::DoL()
	{
	__ASSERT_DEBUG(iContext.iNodeActivity, User::Panic(KIpProtoTMPanic, KPanicNoActivity));
	CSelectProviderActivity* act = static_cast<CSelectProviderActivity*>(iContext.iNodeActivity);

	//Create selector
	//If a ConnPrefList is being used we know what kind of Selector to use
	//(as only one kind supports it) so we can create it without calling
	//the intermidiate DoCreateProviderSelectorL
	act->iSelector = new (ELeave) CIpProtoProviderSelectorConnPrefList(message_cast<TCFSelector::TSelect>(iContext.iMessage).iConnPrefList); 

	//First assumption: At this layer (IpProto) all SelectProvider requests are always comming from MCprs.
#if !defined(__GCCXML__)
	CMetaConnectionProviderBase& mcpr = mnode_cast<CMetaConnectionProviderBase>(address_cast<TNodeId>(iContext.iSender).Node());
#else
	CMetaConnectionProviderBase& mcpr =
		reinterpret_cast<CMetaConnectionProviderBase&>(address_cast<TNodeId>(iContext.iSender).Node());
#endif
		
	if (mcpr.ProviderInfo().Instance() !=0) 
		{ //We have a legacy provider above us
		ASSERT(mcpr.ProviderInfo().TierId() == TUid::Uid(KAfInet)); //Here we can partially test the second assumption

		ASimpleSelectorBase& baseselector = static_cast<ASimpleSelectorBase&>(*act->iSelector);
		if (baseselector.TypeId() == CLinkPrefsSelector::KUid)
			{
			CLinkPrefsSelector& selector = static_cast<CLinkPrefsSelector&>(*act->iSelector);
			selector.iLegacyMCpr = &mcpr;
			}
		}
	else
		{
		//A 399 selection is ongoing. Note: this code is intentionally not generic and needs to be reworking.
		AIpProtoSelectorBase& currentSelector = static_cast<AIpProtoSelectorBase&>(*act->iSelector);
		currentSelector.iNetworkMCpr = &mcpr;
		
		if (iContext.ActivityId() == ESock::ECFActivityConnectionStartRecovery)
			{
			currentSelector.iReselection = ETrue;
			}
		}
	
	ISelectionNotify selectionNotify(act,act->InterfaceVTable());
	act->iSelector->SelectL(selectionNotify);
	//Do not do any further processing here as selectors may be returning synchronously (via ISelectionNotify).
	}

namespace IpProtoTMSelectProviderActivity
{ //Simple parallel activity provider selection, waits untill selection completes (via ISelectionNotify), then gets destroyed
DECLARE_DEFINE_CUSTOM_NODEACTIVITY(ECFActivitySelect, IpProtoTMSelectProvider, TCFSelector::TSimpleSelect, CSelectProviderActivity::NewL)
	FIRST_NODEACTIVITY_ENTRY(TMStates::TAwaitingSelectProvider, MeshMachine::TNoTag)
	NODEACTIVITY_ENTRY(KNoTag, IpProtoTMStates::TSelectProvider, CoreStates::TNeverAccept, MeshMachine::TNoTag)
	LAST_NODEACTIVITY_ENTRY(KNoTag, MeshMachine::TDoNothing)
NODEACTIVITY_END()
}

namespace IpProtoTMSelectProviderConnPrefListActivity
{ //Simple parallel activity provider selection, waits untill selection completes (via ISelectionNotify), then gets destroyed
DECLARE_DEFINE_CUSTOM_NODEACTIVITY(ECFActivitySelect, IpProtoTMSelectConnPrefListProvider, TCFSelector::TSelect, CSelectProviderActivity::NewL)
	FIRST_NODEACTIVITY_ENTRY(TMStates::TAwaitingSelectProviderSuper, MeshMachine::TNoTag)
	NODEACTIVITY_ENTRY(KNoTag, IpProtoTMStates::TSelectProviderConnPrefList, CoreStates::TNeverAccept, MeshMachine::TNoTag)
	LAST_NODEACTIVITY_ENTRY(KNoTag, MeshMachine::TDoNothing)
NODEACTIVITY_END()
}

namespace IpProtoTMActivities
{
DECLARE_DEFINE_ACTIVITY_MAP(stateMap)
	ACTIVITY_MAP_ENTRY(IpProtoTMSelectProviderActivity,IpProtoTMSelectProvider)
	ACTIVITY_MAP_ENTRY(IpProtoTMSelectProviderConnPrefListActivity,IpProtoTMSelectConnPrefListProvider)
ACTIVITY_MAP_END_BASE(TMActivities, coreTMActivities)
}


CIPProtoTierManager* CIPProtoTierManager::NewL(ESock::CTierManagerFactoryBase& aFactory)
	{
 	return new (ELeave) CIPProtoTierManager(aFactory, IpProtoTMActivities::stateMap::Self());
	}

CIPProtoTierManager::CIPProtoTierManager(ESock::CTierManagerFactoryBase& aFactory,
                                         const MeshMachine::TNodeActivityMap& aActivityMap)
:	CCoreTierManager(aFactory,aActivityMap)
	{
	LOG_NODE_CREATE(KIPProtoTierMgrTag, CIPProtoTierManager);
	}

CIPProtoTierManager::~CIPProtoTierManager()
	{
	LOG_NODE_DESTROY(KIPProtoTierMgrTag, CIPProtoTierManager);
	}

ESock::MProviderSelector* CIPProtoTierManager::DoCreateProviderSelectorL(const Meta::SMetaData& aSelectionPreferences)
	{
    return TIpProtoProviderSelectorFactory::NewSelectorL(aSelectionPreferences);
	}

void CIPProtoTierManager::ReceivedL(const TRuntimeCtxId& aSender, const TNodeId& aRecipient, TSignatureBase& aMessage)
	{
   	TNodeContext<CIPProtoTierManager> ctx(*this, aMessage, aSender, aRecipient);
   	CCoreTierManager::Received(ctx);
    User::LeaveIfError(ctx.iReturn);
	}