emulator/emulatorbsp/specific/ethernet.cpp
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Tue, 02 Feb 2010 01:39:10 +0200
changeset 0 cec860690d41
permissions -rw-r--r--
Revision: 201005 Kit: 201005

// Copyright (c) 2002-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:
// wins\specific\ethernet.cpp
// PDD for the Ethernet under the windows emulator
// 
//

/**
 @file wins/specific/ethernet.cpp
*/


#include <ethernet.h>
#include "nk_priv.h"
#include "nk_plat.h"
#include <property.h>

#include "pcap.h"


/** @addtogroup enet Ethernet Drivers
 *  Kernel Ethernet Support
 */

/** @addtogroup enet_pdd Driver PDD's
 * @ingroup enet
 */

/** @addtogroup enet_byplatform Ethernet support by platform
 * @ingroup enet
 */

/** @addtogroup enet_wins WINS (Emulator) Ethernet support
 * @ingroup enet_byplatform
 */


// strings potentially written to ethernet.ini by netcards.exe
#define	KEpocIniEthSpeed10Mbps	"10Mbps"
#define	KEpocIniEthSpeed100Mbps	"100Mbps"

// entries in epoc.ini file:
#define	KEpocIniEthSpeedEntry	"ETHER_SPEED"
#define	KEpocIniEthNIFEntry	"ETHER_NIF"
#define	KEpocIniEthMACEntry	"ETHER_MAC"

#define KEthDrvPanicCategory "D32ETHER"

#define KLocalDriverNameMax				256

#define PCAP_FILTER "ether dst %x:%x:%x:%x:%x:%x or ether multicast and ether src not %x:%x:%x:%x:%x:%x"

_LIT(KPddName, "Ethernet.Wins");

// needs ldd version..
const TInt KMinimumLddMajorVersion=1;
const TInt KMinimumLddMinorVersion=0;
const TInt KMinimumLddBuild=122;

/** @addtogroup enet_windows Windows Emulator Ethernet Pdd
 * @ingroup enet_pdd
 * @ingroup enet_wins
 * @{
 */

/**
 * The Windows specific Ethernet physical device (factory) class 
 * @internalTechnology belongs to PDD which sits internally in kernel 
 */
class DDriverEthernet : public DPhysicalDevice
    {
public:

    /**
     * The constructor
     * Sets the drivers version number. Limits possible
	 * number of units to one only (unit 0)
     */
    DDriverEthernet();

    /**
     * Inherited from DPhysicalDevice.
	 * Install the driver by setting it's name
	 * @return KErrNone on success, other error code on failure
     */
    virtual TInt Install();

    /**
     * Inherited from DPhysicalDevice.
	 * Get the Capabilites of the driver
     * NOT supported but required as implementation of
	 * pure virtual in base class
     */
    virtual void GetCaps(TDes8 &aDes) const;

    /**
     * Inherited from DPhysicalDevice.
	 * Create a channel to a device
	 * @param aChannel a reference to a newly created channel object
	 * @param aUnit a unit for which the channel is being created
	 * @param anInfo pointer to a descriptor with additional info (may be NULL)
	 * @param aVer a requested version
     * @return KErrNone on success, other error code on failure
     */
    virtual TInt Create(DBase*& aChannel, TInt aUnit, 
						const TDesC8* anInfo, const TVersion &aVer);

    /**
     * Inherited from DPhysicalDevice.
	 * Validate that the info supplied would create a valid channel
	 * @param aUnit a unit for which validation is to be done
	 * @param anInfo pointer to a descriptor with additional info (may be NULL)
	 * @param aVer a version to be validated
     * @return KErrNone if valid, KErrNotSupported otherwise
	 */
    virtual TInt Validate(TInt aUnit, const TDesC8* anInfo, const TVersion &aVer);
	};


/**
 * The WINS specific Ethernet channel class for the libpcap library
 * @internalTechnology belongs to PDD which sits internally in kernel 
 */
class DEthernetWins : public DEthernet
    {

	/**
     * The Worker Thread
     * @param aDEthernetWins Pointer to this object
     * @return Should never return
     */
    friend TInt WorkThread(DEthernetWins * aDEthernetWins);

public:

	enum TWinsEtherPanic
	{
	EBadMacAddress = 1, // means bad MAC address in ini file or entry for it missing or ini file missing
	ENoNetInterface, // means entry for network interface name missing in ini file or ini file missing
	EPcapNull	// means Wpcap couldn't be initialised - potentially not installed or name of network interface in ini file wrong
	};

    /**
     * The Constructor.
     */
    DEthernetWins();

    /**
     * The Destructor.
     */
    ~DEthernetWins();

	/**
     * The Isr plays a role of the pcap callback.
     * Do NOT call the function directly
     * @param thisObject a pointer to this object
     * @param header the received packet's pcap header
     * @param pkt_data the received packet
	 * @post new packet is copied to LDD's RX buffers' queue
	 */
    static void Isr(u_char *thisObject,   
                     const struct pcap_pkthdr *header, 
                     const u_char *pkt_data);

    /**
     * The DoCreate Method.
     * Sets up the channel as part of the object creation
     * and retrieves the MAC address from epoc.ini file.
	 * Also creates wpcap handler and thread for wpcap loop.
	 * @pre epoc32\\data\\epoc.ini must exist with entries: "ETHER-NIF=..", "ETHER-MAC=..", "ETHER-SPEED=.."
	 * @param aUnit a unit for which the channel is being created
	 * @panic D32ETHER reason: (1) can't get proper MAC address (2) can't get
	 * network interface name (3) can't initialise wpcap
	 * @return KErrNone on success, other error code on failure
	 */
    TInt DoCreate(TInt aUnit); 

    /**
     * DEthernet implementation.
	 * Start the receiver.
     * Resumes pcap thread. Sets status to ready.
	 * @return KErrNone on success or suitable error code on failure
     */
    virtual TInt Start();

    /**
     * DEthernet implementation.
	 * Stop the receiver.
     * @param aMode possible values are: EStopNormal (=0), EStopEmergency (=1)
	 * @post pcap thread suspended, status set to not ready
     */
    virtual void Stop(TStopMode aMode);

    /**
     * DEthernet implementation.
	 * Validates a new configuration - should be called before Configure()
     * @param aConfig is the configuration to be validated
     * @return KErrNone if aConfig valid, KErrNotSupported otherwise
     * @see Configure()
     */
    virtual TInt ValidateConfig(const TEthernetConfigV01 &aConfig) const;

    /**
     * DEthernet implementation.
	 * Configure the PDD and pcap library
     * Reconfigure the library using the new configuration supplied.
	 * Sets pcap filter to read only frames with destination address set to
	 * broadcast, multicast or MAC addresss from defaultConfig.
     * This will not change the MAC address.
     * @param aConfig The new configuration
	 * @return KErrNone on success, suitable error code otherwise
     * @see ValidateConfig()
     * @see MacConfigure()
     */
    virtual TInt Configure(TEthernetConfigV01 &aConfig);

    /**
     * DEthernet implementation.
	 * Change the MAC address - writes new MAC address in defaultConfig.
	 * If new settings are to have any effect then pcap filter
	 * ought to be set again which is done by 'Configure()'
	 * @param aConfig a configuration structure containing the new MAC
     * @see Configure()
     */
    virtual void MacConfigure(TEthernetConfigV01 &aConfig);

	/**
     * DEthernet implementation.
	 * Get the current config from defaultConfig member varaiable
	 * which is assumed to be up to date.
     * Fills in the following fields:
     * The Transmit Speed
     * The Duplex Setting
     * The MAC address
     * @param aConfig is a TEthernetConfigV01 reference that will be filled in
     */
    virtual void GetConfig(TEthernetConfigV01 &aConfig) const;

	/**
     * DEthernet implementation.
	 * Dummy method, required as pure virtual in base class
     */
    virtual void CheckConfig(TEthernetConfigV01 &aConfig);

	/**
     * DEthernet implementation.
	 * Should query the capabilites.
     * NOT supported but required as pure virtual in base class
     */
    virtual void Caps(TDes8 &aCaps) const;

    /**
     * DEthernet implementation.
	 * Transmit data via wpcap
     * @param aBuffer reference to the data to be sent
     * @return KErrNone on success, other error code on failure
     */
    virtual TInt Send(TBuf8<KMaxEthernetPacket+32> &aBuffer);

    /**
     * DEthernet implementation.
	 * Retrieve data
     * Pull the received data out of the pcap library and into the supplied buffer. 
     * Need to be told if the buffer is OK 
     * @param aBuffer Reference to the buffer to be used to store the data in
     * @param okToUse Bool to indicate if the buffer is usable
     * @return KErrNone on success, other error code on failure
     */
    virtual TInt ReceiveFrame(TBuf8<KMaxEthernetPacket+32> &aBuffer, TBool okToUse);

	/**
     * DEthernet implementation.
	 * Disables all IRQ's 
     * @return The IRQ level before it was changed
     * @see RestoreIrqs()
     */
    virtual TInt DisableIrqs();

    /**
     * DEthernet implementation.
	 * Restore the IRQ's to the supplied level
	 * @param aIrq The level to set the irqs to.
     * @see DisableIrqs()
     */
    virtual void RestoreIrqs(TInt aIrq);

	/**
     * DEthernet implementation.
	 * Return the DFC Queue that this device should use
     * @param aUnit a channel's unit number (ignored - only one unit possible)
     * @return a DFC Queue to use
     */
    virtual TDfcQue* DfcQ(TInt aUnit);
    
		
private:
 	static TBool DEthernetWins::IsTcp(TDesC8 &aFrame);
 	static TInt DEthernetWins::GetTcpAckNumber(TDesC8 &frame);
 	static TInt DEthernetWins::GetTcpSeqNumber(TDesC8 &frame);
	/**
     * Read network interface to be used from configuration file. Panic if
	 * pre-conditions are not satisfied.
	 * @pre epoc32\\data\\epoc.ini must exist with entry: "ETHER-NIF=existing_nif_name"
	 * @post network interface name put in a member variable: iNetInterfaceName
	 * @panic D32ETHER reason: (2) can't get network interface name 
     */
	void  SetDriverName();

	/**
	 * Read MAC address from a configuration file and put it
	 * into defaultConfig member variable checking before if the
	 * one from the file is correct. Panic if pre-conditions are not satisfied
	 * (although in case when MAC address is improper).
	 * @pre epoc32\\data\\epoc.ini must exist with entry: "ETHER-MAC=proper_mac_address"
	 * @panic D32ETHER reason:  EBadMacAddress
	 * @return KErrNone on success (panics on failure)
	 */
	TInt SetMacAddress();

    
private:

    /**
     * The handle to the pcap interface - wpcap specific
     */
    pcap_t * iPcapPtr;

    /**
     * Saved received packet information - wpcap specific:
	 * pointer to a structure that holds general information about the packet:
	 * the time in which it was sniffed, the length of this packet, 
	 * and the length of his specific portion (in case it is fragmented)
     */
    const struct pcap_pkthdr * iPcapHeader; 

    /**
     * Saved receive packet data - wpcap specific
     */
    const u_char * iPcapPktData;

    /**
     * Id of the receiver - wpcap thread
     */
    unsigned long iWorkerThreadId;

	/**
     * Contains the handle to wpcap thread. 
     */
	HANDLE				iThreadHandle;

    /**
     * Stores the unit number (only one interface possible in 
	 * this implementation, so it will have value "0")
     */
    TInt  iUnit;
	
    /**
     * Is ETrue if the chip has been fully configured and is ready
	 * for receiving frames. Is set to ETrue in Start(), to EFalse
	 * in Stop(). Initialized in constructor as EFalse.
     */
    TBool iReady;

    /**
     * Contains the default/current configuration of the driver.
	 * Updated whenever configuration is to be changed.
     */
    TEthernetConfigV01 defaultConfig;

	/**
     * Contains the network interface name to be used 
	 * @see SetDriverName()
     */
	char	iNetInterfaceName[KLocalDriverNameMax];
	
    };

/** @} */ // End of wins ethernet pdd


 
DDriverEthernet::DDriverEthernet()
// Constructor
    {
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DDriverEthernet::DDriverEthernet()"));
    iUnitsMask=0x1;	// support unit 0 only
    iVersion=TVersion(KEthernetMajorVersionNumber,
                      KEthernetMinorVersionNumber,
                      KEthernetBuildVersionNumber);
	
	}


TInt DDriverEthernet::Install()
// Install the driver
    {
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DDriverEthernet::Install()"));
    return SetName(&KPddName);
    }


void GetWinsEthernetsCaps(TDes8 &aCaps, TInt aUnit=0)
    {
    __KTRACE_OPT(KHARDWARE, Kern::Printf("GetWinsEthernetsCaps(TDes8 &aCaps, TInt aUnit)"));
    TEthernetCaps capsBuf;
	
	aUnit=0;

    aCaps.FillZ(aCaps.MaxLength());
    aCaps=capsBuf.Left(Min(capsBuf.Length(),aCaps.MaxLength()));
    }

void PanicFromWinsEtherDrv(TInt aReason)
	{
	Kern::Fault(KEthDrvPanicCategory, aReason);
	}

void DDriverEthernet::GetCaps(TDes8 &aDes) const
// Return the driver's capabilities
    {
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DDriverEthernet::GetCaps(TDes8 &aDes) const"));
    GetWinsEthernetsCaps(aDes);	
    }


TInt DDriverEthernet::Create(DBase*& aChannel, 
                             TInt aUnit, 
                             const TDesC8* aInfo, 
                             const TVersion& aVer)
// Create a driver
    {
    __KTRACE_OPT(KHARDWARE, 
		Kern::Printf("DDriverEthernet::Create(DBase*& aChannel, TInt aUnit, const TDesC8* nInfo, const TVersion& aVer)"));
	
    TInt ret;

	ret = Validate( aUnit, aInfo, aVer);
	if ( KErrNone != ret )
		return ret;

	ret = KErrNoMemory;

	DEthernetWins* ptrPdd = new DEthernetWins;

	
	if ( ptrPdd )
        {
        ret = ptrPdd->DoCreate(aUnit);
        if ( ret != KErrNone)
			{
            delete ptrPdd;
			}	
		else
			aChannel = ptrPdd;
        }

	
    return ret;
    }


TInt DDriverEthernet::Validate(TInt aUnit, 
                               const TDesC8* /*aInfo*/, 
                               const TVersion& aVer)
//	Validate the requested configuration
    {
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DDriverEthernet::Validate(TInt aUnit, const TDesC8* /*aInfo*/, const TVersion& aVer)"));
    if ((!Kern::QueryVersionSupported(iVersion,aVer)) || 
        (!Kern::QueryVersionSupported(aVer,TVersion(KMinimumLddMajorVersion,
                                                    KMinimumLddMinorVersion,
                                                    KMinimumLddBuild))))
		{
        return KErrNotSupported;
		}
    
	if (aUnit != 0)
		{
        return KErrNotSupported;
		}

	return KErrNone;
    }


DEthernetWins::DEthernetWins()
// Constructor
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::DEthernetWins()"));

    iReady = EFalse;
    
	// set default configuration - must be set before DoCreate gets called
	defaultConfig.iEthSpeed  = KEthSpeedUnknown;
    defaultConfig.iEthDuplex = KEthDuplexUnknown;
							
	// MAC address initially set to NULL
	defaultConfig.iEthAddress[0] = 0; 
    defaultConfig.iEthAddress[1] = 0; 
    defaultConfig.iEthAddress[2] = 0; 
	defaultConfig.iEthAddress[3] = 0; 
    defaultConfig.iEthAddress[4] = 0; 
    defaultConfig.iEthAddress[5] = 0; 
	
	iNetInterfaceName[0] = '\0';

	// wpcap
	iPcapPtr = NULL;

	}

DEthernetWins::~DEthernetWins()
// Destructor
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::~DEthernetWins()"));

	if (iPcapPtr)
		{
		pcap_close(iPcapPtr);
		iPcapPtr = NULL;
		}
  
	}


TInt DEthernetWins::DoCreate(TInt aUnit)//, const TDesC8* /*anInfo*/)
// Sets up the PDD
	{
	__KTRACE_OPT(KHARDWARE, 
		Kern::Printf("DEthernetWins::DoCreate(TInt aUnit, const TDesC8* /*anInfo*/)"));
   
	iUnit = aUnit;

	TInt ret = KErrNone;

	SetMacAddress(); // will panic if can't get proper MAC address

	const char* speedProperty = Property::GetString(KEpocIniEthSpeedEntry);

	if( (NULL==speedProperty) ? 0 : (0 == strcmp( speedProperty, KEpocIniEthSpeed10Mbps )) )
		{
		defaultConfig.iEthSpeed = KEthSpeed10BaseT;
		}
	else if ( (NULL==speedProperty) ? 0 : (0 == strcmp( speedProperty, KEpocIniEthSpeed100Mbps )) )
		{
		defaultConfig.iEthSpeed = KEthSpeed100BaseTX;
		}

	SetDriverName();

	char errPcap[PCAP_ERRBUF_SIZE];

	if( iPcapPtr == NULL )
		{
		
		if((iPcapPtr = pcap_open_live( iNetInterfaceName,
									   2000, 
                                      1, 
                                      20, 
                                      errPcap)) != NULL)
			{
	 	    
			if ((iThreadHandle = CreateThread(NULL,
                                           0,
                                           (LPTHREAD_START_ROUTINE)WorkThread,
                                           this,
                                           CREATE_SUSPENDED,
                                           &iWorkerThreadId)) != NULL)
				{
    			ret = Configure(defaultConfig);
				}
			else
				{
				__KTRACE_OPT(KHARDWARE, Kern::Printf("CreateThread() failed to create worker thread"));
				ret = KErrBadHandle; 
				}
			}
		else
			{
		    __KTRACE_OPT(KHARDWARE, Kern::Printf("pcap_open_live() failed to open"));
			//ret = KErrCouldNotConnect;
			PanicFromWinsEtherDrv( EPcapNull );
			}
    
		}
	
    if(ret != KErrNone)
		{
		// Failed to init all so tidy up
		if( iPcapPtr )
			{
			pcap_close(iPcapPtr);
			iPcapPtr = NULL;
			}
		__KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::DoCreate() returned with ERROR"));
		return ret;
		}


    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::DoCreate %2x:%2x:%2x:%2x:%2x:%2x",
                                         defaultConfig.iEthAddress[0], 
                                         defaultConfig.iEthAddress[1],
                                         defaultConfig.iEthAddress[2], 
                                         defaultConfig.iEthAddress[3],
                                         defaultConfig.iEthAddress[4], 
                                         defaultConfig.iEthAddress[5]));

	return ret;
	}


void  DEthernetWins::SetDriverName()
	{
	__KTRACE_OPT(KHARDWARE, 
		Kern::Printf("DEthernetWins::SetDriverName()"));

	const char* property = Property::GetString(KEpocIniEthNIFEntry);

	__ASSERT_ALWAYS( (NULL != property),
		PanicFromWinsEtherDrv(ENoNetInterface) );

	memcpy( iNetInterfaceName, property, strlen(property) );
	}


TInt DEthernetWins::SetMacAddress()
// reads MAC address from epoc.ini file and writes into defaultConfig
	{
	
	__KTRACE_OPT(KHARDWARE, 
		Kern::Printf("DEthernetWins::SetMacAddress()"));

	TUint8 tempAddress[6];

	const char *tempBuffer = Property::GetString(KEpocIniEthMACEntry);

	__ASSERT_ALWAYS( ((tempBuffer != NULL) && (strlen(tempBuffer) >= 6)),
		PanicFromWinsEtherDrv(EBadMacAddress) );


	TBuf<20> validCharsLower(_L("0123456789abcdef"));
	TBuf<20> validCharsUpper(_L("0123456789ABCDEF"));
	TUint8 value;
	TUint8 upper=0;
	TChar c;
	TInt pos; 
	TInt i;
	for( i=0; i<6; i++)
		{
		c = tempBuffer[2*i];
		if(((pos = validCharsLower.Locate(c)) == KErrNotFound) &&
			((pos = validCharsUpper.Locate(c)) == KErrNotFound))
			{
			PanicFromWinsEtherDrv( EBadMacAddress );
			}
		upper = (TUint8)pos;
		c = tempBuffer[(2*i)+1];
		if( ((pos = validCharsLower.Locate(c)) == KErrNotFound) &&
			((pos = validCharsUpper.Locate(c)) == KErrNotFound) )
			{
			PanicFromWinsEtherDrv( EBadMacAddress );
			}

		value = (TUint8)pos;
		value = (TUint8)((upper<<4) | value);
		tempAddress[i] = value;
		}

	for( i=0; i<6; i++ )
		{
		defaultConfig.iEthAddress[i] = tempAddress[i];
		}

	return KErrNone;

	}


TDfcQue* DEthernetWins::DfcQ(TInt /*aUnit*/)
// Return the DFC queue to be used for this device
	{
    __KTRACE_OPT(KHARDWARE, 
		Kern::Printf("DEthernetWins::DfcQ(TInt )"));
    return Kern::DfcQue0();
	}


TInt DEthernetWins::Start()
// Start receiving frames
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Start()"));

	TInt32 ret;
	
	// Start thread
	// ResumeThread() - from MSDN help:
	// This function decrements a thread’s suspend count. 
	// When the suspend count is decremented to zero, 
	// the execution of the thread is resumed. 
	// Return value: The thread’s previous suspend count indicates success. 0xFFFFFFFF indicates failure
	ret = ResumeThread( iThreadHandle );
	if( (0xFFFFFFFF == ret) )//|| (ret > 1) )
		return KErrGeneral;
	
    iReady = ETrue;

    return KErrNone;
	}


void DEthernetWins::Stop(TStopMode aMode)
// Stop receiving frames
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Stop(TStopMode aMode)"));

	
    switch (aMode)
		{
        case EStopNormal:
        case EStopEmergency:
			SuspendThread(iThreadHandle);
			iReady = EFalse;
			break;
		default:
			SuspendThread(iThreadHandle);
			iReady = EFalse;
			break;
		}

	}


TInt DEthernetWins::ValidateConfig(const TEthernetConfigV01 &aConfig) const
// Validate a config structure.
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::ValidateConfig(const TEthernetConfigV01 &aConfig) const"));
    switch(aConfig.iEthSpeed)
        {
        case KEthSpeedUnknown:
		case KEthSpeedAuto: 
        case KEthSpeed10BaseT:
        case KEthSpeed100BaseTX:
			break;
        default:		
        	return KErrNotSupported;
        }

    switch(aConfig.iEthDuplex)
        {
        case KEthDuplexUnknown:
		case KEthDuplexAuto:
        case KEthDuplexHalf:
        case KEthDuplexFull:
			break;
        default:
        	return KErrNotSupported;
        }

    return KErrNone;
	}


void DEthernetWins::CheckConfig(TEthernetConfigV01& /*aConfig*/)
// dummy implementation of pure virtual function
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::CheckConfig(TEthernetConfigV01& aConfig)"));
	}

TInt DEthernetWins::Send(TBuf8<KMaxEthernetPacket+32> &aBuffer)
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Send(TBuf8<KMaxEthernetPacket+32> &aBuffer)"));
    TUint16 * dataP = (TUint16 *)aBuffer.Ptr();
    TUint16 length = TUint16(aBuffer.Length());

	TInt ret;

    if (iPcapPtr)
	    {
		__KTRACE_OPT2(KHARDWARE, KSCRATCH, Kern::Printf("  >pdd tx: sending tcp seq=%u ack=%u", GetTcpSeqNumber(aBuffer), GetTcpAckNumber(aBuffer) ));
        if(pcap_sendpacket(iPcapPtr, (unsigned char *)dataP, length))
		    {
            __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Send - pcap_sendpacket() failed "));

		    ret = KErrWrite; 
	        }
        else
		    {
            __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Send: \n*** completed pcap_sendpacket() ***\n"));
            ret = KErrNone;
            }
		}
    else
		{
        __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Send() pcap adapter NOT open\n"));
        ret = KErrNotReady; 
		}
    
    return ret;
	}


TInt DEthernetWins::DisableIrqs()
// Disable normal interrupts
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::DisableIrqs()"));
    return NKern::DisableInterrupts(1);
	}


void DEthernetWins::RestoreIrqs(TInt aLevel)
// Restore normal interrupts
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::RestoreIrqs(TInt aLevel)"));
    NKern::RestoreInterrupts(aLevel);
	}


void DEthernetWins::MacConfigure(TEthernetConfigV01 &aConfig)
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::MacConfigure(TEthernetConfigV01 &aConfig)"));
    defaultConfig.iEthAddress[0] = aConfig.iEthAddress[0]; 
	defaultConfig.iEthAddress[1] = aConfig.iEthAddress[1];
	defaultConfig.iEthAddress[2] = aConfig.iEthAddress[2];
	defaultConfig.iEthAddress[3] = aConfig.iEthAddress[3];
	defaultConfig.iEthAddress[4] = aConfig.iEthAddress[4];
	defaultConfig.iEthAddress[5] = aConfig.iEthAddress[5];
	}


TInt DEthernetWins::Configure(TEthernetConfigV01 & /*aConfig*/)
// Set a wpcap filter
	{

	__KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Configure(TEthernetConfigV01 &aConfig)"));
	
    char errPcap[PCAP_ERRBUF_SIZE];
	TInt ret = KErrNone;
    char filter[128];
    struct bpf_program fcode;
    bpf_u_int32 SubNet,NetMask;

	//obtain the subnet
    pcap_lookupnet(	iNetInterfaceName, &SubNet, &NetMask, errPcap);

    sprintf(filter, 
            PCAP_FILTER, //pcap_filter, 
            defaultConfig.iEthAddress[0],
            defaultConfig.iEthAddress[1],
            defaultConfig.iEthAddress[2],
            defaultConfig.iEthAddress[3],
            defaultConfig.iEthAddress[4],
            defaultConfig.iEthAddress[5],
            defaultConfig.iEthAddress[0],
            defaultConfig.iEthAddress[1],
            defaultConfig.iEthAddress[2],
            defaultConfig.iEthAddress[3],
            defaultConfig.iEthAddress[4],
            defaultConfig.iEthAddress[5]);


	//compile the filter
    if( 0 != pcap_compile(iPcapPtr, &fcode, filter, 1, NetMask) )
		{
		// failed to compile:
		__KTRACE_OPT(KHARDWARE, Kern::Printf("pcap_compile() failed"));
		ret = KErrUnknown; 
		}
	else
		{
		// compiled successfully: set the filter
		if( 0 != pcap_setfilter(iPcapPtr, &fcode))
			{
			__KTRACE_OPT(KHARDWARE, Kern::Printf("pcap_setfilter() failed"));
			ret = KErrUnknown; 
			}
		}

    return ret; 
	}


void DEthernetWins::GetConfig(TEthernetConfigV01 &aConfig) const
// Get the current config from defaultConfig member
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::GetConfig(TEthernetConfigV01 &aConfig) const"));
    aConfig = defaultConfig;
	}


void DEthernetWins::Caps(TDes8 &aCaps) const
// return PDD's capabilites
    {
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Caps(TDes8 &aCaps) const"));
    GetWinsEthernetsCaps(aCaps,iUnit); 
    }


void DEthernetWins::Isr( u_char *thisObject,  
                       	const struct pcap_pkthdr *header, 
                        const u_char *pkt_data)
// pcap callback fuction
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::Isr()"));

	DEthernetWins& thisPdd = *(DEthernetWins*) thisObject;
    	
	StartOfInterrupt();

	thisPdd.iPcapHeader = header;
	thisPdd.iPcapPktData = pkt_data;

    thisPdd.ReceiveIsr();
	
	EndOfInterrupt();

    return;
	}


TInt DEthernetWins::ReceiveFrame(TBuf8<KMaxEthernetPacket+32> &aBuffer, TBool okToUse)
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("DEthernetWins::ReceiveFrame(TBuf8<KMaxEthernetPacket+32> &aBuffer)"));

	// If no buffer available dump frame
    if(!okToUse)
		{
        return KErrGeneral;
		}

    aBuffer.Copy(iPcapPktData, iPcapHeader->len);
    aBuffer.SetLength(iPcapHeader->len);
	__KTRACE_OPT2(KHARDWARE, KSCRATCH, Kern::Printf("  >pdd rx: tcp seq=%u ack=%u", GetTcpSeqNumber(aBuffer), GetTcpAckNumber(aBuffer) ));
    
	return KErrNone;
	}

#ifdef _DEBUG
const TUint8 ETHER2_TYPE_IP_MSB = 0x08;
const TUint8 ETHER2_TYPE_IP_LSB = 0x00;
const TUint8 IP_TYPE_TCP        = 0x06;

TBool DEthernetWins::IsTcp(TDesC8 &aFrame)
	{
 	return (aFrame[12] == ETHER2_TYPE_IP_MSB && aFrame[13] == ETHER2_TYPE_IP_LSB && aFrame[23] == IP_TYPE_TCP);
	}

TInt DEthernetWins::GetTcpSeqNumber(TDesC8 &aFrame)
	{
 	TInt seqNum = 0;
 	if (IsTcp(aFrame))
 		seqNum = aFrame[38] << 24 | aFrame[39] << 16 | aFrame[40] << 8| aFrame[41];
 	return seqNum;
 	}

TInt DEthernetWins::GetTcpAckNumber(TDesC8 &aFrame)
 	{
 	TInt ackNum = 0;
 	if (IsTcp(aFrame))
 		ackNum = aFrame[42] << 24 | aFrame[43] << 16 | aFrame[44] << 8| aFrame[45];
 	return ackNum;
 	}
#endif

TInt WorkThread(DEthernetWins * aDEthernetWins)
	{
    __KTRACE_OPT(KHARDWARE, Kern::Printf("WorkThread(DEthernetWins * aDEthernetWins)"));

    return pcap_loop(aDEthernetWins->iPcapPtr, 0, DEthernetWins::Isr, (unsigned char *)aDEthernetWins);
	}

/**
 * @addtogroup enet_windows
 * @{
 */

/**
 * Real entry point from the Kernel: return a new driver
 * (Macro wrapping: EXPORT_C DPhysicalDevice *CreatePhysicalDevice() )
 */
DECLARE_STANDARD_PDD()
	{
	return new DDriverEthernet;
	}


/** @} */ // end of windows group