diff options
Diffstat (limited to 'src/VBox/Devices/Network/Dev3C501.cpp')
-rw-r--r-- | src/VBox/Devices/Network/Dev3C501.cpp | 2721 |
1 files changed, 2721 insertions, 0 deletions
diff --git a/src/VBox/Devices/Network/Dev3C501.cpp b/src/VBox/Devices/Network/Dev3C501.cpp new file mode 100644 index 00000000..e6e141f1 --- /dev/null +++ b/src/VBox/Devices/Network/Dev3C501.cpp @@ -0,0 +1,2721 @@ +/* $Id: Dev3C501.cpp $ */ +/** @file + * Dev3C501 - 3Com EtherLink (3C501) Ethernet Adapter Emulation. + */ + +/* + * Copyright (C) 2022-2023 Oracle and/or its affiliates. + * + * This file is part of VirtualBox base platform packages, as + * available from https://www.virtualbox.org. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation, in version 3 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, see <https://www.gnu.org/licenses>. + * + * SPDX-License-Identifier: GPL-3.0-only + */ + +/** @page pg_dev_3c501 3Com 3C501 Ethernet Controller Emulation. + * + * This software was written based on the following documents: + * + * - 3Com IBM Ethernet (IE) Controller/Transceiver + * External Reference Specification, March 15, 1983 + * - 3Com EtherLink (3C501) Adapter Technical Reference + * Manual Part No. 6405-00, November 26, 1988 + * - SEEQ 8001 EDLC Ethernet Data Link Controller + * Preliminary Data Sheet, December 1982 + * + * The emulation is compatible with 3Com 3C501 EtherLink aka IE4. It also + * appears to be compatible with the original 1982 3C500 IBM Ethernet aka + * IE; the IE and IE4 documentation is nearly identical. + * + * The EtherLink is a very early design. It has only a single 2K buffer for + * both send and receive, and was desgined long before full-duplex Ethernet + * was possible (it is capable of simultaneous send and receive, but only in + * loopback mode). If it has just received a packet, the EtherLink can't + * receive another packet until the first one has been processed by the + * host. + * + * The above problem is greatly alleviated in a VM because incoming packets + * can be buffered for a short while and don't have to be immediately + * dropped just because the adapter is currently sending or because the + * receive status register has not been read yet. + * + * The first 8 registers (station address, receive and transmit command and + * status) are implemented in the SEEQ 8001 EDLC chip. The remaining 8 + * registers are provided by the 3Com ASIC (0755-02) on the 3C501 or + * discrete chips on the 3C500. + * + * The '16 collisions' bit in the transmit command/status register is nearly + * useless. The SEEQ 8001 could retransmit automatically, but the IE/IE4 can + * not because the GP Buffer Pointer needs to be reinitialized by software + * prior to each transmit attempt. It is unclear if the 16-collision counter + * only rolls over modulo 16 or if it is cleared by something other than + * reset. + * + * The 3C501 supports DMA transfers to/from the packet buffer. Many drivers + * don't use DMA by default or at all. Due to the overhead of programming + * the DMA controller, direct I/O access (rep insb/outsb) is always faster + * in a VM. DMA would only be a win for very ancient drivers which don't use + * the rep insb/outsb instructions (those didn't exist on the 8086/8088). + * + * NB: The default DMA channel (channel 1) conflicts with the default Sound + * Blaster settings. If both 3C501 and SB16 are used, then one of them + * either needs to be reconfigured to use DMA channel other than 1 or the + * 3C501 must not use DMA. + * + * The 3Com documentation implies that writing the low byte of the Receive + * Buffer Pointer is enough to clear the pointer. Yet almost all drivers, + * including 3Com's sample code, write zeros to both the low and high bytes + * of the Receive Buffer Pointer when clearing it. BSD drivers (if_el.c) + * notably only write the low byte. It has been verified on a real 3C501 + * that the documentation is correct. Writing anything to the Receive Buffer + * Pointer LSB clears the pointer (writing to the MSB appears to have no + * effect whatsoever). + * + * If the Receive Buffer Pointer is not explicitly cleared prior to + * receiving a packet, it will simply keep incrementing from wherever it + * was. Once it hits the end of the buffer (wraps around to zero), a + * receive overflow will be triggered (because the EDLC's FIFO will no + * longer be serviced) but the buffer will contain however much data there + * was room for. Note that the SEEQ 8001 datasheet is not explicit, but the + * EDLC can probably receive frames with more than 1,500 octets of payload. + * + * The GP Buffer Pointer behavior is quite curious. It appears to be + * internally a 12-bit pointer, and its top bit (that is, bit 11) is ignored + * when addressing into the 2K buffer. When writing the MSB, the top 5 bits + * are masked (always written as zero), i.e. only a 11-bit value can be + * written. Through auto-increment, the GP Buffer Pointer can reach values + * that can be read but not written. + * + * The implementation was tested for correctness using 3Com's diagnostic + * utility (3C501.EXE, Version 2.4, 1986 and also DIAGNOSE.COM, Version 2.0, + * 1983) and "passes diagnose with flying colors". Note that the interrupt + * test does not pass in V2.3 diagnostics by default because it writes an + * EOI to port 0F820h instead of 20h, relying on the system board to decode + * only the low 10 bits of the address. PCI-based systems decode all address + * bits and writes to address 0F820h do not reach the interrupt controller. + * The 3C501.EXE utility can be run with the '-i' switch to skip interrupt + * tests; the older DIAGNOSE.COM does not have that problem. In both + * versions, the preliminary test fails if the MAC address OID is not + * 02:60:8C (the utility thinks the PROM is corrupted). + * + * 3Com's XNS driver (ETH.SYS) likewise requires the OID to be 02:60:8C, + * otherwise the driver uses 00:00:00:00:00:00 as its MAC address, which is + * not something that produces useful results. Most old drivers (NetWare, + * NDIS, XENIX) don't care about the OID, but some (BSDs, Linux, some SCO + * UNIX versions) want to see the 3Com OID. + * + * The MS Networks Client setup also requires the OID to match 3Com's when + * detecting the hardware, but the actual NDIS driver does not care. Note + * that the setup fails to detect the emulated 3C501 at the default 0x300 + * base address, but finds it at 0x310 and other addresses. + * + * Note that especially newer Linux/BSD OSes are a lost cause. Their 3C501 + * drivers are very hard to configure, broken in various ways, and likely + * untested. For example the Linux driver clears the receive buffer pointer + * at the end of the interrupt handler, which may easily happen after a + * packet was already received. In FreeBSD 6.4, the kernel crashes when the + * el0 driver is loaded. In FreeBSD 5.0, the el0 driver sends packets and + * reads packets from the card, but the OS never sees any incoming data + * (even though the receive packet counter keeps going up). + * + * The precise receive logic (when a packet is copied to the buffer, when an + * interrupt is signaled, when receive goes idle) is difficult to understand + * from the 3Com documentation, but is extensively tested by the diagnostic + * utility. The SEEQ 8001 datasheet may be easier to understand than the + * EtherLink documentation. + * + * Some drivers (e.g. NetWare DOS IPX shell and ODI drivers) like to reset + * the chip more or less after every packet is sent or received. That leads + * to a situation where the NIC is briefly unable to receive anything. If we + * drop packets in that case, we end up with well over 10% packet loss and + * terrible performance. We have to hold off and not drop packets just + * because the receiver is disabled for a moment. + * + * Note that the reset bit in the auxiliary command register does not nearly + * reset the entire chip as the documentation suggests. It may only truly + * reset the SEEQ 8001 EDLC chip. It is impossible to say how going out of + * reset affects the auxiliary command register itself, since it must be + * written to exit the reset state. The reset bit clears the EDLC transmit + * and command registers, but not the programmed station address. It also + * does not disturb the packet buffer, and it does not clear the GP Buffer + * Pointer. + * + * The default EtherLink configuration uses I/O base 300h, IRQ 3, DMA + * channel 1. Prior to May 1983, the default IRQ was 5. On old EtherLink + * cards, the I/O address was configurable from 200h-3F0h in increments of + * 16, DMA 1 or 3, and IRQ 3 or 5. Newer EtherLinks (starting circa in 1984) + * in addition allow DMA 2 and IRQ 2, 4, 6, and 7. + * + */ + + +/********************************************************************************************************************************* +* Header Files * +*********************************************************************************************************************************/ +#define LOG_GROUP LOG_GROUP_DEV_ELNK +#include <VBox/vmm/pdmdev.h> +#include <VBox/vmm/pdmnetifs.h> +#include <VBox/vmm/pgm.h> +#include <VBox/version.h> +#include <iprt/asm.h> +#include <iprt/assert.h> +#include <iprt/critsect.h> +#include <iprt/net.h> +#include <iprt/string.h> +#include <iprt/time.h> +#ifdef IN_RING3 +# include <iprt/mem.h> +# include <iprt/semaphore.h> +# include <iprt/uuid.h> +#endif + +#include "VBoxDD.h" + + +/********************************************************************************************************************************* +* Defined Constants And Macros * +*********************************************************************************************************************************/ + +#define ELNK_SAVEDSTATE_VERSION 1 + +/** Maximum number of times we report a link down to the guest (failure to send frame) */ +#define ELNK_MAX_LINKDOWN_REPORTED 3 + +/** Maximum number of times we postpone restoring a link that is temporarily down. */ +#define ELNK_MAX_LINKRST_POSTPONED 3 + +/** Maximum frame size we handle */ +#define MAX_FRAME 1536 + +/* Size of the packet buffer. */ +#define ELNK_BUF_SIZE 2048u + +/* The packet buffer address mask. */ +#define ELNK_BUF_ADR_MASK (ELNK_BUF_SIZE - 1) + +/* The GP buffer pointer address within the buffer. */ +#define ELNK_GP(pThis) ((pThis)->uGPBufPtr & ELNK_BUF_ADR_MASK) + +/* The GP buffer pointer mask. + * NB: The GP buffer pointer is internally a 12-bit counter. When addressing into the + * packet buffer, bit 11 is ignored. Required to pass 3C501 diagnostics. + */ +#define ELNK_GP_MASK 0xfff + +/* The EtherLink is an 8-bit adapter, hence DMA channels up to 3 are available. */ +#define ELNK_MAX_VALID_DMA 3 + + +/********************************************************************************************************************************* +* Structures and Typedefs * +*********************************************************************************************************************************/ + + +/** + * EtherLink Transmit Command Register. + */ +typedef struct ELNK_XMIT_CMD { + uint8_t det_ufl : 1; /* Detect underflow. */ + uint8_t det_coll : 1; /* Detect collision. */ + uint8_t det_16col : 1; /* Detect collision 16. */ + uint8_t det_succ : 1; /* Detect successful xmit. */ + uint8_t unused : 4; +} EL_XMT_CMD; + +/** + * EtherLink Transmit Status Register. + * + * We will never see any real collisions, although collisions (including 16 + * successive collisions) may be useful to report when the link is down + * (something the 3C501 does not have a concept of). + */ +typedef struct ELNK_XMIT_STAT { + uint8_t uflow : 1; /* Underflow on transmit. */ + uint8_t coll : 1; /* Collision on transmit. */ + uint8_t coll16 : 1; /* 16 collisions on transmit. */ + uint8_t ready : 1; /* Ready for a new frame. */ + uint8_t undef : 4; +} EL_XMT_STAT; + +/** Address match (adr_match) modes. */ +typedef enum { + EL_ADRM_DISABLED = 0, /* Receiver disabled. */ + EL_ADRM_PROMISC = 1, /* Receive all addresses. */ + EL_ADRM_BCAST = 2, /* Receive station + broadcast. */ + EL_ADRM_MCAST = 3 /* Receive station + multicast. */ +} EL_ADDR_MATCH; + +/** + * EtherLink Receive Command Register. + */ +typedef struct ELNK_RECV_CMD { + uint8_t det_ofl : 1; /* Detect overflow errors. */ + uint8_t det_fcs : 1; /* Detect FCS errors. */ + uint8_t det_drbl : 1; /* Detect dribble error. */ + uint8_t det_runt : 1; /* Detect short frames. */ + uint8_t det_eof : 1; /* Detect EOF (frames without overflow). */ + uint8_t acpt_good : 1; /* Accept good frames. */ + uint8_t adr_match : 2; /* Address match mode. */ +} EL_RCV_CMD; + +/** + * EtherLink Receive Status Register. + */ +typedef struct ELNK_RECV_STAT { + uint8_t oflow : 1; /* Overflow on receive. */ + uint8_t fcs : 1; /* FCS error. */ + uint8_t dribble : 1; /* Dribble error. */ + uint8_t runt : 1; /* Short frame. */ + uint8_t no_ovf : 1; /* Received packet w/o overflow. */ + uint8_t good : 1; /* Received good packet. */ + uint8_t undef : 1; + uint8_t stale : 1; /* Stale receive status. */ +} EL_RCV_STAT; + +/** Buffer control (buf_ctl) modes. */ +typedef enum { + EL_BCTL_SYSTEM = 0, /* Host has buffer access. */ + EL_BCTL_XMT_RCV = 1, /* Transmit, then receive. */ + EL_BCTL_RECEIVE = 2, /* Receive. */ + EL_BCTL_LOOPBACK = 3 /* Loopback. */ +} EL_BUFFER_CONTROL; + +/** + * EtherLink Auxiliary Status Register. + */ +typedef struct ELNK_AUX_CMD { + uint8_t ire : 1; /* Interrupt Request Enable. */ + uint8_t xmit_bf : 1; /* Xmit packets with bad FCS. */ + uint8_t buf_ctl : 2; /* Packet buffer control. */ + uint8_t unused : 1; + uint8_t dma_req : 1; /* DMA request. */ + uint8_t ride : 1; /* Request Interrupt and DMA Enable. */ + uint8_t reset : 1; /* Card in reset while set. */ +} EL_AUX_CMD; + +/** + * EtherLink Auxiliary Status Register. + */ +typedef struct ELNK_AUX_STAT { + uint8_t recv_bsy : 1; /* Receive busy. */ + uint8_t xmit_bf : 1; /* Xmit packets with bad FCS. */ + uint8_t buf_ctl : 2; /* Packet buffer control. */ + uint8_t dma_done : 1; /* DMA done. */ + uint8_t dma_req : 1; /* DMA request. */ + uint8_t ride : 1; /* Request Interrupt and DMA Enable. */ + uint8_t xmit_bsy : 1; /* Transmit busy. */ +} EL_AUX_STAT; + +/** + * Internal interrupt status. + */ +typedef struct ELNK_INTR_STAT { + uint8_t recv_intr : 1; /* Receive interrupt status. */ + uint8_t xmit_intr : 1; /* Transmit interrupt status. */ + uint8_t dma_intr : 1; /* DMA interrupt status. */ + uint8_t unused : 5; +} EL_INTR_STAT; + + +/** + * EtherLink 3C501 state. + */ +typedef struct ELNKSTATE +{ + /** Restore timer. + * This is used to disconnect and reconnect the link after a restore. */ + TMTIMERHANDLE hTimerRestore; + + /** Transmit signaller. */ + PDMTASKHANDLE hXmitTask; + /** Receive ready signaller. */ + PDMTASKHANDLE hCanRxTask; + + /** Internal interrupt flag. */ + bool fISR; + /** Internal DMA active flag. */ + bool fDMA; + /** Internal in-reset flag. */ + bool fInReset; + + /** The PROM contents. Only 8 bytes addressable, R/O. */ + uint8_t aPROM[8]; + + /** The station address programmed by the guest, W/O. */ + uint8_t aStationAddr[6]; + /** General Purpose (GP) Buffer Pointer, R/W. */ + uint16_t uGPBufPtr; + + /** Receive (RCV) Buffer Pointer, R/WC. */ + uint16_t uRCVBufPtr; + /** Transmit Command Register, W/O. */ + union { + uint8_t XmitCmdReg; + EL_XMT_CMD XmitCmd; + }; + /** Transmit Status Register, R/O. */ + union { + uint8_t XmitStatReg; + EL_XMT_STAT XmitStat; + }; + /** Receive Command Register, W/O. */ + union { + uint8_t RcvCmdReg; + EL_RCV_CMD RcvCmd; + }; + /** Receive Status Register, R/O. */ + union { + uint8_t RcvStatReg; + EL_RCV_STAT RcvStat; + }; + /** Auxiliary Command Register, W/O. */ + union { + uint8_t AuxCmdReg; + EL_AUX_CMD AuxCmd; + }; + /** Auxiliary Status Register, R/O. */ + union { + uint8_t AuxStatReg; + EL_AUX_STAT AuxStat; + }; + + /** Base port of the I/O space region. */ + RTIOPORT IOPortBase; + /** The configured ISA IRQ. */ + uint8_t uIsaIrq; + /** The configured ISA DMA channel. */ + uint8_t uIsaDma; + /** If set the link is currently up. */ + bool fLinkUp; + /** If set the link is temporarily down because of a saved state load. */ + bool fLinkTempDown; + /** Number of times we've reported the link down. */ + uint16_t cLinkDownReported; + /** Number of times we've postponed the link restore. */ + uint16_t cLinkRestorePostponed; + + /** The "hardware" MAC address. */ + RTMAC MacConfigured; + /** Internal interrupt state. */ + union { + uint8_t IntrStateReg; + EL_INTR_STAT IntrState; + }; + + /** Set if ELNKSTATER3::pDrv is not NULL. */ + bool fDriverAttached; + /** The LED. */ + PDMLED Led; + /** Status LUN: The LED ports. */ + PDMILEDPORTS ILeds; + /** Partner of ILeds. */ + R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector; + + /** Access critical section. */ + PDMCRITSECT CritSect; + /** Event semaphore for blocking on receive. */ + RTSEMEVENT hEventOutOfRxSpace; + /** We are waiting/about to start waiting for more receive buffers. */ + bool volatile fMaybeOutOfSpace; + + /* MS to wait before we enable the link. */ + uint32_t cMsLinkUpDelay; + /** The device instance number (for logging). */ + uint32_t iInstance; + + STAMCOUNTER StatReceiveBytes; + STAMCOUNTER StatTransmitBytes; + STAMCOUNTER StatPktsLostReset; +#ifdef VBOX_WITH_STATISTICS + STAMPROFILEADV StatIOReadRZ; + STAMPROFILEADV StatIOReadR3; + STAMPROFILEADV StatIOWriteRZ; + STAMPROFILEADV StatIOWriteR3; + STAMPROFILEADV StatReceive; + STAMPROFILEADV StatTransmitR3; + STAMPROFILEADV StatTransmitRZ; + STAMPROFILE StatTransmitSendR3; + STAMPROFILE StatTransmitSendRZ; + STAMPROFILE StatRxOverflow; + STAMCOUNTER StatRxOverflowWakeup; + STAMPROFILEADV StatInterrupt; + STAMCOUNTER StatResets; + STAMCOUNTER StatDropPktAdrmDis; + STAMCOUNTER StatDropPktZeroLen; + STAMCOUNTER StatDropPktVMNotRunning; + STAMCOUNTER StatDropPktNoLink; + STAMCOUNTER StatDropPktStaleRcv; +#endif /* VBOX_WITH_STATISTICS */ + + /** ISA I/O ports. */ + IOMIOPORTHANDLE hIoPortsIsa; + + /** The loopback transmit buffer (avoid stack allocations). */ + uint8_t abLoopBuf[ELNK_BUF_SIZE]; + + /** The runt pad buffer (only really needs 60 bytes). */ + uint8_t abRuntBuf[64]; + + /** The packet buffer. */ + uint8_t abPacketBuf[ELNK_BUF_SIZE]; +} ELNKSTATE, *PELNKSTATE; + + +/** + * EtherLink state for ring-3. + * + * @implements PDMIBASE + * @implements PDMINETWORKDOWN + * @implements PDMINETWORKCONFIG + * @implements PDMILEDPORTS + */ +typedef struct ELNKSTATER3 +{ + /** Pointer to the device instance. */ + PPDMDEVINSR3 pDevIns; + /** Pointer to the connector of the attached network driver. */ + PPDMINETWORKUPR3 pDrv; + /** Pointer to the attached network driver. */ + R3PTRTYPE(PPDMIBASE) pDrvBase; + /** LUN\#0 + status LUN: The base interface. */ + PDMIBASE IBase; + /** LUN\#0: The network port interface. */ + PDMINETWORKDOWN INetworkDown; + /** LUN\#0: The network config port interface. */ + PDMINETWORKCONFIG INetworkConfig; + + /** Status LUN: The LED ports. */ + PDMILEDPORTS ILeds; + /** Partner of ILeds. */ + R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector; +} ELNKSTATER3; +/** Pointer to an EtherLink state structure for ring-3. */ +typedef ELNKSTATER3 *PELNKSTATER3; + + +/** + * EtherLink state for ring-0. + */ +typedef struct ELNKSTATER0 +{ + /** Pointer to the connector of the attached network driver. */ + PPDMINETWORKUPR0 pDrv; +} ELNKSTATER0; +/** Pointer to an EtherLink state structure for ring-0. */ +typedef ELNKSTATER0 *PELNKSTATER0; + + +/** + * EtherLink state for raw-mode. + */ +typedef struct ELNKSTATERC +{ + /** Pointer to the connector of the attached network driver. */ + PPDMINETWORKUPRC pDrv; +} ELNKSTATERC; +/** Pointer to an EtherLink state structure for raw-mode. */ +typedef ELNKSTATERC *PELNKSTATERC; + + +/** The EtherLink state structure for the current context. */ +typedef CTX_SUFF(ELNKSTATE) ELNKSTATECC; +/** Pointer to an EtherLink state structure for the current + * context. */ +typedef CTX_SUFF(PELNKSTATE) PELNKSTATECC; + + +#ifndef VBOX_DEVICE_STRUCT_TESTCASE + + +/********************************************************************************************************************************* +* Internal Functions * +*********************************************************************************************************************************/ + +static int elnkAsyncTransmit(PPDMDEVINS pDevIns, PELNKSTATE pThis, PELNKSTATECC pThisCC, bool fOnWorkerThread); + +/** + * Checks if the link is up. + * @returns true if the link is up. + * @returns false if the link is down. + */ +DECLINLINE(bool) elnkIsLinkUp(PELNKSTATE pThis) +{ + return pThis->fDriverAttached && !pThis->fLinkTempDown && pThis->fLinkUp; +} + + +#ifndef ETHER_IS_MULTICAST /* Net/Open BSD macro it seems */ +#define ETHER_IS_MULTICAST(a) ((*(uint8_t *)(a)) & 1) +#endif + +#define ETHER_ADDR_LEN ETH_ALEN +#define ETH_ALEN 6 +#pragma pack(1) +struct ether_header /** @todo Use RTNETETHERHDR? */ +{ + uint8_t ether_dhost[ETH_ALEN]; /**< destination ethernet address */ + uint8_t ether_shost[ETH_ALEN]; /**< source ethernet address */ + uint16_t ether_type; /**< packet type ID field */ +}; +#pragma pack() + + +/** + * Check if incoming frame matches the station address. + */ +DECLINLINE(int) padr_match(PELNKSTATE pThis, const uint8_t *buf) +{ + struct ether_header *hdr = (struct ether_header *)buf; + int result; + + /* Checks own + broadcast as well as own + multicast. */ + result = (pThis->RcvCmd.adr_match >= EL_ADRM_BCAST) && !memcmp(hdr->ether_dhost, pThis->aStationAddr, 6); + + return result; +} + + +/** + * Check if incoming frame is an accepted broadcast frame. + */ +DECLINLINE(int) padr_bcast(PELNKSTATE pThis, const uint8_t *buf) +{ + static uint8_t aBCAST[6] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff }; + struct ether_header *hdr = (struct ether_header *)buf; + int result = (pThis->RcvCmd.adr_match == EL_ADRM_BCAST) && !memcmp(hdr->ether_dhost, aBCAST, 6); + return result; +} + + +/** + * Check if incoming frame is an accepted multicast frame. + */ +DECLINLINE(int) padr_mcast(PELNKSTATE pThis, const uint8_t *buf) +{ + struct ether_header *hdr = (struct ether_header *)buf; + int result = (pThis->RcvCmd.adr_match == EL_ADRM_MCAST) && ETHER_IS_MULTICAST(hdr->ether_dhost); + return result; +} + + +/** + * Update the device IRQ line based on internal state. + */ +static void elnkUpdateIrq(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + bool fISR = false; + + STAM_PROFILE_ADV_START(&pThis->StatInterrupt, a); + + /* IRQ is active if any interrupt source is active and interrupts + * are enabled via RIDE or IRE. + */ + if (pThis->IntrStateReg && (pThis->AuxCmd.ride || pThis->AuxCmd.ire)) + fISR = true; + + Log2(("#%d set irq fISR=%d\n", pThis->iInstance, fISR)); + + /* The IRQ line typically does not change. */ + if (RT_UNLIKELY(fISR != pThis->fISR)) + { + Log(("#%d IRQ=%d, state=%d\n", pThis->iInstance, pThis->uIsaIrq, fISR)); + PDMDevHlpISASetIrq(pDevIns, pThis->uIsaIrq, fISR); + pThis->fISR = fISR; + } + STAM_PROFILE_ADV_STOP(&pThis->StatInterrupt, a); +} + + +/** + * Perform a software reset of the NIC. + */ +static void elnkSoftReset(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + LogFlowFunc(("#%d:\n", pThis->iInstance)); + + /* Clear some of the user-visible register state. */ + pThis->XmitCmdReg = 0; + pThis->XmitStatReg = 0; + pThis->RcvCmdReg = 0; + pThis->RcvStatReg = 0; + pThis->AuxCmdReg = 0; + pThis->AuxStatReg = 0; + + /* The "stale receive status" is cleared by receiving an "interesting" packet. */ + pThis->RcvStat.stale = 1; + + /* By virtue of setting the buffer control to system, transmit is set to busy. */ + pThis->AuxStat.xmit_bsy = 1; + + /* Clear internal interrupt state. */ + pThis->IntrStateReg = 0; + elnkUpdateIrq(pDevIns, pThis); + + /* Note that a soft reset does not clear the packet buffer; software often + * assumes that it survives soft reset. The programmed station address is + * likewise not reset, and the buffer pointers are not reset either. + * Verified on a real 3C501. + */ + + /* No longer in reset state. */ + pThis->fInReset = false; +} + +#ifdef IN_RING3 + +static DECLCALLBACK(void) elnkR3WakeupReceive(PPDMDEVINS pDevIns) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + STAM_COUNTER_INC(&pThis->StatRxOverflowWakeup); + if (pThis->hEventOutOfRxSpace != NIL_RTSEMEVENT) + RTSemEventSignal(pThis->hEventOutOfRxSpace); +} + +/** + * @callback_method_impl{FNPDMTASKDEV, + * Signal to R3 that NIC is ready to receive a packet. + */ +static DECLCALLBACK(void) elnkR3CanRxTaskCallback(PPDMDEVINS pDevIns, void *pvUser) +{ + RT_NOREF(pvUser); + elnkR3WakeupReceive(pDevIns); +} + +#endif /* IN_RING3 */ + + +/** + * Write incoming data into the packet buffer. + */ +static void elnkReceiveLocked(PPDMDEVINS pDevIns, PELNKSTATE pThis, const uint8_t *src, size_t cbToRecv, bool fLoopback) +{ + int is_padr = 0, is_bcast = 0, is_mcast = 0; + union { + uint8_t RcvStatNewReg; + EL_RCV_STAT RcvStatNew; + }; + + /* + * Drop all packets if the VM is not running yet/anymore. + */ + VMSTATE enmVMState = PDMDevHlpVMState(pDevIns); + if ( enmVMState != VMSTATE_RUNNING + && enmVMState != VMSTATE_RUNNING_LS) + { + STAM_COUNTER_INC(&pThis->StatDropPktVMNotRunning); + return; + } + + /* Drop everything if address matching is disabled. */ + if (RT_UNLIKELY(pThis->RcvCmd.adr_match == EL_ADRM_DISABLED)) + { + STAM_COUNTER_INC(&pThis->StatDropPktAdrmDis); + return; + } + + /* Drop zero-length packets (how does that even happen?). */ + if (RT_UNLIKELY(!cbToRecv)) + { + STAM_COUNTER_INC(&pThis->StatDropPktZeroLen); + return; + } + + /* + * Drop all packets if the cable is not connected (and not in loopback). + */ + if (RT_UNLIKELY(!elnkIsLinkUp(pThis) && !fLoopback)) + { + STAM_COUNTER_INC(&pThis->StatDropPktNoLink); + return; + } + + /* + * Do not receive further packets until receive status was read. + */ + if (RT_UNLIKELY(pThis->RcvStat.stale == 0)) + { + STAM_COUNTER_INC(&pThis->StatDropPktStaleRcv); + return; + } + + LogFlowFunc(("#%d: size on wire=%d, RCV ptr=%u\n", pThis->iInstance, cbToRecv, pThis->uRCVBufPtr)); + + /* + * Perform address matching. Packets which do not pass the address + * filter are always ignored. + */ + /// @todo cbToRecv must be 6 or more (complete address) + if ( pThis->RcvCmd.adr_match == EL_ADRM_PROMISC /* promiscuous enabled */ + || (is_padr = padr_match(pThis, src)) + || (is_bcast = padr_bcast(pThis, src)) + || (is_mcast = padr_mcast(pThis, src))) + { + uint8_t *dst = pThis->abPacketBuf + pThis->uRCVBufPtr; + + Log2Func(("#%d Packet passed address filter (is_padr=%d, is_bcast=%d, is_mcast=%d), size=%d\n", pThis->iInstance, cbToRecv, is_padr, is_bcast, is_mcast)); + + /* Receive status is evaluated from scratch. The stale bit must remain set until we know better. */ + RcvStatNewReg = 0; + RcvStatNew.stale = 1; + pThis->RcvStatReg = 0x80; + + /* Detect errors: Runts, overflow, and FCS errors. + * NB: Dribble errors can not happen because we can only receive an + * integral number of bytes. FCS errors are only possible in loopback + * mode in case the FCS is deliberately corrupted. + */ + + /* See if we need to pad, and how much. Have to be careful because the + * Receive Buffer Pointer might be near the end of the buffer. + */ + if (RT_UNLIKELY(cbToRecv < 60)) + { + /* In loopback mode only, short packets are flagged as errors because + * diagnostic tools want to see the errors. Otherwise they're padded to + * minimum length (if packet came over the wire, it should have been + * properly padded). + */ + /// @todo This really is kind of wrong. We shouldn't be doing any + /// padding here, it should be done by the sending side! + if (!fLoopback) + { + memset(pThis->abRuntBuf, 0, sizeof(pThis->abRuntBuf)); + memcpy(pThis->abRuntBuf, src, cbToRecv); + cbToRecv = 60; + src = pThis->abRuntBuf; + } + else + { + LogFunc(("#%d runt, size=%d\n", pThis->iInstance, cbToRecv)); + RcvStatNew.runt = 1; + } + } + + /* We don't care how big the frame is; if it fits into the buffer, all is + * good. But conversely if the Receive Buffer Pointer is initially near the + * end of the buffer, a small frame can trigger an overflow. + */ + if (pThis->uRCVBufPtr + cbToRecv <= ELNK_BUF_SIZE) + { + RcvStatNew.no_ovf = 1; + } + else + { + LogFunc(("#%d overflow, size=%d\n", pThis->iInstance, cbToRecv)); + RcvStatNew.oflow = 1; + } + + if (fLoopback && pThis->AuxCmd.xmit_bf) + { + LogFunc(("#%d bad FCS\n", pThis->iInstance)); + RcvStatNew.fcs = 1; + } + + /* Error-free packets are considered good. */ + if (RcvStatNew.no_ovf && !RcvStatNew.fcs && !RcvStatNew.runt) + RcvStatNew.good = 1; + + uint16_t cbCopy = (uint16_t)RT_MIN(ELNK_BUF_SIZE - pThis->uRCVBufPtr, cbToRecv); + + /* All packets that passed the address filter are copied to the buffer. */ + + STAM_REL_COUNTER_ADD(&pThis->StatReceiveBytes, cbCopy); + + /* Copy incoming data to the packet buffer. NB: Starts at the current + * Receive Buffer Pointer position. + */ + memcpy(dst, src, cbCopy); + + /* Packet length is indicated via the receive buffer pointer. */ + pThis->uRCVBufPtr = (pThis->uRCVBufPtr + cbCopy) & ELNK_GP_MASK; + + Log2Func(("Received packet, size=%d, RP=%u\n", cbCopy, pThis->uRCVBufPtr)); + + /* + * If one of the "interesting" conditions was hit, stop receiving until + * the status register is read (mark it not stale). + * NB: The precise receive logic is not very well described in the EtherLink + * documentation. It was refined using the 3C501.EXE diagnostic utility. + */ + if ( (RcvStatNew.good && pThis->RcvCmd.acpt_good) + || (RcvStatNew.no_ovf && pThis->RcvCmd.det_eof) + || (RcvStatNew.runt && pThis->RcvCmd.det_runt) + || (RcvStatNew.dribble && pThis->RcvCmd.det_drbl) + || (RcvStatNew.fcs && pThis->RcvCmd.det_fcs) + || (RcvStatNew.oflow && pThis->RcvCmd.det_ofl)) + { + pThis->AuxStat.recv_bsy = 0; + pThis->IntrState.recv_intr = 1; + RcvStatNew.stale = 0; /* Prevents further receive until set again. */ + } + /* Finally update the receive status. */ + pThis->RcvStat = RcvStatNew; + + LogFlowFunc(("#%d: RcvCmd=%02X, RcvStat=%02X, RCVBufPtr=%u\n", pThis->iInstance, pThis->RcvCmdReg, pThis->RcvStatReg, pThis->uRCVBufPtr)); + elnkUpdateIrq(pDevIns, pThis); + } +} + + +/** + * Transmit data from the packet buffer. + * + * @returns VBox status code. VERR_TRY_AGAIN is returned if we're busy. + * + * @param pDevIns The device instance. + * @param pThis The EtherLink shared instance + * data. + * @param pThisCC The EtherLink state data for the + * current context. + * @param fOnWorkerThread Whether we're on a worker thread or on an EMT. + */ +static int elnkXmitBuffer(PPDMDEVINS pDevIns, PELNKSTATE pThis, PELNKSTATECC pThisCC, bool fOnWorkerThread) +{ + RT_NOREF_PV(fOnWorkerThread); + int rc; + + /* + * Grab the xmit lock of the driver as well as the 3C501 device state. + */ + PPDMINETWORKUP pDrv = pThisCC->pDrv; + if (pDrv) + { + rc = pDrv->pfnBeginXmit(pDrv, false /*fOnWorkerThread*/); + if (RT_FAILURE(rc)) + return rc; + } + rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + if (RT_SUCCESS(rc)) + { + /** @todo check if we're supposed to suspend now. */ + /* + * Do the transmitting. + */ + int rc2 = elnkAsyncTransmit(pDevIns, pThis, pThisCC, false /*fOnWorkerThread*/); + AssertReleaseRC(rc2); + + /* + * Release the locks. + */ + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + } + else + AssertLogRelRC(rc); + if (pDrv) + pDrv->pfnEndXmit(pDrv); + + return rc; +} + + +#ifdef IN_RING3 + +/** + * @callback_method_impl{FNPDMTASKDEV, + * This is just a very simple way of delaying sending to R3. + */ +static DECLCALLBACK(void) elnkR3XmitTaskCallback(PPDMDEVINS pDevIns, void *pvUser) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + PELNKSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PELNKSTATECC); + NOREF(pvUser); + + /* + * Transmit if we can. + */ + elnkXmitBuffer(pDevIns, pThis, pThisCC, true /*fOnWorkerThread*/); +} +#endif /* IN_RING3 */ + + +/** + * Allocates a scatter/gather buffer for a transfer. + * + * @returns See PPDMINETWORKUP::pfnAllocBuf. + * @param pThis The shared state data. + * @param pThisCC The current context state data. + * @param cbMin The minimum buffer size. + * @param fLoopback Set if we're in loopback mode. + * @param pSgLoop Pointer to stack storage for the loopback SG. + * @param ppSgBuf Where to return the SG buffer descriptor on success. + * Always set. + */ +DECLINLINE(int) elnkXmitAllocBuf(PELNKSTATE pThis, PELNKSTATECC pThisCC, size_t cbMin, bool fLoopback, + PPDMSCATTERGATHER pSgLoop, PPPDMSCATTERGATHER ppSgBuf) +{ + int rc; + + if (!fLoopback) + { + PPDMINETWORKUP pDrv = pThisCC->pDrv; + if (RT_LIKELY(pDrv)) + { + rc = pDrv->pfnAllocBuf(pDrv, cbMin, NULL /*pGso*/, ppSgBuf); + AssertMsg(rc == VINF_SUCCESS || rc == VERR_TRY_AGAIN || rc == VERR_NET_DOWN || rc == VERR_NO_MEMORY, ("%Rrc\n", rc)); + if (RT_FAILURE(rc)) + *ppSgBuf = NULL; + } + else + { + rc = VERR_NET_DOWN; + *ppSgBuf = NULL; + } + } + else + { + /* Fake loopback allocator. */ + pSgLoop->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1; + pSgLoop->cbUsed = 0; + pSgLoop->cbAvailable = sizeof(pThis->abLoopBuf); + pSgLoop->pvAllocator = pThis; + pSgLoop->pvUser = NULL; + pSgLoop->cSegs = 1; + pSgLoop->aSegs[0].cbSeg = sizeof(pThis->abLoopBuf); + pSgLoop->aSegs[0].pvSeg = pThis->abLoopBuf; + *ppSgBuf = pSgLoop; + rc = VINF_SUCCESS; + } + return rc; +} + + +/** + * Sends the scatter/gather buffer. + * + * Wrapper around PDMINETWORKUP::pfnSendBuf, so check it out for the fine print. + * + * @returns See PDMINETWORKUP::pfnSendBuf. + * @param pDevIns The device instance. + * @param pThis The shared EtherLink state data. + * @param pThisCC The current context state data. + * @param fLoopback Set if we're in loopback mode. + * @param pSgBuf The SG to send. + * @param fOnWorkerThread Set if we're being called on a work thread. Clear + * if an EMT. + */ +DECLINLINE(int) elnkXmitSendBuf(PPDMDEVINS pDevIns, PELNKSTATE pThis, PELNKSTATECC pThisCC, bool fLoopback, PPDMSCATTERGATHER pSgBuf, bool fOnWorkerThread) +{ + int rc; + STAM_REL_COUNTER_ADD(&pThis->StatTransmitBytes, pSgBuf->cbUsed); + if (!fLoopback) + { + STAM_PROFILE_START(&pThis->CTX_SUFF_Z(StatTransmitSend), a); + if (pSgBuf->cbUsed > 70) /* unqualified guess */ + pThis->Led.Asserted.s.fWriting = pThis->Led.Actual.s.fWriting = 1; + + PPDMINETWORKUP pDrv = pThisCC->pDrv; + if (RT_LIKELY(pDrv)) + { + rc = pDrv->pfnSendBuf(pDrv, pSgBuf, fOnWorkerThread); + AssertMsg(rc == VINF_SUCCESS || rc == VERR_NET_DOWN || rc == VERR_NET_NO_BUFFER_SPACE, ("%Rrc\n", rc)); + } + else + rc = VERR_NET_DOWN; + + pThis->Led.Actual.s.fWriting = 0; + STAM_PROFILE_STOP(&pThis->CTX_SUFF_Z(StatTransmitSend), a); + } + else + { + /* Loopback, immediately send buffer to the receive path. */ + Assert(pSgBuf->pvAllocator == (void *)pThis); + pThis->Led.Asserted.s.fReading = pThis->Led.Actual.s.fReading = 1; + + LogFlowFunc(("#%d: loopback (%u bytes)\n", pThis->iInstance, pSgBuf->cbUsed)); + elnkReceiveLocked(pDevIns, pThis, pThis->abLoopBuf, pSgBuf->cbUsed, fLoopback); + pThis->Led.Actual.s.fReading = 0; + rc = VINF_SUCCESS; + } + return rc; +} + + +/** + * Reads the entire frame into the scatter gather buffer. + */ +DECLINLINE(void) elnkXmitRead(PPDMDEVINS pDevIns, PELNKSTATE pThis, const unsigned cbFrame, PPDMSCATTERGATHER pSgBuf) +{ + Assert(PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect)); RT_NOREF(pDevIns); + Assert(pSgBuf->cbAvailable >= cbFrame); + + pSgBuf->cbUsed = cbFrame; + memcpy(pSgBuf->aSegs[0].pvSeg, &pThis->abPacketBuf[ELNK_GP(pThis)], cbFrame); +} + +/** + * Try to transmit a frame. + */ +static void elnkTransmit(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + PELNKSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PELNKSTATECC); + + /* + * Transmit the packet if possible, defer it if we cannot do it + * in the current context. + */ +#if defined(IN_RING0) || defined(IN_RC) + if (!pThisCC->pDrv) + { + int rc = PDMDevHlpTaskTrigger(pDevIns, pThis->hXmitTask); + AssertRC(rc); + } + else +#endif + { + int rc = elnkXmitBuffer(pDevIns, pThis, pThisCC, false /*fOnWorkerThread*/); + if (rc == VERR_TRY_AGAIN) + rc = VINF_SUCCESS; + AssertRC(rc); + } +} + + +/** + * If a packet is waiting, poke the receiving machinery. + * + * @threads EMT. + */ +static void elnkKickReceive(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + /* Some drivers (e.g. NetWare IPX shell/ODI drivers) first go to receive mode through + * the aux command register and only then enable address matching. + */ + if ((pThis->AuxStat.recv_bsy == 1) && (pThis->RcvCmd.adr_match != EL_ADRM_DISABLED)) + { + if (pThis->fMaybeOutOfSpace) + { +#ifdef IN_RING3 + elnkR3WakeupReceive(pDevIns); +#else + int rc = PDMDevHlpTaskTrigger(pDevIns, pThis->hCanRxTask); + AssertRC(rc); +#endif + } + } + +} + +/** + * Try transmitting a frame. + * + * @threads TX or EMT. + */ +static int elnkAsyncTransmit(PPDMDEVINS pDevIns, PELNKSTATE pThis, PELNKSTATECC pThisCC, bool fOnWorkerThread) +{ + Assert(PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect)); + + /* + * Just drop it if not transmitting. Can happen with delayed transmits + * if transmit was disabled in the meantime. + */ + if (RT_UNLIKELY(!pThis->AuxStat.xmit_bsy)) + { + LogFunc(("#%d: Nope, xmit disabled (fOnWorkerThread=%RTbool)\n", pThis->iInstance, fOnWorkerThread)); + return VINF_SUCCESS; + } + + if (RT_UNLIKELY((pThis->AuxCmd.buf_ctl != EL_BCTL_XMT_RCV) && (pThis->AuxCmd.buf_ctl != EL_BCTL_LOOPBACK))) + { + LogFunc(("#%d: Nope, not in xmit-then-receive or loopback state (fOnWorkerThread=%RTbool)\n", pThis->iInstance, fOnWorkerThread)); + return VINF_SUCCESS; + } + + /* + * Blast out data from the packet buffer. + */ + int rc; + STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a); + do + { + /* Don't send anything when the link is down. */ + if (RT_UNLIKELY( !elnkIsLinkUp(pThis) + && pThis->cLinkDownReported > ELNK_MAX_LINKDOWN_REPORTED) + ) + break; + + bool const fLoopback = pThis->AuxCmd.buf_ctl == EL_BCTL_LOOPBACK; + PDMSCATTERGATHER SgLoop; + PPDMSCATTERGATHER pSgBuf; + + /* + * Sending is easy peasy, there is by definition always + * a complete packet on hand. + */ + const unsigned cb = ELNK_BUF_SIZE - ELNK_GP(pThis); /* Packet size. */ + LogFunc(("#%d: cb=%d\n", pThis->iInstance, cb)); + + pThis->XmitStatReg = 0; /* Clear transmit status before filling it out. */ + + if (RT_LIKELY(elnkIsLinkUp(pThis) || fLoopback)) + { + if (RT_LIKELY(cb <= MAX_FRAME)) + { + rc = elnkXmitAllocBuf(pThis, pThisCC, cb, fLoopback, &SgLoop, &pSgBuf); + if (RT_SUCCESS(rc)) + { + elnkXmitRead(pDevIns, pThis, cb, pSgBuf); + rc = elnkXmitSendBuf(pDevIns, pThis, pThisCC, fLoopback, pSgBuf, fOnWorkerThread); + Log2Func(("#%d: rc=%Rrc\n", pThis->iInstance, rc)); + } + else if (rc == VERR_TRY_AGAIN) + { + STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a); + LogFunc(("#%d: rc=%Rrc\n", pThis->iInstance, rc)); + return VINF_SUCCESS; + } + if (RT_SUCCESS(rc)) + pThis->XmitStat.ready = 1; + else + pThis->XmitStat.coll = 1; /* Pretend there was a collision. */ + } + else + { + /* Signal error, as this violates the Ethernet specs. */ + /** @todo check if the correct error is generated. */ + LogRel(("3C501#%d: illegal giant frame (%u bytes) -> signalling error\n", pThis->iInstance, cb)); + } + } + else + { + /* Signal a transmit error pretending there was a collision. */ + pThis->cLinkDownReported++; + pThis->XmitStat.coll = 1; + } + /* Transmit officially done, update register state. */ + pThis->AuxStat.xmit_bsy = 0; + pThis->IntrState.xmit_intr = !!(pThis->XmitCmdReg & pThis->XmitStatReg); + LogFlowFunc(("#%d: XmitCmd=%02X, XmitStat=%02X\n", pThis->iInstance, pThis->XmitCmdReg, pThis->XmitStatReg)); + + /* NB: After a transmit, the GP Buffer Pointer points just past + * the end of the packet buffer (3C501 diagnostics). + */ + pThis->uGPBufPtr = ELNK_BUF_SIZE; + + /* NB: The buffer control does *not* change to Receive and stays the way it was. */ + if (RT_UNLIKELY(!fLoopback)) + { + pThis->AuxStat.recv_bsy = 1; /* Receive Busy now set until a packet is received. */ + elnkKickReceive(pDevIns, pThis); + } + } while (0); /* No loop, because there isn't ever more than one packet to transmit. */ + + elnkUpdateIrq(pDevIns, pThis); + + STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a); + + return VINF_SUCCESS; +} + +/* -=-=-=-=-=- I/O Port access -=-=-=-=-=- */ + + +static int elnkCsrWrite(PPDMDEVINS pDevIns, PELNKSTATE pThis, uint8_t data) +{ + int rc = VINF_SUCCESS; + bool fTransmit = false; + bool fReceive = false; + bool fDMAR; + union { + uint8_t reg; + EL_AUX_CMD val; + }; + + reg = data; + + /* Handle reset first. */ + if (pThis->AuxCmd.reset != val.reset) + { + if (val.reset) + { + /* Card is placed into reset. Just set the flag. NB: When in reset + * state, we permit writes to other registers, but those have no + * effect and will be overwritten when the card is taken out of reset. + */ + LogFunc(("#%d: Card going into reset\n", pThis->iInstance)); + pThis->fInReset = true; + + /* Many EtherLink drivers like to reset the card a lot. That can lead to + * packet loss if a packet was already received before the card was reset. + */ + if (RT_UNLIKELY(!pThis->RcvStat.stale)) + STAM_REL_COUNTER_INC(&pThis->StatPktsLostReset); + } + else + { + /* Card is being taken out of reset. */ + LogFunc(("#%d: Card going out of reset\n", pThis->iInstance)); + STAM_COUNTER_INC(&pThis->StatResets); + elnkSoftReset(pDevIns, pThis); + } + pThis->AuxCmd.reset = val.reset; /* Update the reset bit, if nothing else. */ + } + + /* If the card is in reset, stop right here. */ + if (pThis->fInReset) + return rc; + + /* Evaluate DMA state. If it changed, we'll have to go back to R3. */ + fDMAR = val.dma_req && val.ride; + if (fDMAR != pThis->fDMA) +#ifdef IN_RING3 + { + /* Start/stop DMA as requested. */ + pThis->fDMA = fDMAR; + PDMDevHlpDMASetDREQ(pDevIns, pThis->uIsaDma, fDMAR); + if (fDMAR) + PDMDevHlpDMASchedule(pDevIns); + Log(("3C501#%d: DMARQ for channel %u set to %u\n", pThis->iInstance, pThis->uIsaDma, fDMAR)); + } +#else + return VINF_IOM_R3_IOPORT_WRITE; +#endif + + /* Interrupt enable changes. */ + if ((pThis->AuxCmd.ire != val.ire) || (pThis->AuxCmd.ride != val.ride)) + { + pThis->AuxStat.ride = pThis->AuxCmd.ride = val.ride; + pThis->AuxCmd.ire = val.ire; /* NB: IRE is not visible in the aux status register. */ + } + + /* DMA Request changes. */ + if (pThis->AuxCmd.dma_req != val.dma_req) + { + pThis->AuxStat.dma_req = pThis->AuxCmd.dma_req = val.dma_req; + if (!val.dma_req) + { + /* Clearing the DMA Request bit also clears the DMA Done status bit and any DMA interrupt. */ + pThis->IntrState.dma_intr = 0; + pThis->AuxStat.dma_done = 0; + } + } + + /* Packet buffer control changes. */ + if (pThis->AuxCmd.buf_ctl != val.buf_ctl) + { +#ifdef LOG_ENABLED + static const char *apszBuffCntrl[4] = { "System", "Xmit then Recv", "Receive", "Loopback" }; + Log(("3C501#%d: Packet buffer control `%s' -> `%s'\n", pThis->iInstance, apszBuffCntrl[pThis->AuxCmd.buf_ctl], apszBuffCntrl[val.buf_ctl])); +#endif + if (val.buf_ctl == EL_BCTL_XMT_RCV) + { + /* Transmit, then receive. */ + Log2(("3C501#%d: Transmit %u bytes\n%Rhxs\nxmit_bsy=%u\n", pThis->iInstance, ELNK_BUF_SIZE - pThis->uGPBufPtr, &pThis->abPacketBuf[pThis->uGPBufPtr], pThis->AuxStat.xmit_bsy)); + fTransmit = true; + pThis->AuxStat.recv_bsy = 0; + } + else if (val.buf_ctl == EL_BCTL_SYSTEM) + { + pThis->AuxStat.xmit_bsy = 1; /* Transmit Busy is set here and cleared once actual transmit completes. */ + pThis->AuxStat.recv_bsy = 0; + } + else if (val.buf_ctl == EL_BCTL_RECEIVE) + { + /* Special case: If going from xmit-then-receive mode to receive mode, and we received + * a packet already (right after the receive), don't restart receive and lose the already + * received packet. + */ + if (!pThis->uRCVBufPtr) + fReceive = true; + } + else + { + /* For loopback, we go through the regular transmit and receive path. That may be an + * overkill but the receive path is too complex for a special loopback-only case. + */ + fTransmit = true; + pThis->AuxStat.recv_bsy = 1; /* Receive Busy now set until a packet is received. */ + } + pThis->AuxStat.buf_ctl = pThis->AuxCmd.buf_ctl = val.buf_ctl; + } + + /* NB: Bit 1 (xmit_bf, transmit packets with bad FCS) is a simple control + * bit which does not require special handling here. Just copy it over. + */ + pThis->AuxStat.xmit_bf = pThis->AuxCmd.xmit_bf = val.xmit_bf; + + /* There are multiple bits that affect interrupt state. Handle them now. */ + elnkUpdateIrq(pDevIns, pThis); + + /* After fully updating register state, do a transmit (including loopback) or receive. */ + if (fTransmit) + elnkTransmit(pDevIns, pThis); + else if (fReceive) + { + pThis->AuxStat.recv_bsy = 1; /* Receive Busy now set until a packet is received. */ + elnkKickReceive(pDevIns, pThis); + } + + return rc; +} + +static int elIoWrite(PPDMDEVINS pDevIns, PELNKSTATE pThis, uint32_t addr, uint32_t val) +{ + int reg = addr & 0xf; + int rc = VINF_SUCCESS; + + Log2Func(("#%d: addr=%#06x val=%#04x\n", pThis->iInstance, addr, val & 0xff)); + + switch (reg) + { + case 0x00: /* Six bytes of station address. */ + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + pThis->aStationAddr[reg] = val; + break; + + case 0x06: /* Receive command. */ + { + EL_RCV_CMD OldRcvCmd = pThis->RcvCmd; + + pThis->RcvCmdReg = val; + /* If address filter just got enabled, receive may need a kick. */ + if (OldRcvCmd.adr_match == EL_ADRM_DISABLED && pThis->RcvCmd.adr_match != EL_ADRM_DISABLED) + elnkKickReceive(pDevIns, pThis); + Log2(("Receive Command register set to %02X\n", pThis->RcvCmdReg)); + break; + } + + case 0x07: /* Transmit command. */ + pThis->XmitCmdReg = val; + Log2(("Transmit Command register set to %02X\n", pThis->XmitCmdReg)); + break; + + case 0x08: /* GP Buffer pointer LSB. */ + pThis->uGPBufPtr = (pThis->uGPBufPtr & 0xff00) | (uint8_t)val; + Log2(("GP Buffer Pointer LSB write, now %u\n", pThis->uGPBufPtr)); + break; + + case 0x09: /* GP Buffer pointer MSB. */ + pThis->uGPBufPtr = ((uint8_t)val << 8) | RT_LOBYTE(pThis->uGPBufPtr); + Log2(("GP Buffer Pointer MSB write, now %u\n", pThis->uGPBufPtr)); + break; + + case 0x0a: /* RCV Buffer pointer clear. */ + pThis->uRCVBufPtr = 0; + Log2(("RCV Buffer Pointer cleared (%02X)\n", val)); + break; + + case 0x0b: /* RCV buffer pointer MSB. */ + case 0x0c: /* Ethernet address PROM window. */ + case 0x0d: /* Undocumented. */ + Log(("Writing read-only register %02X!\n", reg)); + break; + + case 0x0e: /* Auxiliary Command (CSR). */ + rc = elnkCsrWrite(pDevIns, pThis, val); + break; + + case 0x0f: /* Buffer window. */ + /* Writes use low 11 bits of GP buffer pointer, auto-increment. */ + if (pThis->AuxCmd.buf_ctl != EL_BCTL_SYSTEM) + { + Log(("Packet buffer write ignored, buf_ctl=%u!\n", pThis->AuxCmd.buf_ctl)); + /// @todo Does this still increment GPBufPtr? + break; + } + pThis->abPacketBuf[ELNK_GP(pThis)] = val; + pThis->uGPBufPtr = (pThis->uGPBufPtr + 1) & ELNK_GP_MASK; + break; + } + + return rc; +} + +static uint32_t elIoRead(PPDMDEVINS pDevIns, PELNKSTATE pThis, uint32_t addr, int *pRC) +{ + uint32_t val = UINT32_MAX; + + *pRC = VINF_SUCCESS; + + switch (addr & 0x0f) + { + case 0x00: /* Receive status register aliases. The SEEQ 8001 */ + case 0x02: /* EDLC clearly only decodes one bit for reads. */ + case 0x04: + case 0x06: /* Receive status register. */ + val = pThis->RcvStatReg; + pThis->RcvStat.stale = 1; /* Allows further reception. */ + pThis->IntrState.recv_intr = 0; /* Reading clears receive interrupt. */ + elnkUpdateIrq(pDevIns, pThis); + break; + + case 0x01: /* Transmit status register aliases. */ + case 0x03: + case 0x05: + case 0x07: /* Transmit status register. */ + val = pThis->XmitStatReg; + pThis->IntrState.xmit_intr = 0; /* Reading clears transmit interrupt. */ + elnkUpdateIrq(pDevIns, pThis); + break; + + case 0x08: /* GP Buffer pointer LSB. */ + val = RT_LOBYTE(pThis->uGPBufPtr); + break; + + case 0x09: /* GP Buffer pointer MSB. */ + val = RT_HIBYTE(pThis->uGPBufPtr); + break; + + case 0x0a: /* RCV Buffer pointer LSB. */ + val = RT_LOBYTE(pThis->uRCVBufPtr); + break; + + case 0x0b: /* RCV Buffer pointer MSB. */ + val = RT_HIBYTE(pThis->uRCVBufPtr); + break; + + case 0x0c: /* Ethernet address PROM window. */ + case 0x0d: /* Alias. */ + /* Reads use low 3 bits of GP buffer pointer, no auto-increment. */ + val = pThis->aPROM[pThis->uGPBufPtr & 7]; + break; + + case 0x0e: /* Auxiliary status register. */ + val = pThis->AuxStatReg; + break; + + case 0x0f: /* Buffer window. */ + /* Reads use low 11 bits of GP buffer pointer, auto-increment. */ + val = pThis->abPacketBuf[ELNK_GP(pThis)]; + pThis->uGPBufPtr = (pThis->uGPBufPtr + 1) & ELNK_GP_MASK; + break; + } + + elnkUpdateIrq(pDevIns, pThis); + + Log2Func(("#%d: addr=%#06x val=%#04x\n", pThis->iInstance, addr, val & 0xff)); + return val; +} + + +/** + * @callback_method_impl{FNIOMIOPORTIN} + */ +static DECLCALLBACK(VBOXSTRICTRC) +elnkIOPortRead(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t *pu32, unsigned cb) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + int rc = VINF_SUCCESS; + uint8_t u8Lo, u8Hi; + STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIORead), a); + Assert(PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect)); + RT_NOREF_PV(pvUser); + + switch (cb) + { + case 1: + *pu32 = elIoRead(pDevIns, pThis, Port, &rc); + break; + case 2: + /* Manually split word access. */ + u8Lo = elIoRead(pDevIns, pThis, Port + 0, &rc); + Assert(RT_SUCCESS(rc)); + u8Hi = elIoRead(pDevIns, pThis, Port + 1, &rc); + Assert(RT_SUCCESS(rc)); + *pu32 = RT_MAKE_U16(u8Lo, u8Hi); + break; + default: + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, + "elnkIOPortRead: unsupported operation size: offset=%#10x cb=%u\n", + Port, cb); + } + + Log2Func(("#%d: Port=%RTiop *pu32=%#RX32 cb=%d rc=%Rrc\n", pThis->iInstance, Port, *pu32, cb, rc)); + STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIORead), a); + return rc; +} + + +/** + * @callback_method_impl{FNIOMIOPORTOUT} + */ +static DECLCALLBACK(VBOXSTRICTRC) +elnkIOPortWrite(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT Port, uint32_t u32, unsigned cb) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + int rc = VINF_SUCCESS; + STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIOWrite), a); + Assert(PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect)); + RT_NOREF_PV(pvUser); + + switch (cb) + { + case 1: + rc = elIoWrite(pDevIns, pThis, Port, RT_LOBYTE(u32)); + break; + case 2: + /* Manually split word access. */ + rc = elIoWrite(pDevIns, pThis, Port + 0, RT_LOBYTE(u32)); + if (!RT_SUCCESS(rc)) + break; + rc = elIoWrite(pDevIns, pThis, Port + 1, RT_HIBYTE(u32)); + break; + default: + rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, + "elnkIOPortWrite: unsupported operation size: offset=%#10x cb=%u\n", + Port, cb); + } + + Log2Func(("#%d: Port=%RTiop u32=%#RX32 cb=%d rc=%Rrc\n", pThis->iInstance, Port, u32, cb, rc)); + STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIOWrite), a); + return rc; +} + + +#ifdef IN_RING3 + +/* Shamelessly stolen from DevDMA.cpp */ + +/* Test the decrement bit of mode register. */ +#define IS_MODE_DEC(c) ((c) & 0x20) +/* Test the auto-init bit of mode register. */ +#define IS_MODE_AI(c) ((c) & 0x10) +/* Extract the transfer type bits of mode register. */ +#define GET_MODE_XTYP(c) (((c) & 0x0c) >> 2) + +/* DMA transfer modes. */ +enum { + DMODE_DEMAND, /* Demand transfer mode. */ + DMODE_SINGLE, /* Single transfer mode. */ + DMODE_BLOCK, /* Block transfer mode. */ + DMODE_CASCADE /* Cascade mode. */ +}; + +/* DMA transfer types. */ +enum { + DTYPE_VERIFY, /* Verify transfer type. */ + DTYPE_WRITE, /* Write transfer type. */ + DTYPE_READ, /* Read transfer type. */ + DTYPE_ILLEGAL /* Undefined. */ +}; + +static DECLCALLBACK(uint32_t) elnkR3DMAXferHandler(PPDMDEVINS pDevIns, void *opaque, + unsigned nchan, uint32_t dma_pos, uint32_t dma_len) +{ + RT_NOREF(pDevIns); + PELNKSTATE pThis = (PELNKSTATE)opaque; + int dma_mode; + int dma_type; + uint32_t cbToXfer; + uint32_t cbXferred; + uint16_t uLastPos; + int rc; + + /* + * The 3C501 EtherLink uses DMA as an alternative to accessing + * the buffer window register. The GP Buffer Pointer controls + * the address into the packet buffer for both writing to and + * reading from the buffer. + */ + dma_mode = PDMDevHlpDMAGetChannelMode(pDevIns, pThis->uIsaDma); + dma_type = GET_MODE_XTYP(dma_mode); + LogFlowFunc(("dma_mode=%d, dma_type=%d, dma_pos=%u, dma_len=%u, GPBP=%u\n", dma_mode, dma_type, dma_pos, dma_len, pThis->uGPBufPtr)); + + cbToXfer = dma_len; + + if (dma_type == DTYPE_WRITE) + { + /* Write transfer type. Reading from device, writing to memory. */ + rc = PDMDevHlpDMAWriteMemory(pDevIns, nchan, + &pThis->abPacketBuf[ELNK_GP(pThis)], + dma_pos, cbToXfer, &cbXferred); + AssertMsgRC(rc, ("DMAWriteMemory -> %Rrc\n", rc)); + uLastPos = pThis->uRCVBufPtr; + } + else + { + /* Read of Verify transfer type. Reading from memory, writing to device. */ + rc = PDMDevHlpDMAReadMemory(pDevIns, nchan, + &pThis->abPacketBuf[ELNK_GP(pThis)], + dma_pos, cbToXfer, &cbXferred); + AssertMsgRC(rc, ("DMAReadMemory -> %Rrc\n", rc)); + uLastPos = 0; /* Stop when buffer address wraps back to zero. */ + } + Log2Func(("After DMA transfer: GPBufPtr=%u, lastpos=%u, cbXferred=%u\n", pThis->uGPBufPtr, uLastPos, cbXferred)); + + /* Advance the GP buffer pointer and see if transfer completed (it almost certainly did). */ + pThis->uGPBufPtr = (pThis->uGPBufPtr + cbXferred) & ELNK_GP_MASK; + if (ELNK_GP(pThis) == uLastPos || 1) + { + Log2(("DMA completed\n")); + PDMDevHlpDMASetDREQ(pDevIns, pThis->uIsaDma, 0); + pThis->IntrState.dma_intr = 1; + pThis->AuxStat.dma_done = 1; + elnkUpdateIrq(pDevIns, pThis); + } + else + { + Log(("DMA continuing: GPBufPtr=%u, lastpos=%u, cbXferred=%u\n", pThis->uGPBufPtr, uLastPos, cbXferred)); + PDMDevHlpDMASchedule(pDevIns); + } + + /* Returns the updated transfer count. */ + return dma_pos + cbXferred; +} + + +/* -=-=-=-=-=- Timer Callbacks -=-=-=-=-=- */ + +/** + * @callback_method_impl{FNTMTIMERDEV, Restore timer callback} + * + * This is only called when we restore a saved state and temporarily + * disconnected the network link to inform the guest that network connections + * should be considered lost. + */ +static DECLCALLBACK(void) elnkR3TimerRestore(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser) +{ + RT_NOREF(pvUser); + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + int rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + AssertReleaseRC(rc); + + rc = VERR_GENERAL_FAILURE; + + /* The EhterLink cards have no concept of a link state, and cables were assumed to be + * permanently attached (AUI or BNC). We can simulate a disconnected cable by reporting + * collisions on transmit, but a guest that waits to receive something will never know. + * For that reason, the link is temporarily down, we will only postpone restoring it + * a couple of times, and then reconnect regardless of whether the guest noticed + * anything or not. + */ + if ( (pThis->cLinkDownReported <= ELNK_MAX_LINKDOWN_REPORTED) + && (pThis->cLinkRestorePostponed <= ELNK_MAX_LINKRST_POSTPONED)) + rc = PDMDevHlpTimerSetMillies(pDevIns, hTimer, 1500); + if (RT_FAILURE(rc)) + { + pThis->fLinkTempDown = false; + if (pThis->fLinkUp) + { + LogRel(("3C501#%d: The link is back up again after the restore.\n", + pThis->iInstance)); + LogFunc(("#%d: cLinkDownReported=%d\n", + pThis->iInstance, pThis->cLinkDownReported)); + pThis->Led.Actual.s.fError = 0; + } + } + else + { + LogFunc(("#%d: cLinkDownReported=%d, cLinkRestorePostponed=%d, wait another 1500ms...\n", + pThis->iInstance, pThis->cLinkDownReported, pThis->cLinkRestorePostponed)); + pThis->cLinkRestorePostponed++; + } + + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); +} + + +/* -=-=-=-=-=- Debug Info Handler -=-=-=-=-=- */ + +/** + * @callback_method_impl{FNDBGFHANDLERDEV} + */ +static DECLCALLBACK(void) elnkR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + bool fStationAddr = false; + bool fRecvBuffer = false; + bool fSendBuffer = false; + static const char *apszAddrMatch[4] = { "Disabled", "Promiscuous", "Broadcast", "Multicast" }; + static const char *apszBuffCntrl[4] = { "System", "Xmit then Recv", "Receive", "Loopback" }; + /* + * Parse args. + */ + if (pszArgs) + { + fStationAddr = strstr(pszArgs, "verbose") || strstr(pszArgs, "addr"); + fRecvBuffer = strstr(pszArgs, "verbose") || strstr(pszArgs, "recvbuf"); + fSendBuffer = strstr(pszArgs, "verbose") || strstr(pszArgs, "sendbuf"); + } + + /* + * Show info. + */ + pHlp->pfnPrintf(pHlp, + "3C501 #%d: port=%RTiop IRQ=%u DMA=%u mac-cfg=%RTmac%s%s %s\n", + pThis->iInstance, + pThis->IOPortBase, pThis->uIsaIrq, pThis->uIsaDma, &pThis->MacConfigured, + pDevIns->fRCEnabled ? " RC" : "", pDevIns->fR0Enabled ? " RZ" : "", + pThis->fDriverAttached ? "attached" : "unattached!"); + + int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_INTERNAL_ERROR); /* Take it here so we know why we're hanging... */ + PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &pThis->CritSect, rcLock); + + pHlp->pfnPrintf(pHlp, " GP Buf Ptr : %u (masked %u)\n", pThis->uGPBufPtr, ELNK_GP(pThis)); + pHlp->pfnPrintf(pHlp, " RCV Buf Ptr: %u\n", pThis->uRCVBufPtr); + pHlp->pfnPrintf(pHlp, " Recv Command: %02X Recv Status: %02X\n", pThis->RcvCmdReg, pThis->RcvStatReg); + pHlp->pfnPrintf(pHlp, " Xmit Command: %02X Xmit Status: %02X\n", pThis->XmitCmdReg, pThis->XmitStatReg); + pHlp->pfnPrintf(pHlp, " Aux Command: %02X Aux Status: %02X\n", pThis->AuxCmdReg, pThis->AuxStatReg); + + pHlp->pfnPrintf(pHlp, " Address matching: %s\n", apszAddrMatch[pThis->RcvCmd.adr_match]); + pHlp->pfnPrintf(pHlp, " Buffer control : %s\n", apszBuffCntrl[pThis->AuxCmd.buf_ctl]); + pHlp->pfnPrintf(pHlp, " Interrupt state : xmit=%u recv=%u dma=%u\n", pThis->IntrState.xmit_intr, pThis->IntrState.recv_intr, pThis->IntrState.dma_intr); + if (pThis->fLinkTempDown) + { + pHlp->pfnPrintf(pHlp, " Link down count : %d\n", pThis->cLinkDownReported); + pHlp->pfnPrintf(pHlp, " Postpone count : %d\n", pThis->cLinkRestorePostponed); + } + + /* Dump the station address. */ + if (fStationAddr) + { + pHlp->pfnPrintf(pHlp, " Station address : %RTmac\n", &pThis->aStationAddr); + } + + /* Dump the beginning of the send buffer. */ + if (fSendBuffer) + { + pHlp->pfnPrintf(pHlp, "Send buffer (start at %u):\n", ELNK_GP(pThis)); + unsigned dump_end = RT_MIN((ELNK_GP(pThis)) + 64, sizeof(pThis->abPacketBuf) - 16); + for (unsigned ofs = ELNK_GP(pThis); ofs < dump_end; ofs += 16) + pHlp->pfnPrintf(pHlp, " %04X: %Rhxs\n", ofs, &pThis->abPacketBuf[ofs]); + pHlp->pfnPrintf(pHlp, "pktbuf at %p, end at %p\n", &pThis->abPacketBuf[ELNK_GP(pThis)], &pThis->abPacketBuf[ELNK_BUF_SIZE]); + } + + /* Dump the beginning of the receive buffer. */ + if (fRecvBuffer) + { + pHlp->pfnPrintf(pHlp, "Receive buffer (start at 0):\n"); + unsigned dump_end = RT_MIN(pThis->uRCVBufPtr, 64); + for (unsigned ofs = 0; ofs < dump_end; ofs += 16) + pHlp->pfnPrintf(pHlp, " %04X: %Rhxs\n", ofs, &pThis->abPacketBuf[ofs]); + pHlp->pfnPrintf(pHlp, "pktbuf at %p, end at %p\n", pThis->abPacketBuf, &pThis->abPacketBuf[pThis->uRCVBufPtr]); + } + + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); +} + + +/* -=-=-=-=-=- Helper(s) -=-=-=-=-=- */ + + +static void elnkR3HardReset(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + LogFlowFunc(("#%d:\n", pThis->iInstance)); + + /* Initialize the PROM */ + Assert(sizeof(pThis->MacConfigured) == 6); + memcpy(pThis->aPROM, &pThis->MacConfigured, sizeof(pThis->MacConfigured)); + pThis->aPROM[6] = pThis->aPROM[7] = 0; /* The two padding bytes. */ + + /* Clear the packet buffer and station address. */ + memset(pThis->abPacketBuf, 0, sizeof(pThis->abPacketBuf)); + memset(pThis->aStationAddr, 0, sizeof(pThis->aStationAddr)); + + /* Reset the buffer pointers. */ + pThis->uGPBufPtr = 0; + pThis->uRCVBufPtr = 0; + + elnkSoftReset(pDevIns, pThis); +} + +/** + * Takes down the link temporarily if it's current status is up. + * + * This is used during restore and when replumbing the network link. + * + * The temporary link outage is supposed to indicate to the OS that all network + * connections have been lost and that it for instance is appropriate to + * renegotiate any DHCP lease. + * + * @param pDevIns The device instance. + * @param pThis The device instance data. + */ +static void elnkTempLinkDown(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + if (pThis->fLinkUp) + { + pThis->fLinkTempDown = true; + pThis->cLinkDownReported = 0; + pThis->cLinkRestorePostponed = 0; + pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1; + int rc = PDMDevHlpTimerSetMillies(pDevIns, pThis->hTimerRestore, pThis->cMsLinkUpDelay); + AssertRC(rc); + } +} + + +/* -=-=-=-=-=- Saved State -=-=-=-=-=- */ + +/** + * @callback_method_impl{FNSSMDEVLIVEEXEC, Pass 0 only.} + */ +static DECLCALLBACK(int) elnkLiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass) +{ + RT_NOREF(uPass); + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + pDevIns->pHlpR3->pfnSSMPutMem(pSSM, &pThis->MacConfigured, sizeof(pThis->MacConfigured)); + return VINF_SSM_DONT_CALL_AGAIN; +} + + +/** + * @callback_method_impl{FNSSMDEVSAVEPREP, + * Serializes the receive thread, it may be working inside the critsect.} + */ +static DECLCALLBACK(int) elnkSavePrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM) +{ + RT_NOREF(pSSM); + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + int rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + AssertRC(rc); + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + + return VINF_SUCCESS; +} + + +/** + * @callback_method_impl{FNSSMDEVSAVEEXEC} + */ +static DECLCALLBACK(int) elnkSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3; + + pHlp->pfnSSMPutU16(pSSM, pThis->uGPBufPtr); + pHlp->pfnSSMPutU16(pSSM, pThis->uRCVBufPtr); + pHlp->pfnSSMPutU8(pSSM, pThis->XmitCmdReg); + pHlp->pfnSSMPutU8(pSSM, pThis->XmitStatReg); + pHlp->pfnSSMPutU8(pSSM, pThis->RcvCmdReg); + pHlp->pfnSSMPutU8(pSSM, pThis->RcvStatReg); + pHlp->pfnSSMPutU8(pSSM, pThis->AuxCmdReg); + pHlp->pfnSSMPutU8(pSSM, pThis->AuxStatReg); + + pHlp->pfnSSMPutU8(pSSM, pThis->IntrStateReg); + pHlp->pfnSSMPutBool(pSSM, pThis->fInReset); + pHlp->pfnSSMPutBool(pSSM, pThis->fLinkUp); + pHlp->pfnSSMPutBool(pSSM, pThis->fISR); + pHlp->pfnSSMPutMem(pSSM, pThis->aStationAddr, sizeof(pThis->aStationAddr)); + + /* Save the configured MAC address. */ + pHlp->pfnSSMPutMem(pSSM, &pThis->MacConfigured, sizeof(pThis->MacConfigured)); + + return VINF_SUCCESS; +} + + +/** + * @callback_method_impl{FNSSMDEVLOADPREP}, + * Serializes the receive thread, it may be working inside the critsect.} + */ +static DECLCALLBACK(int) elnkLoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + RT_NOREF(pSSM); + + int rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + AssertRC(rc); + + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + + return rc; +} + + +/** + * @callback_method_impl{FNSSMDEVLOADEXEC} + */ +static DECLCALLBACK(int) elnkLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + PELNKSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PELNKSTATECC); + PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3; + + if (SSM_VERSION_MAJOR_CHANGED(uVersion, ELNK_SAVEDSTATE_VERSION)) + return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION; + + if (uPass == SSM_PASS_FINAL) + { + /* restore data */ + pHlp->pfnSSMGetU16(pSSM, &pThis->uGPBufPtr); + pHlp->pfnSSMGetU16(pSSM, &pThis->uRCVBufPtr); + pHlp->pfnSSMGetU8(pSSM, &pThis->XmitCmdReg); + pHlp->pfnSSMGetU8(pSSM, &pThis->XmitStatReg); + pHlp->pfnSSMGetU8(pSSM, &pThis->RcvCmdReg); + pHlp->pfnSSMGetU8(pSSM, &pThis->RcvStatReg); + pHlp->pfnSSMGetU8(pSSM, &pThis->AuxCmdReg); + pHlp->pfnSSMGetU8(pSSM, &pThis->AuxStatReg); + + pHlp->pfnSSMGetU8(pSSM, &pThis->IntrStateReg); + pHlp->pfnSSMGetBool(pSSM, &pThis->fInReset); + pHlp->pfnSSMGetBool(pSSM, &pThis->fLinkUp); + pHlp->pfnSSMGetBool(pSSM, &pThis->fISR); + pHlp->pfnSSMGetMem(pSSM, &pThis->aStationAddr, sizeof(pThis->aStationAddr)); + } + + /* check config */ + RTMAC Mac; + int rc = pHlp->pfnSSMGetMem(pSSM, &Mac, sizeof(Mac)); + AssertRCReturn(rc, rc); + if ( memcmp(&Mac, &pThis->MacConfigured, sizeof(Mac)) + && (uPass == 0 || !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)) ) + LogRel(("3C501#%u: The mac address differs: config=%RTmac saved=%RTmac\n", pThis->iInstance, &pThis->MacConfigured, &Mac)); + + if (uPass == SSM_PASS_FINAL) + { + /* update promiscuous mode. */ + if (pThisCC->pDrv) + pThisCC->pDrv->pfnSetPromiscuousMode(pThisCC->pDrv, 0 /* promiscuous enabled */); + + /* Indicate link down to the guest OS that all network connections have + been lost, unless we've been teleported here. */ + if (!PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)) + elnkTempLinkDown(pDevIns, pThis); + } + + return VINF_SUCCESS; +} + + +/* -=-=-=-=-=- ELNKSTATE::INetworkDown -=-=-=-=-=- */ + +/** + * Check if the device/driver can receive data now. + * + * Worker for elnkNet_WaitReceiveAvail(). This must be called before + * the pfnRecieve() method is called. + * + * @returns VBox status code. + * @param pDevIns The device instance data. + * @param pThis The shared instance data. + */ +static int elnkCanReceive(PPDMDEVINS pDevIns, PELNKSTATE pThis) +{ + int rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + AssertReleaseRC(rc); + + rc = VINF_SUCCESS; + + /* + * The real 3C501 is very limited in that the packet buffer can only hold one + * frame and and it is shared between transmit and receive, which means the card + * frequently drops packets on a busy network. We cheat a bit and try to hold + * off when it looks like receive is only temporarily unavailable. + * + * If the receiver is disabled, accept packet and drop it to avoid + * packet pile-ups. If it's enabled, take a closer look. + */ +#if 0 + if (pThis->RcvCmd.adr_match != EL_ADRM_DISABLED) { + /* The 3C501 is only prepared to accept a packet if the receiver is busy. + * When not busy, try to delay packets. + */ + if (!pThis->AuxStat.recv_bsy) + { + rc = VERR_NET_NO_BUFFER_SPACE; + } + } +#else + if (pThis->RcvCmd.adr_match == EL_ADRM_DISABLED || !pThis->AuxStat.recv_bsy) + { + rc = VERR_NET_NO_BUFFER_SPACE; + } +#endif + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + return rc; +} + + +/** + * @interface_method_impl{PDMINETWORKDOWN,pfnWaitReceiveAvail} + */ +static DECLCALLBACK(int) elnkNet_WaitReceiveAvail(PPDMINETWORKDOWN pInterface, RTMSINTERVAL cMillies) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, INetworkDown); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + int rc = elnkCanReceive(pDevIns, pThis); + if (RT_SUCCESS(rc)) + return VINF_SUCCESS; + if (RT_UNLIKELY(cMillies == 0)) + return VERR_NET_NO_BUFFER_SPACE; + + rc = VERR_INTERRUPTED; + ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, true); + STAM_PROFILE_START(&pThis->StatRxOverflow, a); + VMSTATE enmVMState; + while (RT_LIKELY( (enmVMState = PDMDevHlpVMState(pDevIns)) == VMSTATE_RUNNING + || enmVMState == VMSTATE_RUNNING_LS)) + { + int rc2 = elnkCanReceive(pDevIns, pThis); + if (RT_SUCCESS(rc2)) + { + rc = VINF_SUCCESS; + break; + } + LogFlowFunc(("waiting cMillies=%u...\n", cMillies)); + + /* Start the poll timer once which will remain active as long fMaybeOutOfSpace + * is true -- even if (transmit) polling is disabled. */ + rc2 = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + AssertReleaseRC(rc2); + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + RTSemEventWait(pThis->hEventOutOfRxSpace, cMillies); + } + STAM_PROFILE_STOP(&pThis->StatRxOverflow, a); + ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, false); + + return rc; +} + + +/** + * @interface_method_impl{PDMINETWORKDOWN,pfnReceive} + */ +static DECLCALLBACK(int) elnkNet_Receive(PPDMINETWORKDOWN pInterface, const void *pvBuf, size_t cb) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, INetworkDown); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + int rc; + + STAM_PROFILE_ADV_START(&pThis->StatReceive, a); + rc = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + AssertReleaseRC(rc); + + if (cb > 50) /* unqualified guess */ + pThis->Led.Asserted.s.fReading = pThis->Led.Actual.s.fReading = 1; + elnkReceiveLocked(pDevIns, pThis, (const uint8_t *)pvBuf, cb, false); + pThis->Led.Actual.s.fReading = 0; + + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a); + + return VINF_SUCCESS; +} + + +/** + * @interface_method_impl{PDMINETWORKDOWN,pfnXmitPending} + */ +static DECLCALLBACK(void) elnkNet_XmitPending(PPDMINETWORKDOWN pInterface) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, INetworkDown); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + elnkXmitBuffer(pDevIns, pThis, pThisCC, true /*fOnWorkerThread*/); +} + + +/* -=-=-=-=-=- ELNKSTATE::INetworkConfig -=-=-=-=-=- */ + +/** + * @interface_method_impl{PDMINETWORKCONFIG,pfnGetMac} + */ +static DECLCALLBACK(int) elnkGetMac(PPDMINETWORKCONFIG pInterface, PRTMAC pMac) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, INetworkConfig); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + LogFlowFunc(("#%d\n", pThis->iInstance)); + /// @todo This is broken!! We can't properly get the MAC address set by the guest +#if 0 + memcpy(pMac, pThis->aStationAddr, sizeof(*pMac)); +#else + memcpy(pMac, pThis->aPROM, sizeof(*pMac)); +#endif + return VINF_SUCCESS; +} + + +/** + * @interface_method_impl{PDMINETWORKCONFIG,pfnGetLinkState} + */ +static DECLCALLBACK(PDMNETWORKLINKSTATE) elnkGetLinkState(PPDMINETWORKCONFIG pInterface) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, INetworkConfig); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + if (pThis->fLinkUp && !pThis->fLinkTempDown) + return PDMNETWORKLINKSTATE_UP; + if (!pThis->fLinkUp) + return PDMNETWORKLINKSTATE_DOWN; + if (pThis->fLinkTempDown) + return PDMNETWORKLINKSTATE_DOWN_RESUME; + AssertMsgFailed(("Invalid link state!\n")); + return PDMNETWORKLINKSTATE_INVALID; +} + + +/** + * @interface_method_impl{PDMINETWORKCONFIG,pfnSetLinkState} + */ +static DECLCALLBACK(int) elnkSetLinkState(PPDMINETWORKCONFIG pInterface, PDMNETWORKLINKSTATE enmState) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, INetworkConfig); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + bool fLinkUp; + + AssertMsgReturn(enmState > PDMNETWORKLINKSTATE_INVALID && enmState <= PDMNETWORKLINKSTATE_DOWN_RESUME, + ("Invalid link state: enmState=%d\n", enmState), VERR_INVALID_PARAMETER); + + if (enmState == PDMNETWORKLINKSTATE_DOWN_RESUME) + { + elnkTempLinkDown(pDevIns, pThis); + /* + * Note that we do not notify the driver about the link state change because + * the change is only temporary and can be disregarded from the driver's + * point of view (see @bugref{7057}). + */ + return VINF_SUCCESS; + } + /* has the state changed? */ + fLinkUp = enmState == PDMNETWORKLINKSTATE_UP; + if (pThis->fLinkUp != fLinkUp) + { + pThis->fLinkUp = fLinkUp; + if (fLinkUp) + { + /* Connect with a configured delay. */ + pThis->fLinkTempDown = true; + pThis->cLinkDownReported = 0; + pThis->cLinkRestorePostponed = 0; + pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1; + int rc = PDMDevHlpTimerSetMillies(pDevIns, pThis->hTimerRestore, pThis->cMsLinkUpDelay); + AssertRC(rc); + } + else + { + /* Disconnect. */ + pThis->cLinkDownReported = 0; + pThis->cLinkRestorePostponed = 0; + pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1; + } + Assert(!PDMDevHlpCritSectIsOwner(pDevIns, &pThis->CritSect)); + if (pThisCC->pDrv) + pThisCC->pDrv->pfnNotifyLinkChanged(pThisCC->pDrv, enmState); + } + return VINF_SUCCESS; +} + + +/* -=-=-=-=-=- ELNKSTATE::ILeds (LUN#0) -=-=-=-=-=- */ + +/** + * @interface_method_impl{PDMILEDPORTS,pfnQueryStatusLed} + */ +static DECLCALLBACK(int) elnkQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, ILeds); + PPDMDEVINS pDevIns = pThisCC->pDevIns; + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + if (iLUN == 0) + { + *ppLed = &pThis->Led; + return VINF_SUCCESS; + } + return VERR_PDM_LUN_NOT_FOUND; +} + + +/* -=-=-=-=-=- ELNKSTATE::IBase (LUN#0) -=-=-=-=-=- */ + +/** + * @interface_method_impl{PDMIBASE,pfnQueryInterface} + */ +static DECLCALLBACK(void *) elnkQueryInterface(struct PDMIBASE *pInterface, const char *pszIID) +{ + PELNKSTATECC pThisCC = RT_FROM_MEMBER(pInterface, ELNKSTATECC, IBase); + Assert(&pThisCC->IBase == pInterface); + PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThisCC->IBase); + PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKDOWN, &pThisCC->INetworkDown); + PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKCONFIG, &pThisCC->INetworkConfig); + PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThisCC->ILeds); + return NULL; +} + + +/* -=-=-=-=-=- PDMDEVREG -=-=-=-=-=- */ + +/** + * @interface_method_impl{PDMDEVREG,pfnPowerOff} + */ +static DECLCALLBACK(void) elnkR3PowerOff(PPDMDEVINS pDevIns) +{ + /* Poke thread waiting for buffer space. */ + elnkR3WakeupReceive(pDevIns); +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnDetach} + * + * One port on the network card has been disconnected from the network. + */ +static DECLCALLBACK(void) elnkR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + PELNKSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PELNKSTATECC); + RT_NOREF(fFlags); + LogFlowFunc(("#%d:\n", pThis->iInstance)); + + AssertLogRelReturnVoid(iLUN == 0); + + int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &pThis->CritSect, rcLock); + + /* + * Zero some important members. + */ + pThis->fDriverAttached = false; + pThisCC->pDrvBase = NULL; + pThisCC->pDrv = NULL; + + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnAttach} + * One port on the network card has been connected to a network. + */ +static DECLCALLBACK(int) elnkR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + PELNKSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PELNKSTATECC); + RT_NOREF(fFlags); + LogFlowFunc(("#%d:\n", pThis->iInstance)); + + AssertLogRelReturn(iLUN == 0, VERR_PDM_NO_SUCH_LUN); + + int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &pThis->CritSect, VERR_SEM_BUSY); + PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &pThis->CritSect, rcLock); + + /* + * Attach the driver. + */ + int rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->IBase, &pThisCC->pDrvBase, "Network Port"); + if (RT_SUCCESS(rc)) + { + pThisCC->pDrv = PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMINETWORKUP); + AssertMsgStmt(pThisCC->pDrv, ("Failed to obtain the PDMINETWORKUP interface!\n"), + rc = VERR_PDM_MISSING_INTERFACE_BELOW); + pThis->fDriverAttached = true; + } + else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER + || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME) + { + /* This should never happen because this function is not called + * if there is no driver to attach! */ + Log(("#%d: No attached driver!\n", pThis->iInstance)); + } + + /* + * Temporary set the link down if it was up so that the guest + * will know that we have change the configuration of the + * network card + */ + if (RT_SUCCESS(rc)) + elnkTempLinkDown(pDevIns, pThis); + + PDMDevHlpCritSectLeave(pDevIns, &pThis->CritSect); + return rc; +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnSuspend} + */ +static DECLCALLBACK(void) elnkR3Suspend(PPDMDEVINS pDevIns) +{ + /* Poke thread waiting for buffer space. */ + elnkR3WakeupReceive(pDevIns); +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnReset} + */ +static DECLCALLBACK(void) elnkR3Reset(PPDMDEVINS pDevIns) +{ + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + if (pThis->fLinkTempDown) + { + pThis->cLinkDownReported = 0x1000; + pThis->cLinkRestorePostponed = 0x1000; + PDMDevHlpTimerStop(pDevIns, pThis->hTimerRestore); + elnkR3TimerRestore(pDevIns, pThis->hTimerRestore, pThis); + } + + /** @todo How to flush the queues? */ + elnkR3HardReset(pDevIns, pThis); +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnRelocate} + */ +static DECLCALLBACK(void) elnkR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta) +{ + PELNKSTATERC pThisRC = PDMINS_2_DATA_RC(pDevIns, PELNKSTATERC); + pThisRC->pDrv += offDelta; +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnDestruct} + */ +static DECLCALLBACK(int) elnkR3Destruct(PPDMDEVINS pDevIns) +{ + PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns); + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + if (PDMDevHlpCritSectIsInitialized(pDevIns, &pThis->CritSect)) + { + RTSemEventSignal(pThis->hEventOutOfRxSpace); + RTSemEventDestroy(pThis->hEventOutOfRxSpace); + pThis->hEventOutOfRxSpace = NIL_RTSEMEVENT; + PDMDevHlpCritSectDelete(pDevIns, &pThis->CritSect); + } + return VINF_SUCCESS; +} + + +/** + * @interface_method_impl{PDMDEVREG,pfnConstruct} + */ +static DECLCALLBACK(int) elnkR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg) +{ + PDMDEV_CHECK_VERSIONS_RETURN(pDevIns); + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + PELNKSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PELNKSTATECC); + PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3; + PPDMIBASE pBase; + char szTmp[128]; + int rc; + + /* + * Init what's required to make the destructor safe. + */ + pThis->iInstance = iInstance; + pThis->hEventOutOfRxSpace = NIL_RTSEMEVENT; + pThis->hIoPortsIsa = NIL_IOMIOPORTHANDLE; + pThisCC->pDevIns = pDevIns; + + /* + * Validate configuration. + */ + PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns, "MAC|CableConnected|Port|IRQ|DMA|LinkUpDelay|LineSpeed", ""); + + /* + * Read the configuration. + */ + rc = pHlp->pfnCFGMQueryBytes(pCfg, "MAC", &pThis->MacConfigured, sizeof(pThis->MacConfigured)); + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, + N_("Configuration error: Failed to get the \"MAC\" value")); + rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "CableConnected", &pThis->fLinkUp, true); + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, + N_("Configuration error: Failed to get the \"CableConnected\" value")); + + /* + * Process ISA configuration options. + */ + rc = pHlp->pfnCFGMQueryPortDef(pCfg, "Port", &pThis->IOPortBase, 0x300); + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, + N_("Configuration error: Failed to get the \"Port\" value")); + + rc = pHlp->pfnCFGMQueryU8Def(pCfg, "IRQ", &pThis->uIsaIrq, 3); + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, + N_("Configuration error: Failed to get the \"IRQ\" value")); + + rc = pHlp->pfnCFGMQueryU8Def(pCfg, "DMA", &pThis->uIsaDma, 1); + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, + N_("Configuration error: Failed to get the \"DMA\" value")); + + rc = pHlp->pfnCFGMQueryU32Def(pCfg, "LinkUpDelay", (uint32_t*)&pThis->cMsLinkUpDelay, 5000); /* ms */ + if (RT_FAILURE(rc)) + return PDMDEV_SET_ERROR(pDevIns, rc, + N_("Configuration error: Failed to get the value of 'LinkUpDelay'")); + Assert(pThis->cMsLinkUpDelay <= 300000); /* less than 5 minutes */ + if (pThis->cMsLinkUpDelay > 5000 || pThis->cMsLinkUpDelay < 100) + { + LogRel(("3C501#%d WARNING! Link up delay is set to %u seconds!\n", + iInstance, pThis->cMsLinkUpDelay / 1000)); + } + Log(("#%d Link up delay is set to %u seconds\n", + iInstance, pThis->cMsLinkUpDelay / 1000)); + + + /* + * Initialize data (most of it anyway). + */ + pThis->Led.u32Magic = PDMLED_MAGIC; + /* IBase */ + pThisCC->IBase.pfnQueryInterface = elnkQueryInterface; + /* INetworkPort */ + pThisCC->INetworkDown.pfnWaitReceiveAvail = elnkNet_WaitReceiveAvail; + pThisCC->INetworkDown.pfnReceive = elnkNet_Receive; + pThisCC->INetworkDown.pfnXmitPending = elnkNet_XmitPending; + /* INetworkConfig */ + pThisCC->INetworkConfig.pfnGetMac = elnkGetMac; + pThisCC->INetworkConfig.pfnGetLinkState = elnkGetLinkState; + pThisCC->INetworkConfig.pfnSetLinkState = elnkSetLinkState; + /* ILeds */ + pThisCC->ILeds.pfnQueryStatusLed = elnkQueryStatusLed; + + /* + * We use our own critical section (historical reasons). + */ + rc = PDMDevHlpCritSectInit(pDevIns, &pThis->CritSect, RT_SRC_POS, "3C501#%u", iInstance); + AssertRCReturn(rc, rc); + rc = PDMDevHlpSetDeviceCritSect(pDevIns, &pThis->CritSect); + AssertRCReturn(rc, rc); + + rc = RTSemEventCreate(&pThis->hEventOutOfRxSpace); + AssertRCReturn(rc, rc); + + /* + * Register ISA I/O ranges for the EtherLink 3C501. + */ + rc = PDMDevHlpIoPortCreateAndMap(pDevIns, pThis->IOPortBase, 0x10 /*cPorts*/, elnkIOPortWrite, elnkIOPortRead, + "3C501", NULL /*paExtDesc*/, &pThis->hIoPortsIsa); + if (RT_FAILURE(rc)) + return rc; + + /* + * Register DMA channel. + */ + if (pThis->uIsaDma <= ELNK_MAX_VALID_DMA) + { + rc = PDMDevHlpDMARegister(pDevIns, pThis->uIsaDma, elnkR3DMAXferHandler, pThis); + if (RT_FAILURE(rc)) + return rc; + LogRel(("3C501#%d: Enabling DMA channel %u\n", iInstance, pThis->uIsaDma)); + } + else + LogRel(("3C501#%d: Disabling DMA\n", iInstance)); + + rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, elnkR3TimerRestore, NULL, TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_NO_RING0, + "3C501 Restore Timer", &pThis->hTimerRestore); + if (RT_FAILURE(rc)) + return rc; + + rc = PDMDevHlpSSMRegisterEx(pDevIns, ELNK_SAVEDSTATE_VERSION, sizeof(*pThis), NULL, + NULL, elnkLiveExec, NULL, + elnkSavePrep, elnkSaveExec, NULL, + elnkLoadPrep, elnkLoadExec, NULL); + if (RT_FAILURE(rc)) + return rc; + + /* + * Create the transmit queue. + */ + rc = PDMDevHlpTaskCreate(pDevIns, PDMTASK_F_RZ, "3C501-Xmit", elnkR3XmitTaskCallback, NULL /*pvUser*/, &pThis->hXmitTask); + if (RT_FAILURE(rc)) + return rc; + + /* + * Create the RX notifier signaller. + */ + rc = PDMDevHlpTaskCreate(pDevIns, PDMTASK_F_RZ, "3C501-Rcv", elnkR3CanRxTaskCallback, NULL /*pvUser*/, &pThis->hCanRxTask); + if (RT_FAILURE(rc)) + return rc; + + /* + * Register the info item. + */ + RTStrPrintf(szTmp, sizeof(szTmp), "elnk%d", pThis->iInstance); + PDMDevHlpDBGFInfoRegister(pDevIns, szTmp, "3C501 info", elnkR3Info); + + /* + * Attach status driver (optional). + */ + rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThisCC->IBase, &pBase, "Status Port"); + if (RT_SUCCESS(rc)) + pThis->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS); + else if ( rc != VERR_PDM_NO_ATTACHED_DRIVER + && rc != VERR_PDM_CFG_MISSING_DRIVER_NAME) + { + AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc)); + return rc; + } + + /* + * Attach driver. + */ + rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->IBase, &pThisCC->pDrvBase, "Network Port"); + if (RT_SUCCESS(rc)) + { + pThisCC->pDrv = PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMINETWORKUP); + AssertMsgReturn(pThisCC->pDrv, ("Failed to obtain the PDMINETWORKUP interface!\n"), + VERR_PDM_MISSING_INTERFACE_BELOW); + pThis->fDriverAttached = true; + } + else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER + || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME) + { + /* No error! */ + Log(("No attached driver!\n")); + } + else + return rc; + + /* + * Reset the device state. (Do after attaching.) + */ + elnkR3HardReset(pDevIns, pThis); + + /* + * Register statistics counters. + */ + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Amount of data received", "/Public/Net/EtherLink%u/BytesReceived", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Amount of data transmitted", "/Public/Net/EtherLink%u/BytesTransmitted", iInstance); + + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Amount of data received", "/Devices/EtherLink%d/ReceiveBytes", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Amount of data transmitted", "/Devices/EtherLink%d/TransmitBytes", iInstance); + +#ifdef VBOX_WITH_STATISTICS + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOReadRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in RZ", "/Devices/EtherLink%d/IO/ReadRZ", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOReadR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in R3", "/Devices/EtherLink%d/IO/ReadR3", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOWriteRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in RZ", "/Devices/EtherLink%d/IO/WriteRZ", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOWriteR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in R3", "/Devices/EtherLink%d/IO/WriteR3", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceive, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling receive", "/Devices/EtherLink%d/Receive", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRxOverflow, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_OCCURENCE, "Profiling RX overflows", "/Devices/EtherLink%d/RxOverflow", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRxOverflowWakeup, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_OCCURENCE, "Nr of RX overflow wakeups", "/Devices/EtherLink%d/RxOverflowWakeup", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling transmits in RZ", "/Devices/EtherLink%d/Transmit/TotalRZ", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling transmits in R3", "/Devices/EtherLink%d/Transmit/TotalR3", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitSendRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling send transmit in RZ", "/Devices/EtherLink%d/Transmit/SendRZ", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitSendR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling send transmit in R3", "/Devices/EtherLink%d/Transmit/SendR3", iInstance); + + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatInterrupt, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling interrupt checks", "/Devices/EtherLink%d/UpdateIRQ", iInstance); + + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatResets, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of soft resets", "/Devices/EtherLink%d/SoftResets", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatDropPktAdrmDis, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Dropped packet, disabled address match", "/Devices/EtherLink%d/DropPktAdrmDis", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatDropPktZeroLen, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Dropped zero length packet", "/Devices/EtherLink%d/DropPktZeroLen", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatDropPktVMNotRunning,STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Dropped packet, VM not running", "/Devices/EtherLink%d/DropPktVMNotRunning", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatDropPktNoLink, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Dropped packet, no link", "/Devices/EtherLink%d/DropPktNoLink", iInstance); + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatDropPktStaleRcv, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Dropped packet, status register unread", "/Devices/EtherLink%d/DropPktStaleRcv", iInstance); +#endif /* VBOX_WITH_STATISTICS */ + PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatPktsLostReset, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of packets lost due to resets", "/Devices/EtherLink%d/PktsLostByReset", iInstance); + + return VINF_SUCCESS; +} + +#else + +/** + * @callback_method_impl{PDMDEVREGR0,pfnConstruct} + */ +static DECLCALLBACK(int) elnkRZConstruct(PPDMDEVINS pDevIns) +{ + PDMDEV_CHECK_VERSIONS_RETURN(pDevIns); + PELNKSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PELNKSTATE); + + /* Critical section setup: */ + int rc = PDMDevHlpSetDeviceCritSect(pDevIns, &pThis->CritSect); + AssertRCReturn(rc, rc); + + /* ISA I/O ports: */ + if (pThis->hIoPortsIsa != NIL_IOMIOPORTHANDLE) + { + rc = PDMDevHlpIoPortSetUpContext(pDevIns, pThis->hIoPortsIsa, elnkIOPortWrite, elnkIOPortRead, NULL /*pvUser*/); + AssertRCReturn(rc, rc); + } + + return VINF_SUCCESS; +} + +#endif /* IN_RING3 */ + +/** + * The device registration structure. + */ +const PDMDEVREG g_Device3C501 = +{ + /* .u32Version = */ PDM_DEVREG_VERSION, + /* .uReserved0 = */ 0, + /* .szName = */ "3c501", + /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RZ | PDM_DEVREG_FLAGS_NEW_STYLE, + /* .fClass = */ PDM_DEVREG_CLASS_NETWORK, + /* .cMaxInstances = */ ~0U, + /* .uSharedVersion = */ 42, + /* .cbInstanceShared = */ sizeof(ELNKSTATE), + /* .cbInstanceCC = */ sizeof(ELNKSTATECC), + /* .cbInstanceRC = */ sizeof(ELNKSTATERC), + /* .cMaxPciDevices = */ 0, + /* .cMaxMsixVectors = */ 0, + /* .pszDescription = */ "3Com EtherLink 3C501 adapter.\n", +#if defined(IN_RING3) + /* .pszRCMod = */ "VBoxDDRC.rc", + /* .pszR0Mod = */ "VBoxDDR0.r0", + /* .pfnConstruct = */ elnkR3Construct, + /* .pfnDestruct = */ elnkR3Destruct, + /* .pfnRelocate = */ elnkR3Relocate, + /* .pfnMemSetup = */ NULL, + /* .pfnPowerOn = */ NULL, + /* .pfnReset = */ elnkR3Reset, + /* .pfnSuspend = */ elnkR3Suspend, + /* .pfnResume = */ NULL, + /* .pfnAttach = */ elnkR3Attach, + /* .pfnDetach = */ elnkR3Detach, + /* .pfnQueryInterface = */ NULL, + /* .pfnInitComplete = */ NULL, + /* .pfnPowerOff = */ elnkR3PowerOff, + /* .pfnSoftReset = */ NULL, + /* .pfnReserved0 = */ NULL, + /* .pfnReserved1 = */ NULL, + /* .pfnReserved2 = */ NULL, + /* .pfnReserved3 = */ NULL, + /* .pfnReserved4 = */ NULL, + /* .pfnReserved5 = */ NULL, + /* .pfnReserved6 = */ NULL, + /* .pfnReserved7 = */ NULL, +#elif defined(IN_RING0) + /* .pfnEarlyConstruct = */ NULL, + /* .pfnConstruct = */ elnkRZConstruct, + /* .pfnDestruct = */ NULL, + /* .pfnFinalDestruct = */ NULL, + /* .pfnRequest = */ NULL, + /* .pfnReserved0 = */ NULL, + /* .pfnReserved1 = */ NULL, + /* .pfnReserved2 = */ NULL, + /* .pfnReserved3 = */ NULL, + /* .pfnReserved4 = */ NULL, + /* .pfnReserved5 = */ NULL, + /* .pfnReserved6 = */ NULL, + /* .pfnReserved7 = */ NULL, +#elif defined(IN_RC) + /* .pfnConstruct = */ NULL, + /* .pfnReserved0 = */ NULL, + /* .pfnReserved1 = */ NULL, + /* .pfnReserved2 = */ NULL, + /* .pfnReserved3 = */ NULL, + /* .pfnReserved4 = */ NULL, + /* .pfnReserved5 = */ NULL, + /* .pfnReserved6 = */ NULL, + /* .pfnReserved7 = */ NULL, +#else +# error "Not in IN_RING3, IN_RING0 or IN_RC!" +#endif + /* .u32VersionEnd = */ PDM_DEVREG_VERSION +}; + +#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */ |