diff -r 6b1d113cdff3 -r 6638e7f4bd8f telephonyprotocols/rawipnif/src/IPv4Binder.cpp --- a/telephonyprotocols/rawipnif/src/IPv4Binder.cpp Mon May 03 13:37:20 2010 +0300 +++ b/telephonyprotocols/rawipnif/src/IPv4Binder.cpp Thu May 06 15:10:38 2010 +0100 @@ -1,670 +1,670 @@ -// Copyright (c) 2006-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 implements the CIPv4Binder class, which handles the transmission -// of IPv4 data to and from the TCP/IP stack. -// -// - -/** - @file -*/ - -#include -#include -#include "RawIPFlow.h" -#include "IPv4Binder.h" -#include -#include - -using namespace ESock; -#ifdef WCDMA_STUB -#include -#endif - -#define LOG_IP_ADDRESS(desc,addr) _LOG_L1C5(_L8(" " desc " = %d.%d.%d.%d"), \ - addr >> 24, (addr >> 16) & 0xFF, (addr >> 8) & 0xFF, addr & 0xFF); - -#ifdef __EABI__ -// Patch data is used and KMtuIPv4 and KRMtuIPv4 can be modified to a different value in RawIpNif.iby file -extern const TInt KMtuIPv4 = KDefaultMtu; -extern const TInt KRMtuIPv4 = KDefaultMtu; -#endif - -CIPv4Binder::CIPv4Binder(CRawIPFlow& aFlow, CBttLogger* aTheLogger) -/** - * Constructor - */ - : CBinderBase(aFlow,aTheLogger), - iSpeedMetric(KDefaultSpeedMetric) - { - } - -CIPv4Binder::~CIPv4Binder() -/** - * Destructor - */ - { - } - -MLowerDataSender* CIPv4Binder::Bind(MUpperDataReceiver* aUpperReceiver, MUpperControl* aUpperControl) -/** - * Binds TCP/IP protocol to Flow - * - * @param aUpperReceiver A pointer to Upper layer Receive class - * @param aUpperControl A pointer to Upper layer control class - */ - { - CBinderBase::Bind(aUpperReceiver, aUpperControl); // Call the superclass's method. - return this; - } - -TInt CIPv4Binder::Control(TUint aLevel, TUint aName, TDes8& aOption) -/** - * The main function called by the TCP/IP protocol to control the interface. - * Can perform a variety of general IP tasks (such as getting IP config) - * and "3G" specific tasks (such as deleting the context). - * - * @param aLevel The level of the interface to control - always KSOLInterface - * @param aName The command to perform - * @param aOption Data to be input/output as a result of the command - * @return Standard error codes - */ - { - _LOG_L1C3(_L8("CIPv4Binder::Control [aLevel=%d, aName=%d]"), - aLevel, aName); - - if (aLevel == KSOLInterface) - { - switch (aName) - { - case KSoIfHardwareAddr: - // unsupported because we don't have a h/w address - break; - - // 3G-specific configuration commands are below this point. -#ifdef WCDMA_STUB - case KRegisterEventHandler: - // Raw IP NIF Events are not supported - case KContextSetEvents: - // Raw IP NIF Events are not supported - break; - - case KContextCreate: - // We don't support creating new secondary contexts. - break; - - case KContextDelete: - // Deletes the primary PDP context. This will shut down the Nif. - return DeleteContext(aOption); - - case KContextActivate: - // If the IPv4 interface is up, then the context will already have - // been activated. So this command should fail with - // KErrAlreadyExists - { - TUint8* ptr = CONST_CAST(TUint8*, aOption.Ptr()); - TContextParameters* contextParams = - REINTERPRET_CAST(TContextParameters*, ptr); - - if (contextParams->iContextInfo.iContextId != - STATIC_CAST(TInt8, GetFlow().GetBcaController()->Nsapi())) - { - contextParams->iReasonCode = KErrNotFound; - } - else - { - contextParams->iContextInfo.iStatus = - GetFlow().GetContextStatus(); - contextParams->iReasonCode = KErrAlreadyExists; - } - return KErrNone; - } - - case KNifSetDefaultQoS: - case KContextQoSSet: - // Setting the QoS is meaningless over GPRS, so we just return that - // we don't support these operations. - break; - - case KContextTFTModify: - // As we only have one primary context, we don't support anything - // to do with traffic flow templates, which are used by secondary - // contexts. - break; - - case KContextModifyActive: - // This command is only valid aftermodifying TFT/QoS parameters. - // As we don't support any of these operations, - // this command is never valid. - break; -#endif - default: - (void)aOption; - break; - } - } - return KErrNotSupported; - } - -TInt CIPv4Binder::GetConfig(TBinderConfig& aConfig) - { - _LOG_L1C1(_L8("CIPv4Binder::GetConfig")); - - TBinderConfig4* config = TBinderConfig::Cast(aConfig); - - if(config == NULL) - { - return KErrNotSupported; - } - - config->iFamily = KAfInet; /* KAfInet - selects TBinderConfig4 */ - - config->iInfo.iFeatures = KIfCanBroadcast | KIfCanMulticast; /* Feature flags */ - - -#if defined __EABI__ - // Default value for Tx and Rx packet size - config->iInfo.iMtu = KMtuIPv4; - config->iInfo.iRMtu = KRMtuIPv4; -#else // WINS - // Set default values in case patch is not present in epoc.ini - config->iInfo.iMtu = KDefaultMtu; - config->iInfo.iRMtu = KDefaultMtu; - - // for the emulator process is patched via the epoc.ini file - UserSvr::HalFunction(EHalGroupEmulator,EEmulatorHalIntProperty,(TAny*)"rawip_KMtuIPv4",&(config->iInfo.iMtu)); - UserSvr::HalFunction(EHalGroupEmulator,EEmulatorHalIntProperty,(TAny*)"rawip_KRMtuIPv4",&(config->iInfo.iRMtu)); -#endif - - config->iInfo.iSpeedMetric = iSpeedMetric; /* approximation of the interface speed in Kbps. */ - - LOG_IP_ADDRESS("Local IP address from TBinderConfig", iSettings.iLocalAddr); - - config->iAddress.SetAddress(iSettings.iLocalAddr); /* Interface IP address. */ - config->iNetMask.SetAddress(iSettings.iNetMask); /* IP netmask. */ - config->iBrdAddr.SetAddress(iSettings.iBroadcastAddr); /* IP broadcast address. */ - config->iDefGate.SetAddress(iSettings.iDefGateway); /* IP default gateway or peer address (if known). */ - config->iNameSer1.SetAddress(iSettings.iPrimaryDns); /* IP primary name server (if any). */ - config->iNameSer2.SetAddress(iSettings.iSecondaryDns); /* IP secondary name server (if any). */ - - return KErrNone; - } -#ifdef WCDMA_STUB - -TInt CIPv4Binder::DeleteContext(TDes8& aContextParameters) -/** - * Deletes a context. As the NIF is responsible for one primary context, - * this is equivalent to closing down the NIF. - * - * @param aContextParameters Parameters of the context to delete - * @return KErrArgument if an incorrect structure is passed, otherwise KErrNone - */ - { - _LOG_L1C1(_L8("CIPv4Binder::DeleteContext")); - - if (aContextParameters.Length() != sizeof(TContextParameters)) - { - return KErrArgument; - } - - TUint8* ptr = CONST_CAST(TUint8*, aContextParameters.Ptr()); - TContextParameters* params = REINTERPRET_CAST(TContextParameters*, ptr); - - if (params->iContextInfo.iContextId != - STATIC_CAST(TInt8, GetFlow().GetBcaController()->Nsapi())) - { - params->iReasonCode = KErrBadName; - } - else - { - params->iReasonCode = KErrNone; - GetFlow().Stop(KErrNone, MNifIfNotify::EDisconnect); - } - - return KErrNone; - } - -#endif -/** - * Called when the context has been activated to set our IP address and get - * any other required settings from CommDB. - * - * @param aConfig The new context config - */ -void CIPv4Binder::UpdateContextConfigL(const TPacketDataConfigBase& aConfig) - { - _LOG_L1C1(_L8("CIPv4Binder::UpdateContextConfig")); - - // Get our IP address from the GPRS context config. - TInetAddr address; - - TBuf tempAddr; - - const RPacketContext::TProtocolConfigOptionV2* pco; - TInt rel = const_cast(aConfig).ExtensionId(); - if (rel == TPacketDataConfigBase::KConfigGPRS) - { - tempAddr.Copy(static_cast(aConfig).iPdpAddress); - pco = &static_cast(aConfig).iProtocolConfigOption; - } - else - { - ASSERT(rel == TPacketDataConfigBase::KConfigRel99Rel4 || rel == TPacketDataConfigBase::KConfigRel5); - tempAddr.Copy(static_cast(aConfig).iPdpAddress); - pco = &static_cast(aConfig).iProtocolConfigOption; - } - TInt ret = address.Input(tempAddr); - - // We've got our IP address! Let's save it. - if (ret == KErrNone) - { - iSettings.iLocalAddr = address.Address(); - LOG_IP_ADDRESS("Got local IP address from context", iSettings.iLocalAddr); - iSettings.iDefGateway = address.Address(); - _LOG_L1C1(_L8("Set Default Gateway to local IP address")); - } - else - { - _LOG_L2C2(_L8("Couldn't get IP address from GPRS config (err: %d)"), - ret); - - // Don't leave on this error: we may still be OK if we read some - // settings from CommDB. - } - - // @todo - is this correct. We can only get the DNS addresses - // from the TSY using the iProtocolConfigOption data. Yet a client could - // access those DNS config details without knowing about the state of the - // iSettings.iGetDnsFromServer flag. - - if ((iSettings.iGetDnsFromServer) || - ((iSettings.iPrimaryDns == 0) && - (iSettings.iSecondaryDns == 0)) ) - { - TBuf tempAddr; - tempAddr.Copy(pco->iDnsAddresses.iPrimaryDns); - ret = address.Input(tempAddr); - - if (ret == KErrNone) - { - iSettings.iPrimaryDns = address.Address(); - LOG_IP_ADDRESS("Got primary DNS from context PCO", iSettings.iPrimaryDns); - } - else - { - _LOG_L2C2(_L8("Couldn't get primary DNS address from GPRS config (err: %d)"), - ret); - - // Don't leave on this error: we may still be OK if we read some - // settings from CommDB. - } - - tempAddr.Copy(pco->iDnsAddresses.iSecondaryDns); - ret = address.Input(tempAddr); - - if (ret == KErrNone) - { - iSettings.iSecondaryDns = address.Address(); - LOG_IP_ADDRESS("Got secondary DNS from context PCO", iSettings.iPrimaryDns); - } - else - { - _LOG_L2C2(_L8("Couldn't get secondary DNS address from GPRS config (err: %d)"), - ret); - - // Don't leave on this error: we may still be OK if we read some - // settings from CommDB. - } - } - else - { - LOG_IP_ADDRESS("Using CommDB DNS address - Primary ", iSettings.iPrimaryDns); - LOG_IP_ADDRESS(" - Secondary ", iSettings.iSecondaryDns); - } - - - // TProtocolConfigOptionV2::iMiscBuffer is not the correct way to pass a gateway - // address - data needs to be passed in TLV format but there are no TLV content tags - // defined for a gateway address and any UMTS/GPRS hardware which claims to be able - // to supply this address is erroneous (3gpp standard 24.008, section 10.5.6.3). - // This misuse of iMiscBuffer was preventing other correctly formed parameters from - // being passed (INC113612). -/* if (iSettings.iGetGatewayFromServer) - { - tempAddr.Copy(aConfig.iProtocolConfigOption.iMiscBuffer); - ret = address.Input(tempAddr); - - if (ret == KErrNone) - { - iSettings.iDefGateway = address.Address(); - LOG_IP_ADDRESS("Got default gateway", iSettings.iDefGateway); - } - else - { - _LOG_L2C2(_L8("Couldn't get default gateway from GPRS config (err: %d)"), - ret); - } - }*/ - } - -void CIPv4Binder::UpdateConnectionSpeed(TUint aConnectionSpeed) -/** - * Sets the speed metric to return to TCP/IP, based on what the TSY tells us. - * - * @param aConnectionSpeed Our connection speed - */ - { - _LOG_L1C1(_L8("CIPv4Binder::UpdateConnectionSpeed")); - - iSpeedMetric = aConnectionSpeed; - } - - -MLowerDataSender::TSendResult CIPv4Binder::Send(RMBufChain& aPdu) -/** - * Called by the protocol to send an outgoing IP packet to the network. - * - * @param aPdu The outgoing packet - * @return MLowerDataSender::ESendBlocked or ESendAccepted based on state of flow. - */ - { - _LOG_L1C1(_L8("CIPv4Binder::Send")); - -#ifdef __BTT_LOGGING__ - LogPacket(aPdu); -#endif - - // Return ESendBlocked: flow cannot accept any more packets [blocked, queue full, etc] - // Return ESendAccepted: flow has accepted this packet and can accept another. - - return GetFlow().SendPacket(aPdu, NULL, KIp4FrameType); - } - -TInt CIPv4Binder::Notification(TAgentToNifEventType /*aEvent*/, - void* /*aInfo*/) -/** - * The Nif will ignore any notification sent - * - * @param aEvent Not used - * @param aInfo Not used - */ - { - _LOG_L1C1(_L8("CIPv4Binder::Notification")); - - return KErrNone; - } - -void CIPv4Binder::StartSending() -/** - * Indicates to the protocol layer that the NIF is ready to send packets. - * - * @param aProtocol A pointer to a protocol - */ - { - _LOG_L1C1(_L8("CIPv4Binder::StartSending()")); - - CBinderBase::StartSending(); - } - -TBool CIPv4Binder::WantsProtocol(TUint16 aProtocolCode) -/** - * Indicates the type of protocol implemented by this class. - * - * @param aProtocolCode The protocol type - */ - { - _LOG_L1C2(_L8("CIPv4Binder::WantsProtocol [aProtocolCode=%X]"), - aProtocolCode); - -#ifdef RAWIP_HEADER_APPENDED_TO_PACKETS - return ((aProtocolCode & 0x00FF) == KIp4FrameType); -#else - (void) aProtocolCode; - return ETrue; -#endif // RAWIP_HEADER_APPENDED_TO_PACKETS - } - -void CIPv4Binder::Process(RMBufChain& aPdu) -/** - * Called when an incoming IP packet has arrived. Send packets up to the - * TCP/IP stack. - * - * @param aPdu The incoming packet - */ - { - _LOG_L1C1(_L8("CIPv4Binder::Process")); - -#ifdef __BTT_LOGGING__ - LogPacket(aPdu); -#endif - - // Pass incoming packets up to the protocol, unless it hasn't - // been bound yet. - if (iUpperReceiver) // ASSERT(iUpperReceiver) ? - { - _LOG_L1C1(_L8("CIPv4Binder: Packet Sent to TCP/IP Protocol!!!")); - iUpperReceiver->Process(aPdu); - } - else - { - _LOG_L2C1(_L8("WARNING: dumping incoming packet, no protocol bound")); - aPdu.Free(); - } - } - -// -// MLowerControl methods -// - -TInt CIPv4Binder::GetName(TDes& aName) -/** -*/ - { - WriteIfName(aName); - return KErrNone; - } - -// -// CBinderBase methods -// - -void CIPv4Binder::SetProvision(const CIPConfig& aProvision) -/** -Set provisioning information for IPv4 binder. - -Called from RawIP Flow. - -@param aProvision Provisioning structure from Control side. -*/ - { - _LOG_L1C1(_L8("CIPv4Binder::SetProvision")); - - iSettings.iLocalAddr = aProvision.GetIpAddress(); - iSettings.iNetMask = aProvision.GetIpNetMask(); - iSettings.iBroadcastAddr = aProvision.GetBroadCastAddr(); - iSettings.iDefGateway = aProvision.GetIpGateway(); - iSettings.iPrimaryDns = aProvision.GetIp4NameServer1(); - iSettings.iSecondaryDns = aProvision.GetIp4NameServer2(); - iSettings.iGetGatewayFromServer = aProvision.GetIpAddrFromServer(); - iSettings.iGetDnsFromServer = aProvision.GetIp4DNSAddrFromServer(); - - LOG_IP_ADDRESS("Local IP address from Provisioning", iSettings.iLocalAddr); - } - -#ifdef __BTT_LOGGING__ -void CIPv4Binder::LogPacket(const RMBufChain& aPacket) -/** -* Logs packet information into log file. -* -* @param aPacket The packet -*/ - { - _LOG_L1C1(_L8("CIPv4Binder::LogPacket")); - - TInt mBufLength = aPacket.Length() - aPacket.First()->Length(); - - _LOG_L3C2(_L8("Analysis of %d byte packet:"), mBufLength); - - //Note: All the constants used on this method are a pragmatic guess of the - //IP header fields. The only porpose of this method is logging. - - if (mBufLength < 20) - { - _LOG_L3C2(_L8(" -doesn't appear to be a valid IPv4 packet (length=%d)") - , mBufLength); - return; - } - - // Get a pointer to the packet's payload. - const TUint8* payloadPtr = aPacket.First()->Next()->Ptr(); - - if ((payloadPtr[0] & 0xF0) != 0x40) - { - _LOG_L3C2(_L8(" - doesn't appear to be an IPv4 packet (version=0x%X)"), - (payloadPtr[0] & 0xF0) >> 4); - return; - } - - if ((payloadPtr[0] & 0xF) != 0x5) - { - _LOG_L3C2(_L8(" - doesn't have a standard IP header (length=0x%X)"), - payloadPtr[0] & 0xF); - return; - } - - _LOG_L3C5(_L8(" - src addr: %d.%d.%d.%d"), payloadPtr[12], payloadPtr[13], - payloadPtr[14], payloadPtr[15]); - _LOG_L3C5(_L8(" - dst addr: %d.%d.%d.%d"), payloadPtr[16], payloadPtr[17], - payloadPtr[18], payloadPtr[19]); - - if (payloadPtr[9] == 0x06) - { - _LOG_L3C1(_L8(" - appears to be a TCP packet")); - if (mBufLength < 40) - { - _LOG_L3C2(_L8(" - but is too short (length=0x%X)"), mBufLength); - return; - } - _LOG_L3C3(_L8(" - src port: %d, dst port: %d"), - (payloadPtr[20] << 8) + payloadPtr[21], - (payloadPtr[22] << 8) + payloadPtr[23]); - _LOG_L3C3(_L8(" - seq #: 0x%08X, ack #: 0x%08X"), - (payloadPtr[24] << 24) + (payloadPtr[25] << 16) + - (payloadPtr[26] << 8) + payloadPtr[27], - (payloadPtr[28] << 24) + (payloadPtr[29] << 16) + - (payloadPtr[30] << 8) + payloadPtr[31]); - - TBuf8<100> flagsSet; - flagsSet.Copy(_L8(" - flags set: ")); - - // Write description of payload's flags to "flagsSet" - if (payloadPtr[33] & 0x01) - { - flagsSet.Append(_L8("FIN ")); - } - if (payloadPtr[33] & 0x02) - { - flagsSet.Append(_L8("SYN ")); - } - if (payloadPtr[33] & 0x04) - { - flagsSet.Append(_L8("RST ")); - } - if (payloadPtr[33] & 0x08) - { - flagsSet.Append(_L8("PSH ")); - } - if (payloadPtr[33] & 0x10) - { - flagsSet.Append(_L8("ACK ")); - } - if (payloadPtr[33] & 0x20) - { - flagsSet.Append(_L8("URG ")); - } - _LOG_L3C1(flagsSet); - - TInt dataOffset = payloadPtr[32] >> 2; // in bytes - if ((dataOffset > 0) && (mBufLength > dataOffset + 30)) - { - TBuf8<100> data; - data.Copy(_L8(" - data begins: ")); - for (TInt i = dataOffset + 20; i < dataOffset + 100; i++) - { - // We skip some bytes at the end of the MBuf, as they're junk. - if (i >= (mBufLength - 10)) - { - break; - } - if (TChar(payloadPtr[i]).IsPrint()) - { - data.Append(TChar(payloadPtr[i])); - } - else - { - data.Append(TChar('?')); - } - } - _LOG_L3C1(data); - } - } - else if (payloadPtr[9] == 0x01) - { - _LOG_L3C1(_L8(" - appears to be an ICMP packet")); - if (mBufLength < 24) - { - _LOG_L3C2(_L8(" - but is too short (length=0x%X)"), mBufLength); - return; - } - - if (payloadPtr[20] == 0x8) - { - _LOG_L3C1(_L8(" - is an echo request")); - } - else if (payloadPtr[20] == 0x0) - { - _LOG_L3C1(_L8(" - is an echo reply")); - } - else - { - _LOG_L3C2(_L8(" - unknown type (0x%02X)"), payloadPtr[20]); - return; - } - - if (mBufLength >= 28) - { - _LOG_L3C3(_L8(" - ID: 0x%04X, seq #: 0x%04X"), - (payloadPtr[24] << 8) + payloadPtr[25], - (payloadPtr[26] << 8) + payloadPtr[27]); - } - } - else if (payloadPtr[9] == 0x11) - { - _LOG_L3C1(_L8(" - appears to be a UDP packet")); - if (mBufLength < 28) - { - _LOG_L3C2(_L8(" - but is too short (length=0x%X)"), mBufLength); - return; - } - _LOG_L3C3(_L8(" - src port: %d, dst port: %d"), - (payloadPtr[20] << 8) + payloadPtr[21], - (payloadPtr[22] << 8) + payloadPtr[23]); - } - else - { - _LOG_L3C2(_L8(" - appears to be for an unknown protocol (0x%X)"), - payloadPtr[9]); - } - } -#endif // __BTT_LOGGING__ +// Copyright (c) 2006-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 implements the CIPv4Binder class, which handles the transmission +// of IPv4 data to and from the TCP/IP stack. +// +// + +/** + @file +*/ + +#include +#include +#include "RawIPFlow.h" +#include "IPv4Binder.h" +#include +#include + +using namespace ESock; +#ifdef WCDMA_STUB +#include +#endif + +#define LOG_IP_ADDRESS(desc,addr) _LOG_L1C5(_L8(" " desc " = %d.%d.%d.%d"), \ + addr >> 24, (addr >> 16) & 0xFF, (addr >> 8) & 0xFF, addr & 0xFF); + +#ifdef __EABI__ +// Patch data is used and KMtuIPv4 and KRMtuIPv4 can be modified to a different value in RawIpNif.iby file +extern const TInt KMtuIPv4 = KDefaultMtu; +extern const TInt KRMtuIPv4 = KDefaultMtu; +#endif + +CIPv4Binder::CIPv4Binder(CRawIPFlow& aFlow, CBttLogger* aTheLogger) +/** + * Constructor + */ + : CBinderBase(aFlow,aTheLogger), + iSpeedMetric(KDefaultSpeedMetric) + { + } + +CIPv4Binder::~CIPv4Binder() +/** + * Destructor + */ + { + } + +MLowerDataSender* CIPv4Binder::Bind(MUpperDataReceiver* aUpperReceiver, MUpperControl* aUpperControl) +/** + * Binds TCP/IP protocol to Flow + * + * @param aUpperReceiver A pointer to Upper layer Receive class + * @param aUpperControl A pointer to Upper layer control class + */ + { + CBinderBase::Bind(aUpperReceiver, aUpperControl); // Call the superclass's method. + return this; + } + +TInt CIPv4Binder::Control(TUint aLevel, TUint aName, TDes8& aOption) +/** + * The main function called by the TCP/IP protocol to control the interface. + * Can perform a variety of general IP tasks (such as getting IP config) + * and "3G" specific tasks (such as deleting the context). + * + * @param aLevel The level of the interface to control - always KSOLInterface + * @param aName The command to perform + * @param aOption Data to be input/output as a result of the command + * @return Standard error codes + */ + { + _LOG_L1C3(_L8("CIPv4Binder::Control [aLevel=%d, aName=%d]"), + aLevel, aName); + + if (aLevel == KSOLInterface) + { + switch (aName) + { + case KSoIfHardwareAddr: + // unsupported because we don't have a h/w address + break; + + // 3G-specific configuration commands are below this point. +#ifdef WCDMA_STUB + case KRegisterEventHandler: + // Raw IP NIF Events are not supported + case KContextSetEvents: + // Raw IP NIF Events are not supported + break; + + case KContextCreate: + // We don't support creating new secondary contexts. + break; + + case KContextDelete: + // Deletes the primary PDP context. This will shut down the Nif. + return DeleteContext(aOption); + + case KContextActivate: + // If the IPv4 interface is up, then the context will already have + // been activated. So this command should fail with + // KErrAlreadyExists + { + TUint8* ptr = CONST_CAST(TUint8*, aOption.Ptr()); + TContextParameters* contextParams = + REINTERPRET_CAST(TContextParameters*, ptr); + + if (contextParams->iContextInfo.iContextId != + STATIC_CAST(TInt8, GetFlow().GetBcaController()->Nsapi())) + { + contextParams->iReasonCode = KErrNotFound; + } + else + { + contextParams->iContextInfo.iStatus = + GetFlow().GetContextStatus(); + contextParams->iReasonCode = KErrAlreadyExists; + } + return KErrNone; + } + + case KNifSetDefaultQoS: + case KContextQoSSet: + // Setting the QoS is meaningless over GPRS, so we just return that + // we don't support these operations. + break; + + case KContextTFTModify: + // As we only have one primary context, we don't support anything + // to do with traffic flow templates, which are used by secondary + // contexts. + break; + + case KContextModifyActive: + // This command is only valid aftermodifying TFT/QoS parameters. + // As we don't support any of these operations, + // this command is never valid. + break; +#endif + default: + (void)aOption; + break; + } + } + return KErrNotSupported; + } + +TInt CIPv4Binder::GetConfig(TBinderConfig& aConfig) + { + _LOG_L1C1(_L8("CIPv4Binder::GetConfig")); + + TBinderConfig4* config = TBinderConfig::Cast(aConfig); + + if(config == NULL) + { + return KErrNotSupported; + } + + config->iFamily = KAfInet; /* KAfInet - selects TBinderConfig4 */ + + config->iInfo.iFeatures = KIfCanBroadcast | KIfCanMulticast; /* Feature flags */ + + +#if defined __EABI__ + // Default value for Tx and Rx packet size + config->iInfo.iMtu = KMtuIPv4; + config->iInfo.iRMtu = KRMtuIPv4; +#else // WINS + // Set default values in case patch is not present in epoc.ini + config->iInfo.iMtu = KDefaultMtu; + config->iInfo.iRMtu = KDefaultMtu; + + // for the emulator process is patched via the epoc.ini file + UserSvr::HalFunction(EHalGroupEmulator,EEmulatorHalIntProperty,(TAny*)"rawip_KMtuIPv4",&(config->iInfo.iMtu)); + UserSvr::HalFunction(EHalGroupEmulator,EEmulatorHalIntProperty,(TAny*)"rawip_KRMtuIPv4",&(config->iInfo.iRMtu)); +#endif + + config->iInfo.iSpeedMetric = iSpeedMetric; /* approximation of the interface speed in Kbps. */ + + LOG_IP_ADDRESS("Local IP address from TBinderConfig", iSettings.iLocalAddr); + + config->iAddress.SetAddress(iSettings.iLocalAddr); /* Interface IP address. */ + config->iNetMask.SetAddress(iSettings.iNetMask); /* IP netmask. */ + config->iBrdAddr.SetAddress(iSettings.iBroadcastAddr); /* IP broadcast address. */ + config->iDefGate.SetAddress(iSettings.iDefGateway); /* IP default gateway or peer address (if known). */ + config->iNameSer1.SetAddress(iSettings.iPrimaryDns); /* IP primary name server (if any). */ + config->iNameSer2.SetAddress(iSettings.iSecondaryDns); /* IP secondary name server (if any). */ + + return KErrNone; + } +#ifdef WCDMA_STUB + +TInt CIPv4Binder::DeleteContext(TDes8& aContextParameters) +/** + * Deletes a context. As the NIF is responsible for one primary context, + * this is equivalent to closing down the NIF. + * + * @param aContextParameters Parameters of the context to delete + * @return KErrArgument if an incorrect structure is passed, otherwise KErrNone + */ + { + _LOG_L1C1(_L8("CIPv4Binder::DeleteContext")); + + if (aContextParameters.Length() != sizeof(TContextParameters)) + { + return KErrArgument; + } + + TUint8* ptr = CONST_CAST(TUint8*, aContextParameters.Ptr()); + TContextParameters* params = REINTERPRET_CAST(TContextParameters*, ptr); + + if (params->iContextInfo.iContextId != + STATIC_CAST(TInt8, GetFlow().GetBcaController()->Nsapi())) + { + params->iReasonCode = KErrBadName; + } + else + { + params->iReasonCode = KErrNone; + GetFlow().Stop(KErrNone, MNifIfNotify::EDisconnect); + } + + return KErrNone; + } + +#endif +/** + * Called when the context has been activated to set our IP address and get + * any other required settings from CommDB. + * + * @param aConfig The new context config + */ +void CIPv4Binder::UpdateContextConfigL(const TPacketDataConfigBase& aConfig) + { + _LOG_L1C1(_L8("CIPv4Binder::UpdateContextConfig")); + + // Get our IP address from the GPRS context config. + TInetAddr address; + + TBuf tempAddr; + + const RPacketContext::TProtocolConfigOptionV2* pco; + TInt rel = const_cast(aConfig).ExtensionId(); + if (rel == TPacketDataConfigBase::KConfigGPRS) + { + tempAddr.Copy(static_cast(aConfig).iPdpAddress); + pco = &static_cast(aConfig).iProtocolConfigOption; + } + else + { + ASSERT(rel == TPacketDataConfigBase::KConfigRel99Rel4 || rel == TPacketDataConfigBase::KConfigRel5); + tempAddr.Copy(static_cast(aConfig).iPdpAddress); + pco = &static_cast(aConfig).iProtocolConfigOption; + } + TInt ret = address.Input(tempAddr); + + // We've got our IP address! Let's save it. + if (ret == KErrNone) + { + iSettings.iLocalAddr = address.Address(); + LOG_IP_ADDRESS("Got local IP address from context", iSettings.iLocalAddr); + iSettings.iDefGateway = address.Address(); + _LOG_L1C1(_L8("Set Default Gateway to local IP address")); + } + else + { + _LOG_L2C2(_L8("Couldn't get IP address from GPRS config (err: %d)"), + ret); + + // Don't leave on this error: we may still be OK if we read some + // settings from CommDB. + } + + // @todo - is this correct. We can only get the DNS addresses + // from the TSY using the iProtocolConfigOption data. Yet a client could + // access those DNS config details without knowing about the state of the + // iSettings.iGetDnsFromServer flag. + + if ((iSettings.iGetDnsFromServer) || + ((iSettings.iPrimaryDns == 0) && + (iSettings.iSecondaryDns == 0)) ) + { + TBuf tempAddr; + tempAddr.Copy(pco->iDnsAddresses.iPrimaryDns); + ret = address.Input(tempAddr); + + if (ret == KErrNone) + { + iSettings.iPrimaryDns = address.Address(); + LOG_IP_ADDRESS("Got primary DNS from context PCO", iSettings.iPrimaryDns); + } + else + { + _LOG_L2C2(_L8("Couldn't get primary DNS address from GPRS config (err: %d)"), + ret); + + // Don't leave on this error: we may still be OK if we read some + // settings from CommDB. + } + + tempAddr.Copy(pco->iDnsAddresses.iSecondaryDns); + ret = address.Input(tempAddr); + + if (ret == KErrNone) + { + iSettings.iSecondaryDns = address.Address(); + LOG_IP_ADDRESS("Got secondary DNS from context PCO", iSettings.iPrimaryDns); + } + else + { + _LOG_L2C2(_L8("Couldn't get secondary DNS address from GPRS config (err: %d)"), + ret); + + // Don't leave on this error: we may still be OK if we read some + // settings from CommDB. + } + } + else + { + LOG_IP_ADDRESS("Using CommDB DNS address - Primary ", iSettings.iPrimaryDns); + LOG_IP_ADDRESS(" - Secondary ", iSettings.iSecondaryDns); + } + + + // TProtocolConfigOptionV2::iMiscBuffer is not the correct way to pass a gateway + // address - data needs to be passed in TLV format but there are no TLV content tags + // defined for a gateway address and any UMTS/GPRS hardware which claims to be able + // to supply this address is erroneous (3gpp standard 24.008, section 10.5.6.3). + // This misuse of iMiscBuffer was preventing other correctly formed parameters from + // being passed (INC113612). +/* if (iSettings.iGetGatewayFromServer) + { + tempAddr.Copy(aConfig.iProtocolConfigOption.iMiscBuffer); + ret = address.Input(tempAddr); + + if (ret == KErrNone) + { + iSettings.iDefGateway = address.Address(); + LOG_IP_ADDRESS("Got default gateway", iSettings.iDefGateway); + } + else + { + _LOG_L2C2(_L8("Couldn't get default gateway from GPRS config (err: %d)"), + ret); + } + }*/ + } + +void CIPv4Binder::UpdateConnectionSpeed(TUint aConnectionSpeed) +/** + * Sets the speed metric to return to TCP/IP, based on what the TSY tells us. + * + * @param aConnectionSpeed Our connection speed + */ + { + _LOG_L1C1(_L8("CIPv4Binder::UpdateConnectionSpeed")); + + iSpeedMetric = aConnectionSpeed; + } + + +MLowerDataSender::TSendResult CIPv4Binder::Send(RMBufChain& aPdu) +/** + * Called by the protocol to send an outgoing IP packet to the network. + * + * @param aPdu The outgoing packet + * @return MLowerDataSender::ESendBlocked or ESendAccepted based on state of flow. + */ + { + _LOG_L1C1(_L8("CIPv4Binder::Send")); + +#ifdef __BTT_LOGGING__ + LogPacket(aPdu); +#endif + + // Return ESendBlocked: flow cannot accept any more packets [blocked, queue full, etc] + // Return ESendAccepted: flow has accepted this packet and can accept another. + + return GetFlow().SendPacket(aPdu, NULL, KIp4FrameType); + } + +TInt CIPv4Binder::Notification(TAgentToNifEventType /*aEvent*/, + void* /*aInfo*/) +/** + * The Nif will ignore any notification sent + * + * @param aEvent Not used + * @param aInfo Not used + */ + { + _LOG_L1C1(_L8("CIPv4Binder::Notification")); + + return KErrNone; + } + +void CIPv4Binder::StartSending() +/** + * Indicates to the protocol layer that the NIF is ready to send packets. + * + * @param aProtocol A pointer to a protocol + */ + { + _LOG_L1C1(_L8("CIPv4Binder::StartSending()")); + + CBinderBase::StartSending(); + } + +TBool CIPv4Binder::WantsProtocol(TUint16 aProtocolCode) +/** + * Indicates the type of protocol implemented by this class. + * + * @param aProtocolCode The protocol type + */ + { + _LOG_L1C2(_L8("CIPv4Binder::WantsProtocol [aProtocolCode=%X]"), + aProtocolCode); + +#ifdef RAWIP_HEADER_APPENDED_TO_PACKETS + return ((aProtocolCode & 0x00FF) == KIp4FrameType); +#else + (void) aProtocolCode; + return ETrue; +#endif // RAWIP_HEADER_APPENDED_TO_PACKETS + } + +void CIPv4Binder::Process(RMBufChain& aPdu) +/** + * Called when an incoming IP packet has arrived. Send packets up to the + * TCP/IP stack. + * + * @param aPdu The incoming packet + */ + { + _LOG_L1C1(_L8("CIPv4Binder::Process")); + +#ifdef __BTT_LOGGING__ + LogPacket(aPdu); +#endif + + // Pass incoming packets up to the protocol, unless it hasn't + // been bound yet. + if (iUpperReceiver) // ASSERT(iUpperReceiver) ? + { + _LOG_L1C1(_L8("CIPv4Binder: Packet Sent to TCP/IP Protocol!!!")); + iUpperReceiver->Process(aPdu); + } + else + { + _LOG_L2C1(_L8("WARNING: dumping incoming packet, no protocol bound")); + aPdu.Free(); + } + } + +// +// MLowerControl methods +// + +TInt CIPv4Binder::GetName(TDes& aName) +/** +*/ + { + WriteIfName(aName); + return KErrNone; + } + +// +// CBinderBase methods +// + +void CIPv4Binder::SetProvision(const CIPConfig& aProvision) +/** +Set provisioning information for IPv4 binder. + +Called from RawIP Flow. + +@param aProvision Provisioning structure from Control side. +*/ + { + _LOG_L1C1(_L8("CIPv4Binder::SetProvision")); + + iSettings.iLocalAddr = aProvision.GetIpAddress(); + iSettings.iNetMask = aProvision.GetIpNetMask(); + iSettings.iBroadcastAddr = aProvision.GetBroadCastAddr(); + iSettings.iDefGateway = aProvision.GetIpGateway(); + iSettings.iPrimaryDns = aProvision.GetIp4NameServer1(); + iSettings.iSecondaryDns = aProvision.GetIp4NameServer2(); + iSettings.iGetGatewayFromServer = aProvision.GetIpAddrFromServer(); + iSettings.iGetDnsFromServer = aProvision.GetIp4DNSAddrFromServer(); + + LOG_IP_ADDRESS("Local IP address from Provisioning", iSettings.iLocalAddr); + } + +#ifdef __BTT_LOGGING__ +void CIPv4Binder::LogPacket(const RMBufChain& aPacket) +/** +* Logs packet information into log file. +* +* @param aPacket The packet +*/ + { + _LOG_L1C1(_L8("CIPv4Binder::LogPacket")); + + TInt mBufLength = aPacket.Length() - aPacket.First()->Length(); + + _LOG_L3C2(_L8("Analysis of %d byte packet:"), mBufLength); + + //Note: All the constants used on this method are a pragmatic guess of the + //IP header fields. The only porpose of this method is logging. + + if (mBufLength < 20) + { + _LOG_L3C2(_L8(" -doesn't appear to be a valid IPv4 packet (length=%d)") + , mBufLength); + return; + } + + // Get a pointer to the packet's payload. + const TUint8* payloadPtr = aPacket.First()->Next()->Ptr(); + + if ((payloadPtr[0] & 0xF0) != 0x40) + { + _LOG_L3C2(_L8(" - doesn't appear to be an IPv4 packet (version=0x%X)"), + (payloadPtr[0] & 0xF0) >> 4); + return; + } + + if ((payloadPtr[0] & 0xF) != 0x5) + { + _LOG_L3C2(_L8(" - doesn't have a standard IP header (length=0x%X)"), + payloadPtr[0] & 0xF); + return; + } + + _LOG_L3C5(_L8(" - src addr: %d.%d.%d.%d"), payloadPtr[12], payloadPtr[13], + payloadPtr[14], payloadPtr[15]); + _LOG_L3C5(_L8(" - dst addr: %d.%d.%d.%d"), payloadPtr[16], payloadPtr[17], + payloadPtr[18], payloadPtr[19]); + + if (payloadPtr[9] == 0x06) + { + _LOG_L3C1(_L8(" - appears to be a TCP packet")); + if (mBufLength < 40) + { + _LOG_L3C2(_L8(" - but is too short (length=0x%X)"), mBufLength); + return; + } + _LOG_L3C3(_L8(" - src port: %d, dst port: %d"), + (payloadPtr[20] << 8) + payloadPtr[21], + (payloadPtr[22] << 8) + payloadPtr[23]); + _LOG_L3C3(_L8(" - seq #: 0x%08X, ack #: 0x%08X"), + (payloadPtr[24] << 24) + (payloadPtr[25] << 16) + + (payloadPtr[26] << 8) + payloadPtr[27], + (payloadPtr[28] << 24) + (payloadPtr[29] << 16) + + (payloadPtr[30] << 8) + payloadPtr[31]); + + TBuf8<100> flagsSet; + flagsSet.Copy(_L8(" - flags set: ")); + + // Write description of payload's flags to "flagsSet" + if (payloadPtr[33] & 0x01) + { + flagsSet.Append(_L8("FIN ")); + } + if (payloadPtr[33] & 0x02) + { + flagsSet.Append(_L8("SYN ")); + } + if (payloadPtr[33] & 0x04) + { + flagsSet.Append(_L8("RST ")); + } + if (payloadPtr[33] & 0x08) + { + flagsSet.Append(_L8("PSH ")); + } + if (payloadPtr[33] & 0x10) + { + flagsSet.Append(_L8("ACK ")); + } + if (payloadPtr[33] & 0x20) + { + flagsSet.Append(_L8("URG ")); + } + _LOG_L3C1(flagsSet); + + TInt dataOffset = payloadPtr[32] >> 2; // in bytes + if ((dataOffset > 0) && (mBufLength > dataOffset + 30)) + { + TBuf8<100> data; + data.Copy(_L8(" - data begins: ")); + for (TInt i = dataOffset + 20; i < dataOffset + 100; i++) + { + // We skip some bytes at the end of the MBuf, as they're junk. + if (i >= (mBufLength - 10)) + { + break; + } + if (TChar(payloadPtr[i]).IsPrint()) + { + data.Append(TChar(payloadPtr[i])); + } + else + { + data.Append(TChar('?')); + } + } + _LOG_L3C1(data); + } + } + else if (payloadPtr[9] == 0x01) + { + _LOG_L3C1(_L8(" - appears to be an ICMP packet")); + if (mBufLength < 24) + { + _LOG_L3C2(_L8(" - but is too short (length=0x%X)"), mBufLength); + return; + } + + if (payloadPtr[20] == 0x8) + { + _LOG_L3C1(_L8(" - is an echo request")); + } + else if (payloadPtr[20] == 0x0) + { + _LOG_L3C1(_L8(" - is an echo reply")); + } + else + { + _LOG_L3C2(_L8(" - unknown type (0x%02X)"), payloadPtr[20]); + return; + } + + if (mBufLength >= 28) + { + _LOG_L3C3(_L8(" - ID: 0x%04X, seq #: 0x%04X"), + (payloadPtr[24] << 8) + payloadPtr[25], + (payloadPtr[26] << 8) + payloadPtr[27]); + } + } + else if (payloadPtr[9] == 0x11) + { + _LOG_L3C1(_L8(" - appears to be a UDP packet")); + if (mBufLength < 28) + { + _LOG_L3C2(_L8(" - but is too short (length=0x%X)"), mBufLength); + return; + } + _LOG_L3C3(_L8(" - src port: %d, dst port: %d"), + (payloadPtr[20] << 8) + payloadPtr[21], + (payloadPtr[22] << 8) + payloadPtr[23]); + } + else + { + _LOG_L3C2(_L8(" - appears to be for an unknown protocol (0x%X)"), + payloadPtr[9]); + } + } +#endif // __BTT_LOGGING__