summaryrefslogtreecommitdiffstats
path: root/src/VBox/Devices/Network/DevE1000.cpp
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 16:49:04 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 16:49:04 +0000
commit16f504a9dca3fe3b70568f67b7d41241ae485288 (patch)
treec60f36ada0496ba928b7161059ba5ab1ab224f9d /src/VBox/Devices/Network/DevE1000.cpp
parentInitial commit. (diff)
downloadvirtualbox-16f504a9dca3fe3b70568f67b7d41241ae485288.tar.xz
virtualbox-16f504a9dca3fe3b70568f67b7d41241ae485288.zip
Adding upstream version 7.0.6-dfsg.upstream/7.0.6-dfsgupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'src/VBox/Devices/Network/DevE1000.cpp')
-rw-r--r--src/VBox/Devices/Network/DevE1000.cpp8509
1 files changed, 8509 insertions, 0 deletions
diff --git a/src/VBox/Devices/Network/DevE1000.cpp b/src/VBox/Devices/Network/DevE1000.cpp
new file mode 100644
index 00000000..7247925f
--- /dev/null
+++ b/src/VBox/Devices/Network/DevE1000.cpp
@@ -0,0 +1,8509 @@
+/* $Id: DevE1000.cpp $ */
+/** @file
+ * DevE1000 - Intel 82540EM Ethernet Controller Emulation.
+ *
+ * Implemented in accordance with the specification:
+ *
+ * PCI/PCI-X Family of Gigabit Ethernet Controllers Software Developer's Manual
+ * 82540EP/EM, 82541xx, 82544GC/EI, 82545GM/EM, 82546GB/EB, and 82547xx
+ *
+ * 317453-002 Revision 3.5
+ *
+ * @todo IPv6 checksum offloading support
+ * @todo Flexible Filter / Wakeup (optional?)
+ */
+
+/*
+ * Copyright (C) 2007-2022 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
+ */
+
+
+/*********************************************************************************************************************************
+* Header Files *
+*********************************************************************************************************************************/
+#define LOG_GROUP LOG_GROUP_DEV_E1000
+#include <iprt/crc.h>
+#include <iprt/ctype.h>
+#include <iprt/net.h>
+#include <iprt/semaphore.h>
+#include <iprt/string.h>
+#include <iprt/time.h>
+#include <iprt/uuid.h>
+#include <VBox/vmm/pdmdev.h>
+#include <VBox/vmm/pdmnetifs.h>
+#include <VBox/vmm/pdmnetinline.h>
+#include <VBox/param.h>
+#include "VBoxDD.h"
+
+#include "DevEEPROM.h"
+#include "DevE1000Phy.h"
+
+
+/*********************************************************************************************************************************
+* Defined Constants And Macros *
+*********************************************************************************************************************************/
+/** @name E1000 Build Options
+ * @{ */
+/** @def E1K_INIT_RA0
+ * E1K_INIT_RA0 forces E1000 to set the first entry in Receive Address filter
+ * table to MAC address obtained from CFGM. Most guests read MAC address from
+ * EEPROM and write it to RA[0] explicitly, but Mac OS X seems to depend on it
+ * being already set (see @bugref{4657}).
+ */
+#define E1K_INIT_RA0
+/** @def E1K_LSC_ON_RESET
+ * E1K_LSC_ON_RESET causes e1000 to generate Link Status Change
+ * interrupt after hard reset. This makes the E1K_LSC_ON_SLU option unnecessary.
+ * With unplugged cable, LSC is triggerred for 82543GC only.
+ */
+#define E1K_LSC_ON_RESET
+/** @def E1K_LSC_ON_SLU
+ * E1K_LSC_ON_SLU causes E1000 to generate Link Status Change interrupt when
+ * the guest driver brings up the link via STATUS.LU bit. Again the only guest
+ * that requires it is Mac OS X (see @bugref{4657}).
+ */
+//#define E1K_LSC_ON_SLU
+/** @def E1K_INIT_LINKUP_DELAY
+ * E1K_INIT_LINKUP_DELAY prevents the link going up while the driver is still
+ * in init (see @bugref{8624}).
+ */
+#define E1K_INIT_LINKUP_DELAY_US (2000 * 1000)
+/** @def E1K_IMS_INT_DELAY_NS
+ * E1K_IMS_INT_DELAY_NS prevents interrupt storms in Windows guests on enabling
+ * interrupts (see @bugref{8624}).
+ */
+#define E1K_IMS_INT_DELAY_NS 100
+/** @def E1K_TX_DELAY
+ * E1K_TX_DELAY aims to improve guest-host transfer rate for TCP streams by
+ * preventing packets to be sent immediately. It allows to send several
+ * packets in a batch reducing the number of acknowledgments. Note that it
+ * effectively disables R0 TX path, forcing sending in R3.
+ */
+//#define E1K_TX_DELAY 150
+/** @def E1K_USE_TX_TIMERS
+ * E1K_USE_TX_TIMERS aims to reduce the number of generated TX interrupts if a
+ * guest driver set the delays via the Transmit Interrupt Delay Value (TIDV)
+ * register. Enabling it showed no positive effects on existing guests so it
+ * stays disabled. See sections 3.2.7.1 and 3.4.3.1 in "8254x Family of Gigabit
+ * Ethernet Controllers Software Developer’s Manual" for more detailed
+ * explanation.
+ */
+//#define E1K_USE_TX_TIMERS
+/** @def E1K_NO_TAD
+ * E1K_NO_TAD disables one of two timers enabled by E1K_USE_TX_TIMERS, the
+ * Transmit Absolute Delay time. This timer sets the maximum time interval
+ * during which TX interrupts can be postponed (delayed). It has no effect
+ * if E1K_USE_TX_TIMERS is not defined.
+ */
+//#define E1K_NO_TAD
+/** @def E1K_REL_DEBUG
+ * E1K_REL_DEBUG enables debug logging of l1, l2, l3 in release build.
+ */
+//#define E1K_REL_DEBUG
+/** @def E1K_INT_STATS
+ * E1K_INT_STATS enables collection of internal statistics used for
+ * debugging of delayed interrupts, etc.
+ */
+#define E1K_INT_STATS
+/** @def E1K_WITH_MSI
+ * E1K_WITH_MSI enables rudimentary MSI support. Not implemented.
+ */
+//#define E1K_WITH_MSI
+/** @def E1K_WITH_TX_CS
+ * E1K_WITH_TX_CS protects e1kXmitPending with a critical section.
+ */
+#define E1K_WITH_TX_CS
+/** @def E1K_WITH_TXD_CACHE
+ * E1K_WITH_TXD_CACHE causes E1000 to fetch multiple TX descriptors in a
+ * single physical memory read (or two if it wraps around the end of TX
+ * descriptor ring). It is required for proper functioning of bandwidth
+ * resource control as it allows to compute exact sizes of packets prior
+ * to allocating their buffers (see @bugref{5582}).
+ */
+#define E1K_WITH_TXD_CACHE
+/** @def E1K_WITH_RXD_CACHE
+ * E1K_WITH_RXD_CACHE causes E1000 to fetch multiple RX descriptors in a
+ * single physical memory read (or two if it wraps around the end of RX
+ * descriptor ring). Intel's packet driver for DOS needs this option in
+ * order to work properly (see @bugref{6217}).
+ */
+#define E1K_WITH_RXD_CACHE
+/** @def E1K_WITH_PREREG_MMIO
+ * E1K_WITH_PREREG_MMIO enables a new style MMIO registration and is
+ * currently only done for testing the relateted PDM, IOM and PGM code. */
+//#define E1K_WITH_PREREG_MMIO
+/* @} */
+/* End of Options ************************************************************/
+
+#ifdef E1K_WITH_TXD_CACHE
+/**
+ * E1K_TXD_CACHE_SIZE specifies the maximum number of TX descriptors stored
+ * in the state structure. It limits the amount of descriptors loaded in one
+ * batch read. For example, Linux guest may use up to 20 descriptors per
+ * TSE packet. The largest TSE packet seen (Windows guest) was 45 descriptors.
+ */
+# define E1K_TXD_CACHE_SIZE 64u
+#endif /* E1K_WITH_TXD_CACHE */
+
+#ifdef E1K_WITH_RXD_CACHE
+/**
+ * E1K_RXD_CACHE_SIZE specifies the maximum number of RX descriptors stored
+ * in the state structure. It limits the amount of descriptors loaded in one
+ * batch read. For example, XP guest adds 15 RX descriptors at a time.
+ */
+# define E1K_RXD_CACHE_SIZE 16u
+#endif /* E1K_WITH_RXD_CACHE */
+
+
+/* Little helpers ************************************************************/
+#undef htons
+#undef ntohs
+#undef htonl
+#undef ntohl
+#define htons(x) ((((x) & 0xff00) >> 8) | (((x) & 0x00ff) << 8))
+#define ntohs(x) htons(x)
+#define htonl(x) ASMByteSwapU32(x)
+#define ntohl(x) htonl(x)
+
+#ifndef DEBUG
+# ifdef E1K_REL_DEBUG
+# define DEBUG
+# define E1kLog(a) LogRel(a)
+# define E1kLog2(a) LogRel(a)
+# define E1kLog3(a) LogRel(a)
+# define E1kLogX(x, a) LogRel(a)
+//# define E1kLog3(a) do {} while (0)
+# else
+# define E1kLog(a) do {} while (0)
+# define E1kLog2(a) do {} while (0)
+# define E1kLog3(a) do {} while (0)
+# define E1kLogX(x, a) do {} while (0)
+# endif
+#else
+# define E1kLog(a) Log(a)
+# define E1kLog2(a) Log2(a)
+# define E1kLog3(a) Log3(a)
+# define E1kLogX(x, a) LogIt(x, LOG_GROUP, a)
+//# define E1kLog(a) do {} while (0)
+//# define E1kLog2(a) do {} while (0)
+//# define E1kLog3(a) do {} while (0)
+#endif
+
+#if 0
+# define LOG_ENABLED
+# define E1kLogRel(a) LogRel(a)
+# undef Log6
+# define Log6(a) LogRel(a)
+#else
+# define E1kLogRel(a) do { } while (0)
+#endif
+
+//#undef DEBUG
+
+#define E1K_RELOCATE(p, o) *(RTHCUINTPTR *)&p += o
+
+#define E1K_INC_CNT32(cnt) \
+do { \
+ if (cnt < UINT32_MAX) \
+ cnt++; \
+} while (0)
+
+#define E1K_ADD_CNT64(cntLo, cntHi, val) \
+do { \
+ uint64_t u64Cnt = RT_MAKE_U64(cntLo, cntHi); \
+ uint64_t tmp = u64Cnt; \
+ u64Cnt += val; \
+ if (tmp > u64Cnt ) \
+ u64Cnt = UINT64_MAX; \
+ cntLo = (uint32_t)u64Cnt; \
+ cntHi = (uint32_t)(u64Cnt >> 32); \
+} while (0)
+
+#ifdef E1K_INT_STATS
+# define E1K_INC_ISTAT_CNT(cnt) do { ++cnt; } while (0)
+#else /* E1K_INT_STATS */
+# define E1K_INC_ISTAT_CNT(cnt) do { } while (0)
+#endif /* E1K_INT_STATS */
+
+
+/*****************************************************************************/
+
+typedef uint32_t E1KCHIP;
+#define E1K_CHIP_82540EM 0
+#define E1K_CHIP_82543GC 1
+#define E1K_CHIP_82545EM 2
+
+#ifdef IN_RING3
+/** Different E1000 chips. */
+static const struct E1kChips
+{
+ uint16_t uPCIVendorId;
+ uint16_t uPCIDeviceId;
+ uint16_t uPCISubsystemVendorId;
+ uint16_t uPCISubsystemId;
+ const char *pcszName;
+} g_aChips[] =
+{
+ /* Vendor Device SSVendor SubSys Name */
+ { 0x8086,
+ /* Temporary code, as MSI-aware driver dislike 0x100E. How to do that right? */
+# ifdef E1K_WITH_MSI
+ 0x105E,
+# else
+ 0x100E,
+# endif
+ 0x8086, 0x001E, "82540EM" }, /* Intel 82540EM-A in Intel PRO/1000 MT Desktop */
+ { 0x8086, 0x1004, 0x8086, 0x1004, "82543GC" }, /* Intel 82543GC in Intel PRO/1000 T Server */
+ { 0x8086, 0x100F, 0x15AD, 0x0750, "82545EM" } /* Intel 82545EM-A in VMWare Network Adapter */
+};
+#endif /* IN_RING3 */
+
+
+/* The size of register area mapped to I/O space */
+#define E1K_IOPORT_SIZE 0x8
+/* The size of memory-mapped register area */
+#define E1K_MM_SIZE 0x20000
+
+#define E1K_MAX_TX_PKT_SIZE 16288
+#define E1K_MAX_RX_PKT_SIZE 16384
+
+/*****************************************************************************/
+
+#ifndef VBOX_DEVICE_STRUCT_TESTCASE
+/** Gets the specfieid bits from the register. */
+#define GET_BITS(reg, bits) ((reg & reg##_##bits##_MASK) >> reg##_##bits##_SHIFT)
+#define GET_BITS_V(val, reg, bits) ((val & reg##_##bits##_MASK) >> reg##_##bits##_SHIFT)
+#define BITS(reg, bits, bitval) (bitval << reg##_##bits##_SHIFT)
+#define SET_BITS(reg, bits, bitval) do { reg = (reg & ~reg##_##bits##_MASK) | (bitval << reg##_##bits##_SHIFT); } while (0)
+#define SET_BITS_V(val, reg, bits, bitval) do { val = (val & ~reg##_##bits##_MASK) | (bitval << reg##_##bits##_SHIFT); } while (0)
+
+#define CTRL_SLU UINT32_C(0x00000040)
+#define CTRL_MDIO UINT32_C(0x00100000)
+#define CTRL_MDC UINT32_C(0x00200000)
+#define CTRL_MDIO_DIR UINT32_C(0x01000000)
+#define CTRL_MDC_DIR UINT32_C(0x02000000)
+#define CTRL_RESET UINT32_C(0x04000000)
+#define CTRL_VME UINT32_C(0x40000000)
+
+#define STATUS_LU UINT32_C(0x00000002)
+#define STATUS_TXOFF UINT32_C(0x00000010)
+
+#define EECD_EE_WIRES UINT32_C(0x0F)
+#define EECD_EE_REQ UINT32_C(0x40)
+#define EECD_EE_GNT UINT32_C(0x80)
+
+#define EERD_START UINT32_C(0x00000001)
+#define EERD_DONE UINT32_C(0x00000010)
+#define EERD_DATA_MASK UINT32_C(0xFFFF0000)
+#define EERD_DATA_SHIFT 16
+#define EERD_ADDR_MASK UINT32_C(0x0000FF00)
+#define EERD_ADDR_SHIFT 8
+
+#define MDIC_DATA_MASK UINT32_C(0x0000FFFF)
+#define MDIC_DATA_SHIFT 0
+#define MDIC_REG_MASK UINT32_C(0x001F0000)
+#define MDIC_REG_SHIFT 16
+#define MDIC_PHY_MASK UINT32_C(0x03E00000)
+#define MDIC_PHY_SHIFT 21
+#define MDIC_OP_WRITE UINT32_C(0x04000000)
+#define MDIC_OP_READ UINT32_C(0x08000000)
+#define MDIC_READY UINT32_C(0x10000000)
+#define MDIC_INT_EN UINT32_C(0x20000000)
+#define MDIC_ERROR UINT32_C(0x40000000)
+
+#define TCTL_EN UINT32_C(0x00000002)
+#define TCTL_PSP UINT32_C(0x00000008)
+
+#define RCTL_EN UINT32_C(0x00000002)
+#define RCTL_UPE UINT32_C(0x00000008)
+#define RCTL_MPE UINT32_C(0x00000010)
+#define RCTL_LPE UINT32_C(0x00000020)
+#define RCTL_LBM_MASK UINT32_C(0x000000C0)
+#define RCTL_LBM_SHIFT 6
+#define RCTL_RDMTS_MASK UINT32_C(0x00000300)
+#define RCTL_RDMTS_SHIFT 8
+#define RCTL_LBM_TCVR UINT32_C(3) /**< PHY or external SerDes loopback. */
+#define RCTL_MO_MASK UINT32_C(0x00003000)
+#define RCTL_MO_SHIFT 12
+#define RCTL_BAM UINT32_C(0x00008000)
+#define RCTL_BSIZE_MASK UINT32_C(0x00030000)
+#define RCTL_BSIZE_SHIFT 16
+#define RCTL_VFE UINT32_C(0x00040000)
+#define RCTL_CFIEN UINT32_C(0x00080000)
+#define RCTL_CFI UINT32_C(0x00100000)
+#define RCTL_BSEX UINT32_C(0x02000000)
+#define RCTL_SECRC UINT32_C(0x04000000)
+
+#define ICR_TXDW UINT32_C(0x00000001)
+#define ICR_TXQE UINT32_C(0x00000002)
+#define ICR_LSC UINT32_C(0x00000004)
+#define ICR_RXDMT0 UINT32_C(0x00000010)
+#define ICR_RXT0 UINT32_C(0x00000080)
+#define ICR_TXD_LOW UINT32_C(0x00008000)
+#define RDTR_FPD UINT32_C(0x80000000)
+
+#define PBA_st ((PBAST*)(pThis->auRegs + PBA_IDX))
+typedef struct
+{
+ unsigned rxa : 7;
+ unsigned rxa_r : 9;
+ unsigned txa : 16;
+} PBAST;
+AssertCompileSize(PBAST, 4);
+
+#define TXDCTL_WTHRESH_MASK 0x003F0000
+#define TXDCTL_WTHRESH_SHIFT 16
+#define TXDCTL_LWTHRESH_MASK 0xFE000000
+#define TXDCTL_LWTHRESH_SHIFT 25
+
+#define RXCSUM_PCSS_MASK UINT32_C(0x000000FF)
+#define RXCSUM_PCSS_SHIFT 0
+
+/** @name Register access macros
+ * @remarks These ASSUME alocal variable @a pThis of type PE1KSTATE.
+ * @{ */
+#define CTRL pThis->auRegs[CTRL_IDX]
+#define STATUS pThis->auRegs[STATUS_IDX]
+#define EECD pThis->auRegs[EECD_IDX]
+#define EERD pThis->auRegs[EERD_IDX]
+#define CTRL_EXT pThis->auRegs[CTRL_EXT_IDX]
+#define FLA pThis->auRegs[FLA_IDX]
+#define MDIC pThis->auRegs[MDIC_IDX]
+#define FCAL pThis->auRegs[FCAL_IDX]
+#define FCAH pThis->auRegs[FCAH_IDX]
+#define FCT pThis->auRegs[FCT_IDX]
+#define VET pThis->auRegs[VET_IDX]
+#define ICR pThis->auRegs[ICR_IDX]
+#define ITR pThis->auRegs[ITR_IDX]
+#define ICS pThis->auRegs[ICS_IDX]
+#define IMS pThis->auRegs[IMS_IDX]
+#define IMC pThis->auRegs[IMC_IDX]
+#define RCTL pThis->auRegs[RCTL_IDX]
+#define FCTTV pThis->auRegs[FCTTV_IDX]
+#define TXCW pThis->auRegs[TXCW_IDX]
+#define RXCW pThis->auRegs[RXCW_IDX]
+#define TCTL pThis->auRegs[TCTL_IDX]
+#define TIPG pThis->auRegs[TIPG_IDX]
+#define AIFS pThis->auRegs[AIFS_IDX]
+#define LEDCTL pThis->auRegs[LEDCTL_IDX]
+#define PBA pThis->auRegs[PBA_IDX]
+#define FCRTL pThis->auRegs[FCRTL_IDX]
+#define FCRTH pThis->auRegs[FCRTH_IDX]
+#define RDFH pThis->auRegs[RDFH_IDX]
+#define RDFT pThis->auRegs[RDFT_IDX]
+#define RDFHS pThis->auRegs[RDFHS_IDX]
+#define RDFTS pThis->auRegs[RDFTS_IDX]
+#define RDFPC pThis->auRegs[RDFPC_IDX]
+#define RDBAL pThis->auRegs[RDBAL_IDX]
+#define RDBAH pThis->auRegs[RDBAH_IDX]
+#define RDLEN pThis->auRegs[RDLEN_IDX]
+#define RDH pThis->auRegs[RDH_IDX]
+#define RDT pThis->auRegs[RDT_IDX]
+#define RDTR pThis->auRegs[RDTR_IDX]
+#define RXDCTL pThis->auRegs[RXDCTL_IDX]
+#define RADV pThis->auRegs[RADV_IDX]
+#define RSRPD pThis->auRegs[RSRPD_IDX]
+#define TXDMAC pThis->auRegs[TXDMAC_IDX]
+#define TDFH pThis->auRegs[TDFH_IDX]
+#define TDFT pThis->auRegs[TDFT_IDX]
+#define TDFHS pThis->auRegs[TDFHS_IDX]
+#define TDFTS pThis->auRegs[TDFTS_IDX]
+#define TDFPC pThis->auRegs[TDFPC_IDX]
+#define TDBAL pThis->auRegs[TDBAL_IDX]
+#define TDBAH pThis->auRegs[TDBAH_IDX]
+#define TDLEN pThis->auRegs[TDLEN_IDX]
+#define TDH pThis->auRegs[TDH_IDX]
+#define TDT pThis->auRegs[TDT_IDX]
+#define TIDV pThis->auRegs[TIDV_IDX]
+#define TXDCTL pThis->auRegs[TXDCTL_IDX]
+#define TADV pThis->auRegs[TADV_IDX]
+#define TSPMT pThis->auRegs[TSPMT_IDX]
+#define CRCERRS pThis->auRegs[CRCERRS_IDX]
+#define ALGNERRC pThis->auRegs[ALGNERRC_IDX]
+#define SYMERRS pThis->auRegs[SYMERRS_IDX]
+#define RXERRC pThis->auRegs[RXERRC_IDX]
+#define MPC pThis->auRegs[MPC_IDX]
+#define SCC pThis->auRegs[SCC_IDX]
+#define ECOL pThis->auRegs[ECOL_IDX]
+#define MCC pThis->auRegs[MCC_IDX]
+#define LATECOL pThis->auRegs[LATECOL_IDX]
+#define COLC pThis->auRegs[COLC_IDX]
+#define DC pThis->auRegs[DC_IDX]
+#define TNCRS pThis->auRegs[TNCRS_IDX]
+/* #define SEC pThis->auRegs[SEC_IDX] Conflict with sys/time.h */
+#define CEXTERR pThis->auRegs[CEXTERR_IDX]
+#define RLEC pThis->auRegs[RLEC_IDX]
+#define XONRXC pThis->auRegs[XONRXC_IDX]
+#define XONTXC pThis->auRegs[XONTXC_IDX]
+#define XOFFRXC pThis->auRegs[XOFFRXC_IDX]
+#define XOFFTXC pThis->auRegs[XOFFTXC_IDX]
+#define FCRUC pThis->auRegs[FCRUC_IDX]
+#define PRC64 pThis->auRegs[PRC64_IDX]
+#define PRC127 pThis->auRegs[PRC127_IDX]
+#define PRC255 pThis->auRegs[PRC255_IDX]
+#define PRC511 pThis->auRegs[PRC511_IDX]
+#define PRC1023 pThis->auRegs[PRC1023_IDX]
+#define PRC1522 pThis->auRegs[PRC1522_IDX]
+#define GPRC pThis->auRegs[GPRC_IDX]
+#define BPRC pThis->auRegs[BPRC_IDX]
+#define MPRC pThis->auRegs[MPRC_IDX]
+#define GPTC pThis->auRegs[GPTC_IDX]
+#define GORCL pThis->auRegs[GORCL_IDX]
+#define GORCH pThis->auRegs[GORCH_IDX]
+#define GOTCL pThis->auRegs[GOTCL_IDX]
+#define GOTCH pThis->auRegs[GOTCH_IDX]
+#define RNBC pThis->auRegs[RNBC_IDX]
+#define RUC pThis->auRegs[RUC_IDX]
+#define RFC pThis->auRegs[RFC_IDX]
+#define ROC pThis->auRegs[ROC_IDX]
+#define RJC pThis->auRegs[RJC_IDX]
+#define MGTPRC pThis->auRegs[MGTPRC_IDX]
+#define MGTPDC pThis->auRegs[MGTPDC_IDX]
+#define MGTPTC pThis->auRegs[MGTPTC_IDX]
+#define TORL pThis->auRegs[TORL_IDX]
+#define TORH pThis->auRegs[TORH_IDX]
+#define TOTL pThis->auRegs[TOTL_IDX]
+#define TOTH pThis->auRegs[TOTH_IDX]
+#define TPR pThis->auRegs[TPR_IDX]
+#define TPT pThis->auRegs[TPT_IDX]
+#define PTC64 pThis->auRegs[PTC64_IDX]
+#define PTC127 pThis->auRegs[PTC127_IDX]
+#define PTC255 pThis->auRegs[PTC255_IDX]
+#define PTC511 pThis->auRegs[PTC511_IDX]
+#define PTC1023 pThis->auRegs[PTC1023_IDX]
+#define PTC1522 pThis->auRegs[PTC1522_IDX]
+#define MPTC pThis->auRegs[MPTC_IDX]
+#define BPTC pThis->auRegs[BPTC_IDX]
+#define TSCTC pThis->auRegs[TSCTC_IDX]
+#define TSCTFC pThis->auRegs[TSCTFC_IDX]
+#define RXCSUM pThis->auRegs[RXCSUM_IDX]
+#define WUC pThis->auRegs[WUC_IDX]
+#define WUFC pThis->auRegs[WUFC_IDX]
+#define WUS pThis->auRegs[WUS_IDX]
+#define MANC pThis->auRegs[MANC_IDX]
+#define IPAV pThis->auRegs[IPAV_IDX]
+#define WUPL pThis->auRegs[WUPL_IDX]
+/** @} */
+#endif /* VBOX_DEVICE_STRUCT_TESTCASE */
+
+/**
+ * Indices of memory-mapped registers in register table.
+ */
+typedef enum
+{
+ CTRL_IDX,
+ STATUS_IDX,
+ EECD_IDX,
+ EERD_IDX,
+ CTRL_EXT_IDX,
+ FLA_IDX,
+ MDIC_IDX,
+ FCAL_IDX,
+ FCAH_IDX,
+ FCT_IDX,
+ VET_IDX,
+ ICR_IDX,
+ ITR_IDX,
+ ICS_IDX,
+ IMS_IDX,
+ IMC_IDX,
+ RCTL_IDX,
+ FCTTV_IDX,
+ TXCW_IDX,
+ RXCW_IDX,
+ TCTL_IDX,
+ TIPG_IDX,
+ AIFS_IDX,
+ LEDCTL_IDX,
+ PBA_IDX,
+ FCRTL_IDX,
+ FCRTH_IDX,
+ RDFH_IDX,
+ RDFT_IDX,
+ RDFHS_IDX,
+ RDFTS_IDX,
+ RDFPC_IDX,
+ RDBAL_IDX,
+ RDBAH_IDX,
+ RDLEN_IDX,
+ RDH_IDX,
+ RDT_IDX,
+ RDTR_IDX,
+ RXDCTL_IDX,
+ RADV_IDX,
+ RSRPD_IDX,
+ TXDMAC_IDX,
+ TDFH_IDX,
+ TDFT_IDX,
+ TDFHS_IDX,
+ TDFTS_IDX,
+ TDFPC_IDX,
+ TDBAL_IDX,
+ TDBAH_IDX,
+ TDLEN_IDX,
+ TDH_IDX,
+ TDT_IDX,
+ TIDV_IDX,
+ TXDCTL_IDX,
+ TADV_IDX,
+ TSPMT_IDX,
+ CRCERRS_IDX,
+ ALGNERRC_IDX,
+ SYMERRS_IDX,
+ RXERRC_IDX,
+ MPC_IDX,
+ SCC_IDX,
+ ECOL_IDX,
+ MCC_IDX,
+ LATECOL_IDX,
+ COLC_IDX,
+ DC_IDX,
+ TNCRS_IDX,
+ SEC_IDX,
+ CEXTERR_IDX,
+ RLEC_IDX,
+ XONRXC_IDX,
+ XONTXC_IDX,
+ XOFFRXC_IDX,
+ XOFFTXC_IDX,
+ FCRUC_IDX,
+ PRC64_IDX,
+ PRC127_IDX,
+ PRC255_IDX,
+ PRC511_IDX,
+ PRC1023_IDX,
+ PRC1522_IDX,
+ GPRC_IDX,
+ BPRC_IDX,
+ MPRC_IDX,
+ GPTC_IDX,
+ GORCL_IDX,
+ GORCH_IDX,
+ GOTCL_IDX,
+ GOTCH_IDX,
+ RNBC_IDX,
+ RUC_IDX,
+ RFC_IDX,
+ ROC_IDX,
+ RJC_IDX,
+ MGTPRC_IDX,
+ MGTPDC_IDX,
+ MGTPTC_IDX,
+ TORL_IDX,
+ TORH_IDX,
+ TOTL_IDX,
+ TOTH_IDX,
+ TPR_IDX,
+ TPT_IDX,
+ PTC64_IDX,
+ PTC127_IDX,
+ PTC255_IDX,
+ PTC511_IDX,
+ PTC1023_IDX,
+ PTC1522_IDX,
+ MPTC_IDX,
+ BPTC_IDX,
+ TSCTC_IDX,
+ TSCTFC_IDX,
+ RXCSUM_IDX,
+ WUC_IDX,
+ WUFC_IDX,
+ WUS_IDX,
+ MANC_IDX,
+ IPAV_IDX,
+ WUPL_IDX,
+ MTA_IDX,
+ RA_IDX,
+ VFTA_IDX,
+ IP4AT_IDX,
+ IP6AT_IDX,
+ WUPM_IDX,
+ FFLT_IDX,
+ FFMT_IDX,
+ FFVT_IDX,
+ PBM_IDX,
+ RA_82542_IDX,
+ MTA_82542_IDX,
+ VFTA_82542_IDX,
+ E1K_NUM_OF_REGS
+} E1kRegIndex;
+
+#define E1K_NUM_OF_32BIT_REGS MTA_IDX
+/** The number of registers with strictly increasing offset. */
+#define E1K_NUM_OF_BINARY_SEARCHABLE (WUPL_IDX + 1)
+
+
+/**
+ * Define E1000-specific EEPROM layout.
+ */
+struct E1kEEPROM
+{
+ public:
+ EEPROM93C46 eeprom;
+
+#ifdef IN_RING3
+ /**
+ * Initialize EEPROM content.
+ *
+ * @param macAddr MAC address of E1000.
+ */
+ void init(RTMAC &macAddr)
+ {
+ eeprom.init();
+ memcpy(eeprom.m_au16Data, macAddr.au16, sizeof(macAddr.au16));
+ eeprom.m_au16Data[0x04] = 0xFFFF;
+ /*
+ * bit 3 - full support for power management
+ * bit 10 - full duplex
+ */
+ eeprom.m_au16Data[0x0A] = 0x4408;
+ eeprom.m_au16Data[0x0B] = 0x001E;
+ eeprom.m_au16Data[0x0C] = 0x8086;
+ eeprom.m_au16Data[0x0D] = 0x100E;
+ eeprom.m_au16Data[0x0E] = 0x8086;
+ eeprom.m_au16Data[0x0F] = 0x3040;
+ eeprom.m_au16Data[0x21] = 0x7061;
+ eeprom.m_au16Data[0x22] = 0x280C;
+ eeprom.m_au16Data[0x23] = 0x00C8;
+ eeprom.m_au16Data[0x24] = 0x00C8;
+ eeprom.m_au16Data[0x2F] = 0x0602;
+ updateChecksum();
+ };
+
+ /**
+ * Compute the checksum as required by E1000 and store it
+ * in the last word.
+ */
+ void updateChecksum()
+ {
+ uint16_t u16Checksum = 0;
+
+ for (int i = 0; i < eeprom.SIZE-1; i++)
+ u16Checksum += eeprom.m_au16Data[i];
+ eeprom.m_au16Data[eeprom.SIZE-1] = 0xBABA - u16Checksum;
+ };
+
+ /**
+ * First 6 bytes of EEPROM contain MAC address.
+ *
+ * @returns MAC address of E1000.
+ */
+ void getMac(PRTMAC pMac)
+ {
+ memcpy(pMac->au16, eeprom.m_au16Data, sizeof(pMac->au16));
+ };
+
+ uint32_t read()
+ {
+ return eeprom.read();
+ }
+
+ void write(uint32_t u32Wires)
+ {
+ eeprom.write(u32Wires);
+ }
+
+ bool readWord(uint32_t u32Addr, uint16_t *pu16Value)
+ {
+ return eeprom.readWord(u32Addr, pu16Value);
+ }
+
+ int load(PCPDMDEVHLPR3 pHlp, PSSMHANDLE pSSM)
+ {
+ return eeprom.load(pHlp, pSSM);
+ }
+
+ void save(PCPDMDEVHLPR3 pHlp, PSSMHANDLE pSSM)
+ {
+ eeprom.save(pHlp, pSSM);
+ }
+#endif /* IN_RING3 */
+};
+
+
+#define E1K_SPEC_VLAN(s) (s & 0xFFF)
+#define E1K_SPEC_CFI(s) (!!((s>>12) & 0x1))
+#define E1K_SPEC_PRI(s) ((s>>13) & 0x7)
+
+struct E1kRxDStatus
+{
+ /** @name Descriptor Status field (3.2.3.1)
+ * @{ */
+ unsigned fDD : 1; /**< Descriptor Done. */
+ unsigned fEOP : 1; /**< End of packet. */
+ unsigned fIXSM : 1; /**< Ignore checksum indication. */
+ unsigned fVP : 1; /**< VLAN, matches VET. */
+ unsigned : 1;
+ unsigned fTCPCS : 1; /**< RCP Checksum calculated on the packet. */
+ unsigned fIPCS : 1; /**< IP Checksum calculated on the packet. */
+ unsigned fPIF : 1; /**< Passed in-exact filter */
+ /** @} */
+ /** @name Descriptor Errors field (3.2.3.2)
+ * (Only valid when fEOP and fDD are set.)
+ * @{ */
+ unsigned fCE : 1; /**< CRC or alignment error. */
+ unsigned : 4; /**< Reserved, varies with different models... */
+ unsigned fTCPE : 1; /**< TCP/UDP checksum error. */
+ unsigned fIPE : 1; /**< IP Checksum error. */
+ unsigned fRXE : 1; /**< RX Data error. */
+ /** @} */
+ /** @name Descriptor Special field (3.2.3.3)
+ * @{ */
+ unsigned u16Special : 16; /**< VLAN: Id, Canonical form, Priority. */
+ /** @} */
+};
+typedef struct E1kRxDStatus E1KRXDST;
+
+struct E1kRxDesc_st
+{
+ uint64_t u64BufAddr; /**< Address of data buffer */
+ uint16_t u16Length; /**< Length of data in buffer */
+ uint16_t u16Checksum; /**< Packet checksum */
+ E1KRXDST status;
+};
+typedef struct E1kRxDesc_st E1KRXDESC;
+AssertCompileSize(E1KRXDESC, 16);
+
+#define E1K_DTYP_LEGACY -1
+#define E1K_DTYP_CONTEXT 0
+#define E1K_DTYP_DATA 1
+#define E1K_DTYP_INVALID 2
+
+struct E1kTDLegacy
+{
+ uint64_t u64BufAddr; /**< Address of data buffer */
+ struct TDLCmd_st
+ {
+ unsigned u16Length : 16;
+ unsigned u8CSO : 8;
+ /* CMD field : 8 */
+ unsigned fEOP : 1;
+ unsigned fIFCS : 1;
+ unsigned fIC : 1;
+ unsigned fRS : 1;
+ unsigned fRPS : 1;
+ unsigned fDEXT : 1;
+ unsigned fVLE : 1;
+ unsigned fIDE : 1;
+ } cmd;
+ struct TDLDw3_st
+ {
+ /* STA field */
+ unsigned fDD : 1;
+ unsigned fEC : 1;
+ unsigned fLC : 1;
+ unsigned fTURSV : 1;
+ /* RSV field */
+ unsigned u4RSV : 4;
+ /* CSS field */
+ unsigned u8CSS : 8;
+ /* Special field*/
+ unsigned u16Special: 16;
+ } dw3;
+};
+
+/**
+ * TCP/IP Context Transmit Descriptor, section 3.3.6.
+ */
+struct E1kTDContext
+{
+ struct CheckSum_st
+ {
+ /** TSE: Header start. !TSE: Checksum start. */
+ unsigned u8CSS : 8;
+ /** Checksum offset - where to store it. */
+ unsigned u8CSO : 8;
+ /** Checksum ending (inclusive) offset, 0 = end of packet. */
+ unsigned u16CSE : 16;
+ } ip;
+ struct CheckSum_st tu;
+ struct TDCDw2_st
+ {
+ /** TSE: The total number of payload bytes for this context. Sans header. */
+ unsigned u20PAYLEN : 20;
+ /** The descriptor type - E1K_DTYP_CONTEXT (0). */
+ unsigned u4DTYP : 4;
+ /** TUCMD field, 8 bits
+ * @{ */
+ /** TSE: TCP (set) or UDP (clear). */
+ unsigned fTCP : 1;
+ /** TSE: IPv4 (set) or IPv6 (clear) - for finding the payload length field in
+ * the IP header. Does not affect the checksumming.
+ * @remarks 82544GC/EI interprets a cleared field differently. */
+ unsigned fIP : 1;
+ /** TSE: TCP segmentation enable. When clear the context describes */
+ unsigned fTSE : 1;
+ /** Report status (only applies to dw3.fDD for here). */
+ unsigned fRS : 1;
+ /** Reserved, MBZ. */
+ unsigned fRSV1 : 1;
+ /** Descriptor extension, must be set for this descriptor type. */
+ unsigned fDEXT : 1;
+ /** Reserved, MBZ. */
+ unsigned fRSV2 : 1;
+ /** Interrupt delay enable. */
+ unsigned fIDE : 1;
+ /** @} */
+ } dw2;
+ struct TDCDw3_st
+ {
+ /** Descriptor Done. */
+ unsigned fDD : 1;
+ /** Reserved, MBZ. */
+ unsigned u7RSV : 7;
+ /** TSO: The header (prototype) length (Ethernet[, VLAN tag], IP, TCP/UDP. */
+ unsigned u8HDRLEN : 8;
+ /** TSO: Maximum segment size. */
+ unsigned u16MSS : 16;
+ } dw3;
+};
+typedef struct E1kTDContext E1KTXCTX;
+
+/**
+ * TCP/IP Data Transmit Descriptor, section 3.3.7.
+ */
+struct E1kTDData
+{
+ uint64_t u64BufAddr; /**< Address of data buffer */
+ struct TDDCmd_st
+ {
+ /** The total length of data pointed to by this descriptor. */
+ unsigned u20DTALEN : 20;
+ /** The descriptor type - E1K_DTYP_DATA (1). */
+ unsigned u4DTYP : 4;
+ /** @name DCMD field, 8 bits (3.3.7.1).
+ * @{ */
+ /** End of packet. Note TSCTFC update. */
+ unsigned fEOP : 1;
+ /** Insert Ethernet FCS/CRC (requires fEOP to be set). */
+ unsigned fIFCS : 1;
+ /** Use the TSE context when set and the normal when clear. */
+ unsigned fTSE : 1;
+ /** Report status (dw3.STA). */
+ unsigned fRS : 1;
+ /** Reserved. 82544GC/EI defines this report packet set (RPS). */
+ unsigned fRPS : 1;
+ /** Descriptor extension, must be set for this descriptor type. */
+ unsigned fDEXT : 1;
+ /** VLAN enable, requires CTRL.VME, auto enables FCS/CRC.
+ * Insert dw3.SPECIAL after ethernet header. */
+ unsigned fVLE : 1;
+ /** Interrupt delay enable. */
+ unsigned fIDE : 1;
+ /** @} */
+ } cmd;
+ struct TDDDw3_st
+ {
+ /** @name STA field (3.3.7.2)
+ * @{ */
+ unsigned fDD : 1; /**< Descriptor done. */
+ unsigned fEC : 1; /**< Excess collision. */
+ unsigned fLC : 1; /**< Late collision. */
+ /** Reserved, except for the usual oddball (82544GC/EI) where it's called TU. */
+ unsigned fTURSV : 1;
+ /** @} */
+ unsigned u4RSV : 4; /**< Reserved field, MBZ. */
+ /** @name POPTS (Packet Option) field (3.3.7.3)
+ * @{ */
+ unsigned fIXSM : 1; /**< Insert IP checksum. */
+ unsigned fTXSM : 1; /**< Insert TCP/UDP checksum. */
+ unsigned u6RSV : 6; /**< Reserved, MBZ. */
+ /** @} */
+ /** @name SPECIAL field - VLAN tag to be inserted after ethernet header.
+ * Requires fEOP, fVLE and CTRL.VME to be set.
+ * @{ */
+ unsigned u16Special: 16; /**< VLAN: Id, Canonical form, Priority. */
+ /** @} */
+ } dw3;
+};
+typedef struct E1kTDData E1KTXDAT;
+
+union E1kTxDesc
+{
+ struct E1kTDLegacy legacy;
+ struct E1kTDContext context;
+ struct E1kTDData data;
+};
+typedef union E1kTxDesc E1KTXDESC;
+AssertCompileSize(E1KTXDESC, 16);
+
+#define RA_CTL_AS 0x0003
+#define RA_CTL_AV 0x8000
+
+union E1kRecAddr
+{
+ uint32_t au32[32];
+ struct RAArray
+ {
+ uint8_t addr[6];
+ uint16_t ctl;
+ } array[16];
+};
+typedef struct E1kRecAddr::RAArray E1KRAELEM;
+typedef union E1kRecAddr E1KRA;
+AssertCompileSize(E1KRA, 8*16);
+
+#define E1K_IP_RF UINT16_C(0x8000) /**< reserved fragment flag */
+#define E1K_IP_DF UINT16_C(0x4000) /**< dont fragment flag */
+#define E1K_IP_MF UINT16_C(0x2000) /**< more fragments flag */
+#define E1K_IP_OFFMASK UINT16_C(0x1fff) /**< mask for fragmenting bits */
+
+/** @todo use+extend RTNETIPV4 */
+struct E1kIpHeader
+{
+ /* type of service / version / header length */
+ uint16_t tos_ver_hl;
+ /* total length */
+ uint16_t total_len;
+ /* identification */
+ uint16_t ident;
+ /* fragment offset field */
+ uint16_t offset;
+ /* time to live / protocol*/
+ uint16_t ttl_proto;
+ /* checksum */
+ uint16_t chksum;
+ /* source IP address */
+ uint32_t src;
+ /* destination IP address */
+ uint32_t dest;
+};
+AssertCompileSize(struct E1kIpHeader, 20);
+
+#define E1K_TCP_FIN UINT16_C(0x01)
+#define E1K_TCP_SYN UINT16_C(0x02)
+#define E1K_TCP_RST UINT16_C(0x04)
+#define E1K_TCP_PSH UINT16_C(0x08)
+#define E1K_TCP_ACK UINT16_C(0x10)
+#define E1K_TCP_URG UINT16_C(0x20)
+#define E1K_TCP_ECE UINT16_C(0x40)
+#define E1K_TCP_CWR UINT16_C(0x80)
+#define E1K_TCP_FLAGS UINT16_C(0x3f)
+
+/** @todo use+extend RTNETTCP */
+struct E1kTcpHeader
+{
+ uint16_t src;
+ uint16_t dest;
+ uint32_t seqno;
+ uint32_t ackno;
+ uint16_t hdrlen_flags;
+ uint16_t wnd;
+ uint16_t chksum;
+ uint16_t urgp;
+};
+AssertCompileSize(struct E1kTcpHeader, 20);
+
+
+#ifdef E1K_WITH_TXD_CACHE
+/** The current Saved state version. */
+# define E1K_SAVEDSTATE_VERSION 4
+/** Saved state version for VirtualBox 4.2 with VLAN tag fields. */
+# define E1K_SAVEDSTATE_VERSION_VBOX_42_VTAG 3
+#else /* !E1K_WITH_TXD_CACHE */
+/** The current Saved state version. */
+# define E1K_SAVEDSTATE_VERSION 3
+#endif /* !E1K_WITH_TXD_CACHE */
+/** Saved state version for VirtualBox 4.1 and earlier.
+ * These did not include VLAN tag fields. */
+#define E1K_SAVEDSTATE_VERSION_VBOX_41 2
+/** Saved state version for VirtualBox 3.0 and earlier.
+ * This did not include the configuration part nor the E1kEEPROM. */
+#define E1K_SAVEDSTATE_VERSION_VBOX_30 1
+
+/**
+ * E1000 shared device state.
+ *
+ * This is shared between ring-0 and ring-3.
+ */
+typedef struct E1KSTATE
+{
+ char szPrf[8]; /**< Log prefix, e.g. E1000#1. */
+
+ /** Handle to PCI region \#0, the MMIO region. */
+ IOMIOPORTHANDLE hMmioRegion;
+ /** Handle to PCI region \#2, the I/O ports. */
+ IOMIOPORTHANDLE hIoPorts;
+
+ /** Receive Interrupt Delay Timer. */
+ TMTIMERHANDLE hRIDTimer;
+ /** Receive Absolute Delay Timer. */
+ TMTIMERHANDLE hRADTimer;
+ /** Transmit Interrupt Delay Timer. */
+ TMTIMERHANDLE hTIDTimer;
+ /** Transmit Absolute Delay Timer. */
+ TMTIMERHANDLE hTADTimer;
+ /** Transmit Delay Timer. */
+ TMTIMERHANDLE hTXDTimer;
+ /** Late Interrupt Timer. */
+ TMTIMERHANDLE hIntTimer;
+ /** Link Up(/Restore) Timer. */
+ TMTIMERHANDLE hLUTimer;
+
+ /** Transmit task. */
+ PDMTASKHANDLE hTxTask;
+
+ /** Critical section - what is it protecting? */
+ PDMCRITSECT cs;
+ /** RX Critical section. */
+ PDMCRITSECT csRx;
+#ifdef E1K_WITH_TX_CS
+ /** TX Critical section. */
+ PDMCRITSECT csTx;
+#endif /* E1K_WITH_TX_CS */
+ /** MAC address obtained from the configuration. */
+ RTMAC macConfigured;
+ uint16_t u16Padding0;
+ /** EMT: Last time the interrupt was acknowledged. */
+ uint64_t u64AckedAt;
+ /** All: Used for eliminating spurious interrupts. */
+ bool fIntRaised;
+ /** EMT: false if the cable is disconnected by the GUI. */
+ bool fCableConnected;
+ /** true if the device is attached to a driver. */
+ bool fIsAttached;
+ /** EMT: Compute Ethernet CRC for RX packets. */
+ bool fEthernetCRC;
+ /** All: throttle interrupts. */
+ bool fItrEnabled;
+ /** All: throttle RX interrupts. */
+ bool fItrRxEnabled;
+ /** All: Delay TX interrupts using TIDV/TADV. */
+ bool fTidEnabled;
+ bool afPadding[2];
+ /** Link up delay (in milliseconds). */
+ uint32_t cMsLinkUpDelay;
+
+ /** All: Device register storage. */
+ uint32_t auRegs[E1K_NUM_OF_32BIT_REGS];
+ /** TX/RX: Status LED. */
+ PDMLED led;
+ /** TX/RX: Number of packet being sent/received to show in debug log. */
+ uint32_t u32PktNo;
+
+ /** EMT: Offset of the register to be read via IO. */
+ uint32_t uSelectedReg;
+ /** EMT: Multicast Table Array. */
+ uint32_t auMTA[128];
+ /** EMT: Receive Address registers. */
+ E1KRA aRecAddr;
+ /** EMT: VLAN filter table array. */
+ uint32_t auVFTA[128];
+ /** EMT: Receive buffer size. */
+ uint16_t u16RxBSize;
+ /** EMT: Locked state -- no state alteration possible. */
+ bool fLocked;
+ /** EMT: */
+ bool fDelayInts;
+ /** All: */
+ bool fIntMaskUsed;
+
+ /** N/A: */
+ bool volatile fMaybeOutOfSpace;
+ /** EMT: Gets signalled when more RX descriptors become available. */
+ SUPSEMEVENT hEventMoreRxDescAvail;
+#ifdef E1K_WITH_RXD_CACHE
+ /** RX: Fetched RX descriptors. */
+ E1KRXDESC aRxDescriptors[E1K_RXD_CACHE_SIZE];
+ //uint64_t aRxDescAddr[E1K_RXD_CACHE_SIZE];
+ /** RX: Actual number of fetched RX descriptors. */
+ uint32_t nRxDFetched;
+ /** RX: Index in cache of RX descriptor being processed. */
+ uint32_t iRxDCurrent;
+#endif /* E1K_WITH_RXD_CACHE */
+
+ /** TX: Context used for TCP segmentation packets. */
+ E1KTXCTX contextTSE;
+ /** TX: Context used for ordinary packets. */
+ E1KTXCTX contextNormal;
+#ifdef E1K_WITH_TXD_CACHE
+ /** TX: Fetched TX descriptors. */
+ E1KTXDESC aTxDescriptors[E1K_TXD_CACHE_SIZE];
+ /** TX: Validity of TX descriptors. Set by e1kLocateTxPacket, used by e1kXmitPacket. */
+ bool afTxDValid[E1K_TXD_CACHE_SIZE];
+ /** TX: Actual number of fetched TX descriptors. */
+ uint8_t nTxDFetched;
+ /** TX: Index in cache of TX descriptor being processed. */
+ uint8_t iTxDCurrent;
+ /** TX: Will this frame be sent as GSO. */
+ bool fGSO;
+ /** Alignment padding. */
+ bool fReserved;
+ /** TX: Number of bytes in next packet. */
+ uint32_t cbTxAlloc;
+
+#endif /* E1K_WITH_TXD_CACHE */
+ /** GSO context. u8Type is set to PDMNETWORKGSOTYPE_INVALID when not
+ * applicable to the current TSE mode. */
+ PDMNETWORKGSO GsoCtx;
+ /** Scratch space for holding the loopback / fallback scatter / gather
+ * descriptor. */
+ union
+ {
+ PDMSCATTERGATHER Sg;
+ uint8_t padding[8 * sizeof(RTUINTPTR)];
+ } uTxFallback;
+ /** TX: Transmit packet buffer use for TSE fallback and loopback. */
+ uint8_t aTxPacketFallback[E1K_MAX_TX_PKT_SIZE];
+ /** TX: Number of bytes assembled in TX packet buffer. */
+ uint16_t u16TxPktLen;
+ /** TX: False will force segmentation in e1000 instead of sending frames as GSO. */
+ bool fGSOEnabled;
+ /** TX: IP checksum has to be inserted if true. */
+ bool fIPcsum;
+ /** TX: TCP/UDP checksum has to be inserted if true. */
+ bool fTCPcsum;
+ /** TX: VLAN tag has to be inserted if true. */
+ bool fVTag;
+ /** TX: TCI part of VLAN tag to be inserted. */
+ uint16_t u16VTagTCI;
+ /** TX TSE fallback: Number of payload bytes remaining in TSE context. */
+ uint32_t u32PayRemain;
+ /** TX TSE fallback: Number of header bytes remaining in TSE context. */
+ uint16_t u16HdrRemain;
+ /** TX TSE fallback: Flags from template header. */
+ uint16_t u16SavedFlags;
+ /** TX TSE fallback: Partial checksum from template header. */
+ uint32_t u32SavedCsum;
+ /** ?: Emulated controller type. */
+ E1KCHIP eChip;
+
+ /** EMT: Physical interface emulation. */
+ PHY phy;
+
+#if 0
+ /** Alignment padding. */
+ uint8_t Alignment[HC_ARCH_BITS == 64 ? 8 : 4];
+#endif
+
+ STAMCOUNTER StatReceiveBytes;
+ STAMCOUNTER StatTransmitBytes;
+#if defined(VBOX_WITH_STATISTICS)
+ STAMPROFILEADV StatMMIOReadRZ;
+ STAMPROFILEADV StatMMIOReadR3;
+ STAMPROFILEADV StatMMIOWriteRZ;
+ STAMPROFILEADV StatMMIOWriteR3;
+ STAMPROFILEADV StatEEPROMRead;
+ STAMPROFILEADV StatEEPROMWrite;
+ STAMPROFILEADV StatIOReadRZ;
+ STAMPROFILEADV StatIOReadR3;
+ STAMPROFILEADV StatIOWriteRZ;
+ STAMPROFILEADV StatIOWriteR3;
+ STAMPROFILEADV StatLateIntTimer;
+ STAMCOUNTER StatLateInts;
+ STAMCOUNTER StatIntsRaised;
+ STAMCOUNTER StatIntsPrevented;
+ STAMPROFILEADV StatReceive;
+ STAMPROFILEADV StatReceiveCRC;
+ STAMPROFILEADV StatReceiveFilter;
+ STAMPROFILEADV StatReceiveStore;
+ STAMPROFILEADV StatTransmitRZ;
+ STAMPROFILEADV StatTransmitR3;
+ STAMPROFILE StatTransmitSendRZ;
+ STAMPROFILE StatTransmitSendR3;
+ STAMPROFILE StatRxOverflow;
+ STAMCOUNTER StatRxOverflowWakeupRZ;
+ STAMCOUNTER StatRxOverflowWakeupR3;
+ STAMCOUNTER StatTxDescCtxNormal;
+ STAMCOUNTER StatTxDescCtxTSE;
+ STAMCOUNTER StatTxDescLegacy;
+ STAMCOUNTER StatTxDescData;
+ STAMCOUNTER StatTxDescTSEData;
+ STAMCOUNTER StatTxPathFallback;
+ STAMCOUNTER StatTxPathGSO;
+ STAMCOUNTER StatTxPathRegular;
+ STAMCOUNTER StatPHYAccesses;
+ STAMCOUNTER aStatRegWrites[E1K_NUM_OF_REGS];
+ STAMCOUNTER aStatRegReads[E1K_NUM_OF_REGS];
+#endif /* VBOX_WITH_STATISTICS */
+
+#ifdef E1K_INT_STATS
+ /* Internal stats */
+ uint64_t u64ArmedAt;
+ uint64_t uStatMaxTxDelay;
+ uint32_t uStatInt;
+ uint32_t uStatIntTry;
+ uint32_t uStatIntLower;
+ uint32_t uStatNoIntICR;
+ int32_t iStatIntLost;
+ int32_t iStatIntLostOne;
+ uint32_t uStatIntIMS;
+ uint32_t uStatIntSkip;
+ uint32_t uStatIntLate;
+ uint32_t uStatIntMasked;
+ uint32_t uStatIntEarly;
+ uint32_t uStatIntRx;
+ uint32_t uStatIntTx;
+ uint32_t uStatIntICS;
+ uint32_t uStatIntRDTR;
+ uint32_t uStatIntRXDMT0;
+ uint32_t uStatIntTXQE;
+ uint32_t uStatTxNoRS;
+ uint32_t uStatTxIDE;
+ uint32_t uStatTxDelayed;
+ uint32_t uStatTxDelayExp;
+ uint32_t uStatTAD;
+ uint32_t uStatTID;
+ uint32_t uStatRAD;
+ uint32_t uStatRID;
+ uint32_t uStatRxFrm;
+ uint32_t uStatTxFrm;
+ uint32_t uStatDescCtx;
+ uint32_t uStatDescDat;
+ uint32_t uStatDescLeg;
+ uint32_t uStatTx1514;
+ uint32_t uStatTx2962;
+ uint32_t uStatTx4410;
+ uint32_t uStatTx5858;
+ uint32_t uStatTx7306;
+ uint32_t uStatTx8754;
+ uint32_t uStatTx16384;
+ uint32_t uStatTx32768;
+ uint32_t uStatTxLarge;
+ uint32_t uStatAlign;
+#endif /* E1K_INT_STATS */
+} E1KSTATE;
+/** Pointer to the E1000 device state. */
+typedef E1KSTATE *PE1KSTATE;
+
+/**
+ * E1000 ring-3 device state
+ *
+ * @implements PDMINETWORKDOWN
+ * @implements PDMINETWORKCONFIG
+ * @implements PDMILEDPORTS
+ */
+typedef struct E1KSTATER3
+{
+ PDMIBASE IBase;
+ PDMINETWORKDOWN INetworkDown;
+ PDMINETWORKCONFIG INetworkConfig;
+ /** LED interface */
+ PDMILEDPORTS ILeds;
+ /** Attached network driver. */
+ R3PTRTYPE(PPDMIBASE) pDrvBase;
+ R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
+
+ /** Pointer to the shared state. */
+ R3PTRTYPE(PE1KSTATE) pShared;
+
+ /** Device instance. */
+ PPDMDEVINSR3 pDevInsR3;
+ /** Attached network driver. */
+ PPDMINETWORKUPR3 pDrvR3;
+ /** The scatter / gather buffer used for the current outgoing packet. */
+ R3PTRTYPE(PPDMSCATTERGATHER) pTxSgR3;
+
+ /** EMT: EEPROM emulation */
+ E1kEEPROM eeprom;
+} E1KSTATER3;
+/** Pointer to the E1000 ring-3 device state. */
+typedef E1KSTATER3 *PE1KSTATER3;
+
+
+/**
+ * E1000 ring-0 device state
+ */
+typedef struct E1KSTATER0
+{
+ /** Device instance. */
+ PPDMDEVINSR0 pDevInsR0;
+ /** Attached network driver. */
+ PPDMINETWORKUPR0 pDrvR0;
+ /** The scatter / gather buffer used for the current outgoing packet - R0. */
+ R0PTRTYPE(PPDMSCATTERGATHER) pTxSgR0;
+} E1KSTATER0;
+/** Pointer to the E1000 ring-0 device state. */
+typedef E1KSTATER0 *PE1KSTATER0;
+
+
+/**
+ * E1000 raw-mode device state
+ */
+typedef struct E1KSTATERC
+{
+ /** Device instance. */
+ PPDMDEVINSRC pDevInsRC;
+ /** Attached network driver. */
+ PPDMINETWORKUPRC pDrvRC;
+ /** The scatter / gather buffer used for the current outgoing packet. */
+ RCPTRTYPE(PPDMSCATTERGATHER) pTxSgRC;
+} E1KSTATERC;
+/** Pointer to the E1000 raw-mode device state. */
+typedef E1KSTATERC *PE1KSTATERC;
+
+
+/** @def PE1KSTATECC
+ * Pointer to the instance data for the current context. */
+#ifdef IN_RING3
+typedef E1KSTATER3 E1KSTATECC;
+typedef PE1KSTATER3 PE1KSTATECC;
+#elif defined(IN_RING0)
+typedef E1KSTATER0 E1KSTATECC;
+typedef PE1KSTATER0 PE1KSTATECC;
+#elif defined(IN_RC)
+typedef E1KSTATERC E1KSTATECC;
+typedef PE1KSTATERC PE1KSTATECC;
+#else
+# error "Not IN_RING3, IN_RING0 or IN_RC"
+#endif
+
+
+#ifndef VBOX_DEVICE_STRUCT_TESTCASE
+
+/* Forward declarations ******************************************************/
+static int e1kXmitPending(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread);
+
+/**
+ * E1000 register read handler.
+ */
+typedef int (FNE1KREGREAD)(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value);
+/**
+ * E1000 register write handler.
+ */
+typedef int (FNE1KREGWRITE)(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t u32Value);
+
+static FNE1KREGREAD e1kRegReadUnimplemented;
+static FNE1KREGWRITE e1kRegWriteUnimplemented;
+static FNE1KREGREAD e1kRegReadAutoClear;
+static FNE1KREGREAD e1kRegReadDefault;
+static FNE1KREGWRITE e1kRegWriteDefault;
+#if 0 /* unused */
+static FNE1KREGREAD e1kRegReadCTRL;
+#endif
+static FNE1KREGWRITE e1kRegWriteCTRL;
+static FNE1KREGREAD e1kRegReadEECD;
+static FNE1KREGWRITE e1kRegWriteEECD;
+static FNE1KREGWRITE e1kRegWriteEERD;
+static FNE1KREGWRITE e1kRegWriteMDIC;
+static FNE1KREGREAD e1kRegReadICR;
+static FNE1KREGWRITE e1kRegWriteICR;
+static FNE1KREGREAD e1kRegReadICS;
+static FNE1KREGWRITE e1kRegWriteICS;
+static FNE1KREGWRITE e1kRegWriteIMS;
+static FNE1KREGWRITE e1kRegWriteIMC;
+static FNE1KREGWRITE e1kRegWriteRCTL;
+static FNE1KREGWRITE e1kRegWritePBA;
+static FNE1KREGWRITE e1kRegWriteRDT;
+static FNE1KREGWRITE e1kRegWriteRDTR;
+static FNE1KREGWRITE e1kRegWriteTDT;
+static FNE1KREGREAD e1kRegReadMTA;
+static FNE1KREGWRITE e1kRegWriteMTA;
+static FNE1KREGREAD e1kRegReadRA;
+static FNE1KREGWRITE e1kRegWriteRA;
+static FNE1KREGREAD e1kRegReadVFTA;
+static FNE1KREGWRITE e1kRegWriteVFTA;
+
+/**
+ * Register map table.
+ *
+ * Override pfnRead and pfnWrite to get register-specific behavior.
+ */
+static const struct E1kRegMap_st
+{
+ /** Register offset in the register space. */
+ uint32_t offset;
+ /** Size in bytes. Registers of size > 4 are in fact tables. */
+ uint32_t size;
+ /** Readable bits. */
+ uint32_t readable;
+ /** Writable bits. */
+ uint32_t writable;
+ /** Read callback. */
+ FNE1KREGREAD *pfnRead;
+ /** Write callback. */
+ FNE1KREGWRITE *pfnWrite;
+ /** Abbreviated name. */
+ const char *abbrev;
+ /** Full name. */
+ const char *name;
+} g_aE1kRegMap[E1K_NUM_OF_REGS] =
+{
+ /* offset size read mask write mask read callback write callback abbrev full name */
+ /*------- ------- ---------- ---------- ----------------------- ------------------------ ---------- ------------------------------*/
+ { 0x00000, 0x00004, 0xDBF31BE9, 0xDBF31BE9, e1kRegReadDefault , e1kRegWriteCTRL , "CTRL" , "Device Control" },
+ { 0x00008, 0x00004, 0x0000FDFF, 0x00000000, e1kRegReadDefault , e1kRegWriteUnimplemented, "STATUS" , "Device Status" },
+ { 0x00010, 0x00004, 0x000027F0, 0x00000070, e1kRegReadEECD , e1kRegWriteEECD , "EECD" , "EEPROM/Flash Control/Data" },
+ { 0x00014, 0x00004, 0xFFFFFF10, 0xFFFFFF00, e1kRegReadDefault , e1kRegWriteEERD , "EERD" , "EEPROM Read" },
+ { 0x00018, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "CTRL_EXT", "Extended Device Control" },
+ { 0x0001c, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FLA" , "Flash Access (N/A)" },
+ { 0x00020, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteMDIC , "MDIC" , "MDI Control" },
+ { 0x00028, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCAL" , "Flow Control Address Low" },
+ { 0x0002c, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCAH" , "Flow Control Address High" },
+ { 0x00030, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCT" , "Flow Control Type" },
+ { 0x00038, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "VET" , "VLAN EtherType" },
+ { 0x000c0, 0x00004, 0x0001F6DF, 0x0001F6DF, e1kRegReadICR , e1kRegWriteICR , "ICR" , "Interrupt Cause Read" },
+ { 0x000c4, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "ITR" , "Interrupt Throttling" },
+ { 0x000c8, 0x00004, 0x0001F6DF, 0xFFFFFFFF, e1kRegReadICS , e1kRegWriteICS , "ICS" , "Interrupt Cause Set" },
+ { 0x000d0, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteIMS , "IMS" , "Interrupt Mask Set/Read" },
+ { 0x000d8, 0x00004, 0x00000000, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteIMC , "IMC" , "Interrupt Mask Clear" },
+ { 0x00100, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteRCTL , "RCTL" , "Receive Control" },
+ { 0x00170, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCTTV" , "Flow Control Transmit Timer Value" },
+ { 0x00178, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TXCW" , "Transmit Configuration Word (N/A)" },
+ { 0x00180, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RXCW" , "Receive Configuration Word (N/A)" },
+ { 0x00400, 0x00004, 0x017FFFFA, 0x017FFFFA, e1kRegReadDefault , e1kRegWriteDefault , "TCTL" , "Transmit Control" },
+ { 0x00410, 0x00004, 0x3FFFFFFF, 0x3FFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TIPG" , "Transmit IPG" },
+ { 0x00458, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "AIFS" , "Adaptive IFS Throttle - AIT" },
+ { 0x00e00, 0x00004, 0xCFCFCFCF, 0xCFCFCFCF, e1kRegReadDefault , e1kRegWriteDefault , "LEDCTL" , "LED Control" },
+ { 0x01000, 0x00004, 0xFFFF007F, 0x0000007F, e1kRegReadDefault , e1kRegWritePBA , "PBA" , "Packet Buffer Allocation" },
+ { 0x02160, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCRTL" , "Flow Control Receive Threshold Low" },
+ { 0x02168, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCRTH" , "Flow Control Receive Threshold High" },
+ { 0x02410, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFH" , "Receive Data FIFO Head" },
+ { 0x02418, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFT" , "Receive Data FIFO Tail" },
+ { 0x02420, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFHS" , "Receive Data FIFO Head Saved Register" },
+ { 0x02428, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFTS" , "Receive Data FIFO Tail Saved Register" },
+ { 0x02430, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFPC" , "Receive Data FIFO Packet Count" },
+ { 0x02800, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "RDBAL" , "Receive Descriptor Base Low" },
+ { 0x02804, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "RDBAH" , "Receive Descriptor Base High" },
+ { 0x02808, 0x00004, 0x000FFF80, 0x000FFF80, e1kRegReadDefault , e1kRegWriteDefault , "RDLEN" , "Receive Descriptor Length" },
+ { 0x02810, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "RDH" , "Receive Descriptor Head" },
+ { 0x02818, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteRDT , "RDT" , "Receive Descriptor Tail" },
+ { 0x02820, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteRDTR , "RDTR" , "Receive Delay Timer" },
+ { 0x02828, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RXDCTL" , "Receive Descriptor Control" },
+ { 0x0282c, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "RADV" , "Receive Interrupt Absolute Delay Timer" },
+ { 0x02c00, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RSRPD" , "Receive Small Packet Detect Interrupt" },
+ { 0x03000, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TXDMAC" , "TX DMA Control (N/A)" },
+ { 0x03410, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFH" , "Transmit Data FIFO Head" },
+ { 0x03418, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFT" , "Transmit Data FIFO Tail" },
+ { 0x03420, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFHS" , "Transmit Data FIFO Head Saved Register" },
+ { 0x03428, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFTS" , "Transmit Data FIFO Tail Saved Register" },
+ { 0x03430, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFPC" , "Transmit Data FIFO Packet Count" },
+ { 0x03800, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TDBAL" , "Transmit Descriptor Base Low" },
+ { 0x03804, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TDBAH" , "Transmit Descriptor Base High" },
+ { 0x03808, 0x00004, 0x000FFF80, 0x000FFF80, e1kRegReadDefault , e1kRegWriteDefault , "TDLEN" , "Transmit Descriptor Length" },
+ { 0x03810, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "TDH" , "Transmit Descriptor Head" },
+ { 0x03818, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteTDT , "TDT" , "Transmit Descriptor Tail" },
+ { 0x03820, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "TIDV" , "Transmit Interrupt Delay Value" },
+ { 0x03828, 0x00004, 0xFF3F3F3F, 0xFF3F3F3F, e1kRegReadDefault , e1kRegWriteDefault , "TXDCTL" , "Transmit Descriptor Control" },
+ { 0x0382c, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "TADV" , "Transmit Absolute Interrupt Delay Timer" },
+ { 0x03830, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TSPMT" , "TCP Segmentation Pad and Threshold" },
+ { 0x04000, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "CRCERRS" , "CRC Error Count" },
+ { 0x04004, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "ALGNERRC", "Alignment Error Count" },
+ { 0x04008, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "SYMERRS" , "Symbol Error Count" },
+ { 0x0400c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RXERRC" , "RX Error Count" },
+ { 0x04010, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MPC" , "Missed Packets Count" },
+ { 0x04014, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "SCC" , "Single Collision Count" },
+ { 0x04018, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "ECOL" , "Excessive Collisions Count" },
+ { 0x0401c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MCC" , "Multiple Collision Count" },
+ { 0x04020, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "LATECOL" , "Late Collisions Count" },
+ { 0x04028, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "COLC" , "Collision Count" },
+ { 0x04030, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "DC" , "Defer Count" },
+ { 0x04034, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TNCRS" , "Transmit - No CRS" },
+ { 0x04038, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "SEC" , "Sequence Error Count" },
+ { 0x0403c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "CEXTERR" , "Carrier Extension Error Count" },
+ { 0x04040, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RLEC" , "Receive Length Error Count" },
+ { 0x04048, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XONRXC" , "XON Received Count" },
+ { 0x0404c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XONTXC" , "XON Transmitted Count" },
+ { 0x04050, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XOFFRXC" , "XOFF Received Count" },
+ { 0x04054, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XOFFTXC" , "XOFF Transmitted Count" },
+ { 0x04058, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCRUC" , "FC Received Unsupported Count" },
+ { 0x0405c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC64" , "Packets Received (64 Bytes) Count" },
+ { 0x04060, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC127" , "Packets Received (65-127 Bytes) Count" },
+ { 0x04064, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC255" , "Packets Received (128-255 Bytes) Count" },
+ { 0x04068, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC511" , "Packets Received (256-511 Bytes) Count" },
+ { 0x0406c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC1023" , "Packets Received (512-1023 Bytes) Count" },
+ { 0x04070, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC1522" , "Packets Received (1024-Max Bytes)" },
+ { 0x04074, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GPRC" , "Good Packets Received Count" },
+ { 0x04078, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "BPRC" , "Broadcast Packets Received Count" },
+ { 0x0407c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "MPRC" , "Multicast Packets Received Count" },
+ { 0x04080, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GPTC" , "Good Packets Transmitted Count" },
+ { 0x04088, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GORCL" , "Good Octets Received Count (Low)" },
+ { 0x0408c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GORCH" , "Good Octets Received Count (Hi)" },
+ { 0x04090, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GOTCL" , "Good Octets Transmitted Count (Low)" },
+ { 0x04094, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GOTCH" , "Good Octets Transmitted Count (Hi)" },
+ { 0x040a0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RNBC" , "Receive No Buffers Count" },
+ { 0x040a4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RUC" , "Receive Undersize Count" },
+ { 0x040a8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RFC" , "Receive Fragment Count" },
+ { 0x040ac, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "ROC" , "Receive Oversize Count" },
+ { 0x040b0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RJC" , "Receive Jabber Count" },
+ { 0x040b4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MGTPRC" , "Management Packets Received Count" },
+ { 0x040b8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MGTPDC" , "Management Packets Dropped Count" },
+ { 0x040bc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MGTPTC" , "Management Pkts Transmitted Count" },
+ { 0x040c0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TORL" , "Total Octets Received (Lo)" },
+ { 0x040c4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TORH" , "Total Octets Received (Hi)" },
+ { 0x040c8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TOTL" , "Total Octets Transmitted (Lo)" },
+ { 0x040cc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TOTH" , "Total Octets Transmitted (Hi)" },
+ { 0x040d0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TPR" , "Total Packets Received" },
+ { 0x040d4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TPT" , "Total Packets Transmitted" },
+ { 0x040d8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC64" , "Packets Transmitted (64 Bytes) Count" },
+ { 0x040dc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC127" , "Packets Transmitted (65-127 Bytes) Count" },
+ { 0x040e0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC255" , "Packets Transmitted (128-255 Bytes) Count" },
+ { 0x040e4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC511" , "Packets Transmitted (256-511 Bytes) Count" },
+ { 0x040e8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC1023" , "Packets Transmitted (512-1023 Bytes) Count" },
+ { 0x040ec, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC1522" , "Packets Transmitted (1024 Bytes or Greater) Count" },
+ { 0x040f0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "MPTC" , "Multicast Packets Transmitted Count" },
+ { 0x040f4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "BPTC" , "Broadcast Packets Transmitted Count" },
+ { 0x040f8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TSCTC" , "TCP Segmentation Context Transmitted Count" },
+ { 0x040fc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TSCTFC" , "TCP Segmentation Context Tx Fail Count" },
+ { 0x05000, 0x00004, 0x000007FF, 0x000007FF, e1kRegReadDefault , e1kRegWriteDefault , "RXCSUM" , "Receive Checksum Control" },
+ { 0x05800, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUC" , "Wakeup Control" },
+ { 0x05808, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUFC" , "Wakeup Filter Control" },
+ { 0x05810, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUS" , "Wakeup Status" },
+ { 0x05820, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "MANC" , "Management Control" },
+ { 0x05838, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "IPAV" , "IP Address Valid" },
+ { 0x05900, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUPL" , "Wakeup Packet Length" },
+ { 0x05200, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadMTA , e1kRegWriteMTA , "MTA" , "Multicast Table Array (n)" },
+ { 0x05400, 0x00080, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadRA , e1kRegWriteRA , "RA" , "Receive Address (64-bit) (n)" },
+ { 0x05600, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadVFTA , e1kRegWriteVFTA , "VFTA" , "VLAN Filter Table Array (n)" },
+ { 0x05840, 0x0001c, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "IP4AT" , "IPv4 Address Table" },
+ { 0x05880, 0x00010, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "IP6AT" , "IPv6 Address Table" },
+ { 0x05a00, 0x00080, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUPM" , "Wakeup Packet Memory" },
+ { 0x05f00, 0x0001c, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FFLT" , "Flexible Filter Length Table" },
+ { 0x09000, 0x003fc, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FFMT" , "Flexible Filter Mask Table" },
+ { 0x09800, 0x003fc, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FFVT" , "Flexible Filter Value Table" },
+ { 0x10000, 0x10000, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "PBM" , "Packet Buffer Memory (n)" },
+ { 0x00040, 0x00080, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadRA , e1kRegWriteRA , "RA82542" , "Receive Address (64-bit) (n) (82542)" },
+ { 0x00200, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadMTA , e1kRegWriteMTA , "MTA82542", "Multicast Table Array (n) (82542)" },
+ { 0x00600, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadVFTA , e1kRegWriteVFTA , "VFTA82542", "VLAN Filter Table Array (n) (82542)" }
+};
+
+#ifdef LOG_ENABLED
+
+/**
+ * Convert U32 value to hex string. Masked bytes are replaced with dots.
+ *
+ * @remarks The mask has half-byte byte (not bit) granularity (e.g. 0000000F).
+ *
+ * @returns The buffer.
+ *
+ * @param u32 The word to convert into string.
+ * @param mask Selects which bytes to convert.
+ * @param buf Where to put the result.
+ */
+static char *e1kU32toHex(uint32_t u32, uint32_t mask, char *buf)
+{
+ for (char *ptr = buf + 7; ptr >= buf; --ptr, u32 >>=4, mask >>=4)
+ {
+ if (mask & 0xF)
+ *ptr = (u32 & 0xF) + ((u32 & 0xF) > 9 ? '7' : '0');
+ else
+ *ptr = '.';
+ }
+ buf[8] = 0;
+ return buf;
+}
+
+/**
+ * Returns timer name for debug purposes.
+ *
+ * @returns The timer name.
+ *
+ * @param pThis The device state structure.
+ * @param hTimer The timer to name.
+ */
+DECLINLINE(const char *) e1kGetTimerName(PE1KSTATE pThis, TMTIMERHANDLE hTimer)
+{
+ if (hTimer == pThis->hTIDTimer)
+ return "TID";
+ if (hTimer == pThis->hTADTimer)
+ return "TAD";
+ if (hTimer == pThis->hRIDTimer)
+ return "RID";
+ if (hTimer == pThis->hRADTimer)
+ return "RAD";
+ if (hTimer == pThis->hIntTimer)
+ return "Int";
+ if (hTimer == pThis->hTXDTimer)
+ return "TXD";
+ if (hTimer == pThis->hLUTimer)
+ return "LinkUp";
+ return "unknown";
+}
+
+#endif /* LOG_ENABLED */
+
+/**
+ * Arm a timer.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis Pointer to the device state structure.
+ * @param hTimer The timer to arm.
+ * @param uExpireIn Expiration interval in microseconds.
+ */
+DECLINLINE(void) e1kArmTimer(PPDMDEVINS pDevIns, PE1KSTATE pThis, TMTIMERHANDLE hTimer, uint32_t uExpireIn)
+{
+ if (pThis->fLocked)
+ return;
+
+ E1kLog2(("%s Arming %s timer to fire in %d usec...\n",
+ pThis->szPrf, e1kGetTimerName(pThis, hTimer), uExpireIn));
+ int rc = PDMDevHlpTimerSetMicro(pDevIns, hTimer, uExpireIn);
+ AssertRC(rc);
+}
+
+#ifdef IN_RING3
+/**
+ * Cancel a timer.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis Pointer to the device state structure.
+ * @param pTimer Pointer to the timer.
+ */
+DECLINLINE(void) e1kCancelTimer(PPDMDEVINS pDevIns, PE1KSTATE pThis, TMTIMERHANDLE hTimer)
+{
+ E1kLog2(("%s Stopping %s timer...\n",
+ pThis->szPrf, e1kGetTimerName(pThis, hTimer)));
+ int rc = PDMDevHlpTimerStop(pDevIns, hTimer);
+ if (RT_FAILURE(rc))
+ E1kLog2(("%s e1kCancelTimer: TMTimerStop(%s) failed with %Rrc\n",
+ pThis->szPrf, e1kGetTimerName(pThis, hTimer), rc));
+ RT_NOREF_PV(pThis);
+}
+#endif /* IN_RING3 */
+
+
+#define e1kCsEnter(ps, rcBusy) PDMDevHlpCritSectEnter(pDevIns, &(ps)->cs, (rcBusy))
+#define e1kCsEnterReturn(ps, rcBusy) do { \
+ int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->cs, (rcBusy)); \
+ if (rcLock == VINF_SUCCESS) { /* likely */ } \
+ else return rcLock; \
+ } while (0)
+#define e1kR3CsEnterAsserted(ps) do { \
+ int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->cs, VERR_SEM_BUSY); \
+ PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &(ps)->cs, rcLock); \
+ } while (0)
+#define e1kCsLeave(ps) PDMDevHlpCritSectLeave(pDevIns, &(ps)->cs)
+
+
+#define e1kCsRxEnter(ps, rcBusy) PDMDevHlpCritSectEnter(pDevIns, &(ps)->csRx, (rcBusy))
+#define e1kCsRxEnterReturn(ps) do { \
+ int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->csRx, VERR_SEM_BUSY); \
+ AssertRCReturn(rcLock, rcLock); \
+ } while (0)
+#define e1kR3CsRxEnterAsserted(ps) do { \
+ int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->csRx, VERR_SEM_BUSY); \
+ PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &(ps)->csRx, rcLock); \
+ } while (0)
+#define e1kCsRxLeave(ps) PDMDevHlpCritSectLeave(pDevIns, &(ps)->csRx)
+#define e1kCsRxIsOwner(ps) PDMDevHlpCritSectIsOwner(pDevIns, &(ps)->csRx)
+
+
+#ifndef E1K_WITH_TX_CS
+# define e1kCsTxEnter(ps, rcBusy) VINF_SUCCESS
+# define e1kR3CsTxEnterAsserted(ps) do { } while (0)
+# define e1kCsTxLeave(ps) do { } while (0)
+#else /* E1K_WITH_TX_CS */
+# define e1kCsTxEnter(ps, rcBusy) PDMDevHlpCritSectEnter(pDevIns, &(ps)->csTx, (rcBusy))
+# define e1kR3CsTxEnterAsserted(ps) do { \
+ int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->csTx, VERR_SEM_BUSY); \
+ PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &(ps)->csTx, rcLock); \
+ } while (0)
+# define e1kCsTxLeave(ps) PDMDevHlpCritSectLeave(pDevIns, &(ps)->csTx)
+# define e1kCsTxIsOwner(ps) PDMDevHlpCritSectIsOwner(pDevIns, &(ps)->csTx)
+#endif /* E1K_WITH_TX_CS */
+
+
+#ifdef E1K_WITH_TXD_CACHE
+/*
+ * Transmit Descriptor Register Context
+ */
+struct E1kTxDContext
+{
+ uint32_t tdlen;
+ uint32_t tdh;
+ uint32_t tdt;
+ uint8_t nextPacket;
+};
+typedef struct E1kTxDContext E1KTXDC, *PE1KTXDC;
+
+DECLINLINE(bool) e1kUpdateTxDContext(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pContext)
+{
+ Assert(e1kCsTxIsOwner(pThis));
+ if (!e1kCsTxIsOwner(pThis))
+ {
+ memset(pContext, 0, sizeof(E1KTXDC));
+ return false;
+ }
+ pContext->tdlen = TDLEN;
+ pContext->tdh = TDH;
+ pContext->tdt = TDT;
+ uint32_t cTxRingSize = pContext->tdlen / sizeof(E1KTXDESC);
+#ifdef DEBUG
+ if (pContext->tdh >= cTxRingSize)
+ {
+ Log(("%s e1kUpdateTxDContext: will return false because TDH too big (%u >= %u)\n",
+ pThis->szPrf, pContext->tdh, cTxRingSize));
+ return VINF_SUCCESS;
+ }
+ if (pContext->tdt >= cTxRingSize)
+ {
+ Log(("%s e1kUpdateTxDContext: will return false because TDT too big (%u >= %u)\n",
+ pThis->szPrf, pContext->tdt, cTxRingSize));
+ return VINF_SUCCESS;
+ }
+#endif /* DEBUG */
+ return pContext->tdh < cTxRingSize && pContext->tdt < cTxRingSize;
+}
+#endif /* E1K_WITH_TXD_CACHE */
+#ifdef E1K_WITH_RXD_CACHE
+/*
+ * Receive Descriptor Register Context
+ */
+struct E1kRxDContext
+{
+ uint32_t rdlen;
+ uint32_t rdh;
+ uint32_t rdt;
+};
+typedef struct E1kRxDContext E1KRXDC, *PE1KRXDC;
+
+DECLINLINE(bool) e1kUpdateRxDContext(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pContext, const char *pcszCallee)
+{
+ Assert(e1kCsRxIsOwner(pThis));
+ if (!e1kCsRxIsOwner(pThis))
+ return false;
+ pContext->rdlen = RDLEN;
+ pContext->rdh = RDH;
+ pContext->rdt = RDT;
+ uint32_t cRxRingSize = pContext->rdlen / sizeof(E1KRXDESC);
+ /*
+ * Note that the checks for RDT are a bit different. Some guests, OS/2 for
+ * example, intend to use all descriptors in RX ring, so they point RDT
+ * right beyond the last descriptor in the ring. While this is not
+ * acceptable for other registers, it works out fine for RDT.
+ */
+#ifdef DEBUG
+ if (pContext->rdh >= cRxRingSize)
+ {
+ Log(("%s e1kUpdateRxDContext: called from %s, will return false because RDH too big (%u >= %u)\n",
+ pThis->szPrf, pcszCallee, pContext->rdh, cRxRingSize));
+ return VINF_SUCCESS;
+ }
+ if (pContext->rdt > cRxRingSize)
+ {
+ Log(("%s e1kUpdateRxDContext: called from %s, will return false because RDT too big (%u > %u)\n",
+ pThis->szPrf, pcszCallee, pContext->rdt, cRxRingSize));
+ return VINF_SUCCESS;
+ }
+#else /* !DEBUG */
+ RT_NOREF(pcszCallee);
+#endif /* !DEBUG */
+ return pContext->rdh < cRxRingSize && pContext->rdt <= cRxRingSize; // && (RCTL & RCTL_EN);
+}
+#endif /* E1K_WITH_RXD_CACHE */
+
+/**
+ * Wakeup the RX thread.
+ */
+static void e1kWakeupReceive(PPDMDEVINS pDevIns, PE1KSTATE pThis)
+{
+ if ( pThis->fMaybeOutOfSpace
+ && pThis->hEventMoreRxDescAvail != NIL_SUPSEMEVENT)
+ {
+ STAM_COUNTER_INC(&pThis->CTX_SUFF_Z(StatRxOverflowWakeup));
+ E1kLog(("%s Waking up Out-of-RX-space semaphore\n", pThis->szPrf));
+ int rc = PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEventMoreRxDescAvail);
+ AssertRC(rc);
+ }
+}
+
+#ifdef IN_RING3
+
+/**
+ * Hardware reset. Revert all registers to initial values.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ */
+static void e1kR3HardReset(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
+{
+ E1kLog(("%s Hard reset triggered\n", pThis->szPrf));
+ /* No interrupts should survive device reset, see @bugref(9556). */
+ if (pThis->fIntRaised)
+ {
+ /* Lower(0) INTA(0) */
+ PDMDevHlpPCISetIrq(pDevIns, 0, 0);
+ pThis->fIntRaised = false;
+ E1kLog(("%s e1kR3HardReset: Lowered IRQ: ICR=%08x\n", pThis->szPrf, ICR));
+ }
+ memset(pThis->auRegs, 0, sizeof(pThis->auRegs));
+ memset(pThis->aRecAddr.au32, 0, sizeof(pThis->aRecAddr.au32));
+# ifdef E1K_INIT_RA0
+ memcpy(pThis->aRecAddr.au32, pThis->macConfigured.au8,
+ sizeof(pThis->macConfigured.au8));
+ pThis->aRecAddr.array[0].ctl |= RA_CTL_AV;
+# endif /* E1K_INIT_RA0 */
+ STATUS = 0x0081; /* SPEED=10b (1000 Mb/s), FD=1b (Full Duplex) */
+ EECD = 0x0100; /* EE_PRES=1b (EEPROM present) */
+ CTRL = 0x0a09; /* FRCSPD=1b SPEED=10b LRST=1b FD=1b */
+ TSPMT = 0x01000400;/* TSMT=0400h TSPBP=0100h */
+ Assert(GET_BITS(RCTL, BSIZE) == 0);
+ pThis->u16RxBSize = 2048;
+
+ uint16_t u16LedCtl = 0x0602; /* LED0/LINK_UP#, LED2/LINK100# */
+ pThisCC->eeprom.readWord(0x2F, &u16LedCtl); /* Read LEDCTL defaults from EEPROM */
+ LEDCTL = 0x07008300 | (((uint32_t)u16LedCtl & 0xCF00) << 8) | (u16LedCtl & 0xCF); /* Only LED0 and LED2 defaults come from EEPROM */
+
+ /* Reset promiscuous mode */
+ if (pThisCC->pDrvR3)
+ pThisCC->pDrvR3->pfnSetPromiscuousMode(pThisCC->pDrvR3, false);
+
+# ifdef E1K_WITH_TXD_CACHE
+ e1kR3CsTxEnterAsserted(pThis);
+ pThis->nTxDFetched = 0;
+ pThis->iTxDCurrent = 0;
+ pThis->fGSO = false;
+ pThis->cbTxAlloc = 0;
+ e1kCsTxLeave(pThis);
+# endif /* E1K_WITH_TXD_CACHE */
+# ifdef E1K_WITH_RXD_CACHE
+ e1kR3CsRxEnterAsserted(pThis);
+ pThis->iRxDCurrent = pThis->nRxDFetched = 0;
+ e1kCsRxLeave(pThis);
+# endif /* E1K_WITH_RXD_CACHE */
+# ifdef E1K_LSC_ON_RESET
+ E1kLog(("%s Will trigger LSC in %d seconds...\n",
+ pThis->szPrf, pThis->cMsLinkUpDelay / 1000));
+ e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, pThis->cMsLinkUpDelay * 1000);
+# endif /* E1K_LSC_ON_RESET */
+}
+
+#endif /* IN_RING3 */
+
+/**
+ * Compute Internet checksum.
+ *
+ * @remarks Refer to http://www.netfor2.com/checksum.html for short intro.
+ *
+ * @param pThis The device state structure.
+ * @param cpPacket The packet.
+ * @param cb The size of the packet.
+ * @param pszText A string denoting direction of packet transfer.
+ *
+ * @return The 1's complement of the 1's complement sum.
+ *
+ * @thread E1000_TX
+ */
+static uint16_t e1kCSum16(const void *pvBuf, size_t cb)
+{
+ uint32_t csum = 0;
+ uint16_t *pu16 = (uint16_t *)pvBuf;
+
+ while (cb > 1)
+ {
+ csum += *pu16++;
+ cb -= 2;
+ }
+ if (cb)
+ csum += *(uint8_t*)pu16;
+ while (csum >> 16)
+ csum = (csum >> 16) + (csum & 0xFFFF);
+ Assert(csum < 65536);
+ return (uint16_t)~csum;
+}
+
+/**
+ * Dump a packet to debug log.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param cpPacket The packet.
+ * @param cb The size of the packet.
+ * @param pszText A string denoting direction of packet transfer.
+ * @thread E1000_TX
+ */
+DECLINLINE(void) e1kPacketDump(PPDMDEVINS pDevIns, PE1KSTATE pThis, const uint8_t *cpPacket, size_t cb, const char *pszText)
+{
+#ifdef DEBUG
+ if (RT_LIKELY(e1kCsEnter(pThis, VERR_SEM_BUSY) == VINF_SUCCESS))
+ {
+ Log4(("%s --- %s packet #%d: %RTmac => %RTmac (%d bytes) ---\n",
+ pThis->szPrf, pszText, ++pThis->u32PktNo, cpPacket+6, cpPacket, cb));
+ if (ntohs(*(uint16_t*)(cpPacket+12)) == 0x86DD)
+ {
+ Log4(("%s --- IPv6: %RTnaipv6 => %RTnaipv6\n",
+ pThis->szPrf, cpPacket+14+8, cpPacket+14+24));
+ if (*(cpPacket+14+6) == 0x6)
+ Log4(("%s --- TCP: seq=%x ack=%x\n", pThis->szPrf,
+ ntohl(*(uint32_t*)(cpPacket+14+40+4)), ntohl(*(uint32_t*)(cpPacket+14+40+8))));
+ }
+ else if (ntohs(*(uint16_t*)(cpPacket+12)) == 0x800)
+ {
+ Log4(("%s --- IPv4: %RTnaipv4 => %RTnaipv4\n",
+ pThis->szPrf, *(uint32_t*)(cpPacket+14+12), *(uint32_t*)(cpPacket+14+16)));
+ if (*(cpPacket+14+6) == 0x6)
+ Log4(("%s --- TCP: seq=%x ack=%x\n", pThis->szPrf,
+ ntohl(*(uint32_t*)(cpPacket+14+20+4)), ntohl(*(uint32_t*)(cpPacket+14+20+8))));
+ }
+ E1kLog3(("%.*Rhxd\n", cb, cpPacket));
+ e1kCsLeave(pThis);
+ }
+#else
+ if (RT_LIKELY(e1kCsEnter(pThis, VERR_SEM_BUSY) == VINF_SUCCESS))
+ {
+ if (ntohs(*(uint16_t*)(cpPacket+12)) == 0x86DD)
+ E1kLogRel(("E1000: %s packet #%d, %RTmac => %RTmac, %RTnaipv6 => %RTnaipv6, seq=%x ack=%x\n",
+ pszText, ++pThis->u32PktNo, cpPacket+6, cpPacket, cpPacket+14+8, cpPacket+14+24,
+ ntohl(*(uint32_t*)(cpPacket+14+40+4)), ntohl(*(uint32_t*)(cpPacket+14+40+8))));
+ else
+ E1kLogRel(("E1000: %s packet #%d, %RTmac => %RTmac, %RTnaipv4 => %RTnaipv4, seq=%x ack=%x\n",
+ pszText, ++pThis->u32PktNo, cpPacket+6, cpPacket,
+ *(uint32_t*)(cpPacket+14+12), *(uint32_t*)(cpPacket+14+16),
+ ntohl(*(uint32_t*)(cpPacket+14+20+4)), ntohl(*(uint32_t*)(cpPacket+14+20+8))));
+ e1kCsLeave(pThis);
+ }
+ RT_NOREF2(cb, pszText);
+#endif
+}
+
+/**
+ * Determine the type of transmit descriptor.
+ *
+ * @returns Descriptor type. See E1K_DTYP_XXX defines.
+ *
+ * @param pDesc Pointer to descriptor union.
+ * @thread E1000_TX
+ */
+DECLINLINE(int) e1kGetDescType(E1KTXDESC *pDesc)
+{
+ if (pDesc->legacy.cmd.fDEXT)
+ return pDesc->context.dw2.u4DTYP;
+ return E1K_DTYP_LEGACY;
+}
+
+
+#ifdef E1K_WITH_RXD_CACHE
+/**
+ * Return the number of RX descriptor that belong to the hardware.
+ *
+ * @returns the number of available descriptors in RX ring.
+ * @param pRxdc The receive descriptor register context.
+ * @thread ???
+ */
+DECLINLINE(uint32_t) e1kGetRxLen(PE1KRXDC pRxdc)
+{
+ /**
+ * Make sure RDT won't change during computation. EMT may modify RDT at
+ * any moment.
+ */
+ uint32_t rdt = pRxdc->rdt;
+ return (pRxdc->rdh > rdt ? pRxdc->rdlen/sizeof(E1KRXDESC) : 0) + rdt - pRxdc->rdh;
+}
+
+DECLINLINE(unsigned) e1kRxDInCache(PE1KSTATE pThis)
+{
+ return pThis->nRxDFetched > pThis->iRxDCurrent ?
+ pThis->nRxDFetched - pThis->iRxDCurrent : 0;
+}
+
+DECLINLINE(unsigned) e1kRxDIsCacheEmpty(PE1KSTATE pThis)
+{
+ return pThis->iRxDCurrent >= pThis->nRxDFetched;
+}
+
+/**
+ * Load receive descriptors from guest memory. The caller needs to be in Rx
+ * critical section.
+ *
+ * We need two physical reads in case the tail wrapped around the end of RX
+ * descriptor ring.
+ *
+ * @returns the actual number of descriptors fetched.
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @thread EMT, RX
+ */
+DECLINLINE(unsigned) e1kRxDPrefetch(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pRxdc)
+{
+ E1kLog3(("%s e1kRxDPrefetch: RDH=%x RDT=%x RDLEN=%x "
+ "iRxDCurrent=%x nRxDFetched=%x\n",
+ pThis->szPrf, pRxdc->rdh, pRxdc->rdt, pRxdc->rdlen, pThis->iRxDCurrent, pThis->nRxDFetched));
+ /* We've already loaded pThis->nRxDFetched descriptors past RDH. */
+ unsigned nDescsAvailable = e1kGetRxLen(pRxdc) - e1kRxDInCache(pThis);
+ unsigned nDescsToFetch = RT_MIN(nDescsAvailable, E1K_RXD_CACHE_SIZE - pThis->nRxDFetched);
+ unsigned nDescsTotal = pRxdc->rdlen / sizeof(E1KRXDESC);
+ Assert(nDescsTotal != 0);
+ if (nDescsTotal == 0)
+ return 0;
+ unsigned nFirstNotLoaded = (pRxdc->rdh + e1kRxDInCache(pThis)) % nDescsTotal;
+ unsigned nDescsInSingleRead = RT_MIN(nDescsToFetch, nDescsTotal - nFirstNotLoaded);
+ E1kLog3(("%s e1kRxDPrefetch: nDescsAvailable=%u nDescsToFetch=%u "
+ "nDescsTotal=%u nFirstNotLoaded=0x%x nDescsInSingleRead=%u\n",
+ pThis->szPrf, nDescsAvailable, nDescsToFetch, nDescsTotal,
+ nFirstNotLoaded, nDescsInSingleRead));
+ if (nDescsToFetch == 0)
+ return 0;
+ E1KRXDESC* pFirstEmptyDesc = &pThis->aRxDescriptors[pThis->nRxDFetched];
+ PDMDevHlpPCIPhysRead(pDevIns,
+ ((uint64_t)RDBAH << 32) + RDBAL + nFirstNotLoaded * sizeof(E1KRXDESC),
+ pFirstEmptyDesc, nDescsInSingleRead * sizeof(E1KRXDESC));
+ // uint64_t addrBase = ((uint64_t)RDBAH << 32) + RDBAL;
+ // unsigned i, j;
+ // for (i = pThis->nRxDFetched; i < pThis->nRxDFetched + nDescsInSingleRead; ++i)
+ // {
+ // pThis->aRxDescAddr[i] = addrBase + (nFirstNotLoaded + i - pThis->nRxDFetched) * sizeof(E1KRXDESC);
+ // E1kLog3(("%s aRxDescAddr[%d] = %p\n", pThis->szPrf, i, pThis->aRxDescAddr[i]));
+ // }
+ E1kLog3(("%s Fetched %u RX descriptors at %08x%08x(0x%x), RDLEN=%08x, RDH=%08x, RDT=%08x\n",
+ pThis->szPrf, nDescsInSingleRead,
+ RDBAH, RDBAL + pRxdc->rdh * sizeof(E1KRXDESC),
+ nFirstNotLoaded, pRxdc->rdlen, pRxdc->rdh, pRxdc->rdt));
+ if (nDescsToFetch > nDescsInSingleRead)
+ {
+ PDMDevHlpPCIPhysRead(pDevIns,
+ ((uint64_t)RDBAH << 32) + RDBAL,
+ pFirstEmptyDesc + nDescsInSingleRead,
+ (nDescsToFetch - nDescsInSingleRead) * sizeof(E1KRXDESC));
+ // Assert(i == pThis->nRxDFetched + nDescsInSingleRead);
+ // for (j = 0; i < pThis->nRxDFetched + nDescsToFetch; ++i, ++j)
+ // {
+ // pThis->aRxDescAddr[i] = addrBase + j * sizeof(E1KRXDESC);
+ // E1kLog3(("%s aRxDescAddr[%d] = %p\n", pThis->szPrf, i, pThis->aRxDescAddr[i]));
+ // }
+ E1kLog3(("%s Fetched %u RX descriptors at %08x%08x\n",
+ pThis->szPrf, nDescsToFetch - nDescsInSingleRead,
+ RDBAH, RDBAL));
+ }
+ pThis->nRxDFetched += nDescsToFetch;
+ return nDescsToFetch;
+}
+
+# ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
+/**
+ * Dump receive descriptor to debug log.
+ *
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to the descriptor.
+ * @thread E1000_RX
+ */
+static void e1kPrintRDesc(PE1KSTATE pThis, E1KRXDESC *pDesc)
+{
+ RT_NOREF2(pThis, pDesc);
+ E1kLog2(("%s <-- Receive Descriptor (%d bytes):\n", pThis->szPrf, pDesc->u16Length));
+ E1kLog2((" Address=%16LX Length=%04X Csum=%04X\n",
+ pDesc->u64BufAddr, pDesc->u16Length, pDesc->u16Checksum));
+ E1kLog2((" STA: %s %s %s %s %s %s %s ERR: %s %s %s %s SPECIAL: %s VLAN=%03x PRI=%x\n",
+ pDesc->status.fPIF ? "PIF" : "pif",
+ pDesc->status.fIPCS ? "IPCS" : "ipcs",
+ pDesc->status.fTCPCS ? "TCPCS" : "tcpcs",
+ pDesc->status.fVP ? "VP" : "vp",
+ pDesc->status.fIXSM ? "IXSM" : "ixsm",
+ pDesc->status.fEOP ? "EOP" : "eop",
+ pDesc->status.fDD ? "DD" : "dd",
+ pDesc->status.fRXE ? "RXE" : "rxe",
+ pDesc->status.fIPE ? "IPE" : "ipe",
+ pDesc->status.fTCPE ? "TCPE" : "tcpe",
+ pDesc->status.fCE ? "CE" : "ce",
+ E1K_SPEC_CFI(pDesc->status.u16Special) ? "CFI" :"cfi",
+ E1K_SPEC_VLAN(pDesc->status.u16Special),
+ E1K_SPEC_PRI(pDesc->status.u16Special)));
+}
+# endif /* IN_RING3 */
+#endif /* E1K_WITH_RXD_CACHE */
+
+/**
+ * Dump transmit descriptor to debug log.
+ *
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to descriptor union.
+ * @param pszDir A string denoting direction of descriptor transfer
+ * @thread E1000_TX
+ */
+static void e1kPrintTDesc(PE1KSTATE pThis, E1KTXDESC *pDesc, const char *pszDir,
+ unsigned uLevel = RTLOGGRPFLAGS_LEVEL_2)
+{
+ RT_NOREF4(pThis, pDesc, pszDir, uLevel);
+
+ /*
+ * Unfortunately we cannot use our format handler here, we want R0 logging
+ * as well.
+ */
+ switch (e1kGetDescType(pDesc))
+ {
+ case E1K_DTYP_CONTEXT:
+ E1kLogX(uLevel, ("%s %s Context Transmit Descriptor %s\n",
+ pThis->szPrf, pszDir, pszDir));
+ E1kLogX(uLevel, (" IPCSS=%02X IPCSO=%02X IPCSE=%04X TUCSS=%02X TUCSO=%02X TUCSE=%04X\n",
+ pDesc->context.ip.u8CSS, pDesc->context.ip.u8CSO, pDesc->context.ip.u16CSE,
+ pDesc->context.tu.u8CSS, pDesc->context.tu.u8CSO, pDesc->context.tu.u16CSE));
+ E1kLogX(uLevel, (" TUCMD:%s%s%s %s %s PAYLEN=%04x HDRLEN=%04x MSS=%04x STA: %s\n",
+ pDesc->context.dw2.fIDE ? " IDE":"",
+ pDesc->context.dw2.fRS ? " RS" :"",
+ pDesc->context.dw2.fTSE ? " TSE":"",
+ pDesc->context.dw2.fIP ? "IPv4":"IPv6",
+ pDesc->context.dw2.fTCP ? "TCP":"UDP",
+ pDesc->context.dw2.u20PAYLEN,
+ pDesc->context.dw3.u8HDRLEN,
+ pDesc->context.dw3.u16MSS,
+ pDesc->context.dw3.fDD?"DD":""));
+ break;
+ case E1K_DTYP_DATA:
+ E1kLogX(uLevel, ("%s %s Data Transmit Descriptor (%d bytes) %s\n",
+ pThis->szPrf, pszDir, pDesc->data.cmd.u20DTALEN, pszDir));
+ E1kLogX(uLevel, (" Address=%16LX DTALEN=%05X\n",
+ pDesc->data.u64BufAddr,
+ pDesc->data.cmd.u20DTALEN));
+ E1kLogX(uLevel, (" DCMD:%s%s%s%s%s%s%s STA:%s%s%s POPTS:%s%s SPECIAL:%s VLAN=%03x PRI=%x\n",
+ pDesc->data.cmd.fIDE ? " IDE" :"",
+ pDesc->data.cmd.fVLE ? " VLE" :"",
+ pDesc->data.cmd.fRPS ? " RPS" :"",
+ pDesc->data.cmd.fRS ? " RS" :"",
+ pDesc->data.cmd.fTSE ? " TSE" :"",
+ pDesc->data.cmd.fIFCS? " IFCS":"",
+ pDesc->data.cmd.fEOP ? " EOP" :"",
+ pDesc->data.dw3.fDD ? " DD" :"",
+ pDesc->data.dw3.fEC ? " EC" :"",
+ pDesc->data.dw3.fLC ? " LC" :"",
+ pDesc->data.dw3.fTXSM? " TXSM":"",
+ pDesc->data.dw3.fIXSM? " IXSM":"",
+ E1K_SPEC_CFI(pDesc->data.dw3.u16Special) ? "CFI" :"cfi",
+ E1K_SPEC_VLAN(pDesc->data.dw3.u16Special),
+ E1K_SPEC_PRI(pDesc->data.dw3.u16Special)));
+ break;
+ case E1K_DTYP_LEGACY:
+ E1kLogX(uLevel, ("%s %s Legacy Transmit Descriptor (%d bytes) %s\n",
+ pThis->szPrf, pszDir, pDesc->legacy.cmd.u16Length, pszDir));
+ E1kLogX(uLevel, (" Address=%16LX DTALEN=%05X\n",
+ pDesc->data.u64BufAddr,
+ pDesc->legacy.cmd.u16Length));
+ E1kLogX(uLevel, (" CMD:%s%s%s%s%s%s%s STA:%s%s%s CSO=%02x CSS=%02x SPECIAL:%s VLAN=%03x PRI=%x\n",
+ pDesc->legacy.cmd.fIDE ? " IDE" :"",
+ pDesc->legacy.cmd.fVLE ? " VLE" :"",
+ pDesc->legacy.cmd.fRPS ? " RPS" :"",
+ pDesc->legacy.cmd.fRS ? " RS" :"",
+ pDesc->legacy.cmd.fIC ? " IC" :"",
+ pDesc->legacy.cmd.fIFCS? " IFCS":"",
+ pDesc->legacy.cmd.fEOP ? " EOP" :"",
+ pDesc->legacy.dw3.fDD ? " DD" :"",
+ pDesc->legacy.dw3.fEC ? " EC" :"",
+ pDesc->legacy.dw3.fLC ? " LC" :"",
+ pDesc->legacy.cmd.u8CSO,
+ pDesc->legacy.dw3.u8CSS,
+ E1K_SPEC_CFI(pDesc->legacy.dw3.u16Special) ? "CFI" :"cfi",
+ E1K_SPEC_VLAN(pDesc->legacy.dw3.u16Special),
+ E1K_SPEC_PRI(pDesc->legacy.dw3.u16Special)));
+ break;
+ default:
+ E1kLog(("%s %s Invalid Transmit Descriptor %s\n",
+ pThis->szPrf, pszDir, pszDir));
+ break;
+ }
+}
+
+/**
+ * Raise an interrupt later.
+ *
+ * @param pThis The device state structure.
+ */
+DECLINLINE(void) e1kPostponeInterrupt(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint64_t nsDeadline)
+{
+ if (!PDMDevHlpTimerIsActive(pDevIns, pThis->hIntTimer))
+ PDMDevHlpTimerSetNano(pDevIns, pThis->hIntTimer, nsDeadline);
+}
+
+/**
+ * Raise interrupt if not masked.
+ *
+ * @param pThis The device state structure.
+ */
+static int e1kRaiseInterrupt(PPDMDEVINS pDevIns, PE1KSTATE pThis, int rcBusy, uint32_t u32IntCause)
+{
+ /* Do NOT use e1kCsEnterReturn here as most callers doesn't check the
+ status code. They'll pass a negative rcBusy. */
+ int rc = e1kCsEnter(pThis, rcBusy);
+ if (RT_LIKELY(rc == VINF_SUCCESS))
+ { /* likely */ }
+ else
+ {
+ PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &pThis->cs, rc);
+ return rc;
+ }
+
+ E1K_INC_ISTAT_CNT(pThis->uStatIntTry);
+ ICR |= u32IntCause;
+ if (ICR & IMS)
+ {
+ if (pThis->fIntRaised)
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatIntSkip);
+ E1kLog2(("%s e1kRaiseInterrupt: Already raised, skipped. ICR&IMS=%08x\n",
+ pThis->szPrf, ICR & IMS));
+ }
+ else
+ {
+ uint64_t tsNow = PDMDevHlpTimerGet(pDevIns, pThis->hIntTimer);
+ if (!!ITR && tsNow - pThis->u64AckedAt < ITR * 256
+ && pThis->fItrEnabled && (pThis->fItrRxEnabled || !(ICR & ICR_RXT0)))
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatIntEarly);
+ E1kLog2(("%s e1kRaiseInterrupt: Too early to raise again: %d ns < %d ns.\n",
+ pThis->szPrf, (uint32_t)(tsNow - pThis->u64AckedAt), ITR * 256));
+ e1kPostponeInterrupt(pDevIns, pThis, ITR * 256);
+ }
+ else
+ {
+
+ /* Since we are delivering the interrupt now
+ * there is no need to do it later -- stop the timer.
+ */
+ PDMDevHlpTimerStop(pDevIns, pThis->hIntTimer);
+ E1K_INC_ISTAT_CNT(pThis->uStatInt);
+ STAM_COUNTER_INC(&pThis->StatIntsRaised);
+ /* Got at least one unmasked interrupt cause */
+ pThis->fIntRaised = true;
+ /* Raise(1) INTA(0) */
+ E1kLogRel(("E1000: irq RAISED icr&mask=0x%x, icr=0x%x\n", ICR & IMS, ICR));
+ PDMDevHlpPCISetIrq(pDevIns, 0, 1);
+ E1kLog(("%s e1kRaiseInterrupt: Raised. ICR&IMS=%08x\n",
+ pThis->szPrf, ICR & IMS));
+ }
+ }
+ }
+ else
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatIntMasked);
+ E1kLog2(("%s e1kRaiseInterrupt: Not raising, ICR=%08x, IMS=%08x\n",
+ pThis->szPrf, ICR, IMS));
+ }
+ e1kCsLeave(pThis);
+ return VINF_SUCCESS;
+}
+
+/**
+ * Compute the physical address of the descriptor.
+ *
+ * @returns the physical address of the descriptor.
+ *
+ * @param baseHigh High-order 32 bits of descriptor table address.
+ * @param baseLow Low-order 32 bits of descriptor table address.
+ * @param idxDesc The descriptor index in the table.
+ */
+DECLINLINE(RTGCPHYS) e1kDescAddr(uint32_t baseHigh, uint32_t baseLow, uint32_t idxDesc)
+{
+ AssertCompile(sizeof(E1KRXDESC) == sizeof(E1KTXDESC));
+ return ((uint64_t)baseHigh << 32) + baseLow + idxDesc * sizeof(E1KRXDESC);
+}
+
+#ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
+/**
+ * Advance the head pointer of the receive descriptor queue.
+ *
+ * @remarks RDH always points to the next available RX descriptor.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ */
+DECLINLINE(void) e1kAdvanceRDH(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pRxdc)
+{
+ Assert(e1kCsRxIsOwner(pThis));
+ //e1kR3CsEnterAsserted(pThis);
+ if (++pRxdc->rdh * sizeof(E1KRXDESC) >= pRxdc->rdlen)
+ pRxdc->rdh = 0;
+ RDH = pRxdc->rdh; /* Sync the actual register and RXDC */
+#ifdef E1K_WITH_RXD_CACHE
+ /*
+ * We need to fetch descriptors now as the guest may advance RDT all the way
+ * to RDH as soon as we generate RXDMT0 interrupt. This is mostly to provide
+ * compatibility with Phar Lap ETS, see @bugref(7346). Note that we do not
+ * check if the receiver is enabled. It must be, otherwise we won't get here
+ * in the first place.
+ *
+ * Note that we should have moved both RDH and iRxDCurrent by now.
+ */
+ if (e1kRxDIsCacheEmpty(pThis))
+ {
+ /* Cache is empty, reset it and check if we can fetch more. */
+ pThis->iRxDCurrent = pThis->nRxDFetched = 0;
+ E1kLog3(("%s e1kAdvanceRDH: Rx cache is empty, RDH=%x RDT=%x "
+ "iRxDCurrent=%x nRxDFetched=%x\n",
+ pThis->szPrf, pRxdc->rdh, pRxdc->rdt, pThis->iRxDCurrent, pThis->nRxDFetched));
+ e1kRxDPrefetch(pDevIns, pThis, pRxdc);
+ }
+#endif /* E1K_WITH_RXD_CACHE */
+ /*
+ * Compute current receive queue length and fire RXDMT0 interrupt
+ * if we are low on receive buffers
+ */
+ uint32_t uRQueueLen = pRxdc->rdh>pRxdc->rdt ? pRxdc->rdlen/sizeof(E1KRXDESC)-pRxdc->rdh+pRxdc->rdt : pRxdc->rdt-pRxdc->rdh;
+ /*
+ * The minimum threshold is controlled by RDMTS bits of RCTL:
+ * 00 = 1/2 of RDLEN
+ * 01 = 1/4 of RDLEN
+ * 10 = 1/8 of RDLEN
+ * 11 = reserved
+ */
+ uint32_t uMinRQThreshold = pRxdc->rdlen / sizeof(E1KRXDESC) / (2 << GET_BITS(RCTL, RDMTS));
+ if (uRQueueLen <= uMinRQThreshold)
+ {
+ E1kLogRel(("E1000: low on RX descriptors, RDH=%x RDT=%x len=%x threshold=%x\n", pRxdc->rdh, pRxdc->rdt, uRQueueLen, uMinRQThreshold));
+ E1kLog2(("%s Low on RX descriptors, RDH=%x RDT=%x len=%x threshold=%x, raise an interrupt\n",
+ pThis->szPrf, pRxdc->rdh, pRxdc->rdt, uRQueueLen, uMinRQThreshold));
+ E1K_INC_ISTAT_CNT(pThis->uStatIntRXDMT0);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_RXDMT0);
+ }
+ E1kLog2(("%s e1kAdvanceRDH: at exit RDH=%x RDT=%x len=%x\n",
+ pThis->szPrf, pRxdc->rdh, pRxdc->rdt, uRQueueLen));
+ //e1kCsLeave(pThis);
+}
+#endif /* IN_RING3 */
+
+#ifdef E1K_WITH_RXD_CACHE
+
+# ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
+
+/**
+ * Obtain the next RX descriptor from RXD cache, fetching descriptors from the
+ * RX ring if the cache is empty.
+ *
+ * Note that we cannot advance the cache pointer (iRxDCurrent) yet as it will
+ * go out of sync with RDH which will cause trouble when EMT checks if the
+ * cache is empty to do pre-fetch @bugref(6217).
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @thread RX
+ */
+DECLINLINE(E1KRXDESC *) e1kRxDGet(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pRxdc)
+{
+ Assert(e1kCsRxIsOwner(pThis));
+ /* Check the cache first. */
+ if (pThis->iRxDCurrent < pThis->nRxDFetched)
+ return &pThis->aRxDescriptors[pThis->iRxDCurrent];
+ /* Cache is empty, reset it and check if we can fetch more. */
+ pThis->iRxDCurrent = pThis->nRxDFetched = 0;
+ if (e1kRxDPrefetch(pDevIns, pThis, pRxdc))
+ return &pThis->aRxDescriptors[pThis->iRxDCurrent];
+ /* Out of Rx descriptors. */
+ return NULL;
+}
+
+
+/**
+ * Return the RX descriptor obtained with e1kRxDGet() and advance the cache
+ * pointer. The descriptor gets written back to the RXD ring.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pDesc The descriptor being "returned" to the RX ring.
+ * @thread RX
+ */
+DECLINLINE(void) e1kRxDPut(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KRXDESC* pDesc, PE1KRXDC pRxdc)
+{
+ Assert(e1kCsRxIsOwner(pThis));
+ pThis->iRxDCurrent++;
+ // Assert(pDesc >= pThis->aRxDescriptors);
+ // Assert(pDesc < pThis->aRxDescriptors + E1K_RXD_CACHE_SIZE);
+ // uint64_t addr = e1kDescAddr(RDBAH, RDBAL, RDH);
+ // uint32_t rdh = RDH;
+ // Assert(pThis->aRxDescAddr[pDesc - pThis->aRxDescriptors] == addr);
+ PDMDevHlpPCIPhysWrite(pDevIns, e1kDescAddr(RDBAH, RDBAL, pRxdc->rdh), pDesc, sizeof(E1KRXDESC));
+ /*
+ * We need to print the descriptor before advancing RDH as it may fetch new
+ * descriptors into the cache.
+ */
+ e1kPrintRDesc(pThis, pDesc);
+ e1kAdvanceRDH(pDevIns, pThis, pRxdc);
+}
+
+/**
+ * Store a fragment of received packet at the specifed address.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pDesc The next available RX descriptor.
+ * @param pvBuf The fragment.
+ * @param cb The size of the fragment.
+ */
+static void e1kStoreRxFragment(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KRXDESC *pDesc, const void *pvBuf, size_t cb)
+{
+ STAM_PROFILE_ADV_START(&pThis->StatReceiveStore, a);
+ E1kLog2(("%s e1kStoreRxFragment: store fragment of %04X at %016LX, EOP=%d\n",
+ pThis->szPrf, cb, pDesc->u64BufAddr, pDesc->status.fEOP));
+ PDMDevHlpPCIPhysWrite(pDevIns, pDesc->u64BufAddr, pvBuf, cb);
+ pDesc->u16Length = (uint16_t)cb;
+ Assert(pDesc->u16Length == cb);
+ STAM_PROFILE_ADV_STOP(&pThis->StatReceiveStore, a);
+ RT_NOREF(pThis);
+}
+
+# endif /* IN_RING3 */
+
+#else /* !E1K_WITH_RXD_CACHE */
+
+/**
+ * Store a fragment of received packet that fits into the next available RX
+ * buffer.
+ *
+ * @remarks Trigger the RXT0 interrupt if it is the last fragment of the packet.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pDesc The next available RX descriptor.
+ * @param pvBuf The fragment.
+ * @param cb The size of the fragment.
+ */
+static void e1kStoreRxFragment(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KRXDESC *pDesc, const void *pvBuf, size_t cb)
+{
+ STAM_PROFILE_ADV_START(&pThis->StatReceiveStore, a);
+ E1kLog2(("%s e1kStoreRxFragment: store fragment of %04X at %016LX, EOP=%d\n", pThis->szPrf, cb, pDesc->u64BufAddr, pDesc->status.fEOP));
+ PDMDevHlpPCIPhysWrite(pDevIns, pDesc->u64BufAddr, pvBuf, cb);
+ pDesc->u16Length = (uint16_t)cb; Assert(pDesc->u16Length == cb);
+ /* Write back the descriptor */
+ PDMDevHlpPCIPhysWrite(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), pDesc, sizeof(E1KRXDESC));
+ e1kPrintRDesc(pThis, pDesc);
+ E1kLogRel(("E1000: Wrote back RX desc, RDH=%x\n", RDH));
+ /* Advance head */
+ e1kAdvanceRDH(pDevIns, pThis);
+ //E1kLog2(("%s e1kStoreRxFragment: EOP=%d RDTR=%08X RADV=%08X\n", pThis->szPrf, pDesc->fEOP, RDTR, RADV));
+ if (pDesc->status.fEOP)
+ {
+ /* Complete packet has been stored -- it is time to let the guest know. */
+#ifdef E1K_USE_RX_TIMERS
+ if (RDTR)
+ {
+ /* Arm the timer to fire in RDTR usec (discard .024) */
+ e1kArmTimer(pDevIns, pThis, pThis->hRIDTimer, RDTR);
+ /* If absolute timer delay is enabled and the timer is not running yet, arm it. */
+ if (RADV != 0 && !PDMDevHlpTimerIsActive(pDevIns, pThis->CTX_SUFF(pRADTimer)))
+ e1kArmTimer(pThis, pThis->hRADTimer, RADV);
+ }
+ else
+ {
+#endif
+ /* 0 delay means immediate interrupt */
+ E1K_INC_ISTAT_CNT(pThis->uStatIntRx);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_RXT0);
+#ifdef E1K_USE_RX_TIMERS
+ }
+#endif
+ }
+ STAM_PROFILE_ADV_STOP(&pThis->StatReceiveStore, a);
+}
+
+#endif /* !E1K_WITH_RXD_CACHE */
+
+/**
+ * Returns true if it is a broadcast packet.
+ *
+ * @returns true if destination address indicates broadcast.
+ * @param pvBuf The ethernet packet.
+ */
+DECLINLINE(bool) e1kIsBroadcast(const void *pvBuf)
+{
+ static const uint8_t s_abBcastAddr[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
+ return memcmp(pvBuf, s_abBcastAddr, sizeof(s_abBcastAddr)) == 0;
+}
+
+/**
+ * Returns true if it is a multicast packet.
+ *
+ * @remarks returns true for broadcast packets as well.
+ * @returns true if destination address indicates multicast.
+ * @param pvBuf The ethernet packet.
+ */
+DECLINLINE(bool) e1kIsMulticast(const void *pvBuf)
+{
+ return (*(char*)pvBuf) & 1;
+}
+
+#ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
+/**
+ * Set IXSM, IPCS and TCPCS flags according to the packet type.
+ *
+ * @remarks We emulate checksum offloading for major packets types only.
+ *
+ * @returns VBox status code.
+ * @param pThis The device state structure.
+ * @param pFrame The available data.
+ * @param cb Number of bytes available in the buffer.
+ * @param status Bit fields containing status info.
+ */
+static int e1kRxChecksumOffload(PE1KSTATE pThis, const uint8_t *pFrame, size_t cb, E1KRXDST *pStatus)
+{
+ /** @todo
+ * It is not safe to bypass checksum verification for packets coming
+ * from real wire. We currently unable to tell where packets are
+ * coming from so we tell the driver to ignore our checksum flags
+ * and do verification in software.
+ */
+# if 0
+ uint16_t uEtherType = ntohs(*(uint16_t*)(pFrame + 12));
+
+ E1kLog2(("%s e1kRxChecksumOffload: EtherType=%x\n", pThis->szPrf, uEtherType));
+
+ switch (uEtherType)
+ {
+ case 0x800: /* IPv4 */
+ {
+ pStatus->fIXSM = false;
+ pStatus->fIPCS = true;
+ PRTNETIPV4 pIpHdr4 = (PRTNETIPV4)(pFrame + 14);
+ /* TCP/UDP checksum offloading works with TCP and UDP only */
+ pStatus->fTCPCS = pIpHdr4->ip_p == 6 || pIpHdr4->ip_p == 17;
+ break;
+ }
+ case 0x86DD: /* IPv6 */
+ pStatus->fIXSM = false;
+ pStatus->fIPCS = false;
+ pStatus->fTCPCS = true;
+ break;
+ default: /* ARP, VLAN, etc. */
+ pStatus->fIXSM = true;
+ break;
+ }
+# else
+ pStatus->fIXSM = true;
+ RT_NOREF_PV(pThis); RT_NOREF_PV(pFrame); RT_NOREF_PV(cb);
+# endif
+ return VINF_SUCCESS;
+}
+#endif /* IN_RING3 */
+
+/**
+ * Pad and store received packet.
+ *
+ * @remarks Make sure that the packet appears to upper layer as one coming
+ * from real Ethernet: pad it and insert FCS.
+ *
+ * @returns VBox status code.
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pvBuf The available data.
+ * @param cb Number of bytes available in the buffer.
+ * @param status Bit fields containing status info.
+ */
+static int e1kHandleRxPacket(PPDMDEVINS pDevIns, PE1KSTATE pThis, const void *pvBuf, size_t cb, E1KRXDST status)
+{
+#if defined(IN_RING3) /** @todo Remove this extra copying, it's gonna make us run out of kernel / hypervisor stack! */
+ uint8_t rxPacket[E1K_MAX_RX_PKT_SIZE];
+ uint8_t *ptr = rxPacket;
+# ifdef E1K_WITH_RXD_CACHE
+ E1KRXDC rxdc;
+# endif /* E1K_WITH_RXD_CACHE */
+
+ e1kCsRxEnterReturn(pThis);
+# ifdef E1K_WITH_RXD_CACHE
+ if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kHandleRxPacket")))
+ {
+ e1kCsRxLeave(pThis);
+ E1kLog(("%s e1kHandleRxPacket: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
+ return VINF_SUCCESS;
+ }
+# endif /* E1K_WITH_RXD_CACHE */
+
+ if (cb > 70) /* unqualified guess */
+ pThis->led.Asserted.s.fReading = pThis->led.Actual.s.fReading = 1;
+
+ Assert(cb <= E1K_MAX_RX_PKT_SIZE);
+ Assert(cb > 16);
+ size_t cbMax = ((RCTL & RCTL_LPE) ? E1K_MAX_RX_PKT_SIZE - 4 : 1518) - (status.fVP ? 0 : 4);
+ E1kLog3(("%s Max RX packet size is %u\n", pThis->szPrf, cbMax));
+ if (status.fVP)
+ {
+ /* VLAN packet -- strip VLAN tag in VLAN mode */
+ if ((CTRL & CTRL_VME) && cb > 16)
+ {
+ uint16_t *u16Ptr = (uint16_t*)pvBuf;
+ memcpy(rxPacket, pvBuf, 12); /* Copy src and dst addresses */
+ status.u16Special = RT_BE2H_U16(u16Ptr[7]); /* Extract VLAN tag */
+ memcpy(rxPacket + 12, (uint8_t*)pvBuf + 16, cb - 16); /* Copy the rest of the packet */
+ cb -= 4;
+ E1kLog3(("%s Stripped tag for VLAN %u (cb=%u)\n",
+ pThis->szPrf, status.u16Special, cb));
+ }
+ else
+ {
+ status.fVP = false; /* Set VP only if we stripped the tag */
+ memcpy(rxPacket, pvBuf, cb);
+ }
+ }
+ else
+ memcpy(rxPacket, pvBuf, cb);
+ /* Pad short packets */
+ if (cb < 60)
+ {
+ memset(rxPacket + cb, 0, 60 - cb);
+ cb = 60;
+ }
+ if (!(RCTL & RCTL_SECRC) && cb <= cbMax)
+ {
+ STAM_PROFILE_ADV_START(&pThis->StatReceiveCRC, a);
+ /*
+ * Add FCS if CRC stripping is not enabled. Since the value of CRC
+ * is ignored by most of drivers we may as well save us the trouble
+ * of calculating it (see EthernetCRC CFGM parameter).
+ */
+ if (pThis->fEthernetCRC)
+ *(uint32_t*)(rxPacket + cb) = RTCrc32(rxPacket, cb);
+ cb += sizeof(uint32_t);
+ STAM_PROFILE_ADV_STOP(&pThis->StatReceiveCRC, a);
+ E1kLog3(("%s Added FCS (cb=%u)\n", pThis->szPrf, cb));
+ }
+ /* Compute checksum of complete packet */
+ size_t cbCSumStart = RT_MIN(GET_BITS(RXCSUM, PCSS), cb);
+ uint16_t checksum = e1kCSum16(rxPacket + cbCSumStart, cb - cbCSumStart);
+ e1kRxChecksumOffload(pThis, rxPacket, cb, &status);
+
+ /* Update stats */
+ E1K_INC_CNT32(GPRC);
+ if (e1kIsBroadcast(pvBuf))
+ E1K_INC_CNT32(BPRC);
+ else if (e1kIsMulticast(pvBuf))
+ E1K_INC_CNT32(MPRC);
+ /* Update octet receive counter */
+ E1K_ADD_CNT64(GORCL, GORCH, cb);
+ STAM_REL_COUNTER_ADD(&pThis->StatReceiveBytes, cb);
+ if (cb == 64)
+ E1K_INC_CNT32(PRC64);
+ else if (cb < 128)
+ E1K_INC_CNT32(PRC127);
+ else if (cb < 256)
+ E1K_INC_CNT32(PRC255);
+ else if (cb < 512)
+ E1K_INC_CNT32(PRC511);
+ else if (cb < 1024)
+ E1K_INC_CNT32(PRC1023);
+ else
+ E1K_INC_CNT32(PRC1522);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatRxFrm);
+
+# ifdef E1K_WITH_RXD_CACHE
+ while (cb > 0)
+ {
+ E1KRXDESC *pDesc = e1kRxDGet(pDevIns, pThis, &rxdc);
+
+ if (pDesc == NULL)
+ {
+ E1kLog(("%s Out of receive buffers, dropping the packet "
+ "(cb=%u, in_cache=%u, RDH=%x RDT=%x)\n",
+ pThis->szPrf, cb, e1kRxDInCache(pThis), rxdc.rdh, rxdc.rdt));
+ break;
+ }
+# else /* !E1K_WITH_RXD_CACHE */
+ if (RDH == RDT)
+ {
+ E1kLog(("%s Out of receive buffers, dropping the packet\n",
+ pThis->szPrf));
+ }
+ /* Store the packet to receive buffers */
+ while (RDH != RDT)
+ {
+ /* Load the descriptor pointed by head */
+ E1KRXDESC desc, *pDesc = &desc;
+ PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), &desc, sizeof(desc));
+# endif /* !E1K_WITH_RXD_CACHE */
+ if (pDesc->u64BufAddr)
+ {
+ uint16_t u16RxBufferSize = pThis->u16RxBSize; /* see @bugref{9427} */
+
+ /* Update descriptor */
+ pDesc->status = status;
+ pDesc->u16Checksum = checksum;
+ pDesc->status.fDD = true;
+
+ /*
+ * We need to leave Rx critical section here or we risk deadlocking
+ * with EMT in e1kRegWriteRDT when the write is to an unallocated
+ * page or has an access handler associated with it.
+ * Note that it is safe to leave the critical section here since
+ * e1kRegWriteRDT() never modifies RDH. It never touches already
+ * fetched RxD cache entries either.
+ */
+ if (cb > u16RxBufferSize)
+ {
+ pDesc->status.fEOP = false;
+ e1kCsRxLeave(pThis);
+ e1kStoreRxFragment(pDevIns, pThis, pDesc, ptr, u16RxBufferSize);
+ e1kCsRxEnterReturn(pThis);
+# ifdef E1K_WITH_RXD_CACHE
+ if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kHandleRxPacket")))
+ {
+ e1kCsRxLeave(pThis);
+ E1kLog(("%s e1kHandleRxPacket: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
+ return VINF_SUCCESS;
+ }
+# endif /* E1K_WITH_RXD_CACHE */
+ ptr += u16RxBufferSize;
+ cb -= u16RxBufferSize;
+ }
+ else
+ {
+ pDesc->status.fEOP = true;
+ e1kCsRxLeave(pThis);
+ e1kStoreRxFragment(pDevIns, pThis, pDesc, ptr, cb);
+# ifdef E1K_WITH_RXD_CACHE
+ e1kCsRxEnterReturn(pThis);
+ if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kHandleRxPacket")))
+ {
+ e1kCsRxLeave(pThis);
+ E1kLog(("%s e1kHandleRxPacket: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
+ return VINF_SUCCESS;
+ }
+ cb = 0;
+# else /* !E1K_WITH_RXD_CACHE */
+ pThis->led.Actual.s.fReading = 0;
+ return VINF_SUCCESS;
+# endif /* !E1K_WITH_RXD_CACHE */
+ }
+ /*
+ * Note: RDH is advanced by e1kStoreRxFragment if E1K_WITH_RXD_CACHE
+ * is not defined.
+ */
+ }
+# ifdef E1K_WITH_RXD_CACHE
+ /* Write back the descriptor. */
+ pDesc->status.fDD = true;
+ e1kRxDPut(pDevIns, pThis, pDesc, &rxdc);
+# else /* !E1K_WITH_RXD_CACHE */
+ else
+ {
+ /* Write back the descriptor. */
+ pDesc->status.fDD = true;
+ PDMDevHlpPCIPhysWrite(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), pDesc, sizeof(E1KRXDESC));
+ e1kAdvanceRDH(pDevIns, pThis);
+ }
+# endif /* !E1K_WITH_RXD_CACHE */
+ }
+
+ if (cb > 0)
+ E1kLog(("%s Out of receive buffers, dropping %u bytes", pThis->szPrf, cb));
+
+ pThis->led.Actual.s.fReading = 0;
+
+ e1kCsRxLeave(pThis);
+# ifdef E1K_WITH_RXD_CACHE
+ /* Complete packet has been stored -- it is time to let the guest know. */
+# ifdef E1K_USE_RX_TIMERS
+ if (RDTR)
+ {
+ /* Arm the timer to fire in RDTR usec (discard .024) */
+ e1kArmTimer(pThis, pThis->hRIDTimer, RDTR);
+ /* If absolute timer delay is enabled and the timer is not running yet, arm it. */
+ if (RADV != 0 && !PDMDevHlpTimerIsActive(pDevIns, pThis->hRADTimer))
+ e1kArmTimer(pThis, pThis->hRADTimer, RADV);
+ }
+ else
+ {
+# endif /* E1K_USE_RX_TIMERS */
+ /* 0 delay means immediate interrupt */
+ E1K_INC_ISTAT_CNT(pThis->uStatIntRx);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_RXT0);
+# ifdef E1K_USE_RX_TIMERS
+ }
+# endif /* E1K_USE_RX_TIMERS */
+# endif /* E1K_WITH_RXD_CACHE */
+
+ return VINF_SUCCESS;
+#else /* !IN_RING3 */
+ RT_NOREF(pDevIns, pThis, pvBuf, cb, status);
+ return VERR_INTERNAL_ERROR_2;
+#endif /* !IN_RING3 */
+}
+
+
+#ifdef IN_RING3
+/**
+ * Bring the link up after the configured delay, 5 seconds by default.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @thread any
+ */
+DECLINLINE(void) e1kBringLinkUpDelayed(PPDMDEVINS pDevIns, PE1KSTATE pThis)
+{
+ E1kLog(("%s Will bring up the link in %d seconds...\n",
+ pThis->szPrf, pThis->cMsLinkUpDelay / 1000));
+ e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, pThis->cMsLinkUpDelay * 1000);
+}
+
+/**
+ * Bring up the link immediately.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ */
+DECLINLINE(void) e1kR3LinkUp(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
+{
+ E1kLog(("%s Link is up\n", pThis->szPrf));
+ STATUS |= STATUS_LU;
+ Phy::setLinkStatus(&pThis->phy, true);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_LSC);
+ if (pThisCC->pDrvR3)
+ pThisCC->pDrvR3->pfnNotifyLinkChanged(pThisCC->pDrvR3, PDMNETWORKLINKSTATE_UP);
+ /* Trigger processing of pending TX descriptors (see @bugref{8942}). */
+ PDMDevHlpTaskTrigger(pDevIns, pThis->hTxTask);
+}
+
+/**
+ * Bring down the link immediately.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ */
+DECLINLINE(void) e1kR3LinkDown(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
+{
+ E1kLog(("%s Link is down\n", pThis->szPrf));
+ STATUS &= ~STATUS_LU;
+#ifdef E1K_LSC_ON_RESET
+ Phy::setLinkStatus(&pThis->phy, false);
+#endif /* E1K_LSC_ON_RESET */
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_LSC);
+ if (pThisCC->pDrvR3)
+ pThisCC->pDrvR3->pfnNotifyLinkChanged(pThisCC->pDrvR3, PDMNETWORKLINKSTATE_DOWN);
+}
+
+/**
+ * Bring down the link temporarily.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ */
+DECLINLINE(void) e1kR3LinkDownTemp(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
+{
+ E1kLog(("%s Link is down temporarily\n", pThis->szPrf));
+ STATUS &= ~STATUS_LU;
+ Phy::setLinkStatus(&pThis->phy, false);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_LSC);
+ /*
+ * Notifying the associated driver that the link went down (even temporarily)
+ * seems to be the right thing, but it was not done before. This may cause
+ * a regression if the driver does not expect the link to go down as a result
+ * of sending PDMNETWORKLINKSTATE_DOWN_RESUME to this device. Earlier versions
+ * of code notified the driver that the link was up! See @bugref{7057}.
+ */
+ if (pThisCC->pDrvR3)
+ pThisCC->pDrvR3->pfnNotifyLinkChanged(pThisCC->pDrvR3, PDMNETWORKLINKSTATE_DOWN);
+ e1kBringLinkUpDelayed(pDevIns, pThis);
+}
+#endif /* IN_RING3 */
+
+#if 0 /* unused */
+/**
+ * Read handler for Device Status register.
+ *
+ * Get the link status from PHY.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param mask Used to implement partial reads (8 and 16-bit).
+ */
+static int e1kRegReadCTRL(PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ E1kLog(("%s e1kRegReadCTRL: mdio dir=%s mdc dir=%s mdc=%d\n",
+ pThis->szPrf, (CTRL & CTRL_MDIO_DIR)?"OUT":"IN ",
+ (CTRL & CTRL_MDC_DIR)?"OUT":"IN ", !!(CTRL & CTRL_MDC)));
+ if ((CTRL & CTRL_MDIO_DIR) == 0 && (CTRL & CTRL_MDC))
+ {
+ /* MDC is high and MDIO pin is used for input, read MDIO pin from PHY */
+ if (Phy::readMDIO(&pThis->phy))
+ *pu32Value = CTRL | CTRL_MDIO;
+ else
+ *pu32Value = CTRL & ~CTRL_MDIO;
+ E1kLog(("%s e1kRegReadCTRL: Phy::readMDIO(%d)\n",
+ pThis->szPrf, !!(*pu32Value & CTRL_MDIO)));
+ }
+ else
+ {
+ /* MDIO pin is used for output, ignore it */
+ *pu32Value = CTRL;
+ }
+ return VINF_SUCCESS;
+}
+#endif /* unused */
+
+/**
+ * A helper function to detect the link state to the other side of "the wire".
+ *
+ * When deciding to bring up the link we need to take into account both if the
+ * cable is connected and if our device is actually connected to the outside
+ * world. If no driver is attached we won't be able to allocate TX buffers,
+ * which will prevent us from TX descriptor processing, which will result in
+ * "TX unit hang" in the guest.
+ *
+ * @returns true if the device is connected to something.
+ *
+ * @param pDevIns The device instance.
+ */
+DECLINLINE(bool) e1kIsConnected(PPDMDEVINS pDevIns)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ return pThis->fCableConnected && pThis->fIsAttached;
+}
+
+/**
+ * A callback used by PHY to indicate that the link needs to be updated due to
+ * reset of PHY.
+ *
+ * @param pDevIns The device instance.
+ * @thread any
+ */
+void e1kPhyLinkResetCallback(PPDMDEVINS pDevIns)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+
+ /* Make sure we have cable connected and MAC can talk to PHY */
+ if (e1kIsConnected(pDevIns) && (CTRL & CTRL_SLU))
+ e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, E1K_INIT_LINKUP_DELAY_US);
+ else
+ Log(("%s PHY link reset callback ignored (cable %sconnected, driver %stached, CTRL_SLU=%u)\n", pThis->szPrf,
+ pThis->fCableConnected ? "" : "dis", pThis->fIsAttached ? "at" : "de", CTRL & CTRL_SLU ? 1 : 0));
+}
+
+/**
+ * Write handler for Device Control register.
+ *
+ * Handles reset.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteCTRL(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ int rc = VINF_SUCCESS;
+
+ if (value & CTRL_RESET)
+ { /* RST */
+#ifndef IN_RING3
+ return VINF_IOM_R3_MMIO_WRITE;
+#else
+ e1kR3HardReset(pDevIns, pThis, PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC));
+#endif
+ }
+ else
+ {
+#ifdef E1K_LSC_ON_SLU
+ /*
+ * When the guest changes 'Set Link Up' bit from 0 to 1 we check if
+ * the link is down and the cable is connected, and if they are we
+ * bring the link up, see @bugref{8624}.
+ */
+ if ( (value & CTRL_SLU)
+ && !(CTRL & CTRL_SLU)
+ && pThis->fCableConnected
+ && !(STATUS & STATUS_LU))
+ {
+ /* It should take about 2 seconds for the link to come up */
+ e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, E1K_INIT_LINKUP_DELAY_US);
+ }
+#else /* !E1K_LSC_ON_SLU */
+ if ( (value & CTRL_SLU)
+ && !(CTRL & CTRL_SLU)
+ && e1kIsConnected(pDevIns)
+ && !PDMDevHlpTimerIsActive(pDevIns, pThis->hLUTimer))
+ {
+ /* PXE does not use LSC interrupts, see @bugref{9113}. */
+ STATUS |= STATUS_LU;
+ }
+#endif /* !E1K_LSC_ON_SLU */
+ if ((value & CTRL_VME) != (CTRL & CTRL_VME))
+ {
+ E1kLog(("%s VLAN Mode %s\n", pThis->szPrf, (value & CTRL_VME) ? "Enabled" : "Disabled"));
+ }
+ Log7(("%s e1kRegWriteCTRL: mdio dir=%s mdc dir=%s mdc=%s mdio=%d\n",
+ pThis->szPrf, (value & CTRL_MDIO_DIR)?"OUT":"IN ",
+ (value & CTRL_MDC_DIR)?"OUT":"IN ", (value & CTRL_MDC)?"HIGH":"LOW ", !!(value & CTRL_MDIO)));
+ if (value & CTRL_MDC)
+ {
+ if (value & CTRL_MDIO_DIR)
+ {
+ Log7(("%s e1kRegWriteCTRL: Phy::writeMDIO(%d)\n", pThis->szPrf, !!(value & CTRL_MDIO)));
+ /* MDIO direction pin is set to output and MDC is high, write MDIO pin value to PHY */
+ Phy::writeMDIO(&pThis->phy, !!(value & CTRL_MDIO), pDevIns);
+ }
+ else
+ {
+ if (Phy::readMDIO(&pThis->phy))
+ value |= CTRL_MDIO;
+ else
+ value &= ~CTRL_MDIO;
+ Log7(("%s e1kRegWriteCTRL: Phy::readMDIO(%d)\n", pThis->szPrf, !!(value & CTRL_MDIO)));
+ }
+ }
+ rc = e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+ }
+
+ return rc;
+}
+
+/**
+ * Write handler for EEPROM/Flash Control/Data register.
+ *
+ * Handles EEPROM access requests; forwards writes to EEPROM device if access has been granted.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteEECD(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF(pDevIns, offset, index);
+#ifdef IN_RING3
+ /* So far we are concerned with lower byte only */
+ if ((EECD & EECD_EE_GNT) || pThis->eChip == E1K_CHIP_82543GC)
+ {
+ /* Access to EEPROM granted -- forward 4-wire bits to EEPROM device */
+ /* Note: 82543GC does not need to request EEPROM access */
+ STAM_PROFILE_ADV_START(&pThis->StatEEPROMWrite, a);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ pThisCC->eeprom.write(value & EECD_EE_WIRES);
+ STAM_PROFILE_ADV_STOP(&pThis->StatEEPROMWrite, a);
+ }
+ if (value & EECD_EE_REQ)
+ EECD |= EECD_EE_REQ|EECD_EE_GNT;
+ else
+ EECD &= ~EECD_EE_GNT;
+ //e1kRegWriteDefault(pThis, offset, index, value );
+
+ return VINF_SUCCESS;
+#else /* !IN_RING3 */
+ RT_NOREF(pThis, value);
+ return VINF_IOM_R3_MMIO_WRITE;
+#endif /* !IN_RING3 */
+}
+
+/**
+ * Read handler for EEPROM/Flash Control/Data register.
+ *
+ * Lower 4 bits come from EEPROM device if EEPROM access has been granted.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param mask Used to implement partial reads (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegReadEECD(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+#ifdef IN_RING3
+ uint32_t value = 0; /* Get rid of false positive in parfait. */
+ int rc = e1kRegReadDefault(pDevIns, pThis, offset, index, &value);
+ if (RT_SUCCESS(rc))
+ {
+ if ((value & EECD_EE_GNT) || pThis->eChip == E1K_CHIP_82543GC)
+ {
+ /* Note: 82543GC does not need to request EEPROM access */
+ /* Access to EEPROM granted -- get 4-wire bits to EEPROM device */
+ STAM_PROFILE_ADV_START(&pThis->StatEEPROMRead, a);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ value |= pThisCC->eeprom.read();
+ STAM_PROFILE_ADV_STOP(&pThis->StatEEPROMRead, a);
+ }
+ *pu32Value = value;
+ }
+
+ return rc;
+#else /* !IN_RING3 */
+ RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index); RT_NOREF_PV(pu32Value);
+ return VINF_IOM_R3_MMIO_READ;
+#endif /* !IN_RING3 */
+}
+
+/**
+ * Write handler for EEPROM Read register.
+ *
+ * Handles EEPROM word access requests, reads EEPROM and stores the result
+ * into DATA field.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteEERD(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+#ifdef IN_RING3
+ /* Make use of 'writable' and 'readable' masks. */
+ e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+ /* DONE and DATA are set only if read was triggered by START. */
+ if (value & EERD_START)
+ {
+ STAM_PROFILE_ADV_START(&pThis->StatEEPROMRead, a);
+ uint16_t tmp;
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ if (pThisCC->eeprom.readWord(GET_BITS_V(value, EERD, ADDR), &tmp))
+ SET_BITS(EERD, DATA, tmp);
+ EERD |= EERD_DONE;
+ STAM_PROFILE_ADV_STOP(&pThis->StatEEPROMRead, a);
+ }
+
+ return VINF_SUCCESS;
+#else /* !IN_RING3 */
+ RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index); RT_NOREF_PV(value);
+ return VINF_IOM_R3_MMIO_WRITE;
+#endif /* !IN_RING3 */
+}
+
+
+/**
+ * Write handler for MDI Control register.
+ *
+ * Handles PHY read/write requests; forwards requests to internal PHY device.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteMDIC(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ if (value & MDIC_INT_EN)
+ {
+ E1kLog(("%s ERROR! Interrupt at the end of an MDI cycle is not supported yet.\n",
+ pThis->szPrf));
+ }
+ else if (value & MDIC_READY)
+ {
+ E1kLog(("%s ERROR! Ready bit is not reset by software during write operation.\n",
+ pThis->szPrf));
+ }
+ else if (GET_BITS_V(value, MDIC, PHY) != 1)
+ {
+ E1kLog(("%s WARNING! Access to invalid PHY detected, phy=%d.\n",
+ pThis->szPrf, GET_BITS_V(value, MDIC, PHY)));
+ /*
+ * Some drivers scan the MDIO bus for a PHY. We can work with these
+ * drivers if we set MDIC_READY and MDIC_ERROR when there isn't a PHY
+ * at the requested address, see @bugref{7346}.
+ */
+ MDIC = MDIC_READY | MDIC_ERROR;
+ }
+ else
+ {
+ /* Store the value */
+ e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+ STAM_COUNTER_INC(&pThis->StatPHYAccesses);
+ /* Forward op to PHY */
+ if (value & MDIC_OP_READ)
+ SET_BITS(MDIC, DATA, Phy::readRegister(&pThis->phy, GET_BITS_V(value, MDIC, REG), pDevIns));
+ else
+ Phy::writeRegister(&pThis->phy, GET_BITS_V(value, MDIC, REG), value & MDIC_DATA_MASK, pDevIns);
+ /* Let software know that we are done */
+ MDIC |= MDIC_READY;
+ }
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for Interrupt Cause Read register.
+ *
+ * Bits corresponding to 1s in 'value' will be cleared in ICR register.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteICR(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ ICR &= ~value;
+
+ RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index);
+ return VINF_SUCCESS;
+}
+
+/**
+ * Read handler for Interrupt Cause Read register.
+ *
+ * Reading this register acknowledges all interrupts.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param mask Not used.
+ * @thread EMT
+ */
+static int e1kRegReadICR(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ e1kCsEnterReturn(pThis, VINF_IOM_R3_MMIO_READ);
+
+ uint32_t value = 0;
+ int rc = e1kRegReadDefault(pDevIns, pThis, offset, index, &value);
+ if (RT_SUCCESS(rc))
+ {
+ if (value)
+ {
+ if (!pThis->fIntRaised)
+ E1K_INC_ISTAT_CNT(pThis->uStatNoIntICR);
+ /*
+ * Not clearing ICR causes QNX to hang as it reads ICR in a loop
+ * with disabled interrupts.
+ */
+ //if (IMS)
+ if (1)
+ {
+ /*
+ * Interrupts were enabled -- we are supposedly at the very
+ * beginning of interrupt handler
+ */
+ E1kLogRel(("E1000: irq lowered, icr=0x%x\n", ICR));
+ E1kLog(("%s e1kRegReadICR: Lowered IRQ (%08x)\n", pThis->szPrf, ICR));
+ /* Clear all pending interrupts */
+ ICR = 0;
+ pThis->fIntRaised = false;
+ /* Lower(0) INTA(0) */
+ PDMDevHlpPCISetIrq(pDevIns, 0, 0);
+
+ pThis->u64AckedAt = PDMDevHlpTimerGet(pDevIns, pThis->hIntTimer);
+ if (pThis->fIntMaskUsed)
+ pThis->fDelayInts = true;
+ }
+ else
+ {
+ /*
+ * Interrupts are disabled -- in windows guests ICR read is done
+ * just before re-enabling interrupts
+ */
+ E1kLog(("%s e1kRegReadICR: Suppressing auto-clear due to disabled interrupts (%08x)\n", pThis->szPrf, ICR));
+ }
+ }
+ *pu32Value = value;
+ }
+ e1kCsLeave(pThis);
+
+ return rc;
+}
+
+/**
+ * Read handler for Interrupt Cause Set register.
+ *
+ * VxWorks driver uses this undocumented feature of real H/W to read ICR without acknowledging interrupts.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param pu32Value Where to store the value of the register.
+ * @thread EMT
+ */
+static int e1kRegReadICS(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ RT_NOREF_PV(index);
+ return e1kRegReadDefault(pDevIns, pThis, offset, ICR_IDX, pu32Value);
+}
+
+/**
+ * Write handler for Interrupt Cause Set register.
+ *
+ * Bits corresponding to 1s in 'value' will be set in ICR register.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteICS(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(offset); RT_NOREF_PV(index);
+ E1K_INC_ISTAT_CNT(pThis->uStatIntICS);
+ return e1kRaiseInterrupt(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE, value & g_aE1kRegMap[ICS_IDX].writable);
+}
+
+/**
+ * Write handler for Interrupt Mask Set register.
+ *
+ * Will trigger pending interrupts.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteIMS(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(offset); RT_NOREF_PV(index);
+
+ IMS |= value;
+ E1kLogRel(("E1000: irq enabled, RDH=%x RDT=%x TDH=%x TDT=%x\n", RDH, RDT, TDH, TDT));
+ E1kLog(("%s e1kRegWriteIMS: IRQ enabled\n", pThis->szPrf));
+ /*
+ * We cannot raise an interrupt here as it will occasionally cause an interrupt storm
+ * in Windows guests (see @bugref{8624}, @bugref{5023}).
+ */
+ if ((ICR & IMS) && !pThis->fLocked)
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatIntIMS);
+ e1kPostponeInterrupt(pDevIns, pThis, E1K_IMS_INT_DELAY_NS);
+ }
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for Interrupt Mask Clear register.
+ *
+ * Bits corresponding to 1s in 'value' will be cleared in IMS register.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteIMC(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(offset); RT_NOREF_PV(index);
+
+ e1kCsEnterReturn(pThis, VINF_IOM_R3_MMIO_WRITE);
+ if (pThis->fIntRaised)
+ {
+ /*
+ * Technically we should reset fIntRaised in ICR read handler, but it will cause
+ * Windows to freeze since it may receive an interrupt while still in the very beginning
+ * of interrupt handler.
+ */
+ E1K_INC_ISTAT_CNT(pThis->uStatIntLower);
+ STAM_COUNTER_INC(&pThis->StatIntsPrevented);
+ E1kLogRel(("E1000: irq lowered (IMC), icr=0x%x\n", ICR));
+ /* Lower(0) INTA(0) */
+ PDMDevHlpPCISetIrq(pDevIns, 0, 0);
+ pThis->fIntRaised = false;
+ E1kLog(("%s e1kRegWriteIMC: Lowered IRQ: ICR=%08x\n", pThis->szPrf, ICR));
+ }
+ IMS &= ~value;
+ E1kLog(("%s e1kRegWriteIMC: IRQ disabled\n", pThis->szPrf));
+ e1kCsLeave(pThis);
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for Receive Control register.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteRCTL(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ /* Update promiscuous mode */
+ bool fBecomePromiscous = !!(value & (RCTL_UPE | RCTL_MPE));
+ if (fBecomePromiscous != !!( RCTL & (RCTL_UPE | RCTL_MPE)))
+ {
+ /* Promiscuity has changed, pass the knowledge on. */
+#ifndef IN_RING3
+ return VINF_IOM_R3_MMIO_WRITE;
+#else
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ if (pThisCC->pDrvR3)
+ pThisCC->pDrvR3->pfnSetPromiscuousMode(pThisCC->pDrvR3, fBecomePromiscous);
+#endif
+ }
+
+ /* Adjust receive buffer size */
+ unsigned cbRxBuf = 2048 >> GET_BITS_V(value, RCTL, BSIZE);
+ if (value & RCTL_BSEX)
+ cbRxBuf *= 16;
+ if (cbRxBuf > E1K_MAX_RX_PKT_SIZE)
+ cbRxBuf = E1K_MAX_RX_PKT_SIZE;
+ if (cbRxBuf != pThis->u16RxBSize)
+ E1kLog2(("%s e1kRegWriteRCTL: Setting receive buffer size to %d (old %d)\n",
+ pThis->szPrf, cbRxBuf, pThis->u16RxBSize));
+ Assert(cbRxBuf < 65536);
+ pThis->u16RxBSize = (uint16_t)cbRxBuf;
+
+ /* Update the register */
+ return e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+}
+
+/**
+ * Write handler for Packet Buffer Allocation register.
+ *
+ * TXA = 64 - RXA.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWritePBA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+ PBA_st->txa = 64 - PBA_st->rxa;
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for Receive Descriptor Tail register.
+ *
+ * @remarks Write into RDT forces switch to HC and signal to
+ * e1kR3NetworkDown_WaitReceiveAvail().
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteRDT(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+#ifndef IN_RING3
+ /* XXX */
+// return VINF_IOM_R3_MMIO_WRITE;
+#endif
+ int rc = e1kCsRxEnter(pThis, VINF_IOM_R3_MMIO_WRITE);
+ if (RT_LIKELY(rc == VINF_SUCCESS))
+ {
+ E1kLog(("%s e1kRegWriteRDT\n", pThis->szPrf));
+#ifndef E1K_WITH_RXD_CACHE
+ /*
+ * Some drivers advance RDT too far, so that it equals RDH. This
+ * somehow manages to work with real hardware but not with this
+ * emulated device. We can work with these drivers if we just
+ * write 1 less when we see a driver writing RDT equal to RDH,
+ * see @bugref{7346}.
+ */
+ if (value == RDH)
+ {
+ if (RDH == 0)
+ value = (RDLEN / sizeof(E1KRXDESC)) - 1;
+ else
+ value = RDH - 1;
+ }
+#endif /* !E1K_WITH_RXD_CACHE */
+ rc = e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+#ifdef E1K_WITH_RXD_CACHE
+ E1KRXDC rxdc;
+ if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kRegWriteRDT")))
+ {
+ e1kCsRxLeave(pThis);
+ E1kLog(("%s e1kRegWriteRDT: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
+ return VINF_SUCCESS;
+ }
+ /*
+ * We need to fetch descriptors now as RDT may go whole circle
+ * before we attempt to store a received packet. For example,
+ * Intel's DOS drivers use 2 (!) RX descriptors with the total ring
+ * size being only 8 descriptors! Note that we fetch descriptors
+ * only when the cache is empty to reduce the number of memory reads
+ * in case of frequent RDT writes. Don't fetch anything when the
+ * receiver is disabled either as RDH, RDT, RDLEN can be in some
+ * messed up state.
+ * Note that despite the cache may seem empty, meaning that there are
+ * no more available descriptors in it, it may still be used by RX
+ * thread which has not yet written the last descriptor back but has
+ * temporarily released the RX lock in order to write the packet body
+ * to descriptor's buffer. At this point we still going to do prefetch
+ * but it won't actually fetch anything if there are no unused slots in
+ * our "empty" cache (nRxDFetched==E1K_RXD_CACHE_SIZE). We must not
+ * reset the cache here even if it appears empty. It will be reset at
+ * a later point in e1kRxDGet().
+ */
+ if (e1kRxDIsCacheEmpty(pThis) && (RCTL & RCTL_EN))
+ e1kRxDPrefetch(pDevIns, pThis, &rxdc);
+#endif /* E1K_WITH_RXD_CACHE */
+ e1kCsRxLeave(pThis);
+ if (RT_SUCCESS(rc))
+ {
+ /* Signal that we have more receive descriptors available. */
+ e1kWakeupReceive(pDevIns, pThis);
+ }
+ }
+ return rc;
+}
+
+/**
+ * Write handler for Receive Delay Timer register.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteRDTR(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+ if (value & RDTR_FPD)
+ {
+ /* Flush requested, cancel both timers and raise interrupt */
+#ifdef E1K_USE_RX_TIMERS
+ e1kCancelTimer(pDevIns, pThis, pThis->hRIDTimer);
+ e1kCancelTimer(pDevIns, pThis, pThis->hRADTimer);
+#endif
+ E1K_INC_ISTAT_CNT(pThis->uStatIntRDTR);
+ return e1kRaiseInterrupt(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE, ICR_RXT0);
+ }
+
+ return VINF_SUCCESS;
+}
+
+DECLINLINE(uint32_t) e1kGetTxLen(PE1KTXDC pTxdc)
+{
+ /**
+ * Make sure TDT won't change during computation. EMT may modify TDT at
+ * any moment.
+ */
+ uint32_t tdt = pTxdc->tdt;
+ return (pTxdc->tdh > tdt ? pTxdc->tdlen/sizeof(E1KTXDESC) : 0) + tdt - pTxdc->tdh;
+}
+
+#ifdef IN_RING3
+
+# ifdef E1K_TX_DELAY
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Transmit Delay Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3TxDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ Assert(PDMDevHlpCritSectIsOwner(pDevIns, &pThis->csTx));
+ RT_NOREF(hTimer);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatTxDelayExp);
+# ifdef E1K_INT_STATS
+ uint64_t u64Elapsed = RTTimeNanoTS() - pThis->u64ArmedAt;
+ if (u64Elapsed > pThis->uStatMaxTxDelay)
+ pThis->uStatMaxTxDelay = u64Elapsed;
+# endif
+ int rc = e1kXmitPending(pDevIns, pThis, false /*fOnWorkerThread*/);
+ AssertMsg(RT_SUCCESS(rc) || rc == VERR_TRY_AGAIN, ("%Rrc\n", rc));
+}
+# endif /* E1K_TX_DELAY */
+
+//# ifdef E1K_USE_TX_TIMERS
+
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Transmit Interrupt Delay Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3TxIntDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ Assert(hTimer == pThis->hTIDTimer); RT_NOREF(hTimer);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatTID);
+ /* Cancel absolute delay timer as we have already got attention */
+# ifndef E1K_NO_TAD
+ e1kCancelTimer(pDevIns, pThis, pThis->hTADTimer);
+# endif
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_TXDW);
+}
+
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Transmit Absolute Delay Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3TxAbsDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ Assert(hTimer == pThis->hTADTimer); RT_NOREF(hTimer);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatTAD);
+ /* Cancel interrupt delay timer as we have already got attention */
+ e1kCancelTimer(pDevIns, pThis, pThis->hTIDTimer);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_TXDW);
+}
+
+//# endif /* E1K_USE_TX_TIMERS */
+# ifdef E1K_USE_RX_TIMERS
+
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Receive Interrupt Delay Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3RxIntDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ Assert(hTimer == pThis->hRIDTimer); RT_NOREF(hTimer);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatRID);
+ /* Cancel absolute delay timer as we have already got attention */
+ e1kCancelTimer(pDevIns, pThis, pThis->hRADTimer);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_RXT0);
+}
+
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Receive Absolute Delay Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3RxAbsDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ Assert(hTimer == pThis->hRADTimer); RT_NOREF(hTimer);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatRAD);
+ /* Cancel interrupt delay timer as we have already got attention */
+ e1kCancelTimer(pDevIns, pThis, pThis->hRIDTimer);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_RXT0);
+}
+
+# endif /* E1K_USE_RX_TIMERS */
+
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Late Interrupt Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3LateIntTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ Assert(hTimer == pThis->hIntTimer); RT_NOREF(hTimer);
+ RT_NOREF(hTimer);
+
+ STAM_PROFILE_ADV_START(&pThis->StatLateIntTimer, a);
+ STAM_COUNTER_INC(&pThis->StatLateInts);
+ E1K_INC_ISTAT_CNT(pThis->uStatIntLate);
+# if 0
+ if (pThis->iStatIntLost > -100)
+ pThis->iStatIntLost--;
+# endif
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, 0);
+ STAM_PROFILE_ADV_STOP(&pThis->StatLateIntTimer, a);
+}
+
+/**
+ * @callback_method_impl{FNTMTIMERDEV, Link Up Timer handler.}
+ */
+static DECLCALLBACK(void) e1kR3LinkUpTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
+{
+ PE1KSTATE pThis = (PE1KSTATE)pvUser;
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ Assert(hTimer == pThis->hLUTimer); RT_NOREF(hTimer);
+
+ /*
+ * This can happen if we set the link status to down when the Link up timer was
+ * already armed (shortly after e1kR3LoadDone() or when the cable was disconnected
+ * and connect+disconnect the cable very quick. Moreover, 82543GC triggers LSC
+ * on reset even if the cable is unplugged (see @bugref{8942}).
+ */
+ if (e1kIsConnected(pDevIns))
+ {
+ /* 82543GC does not have an internal PHY */
+ if (pThis->eChip == E1K_CHIP_82543GC || (CTRL & CTRL_SLU))
+ e1kR3LinkUp(pDevIns, pThis, pThisCC);
+ }
+# ifdef E1K_LSC_ON_RESET
+ else if (pThis->eChip == E1K_CHIP_82543GC)
+ e1kR3LinkDown(pDevIns, pThis, pThisCC);
+# endif /* E1K_LSC_ON_RESET */
+}
+
+#endif /* IN_RING3 */
+
+/**
+ * Sets up the GSO context according to the TSE new context descriptor.
+ *
+ * @param pGso The GSO context to setup.
+ * @param pCtx The context descriptor.
+ */
+DECLINLINE(bool) e1kSetupGsoCtx(PPDMNETWORKGSO pGso, E1KTXCTX const *pCtx)
+{
+ pGso->u8Type = PDMNETWORKGSOTYPE_INVALID;
+
+ /*
+ * See if the context descriptor describes something that could be TCP or
+ * UDP over IPv[46].
+ */
+ /* Check the header ordering and spacing: 1. Ethernet, 2. IP, 3. TCP/UDP. */
+ if (RT_UNLIKELY( pCtx->ip.u8CSS < sizeof(RTNETETHERHDR) ))
+ {
+ E1kLog(("e1kSetupGsoCtx: IPCSS=%#x\n", pCtx->ip.u8CSS));
+ return false;
+ }
+ if (RT_UNLIKELY( pCtx->tu.u8CSS < (size_t)pCtx->ip.u8CSS + (pCtx->dw2.fIP ? RTNETIPV4_MIN_LEN : RTNETIPV6_MIN_LEN) ))
+ {
+ E1kLog(("e1kSetupGsoCtx: TUCSS=%#x\n", pCtx->tu.u8CSS));
+ return false;
+ }
+ if (RT_UNLIKELY( pCtx->dw2.fTCP
+ ? pCtx->dw3.u8HDRLEN < (size_t)pCtx->tu.u8CSS + RTNETTCP_MIN_LEN
+ : pCtx->dw3.u8HDRLEN != (size_t)pCtx->tu.u8CSS + RTNETUDP_MIN_LEN ))
+ {
+ E1kLog(("e1kSetupGsoCtx: HDRLEN=%#x TCP=%d\n", pCtx->dw3.u8HDRLEN, pCtx->dw2.fTCP));
+ return false;
+ }
+
+ /* The end of the TCP/UDP checksum should stop at the end of the packet or at least after the headers. */
+ if (RT_UNLIKELY( pCtx->tu.u16CSE > 0 && pCtx->tu.u16CSE <= pCtx->dw3.u8HDRLEN ))
+ {
+ E1kLog(("e1kSetupGsoCtx: TUCSE=%#x HDRLEN=%#x\n", pCtx->tu.u16CSE, pCtx->dw3.u8HDRLEN));
+ return false;
+ }
+
+ /* IPv4 checksum offset. */
+ if (RT_UNLIKELY( pCtx->dw2.fIP && (size_t)pCtx->ip.u8CSO - pCtx->ip.u8CSS != RT_UOFFSETOF(RTNETIPV4, ip_sum) ))
+ {
+ E1kLog(("e1kSetupGsoCtx: IPCSO=%#x IPCSS=%#x\n", pCtx->ip.u8CSO, pCtx->ip.u8CSS));
+ return false;
+ }
+
+ /* TCP/UDP checksum offsets. */
+ if (RT_UNLIKELY( (size_t)pCtx->tu.u8CSO - pCtx->tu.u8CSS
+ != ( pCtx->dw2.fTCP
+ ? RT_UOFFSETOF(RTNETTCP, th_sum)
+ : RT_UOFFSETOF(RTNETUDP, uh_sum) ) ))
+ {
+ E1kLog(("e1kSetupGsoCtx: TUCSO=%#x TUCSS=%#x TCP=%d\n", pCtx->ip.u8CSO, pCtx->ip.u8CSS, pCtx->dw2.fTCP));
+ return false;
+ }
+
+ /*
+ * Because of internal networking using a 16-bit size field for GSO context
+ * plus frame, we have to make sure we don't exceed this.
+ */
+ if (RT_UNLIKELY( pCtx->dw3.u8HDRLEN + pCtx->dw2.u20PAYLEN > VBOX_MAX_GSO_SIZE ))
+ {
+ E1kLog(("e1kSetupGsoCtx: HDRLEN(=%#x) + PAYLEN(=%#x) = %#x, max is %#x\n",
+ pCtx->dw3.u8HDRLEN, pCtx->dw2.u20PAYLEN, pCtx->dw3.u8HDRLEN + pCtx->dw2.u20PAYLEN, VBOX_MAX_GSO_SIZE));
+ return false;
+ }
+
+ /*
+ * We're good for now - we'll do more checks when seeing the data.
+ * So, figure the type of offloading and setup the context.
+ */
+ if (pCtx->dw2.fIP)
+ {
+ if (pCtx->dw2.fTCP)
+ {
+ pGso->u8Type = PDMNETWORKGSOTYPE_IPV4_TCP;
+ pGso->cbHdrsSeg = pCtx->dw3.u8HDRLEN;
+ }
+ else
+ {
+ pGso->u8Type = PDMNETWORKGSOTYPE_IPV4_UDP;
+ pGso->cbHdrsSeg = pCtx->tu.u8CSS; /* IP header only */
+ }
+ /** @todo Detect IPv4-IPv6 tunneling (need test setup since linux doesn't do
+ * this yet it seems)... */
+ }
+ else
+ {
+ pGso->cbHdrsSeg = pCtx->dw3.u8HDRLEN; /** @todo IPv6 UFO */
+ if (pCtx->dw2.fTCP)
+ pGso->u8Type = PDMNETWORKGSOTYPE_IPV6_TCP;
+ else
+ pGso->u8Type = PDMNETWORKGSOTYPE_IPV6_UDP;
+ }
+ pGso->offHdr1 = pCtx->ip.u8CSS;
+ pGso->offHdr2 = pCtx->tu.u8CSS;
+ pGso->cbHdrsTotal = pCtx->dw3.u8HDRLEN;
+ pGso->cbMaxSeg = pCtx->dw3.u16MSS + (pGso->u8Type == PDMNETWORKGSOTYPE_IPV4_UDP ? pGso->offHdr2 : 0);
+ Assert(PDMNetGsoIsValid(pGso, sizeof(*pGso), pGso->cbMaxSeg * 5));
+ E1kLog2(("e1kSetupGsoCtx: mss=%#x hdr=%#x hdrseg=%#x hdr1=%#x hdr2=%#x %s\n",
+ pGso->cbMaxSeg, pGso->cbHdrsTotal, pGso->cbHdrsSeg, pGso->offHdr1, pGso->offHdr2, PDMNetGsoTypeName((PDMNETWORKGSOTYPE)pGso->u8Type) ));
+ return PDMNetGsoIsValid(pGso, sizeof(*pGso), pGso->cbMaxSeg * 5);
+}
+
+/**
+ * Checks if we can use GSO processing for the current TSE frame.
+ *
+ * @param pThis The device state structure.
+ * @param pGso The GSO context.
+ * @param pData The first data descriptor of the frame.
+ * @param pCtx The TSO context descriptor.
+ */
+DECLINLINE(bool) e1kCanDoGso(PE1KSTATE pThis, PCPDMNETWORKGSO pGso, E1KTXDAT const *pData, E1KTXCTX const *pCtx)
+{
+ if (!pData->cmd.fTSE)
+ {
+ E1kLog2(("e1kCanDoGso: !TSE\n"));
+ return false;
+ }
+ if (pData->cmd.fVLE) /** @todo VLAN tagging. */
+ {
+ E1kLog(("e1kCanDoGso: VLE\n"));
+ return false;
+ }
+ if (RT_UNLIKELY(!pThis->fGSOEnabled))
+ {
+ E1kLog3(("e1kCanDoGso: GSO disabled via CFGM\n"));
+ return false;
+ }
+
+ switch ((PDMNETWORKGSOTYPE)pGso->u8Type)
+ {
+ case PDMNETWORKGSOTYPE_IPV4_TCP:
+ case PDMNETWORKGSOTYPE_IPV4_UDP:
+ if (!pData->dw3.fIXSM)
+ {
+ E1kLog(("e1kCanDoGso: !IXSM (IPv4)\n"));
+ return false;
+ }
+ if (!pData->dw3.fTXSM)
+ {
+ E1kLog(("e1kCanDoGso: !TXSM (IPv4)\n"));
+ return false;
+ }
+ /** @todo what more check should we perform here? Ethernet frame type? */
+ E1kLog2(("e1kCanDoGso: OK, IPv4\n"));
+ return true;
+
+ case PDMNETWORKGSOTYPE_IPV6_TCP:
+ case PDMNETWORKGSOTYPE_IPV6_UDP:
+ if (pData->dw3.fIXSM && pCtx->ip.u8CSO)
+ {
+ E1kLog(("e1kCanDoGso: IXSM (IPv6)\n"));
+ return false;
+ }
+ if (!pData->dw3.fTXSM)
+ {
+ E1kLog(("e1kCanDoGso: TXSM (IPv6)\n"));
+ return false;
+ }
+ /** @todo what more check should we perform here? Ethernet frame type? */
+ E1kLog2(("e1kCanDoGso: OK, IPv4\n"));
+ return true;
+
+ default:
+ Assert(pGso->u8Type == PDMNETWORKGSOTYPE_INVALID);
+ E1kLog2(("e1kCanDoGso: e1kSetupGsoCtx failed\n"));
+ return false;
+ }
+}
+
+/**
+ * Frees the current xmit buffer.
+ *
+ * @param pThis The device state structure.
+ */
+static void e1kXmitFreeBuf(PE1KSTATE pThis, PE1KSTATECC pThisCC)
+{
+ PPDMSCATTERGATHER pSg = pThisCC->CTX_SUFF(pTxSg);
+ if (pSg)
+ {
+ pThisCC->CTX_SUFF(pTxSg) = NULL;
+
+ if (pSg->pvAllocator != pThis)
+ {
+ PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
+ if (pDrv)
+ pDrv->pfnFreeBuf(pDrv, pSg);
+ }
+ else
+ {
+ /* loopback */
+ AssertCompileMemberSize(E1KSTATE, uTxFallback.Sg, 8 * sizeof(size_t));
+ Assert(pSg->fFlags == (PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_3));
+ pSg->fFlags = 0;
+ pSg->pvAllocator = NULL;
+ }
+ }
+}
+
+#ifndef E1K_WITH_TXD_CACHE
+/**
+ * Allocates an xmit buffer.
+ *
+ * @returns See PDMINETWORKUP::pfnAllocBuf.
+ * @param pThis The device state structure.
+ * @param cbMin The minimum frame size.
+ * @param fExactSize Whether cbMin is exact or if we have to max it
+ * out to the max MTU size.
+ * @param fGso Whether this is a GSO frame or not.
+ */
+DECLINLINE(int) e1kXmitAllocBuf(PE1KSTATE pThis, PE1KSTATECC pThisCC, size_t cbMin, bool fExactSize, bool fGso)
+{
+ /* Adjust cbMin if necessary. */
+ if (!fExactSize)
+ cbMin = RT_MAX(cbMin, E1K_MAX_TX_PKT_SIZE);
+
+ /* Deal with existing buffer (descriptor screw up, reset, etc). */
+ if (RT_UNLIKELY(pThisCC->CTX_SUFF(pTxSg)))
+ e1kXmitFreeBuf(pThis, pThisCC);
+ Assert(pThisCC->CTX_SUFF(pTxSg) == NULL);
+
+ /*
+ * Allocate the buffer.
+ */
+ PPDMSCATTERGATHER pSg;
+ if (RT_LIKELY(GET_BITS(RCTL, LBM) != RCTL_LBM_TCVR))
+ {
+ PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
+ if (RT_UNLIKELY(!pDrv))
+ return VERR_NET_DOWN;
+ int rc = pDrv->pfnAllocBuf(pDrv, cbMin, fGso ? &pThis->GsoCtx : NULL, &pSg);
+ if (RT_FAILURE(rc))
+ {
+ /* Suspend TX as we are out of buffers atm */
+ STATUS |= STATUS_TXOFF;
+ return rc;
+ }
+ }
+ else
+ {
+ /* Create a loopback using the fallback buffer and preallocated SG. */
+ AssertCompileMemberSize(E1KSTATE, uTxFallback.Sg, 8 * sizeof(size_t));
+ pSg = &pThis->uTxFallback.Sg;
+ pSg->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_3;
+ pSg->cbUsed = 0;
+ pSg->cbAvailable = 0;
+ pSg->pvAllocator = pThis;
+ pSg->pvUser = NULL; /* No GSO here. */
+ pSg->cSegs = 1;
+ pSg->aSegs[0].pvSeg = pThis->aTxPacketFallback;
+ pSg->aSegs[0].cbSeg = sizeof(pThis->aTxPacketFallback);
+ }
+
+ pThisCC->CTX_SUFF(pTxSg) = pSg;
+ return VINF_SUCCESS;
+}
+#else /* E1K_WITH_TXD_CACHE */
+/**
+ * Allocates an xmit buffer.
+ *
+ * @returns See PDMINETWORKUP::pfnAllocBuf.
+ * @param pThis The device state structure.
+ * @param cbMin The minimum frame size.
+ * @param fExactSize Whether cbMin is exact or if we have to max it
+ * out to the max MTU size.
+ * @param fGso Whether this is a GSO frame or not.
+ */
+DECLINLINE(int) e1kXmitAllocBuf(PE1KSTATE pThis, PE1KSTATECC pThisCC, bool fGso)
+{
+ /* Deal with existing buffer (descriptor screw up, reset, etc). */
+ if (RT_UNLIKELY(pThisCC->CTX_SUFF(pTxSg)))
+ e1kXmitFreeBuf(pThis, pThisCC);
+ Assert(pThisCC->CTX_SUFF(pTxSg) == NULL);
+
+ /*
+ * Allocate the buffer.
+ */
+ PPDMSCATTERGATHER pSg;
+ if (RT_LIKELY(GET_BITS(RCTL, LBM) != RCTL_LBM_TCVR))
+ {
+ if (pThis->cbTxAlloc == 0)
+ {
+ /* Zero packet, no need for the buffer */
+ return VINF_SUCCESS;
+ }
+ if (fGso && pThis->GsoCtx.u8Type == PDMNETWORKGSOTYPE_INVALID)
+ {
+ E1kLog3(("Invalid GSO context, won't allocate this packet, cb=%u %s%s\n",
+ pThis->cbTxAlloc, pThis->fVTag ? "VLAN " : "", pThis->fGSO ? "GSO " : ""));
+ /* No valid GSO context is available, ignore this packet. */
+ pThis->cbTxAlloc = 0;
+ return VINF_SUCCESS;
+ }
+
+ PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
+ if (RT_UNLIKELY(!pDrv))
+ return VERR_NET_DOWN;
+ int rc = pDrv->pfnAllocBuf(pDrv, pThis->cbTxAlloc, fGso ? &pThis->GsoCtx : NULL, &pSg);
+ if (RT_FAILURE(rc))
+ {
+ /* Suspend TX as we are out of buffers atm */
+ STATUS |= STATUS_TXOFF;
+ return rc;
+ }
+ E1kLog3(("%s Allocated buffer for TX packet: cb=%u %s%s\n",
+ pThis->szPrf, pThis->cbTxAlloc,
+ pThis->fVTag ? "VLAN " : "",
+ pThis->fGSO ? "GSO " : ""));
+ }
+ else
+ {
+ /* Create a loopback using the fallback buffer and preallocated SG. */
+ AssertCompileMemberSize(E1KSTATE, uTxFallback.Sg, 8 * sizeof(size_t));
+ pSg = &pThis->uTxFallback.Sg;
+ pSg->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_3;
+ pSg->cbUsed = 0;
+ pSg->cbAvailable = sizeof(pThis->aTxPacketFallback);
+ pSg->pvAllocator = pThis;
+ pSg->pvUser = NULL; /* No GSO here. */
+ pSg->cSegs = 1;
+ pSg->aSegs[0].pvSeg = pThis->aTxPacketFallback;
+ pSg->aSegs[0].cbSeg = sizeof(pThis->aTxPacketFallback);
+ }
+ pThis->cbTxAlloc = 0;
+
+ pThisCC->CTX_SUFF(pTxSg) = pSg;
+ return VINF_SUCCESS;
+}
+#endif /* E1K_WITH_TXD_CACHE */
+
+/**
+ * Checks if it's a GSO buffer or not.
+ *
+ * @returns true / false.
+ * @param pTxSg The scatter / gather buffer.
+ */
+DECLINLINE(bool) e1kXmitIsGsoBuf(PDMSCATTERGATHER const *pTxSg)
+{
+#if 0
+ if (!pTxSg)
+ E1kLog(("e1kXmitIsGsoBuf: pTxSG is NULL\n"));
+ if (pTxSg && pTxSg->pvUser)
+ E1kLog(("e1kXmitIsGsoBuf: pvUser is NULL\n"));
+#endif
+ return pTxSg && pTxSg->pvUser /* GSO indicator */;
+}
+
+#ifndef E1K_WITH_TXD_CACHE
+/**
+ * Load transmit descriptor from guest memory.
+ *
+ * @param pDevIns The device instance.
+ * @param pDesc Pointer to descriptor union.
+ * @param addr Physical address in guest context.
+ * @thread E1000_TX
+ */
+DECLINLINE(void) e1kLoadDesc(PPDMDEVINS pDevIns, E1KTXDESC *pDesc, RTGCPHYS addr)
+{
+ PDMDevHlpPCIPhysRead(pDevIns, addr, pDesc, sizeof(E1KTXDESC));
+}
+#else /* E1K_WITH_TXD_CACHE */
+/**
+ * Load transmit descriptors from guest memory.
+ *
+ * We need two physical reads in case the tail wrapped around the end of TX
+ * descriptor ring.
+ *
+ * @returns the actual number of descriptors fetched.
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @thread E1000_TX
+ */
+DECLINLINE(unsigned) e1kTxDLoadMore(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pTxdc)
+{
+ Assert(pThis->iTxDCurrent == 0);
+ /* We've already loaded pThis->nTxDFetched descriptors past TDH. */
+ unsigned nDescsAvailable = e1kGetTxLen(pTxdc) - pThis->nTxDFetched;
+ /* The following two lines ensure that pThis->nTxDFetched never overflows. */
+ AssertCompile(E1K_TXD_CACHE_SIZE < (256 * sizeof(pThis->nTxDFetched)));
+ unsigned nDescsToFetch = RT_MIN(nDescsAvailable, E1K_TXD_CACHE_SIZE - pThis->nTxDFetched);
+ unsigned nDescsTotal = pTxdc->tdlen / sizeof(E1KTXDESC);
+ Assert(nDescsTotal != 0);
+ if (nDescsTotal == 0)
+ return 0;
+ unsigned nFirstNotLoaded = (pTxdc->tdh + pThis->nTxDFetched) % nDescsTotal;
+ unsigned nDescsInSingleRead = RT_MIN(nDescsToFetch, nDescsTotal - nFirstNotLoaded);
+ E1kLog3(("%s e1kTxDLoadMore: nDescsAvailable=%u nDescsToFetch=%u nDescsTotal=%u nFirstNotLoaded=0x%x nDescsInSingleRead=%u\n",
+ pThis->szPrf, nDescsAvailable, nDescsToFetch, nDescsTotal,
+ nFirstNotLoaded, nDescsInSingleRead));
+ if (nDescsToFetch == 0)
+ return 0;
+ E1KTXDESC* pFirstEmptyDesc = &pThis->aTxDescriptors[pThis->nTxDFetched];
+ PDMDevHlpPCIPhysRead(pDevIns,
+ ((uint64_t)TDBAH << 32) + TDBAL + nFirstNotLoaded * sizeof(E1KTXDESC),
+ pFirstEmptyDesc, nDescsInSingleRead * sizeof(E1KTXDESC));
+ E1kLog3(("%s Fetched %u TX descriptors at %08x%08x(0x%x), TDLEN=%08x, TDH=%08x, TDT=%08x\n",
+ pThis->szPrf, nDescsInSingleRead,
+ TDBAH, TDBAL + pTxdc->tdh * sizeof(E1KTXDESC),
+ nFirstNotLoaded, pTxdc->tdlen, pTxdc->tdh, pTxdc->tdt));
+ if (nDescsToFetch > nDescsInSingleRead)
+ {
+ PDMDevHlpPCIPhysRead(pDevIns,
+ ((uint64_t)TDBAH << 32) + TDBAL,
+ pFirstEmptyDesc + nDescsInSingleRead,
+ (nDescsToFetch - nDescsInSingleRead) * sizeof(E1KTXDESC));
+ E1kLog3(("%s Fetched %u TX descriptors at %08x%08x\n",
+ pThis->szPrf, nDescsToFetch - nDescsInSingleRead,
+ TDBAH, TDBAL));
+ }
+ pThis->nTxDFetched += (uint8_t)nDescsToFetch;
+ return nDescsToFetch;
+}
+
+/**
+ * Load transmit descriptors from guest memory only if there are no loaded
+ * descriptors.
+ *
+ * @returns true if there are descriptors in cache.
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @thread E1000_TX
+ */
+DECLINLINE(bool) e1kTxDLazyLoad(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pTxdc)
+{
+ if (pThis->nTxDFetched == 0)
+ return e1kTxDLoadMore(pDevIns, pThis, pTxdc) != 0;
+ return true;
+}
+#endif /* E1K_WITH_TXD_CACHE */
+
+/**
+ * Write back transmit descriptor to guest memory.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to descriptor union.
+ * @param addr Physical address in guest context.
+ * @thread E1000_TX
+ */
+DECLINLINE(void) e1kWriteBackDesc(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KTXDESC *pDesc, RTGCPHYS addr)
+{
+ /* Only the last half of the descriptor has to be written back. */
+ e1kPrintTDesc(pThis, pDesc, "^^^");
+ PDMDevHlpPCIPhysWrite(pDevIns, addr, pDesc, sizeof(E1KTXDESC));
+}
+
+/**
+ * Transmit complete frame.
+ *
+ * @remarks We skip the FCS since we're not responsible for sending anything to
+ * a real ethernet wire.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
+ * @thread E1000_TX
+ */
+static void e1kTransmitFrame(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, bool fOnWorkerThread)
+{
+ PPDMSCATTERGATHER pSg = pThisCC->CTX_SUFF(pTxSg);
+ uint32_t cbFrame = pSg ? (uint32_t)pSg->cbUsed : 0;
+ Assert(!pSg || pSg->cSegs == 1);
+
+ if (cbFrame < 14)
+ {
+ Log(("%s Ignoring invalid frame (%u bytes)\n", pThis->szPrf, cbFrame));
+ return;
+ }
+ if (cbFrame > 70) /* unqualified guess */
+ pThis->led.Asserted.s.fWriting = pThis->led.Actual.s.fWriting = 1;
+
+#ifdef E1K_INT_STATS
+ if (cbFrame <= 1514)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx1514);
+ else if (cbFrame <= 2962)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx2962);
+ else if (cbFrame <= 4410)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx4410);
+ else if (cbFrame <= 5858)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx5858);
+ else if (cbFrame <= 7306)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx7306);
+ else if (cbFrame <= 8754)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx8754);
+ else if (cbFrame <= 16384)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx16384);
+ else if (cbFrame <= 32768)
+ E1K_INC_ISTAT_CNT(pThis->uStatTx32768);
+ else
+ E1K_INC_ISTAT_CNT(pThis->uStatTxLarge);
+#endif /* E1K_INT_STATS */
+
+ /* Add VLAN tag */
+ if (cbFrame > 12 && pThis->fVTag && pSg->cbUsed + 4 <= pSg->cbAvailable)
+ {
+ E1kLog3(("%s Inserting VLAN tag %08x\n",
+ pThis->szPrf, RT_BE2H_U16((uint16_t)VET) | (RT_BE2H_U16(pThis->u16VTagTCI) << 16)));
+ memmove((uint8_t*)pSg->aSegs[0].pvSeg + 16, (uint8_t*)pSg->aSegs[0].pvSeg + 12, cbFrame - 12);
+ *((uint32_t*)pSg->aSegs[0].pvSeg + 3) = RT_BE2H_U16((uint16_t)VET) | (RT_BE2H_U16(pThis->u16VTagTCI) << 16);
+ pSg->cbUsed += 4;
+ cbFrame += 4;
+ Assert(pSg->cbUsed == cbFrame);
+ Assert(pSg->cbUsed <= pSg->cbAvailable);
+ }
+/* E1kLog2(("%s < < < Outgoing packet. Dump follows: > > >\n"
+ "%.*Rhxd\n"
+ "%s < < < < < < < < < < < < < End of dump > > > > > > > > > > > >\n",
+ pThis->szPrf, cbFrame, pSg->aSegs[0].pvSeg, pThis->szPrf));*/
+
+ /* Update the stats */
+ E1K_INC_CNT32(TPT);
+ E1K_ADD_CNT64(TOTL, TOTH, cbFrame);
+ E1K_INC_CNT32(GPTC);
+ if (pSg && e1kIsBroadcast(pSg->aSegs[0].pvSeg))
+ E1K_INC_CNT32(BPTC);
+ else if (pSg && e1kIsMulticast(pSg->aSegs[0].pvSeg))
+ E1K_INC_CNT32(MPTC);
+ /* Update octet transmit counter */
+ E1K_ADD_CNT64(GOTCL, GOTCH, cbFrame);
+ if (pThisCC->CTX_SUFF(pDrv))
+ STAM_REL_COUNTER_ADD(&pThis->StatTransmitBytes, cbFrame);
+ if (cbFrame == 64)
+ E1K_INC_CNT32(PTC64);
+ else if (cbFrame < 128)
+ E1K_INC_CNT32(PTC127);
+ else if (cbFrame < 256)
+ E1K_INC_CNT32(PTC255);
+ else if (cbFrame < 512)
+ E1K_INC_CNT32(PTC511);
+ else if (cbFrame < 1024)
+ E1K_INC_CNT32(PTC1023);
+ else
+ E1K_INC_CNT32(PTC1522);
+
+ E1K_INC_ISTAT_CNT(pThis->uStatTxFrm);
+
+ /*
+ * Dump and send the packet.
+ */
+ int rc = VERR_NET_DOWN;
+ if (pSg && pSg->pvAllocator != pThis)
+ {
+ e1kPacketDump(pDevIns, pThis, (uint8_t const *)pSg->aSegs[0].pvSeg, cbFrame, "--> Outgoing");
+
+ pThisCC->CTX_SUFF(pTxSg) = NULL;
+ PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
+ if (pDrv)
+ {
+ /* Release critical section to avoid deadlock in CanReceive */
+ //e1kCsLeave(pThis);
+ STAM_PROFILE_START(&pThis->CTX_SUFF_Z(StatTransmitSend), a);
+ rc = pDrv->pfnSendBuf(pDrv, pSg, fOnWorkerThread);
+ STAM_PROFILE_STOP(&pThis->CTX_SUFF_Z(StatTransmitSend), a);
+ //e1kR3CsEnterAsserted(pThis);
+ }
+ }
+ else if (pSg)
+ {
+ Assert(pSg->aSegs[0].pvSeg == pThis->aTxPacketFallback);
+ e1kPacketDump(pDevIns, pThis, (uint8_t const *)pSg->aSegs[0].pvSeg, cbFrame, "--> Loopback");
+
+ /** @todo do we actually need to check that we're in loopback mode here? */
+ if (GET_BITS(RCTL, LBM) == RCTL_LBM_TCVR)
+ {
+ E1KRXDST status;
+ RT_ZERO(status);
+ status.fPIF = true;
+ e1kHandleRxPacket(pDevIns, pThis, pSg->aSegs[0].pvSeg, cbFrame, status);
+ rc = VINF_SUCCESS;
+ }
+ e1kXmitFreeBuf(pThis, pThisCC);
+ }
+ else
+ rc = VERR_NET_DOWN;
+ if (RT_FAILURE(rc))
+ {
+ E1kLogRel(("E1000: ERROR! pfnSend returned %Rrc\n", rc));
+ /** @todo handle VERR_NET_DOWN and VERR_NET_NO_BUFFER_SPACE. Signal error ? */
+ }
+
+ pThis->led.Actual.s.fWriting = 0;
+}
+
+/**
+ * Compute and write internet checksum (e1kCSum16) at the specified offset.
+ *
+ * @param pThis The device state structure.
+ * @param pPkt Pointer to the packet.
+ * @param u16PktLen Total length of the packet.
+ * @param cso Offset in packet to write checksum at.
+ * @param css Offset in packet to start computing
+ * checksum from.
+ * @param cse Offset in packet to stop computing
+ * checksum at.
+ * @param fUdp Replace 0 checksum with all 1s.
+ * @thread E1000_TX
+ */
+static void e1kInsertChecksum(PE1KSTATE pThis, uint8_t *pPkt, uint16_t u16PktLen, uint8_t cso, uint8_t css, uint16_t cse, bool fUdp = false)
+{
+ RT_NOREF1(pThis);
+
+ if (css >= u16PktLen)
+ {
+ E1kLog2(("%s css(%X) is greater than packet length-1(%X), checksum is not inserted\n",
+ pThis->szPrf, cso, u16PktLen));
+ return;
+ }
+
+ if (cso >= u16PktLen - 1)
+ {
+ E1kLog2(("%s cso(%X) is greater than packet length-2(%X), checksum is not inserted\n",
+ pThis->szPrf, cso, u16PktLen));
+ return;
+ }
+
+ if (cse == 0 || cse >= u16PktLen)
+ cse = u16PktLen - 1;
+ else if (cse < css)
+ {
+ E1kLog2(("%s css(%X) is greater than cse(%X), checksum is not inserted\n",
+ pThis->szPrf, css, cse));
+ return;
+ }
+
+ uint16_t u16ChkSum = e1kCSum16(pPkt + css, cse - css + 1);
+ if (fUdp && u16ChkSum == 0)
+ u16ChkSum = ~u16ChkSum; /* 0 means no checksum computed in case of UDP (see @bugref{9883}) */
+ E1kLog2(("%s Inserting csum: %04X at %02X, old value: %04X\n", pThis->szPrf,
+ u16ChkSum, cso, *(uint16_t*)(pPkt + cso)));
+ *(uint16_t*)(pPkt + cso) = u16ChkSum;
+}
+
+/**
+ * Add a part of descriptor's buffer to transmit frame.
+ *
+ * @remarks data.u64BufAddr is used unconditionally for both data
+ * and legacy descriptors since it is identical to
+ * legacy.u64BufAddr.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to the descriptor to transmit.
+ * @param u16Len Length of buffer to the end of segment.
+ * @param fSend Force packet sending.
+ * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
+ * @thread E1000_TX
+ */
+#ifndef E1K_WITH_TXD_CACHE
+static void e1kFallbackAddSegment(PPDMDEVINS pDevIns, PE1KSTATE pThis, RTGCPHYS PhysAddr, uint16_t u16Len, bool fSend, bool fOnWorkerThread)
+{
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ /* TCP header being transmitted */
+ struct E1kTcpHeader *pTcpHdr = (struct E1kTcpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.tu.u8CSS);
+ /* IP header being transmitted */
+ struct E1kIpHeader *pIpHdr = (struct E1kIpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.ip.u8CSS);
+
+ E1kLog3(("%s e1kFallbackAddSegment: Length=%x, remaining payload=%x, header=%x, send=%RTbool\n",
+ pThis->szPrf, u16Len, pThis->u32PayRemain, pThis->u16HdrRemain, fSend));
+ Assert(pThis->u32PayRemain + pThis->u16HdrRemain > 0);
+
+ PDMDevHlpPCIPhysRead(pDevIns, PhysAddr, pThis->aTxPacketFallback + pThis->u16TxPktLen, u16Len);
+ E1kLog3(("%s Dump of the segment:\n"
+ "%.*Rhxd\n"
+ "%s --- End of dump ---\n",
+ pThis->szPrf, u16Len, pThis->aTxPacketFallback + pThis->u16TxPktLen, pThis->szPrf));
+ pThis->u16TxPktLen += u16Len;
+ E1kLog3(("%s e1kFallbackAddSegment: pThis->u16TxPktLen=%x\n",
+ pThis->szPrf, pThis->u16TxPktLen));
+ if (pThis->u16HdrRemain > 0)
+ {
+ /* The header was not complete, check if it is now */
+ if (u16Len >= pThis->u16HdrRemain)
+ {
+ /* The rest is payload */
+ u16Len -= pThis->u16HdrRemain;
+ pThis->u16HdrRemain = 0;
+ /* Save partial checksum and flags */
+ pThis->u32SavedCsum = pTcpHdr->chksum;
+ pThis->u16SavedFlags = pTcpHdr->hdrlen_flags;
+ /* Clear FIN and PSH flags now and set them only in the last segment */
+ pTcpHdr->hdrlen_flags &= ~htons(E1K_TCP_FIN | E1K_TCP_PSH);
+ }
+ else
+ {
+ /* Still not */
+ pThis->u16HdrRemain -= u16Len;
+ E1kLog3(("%s e1kFallbackAddSegment: Header is still incomplete, 0x%x bytes remain.\n",
+ pThis->szPrf, pThis->u16HdrRemain));
+ return;
+ }
+ }
+
+ pThis->u32PayRemain -= u16Len;
+
+ if (fSend)
+ {
+ /* Leave ethernet header intact */
+ /* IP Total Length = payload + headers - ethernet header */
+ pIpHdr->total_len = htons(pThis->u16TxPktLen - pThis->contextTSE.ip.u8CSS);
+ E1kLog3(("%s e1kFallbackAddSegment: End of packet, pIpHdr->total_len=%x\n",
+ pThis->szPrf, ntohs(pIpHdr->total_len)));
+ /* Update IP Checksum */
+ pIpHdr->chksum = 0;
+ e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
+ pThis->contextTSE.ip.u8CSO,
+ pThis->contextTSE.ip.u8CSS,
+ pThis->contextTSE.ip.u16CSE);
+
+ /* Update TCP flags */
+ /* Restore original FIN and PSH flags for the last segment */
+ if (pThis->u32PayRemain == 0)
+ {
+ pTcpHdr->hdrlen_flags = pThis->u16SavedFlags;
+ E1K_INC_CNT32(TSCTC);
+ }
+ /* Add TCP length to partial pseudo header sum */
+ uint32_t csum = pThis->u32SavedCsum
+ + htons(pThis->u16TxPktLen - pThis->contextTSE.tu.u8CSS);
+ while (csum >> 16)
+ csum = (csum >> 16) + (csum & 0xFFFF);
+ pTcpHdr->chksum = csum;
+ /* Compute final checksum */
+ e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
+ pThis->contextTSE.tu.u8CSO,
+ pThis->contextTSE.tu.u8CSS,
+ pThis->contextTSE.tu.u16CSE);
+
+ /*
+ * Transmit it. If we've use the SG already, allocate a new one before
+ * we copy of the data.
+ */
+ PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
+ if (!pTxSg)
+ {
+ e1kXmitAllocBuf(pThis, pThisCC, pThis->u16TxPktLen + (pThis->fVTag ? 4 : 0), true /*fExactSize*/, false /*fGso*/);
+ pTxSg = pThisCC->CTX_SUFF(pTxSg);
+ }
+ if (pTxSg)
+ {
+ Assert(pThis->u16TxPktLen <= pThisCC->CTX_SUFF(pTxSg)->cbAvailable);
+ Assert(pTxSg->cSegs == 1);
+ if (pThis->CCCTX_SUFF(pTxSg)->aSegs[0].pvSeg != pThis->aTxPacketFallback)
+ memcpy(pTxSg->aSegs[0].pvSeg, pThis->aTxPacketFallback, pThis->u16TxPktLen);
+ pTxSg->cbUsed = pThis->u16TxPktLen;
+ pTxSg->aSegs[0].cbSeg = pThis->u16TxPktLen;
+ }
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+
+ /* Update Sequence Number */
+ pTcpHdr->seqno = htonl(ntohl(pTcpHdr->seqno) + pThis->u16TxPktLen
+ - pThis->contextTSE.dw3.u8HDRLEN);
+ /* Increment IP identification */
+ pIpHdr->ident = htons(ntohs(pIpHdr->ident) + 1);
+ }
+}
+#else /* E1K_WITH_TXD_CACHE */
+static int e1kFallbackAddSegment(PPDMDEVINS pDevIns, PE1KSTATE pThis, RTGCPHYS PhysAddr, uint16_t u16Len, bool fSend, bool fOnWorkerThread)
+{
+ int rc = VINF_SUCCESS;
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ /* TCP header being transmitted */
+ struct E1kTcpHeader *pTcpHdr = (struct E1kTcpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.tu.u8CSS);
+ /* IP header being transmitted */
+ struct E1kIpHeader *pIpHdr = (struct E1kIpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.ip.u8CSS);
+
+ E1kLog3(("%s e1kFallbackAddSegment: Length=%x, remaining payload=%x, header=%x, send=%RTbool\n",
+ pThis->szPrf, u16Len, pThis->u32PayRemain, pThis->u16HdrRemain, fSend));
+ AssertReturn(pThis->u32PayRemain + pThis->u16HdrRemain > 0, VINF_SUCCESS);
+
+ if (pThis->u16TxPktLen + u16Len <= sizeof(pThis->aTxPacketFallback))
+ PDMDevHlpPCIPhysRead(pDevIns, PhysAddr, pThis->aTxPacketFallback + pThis->u16TxPktLen, u16Len);
+ else
+ E1kLog(("%s e1kFallbackAddSegment: writing beyond aTxPacketFallback, u16TxPktLen=%d(0x%x) + u16Len=%d(0x%x) > %d\n",
+ pThis->szPrf, pThis->u16TxPktLen, pThis->u16TxPktLen, u16Len, u16Len, sizeof(pThis->aTxPacketFallback)));
+ E1kLog3(("%s Dump of the segment:\n"
+ "%.*Rhxd\n"
+ "%s --- End of dump ---\n",
+ pThis->szPrf, u16Len, pThis->aTxPacketFallback + pThis->u16TxPktLen, pThis->szPrf));
+ pThis->u16TxPktLen += u16Len;
+ E1kLog3(("%s e1kFallbackAddSegment: pThis->u16TxPktLen=%x\n",
+ pThis->szPrf, pThis->u16TxPktLen));
+ if (pThis->u16HdrRemain > 0)
+ {
+ /* The header was not complete, check if it is now */
+ if (u16Len >= pThis->u16HdrRemain)
+ {
+ /* The rest is payload */
+ u16Len -= pThis->u16HdrRemain;
+ pThis->u16HdrRemain = 0;
+ /* Save partial checksum and flags */
+ pThis->u32SavedCsum = pTcpHdr->chksum;
+ pThis->u16SavedFlags = pTcpHdr->hdrlen_flags;
+ /* Clear FIN and PSH flags now and set them only in the last segment */
+ pTcpHdr->hdrlen_flags &= ~htons(E1K_TCP_FIN | E1K_TCP_PSH);
+ }
+ else
+ {
+ /* Still not */
+ pThis->u16HdrRemain -= u16Len;
+ E1kLog3(("%s e1kFallbackAddSegment: Header is still incomplete, 0x%x bytes remain.\n",
+ pThis->szPrf, pThis->u16HdrRemain));
+ return rc;
+ }
+ }
+
+ if (u16Len > pThis->u32PayRemain)
+ pThis->u32PayRemain = 0;
+ else
+ pThis->u32PayRemain -= u16Len;
+
+ if (fSend)
+ {
+ /* Leave ethernet header intact */
+ /* IP Total Length = payload + headers - ethernet header */
+ pIpHdr->total_len = htons(pThis->u16TxPktLen - pThis->contextTSE.ip.u8CSS);
+ E1kLog3(("%s e1kFallbackAddSegment: End of packet, pIpHdr->total_len=%x\n",
+ pThis->szPrf, ntohs(pIpHdr->total_len)));
+ /* Update IP Checksum */
+ pIpHdr->chksum = 0;
+ e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
+ pThis->contextTSE.ip.u8CSO,
+ pThis->contextTSE.ip.u8CSS,
+ pThis->contextTSE.ip.u16CSE);
+
+ /* Update TCP flags */
+ /* Restore original FIN and PSH flags for the last segment */
+ if (pThis->u32PayRemain == 0)
+ {
+ pTcpHdr->hdrlen_flags = pThis->u16SavedFlags;
+ E1K_INC_CNT32(TSCTC);
+ }
+ /* Add TCP length to partial pseudo header sum */
+ uint32_t csum = pThis->u32SavedCsum
+ + htons(pThis->u16TxPktLen - pThis->contextTSE.tu.u8CSS);
+ while (csum >> 16)
+ csum = (csum >> 16) + (csum & 0xFFFF);
+ Assert(csum < 65536);
+ pTcpHdr->chksum = (uint16_t)csum;
+ /* Compute final checksum */
+ e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
+ pThis->contextTSE.tu.u8CSO,
+ pThis->contextTSE.tu.u8CSS,
+ pThis->contextTSE.tu.u16CSE);
+
+ /*
+ * Transmit it.
+ */
+ PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
+ if (pTxSg)
+ {
+ /* Make sure the packet fits into the allocated buffer */
+ size_t cbCopy = RT_MIN(pThis->u16TxPktLen, pThisCC->CTX_SUFF(pTxSg)->cbAvailable);
+#ifdef DEBUG
+ if (pThis->u16TxPktLen > pTxSg->cbAvailable)
+ E1kLog(("%s e1kFallbackAddSegment: truncating packet, u16TxPktLen=%d(0x%x) > cbAvailable=%d(0x%x)\n",
+ pThis->szPrf, pThis->u16TxPktLen, pThis->u16TxPktLen, pTxSg->cbAvailable, pTxSg->cbAvailable));
+#endif /* DEBUG */
+ Assert(pTxSg->cSegs == 1);
+ if (pTxSg->aSegs[0].pvSeg != pThis->aTxPacketFallback)
+ memcpy(pTxSg->aSegs[0].pvSeg, pThis->aTxPacketFallback, cbCopy);
+ pTxSg->cbUsed = cbCopy;
+ pTxSg->aSegs[0].cbSeg = cbCopy;
+ }
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+
+ /* Update Sequence Number */
+ pTcpHdr->seqno = htonl(ntohl(pTcpHdr->seqno) + pThis->u16TxPktLen
+ - pThis->contextTSE.dw3.u8HDRLEN);
+ /* Increment IP identification */
+ pIpHdr->ident = htons(ntohs(pIpHdr->ident) + 1);
+
+ /* Allocate new buffer for the next segment. */
+ if (pThis->u32PayRemain)
+ {
+ pThis->cbTxAlloc = RT_MIN(pThis->u32PayRemain,
+ pThis->contextTSE.dw3.u16MSS)
+ + pThis->contextTSE.dw3.u8HDRLEN;
+ /* Do not add VLAN tags to empty packets. */
+ if (pThis->fVTag && pThis->cbTxAlloc > 0)
+ pThis->cbTxAlloc += 4;
+ rc = e1kXmitAllocBuf(pThis, pThisCC, false /* fGSO */);
+ }
+ }
+
+ return rc;
+}
+#endif /* E1K_WITH_TXD_CACHE */
+
+#ifndef E1K_WITH_TXD_CACHE
+/**
+ * TCP segmentation offloading fallback: Add descriptor's buffer to transmit
+ * frame.
+ *
+ * We construct the frame in the fallback buffer first and the copy it to the SG
+ * buffer before passing it down to the network driver code.
+ *
+ * @returns true if the frame should be transmitted, false if not.
+ *
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to the descriptor to transmit.
+ * @param cbFragment Length of descriptor's buffer.
+ * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
+ * @thread E1000_TX
+ */
+static bool e1kFallbackAddToFrame(PE1KSTATE pThis, E1KTXDESC *pDesc, uint32_t cbFragment, bool fOnWorkerThread)
+{
+ PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
+ Assert(e1kGetDescType(pDesc) == E1K_DTYP_DATA);
+ Assert(pDesc->data.cmd.fTSE);
+ Assert(!e1kXmitIsGsoBuf(pTxSg));
+
+ uint16_t u16MaxPktLen = pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw3.u16MSS;
+ Assert(u16MaxPktLen != 0);
+ Assert(u16MaxPktLen < E1K_MAX_TX_PKT_SIZE);
+
+ /*
+ * Carve out segments.
+ */
+ do
+ {
+ /* Calculate how many bytes we have left in this TCP segment */
+ uint32_t cb = u16MaxPktLen - pThis->u16TxPktLen;
+ if (cb > cbFragment)
+ {
+ /* This descriptor fits completely into current segment */
+ cb = cbFragment;
+ e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, pDesc->data.cmd.fEOP /*fSend*/, fOnWorkerThread);
+ }
+ else
+ {
+ e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, true /*fSend*/, fOnWorkerThread);
+ /*
+ * Rewind the packet tail pointer to the beginning of payload,
+ * so we continue writing right beyond the header.
+ */
+ pThis->u16TxPktLen = pThis->contextTSE.dw3.u8HDRLEN;
+ }
+
+ pDesc->data.u64BufAddr += cb;
+ cbFragment -= cb;
+ } while (cbFragment > 0);
+
+ if (pDesc->data.cmd.fEOP)
+ {
+ /* End of packet, next segment will contain header. */
+ if (pThis->u32PayRemain != 0)
+ E1K_INC_CNT32(TSCTFC);
+ pThis->u16TxPktLen = 0;
+ e1kXmitFreeBuf(pThis, PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC));
+ }
+
+ return false;
+}
+#else /* E1K_WITH_TXD_CACHE */
+/**
+ * TCP segmentation offloading fallback: Add descriptor's buffer to transmit
+ * frame.
+ *
+ * We construct the frame in the fallback buffer first and the copy it to the SG
+ * buffer before passing it down to the network driver code.
+ *
+ * @returns error code
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to the descriptor to transmit.
+ * @param cbFragment Length of descriptor's buffer.
+ * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
+ * @thread E1000_TX
+ */
+static int e1kFallbackAddToFrame(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KTXDESC *pDesc, bool fOnWorkerThread)
+{
+#ifdef VBOX_STRICT
+ PPDMSCATTERGATHER pTxSg = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC)->CTX_SUFF(pTxSg);
+ Assert(e1kGetDescType(pDesc) == E1K_DTYP_DATA);
+ Assert(pDesc->data.cmd.fTSE);
+ Assert(!e1kXmitIsGsoBuf(pTxSg));
+#endif
+
+ uint16_t u16MaxPktLen = pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw3.u16MSS;
+ /* We cannot produce empty packets, ignore all TX descriptors (see @bugref{9571}) */
+ if (u16MaxPktLen == 0)
+ return VINF_SUCCESS;
+
+ /*
+ * Carve out segments.
+ */
+ int rc = VINF_SUCCESS;
+ do
+ {
+ /* Calculate how many bytes we have left in this TCP segment */
+ uint16_t cb = u16MaxPktLen - pThis->u16TxPktLen;
+ if (cb > pDesc->data.cmd.u20DTALEN)
+ {
+ /* This descriptor fits completely into current segment */
+ cb = (uint16_t)pDesc->data.cmd.u20DTALEN; /* u20DTALEN at this point is guarantied to fit into 16 bits. */
+ rc = e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, pDesc->data.cmd.fEOP /*fSend*/, fOnWorkerThread);
+ }
+ else
+ {
+ rc = e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, true /*fSend*/, fOnWorkerThread);
+ /*
+ * Rewind the packet tail pointer to the beginning of payload,
+ * so we continue writing right beyond the header.
+ */
+ pThis->u16TxPktLen = pThis->contextTSE.dw3.u8HDRLEN;
+ }
+
+ pDesc->data.u64BufAddr += cb;
+ pDesc->data.cmd.u20DTALEN -= cb;
+ } while (pDesc->data.cmd.u20DTALEN > 0 && RT_SUCCESS(rc));
+
+ if (pDesc->data.cmd.fEOP)
+ {
+ /* End of packet, next segment will contain header. */
+ if (pThis->u32PayRemain != 0)
+ E1K_INC_CNT32(TSCTFC);
+ pThis->u16TxPktLen = 0;
+ e1kXmitFreeBuf(pThis, PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC));
+ }
+
+ return VINF_SUCCESS; /// @todo consider rc;
+}
+#endif /* E1K_WITH_TXD_CACHE */
+
+
+/**
+ * Add descriptor's buffer to transmit frame.
+ *
+ * This deals with GSO and normal frames, e1kFallbackAddToFrame deals with the
+ * TSE frames we cannot handle as GSO.
+ *
+ * @returns true on success, false on failure.
+ *
+ * @param pDevIns The device instance.
+ * @param pThisCC The current context instance data.
+ * @param pThis The device state structure.
+ * @param PhysAddr The physical address of the descriptor buffer.
+ * @param cbFragment Length of descriptor's buffer.
+ * @thread E1000_TX
+ */
+static bool e1kAddToFrame(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, RTGCPHYS PhysAddr, uint32_t cbFragment)
+{
+ PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
+ bool const fGso = e1kXmitIsGsoBuf(pTxSg);
+ uint32_t const cbNewPkt = cbFragment + pThis->u16TxPktLen;
+
+ LogFlow(("%s e1kAddToFrame: ENTER cbFragment=%d u16TxPktLen=%d cbUsed=%d cbAvailable=%d fGSO=%s\n",
+ pThis->szPrf, cbFragment, pThis->u16TxPktLen, pTxSg->cbUsed, pTxSg->cbAvailable,
+ fGso ? "true" : "false"));
+ PCPDMNETWORKGSO pGso = (PCPDMNETWORKGSO)pTxSg->pvUser;
+ if (pGso)
+ {
+ if (RT_UNLIKELY(pGso->cbMaxSeg == 0))
+ {
+ E1kLog(("%s zero-sized fragments are not allowed\n", pThis->szPrf));
+ return false;
+ }
+ if (RT_UNLIKELY(pGso->u8Type == PDMNETWORKGSOTYPE_IPV4_UDP))
+ {
+ E1kLog(("%s UDP fragmentation is no longer supported\n", pThis->szPrf));
+ return false;
+ }
+ }
+ if (RT_UNLIKELY( !fGso && cbNewPkt > E1K_MAX_TX_PKT_SIZE ))
+ {
+ E1kLog(("%s Transmit packet is too large: %u > %u(max)\n", pThis->szPrf, cbNewPkt, E1K_MAX_TX_PKT_SIZE));
+ return false;
+ }
+ if (RT_UNLIKELY( cbNewPkt > pTxSg->cbAvailable ))
+ {
+ E1kLog(("%s Transmit packet is too large: %u > %u(max)\n", pThis->szPrf, cbNewPkt, pTxSg->cbAvailable));
+ return false;
+ }
+
+ if (RT_LIKELY(pTxSg))
+ {
+ Assert(pTxSg->cSegs == 1);
+ if (pTxSg->cbUsed != pThis->u16TxPktLen)
+ E1kLog(("%s e1kAddToFrame: pTxSg->cbUsed=%d(0x%x) != u16TxPktLen=%d(0x%x)\n",
+ pThis->szPrf, pTxSg->cbUsed, pTxSg->cbUsed, pThis->u16TxPktLen, pThis->u16TxPktLen));
+
+ PDMDevHlpPCIPhysRead(pDevIns, PhysAddr, (uint8_t *)pTxSg->aSegs[0].pvSeg + pThis->u16TxPktLen, cbFragment);
+
+ pTxSg->cbUsed = cbNewPkt;
+ }
+ pThis->u16TxPktLen = cbNewPkt;
+
+ return true;
+}
+
+
+/**
+ * Write the descriptor back to guest memory and notify the guest.
+ *
+ * @param pThis The device state structure.
+ * @param pDesc Pointer to the descriptor have been transmitted.
+ * @param addr Physical address of the descriptor in guest memory.
+ * @thread E1000_TX
+ */
+static void e1kDescReport(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KTXDESC *pDesc, RTGCPHYS addr)
+{
+ /*
+ * We fake descriptor write-back bursting. Descriptors are written back as they are
+ * processed.
+ */
+ /* Let's pretend we process descriptors. Write back with DD set. */
+ /*
+ * Prior to r71586 we tried to accomodate the case when write-back bursts
+ * are enabled without actually implementing bursting by writing back all
+ * descriptors, even the ones that do not have RS set. This caused kernel
+ * panics with Linux SMP kernels, as the e1000 driver tried to free up skb
+ * associated with written back descriptor if it happened to be a context
+ * descriptor since context descriptors do not have skb associated to them.
+ * Starting from r71586 we write back only the descriptors with RS set,
+ * which is a little bit different from what the real hardware does in
+ * case there is a chain of data descritors where some of them have RS set
+ * and others do not. It is very uncommon scenario imho.
+ * We need to check RPS as well since some legacy drivers use it instead of
+ * RS even with newer cards.
+ */
+ if (pDesc->legacy.cmd.fRS || pDesc->legacy.cmd.fRPS)
+ {
+ pDesc->legacy.dw3.fDD = 1; /* Descriptor Done */
+ e1kWriteBackDesc(pDevIns, pThis, pDesc, addr);
+ if (pDesc->legacy.cmd.fEOP)
+ {
+//#ifdef E1K_USE_TX_TIMERS
+ if (pThis->fTidEnabled && pDesc->legacy.cmd.fIDE)
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatTxIDE);
+ //if (pThis->fIntRaised)
+ //{
+ // /* Interrupt is already pending, no need for timers */
+ // ICR |= ICR_TXDW;
+ //}
+ //else {
+ /* Arm the timer to fire in TIVD usec (discard .024) */
+ e1kArmTimer(pDevIns, pThis, pThis->hTIDTimer, TIDV);
+# ifndef E1K_NO_TAD
+ /* If absolute timer delay is enabled and the timer is not running yet, arm it. */
+ E1kLog2(("%s Checking if TAD timer is running\n",
+ pThis->szPrf));
+ if (TADV != 0 && !PDMDevHlpTimerIsActive(pDevIns, pThis->hTADTimer))
+ e1kArmTimer(pDevIns, pThis, pThis->hTADTimer, TADV);
+# endif /* E1K_NO_TAD */
+ }
+ else
+ {
+ if (pThis->fTidEnabled)
+ {
+ E1kLog2(("%s No IDE set, cancel TAD timer and raise interrupt\n",
+ pThis->szPrf));
+ /* Cancel both timers if armed and fire immediately. */
+# ifndef E1K_NO_TAD
+ PDMDevHlpTimerStop(pDevIns, pThis->hTADTimer);
+# endif
+ PDMDevHlpTimerStop(pDevIns, pThis->hTIDTimer);
+ }
+//#endif /* E1K_USE_TX_TIMERS */
+ E1K_INC_ISTAT_CNT(pThis->uStatIntTx);
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXDW);
+//#ifdef E1K_USE_TX_TIMERS
+ }
+//#endif /* E1K_USE_TX_TIMERS */
+ }
+ }
+ else
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatTxNoRS);
+ }
+}
+
+#ifndef E1K_WITH_TXD_CACHE
+
+/**
+ * Process Transmit Descriptor.
+ *
+ * E1000 supports three types of transmit descriptors:
+ * - legacy data descriptors of older format (context-less).
+ * - data the same as legacy but providing new offloading capabilities.
+ * - context sets up the context for following data descriptors.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ * @param pDesc Pointer to descriptor union.
+ * @param addr Physical address of descriptor in guest memory.
+ * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
+ * @thread E1000_TX
+ */
+static int e1kXmitDesc(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, E1KTXDESC *pDesc,
+ RTGCPHYS addr, bool fOnWorkerThread)
+{
+ int rc = VINF_SUCCESS;
+ uint32_t cbVTag = 0;
+
+ e1kPrintTDesc(pThis, pDesc, "vvv");
+
+//#ifdef E1K_USE_TX_TIMERS
+ if (pThis->fTidEnabled)
+ e1kCancelTimer(pDevIns, pThis, pThis->hTIDTimer);
+//#endif /* E1K_USE_TX_TIMERS */
+
+ switch (e1kGetDescType(pDesc))
+ {
+ case E1K_DTYP_CONTEXT:
+ if (pDesc->context.dw2.fTSE)
+ {
+ pThis->contextTSE = pDesc->context;
+ pThis->u32PayRemain = pDesc->context.dw2.u20PAYLEN;
+ pThis->u16HdrRemain = pDesc->context.dw3.u8HDRLEN;
+ e1kSetupGsoCtx(&pThis->GsoCtx, &pDesc->context);
+ STAM_COUNTER_INC(&pThis->StatTxDescCtxTSE);
+ }
+ else
+ {
+ pThis->contextNormal = pDesc->context;
+ STAM_COUNTER_INC(&pThis->StatTxDescCtxNormal);
+ }
+ E1kLog2(("%s %s context updated: IP CSS=%02X, IP CSO=%02X, IP CSE=%04X"
+ ", TU CSS=%02X, TU CSO=%02X, TU CSE=%04X\n", pThis->szPrf,
+ pDesc->context.dw2.fTSE ? "TSE" : "Normal",
+ pDesc->context.ip.u8CSS,
+ pDesc->context.ip.u8CSO,
+ pDesc->context.ip.u16CSE,
+ pDesc->context.tu.u8CSS,
+ pDesc->context.tu.u8CSO,
+ pDesc->context.tu.u16CSE));
+ E1K_INC_ISTAT_CNT(pThis->uStatDescCtx);
+ e1kDescReport(pThis, pDesc, addr);
+ break;
+
+ case E1K_DTYP_DATA:
+ {
+ if (pDesc->data.cmd.u20DTALEN == 0 || pDesc->data.u64BufAddr == 0)
+ {
+ E1kLog2(("% Empty data descriptor, skipped.\n", pThis->szPrf));
+ /** @todo Same as legacy when !TSE. See below. */
+ break;
+ }
+ STAM_COUNTER_INC(pDesc->data.cmd.fTSE?
+ &pThis->StatTxDescTSEData:
+ &pThis->StatTxDescData);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ E1K_INC_ISTAT_CNT(pThis->uStatDescDat);
+
+ /*
+ * The last descriptor of non-TSE packet must contain VLE flag.
+ * TSE packets have VLE flag in the first descriptor. The later
+ * case is taken care of a bit later when cbVTag gets assigned.
+ *
+ * 1) pDesc->data.cmd.fEOP && !pDesc->data.cmd.fTSE
+ */
+ if (pDesc->data.cmd.fEOP && !pDesc->data.cmd.fTSE)
+ {
+ pThis->fVTag = pDesc->data.cmd.fVLE;
+ pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
+ }
+ /*
+ * First fragment: Allocate new buffer and save the IXSM and TXSM
+ * packet options as these are only valid in the first fragment.
+ */
+ if (pThis->u16TxPktLen == 0)
+ {
+ pThis->fIPcsum = pDesc->data.dw3.fIXSM;
+ pThis->fTCPcsum = pDesc->data.dw3.fTXSM;
+ E1kLog2(("%s Saving checksum flags:%s%s; \n", pThis->szPrf,
+ pThis->fIPcsum ? " IP" : "",
+ pThis->fTCPcsum ? " TCP/UDP" : ""));
+ if (pDesc->data.cmd.fTSE)
+ {
+ /* 2) pDesc->data.cmd.fTSE && pThis->u16TxPktLen == 0 */
+ pThis->fVTag = pDesc->data.cmd.fVLE;
+ pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
+ cbVTag = pThis->fVTag ? 4 : 0;
+ }
+ else if (pDesc->data.cmd.fEOP)
+ cbVTag = pDesc->data.cmd.fVLE ? 4 : 0;
+ else
+ cbVTag = 4;
+ E1kLog3(("%s About to allocate TX buffer: cbVTag=%u\n", pThis->szPrf, cbVTag));
+ if (e1kCanDoGso(pThis, &pThis->GsoCtx, &pDesc->data, &pThis->contextTSE))
+ rc = e1kXmitAllocBuf(pThis, pThisCC, pThis->contextTSE.dw2.u20PAYLEN + pThis->contextTSE.dw3.u8HDRLEN + cbVTag,
+ true /*fExactSize*/, true /*fGso*/);
+ else if (pDesc->data.cmd.fTSE)
+ rc = e1kXmitAllocBuf(pThis, pThisCC, , pThis->contextTSE.dw3.u16MSS + pThis->contextTSE.dw3.u8HDRLEN + cbVTag,
+ pDesc->data.cmd.fTSE /*fExactSize*/, false /*fGso*/);
+ else
+ rc = e1kXmitAllocBuf(pThis, pThisCC, pDesc->data.cmd.u20DTALEN + cbVTag,
+ pDesc->data.cmd.fEOP /*fExactSize*/, false /*fGso*/);
+
+ /**
+ * @todo: Perhaps it is not that simple for GSO packets! We may
+ * need to unwind some changes.
+ */
+ if (RT_FAILURE(rc))
+ {
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ break;
+ }
+ /** @todo Is there any way to indicating errors other than collisions? Like
+ * VERR_NET_DOWN. */
+ }
+
+ /*
+ * Add the descriptor data to the frame. If the frame is complete,
+ * transmit it and reset the u16TxPktLen field.
+ */
+ if (e1kXmitIsGsoBuf(pThisCC->CTX_SUFF(pTxSg)))
+ {
+ STAM_COUNTER_INC(&pThis->StatTxPathGSO);
+ bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
+ if (pDesc->data.cmd.fEOP)
+ {
+ if ( fRc
+ && pThisCC->CTX_SUFF(pTxSg)
+ && pThisCC->CTX_SUFF(pTxSg)->cbUsed == (size_t)pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN)
+ {
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ E1K_INC_CNT32(TSCTC);
+ }
+ else
+ {
+ if (fRc)
+ E1kLog(("%s bad GSO/TSE %p or %u < %u\n" , pThis->szPrf,
+ pThisCC->CTX_SUFF(pTxSg), pThisCC->CTX_SUFF(pTxSg) ? pThisCC->CTX_SUFF(pTxSg)->cbUsed : 0,
+ pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN));
+ e1kXmitFreeBuf(pThis);
+ E1K_INC_CNT32(TSCTFC);
+ }
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ else if (!pDesc->data.cmd.fTSE)
+ {
+ STAM_COUNTER_INC(&pThis->StatTxPathRegular);
+ bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
+ if (pDesc->data.cmd.fEOP)
+ {
+ if (fRc && pThisCC->CTX_SUFF(pTxSg))
+ {
+ Assert(pThisCC->CTX_SUFF(pTxSg)->cSegs == 1);
+ if (pThis->fIPcsum)
+ e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
+ pThis->contextNormal.ip.u8CSO,
+ pThis->contextNormal.ip.u8CSS,
+ pThis->contextNormal.ip.u16CSE);
+ if (pThis->fTCPcsum)
+ e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
+ pThis->contextNormal.tu.u8CSO,
+ pThis->contextNormal.tu.u8CSS,
+ pThis->contextNormal.tu.u16CSE,
+ !pThis->contextNormal.dw2.fTCP);
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ }
+ else
+ e1kXmitFreeBuf(pThis);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ else
+ {
+ STAM_COUNTER_INC(&pThis->StatTxPathFallback);
+ e1kFallbackAddToFrame(pDevIns, pThis, pDesc, pDesc->data.cmd.u20DTALEN, fOnWorkerThread);
+ }
+
+ e1kDescReport(pThis, pDesc, addr);
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ break;
+ }
+
+ case E1K_DTYP_LEGACY:
+ if (pDesc->legacy.cmd.u16Length == 0 || pDesc->legacy.u64BufAddr == 0)
+ {
+ E1kLog(("%s Empty legacy descriptor, skipped.\n", pThis->szPrf));
+ /** @todo 3.3.3, Length/Buffer Address: RS set -> write DD when processing. */
+ break;
+ }
+ STAM_COUNTER_INC(&pThis->StatTxDescLegacy);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
+
+ /* First fragment: allocate new buffer. */
+ if (pThis->u16TxPktLen == 0)
+ {
+ if (pDesc->legacy.cmd.fEOP)
+ cbVTag = pDesc->legacy.cmd.fVLE ? 4 : 0;
+ else
+ cbVTag = 4;
+ E1kLog3(("%s About to allocate TX buffer: cbVTag=%u\n", pThis->szPrf, cbVTag));
+ /** @todo reset status bits? */
+ rc = e1kXmitAllocBuf(pThis, pThisCC, pDesc->legacy.cmd.u16Length + cbVTag, pDesc->legacy.cmd.fEOP, false /*fGso*/);
+ if (RT_FAILURE(rc))
+ {
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ break;
+ }
+
+ /** @todo Is there any way to indicating errors other than collisions? Like
+ * VERR_NET_DOWN. */
+ }
+
+ /* Add fragment to frame. */
+ if (e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->legacy.cmd.u16Length))
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatDescLeg);
+
+ /* Last fragment: Transmit and reset the packet storage counter. */
+ if (pDesc->legacy.cmd.fEOP)
+ {
+ pThis->fVTag = pDesc->legacy.cmd.fVLE;
+ pThis->u16VTagTCI = pDesc->legacy.dw3.u16Special;
+ /** @todo Offload processing goes here. */
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ /* Last fragment + failure: free the buffer and reset the storage counter. */
+ else if (pDesc->legacy.cmd.fEOP)
+ {
+ e1kXmitFreeBuf(pThis);
+ pThis->u16TxPktLen = 0;
+ }
+
+ e1kDescReport(pThis, pDesc, addr);
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ break;
+
+ default:
+ E1kLog(("%s ERROR Unsupported transmit descriptor type: 0x%04x\n",
+ pThis->szPrf, e1kGetDescType(pDesc)));
+ break;
+ }
+
+ return rc;
+}
+
+#else /* E1K_WITH_TXD_CACHE */
+
+/**
+ * Process Transmit Descriptor.
+ *
+ * E1000 supports three types of transmit descriptors:
+ * - legacy data descriptors of older format (context-less).
+ * - data the same as legacy but providing new offloading capabilities.
+ * - context sets up the context for following data descriptors.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param pThisCC The current context instance data.
+ * @param pDesc Pointer to descriptor union.
+ * @param addr Physical address of descriptor in guest memory.
+ * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
+ * @param cbPacketSize Size of the packet as previously computed.
+ * @thread E1000_TX
+ */
+static int e1kXmitDesc(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, E1KTXDESC *pDesc,
+ RTGCPHYS addr, bool fOnWorkerThread)
+{
+ int rc = VINF_SUCCESS;
+
+ e1kPrintTDesc(pThis, pDesc, "vvv");
+
+//#ifdef E1K_USE_TX_TIMERS
+ if (pThis->fTidEnabled)
+ PDMDevHlpTimerStop(pDevIns, pThis->hTIDTimer);
+//#endif /* E1K_USE_TX_TIMERS */
+
+ switch (e1kGetDescType(pDesc))
+ {
+ case E1K_DTYP_CONTEXT:
+ /* The caller have already updated the context */
+ E1K_INC_ISTAT_CNT(pThis->uStatDescCtx);
+ e1kDescReport(pDevIns, pThis, pDesc, addr);
+ break;
+
+ case E1K_DTYP_DATA:
+ {
+ STAM_COUNTER_INC(pDesc->data.cmd.fTSE?
+ &pThis->StatTxDescTSEData:
+ &pThis->StatTxDescData);
+ E1K_INC_ISTAT_CNT(pThis->uStatDescDat);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ if (pDesc->data.cmd.u20DTALEN == 0 || pDesc->data.u64BufAddr == 0)
+ {
+ E1kLog2(("%s Empty data descriptor, skipped.\n", pThis->szPrf));
+ if (pDesc->data.cmd.fEOP)
+ {
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ else
+ {
+ /*
+ * Add the descriptor data to the frame. If the frame is complete,
+ * transmit it and reset the u16TxPktLen field.
+ */
+ if (e1kXmitIsGsoBuf(pThisCC->CTX_SUFF(pTxSg)))
+ {
+ STAM_COUNTER_INC(&pThis->StatTxPathGSO);
+ bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
+ if (pDesc->data.cmd.fEOP)
+ {
+ if ( fRc
+ && pThisCC->CTX_SUFF(pTxSg)
+ && pThisCC->CTX_SUFF(pTxSg)->cbUsed == (size_t)pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN)
+ {
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ E1K_INC_CNT32(TSCTC);
+ }
+ else
+ {
+ if (fRc)
+ E1kLog(("%s bad GSO/TSE %p or %u < %u\n" , pThis->szPrf,
+ pThisCC->CTX_SUFF(pTxSg), pThisCC->CTX_SUFF(pTxSg) ? pThisCC->CTX_SUFF(pTxSg)->cbUsed : 0,
+ pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN));
+ e1kXmitFreeBuf(pThis, pThisCC);
+ E1K_INC_CNT32(TSCTFC);
+ }
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ else if (!pDesc->data.cmd.fTSE)
+ {
+ STAM_COUNTER_INC(&pThis->StatTxPathRegular);
+ bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
+ if (pDesc->data.cmd.fEOP)
+ {
+ if (fRc && pThisCC->CTX_SUFF(pTxSg))
+ {
+ Assert(pThisCC->CTX_SUFF(pTxSg)->cSegs == 1);
+ if (pThis->fIPcsum)
+ e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
+ pThis->contextNormal.ip.u8CSO,
+ pThis->contextNormal.ip.u8CSS,
+ pThis->contextNormal.ip.u16CSE);
+ if (pThis->fTCPcsum)
+ e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
+ pThis->contextNormal.tu.u8CSO,
+ pThis->contextNormal.tu.u8CSS,
+ pThis->contextNormal.tu.u16CSE,
+ !pThis->contextNormal.dw2.fTCP);
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ }
+ else
+ e1kXmitFreeBuf(pThis, pThisCC);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ else
+ {
+ STAM_COUNTER_INC(&pThis->StatTxPathFallback);
+ rc = e1kFallbackAddToFrame(pDevIns, pThis, pDesc, fOnWorkerThread);
+ }
+ }
+ e1kDescReport(pDevIns, pThis, pDesc, addr);
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ break;
+ }
+
+ case E1K_DTYP_LEGACY:
+ STAM_COUNTER_INC(&pThis->StatTxDescLegacy);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ if (pDesc->legacy.cmd.u16Length == 0 || pDesc->legacy.u64BufAddr == 0)
+ {
+ E1kLog(("%s Empty legacy descriptor, skipped.\n", pThis->szPrf));
+ if (pDesc->data.cmd.fEOP)
+ {
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ else
+ {
+ /* Add fragment to frame. */
+ if (e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->legacy.cmd.u16Length))
+ {
+ E1K_INC_ISTAT_CNT(pThis->uStatDescLeg);
+
+ /* Last fragment: Transmit and reset the packet storage counter. */
+ if (pDesc->legacy.cmd.fEOP)
+ {
+ if (pDesc->legacy.cmd.fIC)
+ {
+ e1kInsertChecksum(pThis,
+ (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg,
+ pThis->u16TxPktLen,
+ pDesc->legacy.cmd.u8CSO,
+ pDesc->legacy.dw3.u8CSS,
+ 0);
+ }
+ e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ /* Last fragment + failure: free the buffer and reset the storage counter. */
+ else if (pDesc->legacy.cmd.fEOP)
+ {
+ e1kXmitFreeBuf(pThis, pThisCC);
+ pThis->u16TxPktLen = 0;
+ }
+ }
+ e1kDescReport(pDevIns, pThis, pDesc, addr);
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ break;
+
+ default:
+ E1kLog(("%s ERROR Unsupported transmit descriptor type: 0x%04x\n",
+ pThis->szPrf, e1kGetDescType(pDesc)));
+ break;
+ }
+
+ return rc;
+}
+
+DECLINLINE(bool) e1kUpdateTxContext(PE1KSTATE pThis, E1KTXDESC *pDesc)
+{
+ if (pDesc->context.dw2.fTSE)
+ {
+ if (!e1kSetupGsoCtx(&pThis->GsoCtx, &pDesc->context))
+ {
+ pThis->contextTSE.dw2.u4DTYP = E1K_DTYP_INVALID;
+ return false;
+ }
+ pThis->contextTSE = pDesc->context;
+ uint32_t cbMaxSegmentSize = pThis->contextTSE.dw3.u16MSS + pThis->contextTSE.dw3.u8HDRLEN + 4; /*VTAG*/
+ if (RT_UNLIKELY(cbMaxSegmentSize > E1K_MAX_TX_PKT_SIZE))
+ {
+ pThis->contextTSE.dw3.u16MSS = E1K_MAX_TX_PKT_SIZE - pThis->contextTSE.dw3.u8HDRLEN - 4; /*VTAG*/
+ LogRelMax(10, ("%s: Transmit packet is too large: %u > %u(max). Adjusted MSS to %u.\n",
+ pThis->szPrf, cbMaxSegmentSize, E1K_MAX_TX_PKT_SIZE, pThis->contextTSE.dw3.u16MSS));
+ }
+ pThis->u32PayRemain = pThis->contextTSE.dw2.u20PAYLEN;
+ pThis->u16HdrRemain = pThis->contextTSE.dw3.u8HDRLEN;
+ e1kSetupGsoCtx(&pThis->GsoCtx, &pThis->contextTSE);
+ STAM_COUNTER_INC(&pThis->StatTxDescCtxTSE);
+ }
+ else
+ {
+ pThis->contextNormal = pDesc->context;
+ STAM_COUNTER_INC(&pThis->StatTxDescCtxNormal);
+ }
+ E1kLog2(("%s %s context updated: IP CSS=%02X, IP CSO=%02X, IP CSE=%04X"
+ ", TU CSS=%02X, TU CSO=%02X, TU CSE=%04X\n", pThis->szPrf,
+ pDesc->context.dw2.fTSE ? "TSE" : "Normal",
+ pDesc->context.ip.u8CSS,
+ pDesc->context.ip.u8CSO,
+ pDesc->context.ip.u16CSE,
+ pDesc->context.tu.u8CSS,
+ pDesc->context.tu.u8CSO,
+ pDesc->context.tu.u16CSE));
+ return true; /* Consider returning false for invalid descriptors */
+}
+
+enum E1kPacketType
+{
+ E1K_PACKET_NONE = 0,
+ E1K_PACKET_LEGACY,
+ E1K_PACKET_NORMAL,
+ E1K_PACKET_TSE
+};
+
+static int e1kLocateTxPacket(PE1KSTATE pThis, PE1KTXDC pTxdc)
+{
+ LogFlow(("%s e1kLocateTxPacket: ENTER cbTxAlloc=%d\n",
+ pThis->szPrf, pThis->cbTxAlloc));
+ /* Check if we have located the packet already. */
+ if (pThis->cbTxAlloc)
+ {
+ LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d\n",
+ pThis->szPrf, pThis->cbTxAlloc));
+ return true;
+ }
+
+ pThis->fGSO = false;
+ pThis->fVTag = false;
+ pThis->fIPcsum = false;
+ pThis->fTCPcsum = false;
+ pThis->u16TxPktLen = 0;
+
+ enum E1kPacketType packetType = E1K_PACKET_NONE;
+ enum E1kPacketType expectedPacketType = E1K_PACKET_NONE;
+ /*
+ * Valid packets start with 1 or 0 context descriptors, followed by 1 or
+ * more data descriptors of the same type: legacy, normal or TSE. Note
+ * that legacy descriptors do not belong to neither normal nor segmentation
+ * contexts rendering the sequence (context_descriptor, legacy_descriptor)
+ * invalid, but the context descriptor will still be applied and the legacy
+ * descriptor will be treated as the beginning of next packet.
+ */
+ bool fInvalidPacket = false;
+ bool fTSE = false;
+ uint32_t cbPacket = 0;
+
+ /* Since we process one packet at a time we will only mark current packet's descriptors as valid */
+ memset(pThis->afTxDValid, 0, sizeof(pThis->afTxDValid));
+ for (int i = pThis->iTxDCurrent; i < pThis->nTxDFetched; ++i)
+ {
+ E1KTXDESC *pDesc = &pThis->aTxDescriptors[i];
+
+ switch (e1kGetDescType(pDesc))
+ {
+ case E1K_DTYP_CONTEXT:
+ /* There can be only one context per packet. Each context descriptor starts a new packet. */
+ if (packetType != E1K_PACKET_NONE)
+ {
+ fInvalidPacket = true;
+ break;
+ }
+ packetType = (pDesc->context.dw2.fTSE) ? E1K_PACKET_TSE : E1K_PACKET_NORMAL;
+ if (cbPacket == 0)
+ pThis->afTxDValid[i] = e1kUpdateTxContext(pThis, pDesc);
+ else
+ E1kLog(("%s e1kLocateTxPacket: ignoring a context descriptor in the middle of a packet, cbPacket=%d\n",
+ pThis->szPrf, cbPacket));
+ continue;
+ case E1K_DTYP_LEGACY:
+ if (packetType != E1K_PACKET_NONE && packetType != E1K_PACKET_LEGACY)
+ {
+ fInvalidPacket = true;
+ break;
+ }
+ packetType = E1K_PACKET_LEGACY;
+ /* Skip invalid descriptors. */
+ if (cbPacket > 0 && (pThis->fGSO || fTSE))
+ {
+ E1kLog(("%s e1kLocateTxPacket: ignoring a legacy descriptor in the segmentation context, cbPacket=%d\n",
+ pThis->szPrf, cbPacket));
+ continue;
+ }
+ pThis->afTxDValid[i] = true; /* Passed all checks, process it */
+
+ /* Skip empty descriptors. */
+ if (!pDesc->legacy.u64BufAddr || !pDesc->legacy.cmd.u16Length)
+ break;
+ cbPacket += pDesc->legacy.cmd.u16Length;
+ pThis->fGSO = false;
+ break;
+ case E1K_DTYP_DATA:
+ expectedPacketType = pDesc->data.cmd.fTSE ? E1K_PACKET_TSE : E1K_PACKET_NORMAL;
+ if (packetType != E1K_PACKET_NONE && packetType != expectedPacketType)
+ {
+ fInvalidPacket = true;
+ break;
+ }
+ /* Skip invalid descriptors. */
+ if (pDesc->data.cmd.fTSE)
+ {
+ if (pThis->contextTSE.dw2.u4DTYP == E1K_DTYP_INVALID)
+ {
+ E1kLog(("%s e1kLocateTxPacket: ignoring TSE descriptor in invalid segmentation context, cbPacket=%d\n",
+ pThis->szPrf, cbPacket));
+ continue;
+ }
+ }
+ else /* !TSE */
+ {
+ if (pThis->contextNormal.dw2.u4DTYP == E1K_DTYP_INVALID)
+ {
+ E1kLog(("%s e1kLocateTxPacket: ignoring non-TSE descriptor in invalid normal context, cbPacket=%d\n",
+ pThis->szPrf, cbPacket));
+ continue;
+ }
+ }
+ if (cbPacket > 0 && (bool)pDesc->data.cmd.fTSE != fTSE)
+ {
+ E1kLog(("%s e1kLocateTxPacket: ignoring %sTSE descriptor in the %ssegmentation context, cbPacket=%d\n",
+ pThis->szPrf, pDesc->data.cmd.fTSE ? "" : "non-", fTSE ? "" : "non-", cbPacket));
+ continue;
+ }
+ pThis->afTxDValid[i] = true; /* Passed all checks, process it */
+
+ /* Skip empty descriptors. */
+ if (!pDesc->data.u64BufAddr || !pDesc->data.cmd.u20DTALEN)
+ break;
+ if (cbPacket == 0)
+ {
+ /*
+ * The first fragment: save IXSM and TXSM options
+ * as these are only valid in the first fragment.
+ */
+ pThis->fIPcsum = pDesc->data.dw3.fIXSM;
+ pThis->fTCPcsum = pDesc->data.dw3.fTXSM;
+ fTSE = pDesc->data.cmd.fTSE;
+ /*
+ * TSE descriptors have VLE bit properly set in
+ * the first fragment.
+ */
+ if (fTSE)
+ {
+ pThis->fVTag = pDesc->data.cmd.fVLE;
+ pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
+ }
+ pThis->fGSO = e1kCanDoGso(pThis, &pThis->GsoCtx, &pDesc->data, &pThis->contextTSE);
+ }
+ cbPacket += pDesc->data.cmd.u20DTALEN;
+ break;
+ default:
+ AssertMsgFailed(("Impossible descriptor type!"));
+ continue;
+ }
+ if (fInvalidPacket)
+ {
+ for (int index = pThis->iTxDCurrent; index < i; ++index)
+ pThis->afTxDValid[index] = false; /* Make sure all descriptors for this packet are skipped by processing */
+ LogFlow(("%s e1kLocateTxPacket: marked %d descriptors as invalid\n", pThis->szPrf, i - pThis->iTxDCurrent));
+ LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d cbPacket=%d%s%s\n",
+ pThis->szPrf, pThis->cbTxAlloc, cbPacket,
+ pThis->fGSO ? " GSO" : "", fTSE ? " TSE" : ""));
+ pTxdc->nextPacket = i;
+ return true;
+ }
+ if (pDesc->legacy.cmd.fEOP)
+ {
+ /*
+ * Non-TSE descriptors have VLE bit properly set in
+ * the last fragment.
+ */
+ if (!fTSE)
+ {
+ pThis->fVTag = pDesc->data.cmd.fVLE;
+ pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
+ }
+ /*
+ * Compute the required buffer size. If we cannot do GSO but still
+ * have to do segmentation we allocate the first segment only.
+ */
+ pThis->cbTxAlloc = (!fTSE || pThis->fGSO) ?
+ cbPacket :
+ RT_MIN(cbPacket, pThis->contextTSE.dw3.u16MSS + pThis->contextTSE.dw3.u8HDRLEN);
+ /* Do not add VLAN tags to empty packets. */
+ if (pThis->fVTag && pThis->cbTxAlloc > 0)
+ pThis->cbTxAlloc += 4;
+ LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d cbPacket=%d%s%s\n",
+ pThis->szPrf, pThis->cbTxAlloc, cbPacket,
+ pThis->fGSO ? " GSO" : "", fTSE ? " TSE" : ""));
+ pTxdc->nextPacket = i + 1;
+ return true;
+ }
+ }
+
+ if (cbPacket == 0 && pThis->nTxDFetched - pThis->iTxDCurrent > 0)
+ {
+ /* All descriptors were empty, we need to process them as a dummy packet */
+ LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d, zero packet!\n",
+ pThis->szPrf, pThis->cbTxAlloc));
+ pTxdc->nextPacket = pThis->nTxDFetched;
+ return true;
+ }
+ LogFlow(("%s e1kLocateTxPacket: RET false cbTxAlloc=%d cbPacket=%d\n",
+ pThis->szPrf, pThis->cbTxAlloc, cbPacket));
+ return false;
+}
+
+static int e1kXmitPacket(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread, PE1KTXDC pTxdc)
+{
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ int rc = VINF_SUCCESS;
+
+ LogFlow(("%s e1kXmitPacket: ENTER current=%d fetched=%d\n",
+ pThis->szPrf, pThis->iTxDCurrent, pThis->nTxDFetched));
+
+ while (pThis->iTxDCurrent < pTxdc->nextPacket && pThis->iTxDCurrent < pThis->nTxDFetched)
+ {
+ E1KTXDESC *pDesc = &pThis->aTxDescriptors[pThis->iTxDCurrent];
+ E1kLog3(("%s About to process new TX descriptor at %08x%08x, TDLEN=%08x, TDH=%08x, TDT=%08x\n",
+ pThis->szPrf, TDBAH, TDBAL + pTxdc->tdh * sizeof(E1KTXDESC), pTxdc->tdlen, pTxdc->tdh, pTxdc->tdt));
+ if (!pThis->afTxDValid[pThis->iTxDCurrent])
+ {
+ e1kPrintTDesc(pThis, pDesc, "vvv");
+ E1kLog(("%s e1kXmitDesc: skipping bad descriptor ^^^\n", pThis->szPrf));
+ e1kDescReport(pDevIns, pThis, pDesc, e1kDescAddr(TDBAH, TDBAL, pTxdc->tdh));
+ rc = VINF_SUCCESS;
+ }
+ else
+ rc = e1kXmitDesc(pDevIns, pThis, pThisCC, pDesc, e1kDescAddr(TDBAH, TDBAL, pTxdc->tdh), fOnWorkerThread);
+ if (RT_FAILURE(rc))
+ break;
+ if (++pTxdc->tdh * sizeof(E1KTXDESC) >= pTxdc->tdlen)
+ pTxdc->tdh = 0;
+ TDH = pTxdc->tdh; /* Sync the actual register and TXDC */
+ uint32_t uLowThreshold = GET_BITS(TXDCTL, LWTHRESH)*8;
+ if (uLowThreshold != 0 && e1kGetTxLen(pTxdc) <= uLowThreshold)
+ {
+ E1kLog2(("%s Low on transmit descriptors, raise ICR.TXD_LOW, len=%x thresh=%x\n",
+ pThis->szPrf, e1kGetTxLen(pTxdc), GET_BITS(TXDCTL, LWTHRESH)*8));
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXD_LOW);
+ }
+ ++pThis->iTxDCurrent;
+ if (e1kGetDescType(pDesc) != E1K_DTYP_CONTEXT && pDesc->legacy.cmd.fEOP)
+ break;
+ }
+
+ LogFlow(("%s e1kXmitPacket: RET %Rrc current=%d fetched=%d\n",
+ pThis->szPrf, rc, pThis->iTxDCurrent, pThis->nTxDFetched));
+ return rc;
+}
+
+#endif /* E1K_WITH_TXD_CACHE */
+#ifndef E1K_WITH_TXD_CACHE
+
+/**
+ * Transmit pending descriptors.
+ *
+ * @returns VBox status code. VERR_TRY_AGAIN is returned if we're busy.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The E1000 state.
+ * @param fOnWorkerThread Whether we're on a worker thread or on an EMT.
+ */
+static int e1kXmitPending(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread)
+{
+ int rc = VINF_SUCCESS;
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+
+ /* Check if transmitter is enabled. */
+ if (!(TCTL & TCTL_EN))
+ return VINF_SUCCESS;
+ /*
+ * Grab the xmit lock of the driver as well as the E1K device state.
+ */
+ rc = e1kCsTxEnter(pThis, VERR_SEM_BUSY);
+ if (RT_LIKELY(rc == VINF_SUCCESS))
+ {
+ PPDMINETWORKUP pDrv = pThis->CTX_SUFF(pDrv);
+ if (pDrv)
+ {
+ rc = pDrv->pfnBeginXmit(pDrv, fOnWorkerThread);
+ if (RT_FAILURE(rc))
+ {
+ e1kCsTxLeave(pThis);
+ return rc;
+ }
+ }
+ /*
+ * Process all pending descriptors.
+ * Note! Do not process descriptors in locked state
+ */
+ while (TDH != TDT && !pThis->fLocked)
+ {
+ E1KTXDESC desc;
+ E1kLog3(("%s About to process new TX descriptor at %08x%08x, TDLEN=%08x, TDH=%08x, TDT=%08x\n",
+ pThis->szPrf, TDBAH, TDBAL + TDH * sizeof(desc), TDLEN, TDH, TDT));
+
+ e1kLoadDesc(pDevIns, &desc, ((uint64_t)TDBAH << 32) + TDBAL + TDH * sizeof(desc));
+ rc = e1kXmitDesc(pDevIns, pThis, pThisCC, &desc, e1kDescAddr(TDBAH, TDBAL, TDH), fOnWorkerThread);
+ /* If we failed to transmit descriptor we will try it again later */
+ if (RT_FAILURE(rc))
+ break;
+ if (++TDH * sizeof(desc) >= TDLEN)
+ TDH = 0;
+
+ if (e1kGetTxLen(pThis) <= GET_BITS(TXDCTL, LWTHRESH)*8)
+ {
+ E1kLog2(("%s Low on transmit descriptors, raise ICR.TXD_LOW, len=%x thresh=%x\n",
+ pThis->szPrf, e1kGetTxLen(pThis), GET_BITS(TXDCTL, LWTHRESH)*8));
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXD_LOW);
+ }
+
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ }
+
+ /// @todo uncomment: pThis->uStatIntTXQE++;
+ /// @todo uncomment: e1kRaiseInterrupt(pDevIns, pThis, ICR_TXQE);
+ /*
+ * Release the lock.
+ */
+ if (pDrv)
+ pDrv->pfnEndXmit(pDrv);
+ e1kCsTxLeave(pThis);
+ }
+
+ return rc;
+}
+
+#else /* E1K_WITH_TXD_CACHE */
+
+static void e1kDumpTxDCache(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pTxdc)
+{
+ unsigned i, cDescs = pTxdc->tdlen / sizeof(E1KTXDESC);
+ uint32_t tdh = pTxdc->tdh;
+ LogRel(("E1000: -- Transmit Descriptors (%d total) --\n", cDescs));
+ for (i = 0; i < cDescs; ++i)
+ {
+ E1KTXDESC desc;
+ PDMDevHlpPCIPhysRead(pDevIns , e1kDescAddr(TDBAH, TDBAL, i), &desc, sizeof(desc));
+ if (i == tdh)
+ LogRel(("E1000: >>> "));
+ LogRel(("E1000: %RGp: %R[e1ktxd]\n", e1kDescAddr(TDBAH, TDBAL, i), &desc));
+ }
+ LogRel(("E1000: -- Transmit Descriptors in Cache (at %d (TDH %d)/ fetched %d / max %d) --\n",
+ pThis->iTxDCurrent, pTxdc->tdh, pThis->nTxDFetched, E1K_TXD_CACHE_SIZE));
+ if (tdh > pThis->iTxDCurrent)
+ tdh -= pThis->iTxDCurrent;
+ else
+ tdh = cDescs + tdh - pThis->iTxDCurrent;
+ for (i = 0; i < pThis->nTxDFetched; ++i)
+ {
+ if (i == pThis->iTxDCurrent)
+ LogRel(("E1000: >>> "));
+ if (cDescs)
+ LogRel(("E1000: %RGp: %R[e1ktxd]\n", e1kDescAddr(TDBAH, TDBAL, tdh++ % cDescs), &pThis->aTxDescriptors[i]));
+ else
+ LogRel(("E1000: <lost>: %R[e1ktxd]\n", &pThis->aTxDescriptors[i]));
+ }
+}
+
+/**
+ * Transmit pending descriptors.
+ *
+ * @returns VBox status code. VERR_TRY_AGAIN is returned if we're busy.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The E1000 state.
+ * @param fOnWorkerThread Whether we're on a worker thread or on an EMT.
+ */
+static int e1kXmitPending(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread)
+{
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ int rc = VINF_SUCCESS;
+
+ /* Check if transmitter is enabled. */
+ if (!(TCTL & TCTL_EN))
+ return VINF_SUCCESS;
+ /*
+ * Grab the xmit lock of the driver as well as the E1K device state.
+ */
+ PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
+ if (pDrv)
+ {
+ rc = pDrv->pfnBeginXmit(pDrv, fOnWorkerThread);
+ if (RT_FAILURE(rc))
+ return rc;
+ }
+
+ /*
+ * Process all pending descriptors.
+ * Note! Do not process descriptors in locked state
+ */
+ rc = e1kCsTxEnter(pThis, VERR_SEM_BUSY);
+ if (RT_LIKELY(rc == VINF_SUCCESS && (TCTL & TCTL_EN)))
+ {
+ E1KTXDC txdc;
+ bool fTxContextValid = e1kUpdateTxDContext(pDevIns, pThis, &txdc);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
+ /*
+ * fIncomplete is set whenever we try to fetch additional descriptors
+ * for an incomplete packet. If fail to locate a complete packet on
+ * the next iteration we need to reset the cache or we risk to get
+ * stuck in this loop forever.
+ */
+ bool fIncomplete = false;
+ while (fTxContextValid && !pThis->fLocked && e1kTxDLazyLoad(pDevIns, pThis, &txdc))
+ {
+ while (e1kLocateTxPacket(pThis, &txdc))
+ {
+ Log4(("%s e1kXmitPending: Located packet at %d. Next packet at %d\n",
+ pThis->szPrf, pThis->iTxDCurrent, txdc.nextPacket));
+ fIncomplete = false;
+ /* Found a complete packet, allocate it. */
+ rc = e1kXmitAllocBuf(pThis, pThisCC, pThis->fGSO);
+ /* If we're out of bandwidth we'll come back later. */
+ if (RT_FAILURE(rc))
+ goto out;
+ /* Copy the packet to allocated buffer and send it. */
+ rc = e1kXmitPacket(pDevIns, pThis, fOnWorkerThread, &txdc);
+ /* If we're out of bandwidth we'll come back later. */
+ if (RT_FAILURE(rc))
+ goto out;
+ }
+ uint8_t u8Remain = pThis->nTxDFetched - pThis->iTxDCurrent;
+ if (RT_UNLIKELY(fIncomplete))
+ {
+ static bool fTxDCacheDumped = false;
+ /*
+ * The descriptor cache is full, but we were unable to find
+ * a complete packet in it. Drop the cache and hope that
+ * the guest driver can recover from network card error.
+ */
+ LogRel(("%s: No complete packets in%s TxD cache! "
+ "Fetched=%d, current=%d, TX len=%d.\n",
+ pThis->szPrf,
+ u8Remain == E1K_TXD_CACHE_SIZE ? " full" : "",
+ pThis->nTxDFetched, pThis->iTxDCurrent,
+ e1kGetTxLen(&txdc)));
+ if (!fTxDCacheDumped)
+ {
+ fTxDCacheDumped = true;
+ e1kDumpTxDCache(pDevIns, pThis, &txdc);
+ }
+ pThis->iTxDCurrent = pThis->nTxDFetched = 0;
+ /*
+ * Returning an error at this point means Guru in R0
+ * (see @bugref{6428}).
+ */
+# ifdef IN_RING3
+ rc = VERR_NET_INCOMPLETE_TX_PACKET;
+# else /* !IN_RING3 */
+ rc = VINF_IOM_R3_MMIO_WRITE;
+# endif /* !IN_RING3 */
+ goto out;
+ }
+ if (u8Remain > 0)
+ {
+ Log4(("%s Incomplete packet at %d. Already fetched %d, "
+ "%d more are available\n",
+ pThis->szPrf, pThis->iTxDCurrent, u8Remain,
+ e1kGetTxLen(&txdc) - u8Remain));
+
+ /*
+ * A packet was partially fetched. Move incomplete packet to
+ * the beginning of cache buffer, then load more descriptors.
+ */
+ memmove(pThis->aTxDescriptors,
+ &pThis->aTxDescriptors[pThis->iTxDCurrent],
+ u8Remain * sizeof(E1KTXDESC));
+ pThis->iTxDCurrent = 0;
+ pThis->nTxDFetched = u8Remain;
+ e1kTxDLoadMore(pDevIns, pThis, &txdc);
+ fIncomplete = true;
+ }
+ else
+ pThis->nTxDFetched = 0;
+ pThis->iTxDCurrent = 0;
+ }
+ if (!pThis->fLocked && GET_BITS(TXDCTL, LWTHRESH) == 0)
+ {
+ E1kLog2(("%s Out of transmit descriptors, raise ICR.TXD_LOW\n",
+ pThis->szPrf));
+ e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXD_LOW);
+ }
+out:
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
+
+ /// @todo uncomment: pThis->uStatIntTXQE++;
+ /// @todo uncomment: e1kRaiseInterrupt(pDevIns, pThis, ICR_TXQE);
+
+ e1kCsTxLeave(pThis);
+ }
+
+
+ /*
+ * Release the lock.
+ */
+ if (pDrv)
+ pDrv->pfnEndXmit(pDrv);
+ return rc;
+}
+
+#endif /* E1K_WITH_TXD_CACHE */
+#ifdef IN_RING3
+
+/**
+ * @interface_method_impl{PDMINETWORKDOWN,pfnXmitPending}
+ */
+static DECLCALLBACK(void) e1kR3NetworkDown_XmitPending(PPDMINETWORKDOWN pInterface)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkDown);
+ PE1KSTATE pThis = pThisCC->pShared;
+ /* Resume suspended transmission */
+ STATUS &= ~STATUS_TXOFF;
+ e1kXmitPending(pThisCC->pDevInsR3, pThis, true /*fOnWorkerThread*/);
+}
+
+/**
+ * @callback_method_impl{FNPDMTASKDEV,
+ * Executes e1kXmitPending at the behest of ring-0/raw-mode.}
+ * @note Not executed on EMT.
+ */
+static DECLCALLBACK(void) e1kR3TxTaskCallback(PPDMDEVINS pDevIns, void *pvUser)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ E1kLog2(("%s e1kR3TxTaskCallback:\n", pThis->szPrf));
+
+ int rc = e1kXmitPending(pDevIns, pThis, false /*fOnWorkerThread*/);
+ AssertMsg(RT_SUCCESS(rc) || rc == VERR_TRY_AGAIN || rc == VERR_NET_DOWN, ("%Rrc\n", rc));
+
+ RT_NOREF(rc, pvUser);
+}
+
+#endif /* IN_RING3 */
+
+/**
+ * Write handler for Transmit Descriptor Tail register.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+static int e1kRegWriteTDT(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ int rc = e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
+
+ /* All descriptors starting with head and not including tail belong to us. */
+ /* Process them. */
+ E1kLog2(("%s e1kRegWriteTDT: TDBAL=%08x, TDBAH=%08x, TDLEN=%08x, TDH=%08x, TDT=%08x\n",
+ pThis->szPrf, TDBAL, TDBAH, TDLEN, TDH, TDT));
+
+ /* Compose a temporary TX context, breaking TX CS rule, for debugging purposes. */
+ /* If we decide to transmit, the TX critical section will be entered later in e1kXmitPending(). */
+ E1KTXDC txdc;
+ txdc.tdlen = TDLEN;
+ txdc.tdh = TDH;
+ txdc.tdt = TDT;
+ /* Ignore TDT writes when the link is down. */
+ if (txdc.tdh != txdc.tdt && (STATUS & STATUS_LU))
+ {
+ Log5(("E1000: TDT write: TDH=%08x, TDT=%08x, %d descriptors to process\n", txdc.tdh, txdc.tdt, e1kGetTxLen(&txdc)));
+ E1kLog(("%s e1kRegWriteTDT: %d descriptors to process\n",
+ pThis->szPrf, e1kGetTxLen(&txdc)));
+
+ /* Transmit pending packets if possible, defer it if we cannot do it
+ in the current context. */
+#ifdef E1K_TX_DELAY
+ rc = e1kCsTxEnter(pThis, VERR_SEM_BUSY);
+ if (RT_LIKELY(rc == VINF_SUCCESS))
+ {
+ if (!PDMDevInsTimerIsActive(pDevIns, pThis->hTXDTimer))
+ {
+# ifdef E1K_INT_STATS
+ pThis->u64ArmedAt = RTTimeNanoTS();
+# endif
+ e1kArmTimer(pDevIns, pThis, pThis->hTXDTimer, E1K_TX_DELAY);
+ }
+ E1K_INC_ISTAT_CNT(pThis->uStatTxDelayed);
+ e1kCsTxLeave(pThis);
+ return rc;
+ }
+ /* We failed to enter the TX critical section -- transmit as usual. */
+#endif /* E1K_TX_DELAY */
+#ifndef IN_RING3
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ if (!pThisCC->CTX_SUFF(pDrv))
+ {
+ PDMDevHlpTaskTrigger(pDevIns, pThis->hTxTask);
+ rc = VINF_SUCCESS;
+ }
+ else
+#endif
+ {
+ rc = e1kXmitPending(pDevIns, pThis, false /*fOnWorkerThread*/);
+ if (rc == VERR_TRY_AGAIN)
+ rc = VINF_SUCCESS;
+#ifndef IN_RING3
+ else if (rc == VERR_SEM_BUSY)
+ rc = VINF_IOM_R3_MMIO_WRITE;
+#endif
+ AssertRC(rc);
+ }
+ }
+
+ return rc;
+}
+
+/**
+ * Write handler for Multicast Table Array registers.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @thread EMT
+ */
+static int e1kRegWriteMTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(pDevIns);
+ AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->auMTA), VERR_DEV_IO_ERROR);
+ pThis->auMTA[(offset - g_aE1kRegMap[index].offset) / sizeof(pThis->auMTA[0])] = value;
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Read handler for Multicast Table Array registers.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @thread EMT
+ */
+static int e1kRegReadMTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ RT_NOREF_PV(pDevIns);
+ AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->auMTA), VERR_DEV_IO_ERROR);
+ *pu32Value = pThis->auMTA[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->auMTA[0])];
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for Receive Address registers.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @thread EMT
+ */
+static int e1kRegWriteRA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(pDevIns);
+ AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->aRecAddr.au32), VERR_DEV_IO_ERROR);
+ pThis->aRecAddr.au32[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->aRecAddr.au32[0])] = value;
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Read handler for Receive Address registers.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @thread EMT
+ */
+static int e1kRegReadRA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ RT_NOREF_PV(pDevIns);
+ AssertReturn(offset - g_aE1kRegMap[index].offset< sizeof(pThis->aRecAddr.au32), VERR_DEV_IO_ERROR);
+ *pu32Value = pThis->aRecAddr.au32[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->aRecAddr.au32[0])];
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for VLAN Filter Table Array registers.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @thread EMT
+ */
+static int e1kRegWriteVFTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(pDevIns);
+ AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->auVFTA), VINF_SUCCESS);
+ pThis->auVFTA[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->auVFTA[0])] = value;
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Read handler for VLAN Filter Table Array registers.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @thread EMT
+ */
+static int e1kRegReadVFTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ RT_NOREF_PV(pDevIns);
+ AssertReturn(offset - g_aE1kRegMap[index].offset< sizeof(pThis->auVFTA), VERR_DEV_IO_ERROR);
+ *pu32Value = pThis->auVFTA[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->auVFTA[0])];
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Read handler for unimplemented registers.
+ *
+ * Merely reports reads from unimplemented registers.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @thread EMT
+ */
+static int e1kRegReadUnimplemented(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ RT_NOREF(pDevIns, pThis, offset, index);
+ E1kLog(("%s At %08X read (00000000) attempt from unimplemented register %s (%s)\n",
+ pThis->szPrf, offset, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+ *pu32Value = 0;
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Default register read handler with automatic clear operation.
+ *
+ * Retrieves the value of register from register array in device state structure.
+ * Then resets all bits.
+ *
+ * @remarks The 'mask' parameter is simply ignored as masking and shifting is
+ * done in the caller.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @thread EMT
+ */
+static int e1kRegReadAutoClear(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ AssertReturn(index < E1K_NUM_OF_32BIT_REGS, VERR_DEV_IO_ERROR);
+ int rc = e1kRegReadDefault(pDevIns, pThis, offset, index, pu32Value);
+ pThis->auRegs[index] = 0;
+
+ return rc;
+}
+
+/**
+ * Default register read handler.
+ *
+ * Retrieves the value of register from register array in device state structure.
+ * Bits corresponding to 0s in 'readable' mask will always read as 0s.
+ *
+ * @remarks The 'mask' parameter is simply ignored as masking and shifting is
+ * done in the caller.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @thread EMT
+ */
+static int e1kRegReadDefault(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
+{
+ RT_NOREF_PV(pDevIns); RT_NOREF_PV(offset);
+
+ AssertReturn(index < E1K_NUM_OF_32BIT_REGS, VERR_DEV_IO_ERROR);
+ *pu32Value = pThis->auRegs[index] & g_aE1kRegMap[index].readable;
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Write handler for unimplemented registers.
+ *
+ * Merely reports writes to unimplemented registers.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @thread EMT
+ */
+
+ static int e1kRegWriteUnimplemented(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index); RT_NOREF_PV(value);
+
+ E1kLog(("%s At %08X write attempt (%08X) to unimplemented register %s (%s)\n",
+ pThis->szPrf, offset, value, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Default register write handler.
+ *
+ * Stores the value to the register array in device state structure. Only bits
+ * corresponding to 1s both in 'writable' and 'mask' will be stored.
+ *
+ * @returns VBox status code.
+ *
+ * @param pThis The device state structure.
+ * @param offset Register offset in memory-mapped frame.
+ * @param index Register index in register array.
+ * @param value The value to store.
+ * @param mask Used to implement partial writes (8 and 16-bit).
+ * @thread EMT
+ */
+
+static int e1kRegWriteDefault(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
+{
+ RT_NOREF(pDevIns, offset);
+
+ AssertReturn(index < E1K_NUM_OF_32BIT_REGS, VERR_DEV_IO_ERROR);
+ pThis->auRegs[index] = (value & g_aE1kRegMap[index].writable)
+ | (pThis->auRegs[index] & ~g_aE1kRegMap[index].writable);
+
+ return VINF_SUCCESS;
+}
+
+/**
+ * Search register table for matching register.
+ *
+ * @returns Index in the register table or -1 if not found.
+ *
+ * @param offReg Register offset in memory-mapped region.
+ * @thread EMT
+ */
+static int e1kRegLookup(uint32_t offReg)
+{
+
+#if 0
+ int index;
+
+ for (index = 0; index < E1K_NUM_OF_REGS; index++)
+ {
+ if (g_aE1kRegMap[index].offset <= offReg && offReg < g_aE1kRegMap[index].offset + g_aE1kRegMap[index].size)
+ {
+ return index;
+ }
+ }
+#else
+ int iStart = 0;
+ int iEnd = E1K_NUM_OF_BINARY_SEARCHABLE;
+ for (;;)
+ {
+ int i = (iEnd - iStart) / 2 + iStart;
+ uint32_t offCur = g_aE1kRegMap[i].offset;
+ if (offReg < offCur)
+ {
+ if (i == iStart)
+ break;
+ iEnd = i;
+ }
+ else if (offReg >= offCur + g_aE1kRegMap[i].size)
+ {
+ i++;
+ if (i == iEnd)
+ break;
+ iStart = i;
+ }
+ else
+ return i;
+ Assert(iEnd > iStart);
+ }
+
+ for (unsigned i = E1K_NUM_OF_BINARY_SEARCHABLE; i < RT_ELEMENTS(g_aE1kRegMap); i++)
+ if (offReg - g_aE1kRegMap[i].offset < g_aE1kRegMap[i].size)
+ return (int)i;
+
+# ifdef VBOX_STRICT
+ for (unsigned i = 0; i < RT_ELEMENTS(g_aE1kRegMap); i++)
+ Assert(offReg - g_aE1kRegMap[i].offset >= g_aE1kRegMap[i].size);
+# endif
+
+#endif
+
+ return -1;
+}
+
+/**
+ * Handle unaligned register read operation.
+ *
+ * Looks up and calls appropriate handler.
+ *
+ * @returns VBox status code.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param offReg Register offset in memory-mapped frame.
+ * @param pv Where to store the result.
+ * @param cb Number of bytes to read.
+ * @thread EMT
+ * @remarks IOM takes care of unaligned and small reads via MMIO. For I/O port
+ * accesses we have to take care of that ourselves.
+ */
+static int e1kRegReadUnaligned(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offReg, void *pv, uint32_t cb)
+{
+ uint32_t u32 = 0;
+ uint32_t shift;
+ int rc = VINF_SUCCESS;
+ int index = e1kRegLookup(offReg);
+#ifdef LOG_ENABLED
+ char buf[9];
+#endif
+
+ /*
+ * From the spec:
+ * For registers that should be accessed as 32-bit double words, partial writes (less than a 32-bit
+ * double word) is ignored. Partial reads return all 32 bits of data regardless of the byte enables.
+ */
+
+ /*
+ * To be able to read bytes and short word we convert them to properly
+ * shifted 32-bit words and masks. The idea is to keep register-specific
+ * handlers simple. Most accesses will be 32-bit anyway.
+ */
+ uint32_t mask;
+ switch (cb)
+ {
+ case 4: mask = 0xFFFFFFFF; break;
+ case 2: mask = 0x0000FFFF; break;
+ case 1: mask = 0x000000FF; break;
+ default:
+ return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "unsupported op size: offset=%#10x cb=%#10x\n", offReg, cb);
+ }
+ if (index >= 0)
+ {
+ RT_UNTRUSTED_VALIDATED_FENCE(); /* paranoia because of port I/O. */
+ if (g_aE1kRegMap[index].readable)
+ {
+ /* Make the mask correspond to the bits we are about to read. */
+ shift = (offReg - g_aE1kRegMap[index].offset) % sizeof(uint32_t) * 8;
+ mask <<= shift;
+ if (!mask)
+ return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Zero mask: offset=%#10x cb=%#10x\n", offReg, cb);
+ /*
+ * Read it. Pass the mask so the handler knows what has to be read.
+ * Mask out irrelevant bits.
+ */
+ //e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
+ //pThis->fDelayInts = false;
+ //pThis->iStatIntLost += pThis->iStatIntLostOne;
+ //pThis->iStatIntLostOne = 0;
+ rc = g_aE1kRegMap[index].pfnRead(pDevIns, pThis, offReg & 0xFFFFFFFC, (uint32_t)index, &u32);
+ u32 &= mask;
+ //e1kCsLeave(pThis);
+ E1kLog2(("%s At %08X read %s from %s (%s)\n",
+ pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf), g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+ Log6(("%s At %08X read %s from %s (%s) [UNALIGNED]\n",
+ pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf), g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+ /* Shift back the result. */
+ u32 >>= shift;
+ }
+ else
+ E1kLog(("%s At %08X read (%s) attempt from write-only register %s (%s)\n",
+ pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf), g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+ if (IOM_SUCCESS(rc))
+ STAM_COUNTER_INC(&pThis->aStatRegReads[index]);
+ }
+ else
+ E1kLog(("%s At %08X read (%s) attempt from non-existing register\n",
+ pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf)));
+
+ memcpy(pv, &u32, cb);
+ return rc;
+}
+
+/**
+ * Handle 4 byte aligned and sized read operation.
+ *
+ * Looks up and calls appropriate handler.
+ *
+ * @returns VBox status code.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param offReg Register offset in memory-mapped frame.
+ * @param pu32 Where to store the result.
+ * @thread EMT
+ */
+static VBOXSTRICTRC e1kRegReadAlignedU32(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offReg, uint32_t *pu32)
+{
+ Assert(!(offReg & 3));
+
+ /*
+ * Lookup the register and check that it's readable.
+ */
+ VBOXSTRICTRC rc = VINF_SUCCESS;
+ int idxReg = e1kRegLookup(offReg);
+ if (RT_LIKELY(idxReg >= 0))
+ {
+ RT_UNTRUSTED_VALIDATED_FENCE(); /* paranoia because of port I/O. */
+ if (RT_UNLIKELY(g_aE1kRegMap[idxReg].readable))
+ {
+ /*
+ * Read it. Pass the mask so the handler knows what has to be read.
+ * Mask out irrelevant bits.
+ */
+ //e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
+ //pThis->fDelayInts = false;
+ //pThis->iStatIntLost += pThis->iStatIntLostOne;
+ //pThis->iStatIntLostOne = 0;
+ rc = g_aE1kRegMap[idxReg].pfnRead(pDevIns, pThis, offReg & 0xFFFFFFFC, (uint32_t)idxReg, pu32);
+ //e1kCsLeave(pThis);
+ Log6(("%s At %08X read %08X from %s (%s)\n",
+ pThis->szPrf, offReg, *pu32, g_aE1kRegMap[idxReg].abbrev, g_aE1kRegMap[idxReg].name));
+ if (IOM_SUCCESS(rc))
+ STAM_COUNTER_INC(&pThis->aStatRegReads[idxReg]);
+ }
+ else
+ E1kLog(("%s At %08X read attempt from non-readable register %s (%s)\n",
+ pThis->szPrf, offReg, g_aE1kRegMap[idxReg].abbrev, g_aE1kRegMap[idxReg].name));
+ }
+ else
+ E1kLog(("%s At %08X read attempt from non-existing register\n", pThis->szPrf, offReg));
+ return rc;
+}
+
+/**
+ * Handle 4 byte sized and aligned register write operation.
+ *
+ * Looks up and calls appropriate handler.
+ *
+ * @returns VBox status code.
+ *
+ * @param pDevIns The device instance.
+ * @param pThis The device state structure.
+ * @param offReg Register offset in memory-mapped frame.
+ * @param u32Value The value to write.
+ * @thread EMT
+ */
+static VBOXSTRICTRC e1kRegWriteAlignedU32(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offReg, uint32_t u32Value)
+{
+ VBOXSTRICTRC rc = VINF_SUCCESS;
+ int index = e1kRegLookup(offReg);
+ if (RT_LIKELY(index >= 0))
+ {
+ RT_UNTRUSTED_VALIDATED_FENCE(); /* paranoia because of port I/O. */
+ if (RT_LIKELY(g_aE1kRegMap[index].writable))
+ {
+ /*
+ * Write it. Pass the mask so the handler knows what has to be written.
+ * Mask out irrelevant bits.
+ */
+ Log6(("%s At %08X write %08X to %s (%s)\n",
+ pThis->szPrf, offReg, u32Value, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+ //e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
+ //pThis->fDelayInts = false;
+ //pThis->iStatIntLost += pThis->iStatIntLostOne;
+ //pThis->iStatIntLostOne = 0;
+ rc = g_aE1kRegMap[index].pfnWrite(pDevIns, pThis, offReg, (uint32_t)index, u32Value);
+ //e1kCsLeave(pThis);
+ }
+ else
+ E1kLog(("%s At %08X write attempt (%08X) to read-only register %s (%s)\n",
+ pThis->szPrf, offReg, u32Value, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
+ if (IOM_SUCCESS(rc))
+ STAM_COUNTER_INC(&pThis->aStatRegWrites[index]);
+ }
+ else
+ E1kLog(("%s At %08X write attempt (%08X) to non-existing register\n",
+ pThis->szPrf, offReg, u32Value));
+ return rc;
+}
+
+
+/* -=-=-=-=- MMIO and I/O Port Callbacks -=-=-=-=- */
+
+/**
+ * @callback_method_impl{FNIOMMMIONEWREAD}
+ */
+static DECLCALLBACK(VBOXSTRICTRC) e1kMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void *pv, uint32_t cb)
+{
+ RT_NOREF2(pvUser, cb);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatMMIORead), a);
+
+ Assert(off < E1K_MM_SIZE);
+ Assert(cb == 4);
+ Assert(!(off & 3));
+
+ VBOXSTRICTRC rcStrict = e1kRegReadAlignedU32(pDevIns, pThis, (uint32_t)off, (uint32_t *)pv);
+
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatMMIORead), a);
+ return rcStrict;
+}
+
+/**
+ * @callback_method_impl{FNIOMMMIONEWWRITE}
+ */
+static DECLCALLBACK(VBOXSTRICTRC) e1kMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void const *pv, uint32_t cb)
+{
+ RT_NOREF2(pvUser, cb);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatMMIOWrite), a);
+
+ Assert(off < E1K_MM_SIZE);
+ Assert(cb == 4);
+ Assert(!(off & 3));
+
+ VBOXSTRICTRC rcStrict = e1kRegWriteAlignedU32(pDevIns, pThis, (uint32_t)off, *(uint32_t const *)pv);
+
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatMMIOWrite), a);
+ return rcStrict;
+}
+
+/**
+ * @callback_method_impl{FNIOMIOPORTNEWIN}
+ */
+static DECLCALLBACK(VBOXSTRICTRC) e1kIOPortIn(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ VBOXSTRICTRC rc;
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIORead), a);
+ RT_NOREF_PV(pvUser);
+
+ if (RT_LIKELY(cb == 4))
+ switch (offPort)
+ {
+ case 0x00: /* IOADDR */
+ *pu32 = pThis->uSelectedReg;
+ Log9(("%s e1kIOPortIn: IOADDR(0), selecting register %#010x, val=%#010x\n", pThis->szPrf, pThis->uSelectedReg, *pu32));
+ rc = VINF_SUCCESS;
+ break;
+
+ case 0x04: /* IODATA */
+ if (!(pThis->uSelectedReg & 3))
+ rc = e1kRegReadAlignedU32(pDevIns, pThis, pThis->uSelectedReg, pu32);
+ else /** @todo r=bird: I wouldn't be surprised if this unaligned branch wasn't necessary. */
+ rc = e1kRegReadUnaligned(pDevIns, pThis, pThis->uSelectedReg, pu32, cb);
+ if (rc == VINF_IOM_R3_MMIO_READ)
+ rc = VINF_IOM_R3_IOPORT_READ;
+ Log9(("%s e1kIOPortIn: IODATA(4), reading from selected register %#010x, val=%#010x\n", pThis->szPrf, pThis->uSelectedReg, *pu32));
+ break;
+
+ default:
+ E1kLog(("%s e1kIOPortIn: invalid port %#010x\n", pThis->szPrf, offPort));
+ /** @todo r=bird: Check what real hardware returns here. */
+ //rc = VERR_IOM_IOPORT_UNUSED; /* Why not? */
+ rc = VINF_IOM_MMIO_UNUSED_00; /* used to return VINF_SUCCESS and not touch *pu32, which amounted to this. */
+ break;
+ }
+ else
+ {
+ E1kLog(("%s e1kIOPortIn: invalid op size: offPort=%RTiop cb=%08x", pThis->szPrf, offPort, cb));
+ rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "%s e1kIOPortIn: invalid op size: offPort=%RTiop cb=%08x\n", pThis->szPrf, offPort, cb);
+ *pu32 = 0; /** @todo r=bird: Check what real hardware returns here. (Didn't used to set a value here, picked zero as that's what we'd end up in most cases.) */
+ }
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIORead), a);
+ return rc;
+}
+
+
+/**
+ * @callback_method_impl{FNIOMIOPORTNEWOUT}
+ */
+static DECLCALLBACK(VBOXSTRICTRC) e1kIOPortOut(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ VBOXSTRICTRC rc;
+ STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIOWrite), a);
+ RT_NOREF_PV(pvUser);
+
+ Log9(("%s e1kIOPortOut: offPort=%RTiop value=%08x\n", pThis->szPrf, offPort, u32));
+ if (RT_LIKELY(cb == 4))
+ {
+ switch (offPort)
+ {
+ case 0x00: /* IOADDR */
+ pThis->uSelectedReg = u32;
+ Log9(("%s e1kIOPortOut: IOADDR(0), selected register %08x\n", pThis->szPrf, pThis->uSelectedReg));
+ rc = VINF_SUCCESS;
+ break;
+
+ case 0x04: /* IODATA */
+ Log9(("%s e1kIOPortOut: IODATA(4), writing to selected register %#010x, value=%#010x\n", pThis->szPrf, pThis->uSelectedReg, u32));
+ if (RT_LIKELY(!(pThis->uSelectedReg & 3)))
+ {
+ rc = e1kRegWriteAlignedU32(pDevIns, pThis, pThis->uSelectedReg, u32);
+ if (rc == VINF_IOM_R3_MMIO_WRITE)
+ rc = VINF_IOM_R3_IOPORT_WRITE;
+ }
+ else
+ rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS,
+ "Spec violation: misaligned offset: %#10x, ignored.\n", pThis->uSelectedReg);
+ break;
+
+ default:
+ E1kLog(("%s e1kIOPortOut: invalid port %#010x\n", pThis->szPrf, offPort));
+ rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "invalid port %#010x\n", offPort);
+ }
+ }
+ else
+ {
+ E1kLog(("%s e1kIOPortOut: invalid op size: offPort=%RTiop cb=%08x\n", pThis->szPrf, offPort, cb));
+ rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "%s: invalid op size: offPort=%RTiop cb=%#x\n", pThis->szPrf, offPort, cb);
+ }
+
+ STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIOWrite), a);
+ return rc;
+}
+
+#ifdef IN_RING3
+
+/**
+ * Dump complete device state to log.
+ *
+ * @param pThis Pointer to device state.
+ */
+static void e1kDumpState(PE1KSTATE pThis)
+{
+ RT_NOREF(pThis);
+ for (int i = 0; i < E1K_NUM_OF_32BIT_REGS; ++i)
+ E1kLog2(("%s: %8.8s = %08x\n", pThis->szPrf, g_aE1kRegMap[i].abbrev, pThis->auRegs[i]));
+# ifdef E1K_INT_STATS
+ LogRel(("%s: Interrupt attempts: %d\n", pThis->szPrf, pThis->uStatIntTry));
+ LogRel(("%s: Interrupts raised : %d\n", pThis->szPrf, pThis->uStatInt));
+ LogRel(("%s: Interrupts lowered: %d\n", pThis->szPrf, pThis->uStatIntLower));
+ LogRel(("%s: ICR outside ISR : %d\n", pThis->szPrf, pThis->uStatNoIntICR));
+ LogRel(("%s: IMS raised ints : %d\n", pThis->szPrf, pThis->uStatIntIMS));
+ LogRel(("%s: Interrupts skipped: %d\n", pThis->szPrf, pThis->uStatIntSkip));
+ LogRel(("%s: Masked interrupts : %d\n", pThis->szPrf, pThis->uStatIntMasked));
+ LogRel(("%s: Early interrupts : %d\n", pThis->szPrf, pThis->uStatIntEarly));
+ LogRel(("%s: Late interrupts : %d\n", pThis->szPrf, pThis->uStatIntLate));
+ LogRel(("%s: Lost interrupts : %d\n", pThis->szPrf, pThis->iStatIntLost));
+ LogRel(("%s: Interrupts by RX : %d\n", pThis->szPrf, pThis->uStatIntRx));
+ LogRel(("%s: Interrupts by TX : %d\n", pThis->szPrf, pThis->uStatIntTx));
+ LogRel(("%s: Interrupts by ICS : %d\n", pThis->szPrf, pThis->uStatIntICS));
+ LogRel(("%s: Interrupts by RDTR: %d\n", pThis->szPrf, pThis->uStatIntRDTR));
+ LogRel(("%s: Interrupts by RDMT: %d\n", pThis->szPrf, pThis->uStatIntRXDMT0));
+ LogRel(("%s: Interrupts by TXQE: %d\n", pThis->szPrf, pThis->uStatIntTXQE));
+ LogRel(("%s: TX int delay asked: %d\n", pThis->szPrf, pThis->uStatTxIDE));
+ LogRel(("%s: TX delayed: %d\n", pThis->szPrf, pThis->uStatTxDelayed));
+ LogRel(("%s: TX delay expired: %d\n", pThis->szPrf, pThis->uStatTxDelayExp));
+ LogRel(("%s: TX no report asked: %d\n", pThis->szPrf, pThis->uStatTxNoRS));
+ LogRel(("%s: TX abs timer expd : %d\n", pThis->szPrf, pThis->uStatTAD));
+ LogRel(("%s: TX int timer expd : %d\n", pThis->szPrf, pThis->uStatTID));
+ LogRel(("%s: RX abs timer expd : %d\n", pThis->szPrf, pThis->uStatRAD));
+ LogRel(("%s: RX int timer expd : %d\n", pThis->szPrf, pThis->uStatRID));
+ LogRel(("%s: TX CTX descriptors: %d\n", pThis->szPrf, pThis->uStatDescCtx));
+ LogRel(("%s: TX DAT descriptors: %d\n", pThis->szPrf, pThis->uStatDescDat));
+ LogRel(("%s: TX LEG descriptors: %d\n", pThis->szPrf, pThis->uStatDescLeg));
+ LogRel(("%s: Received frames : %d\n", pThis->szPrf, pThis->uStatRxFrm));
+ LogRel(("%s: Transmitted frames: %d\n", pThis->szPrf, pThis->uStatTxFrm));
+ LogRel(("%s: TX frames up to 1514: %d\n", pThis->szPrf, pThis->uStatTx1514));
+ LogRel(("%s: TX frames up to 2962: %d\n", pThis->szPrf, pThis->uStatTx2962));
+ LogRel(("%s: TX frames up to 4410: %d\n", pThis->szPrf, pThis->uStatTx4410));
+ LogRel(("%s: TX frames up to 5858: %d\n", pThis->szPrf, pThis->uStatTx5858));
+ LogRel(("%s: TX frames up to 7306: %d\n", pThis->szPrf, pThis->uStatTx7306));
+ LogRel(("%s: TX frames up to 8754: %d\n", pThis->szPrf, pThis->uStatTx8754));
+ LogRel(("%s: TX frames up to 16384: %d\n", pThis->szPrf, pThis->uStatTx16384));
+ LogRel(("%s: TX frames up to 32768: %d\n", pThis->szPrf, pThis->uStatTx32768));
+ LogRel(("%s: Larger TX frames : %d\n", pThis->szPrf, pThis->uStatTxLarge));
+ LogRel(("%s: Max TX Delay : %lld\n", pThis->szPrf, pThis->uStatMaxTxDelay));
+# endif /* E1K_INT_STATS */
+}
+
+
+/* -=-=-=-=- PDMINETWORKDOWN -=-=-=-=- */
+
+/**
+ * Check if the device can receive data now.
+ * This must be called before the pfnRecieve() method is called.
+ *
+ * @returns VBox status code.
+ * @retval VERR_NET_NO_BUFFER_SPACE if we cannot receive.
+ * @param pDevIns The device instance.
+ * @param pThis The instance data.
+ * @thread EMT
+ */
+static int e1kR3CanReceive(PPDMDEVINS pDevIns, PE1KSTATE pThis)
+{
+# ifndef E1K_WITH_RXD_CACHE
+ size_t cb;
+
+ e1kCsRxEnterReturn(pThis);
+
+ if (RT_UNLIKELY(RDLEN == sizeof(E1KRXDESC)))
+ {
+ E1KRXDESC desc;
+ PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), &desc, sizeof(desc));
+ if (desc.status.fDD)
+ cb = 0;
+ else
+ cb = pThis->u16RxBSize;
+ }
+ else if (RDH < RDT)
+ cb = (RDT - RDH) * pThis->u16RxBSize;
+ else if (RDH > RDT)
+ cb = (RDLEN / sizeof(E1KRXDESC) - RDH + RDT) * pThis->u16RxBSize;
+ else
+ {
+ cb = 0;
+ E1kLogRel(("E1000: OUT of RX descriptors!\n"));
+ }
+ E1kLog2(("%s e1kR3CanReceive: at exit RDH=%d RDT=%d RDLEN=%d u16RxBSize=%d cb=%lu\n",
+ pThis->szPrf, RDH, RDT, RDLEN, pThis->u16RxBSize, cb));
+
+ e1kCsRxLeave(pThis);
+ return cb > 0 ? VINF_SUCCESS : VERR_NET_NO_BUFFER_SPACE;
+# else /* E1K_WITH_RXD_CACHE */
+
+ e1kCsRxEnterReturn(pThis);
+
+ E1KRXDC rxdc;
+ if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kR3CanReceive")))
+ {
+ e1kCsRxLeave(pThis);
+ E1kLog(("%s e1kR3CanReceive: failed to update Rx context, returning VERR_NET_NO_BUFFER_SPACE\n", pThis->szPrf));
+ return VERR_NET_NO_BUFFER_SPACE;
+ }
+
+ int rc = VINF_SUCCESS;
+ if (RT_UNLIKELY(rxdc.rdlen == sizeof(E1KRXDESC)))
+ {
+ E1KRXDESC desc;
+ PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, rxdc.rdh), &desc, sizeof(desc));
+ if (desc.status.fDD)
+ rc = VERR_NET_NO_BUFFER_SPACE;
+ }
+ else if (e1kRxDIsCacheEmpty(pThis) && rxdc.rdh == rxdc.rdt)
+ {
+ /* Cache is empty, so is the RX ring. */
+ rc = VERR_NET_NO_BUFFER_SPACE;
+ }
+ E1kLog2(("%s e1kR3CanReceive: at exit in_cache=%d RDH=%d RDT=%d RDLEN=%d u16RxBSize=%d rc=%Rrc\n", pThis->szPrf,
+ e1kRxDInCache(pThis), rxdc.rdh, rxdc.rdt, rxdc.rdlen, pThis->u16RxBSize, rc));
+
+ e1kCsRxLeave(pThis);
+ return rc;
+# endif /* E1K_WITH_RXD_CACHE */
+}
+
+/**
+ * @interface_method_impl{PDMINETWORKDOWN,pfnWaitReceiveAvail}
+ */
+static DECLCALLBACK(int) e1kR3NetworkDown_WaitReceiveAvail(PPDMINETWORKDOWN pInterface, RTMSINTERVAL cMillies)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkDown);
+ PE1KSTATE pThis = pThisCC->pShared;
+ PPDMDEVINS pDevIns = pThisCC->pDevInsR3;
+
+ int rc = e1kR3CanReceive(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 = e1kR3CanReceive(pDevIns, pThis);
+ if (RT_SUCCESS(rc2))
+ {
+ rc = VINF_SUCCESS;
+ break;
+ }
+ E1kLogRel(("E1000: e1kR3NetworkDown_WaitReceiveAvail: waiting cMillies=%u...\n", cMillies));
+ E1kLog(("%s: e1kR3NetworkDown_WaitReceiveAvail: waiting cMillies=%u...\n", pThis->szPrf, cMillies));
+ PDMDevHlpSUPSemEventWaitNoResume(pDevIns, pThis->hEventMoreRxDescAvail, cMillies);
+ }
+ STAM_PROFILE_STOP(&pThis->StatRxOverflow, a);
+ ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, false);
+
+ return rc;
+}
+
+
+/**
+ * Matches the packet addresses against Receive Address table. Looks for
+ * exact matches only.
+ *
+ * @returns true if address matches.
+ * @param pThis Pointer to the state structure.
+ * @param pvBuf The ethernet packet.
+ * @param cb Number of bytes available in the packet.
+ * @thread EMT
+ */
+static bool e1kPerfectMatch(PE1KSTATE pThis, const void *pvBuf)
+{
+ for (unsigned i = 0; i < RT_ELEMENTS(pThis->aRecAddr.array); i++)
+ {
+ E1KRAELEM* ra = pThis->aRecAddr.array + i;
+
+ /* Valid address? */
+ if (ra->ctl & RA_CTL_AV)
+ {
+ Assert((ra->ctl & RA_CTL_AS) < 2);
+ //unsigned char *pAddr = (unsigned char*)pvBuf + sizeof(ra->addr)*(ra->ctl & RA_CTL_AS);
+ //E1kLog3(("%s Matching %02x:%02x:%02x:%02x:%02x:%02x against %02x:%02x:%02x:%02x:%02x:%02x...\n",
+ // pThis->szPrf, pAddr[0], pAddr[1], pAddr[2], pAddr[3], pAddr[4], pAddr[5],
+ // ra->addr[0], ra->addr[1], ra->addr[2], ra->addr[3], ra->addr[4], ra->addr[5]));
+ /*
+ * Address Select:
+ * 00b = Destination address
+ * 01b = Source address
+ * 10b = Reserved
+ * 11b = Reserved
+ * Since ethernet header is (DA, SA, len) we can use address
+ * select as index.
+ */
+ if (memcmp((char*)pvBuf + sizeof(ra->addr)*(ra->ctl & RA_CTL_AS),
+ ra->addr, sizeof(ra->addr)) == 0)
+ return true;
+ }
+ }
+
+ return false;
+}
+
+/**
+ * Matches the packet addresses against Multicast Table Array.
+ *
+ * @remarks This is imperfect match since it matches not exact address but
+ * a subset of addresses.
+ *
+ * @returns true if address matches.
+ * @param pThis Pointer to the state structure.
+ * @param pvBuf The ethernet packet.
+ * @param cb Number of bytes available in the packet.
+ * @thread EMT
+ */
+static bool e1kImperfectMatch(PE1KSTATE pThis, const void *pvBuf)
+{
+ /* Get bits 32..47 of destination address */
+ uint16_t u16Bit = ((uint16_t*)pvBuf)[2];
+
+ unsigned offset = GET_BITS(RCTL, MO);
+ /*
+ * offset means:
+ * 00b = bits 36..47
+ * 01b = bits 35..46
+ * 10b = bits 34..45
+ * 11b = bits 32..43
+ */
+ if (offset < 3)
+ u16Bit = u16Bit >> (4 - offset);
+ return ASMBitTest(pThis->auMTA, u16Bit & 0xFFF);
+}
+
+/**
+ * Determines if the packet is to be delivered to upper layer.
+ *
+ * The following filters supported:
+ * - Exact Unicast/Multicast
+ * - Promiscuous Unicast/Multicast
+ * - Multicast
+ * - VLAN
+ *
+ * @returns true if packet is intended for this node.
+ * @param pThis Pointer to the state structure.
+ * @param pvBuf The ethernet packet.
+ * @param cb Number of bytes available in the packet.
+ * @param pStatus Bit field to store status bits.
+ * @thread EMT
+ */
+static bool e1kAddressFilter(PE1KSTATE pThis, const void *pvBuf, size_t cb, E1KRXDST *pStatus)
+{
+ Assert(cb > 14);
+ /* Assume that we fail to pass exact filter. */
+ pStatus->fPIF = false;
+ pStatus->fVP = false;
+ /* Discard oversized packets */
+ if (cb > E1K_MAX_RX_PKT_SIZE)
+ {
+ E1kLog(("%s ERROR: Incoming packet is too big, cb=%d > max=%d\n",
+ pThis->szPrf, cb, E1K_MAX_RX_PKT_SIZE));
+ E1K_INC_CNT32(ROC);
+ return false;
+ }
+ else if (!(RCTL & RCTL_LPE) && cb > 1522)
+ {
+ /* When long packet reception is disabled packets over 1522 are discarded */
+ E1kLog(("%s Discarding incoming packet (LPE=0), cb=%d\n",
+ pThis->szPrf, cb));
+ E1K_INC_CNT32(ROC);
+ return false;
+ }
+
+ uint16_t *u16Ptr = (uint16_t*)pvBuf;
+ /* Compare TPID with VLAN Ether Type */
+ if (RT_BE2H_U16(u16Ptr[6]) == VET)
+ {
+ pStatus->fVP = true;
+ /* Is VLAN filtering enabled? */
+ if (RCTL & RCTL_VFE)
+ {
+ /* It is 802.1q packet indeed, let's filter by VID */
+ if (RCTL & RCTL_CFIEN)
+ {
+ E1kLog3(("%s VLAN filter: VLAN=%d CFI=%d RCTL_CFI=%d\n", pThis->szPrf,
+ E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7])),
+ E1K_SPEC_CFI(RT_BE2H_U16(u16Ptr[7])),
+ !!(RCTL & RCTL_CFI)));
+ if (E1K_SPEC_CFI(RT_BE2H_U16(u16Ptr[7])) != !!(RCTL & RCTL_CFI))
+ {
+ E1kLog2(("%s Packet filter: CFIs do not match in packet and RCTL (%d!=%d)\n",
+ pThis->szPrf, E1K_SPEC_CFI(RT_BE2H_U16(u16Ptr[7])), !!(RCTL & RCTL_CFI)));
+ return false;
+ }
+ }
+ else
+ E1kLog3(("%s VLAN filter: VLAN=%d\n", pThis->szPrf,
+ E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7]))));
+ if (!ASMBitTest(pThis->auVFTA, E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7]))))
+ {
+ E1kLog2(("%s Packet filter: no VLAN match (id=%d)\n",
+ pThis->szPrf, E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7]))));
+ return false;
+ }
+ }
+ }
+ /* Broadcast filtering */
+ if (e1kIsBroadcast(pvBuf) && (RCTL & RCTL_BAM))
+ return true;
+ E1kLog2(("%s Packet filter: not a broadcast\n", pThis->szPrf));
+ if (e1kIsMulticast(pvBuf))
+ {
+ /* Is multicast promiscuous enabled? */
+ if (RCTL & RCTL_MPE)
+ return true;
+ E1kLog2(("%s Packet filter: no promiscuous multicast\n", pThis->szPrf));
+ /* Try perfect matches first */
+ if (e1kPerfectMatch(pThis, pvBuf))
+ {
+ pStatus->fPIF = true;
+ return true;
+ }
+ E1kLog2(("%s Packet filter: no perfect match\n", pThis->szPrf));
+ if (e1kImperfectMatch(pThis, pvBuf))
+ return true;
+ E1kLog2(("%s Packet filter: no imperfect match\n", pThis->szPrf));
+ }
+ else {
+ /* Is unicast promiscuous enabled? */
+ if (RCTL & RCTL_UPE)
+ return true;
+ E1kLog2(("%s Packet filter: no promiscuous unicast\n", pThis->szPrf));
+ if (e1kPerfectMatch(pThis, pvBuf))
+ {
+ pStatus->fPIF = true;
+ return true;
+ }
+ E1kLog2(("%s Packet filter: no perfect match\n", pThis->szPrf));
+ }
+ E1kLog2(("%s Packet filter: packet discarded\n", pThis->szPrf));
+ return false;
+}
+
+/**
+ * @interface_method_impl{PDMINETWORKDOWN,pfnReceive}
+ */
+static DECLCALLBACK(int) e1kR3NetworkDown_Receive(PPDMINETWORKDOWN pInterface, const void *pvBuf, size_t cb)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkDown);
+ PE1KSTATE pThis = pThisCC->pShared;
+ PPDMDEVINS pDevIns = pThisCC->pDevInsR3;
+ int rc = VINF_SUCCESS;
+
+ /*
+ * Drop packets if the VM is not running yet/anymore.
+ */
+ VMSTATE enmVMState = PDMDevHlpVMState(pDevIns);
+ if ( enmVMState != VMSTATE_RUNNING
+ && enmVMState != VMSTATE_RUNNING_LS)
+ {
+ E1kLog(("%s Dropping incoming packet as VM is not running.\n", pThis->szPrf));
+ return VINF_SUCCESS;
+ }
+
+ /* Discard incoming packets in locked state */
+ if (!(RCTL & RCTL_EN) || pThis->fLocked || !(STATUS & STATUS_LU))
+ {
+ E1kLog(("%s Dropping incoming packet as receive operation is disabled.\n", pThis->szPrf));
+ return VINF_SUCCESS;
+ }
+
+ STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
+
+ //e1kR3CsEnterAsserted(pThis);
+
+ e1kPacketDump(pDevIns, pThis, (const uint8_t*)pvBuf, cb, "<-- Incoming");
+
+ /* Update stats */
+ e1kR3CsEnterAsserted(pThis);
+ E1K_INC_CNT32(TPR);
+ E1K_ADD_CNT64(TORL, TORH, cb < 64? 64 : cb);
+ e1kCsLeave(pThis);
+
+ STAM_PROFILE_ADV_START(&pThis->StatReceiveFilter, a);
+ E1KRXDST status;
+ RT_ZERO(status);
+ bool fPassed = e1kAddressFilter(pThis, pvBuf, cb, &status);
+ STAM_PROFILE_ADV_STOP(&pThis->StatReceiveFilter, a);
+ if (fPassed)
+ {
+ rc = e1kHandleRxPacket(pDevIns, pThis, pvBuf, cb, status);
+ }
+ //e1kCsLeave(pThis);
+ STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
+
+ return rc;
+}
+
+
+/* -=-=-=-=- PDMILEDPORTS -=-=-=-=- */
+
+/**
+ * @interface_method_impl{PDMILEDPORTS,pfnQueryStatusLed}
+ */
+static DECLCALLBACK(int) e1kR3QueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
+{
+ if (iLUN == 0)
+ {
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, ILeds);
+ *ppLed = &pThisCC->pShared->led;
+ return VINF_SUCCESS;
+ }
+ return VERR_PDM_LUN_NOT_FOUND;
+}
+
+
+/* -=-=-=-=- PDMINETWORKCONFIG -=-=-=-=- */
+
+/**
+ * @interface_method_impl{PDMINETWORKCONFIG,pfnGetMac}
+ */
+static DECLCALLBACK(int) e1kR3GetMac(PPDMINETWORKCONFIG pInterface, PRTMAC pMac)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkConfig);
+ pThisCC->eeprom.getMac(pMac);
+ return VINF_SUCCESS;
+}
+
+/**
+ * @interface_method_impl{PDMINETWORKCONFIG,pfnGetLinkState}
+ */
+static DECLCALLBACK(PDMNETWORKLINKSTATE) e1kR3GetLinkState(PPDMINETWORKCONFIG pInterface)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkConfig);
+ PE1KSTATE pThis = pThisCC->pShared;
+ if (STATUS & STATUS_LU)
+ return PDMNETWORKLINKSTATE_UP;
+ return PDMNETWORKLINKSTATE_DOWN;
+}
+
+/**
+ * @interface_method_impl{PDMINETWORKCONFIG,pfnSetLinkState}
+ */
+static DECLCALLBACK(int) e1kR3SetLinkState(PPDMINETWORKCONFIG pInterface, PDMNETWORKLINKSTATE enmState)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkConfig);
+ PE1KSTATE pThis = pThisCC->pShared;
+ PPDMDEVINS pDevIns = pThisCC->pDevInsR3;
+
+ E1kLog(("%s e1kR3SetLinkState: enmState=%d\n", pThis->szPrf, enmState));
+ switch (enmState)
+ {
+ case PDMNETWORKLINKSTATE_UP:
+ pThis->fCableConnected = true;
+ /* If link was down, bring it up after a while. */
+ if (!(STATUS & STATUS_LU))
+ e1kBringLinkUpDelayed(pDevIns, pThis);
+ break;
+ case PDMNETWORKLINKSTATE_DOWN:
+ pThis->fCableConnected = false;
+ /* Always set the phy link state to down, regardless of the STATUS_LU bit.
+ * We might have to set the link state before the driver initializes us. */
+ Phy::setLinkStatus(&pThis->phy, false);
+ /* If link was up, bring it down. */
+ if (STATUS & STATUS_LU)
+ e1kR3LinkDown(pDevIns, pThis, pThisCC);
+ break;
+ case PDMNETWORKLINKSTATE_DOWN_RESUME:
+ /*
+ * There is not much sense in bringing down the link if it has not come up yet.
+ * If it is up though, we bring it down temporarely, then bring it up again.
+ */
+ if (STATUS & STATUS_LU)
+ e1kR3LinkDownTemp(pDevIns, pThis, pThisCC);
+ break;
+ default:
+ ;
+ }
+ return VINF_SUCCESS;
+}
+
+
+/* -=-=-=-=- PDMIBASE -=-=-=-=- */
+
+/**
+ * @interface_method_impl{PDMIBASE,pfnQueryInterface}
+ */
+static DECLCALLBACK(void *) e1kR3QueryInterface(struct PDMIBASE *pInterface, const char *pszIID)
+{
+ PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, 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;
+}
+
+
+/* -=-=-=-=- Saved State -=-=-=-=- */
+
+/**
+ * Saves the configuration.
+ *
+ * @param pThis The E1K state.
+ * @param pSSM The handle to the saved state.
+ */
+static void e1kR3SaveConfig(PCPDMDEVHLPR3 pHlp, PE1KSTATE pThis, PSSMHANDLE pSSM)
+{
+ pHlp->pfnSSMPutMem(pSSM, &pThis->macConfigured, sizeof(pThis->macConfigured));
+ pHlp->pfnSSMPutU32(pSSM, pThis->eChip);
+}
+
+/**
+ * @callback_method_impl{FNSSMDEVLIVEEXEC,Save basic configuration.}
+ */
+static DECLCALLBACK(int) e1kR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
+{
+ RT_NOREF(uPass);
+ e1kR3SaveConfig(pDevIns->pHlpR3, PDMDEVINS_2_DATA(pDevIns, PE1KSTATE), pSSM);
+ return VINF_SSM_DONT_CALL_AGAIN;
+}
+
+/**
+ * @callback_method_impl{FNSSMDEVSAVEPREP,Synchronize.}
+ */
+static DECLCALLBACK(int) e1kR3SavePrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
+{
+ RT_NOREF(pSSM);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+
+ e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
+ e1kCsLeave(pThis);
+ return VINF_SUCCESS;
+#if 0
+ /* 1) Prevent all threads from modifying the state and memory */
+ //pThis->fLocked = true;
+ /* 2) Cancel all timers */
+#ifdef E1K_TX_DELAY
+ e1kCancelTimer(pThis, pThis->CTX_SUFF(pTXDTimer));
+#endif /* E1K_TX_DELAY */
+//#ifdef E1K_USE_TX_TIMERS
+ if (pThis->fTidEnabled)
+ {
+ e1kCancelTimer(pThis, pThis->CTX_SUFF(pTIDTimer));
+#ifndef E1K_NO_TAD
+ e1kCancelTimer(pThis, pThis->CTX_SUFF(pTADTimer));
+#endif /* E1K_NO_TAD */
+ }
+//#endif /* E1K_USE_TX_TIMERS */
+#ifdef E1K_USE_RX_TIMERS
+ e1kCancelTimer(pThis, pThis->CTX_SUFF(pRIDTimer));
+ e1kCancelTimer(pThis, pThis->CTX_SUFF(pRADTimer));
+#endif /* E1K_USE_RX_TIMERS */
+ e1kCancelTimer(pThis, pThis->CTX_SUFF(pIntTimer));
+ /* 3) Did I forget anything? */
+ E1kLog(("%s Locked\n", pThis->szPrf));
+ return VINF_SUCCESS;
+#endif
+}
+
+/**
+ * @callback_method_impl{FNSSMDEVSAVEEXEC}
+ */
+static DECLCALLBACK(int) e1kR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
+
+ e1kR3SaveConfig(pHlp, pThis, pSSM);
+ pThisCC->eeprom.save(pHlp, pSSM);
+ e1kDumpState(pThis);
+ pHlp->pfnSSMPutMem(pSSM, pThis->auRegs, sizeof(pThis->auRegs));
+ pHlp->pfnSSMPutBool(pSSM, pThis->fIntRaised);
+ Phy::saveState(pHlp, pSSM, &pThis->phy);
+ pHlp->pfnSSMPutU32(pSSM, pThis->uSelectedReg);
+ pHlp->pfnSSMPutMem(pSSM, pThis->auMTA, sizeof(pThis->auMTA));
+ pHlp->pfnSSMPutMem(pSSM, &pThis->aRecAddr, sizeof(pThis->aRecAddr));
+ pHlp->pfnSSMPutMem(pSSM, pThis->auVFTA, sizeof(pThis->auVFTA));
+ pHlp->pfnSSMPutU64(pSSM, pThis->u64AckedAt);
+ pHlp->pfnSSMPutU16(pSSM, pThis->u16RxBSize);
+ //pHlp->pfnSSMPutBool(pSSM, pThis->fDelayInts);
+ //pHlp->pfnSSMPutBool(pSSM, pThis->fIntMaskUsed);
+ pHlp->pfnSSMPutU16(pSSM, pThis->u16TxPktLen);
+/** @todo State wrt to the TSE buffer is incomplete, so little point in
+ * saving this actually. */
+ pHlp->pfnSSMPutMem(pSSM, pThis->aTxPacketFallback, pThis->u16TxPktLen);
+ pHlp->pfnSSMPutBool(pSSM, pThis->fIPcsum);
+ pHlp->pfnSSMPutBool(pSSM, pThis->fTCPcsum);
+ pHlp->pfnSSMPutMem(pSSM, &pThis->contextTSE, sizeof(pThis->contextTSE));
+ pHlp->pfnSSMPutMem(pSSM, &pThis->contextNormal, sizeof(pThis->contextNormal));
+ pHlp->pfnSSMPutBool(pSSM, pThis->fVTag);
+ pHlp->pfnSSMPutU16(pSSM, pThis->u16VTagTCI);
+#ifdef E1K_WITH_TXD_CACHE
+# if 0
+ pHlp->pfnSSMPutU8(pSSM, pThis->nTxDFetched);
+ pHlp->pfnSSMPutMem(pSSM, pThis->aTxDescriptors,
+ pThis->nTxDFetched * sizeof(pThis->aTxDescriptors[0]));
+# else
+ /*
+ * There is no point in storing TX descriptor cache entries as we can simply
+ * fetch them again. Moreover, normally the cache is always empty when we
+ * save the state. Store zero entries for compatibility.
+ */
+ pHlp->pfnSSMPutU8(pSSM, 0);
+# endif
+#endif /* E1K_WITH_TXD_CACHE */
+/** @todo GSO requires some more state here. */
+ E1kLog(("%s State has been saved\n", pThis->szPrf));
+ return VINF_SUCCESS;
+}
+
+#if 0
+/**
+ * @callback_method_impl{FNSSMDEVSAVEDONE}
+ */
+static DECLCALLBACK(int) e1kSaveDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+
+ /* If VM is being powered off unlocking will result in assertions in PGM */
+ if (PDMDevHlpGetVM(pDevIns)->enmVMState == VMSTATE_RUNNING)
+ pThis->fLocked = false;
+ else
+ E1kLog(("%s VM is not running -- remain locked\n", pThis->szPrf));
+ E1kLog(("%s Unlocked\n", pThis->szPrf));
+ return VINF_SUCCESS;
+}
+#endif
+
+/**
+ * @callback_method_impl{FNSSMDEVLOADPREP,Synchronize.}
+ */
+static DECLCALLBACK(int) e1kR3LoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
+{
+ RT_NOREF(pSSM);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+
+ e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
+ e1kCsLeave(pThis);
+ return VINF_SUCCESS;
+}
+
+/**
+ * @callback_method_impl{FNSSMDEVLOADEXEC}
+ */
+static DECLCALLBACK(int) e1kR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
+ int rc;
+
+ if ( uVersion != E1K_SAVEDSTATE_VERSION
+#ifdef E1K_WITH_TXD_CACHE
+ && uVersion != E1K_SAVEDSTATE_VERSION_VBOX_42_VTAG
+#endif /* E1K_WITH_TXD_CACHE */
+ && uVersion != E1K_SAVEDSTATE_VERSION_VBOX_41
+ && uVersion != E1K_SAVEDSTATE_VERSION_VBOX_30)
+ return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
+
+ if ( uVersion > E1K_SAVEDSTATE_VERSION_VBOX_30
+ || uPass != SSM_PASS_FINAL)
+ {
+ /* config checks */
+ RTMAC macConfigured;
+ rc = pHlp->pfnSSMGetMem(pSSM, &macConfigured, sizeof(macConfigured));
+ AssertRCReturn(rc, rc);
+ if ( memcmp(&macConfigured, &pThis->macConfigured, sizeof(macConfigured))
+ && (uPass == 0 || !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)) )
+ LogRel(("%s: The mac address differs: config=%RTmac saved=%RTmac\n", pThis->szPrf, &pThis->macConfigured, &macConfigured));
+
+ E1KCHIP eChip;
+ rc = pHlp->pfnSSMGetU32(pSSM, &eChip);
+ AssertRCReturn(rc, rc);
+ if (eChip != pThis->eChip)
+ return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("The chip type differs: config=%u saved=%u"), pThis->eChip, eChip);
+ }
+
+ if (uPass == SSM_PASS_FINAL)
+ {
+ if (uVersion > E1K_SAVEDSTATE_VERSION_VBOX_30)
+ {
+ rc = pThisCC->eeprom.load(pHlp, pSSM);
+ AssertRCReturn(rc, rc);
+ }
+ /* the state */
+ pHlp->pfnSSMGetMem(pSSM, &pThis->auRegs, sizeof(pThis->auRegs));
+ pHlp->pfnSSMGetBool(pSSM, &pThis->fIntRaised);
+ /** @todo PHY could be made a separate device with its own versioning */
+ Phy::loadState(pHlp, pSSM, &pThis->phy);
+ pHlp->pfnSSMGetU32(pSSM, &pThis->uSelectedReg);
+ pHlp->pfnSSMGetMem(pSSM, &pThis->auMTA, sizeof(pThis->auMTA));
+ pHlp->pfnSSMGetMem(pSSM, &pThis->aRecAddr, sizeof(pThis->aRecAddr));
+ pHlp->pfnSSMGetMem(pSSM, &pThis->auVFTA, sizeof(pThis->auVFTA));
+ pHlp->pfnSSMGetU64(pSSM, &pThis->u64AckedAt);
+ pHlp->pfnSSMGetU16(pSSM, &pThis->u16RxBSize);
+ //pHlp->pfnSSMGetBool(pSSM, pThis->fDelayInts);
+ //pHlp->pfnSSMGetBool(pSSM, pThis->fIntMaskUsed);
+ rc = pHlp->pfnSSMGetU16(pSSM, &pThis->u16TxPktLen);
+ AssertRCReturn(rc, rc);
+ if (pThis->u16TxPktLen > sizeof(pThis->aTxPacketFallback))
+ pThis->u16TxPktLen = sizeof(pThis->aTxPacketFallback);
+ pHlp->pfnSSMGetMem(pSSM, &pThis->aTxPacketFallback[0], pThis->u16TxPktLen);
+ pHlp->pfnSSMGetBool(pSSM, &pThis->fIPcsum);
+ pHlp->pfnSSMGetBool(pSSM, &pThis->fTCPcsum);
+ pHlp->pfnSSMGetMem(pSSM, &pThis->contextTSE, sizeof(pThis->contextTSE));
+ rc = pHlp->pfnSSMGetMem(pSSM, &pThis->contextNormal, sizeof(pThis->contextNormal));
+ AssertRCReturn(rc, rc);
+ if (uVersion > E1K_SAVEDSTATE_VERSION_VBOX_41)
+ {
+ pHlp->pfnSSMGetBool(pSSM, &pThis->fVTag);
+ rc = pHlp->pfnSSMGetU16(pSSM, &pThis->u16VTagTCI);
+ AssertRCReturn(rc, rc);
+ }
+ else
+ {
+ pThis->fVTag = false;
+ pThis->u16VTagTCI = 0;
+ }
+#ifdef E1K_WITH_TXD_CACHE
+ if (uVersion > E1K_SAVEDSTATE_VERSION_VBOX_42_VTAG)
+ {
+ rc = pHlp->pfnSSMGetU8(pSSM, &pThis->nTxDFetched);
+ AssertRCReturn(rc, rc);
+ if (pThis->nTxDFetched)
+ pHlp->pfnSSMGetMem(pSSM, pThis->aTxDescriptors,
+ pThis->nTxDFetched * sizeof(pThis->aTxDescriptors[0]));
+ }
+ else
+ pThis->nTxDFetched = 0;
+ /**
+ * @todo Perhaps we should not store TXD cache as the entries can be
+ * simply fetched again from guest's memory. Or can't they?
+ */
+#endif /* E1K_WITH_TXD_CACHE */
+#ifdef E1K_WITH_RXD_CACHE
+ /*
+ * There is no point in storing the RX descriptor cache in the saved
+ * state, we just need to make sure it is empty.
+ */
+ pThis->iRxDCurrent = pThis->nRxDFetched = 0;
+#endif /* E1K_WITH_RXD_CACHE */
+ rc = pHlp->pfnSSMHandleGetStatus(pSSM);
+ AssertRCReturn(rc, rc);
+
+ /* derived state */
+ e1kSetupGsoCtx(&pThis->GsoCtx, &pThis->contextTSE);
+
+ E1kLog(("%s State has been restored\n", pThis->szPrf));
+ e1kDumpState(pThis);
+ }
+ return VINF_SUCCESS;
+}
+
+/**
+ * @callback_method_impl{FNSSMDEVLOADDONE, Link status adjustments after loading.}
+ */
+static DECLCALLBACK(int) e1kR3LoadDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ RT_NOREF(pSSM);
+
+ /* Update promiscuous mode */
+ if (pThisCC->pDrvR3)
+ pThisCC->pDrvR3->pfnSetPromiscuousMode(pThisCC->pDrvR3, !!(RCTL & (RCTL_UPE | RCTL_MPE)));
+
+ /*
+ * Force the link down here, since PDMNETWORKLINKSTATE_DOWN_RESUME is never
+ * passed to us. We go through all this stuff if the link was up and we
+ * wasn't teleported.
+ */
+ if ( (STATUS & STATUS_LU)
+ && !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)
+ && pThis->cMsLinkUpDelay)
+ {
+ e1kR3LinkDownTemp(pDevIns, pThis, pThisCC);
+ }
+ return VINF_SUCCESS;
+}
+
+
+
+/* -=-=-=-=- Debug Info + Log Types -=-=-=-=- */
+
+/**
+ * @callback_method_impl{FNRTSTRFORMATTYPE}
+ */
+static DECLCALLBACK(size_t) e1kR3FmtRxDesc(PFNRTSTROUTPUT pfnOutput,
+ void *pvArgOutput,
+ const char *pszType,
+ void const *pvValue,
+ int cchWidth,
+ int cchPrecision,
+ unsigned fFlags,
+ void *pvUser)
+{
+ RT_NOREF(cchWidth, cchPrecision, fFlags, pvUser);
+ AssertReturn(strcmp(pszType, "e1krxd") == 0, 0);
+ E1KRXDESC* pDesc = (E1KRXDESC*)pvValue;
+ if (!pDesc)
+ return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "NULL_RXD");
+
+ size_t cbPrintf = 0;
+ cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Address=%16LX Length=%04X Csum=%04X\n",
+ pDesc->u64BufAddr, pDesc->u16Length, pDesc->u16Checksum);
+ cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, " STA: %s %s %s %s %s %s %s ERR: %s %s %s %s SPECIAL: %s VLAN=%03x PRI=%x",
+ pDesc->status.fPIF ? "PIF" : "pif",
+ pDesc->status.fIPCS ? "IPCS" : "ipcs",
+ pDesc->status.fTCPCS ? "TCPCS" : "tcpcs",
+ pDesc->status.fVP ? "VP" : "vp",
+ pDesc->status.fIXSM ? "IXSM" : "ixsm",
+ pDesc->status.fEOP ? "EOP" : "eop",
+ pDesc->status.fDD ? "DD" : "dd",
+ pDesc->status.fRXE ? "RXE" : "rxe",
+ pDesc->status.fIPE ? "IPE" : "ipe",
+ pDesc->status.fTCPE ? "TCPE" : "tcpe",
+ pDesc->status.fCE ? "CE" : "ce",
+ E1K_SPEC_CFI(pDesc->status.u16Special) ? "CFI" :"cfi",
+ E1K_SPEC_VLAN(pDesc->status.u16Special),
+ E1K_SPEC_PRI(pDesc->status.u16Special));
+ return cbPrintf;
+}
+
+/**
+ * @callback_method_impl{FNRTSTRFORMATTYPE}
+ */
+static DECLCALLBACK(size_t) e1kR3FmtTxDesc(PFNRTSTROUTPUT pfnOutput,
+ void *pvArgOutput,
+ const char *pszType,
+ void const *pvValue,
+ int cchWidth,
+ int cchPrecision,
+ unsigned fFlags,
+ void *pvUser)
+{
+ RT_NOREF(cchWidth, cchPrecision, fFlags, pvUser);
+ AssertReturn(strcmp(pszType, "e1ktxd") == 0, 0);
+ E1KTXDESC *pDesc = (E1KTXDESC*)pvValue;
+ if (!pDesc)
+ return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "NULL_TXD");
+
+ size_t cbPrintf = 0;
+ switch (e1kGetDescType(pDesc))
+ {
+ case E1K_DTYP_CONTEXT:
+ cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Type=Context\n"
+ " IPCSS=%02X IPCSO=%02X IPCSE=%04X TUCSS=%02X TUCSO=%02X TUCSE=%04X\n"
+ " TUCMD:%s%s%s %s %s PAYLEN=%04x HDRLEN=%04x MSS=%04x STA: %s",
+ pDesc->context.ip.u8CSS, pDesc->context.ip.u8CSO, pDesc->context.ip.u16CSE,
+ pDesc->context.tu.u8CSS, pDesc->context.tu.u8CSO, pDesc->context.tu.u16CSE,
+ pDesc->context.dw2.fIDE ? " IDE":"",
+ pDesc->context.dw2.fRS ? " RS" :"",
+ pDesc->context.dw2.fTSE ? " TSE":"",
+ pDesc->context.dw2.fIP ? "IPv4":"IPv6",
+ pDesc->context.dw2.fTCP ? "TCP":"UDP",
+ pDesc->context.dw2.u20PAYLEN,
+ pDesc->context.dw3.u8HDRLEN,
+ pDesc->context.dw3.u16MSS,
+ pDesc->context.dw3.fDD?"DD":"");
+ break;
+ case E1K_DTYP_DATA:
+ cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Type=Data Address=%16LX DTALEN=%05X\n"
+ " DCMD:%s%s%s%s%s%s%s STA:%s%s%s POPTS:%s%s SPECIAL:%s VLAN=%03x PRI=%x",
+ pDesc->data.u64BufAddr,
+ pDesc->data.cmd.u20DTALEN,
+ pDesc->data.cmd.fIDE ? " IDE" :"",
+ pDesc->data.cmd.fVLE ? " VLE" :"",
+ pDesc->data.cmd.fRPS ? " RPS" :"",
+ pDesc->data.cmd.fRS ? " RS" :"",
+ pDesc->data.cmd.fTSE ? " TSE" :"",
+ pDesc->data.cmd.fIFCS? " IFCS":"",
+ pDesc->data.cmd.fEOP ? " EOP" :"",
+ pDesc->data.dw3.fDD ? " DD" :"",
+ pDesc->data.dw3.fEC ? " EC" :"",
+ pDesc->data.dw3.fLC ? " LC" :"",
+ pDesc->data.dw3.fTXSM? " TXSM":"",
+ pDesc->data.dw3.fIXSM? " IXSM":"",
+ E1K_SPEC_CFI(pDesc->data.dw3.u16Special) ? "CFI" :"cfi",
+ E1K_SPEC_VLAN(pDesc->data.dw3.u16Special),
+ E1K_SPEC_PRI(pDesc->data.dw3.u16Special));
+ break;
+ case E1K_DTYP_LEGACY:
+ cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Type=Legacy Address=%16LX DTALEN=%05X\n"
+ " CMD:%s%s%s%s%s%s%s STA:%s%s%s CSO=%02x CSS=%02x SPECIAL:%s VLAN=%03x PRI=%x",
+ pDesc->data.u64BufAddr,
+ pDesc->legacy.cmd.u16Length,
+ pDesc->legacy.cmd.fIDE ? " IDE" :"",
+ pDesc->legacy.cmd.fVLE ? " VLE" :"",
+ pDesc->legacy.cmd.fRPS ? " RPS" :"",
+ pDesc->legacy.cmd.fRS ? " RS" :"",
+ pDesc->legacy.cmd.fIC ? " IC" :"",
+ pDesc->legacy.cmd.fIFCS? " IFCS":"",
+ pDesc->legacy.cmd.fEOP ? " EOP" :"",
+ pDesc->legacy.dw3.fDD ? " DD" :"",
+ pDesc->legacy.dw3.fEC ? " EC" :"",
+ pDesc->legacy.dw3.fLC ? " LC" :"",
+ pDesc->legacy.cmd.u8CSO,
+ pDesc->legacy.dw3.u8CSS,
+ E1K_SPEC_CFI(pDesc->legacy.dw3.u16Special) ? "CFI" :"cfi",
+ E1K_SPEC_VLAN(pDesc->legacy.dw3.u16Special),
+ E1K_SPEC_PRI(pDesc->legacy.dw3.u16Special));
+ break;
+ default:
+ cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Invalid Transmit Descriptor");
+ break;
+ }
+
+ return cbPrintf;
+}
+
+/** Initializes debug helpers (logging format types). */
+static int e1kR3InitDebugHelpers(void)
+{
+ int rc = VINF_SUCCESS;
+ static bool s_fHelpersRegistered = false;
+ if (!s_fHelpersRegistered)
+ {
+ s_fHelpersRegistered = true;
+ rc = RTStrFormatTypeRegister("e1krxd", e1kR3FmtRxDesc, NULL);
+ AssertRCReturn(rc, rc);
+ rc = RTStrFormatTypeRegister("e1ktxd", e1kR3FmtTxDesc, NULL);
+ AssertRCReturn(rc, rc);
+ }
+ return rc;
+}
+
+/**
+ * Status info callback.
+ *
+ * @param pDevIns The device instance.
+ * @param pHlp The output helpers.
+ * @param pszArgs The arguments.
+ */
+static DECLCALLBACK(void) e1kR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
+{
+ RT_NOREF(pszArgs);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ unsigned i;
+ // bool fRcvRing = false;
+ // bool fXmtRing = false;
+
+ /*
+ * Parse args.
+ if (pszArgs)
+ {
+ fRcvRing = strstr(pszArgs, "verbose") || strstr(pszArgs, "rcv");
+ fXmtRing = strstr(pszArgs, "verbose") || strstr(pszArgs, "xmt");
+ }
+ */
+
+ /*
+ * Show info.
+ */
+ pHlp->pfnPrintf(pHlp, "E1000 #%d: port=%04x mmio=%RGp mac-cfg=%RTmac %s%s%s\n",
+ pDevIns->iInstance,
+ PDMDevHlpIoPortGetMappingAddress(pDevIns, pThis->hIoPorts),
+ PDMDevHlpMmioGetMappingAddress(pDevIns, pThis->hMmioRegion),
+ &pThis->macConfigured, g_aChips[pThis->eChip].pcszName,
+ pDevIns->fRCEnabled ? " RC" : "", pDevIns->fR0Enabled ? " R0" : "");
+
+ e1kR3CsEnterAsserted(pThis); /* Not sure why but PCNet does it */
+
+ for (i = 0; i < E1K_NUM_OF_32BIT_REGS; ++i)
+ pHlp->pfnPrintf(pHlp, "%8.8s = %08x\n", g_aE1kRegMap[i].abbrev, pThis->auRegs[i]);
+
+ for (i = 0; i < RT_ELEMENTS(pThis->aRecAddr.array); i++)
+ {
+ E1KRAELEM* ra = pThis->aRecAddr.array + i;
+ if (ra->ctl & RA_CTL_AV)
+ {
+ const char *pcszTmp;
+ switch (ra->ctl & RA_CTL_AS)
+ {
+ case 0: pcszTmp = "DST"; break;
+ case 1: pcszTmp = "SRC"; break;
+ default: pcszTmp = "reserved";
+ }
+ pHlp->pfnPrintf(pHlp, "RA%02d: %s %RTmac\n", i, pcszTmp, ra->addr);
+ }
+ }
+ unsigned cDescs = RDLEN / sizeof(E1KRXDESC);
+ uint32_t rdh = RDH;
+ pHlp->pfnPrintf(pHlp, "\n-- Receive Descriptors (%d total) --\n", cDescs);
+ for (i = 0; i < cDescs; ++i)
+ {
+ E1KRXDESC desc;
+ PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, i),
+ &desc, sizeof(desc));
+ if (i == rdh)
+ pHlp->pfnPrintf(pHlp, ">>> ");
+ pHlp->pfnPrintf(pHlp, "%RGp: %R[e1krxd]\n", e1kDescAddr(RDBAH, RDBAL, i), &desc);
+ }
+#ifdef E1K_WITH_RXD_CACHE
+ pHlp->pfnPrintf(pHlp, "\n-- Receive Descriptors in Cache (at %d (RDH %d)/ fetched %d / max %d) --\n",
+ pThis->iRxDCurrent, RDH, pThis->nRxDFetched, E1K_RXD_CACHE_SIZE);
+ if (rdh > pThis->iRxDCurrent)
+ rdh -= pThis->iRxDCurrent;
+ else
+ rdh = cDescs + rdh - pThis->iRxDCurrent;
+ for (i = 0; i < pThis->nRxDFetched; ++i)
+ {
+ if (i == pThis->iRxDCurrent)
+ pHlp->pfnPrintf(pHlp, ">>> ");
+ if (cDescs)
+ pHlp->pfnPrintf(pHlp, "%RGp: %R[e1krxd]\n",
+ e1kDescAddr(RDBAH, RDBAL, rdh++ % cDescs),
+ &pThis->aRxDescriptors[i]);
+ else
+ pHlp->pfnPrintf(pHlp, "<lost>: %R[e1krxd]\n",
+ &pThis->aRxDescriptors[i]);
+ }
+#endif /* E1K_WITH_RXD_CACHE */
+
+ cDescs = TDLEN / sizeof(E1KTXDESC);
+ uint32_t tdh = TDH;
+ pHlp->pfnPrintf(pHlp, "\n-- Transmit Descriptors (%d total) --\n", cDescs);
+ for (i = 0; i < cDescs; ++i)
+ {
+ E1KTXDESC desc;
+ PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(TDBAH, TDBAL, i),
+ &desc, sizeof(desc));
+ if (i == tdh)
+ pHlp->pfnPrintf(pHlp, ">>> ");
+ pHlp->pfnPrintf(pHlp, "%RGp: %R[e1ktxd]\n", e1kDescAddr(TDBAH, TDBAL, i), &desc);
+ }
+#ifdef E1K_WITH_TXD_CACHE
+ pHlp->pfnPrintf(pHlp, "\n-- Transmit Descriptors in Cache (at %d (TDH %d)/ fetched %d / max %d) --\n",
+ pThis->iTxDCurrent, TDH, pThis->nTxDFetched, E1K_TXD_CACHE_SIZE);
+ if (tdh > pThis->iTxDCurrent)
+ tdh -= pThis->iTxDCurrent;
+ else
+ tdh = cDescs + tdh - pThis->iTxDCurrent;
+ for (i = 0; i < pThis->nTxDFetched; ++i)
+ {
+ if (i == pThis->iTxDCurrent)
+ pHlp->pfnPrintf(pHlp, ">>> ");
+ if (cDescs)
+ pHlp->pfnPrintf(pHlp, "%RGp: %R[e1ktxd]\n",
+ e1kDescAddr(TDBAH, TDBAL, tdh++ % cDescs),
+ &pThis->aTxDescriptors[i]);
+ else
+ pHlp->pfnPrintf(pHlp, "<lost>: %R[e1ktxd]\n",
+ &pThis->aTxDescriptors[i]);
+ }
+#endif /* E1K_WITH_TXD_CACHE */
+
+
+#ifdef E1K_INT_STATS
+ pHlp->pfnPrintf(pHlp, "Interrupt attempts: %d\n", pThis->uStatIntTry);
+ pHlp->pfnPrintf(pHlp, "Interrupts raised : %d\n", pThis->uStatInt);
+ pHlp->pfnPrintf(pHlp, "Interrupts lowered: %d\n", pThis->uStatIntLower);
+ pHlp->pfnPrintf(pHlp, "ICR outside ISR : %d\n", pThis->uStatNoIntICR);
+ pHlp->pfnPrintf(pHlp, "IMS raised ints : %d\n", pThis->uStatIntIMS);
+ pHlp->pfnPrintf(pHlp, "Interrupts skipped: %d\n", pThis->uStatIntSkip);
+ pHlp->pfnPrintf(pHlp, "Masked interrupts : %d\n", pThis->uStatIntMasked);
+ pHlp->pfnPrintf(pHlp, "Early interrupts : %d\n", pThis->uStatIntEarly);
+ pHlp->pfnPrintf(pHlp, "Late interrupts : %d\n", pThis->uStatIntLate);
+ pHlp->pfnPrintf(pHlp, "Lost interrupts : %d\n", pThis->iStatIntLost);
+ pHlp->pfnPrintf(pHlp, "Interrupts by RX : %d\n", pThis->uStatIntRx);
+ pHlp->pfnPrintf(pHlp, "Interrupts by TX : %d\n", pThis->uStatIntTx);
+ pHlp->pfnPrintf(pHlp, "Interrupts by ICS : %d\n", pThis->uStatIntICS);
+ pHlp->pfnPrintf(pHlp, "Interrupts by RDTR: %d\n", pThis->uStatIntRDTR);
+ pHlp->pfnPrintf(pHlp, "Interrupts by RDMT: %d\n", pThis->uStatIntRXDMT0);
+ pHlp->pfnPrintf(pHlp, "Interrupts by TXQE: %d\n", pThis->uStatIntTXQE);
+ pHlp->pfnPrintf(pHlp, "TX int delay asked: %d\n", pThis->uStatTxIDE);
+ pHlp->pfnPrintf(pHlp, "TX delayed: %d\n", pThis->uStatTxDelayed);
+ pHlp->pfnPrintf(pHlp, "TX delayed expired: %d\n", pThis->uStatTxDelayExp);
+ pHlp->pfnPrintf(pHlp, "TX no report asked: %d\n", pThis->uStatTxNoRS);
+ pHlp->pfnPrintf(pHlp, "TX abs timer expd : %d\n", pThis->uStatTAD);
+ pHlp->pfnPrintf(pHlp, "TX int timer expd : %d\n", pThis->uStatTID);
+ pHlp->pfnPrintf(pHlp, "RX abs timer expd : %d\n", pThis->uStatRAD);
+ pHlp->pfnPrintf(pHlp, "RX int timer expd : %d\n", pThis->uStatRID);
+ pHlp->pfnPrintf(pHlp, "TX CTX descriptors: %d\n", pThis->uStatDescCtx);
+ pHlp->pfnPrintf(pHlp, "TX DAT descriptors: %d\n", pThis->uStatDescDat);
+ pHlp->pfnPrintf(pHlp, "TX LEG descriptors: %d\n", pThis->uStatDescLeg);
+ pHlp->pfnPrintf(pHlp, "Received frames : %d\n", pThis->uStatRxFrm);
+ pHlp->pfnPrintf(pHlp, "Transmitted frames: %d\n", pThis->uStatTxFrm);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 1514: %d\n", pThis->uStatTx1514);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 2962: %d\n", pThis->uStatTx2962);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 4410: %d\n", pThis->uStatTx4410);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 5858: %d\n", pThis->uStatTx5858);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 7306: %d\n", pThis->uStatTx7306);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 8754: %d\n", pThis->uStatTx8754);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 16384: %d\n", pThis->uStatTx16384);
+ pHlp->pfnPrintf(pHlp, "TX frames up to 32768: %d\n", pThis->uStatTx32768);
+ pHlp->pfnPrintf(pHlp, "Larger TX frames : %d\n", pThis->uStatTxLarge);
+#endif /* E1K_INT_STATS */
+
+ e1kCsLeave(pThis);
+}
+
+
+
+/* -=-=-=-=- PDMDEVREG -=-=-=-=- */
+
+/**
+ * Detach notification.
+ *
+ * One port on the network card has been disconnected from the network.
+ *
+ * @param pDevIns The device instance.
+ * @param iLUN The logical unit which is being detached.
+ * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
+ */
+static DECLCALLBACK(void) e1kR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ Log(("%s e1kR3Detach:\n", pThis->szPrf));
+ RT_NOREF(fFlags);
+
+ AssertLogRelReturnVoid(iLUN == 0);
+
+ e1kR3CsEnterAsserted(pThis);
+
+ /* Mark device as detached. */
+ pThis->fIsAttached = false;
+ /*
+ * Zero some important members.
+ */
+ pThisCC->pDrvBase = NULL;
+ pThisCC->pDrvR3 = NULL;
+#if 0 /** @todo @bugref{9218} ring-0 driver stuff */
+ pThisR0->pDrvR0 = NIL_RTR0PTR;
+ pThisRC->pDrvRC = NIL_RTRCPTR;
+#endif
+
+ PDMDevHlpCritSectLeave(pDevIns, &pThis->cs);
+}
+
+/**
+ * Attach the Network attachment.
+ *
+ * One port on the network card has been connected to a network.
+ *
+ * @returns VBox status code.
+ * @param pDevIns The device instance.
+ * @param iLUN The logical unit which is being attached.
+ * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
+ *
+ * @remarks This code path is not used during construction.
+ */
+static DECLCALLBACK(int) e1kR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ LogFlow(("%s e1kR3Attach:\n", pThis->szPrf));
+ RT_NOREF(fFlags);
+
+ AssertLogRelReturn(iLUN == 0, VERR_PDM_NO_SUCH_LUN);
+
+ e1kR3CsEnterAsserted(pThis);
+
+ /*
+ * Attach the driver.
+ */
+ int rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->IBase, &pThisCC->pDrvBase, "Network Port");
+ if (RT_SUCCESS(rc))
+ {
+ pThisCC->pDrvR3 = PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMINETWORKUP);
+ AssertMsgStmt(pThisCC->pDrvR3, ("Failed to obtain the PDMINETWORKUP interface!\n"),
+ rc = VERR_PDM_MISSING_INTERFACE_BELOW);
+ if (RT_SUCCESS(rc))
+ {
+#if 0 /** @todo @bugref{9218} ring-0 driver stuff */
+ pThisR0->pDrvR0 = PDMIBASER0_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASER0), PDMINETWORKUP);
+ pThisRC->pDrvRC = PDMIBASERC_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASERC), PDMINETWORKUP);
+#endif
+ /* Mark device as attached. */
+ pThis->fIsAttached = 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(("%s No attached driver!\n", pThis->szPrf));
+ }
+
+ /*
+ * 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 ((STATUS & STATUS_LU) && RT_SUCCESS(rc))
+ e1kR3LinkDownTemp(pDevIns, pThis, pThisCC);
+
+ PDMDevHlpCritSectLeave(pDevIns, &pThis->cs);
+ return rc;
+}
+
+/**
+ * @copydoc FNPDMDEVPOWEROFF
+ */
+static DECLCALLBACK(void) e1kR3PowerOff(PPDMDEVINS pDevIns)
+{
+ /* Poke thread waiting for buffer space. */
+ e1kWakeupReceive(pDevIns, PDMDEVINS_2_DATA(pDevIns, PE1KSTATE));
+}
+
+/**
+ * @copydoc FNPDMDEVRESET
+ */
+static DECLCALLBACK(void) e1kR3Reset(PPDMDEVINS pDevIns)
+{
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+#ifdef E1K_TX_DELAY
+ e1kCancelTimer(pDevIns, pThis, pThis->hTXDTimer);
+#endif /* E1K_TX_DELAY */
+ e1kCancelTimer(pDevIns, pThis, pThis->hIntTimer);
+ e1kCancelTimer(pDevIns, pThis, pThis->hLUTimer);
+ e1kXmitFreeBuf(pThis, pThisCC);
+ pThis->u16TxPktLen = 0;
+ pThis->fIPcsum = false;
+ pThis->fTCPcsum = false;
+ pThis->fIntMaskUsed = false;
+ pThis->fDelayInts = false;
+ pThis->fLocked = false;
+ pThis->u64AckedAt = 0;
+ e1kR3HardReset(pDevIns, pThis, pThisCC);
+}
+
+/**
+ * @copydoc FNPDMDEVSUSPEND
+ */
+static DECLCALLBACK(void) e1kR3Suspend(PPDMDEVINS pDevIns)
+{
+ /* Poke thread waiting for buffer space. */
+ e1kWakeupReceive(pDevIns, PDMDEVINS_2_DATA(pDevIns, PE1KSTATE));
+}
+
+/**
+ * Device relocation callback.
+ *
+ * When this callback is called the device instance data, and if the
+ * device have a GC component, is being relocated, or/and the selectors
+ * have been changed. The device must use the chance to perform the
+ * necessary pointer relocations and data updates.
+ *
+ * Before the GC code is executed the first time, this function will be
+ * called with a 0 delta so GC pointer calculations can be one in one place.
+ *
+ * @param pDevIns Pointer to the device instance.
+ * @param offDelta The relocation delta relative to the old location.
+ *
+ * @remark A relocation CANNOT fail.
+ */
+static DECLCALLBACK(void) e1kR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
+{
+ PE1KSTATERC pThisRC = PDMINS_2_DATA_RC(pDevIns, PE1KSTATERC);
+ if (pThisRC)
+ pThisRC->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
+ RT_NOREF(offDelta);
+}
+
+/**
+ * Destruct a device instance.
+ *
+ * We need to free non-VM resources only.
+ *
+ * @returns VBox status code.
+ * @param pDevIns The device instance data.
+ * @thread EMT
+ */
+static DECLCALLBACK(int) e1kR3Destruct(PPDMDEVINS pDevIns)
+{
+ PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+
+ e1kDumpState(pThis);
+ E1kLog(("%s Destroying instance\n", pThis->szPrf));
+ if (PDMDevHlpCritSectIsInitialized(pDevIns, &pThis->cs))
+ {
+ if (pThis->hEventMoreRxDescAvail != NIL_SUPSEMEVENT)
+ {
+ PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEventMoreRxDescAvail);
+ RTThreadYield();
+ PDMDevHlpSUPSemEventClose(pDevIns, pThis->hEventMoreRxDescAvail);
+ pThis->hEventMoreRxDescAvail = NIL_SUPSEMEVENT;
+ }
+#ifdef E1K_WITH_TX_CS
+ PDMDevHlpCritSectDelete(pDevIns, &pThis->csTx);
+#endif /* E1K_WITH_TX_CS */
+ PDMDevHlpCritSectDelete(pDevIns, &pThis->csRx);
+ PDMDevHlpCritSectDelete(pDevIns, &pThis->cs);
+ }
+ return VINF_SUCCESS;
+}
+
+
+/**
+ * Set PCI configuration space registers.
+ *
+ * @param pci Reference to PCI device structure.
+ * @thread EMT
+ */
+static void e1kR3ConfigurePciDev(PPDMPCIDEV pPciDev, E1KCHIP eChip)
+{
+ Assert(eChip < RT_ELEMENTS(g_aChips));
+ /* Configure PCI Device, assume 32-bit mode ******************************/
+ PDMPciDevSetVendorId(pPciDev, g_aChips[eChip].uPCIVendorId);
+ PDMPciDevSetDeviceId(pPciDev, g_aChips[eChip].uPCIDeviceId);
+ PDMPciDevSetWord( pPciDev, VBOX_PCI_SUBSYSTEM_VENDOR_ID, g_aChips[eChip].uPCISubsystemVendorId);
+ PDMPciDevSetWord( pPciDev, VBOX_PCI_SUBSYSTEM_ID, g_aChips[eChip].uPCISubsystemId);
+
+ PDMPciDevSetWord( pPciDev, VBOX_PCI_COMMAND, 0x0000);
+ /* DEVSEL Timing (medium device), 66 MHz Capable, New capabilities */
+ PDMPciDevSetWord( pPciDev, VBOX_PCI_STATUS,
+ VBOX_PCI_STATUS_DEVSEL_MEDIUM | VBOX_PCI_STATUS_CAP_LIST | VBOX_PCI_STATUS_66MHZ);
+ /* Stepping A2 */
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_REVISION_ID, 0x02);
+ /* Ethernet adapter */
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_CLASS_PROG, 0x00);
+ PDMPciDevSetWord( pPciDev, VBOX_PCI_CLASS_DEVICE, 0x0200);
+ /* normal single function Ethernet controller */
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_HEADER_TYPE, 0x00);
+ /* Memory Register Base Address */
+ PDMPciDevSetDWord(pPciDev, VBOX_PCI_BASE_ADDRESS_0, 0x00000000);
+ /* Memory Flash Base Address */
+ PDMPciDevSetDWord(pPciDev, VBOX_PCI_BASE_ADDRESS_1, 0x00000000);
+ /* IO Register Base Address */
+ PDMPciDevSetDWord(pPciDev, VBOX_PCI_BASE_ADDRESS_2, 0x00000001);
+ /* Expansion ROM Base Address */
+ PDMPciDevSetDWord(pPciDev, VBOX_PCI_ROM_ADDRESS, 0x00000000);
+ /* Capabilities Pointer */
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_CAPABILITY_LIST, 0xDC);
+ /* Interrupt Pin: INTA# */
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_INTERRUPT_PIN, 0x01);
+ /* Max_Lat/Min_Gnt: very high priority and time slice */
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_MIN_GNT, 0xFF);
+ PDMPciDevSetByte( pPciDev, VBOX_PCI_MAX_LAT, 0x00);
+
+ /* PCI Power Management Registers ****************************************/
+ /* Capability ID: PCI Power Management Registers */
+ PDMPciDevSetByte( pPciDev, 0xDC, VBOX_PCI_CAP_ID_PM);
+ /* Next Item Pointer: PCI-X */
+ PDMPciDevSetByte( pPciDev, 0xDC + 1, 0xE4);
+ /* Power Management Capabilities: PM disabled, DSI */
+ PDMPciDevSetWord( pPciDev, 0xDC + 2,
+ 0x0002 | VBOX_PCI_PM_CAP_DSI);
+ /* Power Management Control / Status Register: PM disabled */
+ PDMPciDevSetWord( pPciDev, 0xDC + 4, 0x0000);
+ /* PMCSR_BSE Bridge Support Extensions: Not supported */
+ PDMPciDevSetByte( pPciDev, 0xDC + 6, 0x00);
+ /* Data Register: PM disabled, always 0 */
+ PDMPciDevSetByte( pPciDev, 0xDC + 7, 0x00);
+
+ /* PCI-X Configuration Registers *****************************************/
+ /* Capability ID: PCI-X Configuration Registers */
+ PDMPciDevSetByte( pPciDev, 0xE4, VBOX_PCI_CAP_ID_PCIX);
+#ifdef E1K_WITH_MSI
+ PDMPciDevSetByte( pPciDev, 0xE4 + 1, 0x80);
+#else
+ /* Next Item Pointer: None (Message Signalled Interrupts are disabled) */
+ PDMPciDevSetByte( pPciDev, 0xE4 + 1, 0x00);
+#endif
+ /* PCI-X Command: Enable Relaxed Ordering */
+ PDMPciDevSetWord( pPciDev, 0xE4 + 2, VBOX_PCI_X_CMD_ERO);
+ /* PCI-X Status: 32-bit, 66MHz*/
+ /** @todo is this value really correct? fff8 doesn't look like actual PCI address */
+ PDMPciDevSetDWord(pPciDev, 0xE4 + 4, 0x0040FFF8);
+}
+
+/**
+ * @interface_method_impl{PDMDEVREG,pfnConstruct}
+ */
+static DECLCALLBACK(int) e1kR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
+{
+ PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+ int rc;
+
+ /*
+ * Initialize the instance data (state).
+ * Note! Caller has initialized it to ZERO already.
+ */
+ RTStrPrintf(pThis->szPrf, sizeof(pThis->szPrf), "E1000#%d", iInstance);
+ E1kLog(("%s Constructing new instance sizeof(E1KRXDESC)=%d\n", pThis->szPrf, sizeof(E1KRXDESC)));
+ pThis->hEventMoreRxDescAvail = NIL_SUPSEMEVENT;
+ pThis->u16TxPktLen = 0;
+ pThis->fIPcsum = false;
+ pThis->fTCPcsum = false;
+ pThis->fIntMaskUsed = false;
+ pThis->fDelayInts = false;
+ pThis->fLocked = false;
+ pThis->u64AckedAt = 0;
+ pThis->led.u32Magic = PDMLED_MAGIC;
+ pThis->u32PktNo = 1;
+ pThis->fIsAttached = false;
+
+ pThisCC->pDevInsR3 = pDevIns;
+ pThisCC->pShared = pThis;
+
+ /* Interfaces */
+ pThisCC->IBase.pfnQueryInterface = e1kR3QueryInterface;
+
+ pThisCC->INetworkDown.pfnWaitReceiveAvail = e1kR3NetworkDown_WaitReceiveAvail;
+ pThisCC->INetworkDown.pfnReceive = e1kR3NetworkDown_Receive;
+ pThisCC->INetworkDown.pfnXmitPending = e1kR3NetworkDown_XmitPending;
+
+ pThisCC->ILeds.pfnQueryStatusLed = e1kR3QueryStatusLed;
+
+ pThisCC->INetworkConfig.pfnGetMac = e1kR3GetMac;
+ pThisCC->INetworkConfig.pfnGetLinkState = e1kR3GetLinkState;
+ pThisCC->INetworkConfig.pfnSetLinkState = e1kR3SetLinkState;
+
+ /*
+ * Internal validations.
+ */
+ for (uint32_t iReg = 1; iReg < E1K_NUM_OF_BINARY_SEARCHABLE; iReg++)
+ AssertLogRelMsgReturn( g_aE1kRegMap[iReg].offset > g_aE1kRegMap[iReg - 1].offset
+ && g_aE1kRegMap[iReg].offset + g_aE1kRegMap[iReg].size
+ >= g_aE1kRegMap[iReg - 1].offset + g_aE1kRegMap[iReg - 1].size,
+ ("%s@%#xLB%#x vs %s@%#xLB%#x\n",
+ g_aE1kRegMap[iReg].abbrev, g_aE1kRegMap[iReg].offset, g_aE1kRegMap[iReg].size,
+ g_aE1kRegMap[iReg - 1].abbrev, g_aE1kRegMap[iReg - 1].offset, g_aE1kRegMap[iReg - 1].size),
+ VERR_INTERNAL_ERROR_4);
+
+ /*
+ * Validate configuration.
+ */
+ PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns,
+ "MAC|"
+ "CableConnected|"
+ "AdapterType|"
+ "LineSpeed|"
+ "ItrEnabled|"
+ "ItrRxEnabled|"
+ "EthernetCRC|"
+ "GSOEnabled|"
+ "LinkUpDelay|"
+ "StatNo",
+ "");
+
+ /** @todo LineSpeed unused! */
+
+ /*
+ * Get config params
+ */
+ PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
+ rc = pHlp->pfnCFGMQueryBytes(pCfg, "MAC", pThis->macConfigured.au8, sizeof(pThis->macConfigured.au8));
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get MAC address"));
+ rc = pHlp->pfnCFGMQueryBool(pCfg, "CableConnected", &pThis->fCableConnected);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'CableConnected'"));
+ rc = pHlp->pfnCFGMQueryU32(pCfg, "AdapterType", (uint32_t*)&pThis->eChip);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'AdapterType'"));
+ Assert(pThis->eChip <= E1K_CHIP_82545EM);
+
+ rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "EthernetCRC", &pThis->fEthernetCRC, true);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'EthernetCRC'"));
+
+ rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "GSOEnabled", &pThis->fGSOEnabled, true);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'GSOEnabled'"));
+
+ rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "ItrEnabled", &pThis->fItrEnabled, false);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'ItrEnabled'"));
+
+ rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "ItrRxEnabled", &pThis->fItrRxEnabled, true);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'ItrRxEnabled'"));
+
+ rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "TidEnabled", &pThis->fTidEnabled, false);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc,
+ N_("Configuration error: Failed to get the value of 'TidEnabled'"));
+
+ /*
+ * Increased the link up delay from 3 to 5 seconds to make sure a guest notices the link loss
+ * and updates its network configuration when the link is restored. See @bugref{10114}.
+ */
+ 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)
+ LogRel(("%s: WARNING! Link up delay is set to %u seconds!\n", pThis->szPrf, pThis->cMsLinkUpDelay / 1000));
+ else if (pThis->cMsLinkUpDelay == 0)
+ LogRel(("%s: WARNING! Link up delay is disabled!\n", pThis->szPrf));
+
+ uint32_t uStatNo = (uint32_t)iInstance;
+ rc = pHlp->pfnCFGMQueryU32Def(pCfg, "StatNo", &uStatNo, (uint32_t)iInstance);
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Failed to get the \"StatNo\" value"));
+
+ LogRel(("%s: Chip=%s LinkUpDelay=%ums EthernetCRC=%s GSO=%s Itr=%s ItrRx=%s TID=%s R0=%s RC=%s\n", pThis->szPrf,
+ g_aChips[pThis->eChip].pcszName, pThis->cMsLinkUpDelay,
+ pThis->fEthernetCRC ? "on" : "off",
+ pThis->fGSOEnabled ? "enabled" : "disabled",
+ pThis->fItrEnabled ? "enabled" : "disabled",
+ pThis->fItrRxEnabled ? "enabled" : "disabled",
+ pThis->fTidEnabled ? "enabled" : "disabled",
+ pDevIns->fR0Enabled ? "enabled" : "disabled",
+ pDevIns->fRCEnabled ? "enabled" : "disabled"));
+
+ /*
+ * Initialize sub-components and register everything with the VMM.
+ */
+
+ /* Initialize the EEPROM. */
+ pThisCC->eeprom.init(pThis->macConfigured);
+
+ /* Initialize internal PHY. */
+ Phy::init(&pThis->phy, iInstance, pThis->eChip == E1K_CHIP_82543GC ? PHY_EPID_M881000 : PHY_EPID_M881011);
+
+ /* Initialize critical sections. We do our own locking. */
+ rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
+ AssertRCReturn(rc, rc);
+
+ rc = PDMDevHlpCritSectInit(pDevIns, &pThis->cs, RT_SRC_POS, "E1000#%d", iInstance);
+ AssertRCReturn(rc, rc);
+ rc = PDMDevHlpCritSectInit(pDevIns, &pThis->csRx, RT_SRC_POS, "E1000#%dRX", iInstance);
+ AssertRCReturn(rc, rc);
+#ifdef E1K_WITH_TX_CS
+ rc = PDMDevHlpCritSectInit(pDevIns, &pThis->csTx, RT_SRC_POS, "E1000#%dTX", iInstance);
+ AssertRCReturn(rc, rc);
+#endif
+
+ /* Saved state registration. */
+ rc = PDMDevHlpSSMRegisterEx(pDevIns, E1K_SAVEDSTATE_VERSION, sizeof(E1KSTATE), NULL,
+ NULL, e1kR3LiveExec, NULL,
+ e1kR3SavePrep, e1kR3SaveExec, NULL,
+ e1kR3LoadPrep, e1kR3LoadExec, e1kR3LoadDone);
+ AssertRCReturn(rc, rc);
+
+ /* Set PCI config registers and register ourselves with the PCI bus. */
+ PDMPCIDEV_ASSERT_VALID(pDevIns, pDevIns->apPciDevs[0]);
+ e1kR3ConfigurePciDev(pDevIns->apPciDevs[0], pThis->eChip);
+ rc = PDMDevHlpPCIRegister(pDevIns, pDevIns->apPciDevs[0]);
+ AssertRCReturn(rc, rc);
+
+#ifdef E1K_WITH_MSI
+ PDMMSIREG MsiReg;
+ RT_ZERO(MsiReg);
+ MsiReg.cMsiVectors = 1;
+ MsiReg.iMsiCapOffset = 0x80;
+ MsiReg.iMsiNextOffset = 0x0;
+ MsiReg.fMsi64bit = false;
+ rc = PDMDevHlpPCIRegisterMsi(pDevIns, &MsiReg);
+ AssertRCReturn(rc, rc);
+#endif
+
+ /*
+ * Map our registers to memory space (region 0, see e1kR3ConfigurePciDev)
+ * From the spec (regarding flags):
+ * For registers that should be accessed as 32-bit double words,
+ * partial writes (less than a 32-bit double word) is ignored.
+ * Partial reads return all 32 bits of data regardless of the
+ * byte enables.
+ */
+ rc = PDMDevHlpMmioCreateEx(pDevIns, E1K_MM_SIZE, IOMMMIO_FLAGS_READ_DWORD | IOMMMIO_FLAGS_WRITE_ONLY_DWORD,
+ pDevIns->apPciDevs[0], 0 /*iPciRegion*/,
+ e1kMMIOWrite, e1kMMIORead, NULL /*pfnFill*/, NULL /*pvUser*/, "E1000", &pThis->hMmioRegion);
+ AssertRCReturn(rc, rc);
+ rc = PDMDevHlpPCIIORegionRegisterMmio(pDevIns, 0, E1K_MM_SIZE, PCI_ADDRESS_SPACE_MEM, pThis->hMmioRegion, NULL);
+ AssertRCReturn(rc, rc);
+
+ /* Map our registers to IO space (region 2, see e1kR3ConfigurePciDev) */
+ static IOMIOPORTDESC const s_aExtDescs[] =
+ {
+ { "IOADDR", "IOADDR", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL },
+ { "IODATA", "IODATA", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL },
+ { NULL, NULL, NULL, NULL }
+ };
+ rc = PDMDevHlpIoPortCreate(pDevIns, E1K_IOPORT_SIZE, pDevIns->apPciDevs[0], 2 /*iPciRegion*/,
+ e1kIOPortOut, e1kIOPortIn, NULL /*pvUser*/, "E1000", s_aExtDescs, &pThis->hIoPorts);
+ AssertRCReturn(rc, rc);
+ rc = PDMDevHlpPCIIORegionRegisterIo(pDevIns, 2, E1K_IOPORT_SIZE, pThis->hIoPorts);
+ AssertRCReturn(rc, rc);
+
+ /* Create transmit queue */
+ rc = PDMDevHlpTaskCreate(pDevIns, PDMTASK_F_RZ, "E1000-Xmit", e1kR3TxTaskCallback, NULL, &pThis->hTxTask);
+ AssertRCReturn(rc, rc);
+
+#ifdef E1K_TX_DELAY
+ /* Create Transmit Delay Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3TxDelayTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Xmit Delay", &pThis->hTXDTimer);
+ AssertRCReturn(rc, rc);
+ rc = PDMDevHlpTimerSetCritSect(pDevIns, pThis->hTXDTimer, &pThis->csTx);
+ AssertRCReturn(rc, rc);
+#endif /* E1K_TX_DELAY */
+
+//#ifdef E1K_USE_TX_TIMERS
+ if (pThis->fTidEnabled)
+ {
+ /* Create Transmit Interrupt Delay Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3TxIntDelayTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Xmit IRQ Delay", &pThis->hTIDTimer);
+ AssertRCReturn(rc, rc);
+
+# ifndef E1K_NO_TAD
+ /* Create Transmit Absolute Delay Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3TxAbsDelayTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Xmit Abs Delay", &pThis->hTADTimer);
+ AssertRCReturn(rc, rc);
+# endif /* E1K_NO_TAD */
+ }
+//#endif /* E1K_USE_TX_TIMERS */
+
+#ifdef E1K_USE_RX_TIMERS
+ /* Create Receive Interrupt Delay Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3RxIntDelayTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Recv IRQ Delay", &pThis->hRIDTimer);
+ AssertRCReturn(rc, rc);
+
+ /* Create Receive Absolute Delay Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3RxAbsDelayTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Recv Abs Delay", &pThis->hRADTimer);
+ AssertRCReturn(rc, rc);
+#endif /* E1K_USE_RX_TIMERS */
+
+ /* Create Late Interrupt Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3LateIntTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Late IRQ", &pThis->hIntTimer);
+ AssertRCReturn(rc, rc);
+
+ /* Create Link Up Timer */
+ rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3LinkUpTimer, pThis,
+ TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Link Up", &pThis->hLUTimer);
+ AssertRCReturn(rc, rc);
+
+ /* Register the info item */
+ char szTmp[20];
+ RTStrPrintf(szTmp, sizeof(szTmp), "e1k%d", iInstance);
+ PDMDevHlpDBGFInfoRegister(pDevIns, szTmp, "E1000 info.", e1kR3Info);
+
+ /* Status driver */
+ PPDMIBASE pBase;
+ rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThisCC->IBase, &pBase, "Status Port");
+ if (RT_FAILURE(rc))
+ return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach the status LUN"));
+ pThisCC->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
+
+ /* Network driver */
+ rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->IBase, &pThisCC->pDrvBase, "Network Port");
+ if (RT_SUCCESS(rc))
+ {
+ pThisCC->pDrvR3 = PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMINETWORKUP);
+ AssertMsgReturn(pThisCC->pDrvR3, ("Failed to obtain the PDMINETWORKUP interface!\n"), VERR_PDM_MISSING_INTERFACE_BELOW);
+
+#if 0 /** @todo @bugref{9218} ring-0 driver stuff */
+ pThisR0->pDrvR0 = PDMIBASER0_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASER0), PDMINETWORKUP);
+ pThisRC->pDrvRC = PDMIBASERC_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASERC), PDMINETWORKUP);
+#endif
+ /* Mark device as attached. */
+ pThis->fIsAttached = true;
+ }
+ else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER
+ || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME)
+ {
+ /* No error! */
+ E1kLog(("%s This adapter is not attached to any network!\n", pThis->szPrf));
+ }
+ else
+ return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach the network LUN"));
+
+ rc = PDMDevHlpSUPSemEventCreate(pDevIns, &pThis->hEventMoreRxDescAvail);
+ AssertRCReturn(rc, rc);
+
+ rc = e1kR3InitDebugHelpers();
+ AssertRCReturn(rc, rc);
+
+ e1kR3HardReset(pDevIns, pThis, pThisCC);
+
+ /*
+ * Register statistics.
+ * The /Public/ bits are official and used by session info in the GUI.
+ */
+ PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
+ "Amount of data received", "/Public/NetAdapter/%u/BytesReceived", uStatNo);
+ PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
+ "Amount of data transmitted", "/Public/NetAdapter/%u/BytesTransmitted", uStatNo);
+ PDMDevHlpSTAMRegisterF(pDevIns, &pDevIns->iInstance, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE,
+ "Device instance number", "/Public/NetAdapter/%u/%s", uStatNo, pDevIns->pReg->szName);
+
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, "ReceiveBytes", STAMUNIT_BYTES, "Amount of data received");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, "TransmitBytes", STAMUNIT_BYTES, "Amount of data transmitted");
+
+#if defined(VBOX_WITH_STATISTICS)
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOReadRZ, STAMTYPE_PROFILE, "MMIO/ReadRZ", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO reads in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOReadR3, STAMTYPE_PROFILE, "MMIO/ReadR3", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO reads in R3");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOWriteRZ, STAMTYPE_PROFILE, "MMIO/WriteRZ", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO writes in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOWriteR3, STAMTYPE_PROFILE, "MMIO/WriteR3", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO writes in R3");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatEEPROMRead, STAMTYPE_PROFILE, "EEPROM/Read", STAMUNIT_TICKS_PER_CALL, "Profiling EEPROM reads");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatEEPROMWrite, STAMTYPE_PROFILE, "EEPROM/Write", STAMUNIT_TICKS_PER_CALL, "Profiling EEPROM writes");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOReadRZ, STAMTYPE_PROFILE, "IO/ReadRZ", STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOReadR3, STAMTYPE_PROFILE, "IO/ReadR3", STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in R3");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOWriteRZ, STAMTYPE_PROFILE, "IO/WriteRZ", STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOWriteR3, STAMTYPE_PROFILE, "IO/WriteR3", STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in R3");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatLateIntTimer, STAMTYPE_PROFILE, "LateInt/Timer", STAMUNIT_TICKS_PER_CALL, "Profiling late int timer");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatLateInts, STAMTYPE_COUNTER, "LateInt/Occured", STAMUNIT_OCCURENCES, "Number of late interrupts");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIntsRaised, STAMTYPE_COUNTER, "Interrupts/Raised", STAMUNIT_OCCURENCES, "Number of raised interrupts");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIntsPrevented, STAMTYPE_COUNTER, "Interrupts/Prevented", STAMUNIT_OCCURENCES, "Number of prevented interrupts");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceive, STAMTYPE_PROFILE, "Receive/Total", STAMUNIT_TICKS_PER_CALL, "Profiling receive");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveCRC, STAMTYPE_PROFILE, "Receive/CRC", STAMUNIT_TICKS_PER_CALL, "Profiling receive checksumming");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveFilter, STAMTYPE_PROFILE, "Receive/Filter", STAMUNIT_TICKS_PER_CALL, "Profiling receive filtering");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveStore, STAMTYPE_PROFILE, "Receive/Store", STAMUNIT_TICKS_PER_CALL, "Profiling receive storing");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatRxOverflow, STAMTYPE_PROFILE, "RxOverflow", STAMUNIT_TICKS_PER_OCCURENCE, "Profiling RX overflows");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatRxOverflowWakeupRZ, STAMTYPE_COUNTER, "RxOverflowWakeupRZ", STAMUNIT_OCCURENCES, "Nr of RX overflow wakeups in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatRxOverflowWakeupR3, STAMTYPE_COUNTER, "RxOverflowWakeupR3", STAMUNIT_OCCURENCES, "Nr of RX overflow wakeups in R3");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitRZ, STAMTYPE_PROFILE, "Transmit/TotalRZ", STAMUNIT_TICKS_PER_CALL, "Profiling transmits in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitR3, STAMTYPE_PROFILE, "Transmit/TotalR3", STAMUNIT_TICKS_PER_CALL, "Profiling transmits in R3");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitSendRZ, STAMTYPE_PROFILE, "Transmit/SendRZ", STAMUNIT_TICKS_PER_CALL, "Profiling send transmit in RZ");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitSendR3, STAMTYPE_PROFILE, "Transmit/SendR3", STAMUNIT_TICKS_PER_CALL, "Profiling send transmit in R3");
+
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescCtxNormal, STAMTYPE_COUNTER, "TxDesc/ContexNormal", STAMUNIT_OCCURENCES, "Number of normal context descriptors");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescCtxTSE, STAMTYPE_COUNTER, "TxDesc/ContextTSE", STAMUNIT_OCCURENCES, "Number of TSE context descriptors");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescData, STAMTYPE_COUNTER, "TxDesc/Data", STAMUNIT_OCCURENCES, "Number of TX data descriptors");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescLegacy, STAMTYPE_COUNTER, "TxDesc/Legacy", STAMUNIT_OCCURENCES, "Number of TX legacy descriptors");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescTSEData, STAMTYPE_COUNTER, "TxDesc/TSEData", STAMUNIT_OCCURENCES, "Number of TX TSE data descriptors");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxPathFallback, STAMTYPE_COUNTER, "TxPath/Fallback", STAMUNIT_OCCURENCES, "Fallback TSE descriptor path");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxPathGSO, STAMTYPE_COUNTER, "TxPath/GSO", STAMUNIT_OCCURENCES, "GSO TSE descriptor path");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxPathRegular, STAMTYPE_COUNTER, "TxPath/Normal", STAMUNIT_OCCURENCES, "Regular descriptor path");
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->StatPHYAccesses, STAMTYPE_COUNTER, "PHYAccesses", STAMUNIT_OCCURENCES, "Number of PHY accesses");
+ for (unsigned iReg = 0; iReg < E1K_NUM_OF_REGS; iReg++)
+ {
+ PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatRegReads[iReg], STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
+ g_aE1kRegMap[iReg].name, "Regs/%s-Reads", g_aE1kRegMap[iReg].abbrev);
+ PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatRegWrites[iReg], STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
+ g_aE1kRegMap[iReg].name, "Regs/%s-Writes", g_aE1kRegMap[iReg].abbrev);
+ }
+#endif /* VBOX_WITH_STATISTICS */
+
+#ifdef E1K_INT_STATS
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->u64ArmedAt, STAMTYPE_U64, "u64ArmedAt", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatMaxTxDelay, STAMTYPE_U64, "uStatMaxTxDelay", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatInt, STAMTYPE_U32, "uStatInt", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntTry, STAMTYPE_U32, "uStatIntTry", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntLower, STAMTYPE_U32, "uStatIntLower", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatNoIntICR, STAMTYPE_U32, "uStatNoIntICR", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->iStatIntLost, STAMTYPE_U32, "iStatIntLost", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->iStatIntLostOne, STAMTYPE_U32, "iStatIntLostOne", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntIMS, STAMTYPE_U32, "uStatIntIMS", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntSkip, STAMTYPE_U32, "uStatIntSkip", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntLate, STAMTYPE_U32, "uStatIntLate", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntMasked, STAMTYPE_U32, "uStatIntMasked", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntEarly, STAMTYPE_U32, "uStatIntEarly", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntRx, STAMTYPE_U32, "uStatIntRx", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntTx, STAMTYPE_U32, "uStatIntTx", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntICS, STAMTYPE_U32, "uStatIntICS", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntRDTR, STAMTYPE_U32, "uStatIntRDTR", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntRXDMT0, STAMTYPE_U32, "uStatIntRXDMT0", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntTXQE, STAMTYPE_U32, "uStatIntTXQE", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxNoRS, STAMTYPE_U32, "uStatTxNoRS", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxIDE, STAMTYPE_U32, "uStatTxIDE", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxDelayed, STAMTYPE_U32, "uStatTxDelayed", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxDelayExp, STAMTYPE_U32, "uStatTxDelayExp", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTAD, STAMTYPE_U32, "uStatTAD", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTID, STAMTYPE_U32, "uStatTID", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatRAD, STAMTYPE_U32, "uStatRAD", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatRID, STAMTYPE_U32, "uStatRID", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatRxFrm, STAMTYPE_U32, "uStatRxFrm", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxFrm, STAMTYPE_U32, "uStatTxFrm", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatDescCtx, STAMTYPE_U32, "uStatDescCtx", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatDescDat, STAMTYPE_U32, "uStatDescDat", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatDescLeg, STAMTYPE_U32, "uStatDescLeg", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx1514, STAMTYPE_U32, "uStatTx1514", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx2962, STAMTYPE_U32, "uStatTx2962", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx4410, STAMTYPE_U32, "uStatTx4410", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx5858, STAMTYPE_U32, "uStatTx5858", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx7306, STAMTYPE_U32, "uStatTx7306", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx8754, STAMTYPE_U32, "uStatTx8754", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx16384, STAMTYPE_U32, "uStatTx16384", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx32768, STAMTYPE_U32, "uStatTx32768", STAMUNIT_NS, NULL);
+ PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxLarge, STAMTYPE_U32, "uStatTxLarge", STAMUNIT_NS, NULL);
+#endif /* E1K_INT_STATS */
+
+ return VINF_SUCCESS;
+}
+
+#else /* !IN_RING3 */
+
+/**
+ * @callback_method_impl{PDMDEVREGR0,pfnConstruct}
+ */
+static DECLCALLBACK(int) e1kRZConstruct(PPDMDEVINS pDevIns)
+{
+ PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
+ PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
+ PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
+
+ /* Initialize context specific state data: */
+ pThisCC->CTX_SUFF(pDevIns) = pDevIns;
+ /** @todo @bugref{9218} ring-0 driver stuff */
+ pThisCC->CTX_SUFF(pDrv) = NULL;
+ pThisCC->CTX_SUFF(pTxSg) = NULL;
+
+ /* Configure critical sections the same way: */
+ int rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
+ AssertRCReturn(rc, rc);
+
+ /* Set up MMIO and I/O port callbacks for this context: */
+ rc = PDMDevHlpMmioSetUpContext(pDevIns, pThis->hMmioRegion, e1kMMIOWrite, e1kMMIORead, NULL /*pvUser*/);
+ AssertRCReturn(rc, rc);
+
+ rc = PDMDevHlpIoPortSetUpContext(pDevIns, pThis->hIoPorts, e1kIOPortOut, e1kIOPortIn, NULL /*pvUser*/);
+ AssertRCReturn(rc, rc);
+
+ return VINF_SUCCESS;
+}
+
+#endif /* !IN_RING3 */
+
+/**
+ * The device registration structure.
+ */
+const PDMDEVREG g_DeviceE1000 =
+{
+ /* .u32version = */ PDM_DEVREG_VERSION,
+ /* .uReserved0 = */ 0,
+ /* .szName = */ "e1000",
+ /* .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(E1KSTATE),
+ /* .cbInstanceCC = */ sizeof(E1KSTATECC),
+ /* .cbInstanceRC = */ sizeof(E1KSTATERC),
+ /* .cMaxPciDevices = */ 1,
+ /* .cMaxMsixVectors = */ 0,
+ /* .pszDescription = */ "Intel PRO/1000 MT Desktop Ethernet.",
+#if defined(IN_RING3)
+ /* .pszRCMod = */ "VBoxDDRC.rc",
+ /* .pszR0Mod = */ "VBoxDDR0.r0",
+ /* .pfnConstruct = */ e1kR3Construct,
+ /* .pfnDestruct = */ e1kR3Destruct,
+ /* .pfnRelocate = */ e1kR3Relocate,
+ /* .pfnMemSetup = */ NULL,
+ /* .pfnPowerOn = */ NULL,
+ /* .pfnReset = */ e1kR3Reset,
+ /* .pfnSuspend = */ e1kR3Suspend,
+ /* .pfnResume = */ NULL,
+ /* .pfnAttach = */ e1kR3Attach,
+ /* .pfnDeatch = */ e1kR3Detach,
+ /* .pfnQueryInterface = */ NULL,
+ /* .pfnInitComplete = */ NULL,
+ /* .pfnPowerOff = */ e1kR3PowerOff,
+ /* .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 = */ e1kRZConstruct,
+ /* .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 = */ e1kRZConstruct,
+ /* .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 */