telephonyserverplugins/multimodetsy/Multimode/gprs/ATGprsConfig.cpp
author ivan.fildichev@opencode.com
Thu, 04 Nov 2010 20:30:45 +0200
branchopencode
changeset 85 96b4f933d69a
parent 24 6638e7f4bd8f
permissions -rw-r--r--
Bugfixes and added new tests

// Copyright (c) 2001-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:
//

#include "gprs.h"
#include "Gprscontext.h"
#include "mSLOGGER.H"
#include <pcktcs.h>
#include "ATGprsConfig.h"
#include "ATIO.H"
#include <etelpckt.h>
#include "NOTIFY.H"
#include "Matstd.h"

_LIT8(KGetCGDCONTCommand,"AT+CGDCONT?\r");
_LIT8(KIPType4	, "IP");
_LIT8(KIPType6	, "IP");
_LIT8(KX25		, "X25");


 /**
 * @file
 * This file implements the CATGPRSSetConfig class and the CATGPRSGetConfig. These two classes are used by the 
 * GPRS AT TSY library. 
 * This state machine uses "AT+CGQREQ" "AT+CGDCONT" "AT+CGQMIN" commands.
 */
CATGPRSSetConfig* CATGPRSSetConfig::NewL(TInt aCid, CATIO* aIo, CTelObject* aTelObject, CATInit* aInit, CPhoneGlobals* aPhoneGlobals)
/**
 *  Standard 2 phase constructor.
 * @param aIo pointer to communication object.
 * @param aTelObject pointer to parent.
 * @param aInit pointer to AT phone init object.
 * @param aPhoneGlobals pointer to phone global wide states.
 */
	{
	CATGPRSSetConfig* p=new(ELeave) CATGPRSSetConfig(aCid, aIo, aTelObject, aInit, aPhoneGlobals);
	CleanupStack::PushL(p);
	p->ConstructL();
	CleanupStack::Pop();
	return p;
	}

void CATGPRSSetConfig::ConstructL()
/**
 * Construct all objects that can leave.
 */
	{
	CATCommands::ConstructL();
	}


CATGPRSSetConfig::CATGPRSSetConfig(TInt aCid, CATIO* aIo, CTelObject *aTelObject, CATInit* aInit, CPhoneGlobals* aPhoneGlobals)	
	: CATCommands(aIo, aTelObject, aInit, aPhoneGlobals), iCid(aCid)
/**
 * Constructor.
 *
 * @param aIo pointer to communication object.
 * @param aTelObject pointer to parent.
 * @param aPhoneGlobals pointer to phone global wide states.
 */
	{
	LOGTEXT(_L8("CATGprsClass::CATGprsClass called"));
	}


CATGPRSSetConfig::~CATGPRSSetConfig()
/**
 * Destructor.
 */ 
	{
	LOGTEXT(_L8("CATGPRSSetConfig::~CATGPRSSetConfig called"));
	}


void CATGPRSSetConfig::Start(TTsyReqHandle aTsyReqHandle, TAny* aConfig)
/**
 * This starts the sending of the set commands.
 *
 * @param aConfig config package.
 * @param aTsyReqHandle handle to the client.
 */
	{
	iReqHandle = aTsyReqHandle;
	TPckg<RPacketContext::TContextConfigGPRS>* contextConfigV1Pckg = (TPckg<RPacketContext::TContextConfigGPRS>*)aConfig;
	iContextConfigGPRS= &(*contextConfigV1Pckg)();
	TInt ret=MakeupCGDCONT();
	if(ret)
		{
		Complete(KErrNotSupported,ETimeOutCompletion);
		return;
		}
	
	LOGTEXT(_L8("CATGPRSSetConfig:\tCATConfigGPRS::Start function called in TSY"));
	
	__ASSERT_ALWAYS(iIo->AddExpectString(this,KNotifyMeIfErrorString) != NULL, Panic(EGeneral));
	iState=EWaitForSetCGDCONTComplete;
	Write(KGprsCommandTimeOut);	
	}

void CATGPRSSetConfig::Stop(TTsyReqHandle aTsyReqHandle)
/**
 * This function cancels the outstanding read and sets the state to EWaitForDSR.
 *
 * @param aTsyReqHandle handle to the client.
 */
	{
	LOGTEXT(_L8("CATGPRSSetConfig::Stop called"));
	if(iState!=EATNotInProgress && aTsyReqHandle==iReqHandle)
		{
		LOGTEXT(_L8("CATGPRSSetConfig::Stop Completing client request with KErrCancel"));
		Complete(KErrCancel,ETimeOutCompletion);
		}
	}


void CATGPRSSetConfig::CompleteWithIOError(TEventSource aSource,TInt aStatus)
/**
 * This Function completes the command from the client whith an error.
 *
 * @param aSource source of event from communication class.
 * @param aStatus status of event.
 */
	{
	Complete(aStatus, aSource);
	}


void CATGPRSSetConfig::Complete(TInt aErr, TEventSource aSource)
/**
 * This Function completes the get or set command from the client.
 * @param aErr an error code to relay to client.
 */
	{
	LOGTEXT(_L8("CATGPRSSetConfig::Complete"));
	RemoveStdExpectStrings();
	iIo->WriteAndTimerCancel(this);
	iIo->RemoveExpectStrings(this); 
	if (aErr == KErrNone)
		{
		((CGprsContext*)iTelObject)->SetConfig(iContextConfigGPRS);
		iPhoneGlobals->iNotificationStore->CheckNotification(iTelObject, EPacketContextConfigChanged);
		}

	// Allow our base class to do its thing and then complete the client request
	CATCommands::Complete(aErr,aSource);		
	iTelObject->ReqCompleted(iReqHandle, aErr);

	iState = EATNotInProgress;
	}


void CATGPRSSetConfig::EventSignal(TEventSource aSource)
/**
 * This function contains the state machine for the command.  The states flow consecutively in 
 * get and set states and are described below.
 *	
 * @par  aSource Source of function call.
 *
 * @par EWaitForDSR
 * Wait for the DSR command to complete. Also sends AT+CGDCONT set or get command to the phone.
 *
 * @par	EWaitForSetCGDCONTComplete,
 * Wait for response from the phone on the set command. 
 *
 * @par		EWaitForSetCGDCONTOK,
 * Validate phone response and send set AT+CGQMIN command.
 * 
 *
 */
	{
	LOGTEXT2(_L8("CATGPRSSetConfig::EventSignal with iState %d"),iState);
	if ((aSource==ETimeOutCompletion))
		{
		LOGTEXT(_L8("CATGPRSSetConfig:\tTimeout Error during Config"));
		Complete(KErrTimedOut,aSource);
		return;
		}
	switch(iState)
		{
		case EWaitForSetCGDCONTComplete:
			LOGTEXT(_L8("CATGPRSSetConfig::EventSignal, EWaitForSetCGDCONTComplete"));
			StandardWriteCompletionHandler(aSource, KGprsCommandTimeOut);
			iState = EWaitForSetCGDCONTOK;
			break;

		case EWaitForSetCGDCONTOK:
			{
			// Called when OK is received
			LOGTEXT(_L8("CATGPRSSetConfig::EventSignal, EWaitForSetCGDCONTOK"));
			__ASSERT_ALWAYS(aSource==EReadCompletion,Panic(EATCommand_IllegalCompletionReadExpected));
			TInt ret = ValidateExpectString();
			Complete(ret, aSource);
			}
			break;

		case EATNotInProgress:
			break;
		default:
			LOGTEXT(_L8("CATGPRSSetConfig::EventSignal, default. Panic"));
			Panic(EIllegalEvent);
			break;
		}			
	}


TInt CATGPRSSetConfig::MakeupCGDCONT()
/**
 * This Function creates the at set string for the AT+CGDCONT command.
 */
	{
	const TInt KPDPTypeIdent=5;
	TBuf8<KPDPTypeIdent>						pdpType;		// PDP Type identifier
	TBuf8<RPacketContext::KGSNNameLength>		gsnName;			// Access point Name
	TBuf8<RPacketContext::KMaxPDPAddressLength>	pdpAddress;	// PDP pre-assigned address
	switch(iContextConfigGPRS->iPdpType)
		{
	
		case RPacketContext::EPdpTypeIPv4:
			pdpType.Format(KIPType4);
		break;

		case RPacketContext::EPdpTypeIPv6:
			pdpType.Format(KIPType6);
		break;

		case RPacketContext::EPdpTypeX25:
			pdpType.Format(KX25);
		break;

		default:
			return KErrUnknown;
		}

	gsnName.Copy(iContextConfigGPRS->iAccessPointName);
	pdpAddress.Copy(iContextConfigGPRS->iPdpAddress);
	const TBool dataCompression=iContextConfigGPRS->iPdpCompression & RPacketContext::KPdpDataCompression;
	const TBool headerCompression=iContextConfigGPRS->iPdpCompression & RPacketContext::KPdpHeaderCompression;

	iTxBuffer.Format(_L8("AT+CGDCONT=%d,\"%S\",\"%S\",\"%S\",%d,%d\r"), iCid, &pdpType,
			&gsnName, &pdpAddress,dataCompression,headerCompression);

	return KErrNone;
	}



CATGPRSGetConfig* CATGPRSGetConfig::NewL(TInt aCid, CATIO* aIo, CTelObject* aTelObject, CATInit* aInit, CPhoneGlobals* aPhoneGlobals)
/**
 *  Standard 2 phase constructor.
 *
 * @param aIo pointer to communication object.
 * @param aTelObject pointer to parent.
 * @param aPhoneGlobals pointer to phone global wide states.
 */
	{
	CATGPRSGetConfig* p=new(ELeave) CATGPRSGetConfig(aCid, aIo, aTelObject, aInit, aPhoneGlobals);
	CleanupStack::PushL(p);
	p->ConstructL();
	CleanupStack::Pop();
	return p;
	}

void CATGPRSGetConfig::ConstructL()
/**
 * Construct all objects that can leave.
 */
	{
	CATCommands::ConstructL();
	}

CATGPRSGetConfig::CATGPRSGetConfig(TInt aCid, CATIO* aIo, CTelObject *aTelObject, CATInit* aInit, CPhoneGlobals* aPhoneGlobals)	
	: CATCommands(aIo, aTelObject, aInit, aPhoneGlobals), iCid(aCid)
/**
 * Constructor.
 *
 * @param aIo pointer to communication object.
 * @param aTelObject pointer to parent.
 * @param aPhoneGlobals pointer to phone global wide states.
 */
	{}


CATGPRSGetConfig::~CATGPRSGetConfig()
/**
 * Destructor.
 */
	{}


void CATGPRSGetConfig::Start(TTsyReqHandle aTsyReqHandle, TAny* aConfig)
/**
 * This starts the sending of the get commands.
 * 
 * @param aTsyReqHandle handle to the client.
 * @param aConfig Pointer to a RPacketContext::TContextConfigGPRS. 
 */

	{
	LOGTEXT(_L8("CATGPRSGetConfig::Start called"));

	TPckg<RPacketContext::TContextConfigGPRS>* contextConfigV1Pckg = (TPckg<RPacketContext::TContextConfigGPRS>*)aConfig;
	iContextConfigV1 = &(*contextConfigV1Pckg)();
	
	__ASSERT_ALWAYS(iIo->AddExpectString(this,KNotifyMeIfErrorString) != NULL, Panic(EGeneral));
	iReqHandle = aTsyReqHandle;
	iState=EWaitForGetCGDCONTComplete;
	Write(KGetCGDCONTCommand, KGprsCommandTimeOut);
	}



void CATGPRSGetConfig::Stop(TTsyReqHandle aTsyReqHandle)
/**
 * This function cancels the outstanding read and sets the state to EWaitForDSR.
 * @param aTsyReqHandle handle to the client.
 */
	{
	LOGTEXT(_L8("CATGPRSGetConfig::Stop called"));
	if(iState!=EATNotInProgress && aTsyReqHandle==iReqHandle)
		{
		LOGTEXT(_L8("CATGPRSGetConfig::Stop Completing client request with KErrCancel"));
		Complete(KErrCancel,ETimeOutCompletion);
		}
	}

void CATGPRSGetConfig::Complete(TInt aErr, TEventSource	aSource)
/**
 * This Function completes the get or set command from the client.
 * @param aErr and error to relay to the client.
 */
	{
	LOGTEXT(_L8("CATGPRSGetConfig::Complete"));
	RemoveStdExpectStrings();
	iIo->WriteAndTimerCancel(this);
	iIo->RemoveExpectStrings(this); 
	if(aErr == KErrNone)		
		{
		((CGprsContext*)iTelObject)->SetConfig(iContextConfigV1);
		}

	// Allow our base class to do its thing and then complete the client request
	CATCommands::Complete(aErr,aSource);		
	iTelObject->ReqCompleted(iReqHandle, aErr);

	iState = EATNotInProgress;
	}

void CATGPRSGetConfig::CompleteWithIOError(TEventSource aSource,TInt aStatus)
/**
 * This Function completes the command from the client whith an error.
 * @param aSource source of event from communication class.
 * @param aStatus status of event.
 */
	{
	Complete(aStatus, aSource);
	}

void CATGPRSGetConfig::EventSignal(TEventSource aSource)
/**
 * This function contains the state machine for the command.  The states flow consecutively in 
 * get and set states and are described below.
 *	
 * @par  aSource Source of function call. 
 *
 *
 * @par		EWaitForGetCGDCONTComplete,
 * Wait for response from the phone on the set command.
 *
 * @par		EWaitForGetCGDCONTOK,
 * Validate phone response and send set AT+CGQMIN command.
 *
 */
	{
	LOGTEXT2(_L8("CATGPRSGetConfig::EventSignal with iState %d"),iState);
	if ((aSource==ETimeOutCompletion))
		{
		LOGTEXT(_L8("CATGPRSGetConfig:\tTimeout Error during Config"));
		Complete(KErrTimedOut,aSource);
		return;
		}
	switch(iState)
		{
		case EWaitForGetCGDCONTComplete:
				{
				iIo->WriteAndTimerCancel(this);
				StandardWriteCompletionHandler(aSource, KGprsCommandTimeOut);
				iState = EWaitForGetCGDCONTOK;
				}
			break;
		case EWaitForGetCGDCONTOK:
				{
				// Called when OK is received
				__ASSERT_ALWAYS(aSource==EReadCompletion,Panic(EATCommand_IllegalCompletionReadExpected));
				TInt ret = ValidateExpectString();
				if(ret)
					{	
					Complete(ret, aSource);
					return;
					}
				TRAP_IGNORE(ParseCGDCONTResponseL());
				Complete(ret, aSource);
				}
			break;
			

		case EATNotInProgress:
			break;
		
		default:
				{
				LOGTEXT(_L8("CATGPRSGetConfig::EventSignal, Default, panic"));
				Panic(EIllegalEvent);
				}
			break;
		}			
	}


void CATGPRSGetConfig::ParseCGDCONTResponseL()
/**
 * This Function parses the response from the get AT+CGDCONT? command to the phone
 */
	{
	ParseBufferLC();
	CATParamListEntry* entry;
	TDblQueIter<CATParamListEntry> iter(iRxResults);
	while(entry = iter++,entry!=NULL)
		{
		if (entry->iResultPtr.MatchF(KCGDCONTResponseString)!=0)
			continue;
		entry = iter++;
		if(entry == NULL)
			User::Leave(KErrNotFound);
		TLex8 lex(entry->iResultPtr);
		TInt val;
		(void)User::LeaveIfError(lex.Val(val));
		if(iCid == val)
			{
			entry = iter++;
			break;
			}
		}
	if(entry == NULL)
		User::Leave(KErrNotFound);
	// we are now pointing to the correct context, just parse the values
	// Get the pdp type
	TPtrC8 result(entry->iResultPtr);
	if(result == KIPType4)	
		iContextConfigV1->iPdpType = RPacketContext::EPdpTypeIPv4;
	else if(result == KIPType6)//EPdpTypeIPv6,
		iContextConfigV1->iPdpType = RPacketContext::EPdpTypeIPv6;
	else if(result == KX25)
		iContextConfigV1->iPdpType = RPacketContext::EPdpTypeX25;
	else
		User::Leave(KErrNotSupported);		
	//Get the gsnName
	entry = iter++;
	if(entry == NULL)
		User::Leave(KErrNotFound);
	result.Set(entry->iResultPtr);
	iContextConfigV1->iAccessPointName.Copy(result);
	// Get pdp address
	entry = iter++;
	if(entry == NULL)
		User::Leave(KErrNotFound);
	result.Set(entry->iResultPtr);
	iContextConfigV1->iPdpAddress.Copy(result);
	// get compression
	entry = iter++;
	if(entry == NULL)
		User::Leave(KErrGeneral);
	iContextConfigV1->iPdpCompression = CATParamListEntry::EntryValL(entry);
	CleanupStack::PopAndDestroy();
	}