basebandadaptationplugins/basebandchanneladaptorforc32/c32bca2/src/MBca2Impl.cpp
author hgs
Thu, 13 May 2010 20:08:44 +0100
changeset 36 2e966480abb1
parent 0 3553901f7fa8
child 24 6638e7f4bd8f
permissions -rw-r--r--
201019_01

// Copyright (c) 2004-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:
// C32 Baseband Channel Adaptor(BCA) Factory APIs implementation.
// This file contains all the API implementations of C32BcaFactory for Symbian OS.
// 
//

/**
 @file 
 @internalComponent
*/
#include "MBca2Impl.h"
#include <networking/bca.h>
#include <comms-infras/commsbuf.h>
#include "C32BcaFactory.h"

using namespace BasebandChannelAdaptation;
using namespace BasebandChannelAdaptation::C32Bca;
using namespace BasebandChannelAdaptation2;
using namespace BasebandChannelAdaptation2::C32BcaShim;

MBca2* MBca2FactoryImpl::NewBcaL()
	{
	MBca2Impl* instance = new(ELeave) MBca2Impl;
	CleanupStack::PushL(instance);
	instance->ConstructL();
	CleanupStack::Pop(instance);
	return instance;
	}
	
void MBca2FactoryImpl::Release()
	{
	delete this;
	}

MBca2Impl::MBca2Impl() : iControl(NULL),
	iBcaParams(NULL),
	iLowerDataSender(NULL),
	iReceiver(NULL),
	iMBca(NULL),
	iC32BcaFactory(NULL),
	iRecvBlocked(EFalse)
	{
	
	}

MBca2Impl::~MBca2Impl()
	{	
	iReceiver->Cancel();
	delete iReceiver;
	delete iLowerDataSender;
	iMBca->Release();
	iC32BcaFactory->Release();
	}

void MBca2Impl::ConstructL()
	{
	iC32BcaFactory = new(ELeave) CC32BcaFactory;
	iMBca = iC32BcaFactory->NewBcaL();
	iLowerDataSender = MLowerDataSenderImpl::NewL(iMBca);
	iReceiver = new (ELeave) CReceiver(iMBca);
	}

TInt MBca2Impl::Open(MUpperControl& aControl, 
						MUpperDataReceiver& aData, 
						const TBcaParams& aBcaParams)
	{
	iControl = &aControl;
	iLowerDataSender->SetControl(&aControl);
	iReceiver->SetParams(&aData,&aBcaParams);
	iBcaParams = &aBcaParams;
	return KErrNone;
	}
		
void MBca2Impl::Start()
	{
	TRequestStatus status;
	iMBca->Open(status,iBcaParams->iChannelId);
	User::WaitForRequest(status);
	TInt ret = status.Int();

	if (ret != KErrNone && ret != KErrAlreadyExists )
		{
		TRequestStatus shutdownStatus;
		iMBca->Shutdown(shutdownStatus);
		User::WaitForRequest(shutdownStatus);
		}
	else
		{
		iControl->StartSending();
		iReceiver->StartListening();
		}
	return;
	}
		
MLowerDataSender& MBca2Impl::GetSender()
	{
	return *iLowerDataSender;
	}
		
void MBca2Impl::Close()
	{
	iMBca->Close();
	}
		
void MBca2Impl::Release()
	{	
	delete this;
	}
		
void MBca2Impl::SetFlowControl(TBlockOption aOption)
	{
	if(aOption == EBlockFlow)
		{
		iRecvBlocked = ETrue;
		iReceiver->Cancel();
		}
	else if(aOption == EUnblockFlow)
		{
		if( iRecvBlocked)
			{
			iReceiver->StartListening();
			}
		else
			{
			iRecvBlocked = EFalse;
			}
		}
	}
		
void MBca2Impl::Control(TRequestStatus& aStatus,
								TUint aOptLevel,
								TUint aOptName,
								TDes8& aOpt)
	{
	iMBca->Ioctl(aStatus,aOptLevel,aOptName,aOpt);
	}
		
void MBca2Impl::CancelControl()
	{
	iMBca->CancelIoctl();
	}

MLowerDataSenderImpl* MLowerDataSenderImpl::NewL(BasebandChannelAdaptation::MBca* aMBca)
	{
	MLowerDataSenderImpl* instance = new (ELeave) MLowerDataSenderImpl;
	CleanupStack::PushL(instance);
	instance->ConstructL(aMBca);
	CleanupStack::Pop(instance);
	return instance;	 
	}

MLowerDataSenderImpl::MLowerDataSenderImpl() : iSender(NULL), iControl(NULL), iSendBlocked(EFalse)
	{
	
	}

void MLowerDataSenderImpl::ConstructL(BasebandChannelAdaptation::MBca* aMBca)
	{
	iSender = new (ELeave) CSender(aMBca);
	}

MLowerDataSenderImpl::~MLowerDataSenderImpl()
	{
	iSender->Cancel();
	delete iSender;
	}

void MLowerDataSenderImpl::SetControl(MUpperControl* aControl)
	{
	iControl = aControl;
	}

MLowerDataSender::TSendResult MLowerDataSenderImpl::Send(RCommsBufChain& aCommsBufChain)
	{	
	TRequestStatus status;
	TInt retSend = iSender->Send(status, aCommsBufChain);
	if(retSend == -1)
		{
		iSendBlocked = ETrue;
		return ESendBlocked;
		}	
	User::WaitForRequest(status);
	TInt ret = status.Int();
	if(ret == KErrNone)
		{
		if(iSendBlocked)
			{
			iControl->StartSending();
			iSendBlocked = EFalse;
			}
		return ESendAccepted;
		}
	else
		{
		iControl->Error(ret);
		return ESendBlocked;
		}
	}

CReceiver::CReceiver(BasebandChannelAdaptation::MBca* aMBca) : CActive(EPriorityNormal), iMBca(aMBca)
	{
	CActiveScheduler::Add(this);
	}

CReceiver::~CReceiver()
	{
	
	}

void CReceiver::SetParams(MUpperDataReceiver* aDataReceiver, const MBca2::TBcaParams* aBcaParams)
	{
	iDataReceiver = aDataReceiver;
	iBcaParams = aBcaParams;
	}

void CReceiver::RunL()
	{	
	TCommsBufAllocator allocator = const_cast<RCommsBufPond&>(iBcaParams->iCommsBufPond).Allocator();
	iBufChain.Alloc(iData.Length(),allocator);
	iBufChain.Write(iData);
	iDataReceiver->Process(iBufChain);
	StartListening();
	}

void CReceiver::DoCancel()
	{
	iMBca->CancelRead();
	}

void CReceiver::StartListening()
	{
	// call read on MBca
	iData.SetMax();
	iMBca->Read(iStatus,iData);
	SetActive();
	}

CSender::CSender(BasebandChannelAdaptation::MBca* aMBca) : CActive(EPriorityNormal), iMBca(aMBca)
	{
	CActiveScheduler::Add(this);
	}
CSender::~CSender()
	{
	
	}

void CSender::RunL()
	{
	User::RequestComplete(iSendRequest, iStatus.Int());
	}

void CSender::DoCancel()
	{
	iMBca->CancelWrite();
	}

TInt CSender::Send(TRequestStatus& aStatus,RCommsBufChain& aPdu)
	{
	if(iStatus == KRequestPending)
		{
			return -1;
		}
	else
		{
		aStatus = KRequestPending;
		iSendRequest = &aStatus;
		TBuf8<KMaxIPPacketAndHeader> buffer;
		if(aPdu.Length() > KMaxIPPacketAndHeader)
			{
			// discard packet and return			
			aPdu.Free();
			User::RequestComplete(iSendRequest, KErrNone);
			return 0;
			}
		buffer.SetLength(aPdu.Length());
		aPdu.Read(buffer);
		aPdu.Free();
		SendBuffer(buffer);
		return 0;
		}
		
	}
void CSender::SendBuffer(const TDesC8& aBuffer)
	{
	iMBca->Write(iStatus,aBuffer);
	SetActive();		
	}