telephonyserverplugins/simtsy/src/CSimMbmsPacketContext.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 11 May 2010 17:18:08 +0300
branchRCL_3
changeset 10 4284d6390a82
parent 0 3553901f7fa8
child 19 630d2f34d719
permissions -rw-r--r--
Revision: 201019 Kit: 201019

// Copyright (c) 2008-2010 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:
// This file contains the implementation of the Similator TSY MBMS Packet Context functionality.  
// 
//

/**
 @file
*/

#include "Simlog.h"
#include "CSimPhone.h"
#include "CSimMbmsPacketContext.h"
#include "CSimPacketQoS.h"
#include "utils.h"
#include <e32math.h>
#include "CSimPacketService.h"
#include "CSimContextHelper.h"

CSimMbmsPacketContext* CSimMbmsPacketContext::NewL(CSimPhone* aPhone, CSimPacketService* aPacketService, const TDesC& aContextName)
/**
* Standard two phase constructor
*
* @param aPacketService Pointer to the Packet Service object (CSimPacketService)
* @param aContextName name  for this packet context
* @return CSimMbmsPacketContext pointer to the packet context object created.
* @leave Leaves if no memory or object is not created for any reason.
*/
	{
	CSimMbmsPacketContext* packetContext = new(ELeave) CSimMbmsPacketContext(aPhone, aPacketService, aContextName);
	CleanupStack::PushL(packetContext);
	packetContext->ConstructL();
	CleanupStack::Pop();
	return packetContext;
	}

CSimMbmsPacketContext::CSimMbmsPacketContext(CSimPhone* aPhone, CSimPacketService* aPacketService, const TName& aContextName)
	:CSimPacketContext(aPhone,aPacketService,aContextName),iContextName(aContextName), iContextType(TPacketDataConfigBase::KConfigMBMS),
	iPhone(aPhone),iPacketService(aPacketService)
/**
* Trivial Constructor.  Initialises all the data members
*
* @param aPacketService Pointer to the Packet Service object (CSimPacketService)
* @param aContextName name  for this packet context
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: Entered constructor");
	iNotifyConfigMBMS.iNotifyPending = EFalse;
	iNotifyStatusChange.iNotifyPending = EFalse;
	}

void CSimMbmsPacketContext::ConstructL()
/**
* Second phase of the 2-phase constructor.
* Constructs all the member data and retrieves all the data from the config file specific to this class.
*
* @leave Leaves no memory or any data member does not construct for any reason.
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: Entered constructL function");
	CSimPacketContext::ConstructL();
	
	iMbmsSetConfigTimer = CSimTimer::NewL(iPhone);
	iMbmsContextTimer = CSimTimer::NewL(iPhone);
	iMbmsUpdateSessionTimer = CSimTimer::NewL(iPhone);
	iMbmsContextInitTimer = CSimTimer::NewL(iPhone);
	iMbmsGetSessionTimer = CSimTimer::NewL(iPhone);
	
	iMbmsContextConfigParams = new (ELeave) CArrayFixFlat<TMbmsConfigParams>(KNumberofConfigs);
	iSessionIdList = new (ELeave) CArrayFixFlat<TUint>(1);
	iMbmsSessionList = new CArrayPtrFlat<CPcktListReadAllAttempt>(1);
		
	const CTestConfigItem* item=NULL;
	TInt ret=KErrNone;

	/* Reading the MBMS Context configuration parameters */
	TUint count=CfgFile()->ItemCount(KMBMSBroadcastMonitorList); //< Retrieves the Count of MBMS related information

	for(TInt i=0;i<count;i++)
		{
		item = CfgFile()->Item(KMBMSBroadcastMonitorList,i); //< Retrieves the MBMS related information

		TInt mnc =0; 
		TInt mcc =0;
		TInt serviceId =0;
		TInt accessBearer=0;
		TInt serviceMode =0;
		TInt activateCode =0;
		TInt servicePriority=0;
		if(!item)
			break;

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,1,mcc);
		if(ret!=KErrNone)
			{
			break;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,2,mnc);
		if(ret!=KErrNone)
			{
			break;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,3,serviceId);
		if(ret!=KErrNone)
			{
			break;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,4,accessBearer);
		if(ret!=KErrNone)
			{
			break;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,5,serviceMode);
		if(ret!=KErrNone)
			{
			break;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,7,activateCode);
		if(ret!=KErrNone)
			{
			break;
			}

		ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,8,servicePriority);
		if(ret!=KErrNone)
			{
			break;
			}

		TMbmsConfigParams mbmsConfigParams;

		mbmsConfigParams.iMCC = mcc;
		mbmsConfigParams.iMNC = mnc;
		mbmsConfigParams.iServiceId = serviceId;
		mbmsConfigParams.iAccessBearer = reinterpret_cast<TMbmsScope&> (accessBearer);
		mbmsConfigParams.iServiceMode = reinterpret_cast<TMbmsServiceMode&> (serviceMode);
		mbmsConfigParams.iServicePriority = reinterpret_cast<TMbmsServicePriority&> (servicePriority);
		mbmsConfigParams.iActivateCode = activateCode;

		iMbmsContextConfigParams->AppendL(mbmsConfigParams);
		}
	
	//iSessionIdList->AppendL(iNumOfSessionId);
	iSessionIdList->InsertL(0,iNumOfSessionId);
	
	/* Reading the list of session ids mentioned in the configuration file*/
	/*item = CfgFile()->Item(KMBMSSessionIdList,0); //< Retrieves the MBMS related information
	
	if(item != NULL)
		{
		TInt sessionId =0;
		TInt numOfSessions =0;
		
		ret = CTestConfig::GetElement(item->Value(),KStdDelimiter,0,numOfSessions);
		iSessionIdList->AppendL(numOfSessions);
		
		for(TInt j=1;j<=numOfSessions;j++)
			{
			ret=CTestConfig::GetElement(item->Value(),KStdDelimiter,j,sessionId);
			if(ret!=KErrNone)
				{
				break;
				}
			else
				{
				iSessionIdList->AppendL(sessionId);
				}
			}
		}*/
	}

void CSimMbmsPacketContext::Init()
/*
 * Initialisation function; Not used in the CSimMbmsPacketContext class.
 * but should be implemented as it is a pure virtual function of MTelObjectTSY. 
 */
	{}
	
CSimMbmsPacketContext::~CSimMbmsPacketContext()
/**
* Trivial Destructor
* Closes all CObject type objects and destroys all other objects created in the ConstructL()
*
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: Entered destructor");

	if (iMbmsSetConfigTimer != NULL)
		{
		delete iMbmsSetConfigTimer;
		}

	if(iMbmsContextTimer != NULL)
		{
		delete iMbmsContextTimer;
		}

	if(iMbmsUpdateSessionTimer != NULL)
		{	
		delete iMbmsUpdateSessionTimer;
		}

	if(iMbmsContextInitTimer)
		{
		delete iMbmsContextInitTimer;
		}

	if(iMbmsGetSessionTimer)
		{
		delete iMbmsGetSessionTimer;
		}
	
	if(iMbmsContextConfigParams !=NULL)
		{
		iMbmsContextConfigParams->Delete(0,iMbmsContextConfigParams->Count());
		delete iMbmsContextConfigParams;
		}

	if(iSessionIdList != NULL)
		{
		iSessionIdList->Delete(0,iSessionIdList->Count());
		delete iSessionIdList;
		}

	if(iMbmsSessionList)
		{
		iMbmsSessionList->ResetAndDestroy();
		}
	delete iMbmsSessionList;

	iPacketService->DecrementMbmsContextCount();
	}

CTelObject* CSimMbmsPacketContext::OpenNewObjectByNameL(const TDesC& /*aName*/)
/**
 * No Qos settings for the MBMS broadcast context 
 * This is not supported and if called, will leave.
 *
 * @param aName name of the QoS object to be opened
 * @leave Leaves with KErrNotSupported if the name is not as expected.
 * @ return NULL.
 */
	{	
	LOGPACKET1("Unexpected call to CSimMbmsPacketContext: OpenNewObjectByName");
	User::Leave(KErrNotSupported);
	return NULL;
	}

CTelObject* CSimMbmsPacketContext::OpenNewObjectL(TDes& /*aNewName*/)
/**
 * No Qos settings for the MBMS broadcast context 
 * This is not supported and if called, will leave.
 *
 * @param aNewName new name of the object created
 * @return NULL
 * @leave Leaves if out of memory.
 */
	{
	LOGPACKET1("Unexpected call to CSimMbmsPacketContext: OpenNewObjectL");
	User::Leave(KErrNotSupported);
	return NULL;
	}

CTelObject::TReqMode CSimMbmsPacketContext::ReqModeL(const TInt aIpc)
/**
* ReqModeL is called from the server's CTelObject::ReqAnalyserL
* in order to check the type of request it has.
* 
* @param aIpc the ipc number that identifies the client request
* @return CTelObject::TReqMode The request mode to be used for this request
* @leave Leaves if not supported by this tsy 
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: ReqModeL");
	CTelObject::TReqMode ret=0;
	switch (aIpc)
		{
		case EPacketContextGetConfig:
		case EPacketContextSetConfig:
		case EPacketContextActivate:
		case EPacketContextDeactivate:
		case EPacketContextDelete:
		case EPacketContextGetStatus:
		case EPacketContextUpdateMbmsSessionList:
		case EPacketContextInitialiseContext:
		case EPacketGetMbmsSessionListPhase1:
		case EPacketGetMbmsSessionListPhase2:
		case EPacketContextGetLastErrorCause:
			break;
						
		case EPacketContextNotifyConfigChanged:
		case EPacketContextNotifyStatusChange:
		case EPacketContextNotifyDataTransferred:
			ret=KReqModeMultipleCompletionEnabled | KReqModeRePostImmediately;
			break;
		default:
			LOGPACKET1("CSimMbmsPacketContext: ReqModeL error, unknown IPC");
			LOGPACKET1("CSimMbmsPacketContext: ReqModeL sending the request to CSimPacketContext");
			ret = CSimPacketContext::ReqModeL(aIpc);
			break;
		}
	return ret;
	}

TInt CSimMbmsPacketContext::RegisterNotification(const TInt aIpc)
/**
* RegisterNotification is called when the server recognises that this notification
* is being posted for the first time on this sub-session object.
* 
* It enables the TSY to "turn on" any regular notification messages that it may receive 
* from the phone
*
* @param aIpc the ipc number that identifies the client request
* @return err KErrNone if fine
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: RegisterNotification called");
	switch (aIpc)
		{
		case EPacketContextNotifyConfigChanged:
		case EPacketContextNotifyStatusChange:
		case EPacketContextNotifyDataTransferred:
			LOGPACKET1("CSimMbmsPacketContext: RegisterNotification");
			return KErrNone;
		default:
			// Unknown or invalid IPC
			LOGPACKET1("CSimMbmsPacketContext: Register error, unknown IPC");
			LOGPACKET1("CSimMbmsPacketContext: Register sending the request to CSimPacketContext");
			return CSimPacketContext::RegisterNotification(aIpc);
		}
	}

TInt CSimMbmsPacketContext::DeregisterNotification(const TInt aIpc)
/**
* DeregisterNotification is called when the server recognises that this notification
* will not be posted again because the last client to have a handle on this sub-session
* object has just closed the handle.
*
* It enables the TSY to "turn off" any regular notification messages that it may 
* receive from the phone
*
* @param aIpc the ipc number that identifies the client request
* @return err KErrNone if fine
*/
	{	
	LOGPACKET1("CSimMbmsPacketContext: DeregisterNotification called");
	switch (aIpc)
		{
		case EPacketContextNotifyConfigChanged:
		case EPacketContextNotifyStatusChange:
		case EPacketContextNotifyDataTransferred:
			LOGPACKET1("CSimMbmsPacketContext: DeregisterNotification");
			return KErrNone;
		default:
			// Unknown or invalid IPC
			LOGPACKET1("CSimMbmsPacketContext: Deregister error, unknown IPC");
			LOGPACKET1("CSimMbmsPacketContext: Deregister sending the request to CSimPacketContext");
			return CSimPacketContext::DeregisterNotification(aIpc);
		}
	}

TInt CSimMbmsPacketContext::NumberOfSlotsL(const TInt aIpc)
/**
* NumberOfSlotsL is called by the server when it is registering a new notification
* It enables the TSY to tell the server how many buffer slots to allocate for
* "repost immediately" notifications that may trigger before clients collect them
*
* @param aIpc the ipc number that identifies the client request
* @return err KErrNone if fine
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: NumberOfSlotsL called");
	TInt numberOfSlots=1;
	switch (aIpc)
		{
		case EPacketContextNotifyConfigChanged:
		case EPacketContextNotifyStatusChange:
		case EPacketContextNotifyDataTransferred:
			LOGPACKET1("CSimMbmsPacketContext: Registered with 5 slots");
			numberOfSlots=5;
			break;
		default:
			// Unknown or invalid IPC
			LOGPACKET1("CSimMbmsPacketContext: Number of Slots error, unknown IPC");
			LOGPACKET1("CSimMbmsPacketContext: Number of Slots: sending the request to CSimPacketContext");
			return CSimPacketContext::NumberOfSlotsL(aIpc);
		}  
	return numberOfSlots;
	}

TInt CSimMbmsPacketContext::ExtFunc(const TTsyReqHandle aTsyReqHandle,const TInt aIpc,
							  const TDataPackage& aPackage)
/**
* ExtFunc is called by the server when it has a "extended", i.e. non-core ETel request
* for the TSY to process.
* A request handle, request type and request data are passed to the TSY
* 
* @param aTsyReqHandle  The request handle for completing the request 
* @param aIpc Ipc representing the request
* @param aPackage any data associated with the request
* @return err KErrNone if request completes ok
*/
	{
	//< if this context has been deleted then the client can not do anything with it
	//< if the client wants to use this class, the client must open a new object by 
	//< name with the name for this context and this will reactivate the context.
	if(iDeleted && (aIpc != EPacketContextGetStatus))  
		{
		ReqCompleted(aTsyReqHandle,KErrNotReady);
		return KErrNone;
		}
	LOGPACKET1("CSimMbmsPacketContext: ExtFunc Called");
	TAny* dataPtr=aPackage.Ptr1();
	TAny* dataPtr2=aPackage.Ptr2();

	switch (aIpc)
		{
		case EPacketContextGetConfig:
			return GetConfig(aTsyReqHandle, aPackage.Des1n());
		case EPacketContextGetConfigCancel:
			return GetConfigCancel(aTsyReqHandle);
		case EPacketContextSetConfig:
			return SetConfig(aTsyReqHandle, aPackage.Des1n());
		case EPacketContextSetConfigCancel:
			return SetConfigCancel(aTsyReqHandle);
		case EPacketContextActivate:
			{	//< Ensures that theres only one context active as only one active context is 
			//<permitted at any one time
			if(iIsActive)
				{
				LOGPACKET1("Trying to activate an already activated context");
				ReqCompleted(aTsyReqHandle,KErrEtelCallAlreadyActive);
				return KErrNone;
				}
			else
				return Activate(aTsyReqHandle);
			}
		case EPacketContextActivateCancel:
			return ActivateCancel(aTsyReqHandle);
		case EPacketContextDeactivate:
			return Deactivate(aTsyReqHandle);
		case EPacketContextDeactivateCancel:
			return DeactivateCancel(aTsyReqHandle);
		case EPacketContextDelete:
			return Delete(aTsyReqHandle);
		case EPacketContextDeleteCancel:
			return DeleteCancel(aTsyReqHandle);
		case EPacketContextGetLastErrorCause:
			return GetLastErrorCause(aTsyReqHandle,
			REINTERPRET_CAST(TInt*, dataPtr));
		case EPacketContextNotifyConfigChanged:
			return NotifyConfigChanged(aTsyReqHandle, aPackage.Des1n());
		case EPacketContextNotifyConfigChangedCancel:
			return NotifyConfigChangedCancel(aTsyReqHandle);
		case EPacketContextNotifyStatusChange:
			return NotifyStatusChange(aTsyReqHandle,
					REINTERPRET_CAST(RPacketContext::TContextStatus*, dataPtr));
		case EPacketContextNotifyStatusChangeCancel:
			return NotifyStatusChangeCancel(aTsyReqHandle);
		case EPacketContextGetStatus:
			return GetStatus(aTsyReqHandle,
					REINTERPRET_CAST(RPacketContext::TContextStatus*, dataPtr));
		case EPacketContextUpdateMbmsSessionList:
			return UpdateMbmsSessionList(aTsyReqHandle,
					REINTERPRET_CAST(TMbmsAction*, dataPtr),
					REINTERPRET_CAST(TUint*, dataPtr2));	
		case EPacketContextInitialiseContext:
			return InitialiseContext(aTsyReqHandle, aPackage.Des1n());
		case EPacketContextInitialiseContextCancel:
			return InitialiseContextCancel(aTsyReqHandle);
		case EPacketGetMbmsSessionListPhase1:
			return GetMbmsSessionsPhase1(aTsyReqHandle, 
			REINTERPRET_CAST(TClientId*, dataPtr), 
			REINTERPRET_CAST(TInt*, dataPtr2));
		case EPacketGetMbmsSessionListPhase2:
			return GetMbmsSessionsPhase2(aTsyReqHandle, 
			REINTERPRET_CAST(TClientId*, dataPtr), aPackage.Des2n());		
		default:
			LOGPACKET1("CSimMbmsPacketContext: ExtFunc Unknown IPC");
			LOGPACKET1("CSimMbmsPacketContext: sending the request to CSimPacketContext::ExtFunc Unknown IPC");
			return CSimPacketContext::ExtFunc(aTsyReqHandle,aIpc,aPackage);
		}
	}
		
TInt CSimMbmsPacketContext::CancelService(const TInt aIpc,const TTsyReqHandle aTsyReqHandle)
/**
* CancelService is called by the server when it is "cleaning-up" any still outstanding
* asynchronous requests before closing a client's sub-session.
* This will happen if a client closes its R-class handle without cancelling outstanding asynchronous requests.
* 
* @param aTsyReqHandle  The request handle for completing the request 
* @param aIpc Ipc representing the request
* @return err KErrNone if request completes ok
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: - CancelService called");

	switch (aIpc)
		{
		case EPacketContextSetConfig:
			return SetConfigCancel(aTsyReqHandle);
		case EPacketContextGetConfig:
			return GetConfigCancel(aTsyReqHandle);
		case EPacketContextActivate:
			return ActivateCancel(aTsyReqHandle);
		case EPacketContextDeactivate:
			return DeactivateCancel(aTsyReqHandle);
		case EPacketContextDeleteCancel:
			return DeleteCancel(aTsyReqHandle);
		case EPacketContextNotifyConfigChanged:
			return NotifyConfigChangedCancel(aTsyReqHandle);
		case EPacketContextNotifyStatusChange:
			return NotifyStatusChangeCancel(aTsyReqHandle);
		case EPacketContextInitialiseContext:
			return InitialiseContextCancel(aTsyReqHandle);
		case EPacketGetMbmsSessionListPhase1:
		case EPacketGetMbmsSessionListPhase2:
			return GetMbmsSessionsCancel(aTsyReqHandle);
		default:
			LOGPACKET1("CSimMbmsPacketContext: - CancelService unknown IPC called");
			LOGPACKET1("CSimMbmsPacketContext: sending the request to CSimPacketContext::ExtFunc Unknown IPC");
			return CSimPacketContext::CancelService(aIpc,aTsyReqHandle);
		}
	}

void CSimMbmsPacketContext::TimerCallBack(TInt aId)
/**
* Timer callback function.  When the timer goes off, it will call back into this
* function for further processing.
* param - aId indicates which Timer Event has occured.
*/
	{
	LOGPACKET1("CSimMbmsPacketContext: - TimerCallBack(TInt aId) called");
	switch(aId)
		{
		case ETimerIdMbmsPcktContextSetConfig:
			LOGPACKET1("CSimMbmsPacketContext: - TimerCallBack SetConfig called");
			if(iSetConfigValues.iErrorValue == KErrNone)
				{
				iMbmsContextConfigParamsIndex = iSetConfigValues.iIndex ;
				}
			ReqCompleted(iSetConfigRequestHandle,iSetConfigValues.iErrorValue);
			break;
		case ETimerIdMbmsUpdateSessionId:
			LOGPACKET1("CSimMbmsPacketContext: - TimerCallBack UpdateSessionID called");
			if(iUpdateSessionHandle != NULL)
				{
				ReqCompleted(iUpdateSessionHandle,KErrNone);
				}
			break;
		case ETimerIdPcktContextInitContext:
			DoInitialiseContext();
			break;
		case ETimerIdPcktContextCreateTft:
			CSimPacketContext::TimerCallBack(aId);
			break;
		default:
			{
			LOGPACKET1(">>CSimMbmsPacketContext::TimerCallBack Default Case for Context Events");
			switch(iCurrentEvent)
				{
				case EMbmsContextEventNone:
					LOGPACKET1("TimerCallBack - iCurrentEvent = [EMbmsContextEventNone]");
					break;
				case EMbmsContextEventActivate:
					LOGPACKET1("TimerCallBack - iCurrentEvent = [EMbmsContextEventActivate]");
					if(iState==RPacketContext::EStatusActivating)
						{
						TInt activateValue = iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iActivateCode;
						TInt ret;
						if(activateValue != KErrNone) 
							{ // Reject the Activation request.
							ret=ChangeState(RPacketContext::EStatusInactive);
							ReqCompleted(iActivateRequestHandle,KErrGprsActivationRejected);
							iCurrentEvent=EMbmsContextEventNone;
							}
						else
							{
							ret=ChangeState(RPacketContext::EStatusActive);
							ReqCompleted(iActivateRequestHandle,ret);
							iCurrentEvent=EMbmsContextEventNone;
							iIsActive = ETrue;
							}
						
						// Check for and complete any pending notifications
						if(iNotifyConfigMBMS.iNotifyPending)
							{
							TMbmsConfigParams thisConfig = iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex);
							iNotifyConfigMBMS.iNotifyPending=EFalse;
							RPacketMbmsContext::TContextConfigMbmsV1 *tempConfig = (RPacketMbmsContext::TContextConfigMbmsV1*)iNotifyConfigMBMS.iNotifyData;
							tempConfig->iTmgi.SetServiceId(thisConfig.iServiceId);
							tempConfig->iTmgi.SetMCC(thisConfig.iMCC);
							tempConfig->iTmgi.SetMNC(thisConfig.iMNC);
							tempConfig->iMbmsAccessBearer= thisConfig.iAccessBearer;
							tempConfig->iMbmsServicePriority = thisConfig.iServicePriority;
							tempConfig->iMbmsServiceMode = thisConfig.iServiceMode;
							ReqCompleted(iNotifyConfigMBMS.iNotifyHandle,KErrNone);
							}
						}
					else
						{
						LOGPACKET1("Unexpected iState for iCurrentEvent[EMbmsContextEventActivate]");
						}
					break;
				case EMbmsContextEventDeactivate:
					LOGPACKET1("TimerCallBack - iCurrentEvent = [EMbmsContextEventDeactivate]");
					if(((iState == RPacketContext::EStatusInactive) || (iState == RPacketContext::EStatusDeactivating)) && 
									(iCurrentEvent==EMbmsContextEventDeactivate))
						{
						TInt ret=ChangeState(RPacketContext::EStatusInactive);
						ReqCompleted(iDeactivateRequestHandle,ret);
						iCurrentEvent=EMbmsContextEventNone;
						iIsActive = EFalse;
						}
					else
						{
						LOGPACKET1("Unexpected iState for iCurrentEvent[EMbmsContextEventActivate]");
						}
					break;
				case EMbmsContextEventDelete:
					LOGPACKET1("TimerCallBack - iCurrentEvent = [EMbmsContextEventDelete]");
					if(iState == RPacketContext::EStatusInactive && !iIsActive )
						{
						TInt ret = ChangeState(RPacketContext::EStatusDeleted);
						ReqCompleted(iDeleteRequestHandle,ret);
						iCurrentEvent=EMbmsContextEventNone;
						}
					else
						{
						ReqCompleted(iDeleteRequestHandle,KErrCorrupt); // KErrCorrupt is sent to indicate that the 
						//delete request is received when the context is Not inactive.
						}
					break;
				default:
					{
					LOGPACKET1("CSimMbmsPacketContext::TimerCallBack");
					LOGPACKET1("<<Unexpected iCurrentEvent @ TimerCallBack");
					break;
					}
				}
			}
		}
	}

TName CSimMbmsPacketContext::ContextName() const 
/**
* Returns the context name to the requested client
*
* @return TName	Name of the MBMS context
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::ContextName called");
	return iContextName; 
	}

const CTestConfigSection* CSimMbmsPacketContext::CfgFile()
/**
* Return a pointer to the Configuration File Section
*
* @return CTestConfigSection	pointer to the configuration file section
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::CfgFile called");
	return iPacketService->CfgFile();
	}

TInt CSimMbmsPacketContext::SetConfig(const TTsyReqHandle aTsyReqHandle,const TDesC8* aConfig)
/**
* Sets the MBMS configuration parameters and notifies if
* there is a pending notification for the config change.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aConfig pointer containing the parameters to set to.
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::SetConfig called");

	TPckg<TPacketDataConfigBase>* configBase = (TPckg<TPacketDataConfigBase>*)aConfig;
	TPacketDataConfigBase& configBaseV1 = (*configBase)();
	iSetConfigRequestHandle = aTsyReqHandle;
	TInt err = KErrCorrupt;

	if(configBaseV1.ExtensionId()==TPacketDataConfigBase::KConfigMBMS)
		{
		TPckg<RPacketMbmsContext::TContextConfigMbmsV1>* configMBMSPckg = (TPckg<RPacketMbmsContext::TContextConfigMbmsV1>*)aConfig;
		RPacketMbmsContext::TContextConfigMbmsV1& configMBMS = (*configMBMSPckg)();

		TMbmsConfigParams mbmsConfigParams;

		mbmsConfigParams.iMCC = configMBMS.iTmgi.GetMCC();
		mbmsConfigParams.iMNC = configMBMS.iTmgi.GetMNC();
		mbmsConfigParams.iServiceId = configMBMS.iTmgi.GetServiceId() ;
		mbmsConfigParams.iAccessBearer = configMBMS.iMbmsAccessBearer;
		mbmsConfigParams.iServiceMode = configMBMS.iMbmsServiceMode;
		mbmsConfigParams.iServicePriority = configMBMS.iMbmsServicePriority;

		TUint index;
		for(index=0; index < iMbmsContextConfigParams->Count(); index++)
			{
			TMbmsConfigParams thisConfig = iMbmsContextConfigParams->At(index);

			if ((mbmsConfigParams.iAccessBearer != thisConfig.iAccessBearer) ||
					(mbmsConfigParams.iServiceMode != thisConfig.iServiceMode) ||
					(mbmsConfigParams.iServicePriority != thisConfig.iServicePriority) ||
					(mbmsConfigParams.iMCC != thisConfig.iMCC) ||
					(mbmsConfigParams.iMNC != thisConfig.iMNC) ||
					(mbmsConfigParams.iServiceId != thisConfig.iServiceId))
				{
				err = KErrCorrupt;
				}
			else
				{
				err = KErrNone;
				//Check for and complete any pending notifications
				if(iNotifyConfigMBMS.iNotifyPending)
					{
					iNotifyConfigMBMS.iNotifyPending=EFalse;
					RPacketMbmsContext::TContextConfigMbmsV1 *tempConfig = (RPacketMbmsContext::TContextConfigMbmsV1*)iNotifyConfigMBMS.iNotifyData;
					tempConfig->iTmgi.SetServiceId(thisConfig.iServiceId);
					tempConfig->iTmgi.SetMCC(thisConfig.iMCC);
					tempConfig->iTmgi.SetMNC(thisConfig.iMNC);
					tempConfig->iMbmsAccessBearer= thisConfig.iAccessBearer;
					tempConfig->iMbmsServicePriority = thisConfig.iServicePriority;
					tempConfig->iMbmsServiceMode = thisConfig.iServiceMode;
					ReqCompleted(iNotifyConfigMBMS.iNotifyHandle,KErrNone);
					}
				break;
				}
			}
		iSetConfigValues.iErrorValue = err;
		iSetConfigValues.iIndex = index;
		}
	else
		{ // not expected to reach here in case of MBMS context
		TUint index=0;
		iSetConfigValues.iErrorValue = err;
		iSetConfigValues.iIndex = index;
		}
	iMbmsSetConfigTimer->Start(RandTime(),this,ETimerIdMbmsPcktContextSetConfig);

	return KErrNone;
	}

TInt CSimMbmsPacketContext::SetConfigCancel(const TTsyReqHandle aTsyReqHandle)
/**
* Cancels the Setconfig request.
* This method will do nothing in this version of SimTSY.
* @param aTsyReqHandle Tsy Request handle for the client request
* @return err KErrNone 
*/
	{
	iMbmsSetConfigTimer->Cancel();
	ReqCompleted(aTsyReqHandle, KErrCancel);
	return KErrNone;
	}

TInt CSimMbmsPacketContext::GetConfig(const TTsyReqHandle aTsyReqHandle,TDes8* aConfig)
/**
* Retrieves and returns the MBMS context configuration
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aConfig pointer to the configuration params to retrieve
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetConfig called");
	TPckg<TPacketDataConfigBase>* configBase = (TPckg<TPacketDataConfigBase>*)aConfig;
	TPacketDataConfigBase& configBaseV1 = (*configBase)();
	
	if(configBaseV1.ExtensionId()==TPacketDataConfigBase::KConfigMBMS)
		{

		TPckg<RPacketMbmsContext::TContextConfigMbmsV1>* configMBMSPckg = (TPckg<RPacketMbmsContext::TContextConfigMbmsV1>*)aConfig;
		RPacketMbmsContext::TContextConfigMbmsV1& configMBMS = (*configMBMSPckg)();
		
		configMBMS.iTmgi.SetServiceId(iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iServiceId);
		configMBMS.iTmgi.SetMCC(iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iMCC);
		configMBMS.iTmgi.SetMNC(iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iMNC);
		configMBMS.iMbmsAccessBearer = reinterpret_cast<TMbmsScope&>(iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iAccessBearer);
		configMBMS.iMbmsServicePriority = reinterpret_cast<TMbmsServicePriority&>(iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iServicePriority);
		configMBMS.iMbmsServiceMode = reinterpret_cast<TMbmsServiceMode&>(iMbmsContextConfigParams->At(iMbmsContextConfigParamsIndex).iServiceMode);

		ReqCompleted(aTsyReqHandle,KErrNone);
		}
	else
		{
		ReqCompleted(aTsyReqHandle,KErrNotSupported);
		}

	return KErrNone;
	}

TInt CSimMbmsPacketContext::GetConfigCancel(const TTsyReqHandle /*aTsyReqHandle*/)
/**
* Cancels the Getconfig request.  
* This method will do nothing in this version of SimTSY.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @return err KErrNone 
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetConfigCancel called");
	return KErrNone;
	}

TInt CSimMbmsPacketContext::GetStatus(const TTsyReqHandle aTsyReqHandle,RPacketContext::TContextStatus* aContextStatus)
/**
* Retrieves and returns the current status of the MBMS context 
* 
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @param aContextStatus pointer to the status of the mbms packet context.
* @return KerrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetStatus called");
	*aContextStatus = iState;
	ReqCompleted(aTsyReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimMbmsPacketContext::Activate(const TTsyReqHandle aTsyReqHandle)
/**
* Client calls this method to try to Activate a mbms context.
* In this method, we'll store the TsyReqHandle for later completion then call ActionEvent
* state machine to handle the event which will kick off a timer.
* The Actual Activation will take place in the TimerCallBack method.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @return KErrNone
*/
	{
	iActivateRequestHandle = aTsyReqHandle;
	TInt ret;
	ret = ActionEvent(EMbmsContextEventActivate);
	return ret;
	}

TInt CSimMbmsPacketContext::ActivateCancel(const TTsyReqHandle aTsyReqHandle)
/**
* Client calls this method to try and cancel an Activate request.
* This will only complete successfully if and only if the Activate request 
* has not completed.i.e. if we haven't already activated the context.
* Otherwise, the request will be completed with KErrNone
*
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::ActivateCancel called");
	iMbmsContextTimer->Cancel();
	if(((iState == RPacketContext::EStatusInactive) || (iState == RPacketContext::EStatusActivating)) && 
					(iCurrentEvent==EMbmsContextEventActivate))
		{
		iCurrentEvent= EMbmsContextEventNone;
		TInt ret = ChangeState(RPacketContext::EStatusInactive);
		if(ret!= KErrNone)
			{
			ReqCompleted(aTsyReqHandle,ret);
			}
		else
			{
			ReqCompleted(aTsyReqHandle,KErrCancel);
			}
		}
	else if (iState == RPacketContext::EStatusDeleted)
		{
		ReqCompleted(aTsyReqHandle,KErrCancel);
		}
	else
		{
		ReqCompleted(aTsyReqHandle,KErrNone);
		}

	return KErrNone;
	}

TInt CSimMbmsPacketContext::Deactivate(const TTsyReqHandle aTsyReqHandle)
/**
* Client calls this method to try to Deactivate an active context.
* In this method, we'll store the TsyReqHandle for later completion then call ActionEvent
* state machine to handle the event which will kick off a timer.
* The Actual Deactivation will take place in the TimerCallBack method.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::Deactivate called");
	iDeactivateRequestHandle = aTsyReqHandle;
	TInt ret = ActionEvent(EMbmsContextEventDeactivate);
	return ret;
	}

TInt CSimMbmsPacketContext::DeactivateCancel(const TTsyReqHandle aTsyReqHandle)
/**
* Client calls this method to try and cancel a Deactivate request.
* This will only complete successfully if and only if the Deactivate request has not completed.
* i.e. if we haven't already deactivated the context.
* Otherwise, the request will be completed with KErrNone.
* 
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::DeactivateCancel called");
	iMbmsContextTimer->Cancel();
	if(((iState == RPacketContext::EStatusInactive) || (iState == RPacketContext::EStatusDeactivating)) && 
					(iCurrentEvent==EMbmsContextEventDeactivate))
		{
		iCurrentEvent= EMbmsContextEventNone;
		TInt ret = ChangeState(RPacketContext::EStatusActive);
		if(ret!= KErrNone)
			{
			ReqCompleted(aTsyReqHandle,ret);
			}
		else
			{
			ReqCompleted(aTsyReqHandle,KErrCancel);
			}
		}
	else
		{
		ReqCompleted(aTsyReqHandle,KErrNone);
		}
	
	return KErrNone;
	}

TInt CSimMbmsPacketContext::ReActivate(CSimPacketService* aPacketService, const TName& aContextName)
/**
* This method is called by the packet Service to Re-Initialise a Deleted context.
* If the context is deleted then the client cannot retrieve any 
* infomation or use the context in any way.
* In order for the client to use the context again, the client must 
* open an object by name passing the name of this context as a pararemter.
*
* @param aPacketService pointer to the parent class CSimPacketService.
* @param aContextName Must be the same name as the name for this context.
* @return KErrNone if no problem in re-initialising this context.
*/
	{
	if(iContextName.Compare(aContextName) != KErrNone)
		{
		return KErrBadName;
		}

	iPacketService = aPacketService;

	if(iDeleted)
		{
		TInt ret = ChangeState(RPacketContext::EStatusInactive);
		if (ret != KErrNone)
			{
			return ret;
			}

		iDeleted = EFalse;
		iMbmsContextConfigParamsIndex = 0;
		iNotifyStatusChange.iNotifyPending = EFalse;
		iNotifyConfigMBMS.iNotifyPending = EFalse;
		}
	return KErrNone;
	}

TInt CSimMbmsPacketContext::Delete(const TTsyReqHandle aTsyReqHandle)
/**
* Client calls this method to try to Delete this context.
* In this method, we'll store the TsyReqHandle for later completion then call ActionEvent
* state machine to handle the event which will kick off a timer.
* The Actual Delete will take place in the TimerCallBack method.
*
* This request will only succeed if the context already exist (had been created) and also not
* previously deleted.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::Delete called");
	iDeleteRequestHandle = aTsyReqHandle;
	TInt ret = ActionEvent(EMbmsContextEventDelete);
	return ret;
	}

TInt CSimMbmsPacketContext::DeleteCancel(const TTsyReqHandle aTsyReqHandle)
/**
* Client calls this method to try and cancel a Delete context request.
* This will only complete successfully if and only if the Delete request has not completed.
* i.e. if we haven't already Deleted the context.
*
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::DeleteCancel called");
	iMbmsContextTimer->Cancel();
	if((iState == RPacketContext::EStatusInactive) && (iCurrentEvent==EMbmsContextEventDelete))
		{
		iCurrentEvent= EMbmsContextEventNone;
		ReqCompleted(aTsyReqHandle,KErrCancel);
		}
	return KErrNone;
	}

TInt CSimMbmsPacketContext::GetLastErrorCause(const TTsyReqHandle aTsyReqHandle,TInt* /*aError*/)
/**
* This method is supported in only for the MBMS Packet Context
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aError pointer to the last error cause
* @return KerrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetLastErrorCause called");
	
	ReqCompleted(aTsyReqHandle,iLastError);
	return KErrNone;
	}


TInt CSimMbmsPacketContext::NotifyConfigChanged(const TTsyReqHandle aTsyReqHandle, TDes8* aConfig)
/**
* Records a client interest in being notified of a change in the Context Configuration.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aConfig pointer to the context config to store the config info when the config changes.
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::NotifyConfigChanged called");
	TPckg<TPacketDataConfigBase>* configBase = (TPckg<TPacketDataConfigBase>*)aConfig;
	TPacketDataConfigBase& configBaseV1 = (*configBase)();
	
	if(configBaseV1.ExtensionId() == TPacketDataConfigBase::KConfigMBMS)
		{
	    __ASSERT_ALWAYS(!iNotifyConfigMBMS.iNotifyPending,SimPanic(ENotificationAlreadyPending));

	    iNotifyConfigMBMS.iNotifyPending = ETrue;
	    iNotifyConfigMBMS.iNotifyHandle = aTsyReqHandle;
	    iNotifyConfigMBMS.iNotifyData = &configBaseV1;
		}
	else
		{
		LOGPACKET1("CSimMbmsPacketContext::NotifyConfigChanged Unexpected config for MBMS context");
		}

	return KErrNone;
	}

TInt CSimMbmsPacketContext::NotifyConfigChangedCancel(const TTsyReqHandle aTsyReqHandle)
/**
* Cancel a client's interest in being notified when the context configuration changes
* This is acheived simply by resetting the flag that indicates a notification is pending.
* 
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::NotifyConfigChangedCancel called");
	if(iNotifyConfigMBMS.iNotifyPending &&
		iNotifyConfigMBMS.iNotifyHandle == aTsyReqHandle)
		{
		iNotifyConfigMBMS.iNotifyPending=EFalse;
		ReqCompleted(aTsyReqHandle,KErrCancel);
		}
	else
		{
		ReqCompleted(aTsyReqHandle,KErrNone);
		}
		
	return KErrNone;
	}

TInt CSimMbmsPacketContext::NotifyStatusChange(const TTsyReqHandle aTsyReqHandle,RPacketContext::TContextStatus* aContextStatus)
/**
* Record a client's interest in being notified of the state of the packet contexts.
* First check that there isn't already a notification pending (the ETel Server should protect
* against this) and then record the information necessary to complete the request later, when
* the status does actually change.
*
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aContextStatus pointer to the packet context status
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::NotifyStatusChange called");
	__ASSERT_ALWAYS(!iNotifyStatusChange.iNotifyPending,SimPanic(ENotificationAlreadyPending));
	iNotifyStatusChange.iNotifyPending = ETrue;
	iNotifyStatusChange.iNotifyHandle = aTsyReqHandle;
	iNotifyStatusChange.iNotifyData = aContextStatus;
	return KErrNone;
	}

TInt CSimMbmsPacketContext::NotifyStatusChangeCancel(const TTsyReqHandle aTsyReqHandle)
/**
* Cancel a client's interest in being notified when the packet context status changes.
* This is acheived simply by resetting the flag that indicates a notification is pending.
* 
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::NotifyStatusChangeCancel called");
	if(iNotifyStatusChange.iNotifyPending)
		{
		iNotifyStatusChange.iNotifyPending=EFalse;
		ReqCompleted(aTsyReqHandle,KErrCancel);
		}
	else
		{
		ReqCompleted(aTsyReqHandle,KErrNone);
		}
	return KErrNone;
	}

TInt CSimMbmsPacketContext::UpdateMbmsSessionList(const TTsyReqHandle aTsyReqHandle,TMbmsAction* aAction, TUint* aSessionId)
/**
* client's interest in updating the MBMS session's list.
* This is achieved  by updating the list maintained internally by the context
* 
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aAction TMbmsAction - ADD/ REMOVE / REMOVE_ALL
* @param aSessionId sessoinID to be used for the updation.
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::UpdateMbmsSessionList called");

	TInt error=KErrNone;
	iUpdateSessionHandle = aTsyReqHandle;
	TKeyArrayFix key(0, ECmpTUint);
	TInt position =-1;

	switch(*aAction)
		{
		case SIMTSY_PACKET_MBMS_ADD_ENTRIES:
			LOGPACKET1("CSimMbmsPacketContext::UpdateMbmsSessionList Action: ADD ");
			
			// TRAP can contain multiple statments
			TRAP(error,iSessionIdList->AppendL(*aSessionId);
                    //iSessionIdList->At(0) = ++iNumOfSessionId;
                    iSessionIdList->Delete(0);
                    iSessionIdList->InsertL(0,++iNumOfSessionId));
			
            iMbmsUpdateSessionTimer->Start(RandTime(),this,ETimerIdMbmsUpdateSessionId);

			if(error != KErrNone)
				{
				ReqCompleted(iUpdateSessionHandle,error);
				}
			break;

		case SIMTSY_PACKET_MBMS_REM_ENTRIES:
			LOGPACKET1("CSimMbmsPacketContext::UpdateMbmsSessionList Action: REMOVE ");
			iSessionIdList->Find(*aSessionId,key,position);
			if(position != -1)
				{
				iSessionIdList->Delete(position);
				//iSessionIdList->At(0) = --iNumOfSessionId;
				TRAP(error, iSessionIdList->InsertL(0,--iNumOfSessionId));
	            if(error != KErrNone)
	                {
	                ReqCompleted(iUpdateSessionHandle,error);
	                }
				iMbmsUpdateSessionTimer->Start(RandTime(),this,ETimerIdMbmsUpdateSessionId);
				}
			else
				{
				error = KErrNotFound;
				ReqCompleted(iUpdateSessionHandle,error);
				}
			break;

		case SIMTSY_PACKET_MBMS_REM_ALL_ENTRIES:
			LOGPACKET1("CSimMbmsPacketContext::UpdateMbmsSessionList Action: REMOVE_ALL ");
			iNumOfSessionId=0;
			//iSessionIdList->At(0) = iNumOfSessionId;
			TRAP(error, iSessionIdList->InsertL(0,iNumOfSessionId));
            if(error != KErrNone)
                {
                ReqCompleted(iUpdateSessionHandle,error);
                }
			iSessionIdList->Delete(1,iSessionIdList->Count()); // removing all session ids from the list
			ReqCompleted(aTsyReqHandle, KErrNone);
			break;

		default:
			LOGPACKET1("CSimMbmsPacketContext::UpdateMbmsSessionList Action: Default ");
			ReqCompleted(aTsyReqHandle,KErrNotFound);
			break;
		}
	return KErrNone;
	}

TInt CSimMbmsPacketContext::UpdateMbmsSessionListCancel(const TTsyReqHandle aTsyReqHandle)
/**
* client's interest in cancelling the updation the MBMS session's list.
* 
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::UpdateMbmsSessionListCancel called");
	iMbmsUpdateSessionTimer->Cancel();
	if( iUpdateSessionHandle == aTsyReqHandle )
		{
		ReqCompleted(iUpdateSessionHandle,KErrCancel);
		iUpdateSessionHandle=NULL;
		}
	else
		{
		ReqCompleted(iUpdateSessionHandle,KErrNone);
		}
	return KErrNone;
	}

TInt CSimMbmsPacketContext::InitialiseContext(const TTsyReqHandle aTsyReqHandle, TDes8* aDataChannelV2Pckg)
/**
* client's interest in Initialising the context.
* 
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aDataChannelV2Pckg the information to be filled once the initialisation is complete.
* @return KErrNone
*/
	{
	TCommSetupItem item = FindCommSettings();
	iInitContextData.iTsyReqHandle = aTsyReqHandle;
	iInitContextData.iChannel = aDataChannelV2Pckg;
	iMbmsContextInitTimer->Start(RandTime(), this, ETimerIdPcktContextInitContext);

	return KErrNone;
	}
	
TInt CSimMbmsPacketContext::DoInitialiseContext()
/* Function completes the request from the client for the initialisation of the context.
 * Called when the timer event occurs after receiving the InitialiseContext request.
 * returns KErrNone.
 */
	{
	RPacketContext::TDataChannelV2Pckg* channel = (RPacketContext::TDataChannelV2Pckg*)iInitContextData.iChannel;
	RPacketContext::TDataChannelV2& channelV2 = (*channel)();

	// Check that the data structure is supported by the simulated TSY version
	TInt err = iPhone->CheckSimTsyVersion(channelV2);
	if(err != KErrNone)
		{
		ReqCompleted(iInitContextData.iTsyReqHandle, err);
		return KErrNone;
		}

	TCommSetupItem item = FindCommSettings();
	channelV2.iPort = item.iPortName;
	channelV2.iCsy = item.iCsyName;
	channelV2.iChannelId = item.iPortName;
	
	ReqCompleted(iInitContextData.iTsyReqHandle, KErrNone);

	return KErrNone;
	}

TInt CSimMbmsPacketContext::InitialiseContextCancel(const TTsyReqHandle aTsyReqHandle)
/**
* client's interest in cancelling the Initialisation of the MBMS Context.
* 
* @param aTsyReqHandle Tsy Request handle for the client cancel request
* @return KErrNone
*/
	{
	iMbmsContextInitTimer->Cancel();
	ReqCompleted(aTsyReqHandle, KErrCancel);
	
	return KErrNone;
	}

TInt CSimMbmsPacketContext::GetMbmsSessionsPhase1(const TTsyReqHandle aTsyReqHandle, TClientId* aClient, TInt* aBufSize)
/**
* client's interest in retrieving the MBMS Session information.
* This is the phase1 retrieval
* 
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aClient ClientId from which the request is sent
* @param aBufSize  Size of the buffer to be allocated to retrieve the data in phase2.
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetMbmsSessionsPhase1 called");
	iGetMbmsSessionHandle = aTsyReqHandle;
	TInt ret=KErrNone;
	TInt leaveCode=KErrNone;
	TRAP(leaveCode, ret=ProcessGetMbmsSessionsPhase1L(aTsyReqHandle, aClient, aBufSize););
	if (leaveCode != KErrNone)
		{
		ReqCompleted(aTsyReqHandle,leaveCode);
		}
	return ret;
	}	

TInt CSimMbmsPacketContext::ProcessGetMbmsSessionsPhase1L(const TTsyReqHandle aTsyReqHandle, 
													 TClientId* aClient, 
													 TInt* aBufSize)
/**
* The Actual function which does the phase1 retrieval.
* 
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aClient ClientId from which the request is sent
* @param aBufSize  Size of the buffer to be allocated to retrieve the data in phase2.
* @return KErrNone
*/
	{
	// Retrieve MBMS session list,
	// Store the sessions and then return the size of the buffer to the client
	LOGPACKET1("CSimMbmsPacketContext::ProcessGetMbmsSessionsPhase1L called");
	RPacketMbmsContext::CMbmsSession* mbmsSession=RPacketMbmsContext::CMbmsSession::NewL();
	CleanupStack::PushL(mbmsSession);

	for(TInt i=1;i<=iNumOfSessionId;i++)
		{
		mbmsSession->iSessionIdList.Append(iSessionIdList->At(i));
		}

	
	// Store the streamed list and the client ID
	CPcktListReadAllAttempt* read = CPcktListReadAllAttempt::NewL(*aClient,aTsyReqHandle);
	CleanupStack::PushL(read);
	
	HBufC8* iBuff=NULL;
	mbmsSession->ExternalizeL(iBuff);	
	CleanupStack::PushL(iBuff);
		
	CBufFlat* buf=CBufFlat::NewL(iBuff->Length());
	CleanupStack::PushL(buf);
	buf->InsertL(0,*iBuff);	
	
	read->iListBuf = buf;
	CleanupStack::Pop(buf);	
	CleanupStack::PopAndDestroy(iBuff);
	
	iMbmsSessionList->AppendL(read);
	CleanupStack::Pop(); // pop the CListReadAllAttempt
	
	// return the CBufBase’s size to client
	*aBufSize=(read->iListBuf)->Size();
	CleanupStack::PopAndDestroy(mbmsSession);
	
	// Complete first phase of list retrieval
	ReqCompleted(aTsyReqHandle,KErrNone);
	return KErrNone;
	}

TInt CSimMbmsPacketContext::GetMbmsSessionsPhase2(const TTsyReqHandle aTsyReqHandle,TClientId* aClient, TDes8* aBuf)
/**
* client's interest in retrieving the MBMS Session information.
* This is the phase2 retrieval
* 
* @param aTsyReqHandle Tsy Request handle for the client request
* @param aClient ClientId from which the request is sent
* @param aBuf  Buffer allocated to retrieve the data.
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetMbmsSessionsPhase2 called");
	CPcktListReadAllAttempt* read=NULL;
	// Find the get Mbms monitored services from this client
	for (TInt i=0; i<iMbmsSessionList->Count(); ++i)
		{
		read = iMbmsSessionList->At(i);
		if ((read->iClient.iSessionHandle==aClient->iSessionHandle) &&
		    (read->iClient.iSubSessionHandle==aClient->iSubSessionHandle))
			{
			TPtr8 bufPtr((read->iListBuf)->Ptr(0));
			// Copy the streamed list to the client
			aBuf->Copy(bufPtr);
			delete read;
			iMbmsSessionList->Delete(i);
			ReqCompleted(aTsyReqHandle,KErrNone);
			return KErrNone;
			}
		}
	// Should handle error case of not finding the matching client from read all phase 1
	ReqCompleted(aTsyReqHandle,KErrBadHandle);
	return KErrNone;
	}
	
TInt CSimMbmsPacketContext::GetMbmsSessionsCancel(const TTsyReqHandle aTsyReqHandle)
/**
* client's interest in cancelling the MBMS Get session request.
* 
* @param aTsyReqHandle Tsy Request handle for the client request
* @return KErrNone
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::GetMbmsSessionsCancel called");
	// Remove the MBMS sessions from iMbmsSessionList
	
	CPcktListReadAllAttempt* read=NULL;
	for (TInt i=0; i<iMbmsSessionList->Count(); ++i)
		{
		read = iMbmsSessionList->At(i);
		if (read->iReqHandle == aTsyReqHandle)
			{
			delete read;
			iMbmsSessionList->Delete(i);
			break;
			}
		}
	ReqCompleted(aTsyReqHandle,KErrCancel);
	return KErrNone;
	}

TInt CSimMbmsPacketContext::ChangeState(RPacketContext::TContextStatus aNewState)
/**
* Attempt to change state.
* First validate that the requested state change is ok.  If it is then proceed to change
* the state and complete any pending state change notification.
*
* @param aNewState the new state to change to
* @return Error indication if change of state is successful or not
*/
	{
	LOGPACKET2("CSimMbmsPacketContext::ChangeState [newState=%d]", aNewState);
	__ASSERT_ALWAYS(aNewState!=RPacketContext::EStatusUnknown,SimPanic(ECallStatusUnknownIllegal));

	if(iState==aNewState)
		return KErrNone;

	TInt ret=KErrGeneral;
	const struct TMbmsContextStateChangeValidity* stateChangePnt=KMbmsContextStateChangeValidity;
	while(stateChangePnt->iOldState!=KMbmsContextStateTableTerminator)
		{
		if(stateChangePnt->iOldState == RPacketContext::EStatusReceiving && stateChangePnt->iNewState == RPacketContext::EStatusActive)
			{
			iLastError = KErrMbmsNotSupported;
			}
		else if(stateChangePnt->iOldState == RPacketContext::EStatusReceiving && stateChangePnt->iNewState == RPacketContext::EStatusSuspended)
			{
			iLastError = KErrMbmsServicePreempted;
			}
		else if(stateChangePnt->iOldState == RPacketContext::EStatusSuspended && stateChangePnt->iNewState == RPacketContext::EStatusActive)
			{
			iLastError = KErrMbmsServiceUnavailable;
			}
		if((stateChangePnt->iOldState==iState) && (stateChangePnt->iNewState==aNewState))
			{
			ret=stateChangePnt->iError;
			break;
			}
		stateChangePnt++;
		}

	if(ret!=KErrNone)
		return ret;

	//Request permission from the phone to change the state of the packet connection
	ret = iPacketService->ChangeState(ConvertToPacketServiceStatus(aNewState));
	if(ret!=KErrNone)
		return ret;

	// Actually change the state.
	iState=aNewState;

	// Check for a pending line state notification.
	if(iNotifyStatusChange.iNotifyPending)
		{
		iNotifyStatusChange.iNotifyPending = EFalse;
		*(RPacketContext::TContextStatus*)iNotifyStatusChange.iNotifyData = iState;
		ReqCompleted(iNotifyStatusChange.iNotifyHandle,KErrNone);
		}

	LOGPACKET1("<<CSimMbmsPacketContext::ChangeState Completed");
	return KErrNone;
	}

TInt CSimMbmsPacketContext::ActionEvent(TMbmsContextEvent aEvent)
/**
* Entry point when an event has occured that may advance the state machine.
* The aEvent parameter describes the event.
*
* This function contains the main state machine for the packet Context.  
*
* @param aEvent The Packet Context event to handle
* @return value represents the error state caused by the attempted state machine jump.
*/
	{
	LOGPACKET1("<<CSimMbmsPacketContext::ActionEvent Completed");
	TInt ret=KErrNone;
	__ASSERT_ALWAYS(iState!=RPacketContext::EStatusUnknown,SimPanic(ECallStatusUnknownIllegal));

	switch(aEvent)
		{
		case EMbmsContextEventActivate:
			LOGPACKET1("CSimMbmsPacketContext::ActionEvent = [EMbmsContextEventActivate]");
			if(iState==RPacketContext::EStatusInactive)
				{
				iCurrentEvent=EMbmsContextEventActivate;
				ret = ChangeState(RPacketContext::EStatusActivating);
				iMbmsContextTimer->Start(RandTime(),this);  
				}
			else
				{
				ReqCompleted(iActivateRequestHandle, KErrGprsActivationRejected);
				}
			break;

		case EMbmsContextEventDeactivate:
			LOGPACKET1("CSimMbmsPacketContext::ActionEvent = [EMbmsContextEventDeactivate]");
			if(iState==RPacketContext::EStatusActive || iState==RPacketContext::EStatusActivating)
				{
				iCurrentEvent=EMbmsContextEventDeactivate;
				ret = ChangeState(RPacketContext::EStatusDeactivating);
				iMbmsContextTimer->Start(RandTime(),this);
				}
			else
				{
				ReqCompleted(iDeactivateRequestHandle, KErrNone);
				}
			break;

		case EMbmsContextEventDelete:
			LOGPACKET1("CSimMbmsPacketContext::ActionEvent = [EMbmsContextEventDelete]");
			iDeleted = ETrue;
			if(iState==RPacketContext::EStatusInactive)
				{
				iCurrentEvent=EMbmsContextEventDelete;
				iMbmsContextTimer->Start(RandTime(),this);
				}
			else
				ReqCompleted(iDeleteRequestHandle, KErrInUse);
			break;
		default:
			LOGPACKET1("CSimMbmsPacketContext::ActionEvent = [Default Case]");
			break;
		}
	return ret;
	}

RPacketService::TStatus CSimMbmsPacketContext::ConvertToPacketServiceStatus(RPacketContext::TContextStatus aNewState)
/**
* Converts Packet Context Status (RPacketContext::TContextStatus) to 
*          Packet Service Status (RPacketService::TStatus)
*
* @param aNewState the Context status to convert to packet service status
* @return RPacketService::TStatus The packet service status conversion
*/
	{
	LOGPACKET1("CSimMbmsPacketContext::ConvertToPacketServiceStatus called");
	RPacketService::TStatus serviceStatus;
	switch (aNewState)
		{
		case RPacketContext::EStatusInactive:
		case RPacketContext::EStatusActivating:
		case RPacketContext::EStatusDeleted:
			serviceStatus = RPacketService::EStatusAttached;
			break;
		case RPacketContext::EStatusActive:
		case RPacketContext::EStatusDeactivating:
			serviceStatus = RPacketService::EStatusActive;
			break;
		case RPacketContext::EStatusSuspended:
			serviceStatus = RPacketService::EStatusSuspended;
			break;
		default:
			serviceStatus = RPacketService::EStatusUnattached;
			break;
		}
	return serviceStatus;
	}

TInt CSimMbmsPacketContext::RandTime()
/* 
 * Function that generates time(in seconds) at random for the SimTSY 
 * Maximum of four seconds; zero seconds also included just to simulate the 
 * synchronous call scenario 
 */
	{
	LOGPACKET1("CSimMbmsPacketContext::RandTime called");
	TInt ranT= Math::Random()%4; 
	return(ranT);
	}