VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DevE1000.cpp@ 94959

Last change on this file since 94959 was 94274, checked in by vboxsync, 3 years ago

Dev/E1000: bugref:10197 Improved handling of mixed descriptor types, VLAN tag additional check

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 339.1 KB
Line 
1/* $Id: DevE1000.cpp 94274 2022-03-16 17:36:35Z vboxsync $ */
2/** @file
3 * DevE1000 - Intel 82540EM Ethernet Controller Emulation.
4 *
5 * Implemented in accordance with the specification:
6 *
7 * PCI/PCI-X Family of Gigabit Ethernet Controllers Software Developer's Manual
8 * 82540EP/EM, 82541xx, 82544GC/EI, 82545GM/EM, 82546GB/EB, and 82547xx
9 *
10 * 317453-002 Revision 3.5
11 *
12 * @todo IPv6 checksum offloading support
13 * @todo Flexible Filter / Wakeup (optional?)
14 */
15
16/*
17 * Copyright (C) 2007-2022 Oracle Corporation
18 *
19 * This file is part of VirtualBox Open Source Edition (OSE), as
20 * available from http://www.virtualbox.org. This file is free software;
21 * you can redistribute it and/or modify it under the terms of the GNU
22 * General Public License (GPL) as published by the Free Software
23 * Foundation, in version 2 as it comes in the "COPYING" file of the
24 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
25 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
26 */
27
28
29/*********************************************************************************************************************************
30* Header Files *
31*********************************************************************************************************************************/
32#define LOG_GROUP LOG_GROUP_DEV_E1000
33#include <iprt/crc.h>
34#include <iprt/ctype.h>
35#include <iprt/net.h>
36#include <iprt/semaphore.h>
37#include <iprt/string.h>
38#include <iprt/time.h>
39#include <iprt/uuid.h>
40#include <VBox/vmm/pdmdev.h>
41#include <VBox/vmm/pdmnetifs.h>
42#include <VBox/vmm/pdmnetinline.h>
43#include <VBox/param.h>
44#include "VBoxDD.h"
45
46#include "DevEEPROM.h"
47#include "DevE1000Phy.h"
48
49
50/*********************************************************************************************************************************
51* Defined Constants And Macros *
52*********************************************************************************************************************************/
53/** @name E1000 Build Options
54 * @{ */
55/** @def E1K_INIT_RA0
56 * E1K_INIT_RA0 forces E1000 to set the first entry in Receive Address filter
57 * table to MAC address obtained from CFGM. Most guests read MAC address from
58 * EEPROM and write it to RA[0] explicitly, but Mac OS X seems to depend on it
59 * being already set (see @bugref{4657}).
60 */
61#define E1K_INIT_RA0
62/** @def E1K_LSC_ON_RESET
63 * E1K_LSC_ON_RESET causes e1000 to generate Link Status Change
64 * interrupt after hard reset. This makes the E1K_LSC_ON_SLU option unnecessary.
65 * With unplugged cable, LSC is triggerred for 82543GC only.
66 */
67#define E1K_LSC_ON_RESET
68/** @def E1K_LSC_ON_SLU
69 * E1K_LSC_ON_SLU causes E1000 to generate Link Status Change interrupt when
70 * the guest driver brings up the link via STATUS.LU bit. Again the only guest
71 * that requires it is Mac OS X (see @bugref{4657}).
72 */
73//#define E1K_LSC_ON_SLU
74/** @def E1K_INIT_LINKUP_DELAY
75 * E1K_INIT_LINKUP_DELAY prevents the link going up while the driver is still
76 * in init (see @bugref{8624}).
77 */
78#define E1K_INIT_LINKUP_DELAY_US (2000 * 1000)
79/** @def E1K_IMS_INT_DELAY_NS
80 * E1K_IMS_INT_DELAY_NS prevents interrupt storms in Windows guests on enabling
81 * interrupts (see @bugref{8624}).
82 */
83#define E1K_IMS_INT_DELAY_NS 100
84/** @def E1K_TX_DELAY
85 * E1K_TX_DELAY aims to improve guest-host transfer rate for TCP streams by
86 * preventing packets to be sent immediately. It allows to send several
87 * packets in a batch reducing the number of acknowledgments. Note that it
88 * effectively disables R0 TX path, forcing sending in R3.
89 */
90//#define E1K_TX_DELAY 150
91/** @def E1K_USE_TX_TIMERS
92 * E1K_USE_TX_TIMERS aims to reduce the number of generated TX interrupts if a
93 * guest driver set the delays via the Transmit Interrupt Delay Value (TIDV)
94 * register. Enabling it showed no positive effects on existing guests so it
95 * stays disabled. See sections 3.2.7.1 and 3.4.3.1 in "8254x Family of Gigabit
96 * Ethernet Controllers Software Developer’s Manual" for more detailed
97 * explanation.
98 */
99//#define E1K_USE_TX_TIMERS
100/** @def E1K_NO_TAD
101 * E1K_NO_TAD disables one of two timers enabled by E1K_USE_TX_TIMERS, the
102 * Transmit Absolute Delay time. This timer sets the maximum time interval
103 * during which TX interrupts can be postponed (delayed). It has no effect
104 * if E1K_USE_TX_TIMERS is not defined.
105 */
106//#define E1K_NO_TAD
107/** @def E1K_REL_DEBUG
108 * E1K_REL_DEBUG enables debug logging of l1, l2, l3 in release build.
109 */
110//#define E1K_REL_DEBUG
111/** @def E1K_INT_STATS
112 * E1K_INT_STATS enables collection of internal statistics used for
113 * debugging of delayed interrupts, etc.
114 */
115#define E1K_INT_STATS
116/** @def E1K_WITH_MSI
117 * E1K_WITH_MSI enables rudimentary MSI support. Not implemented.
118 */
119//#define E1K_WITH_MSI
120/** @def E1K_WITH_TX_CS
121 * E1K_WITH_TX_CS protects e1kXmitPending with a critical section.
122 */
123#define E1K_WITH_TX_CS
124/** @def E1K_WITH_TXD_CACHE
125 * E1K_WITH_TXD_CACHE causes E1000 to fetch multiple TX descriptors in a
126 * single physical memory read (or two if it wraps around the end of TX
127 * descriptor ring). It is required for proper functioning of bandwidth
128 * resource control as it allows to compute exact sizes of packets prior
129 * to allocating their buffers (see @bugref{5582}).
130 */
131#define E1K_WITH_TXD_CACHE
132/** @def E1K_WITH_RXD_CACHE
133 * E1K_WITH_RXD_CACHE causes E1000 to fetch multiple RX descriptors in a
134 * single physical memory read (or two if it wraps around the end of RX
135 * descriptor ring). Intel's packet driver for DOS needs this option in
136 * order to work properly (see @bugref{6217}).
137 */
138#define E1K_WITH_RXD_CACHE
139/** @def E1K_WITH_PREREG_MMIO
140 * E1K_WITH_PREREG_MMIO enables a new style MMIO registration and is
141 * currently only done for testing the relateted PDM, IOM and PGM code. */
142//#define E1K_WITH_PREREG_MMIO
143/* @} */
144/* End of Options ************************************************************/
145
146#ifdef E1K_WITH_TXD_CACHE
147/**
148 * E1K_TXD_CACHE_SIZE specifies the maximum number of TX descriptors stored
149 * in the state structure. It limits the amount of descriptors loaded in one
150 * batch read. For example, Linux guest may use up to 20 descriptors per
151 * TSE packet. The largest TSE packet seen (Windows guest) was 45 descriptors.
152 */
153# define E1K_TXD_CACHE_SIZE 64u
154#endif /* E1K_WITH_TXD_CACHE */
155
156#ifdef E1K_WITH_RXD_CACHE
157/**
158 * E1K_RXD_CACHE_SIZE specifies the maximum number of RX descriptors stored
159 * in the state structure. It limits the amount of descriptors loaded in one
160 * batch read. For example, XP guest adds 15 RX descriptors at a time.
161 */
162# define E1K_RXD_CACHE_SIZE 16u
163#endif /* E1K_WITH_RXD_CACHE */
164
165
166/* Little helpers ************************************************************/
167#undef htons
168#undef ntohs
169#undef htonl
170#undef ntohl
171#define htons(x) ((((x) & 0xff00) >> 8) | (((x) & 0x00ff) << 8))
172#define ntohs(x) htons(x)
173#define htonl(x) ASMByteSwapU32(x)
174#define ntohl(x) htonl(x)
175
176#ifndef DEBUG
177# ifdef E1K_REL_DEBUG
178# define DEBUG
179# define E1kLog(a) LogRel(a)
180# define E1kLog2(a) LogRel(a)
181# define E1kLog3(a) LogRel(a)
182# define E1kLogX(x, a) LogRel(a)
183//# define E1kLog3(a) do {} while (0)
184# else
185# define E1kLog(a) do {} while (0)
186# define E1kLog2(a) do {} while (0)
187# define E1kLog3(a) do {} while (0)
188# define E1kLogX(x, a) do {} while (0)
189# endif
190#else
191# define E1kLog(a) Log(a)
192# define E1kLog2(a) Log2(a)
193# define E1kLog3(a) Log3(a)
194# define E1kLogX(x, a) LogIt(x, LOG_GROUP, a)
195//# define E1kLog(a) do {} while (0)
196//# define E1kLog2(a) do {} while (0)
197//# define E1kLog3(a) do {} while (0)
198#endif
199
200#if 0
201# define LOG_ENABLED
202# define E1kLogRel(a) LogRel(a)
203# undef Log6
204# define Log6(a) LogRel(a)
205#else
206# define E1kLogRel(a) do { } while (0)
207#endif
208
209//#undef DEBUG
210
211#define E1K_RELOCATE(p, o) *(RTHCUINTPTR *)&p += o
212
213#define E1K_INC_CNT32(cnt) \
214do { \
215 if (cnt < UINT32_MAX) \
216 cnt++; \
217} while (0)
218
219#define E1K_ADD_CNT64(cntLo, cntHi, val) \
220do { \
221 uint64_t u64Cnt = RT_MAKE_U64(cntLo, cntHi); \
222 uint64_t tmp = u64Cnt; \
223 u64Cnt += val; \
224 if (tmp > u64Cnt ) \
225 u64Cnt = UINT64_MAX; \
226 cntLo = (uint32_t)u64Cnt; \
227 cntHi = (uint32_t)(u64Cnt >> 32); \
228} while (0)
229
230#ifdef E1K_INT_STATS
231# define E1K_INC_ISTAT_CNT(cnt) do { ++cnt; } while (0)
232#else /* E1K_INT_STATS */
233# define E1K_INC_ISTAT_CNT(cnt) do { } while (0)
234#endif /* E1K_INT_STATS */
235
236
237/*****************************************************************************/
238
239typedef uint32_t E1KCHIP;
240#define E1K_CHIP_82540EM 0
241#define E1K_CHIP_82543GC 1
242#define E1K_CHIP_82545EM 2
243
244#ifdef IN_RING3
245/** Different E1000 chips. */
246static const struct E1kChips
247{
248 uint16_t uPCIVendorId;
249 uint16_t uPCIDeviceId;
250 uint16_t uPCISubsystemVendorId;
251 uint16_t uPCISubsystemId;
252 const char *pcszName;
253} g_aChips[] =
254{
255 /* Vendor Device SSVendor SubSys Name */
256 { 0x8086,
257 /* Temporary code, as MSI-aware driver dislike 0x100E. How to do that right? */
258# ifdef E1K_WITH_MSI
259 0x105E,
260# else
261 0x100E,
262# endif
263 0x8086, 0x001E, "82540EM" }, /* Intel 82540EM-A in Intel PRO/1000 MT Desktop */
264 { 0x8086, 0x1004, 0x8086, 0x1004, "82543GC" }, /* Intel 82543GC in Intel PRO/1000 T Server */
265 { 0x8086, 0x100F, 0x15AD, 0x0750, "82545EM" } /* Intel 82545EM-A in VMWare Network Adapter */
266};
267#endif /* IN_RING3 */
268
269
270/* The size of register area mapped to I/O space */
271#define E1K_IOPORT_SIZE 0x8
272/* The size of memory-mapped register area */
273#define E1K_MM_SIZE 0x20000
274
275#define E1K_MAX_TX_PKT_SIZE 16288
276#define E1K_MAX_RX_PKT_SIZE 16384
277
278/*****************************************************************************/
279
280#ifndef VBOX_DEVICE_STRUCT_TESTCASE
281/** Gets the specfieid bits from the register. */
282#define GET_BITS(reg, bits) ((reg & reg##_##bits##_MASK) >> reg##_##bits##_SHIFT)
283#define GET_BITS_V(val, reg, bits) ((val & reg##_##bits##_MASK) >> reg##_##bits##_SHIFT)
284#define BITS(reg, bits, bitval) (bitval << reg##_##bits##_SHIFT)
285#define SET_BITS(reg, bits, bitval) do { reg = (reg & ~reg##_##bits##_MASK) | (bitval << reg##_##bits##_SHIFT); } while (0)
286#define SET_BITS_V(val, reg, bits, bitval) do { val = (val & ~reg##_##bits##_MASK) | (bitval << reg##_##bits##_SHIFT); } while (0)
287
288#define CTRL_SLU UINT32_C(0x00000040)
289#define CTRL_MDIO UINT32_C(0x00100000)
290#define CTRL_MDC UINT32_C(0x00200000)
291#define CTRL_MDIO_DIR UINT32_C(0x01000000)
292#define CTRL_MDC_DIR UINT32_C(0x02000000)
293#define CTRL_RESET UINT32_C(0x04000000)
294#define CTRL_VME UINT32_C(0x40000000)
295
296#define STATUS_LU UINT32_C(0x00000002)
297#define STATUS_TXOFF UINT32_C(0x00000010)
298
299#define EECD_EE_WIRES UINT32_C(0x0F)
300#define EECD_EE_REQ UINT32_C(0x40)
301#define EECD_EE_GNT UINT32_C(0x80)
302
303#define EERD_START UINT32_C(0x00000001)
304#define EERD_DONE UINT32_C(0x00000010)
305#define EERD_DATA_MASK UINT32_C(0xFFFF0000)
306#define EERD_DATA_SHIFT 16
307#define EERD_ADDR_MASK UINT32_C(0x0000FF00)
308#define EERD_ADDR_SHIFT 8
309
310#define MDIC_DATA_MASK UINT32_C(0x0000FFFF)
311#define MDIC_DATA_SHIFT 0
312#define MDIC_REG_MASK UINT32_C(0x001F0000)
313#define MDIC_REG_SHIFT 16
314#define MDIC_PHY_MASK UINT32_C(0x03E00000)
315#define MDIC_PHY_SHIFT 21
316#define MDIC_OP_WRITE UINT32_C(0x04000000)
317#define MDIC_OP_READ UINT32_C(0x08000000)
318#define MDIC_READY UINT32_C(0x10000000)
319#define MDIC_INT_EN UINT32_C(0x20000000)
320#define MDIC_ERROR UINT32_C(0x40000000)
321
322#define TCTL_EN UINT32_C(0x00000002)
323#define TCTL_PSP UINT32_C(0x00000008)
324
325#define RCTL_EN UINT32_C(0x00000002)
326#define RCTL_UPE UINT32_C(0x00000008)
327#define RCTL_MPE UINT32_C(0x00000010)
328#define RCTL_LPE UINT32_C(0x00000020)
329#define RCTL_LBM_MASK UINT32_C(0x000000C0)
330#define RCTL_LBM_SHIFT 6
331#define RCTL_RDMTS_MASK UINT32_C(0x00000300)
332#define RCTL_RDMTS_SHIFT 8
333#define RCTL_LBM_TCVR UINT32_C(3) /**< PHY or external SerDes loopback. */
334#define RCTL_MO_MASK UINT32_C(0x00003000)
335#define RCTL_MO_SHIFT 12
336#define RCTL_BAM UINT32_C(0x00008000)
337#define RCTL_BSIZE_MASK UINT32_C(0x00030000)
338#define RCTL_BSIZE_SHIFT 16
339#define RCTL_VFE UINT32_C(0x00040000)
340#define RCTL_CFIEN UINT32_C(0x00080000)
341#define RCTL_CFI UINT32_C(0x00100000)
342#define RCTL_BSEX UINT32_C(0x02000000)
343#define RCTL_SECRC UINT32_C(0x04000000)
344
345#define ICR_TXDW UINT32_C(0x00000001)
346#define ICR_TXQE UINT32_C(0x00000002)
347#define ICR_LSC UINT32_C(0x00000004)
348#define ICR_RXDMT0 UINT32_C(0x00000010)
349#define ICR_RXT0 UINT32_C(0x00000080)
350#define ICR_TXD_LOW UINT32_C(0x00008000)
351#define RDTR_FPD UINT32_C(0x80000000)
352
353#define PBA_st ((PBAST*)(pThis->auRegs + PBA_IDX))
354typedef struct
355{
356 unsigned rxa : 7;
357 unsigned rxa_r : 9;
358 unsigned txa : 16;
359} PBAST;
360AssertCompileSize(PBAST, 4);
361
362#define TXDCTL_WTHRESH_MASK 0x003F0000
363#define TXDCTL_WTHRESH_SHIFT 16
364#define TXDCTL_LWTHRESH_MASK 0xFE000000
365#define TXDCTL_LWTHRESH_SHIFT 25
366
367#define RXCSUM_PCSS_MASK UINT32_C(0x000000FF)
368#define RXCSUM_PCSS_SHIFT 0
369
370/** @name Register access macros
371 * @remarks These ASSUME alocal variable @a pThis of type PE1KSTATE.
372 * @{ */
373#define CTRL pThis->auRegs[CTRL_IDX]
374#define STATUS pThis->auRegs[STATUS_IDX]
375#define EECD pThis->auRegs[EECD_IDX]
376#define EERD pThis->auRegs[EERD_IDX]
377#define CTRL_EXT pThis->auRegs[CTRL_EXT_IDX]
378#define FLA pThis->auRegs[FLA_IDX]
379#define MDIC pThis->auRegs[MDIC_IDX]
380#define FCAL pThis->auRegs[FCAL_IDX]
381#define FCAH pThis->auRegs[FCAH_IDX]
382#define FCT pThis->auRegs[FCT_IDX]
383#define VET pThis->auRegs[VET_IDX]
384#define ICR pThis->auRegs[ICR_IDX]
385#define ITR pThis->auRegs[ITR_IDX]
386#define ICS pThis->auRegs[ICS_IDX]
387#define IMS pThis->auRegs[IMS_IDX]
388#define IMC pThis->auRegs[IMC_IDX]
389#define RCTL pThis->auRegs[RCTL_IDX]
390#define FCTTV pThis->auRegs[FCTTV_IDX]
391#define TXCW pThis->auRegs[TXCW_IDX]
392#define RXCW pThis->auRegs[RXCW_IDX]
393#define TCTL pThis->auRegs[TCTL_IDX]
394#define TIPG pThis->auRegs[TIPG_IDX]
395#define AIFS pThis->auRegs[AIFS_IDX]
396#define LEDCTL pThis->auRegs[LEDCTL_IDX]
397#define PBA pThis->auRegs[PBA_IDX]
398#define FCRTL pThis->auRegs[FCRTL_IDX]
399#define FCRTH pThis->auRegs[FCRTH_IDX]
400#define RDFH pThis->auRegs[RDFH_IDX]
401#define RDFT pThis->auRegs[RDFT_IDX]
402#define RDFHS pThis->auRegs[RDFHS_IDX]
403#define RDFTS pThis->auRegs[RDFTS_IDX]
404#define RDFPC pThis->auRegs[RDFPC_IDX]
405#define RDBAL pThis->auRegs[RDBAL_IDX]
406#define RDBAH pThis->auRegs[RDBAH_IDX]
407#define RDLEN pThis->auRegs[RDLEN_IDX]
408#define RDH pThis->auRegs[RDH_IDX]
409#define RDT pThis->auRegs[RDT_IDX]
410#define RDTR pThis->auRegs[RDTR_IDX]
411#define RXDCTL pThis->auRegs[RXDCTL_IDX]
412#define RADV pThis->auRegs[RADV_IDX]
413#define RSRPD pThis->auRegs[RSRPD_IDX]
414#define TXDMAC pThis->auRegs[TXDMAC_IDX]
415#define TDFH pThis->auRegs[TDFH_IDX]
416#define TDFT pThis->auRegs[TDFT_IDX]
417#define TDFHS pThis->auRegs[TDFHS_IDX]
418#define TDFTS pThis->auRegs[TDFTS_IDX]
419#define TDFPC pThis->auRegs[TDFPC_IDX]
420#define TDBAL pThis->auRegs[TDBAL_IDX]
421#define TDBAH pThis->auRegs[TDBAH_IDX]
422#define TDLEN pThis->auRegs[TDLEN_IDX]
423#define TDH pThis->auRegs[TDH_IDX]
424#define TDT pThis->auRegs[TDT_IDX]
425#define TIDV pThis->auRegs[TIDV_IDX]
426#define TXDCTL pThis->auRegs[TXDCTL_IDX]
427#define TADV pThis->auRegs[TADV_IDX]
428#define TSPMT pThis->auRegs[TSPMT_IDX]
429#define CRCERRS pThis->auRegs[CRCERRS_IDX]
430#define ALGNERRC pThis->auRegs[ALGNERRC_IDX]
431#define SYMERRS pThis->auRegs[SYMERRS_IDX]
432#define RXERRC pThis->auRegs[RXERRC_IDX]
433#define MPC pThis->auRegs[MPC_IDX]
434#define SCC pThis->auRegs[SCC_IDX]
435#define ECOL pThis->auRegs[ECOL_IDX]
436#define MCC pThis->auRegs[MCC_IDX]
437#define LATECOL pThis->auRegs[LATECOL_IDX]
438#define COLC pThis->auRegs[COLC_IDX]
439#define DC pThis->auRegs[DC_IDX]
440#define TNCRS pThis->auRegs[TNCRS_IDX]
441/* #define SEC pThis->auRegs[SEC_IDX] Conflict with sys/time.h */
442#define CEXTERR pThis->auRegs[CEXTERR_IDX]
443#define RLEC pThis->auRegs[RLEC_IDX]
444#define XONRXC pThis->auRegs[XONRXC_IDX]
445#define XONTXC pThis->auRegs[XONTXC_IDX]
446#define XOFFRXC pThis->auRegs[XOFFRXC_IDX]
447#define XOFFTXC pThis->auRegs[XOFFTXC_IDX]
448#define FCRUC pThis->auRegs[FCRUC_IDX]
449#define PRC64 pThis->auRegs[PRC64_IDX]
450#define PRC127 pThis->auRegs[PRC127_IDX]
451#define PRC255 pThis->auRegs[PRC255_IDX]
452#define PRC511 pThis->auRegs[PRC511_IDX]
453#define PRC1023 pThis->auRegs[PRC1023_IDX]
454#define PRC1522 pThis->auRegs[PRC1522_IDX]
455#define GPRC pThis->auRegs[GPRC_IDX]
456#define BPRC pThis->auRegs[BPRC_IDX]
457#define MPRC pThis->auRegs[MPRC_IDX]
458#define GPTC pThis->auRegs[GPTC_IDX]
459#define GORCL pThis->auRegs[GORCL_IDX]
460#define GORCH pThis->auRegs[GORCH_IDX]
461#define GOTCL pThis->auRegs[GOTCL_IDX]
462#define GOTCH pThis->auRegs[GOTCH_IDX]
463#define RNBC pThis->auRegs[RNBC_IDX]
464#define RUC pThis->auRegs[RUC_IDX]
465#define RFC pThis->auRegs[RFC_IDX]
466#define ROC pThis->auRegs[ROC_IDX]
467#define RJC pThis->auRegs[RJC_IDX]
468#define MGTPRC pThis->auRegs[MGTPRC_IDX]
469#define MGTPDC pThis->auRegs[MGTPDC_IDX]
470#define MGTPTC pThis->auRegs[MGTPTC_IDX]
471#define TORL pThis->auRegs[TORL_IDX]
472#define TORH pThis->auRegs[TORH_IDX]
473#define TOTL pThis->auRegs[TOTL_IDX]
474#define TOTH pThis->auRegs[TOTH_IDX]
475#define TPR pThis->auRegs[TPR_IDX]
476#define TPT pThis->auRegs[TPT_IDX]
477#define PTC64 pThis->auRegs[PTC64_IDX]
478#define PTC127 pThis->auRegs[PTC127_IDX]
479#define PTC255 pThis->auRegs[PTC255_IDX]
480#define PTC511 pThis->auRegs[PTC511_IDX]
481#define PTC1023 pThis->auRegs[PTC1023_IDX]
482#define PTC1522 pThis->auRegs[PTC1522_IDX]
483#define MPTC pThis->auRegs[MPTC_IDX]
484#define BPTC pThis->auRegs[BPTC_IDX]
485#define TSCTC pThis->auRegs[TSCTC_IDX]
486#define TSCTFC pThis->auRegs[TSCTFC_IDX]
487#define RXCSUM pThis->auRegs[RXCSUM_IDX]
488#define WUC pThis->auRegs[WUC_IDX]
489#define WUFC pThis->auRegs[WUFC_IDX]
490#define WUS pThis->auRegs[WUS_IDX]
491#define MANC pThis->auRegs[MANC_IDX]
492#define IPAV pThis->auRegs[IPAV_IDX]
493#define WUPL pThis->auRegs[WUPL_IDX]
494/** @} */
495#endif /* VBOX_DEVICE_STRUCT_TESTCASE */
496
497/**
498 * Indices of memory-mapped registers in register table.
499 */
500typedef enum
501{
502 CTRL_IDX,
503 STATUS_IDX,
504 EECD_IDX,
505 EERD_IDX,
506 CTRL_EXT_IDX,
507 FLA_IDX,
508 MDIC_IDX,
509 FCAL_IDX,
510 FCAH_IDX,
511 FCT_IDX,
512 VET_IDX,
513 ICR_IDX,
514 ITR_IDX,
515 ICS_IDX,
516 IMS_IDX,
517 IMC_IDX,
518 RCTL_IDX,
519 FCTTV_IDX,
520 TXCW_IDX,
521 RXCW_IDX,
522 TCTL_IDX,
523 TIPG_IDX,
524 AIFS_IDX,
525 LEDCTL_IDX,
526 PBA_IDX,
527 FCRTL_IDX,
528 FCRTH_IDX,
529 RDFH_IDX,
530 RDFT_IDX,
531 RDFHS_IDX,
532 RDFTS_IDX,
533 RDFPC_IDX,
534 RDBAL_IDX,
535 RDBAH_IDX,
536 RDLEN_IDX,
537 RDH_IDX,
538 RDT_IDX,
539 RDTR_IDX,
540 RXDCTL_IDX,
541 RADV_IDX,
542 RSRPD_IDX,
543 TXDMAC_IDX,
544 TDFH_IDX,
545 TDFT_IDX,
546 TDFHS_IDX,
547 TDFTS_IDX,
548 TDFPC_IDX,
549 TDBAL_IDX,
550 TDBAH_IDX,
551 TDLEN_IDX,
552 TDH_IDX,
553 TDT_IDX,
554 TIDV_IDX,
555 TXDCTL_IDX,
556 TADV_IDX,
557 TSPMT_IDX,
558 CRCERRS_IDX,
559 ALGNERRC_IDX,
560 SYMERRS_IDX,
561 RXERRC_IDX,
562 MPC_IDX,
563 SCC_IDX,
564 ECOL_IDX,
565 MCC_IDX,
566 LATECOL_IDX,
567 COLC_IDX,
568 DC_IDX,
569 TNCRS_IDX,
570 SEC_IDX,
571 CEXTERR_IDX,
572 RLEC_IDX,
573 XONRXC_IDX,
574 XONTXC_IDX,
575 XOFFRXC_IDX,
576 XOFFTXC_IDX,
577 FCRUC_IDX,
578 PRC64_IDX,
579 PRC127_IDX,
580 PRC255_IDX,
581 PRC511_IDX,
582 PRC1023_IDX,
583 PRC1522_IDX,
584 GPRC_IDX,
585 BPRC_IDX,
586 MPRC_IDX,
587 GPTC_IDX,
588 GORCL_IDX,
589 GORCH_IDX,
590 GOTCL_IDX,
591 GOTCH_IDX,
592 RNBC_IDX,
593 RUC_IDX,
594 RFC_IDX,
595 ROC_IDX,
596 RJC_IDX,
597 MGTPRC_IDX,
598 MGTPDC_IDX,
599 MGTPTC_IDX,
600 TORL_IDX,
601 TORH_IDX,
602 TOTL_IDX,
603 TOTH_IDX,
604 TPR_IDX,
605 TPT_IDX,
606 PTC64_IDX,
607 PTC127_IDX,
608 PTC255_IDX,
609 PTC511_IDX,
610 PTC1023_IDX,
611 PTC1522_IDX,
612 MPTC_IDX,
613 BPTC_IDX,
614 TSCTC_IDX,
615 TSCTFC_IDX,
616 RXCSUM_IDX,
617 WUC_IDX,
618 WUFC_IDX,
619 WUS_IDX,
620 MANC_IDX,
621 IPAV_IDX,
622 WUPL_IDX,
623 MTA_IDX,
624 RA_IDX,
625 VFTA_IDX,
626 IP4AT_IDX,
627 IP6AT_IDX,
628 WUPM_IDX,
629 FFLT_IDX,
630 FFMT_IDX,
631 FFVT_IDX,
632 PBM_IDX,
633 RA_82542_IDX,
634 MTA_82542_IDX,
635 VFTA_82542_IDX,
636 E1K_NUM_OF_REGS
637} E1kRegIndex;
638
639#define E1K_NUM_OF_32BIT_REGS MTA_IDX
640/** The number of registers with strictly increasing offset. */
641#define E1K_NUM_OF_BINARY_SEARCHABLE (WUPL_IDX + 1)
642
643
644/**
645 * Define E1000-specific EEPROM layout.
646 */
647struct E1kEEPROM
648{
649 public:
650 EEPROM93C46 eeprom;
651
652#ifdef IN_RING3
653 /**
654 * Initialize EEPROM content.
655 *
656 * @param macAddr MAC address of E1000.
657 */
658 void init(RTMAC &macAddr)
659 {
660 eeprom.init();
661 memcpy(eeprom.m_au16Data, macAddr.au16, sizeof(macAddr.au16));
662 eeprom.m_au16Data[0x04] = 0xFFFF;
663 /*
664 * bit 3 - full support for power management
665 * bit 10 - full duplex
666 */
667 eeprom.m_au16Data[0x0A] = 0x4408;
668 eeprom.m_au16Data[0x0B] = 0x001E;
669 eeprom.m_au16Data[0x0C] = 0x8086;
670 eeprom.m_au16Data[0x0D] = 0x100E;
671 eeprom.m_au16Data[0x0E] = 0x8086;
672 eeprom.m_au16Data[0x0F] = 0x3040;
673 eeprom.m_au16Data[0x21] = 0x7061;
674 eeprom.m_au16Data[0x22] = 0x280C;
675 eeprom.m_au16Data[0x23] = 0x00C8;
676 eeprom.m_au16Data[0x24] = 0x00C8;
677 eeprom.m_au16Data[0x2F] = 0x0602;
678 updateChecksum();
679 };
680
681 /**
682 * Compute the checksum as required by E1000 and store it
683 * in the last word.
684 */
685 void updateChecksum()
686 {
687 uint16_t u16Checksum = 0;
688
689 for (int i = 0; i < eeprom.SIZE-1; i++)
690 u16Checksum += eeprom.m_au16Data[i];
691 eeprom.m_au16Data[eeprom.SIZE-1] = 0xBABA - u16Checksum;
692 };
693
694 /**
695 * First 6 bytes of EEPROM contain MAC address.
696 *
697 * @returns MAC address of E1000.
698 */
699 void getMac(PRTMAC pMac)
700 {
701 memcpy(pMac->au16, eeprom.m_au16Data, sizeof(pMac->au16));
702 };
703
704 uint32_t read()
705 {
706 return eeprom.read();
707 }
708
709 void write(uint32_t u32Wires)
710 {
711 eeprom.write(u32Wires);
712 }
713
714 bool readWord(uint32_t u32Addr, uint16_t *pu16Value)
715 {
716 return eeprom.readWord(u32Addr, pu16Value);
717 }
718
719 int load(PCPDMDEVHLPR3 pHlp, PSSMHANDLE pSSM)
720 {
721 return eeprom.load(pHlp, pSSM);
722 }
723
724 void save(PCPDMDEVHLPR3 pHlp, PSSMHANDLE pSSM)
725 {
726 eeprom.save(pHlp, pSSM);
727 }
728#endif /* IN_RING3 */
729};
730
731
732#define E1K_SPEC_VLAN(s) (s & 0xFFF)
733#define E1K_SPEC_CFI(s) (!!((s>>12) & 0x1))
734#define E1K_SPEC_PRI(s) ((s>>13) & 0x7)
735
736struct E1kRxDStatus
737{
738 /** @name Descriptor Status field (3.2.3.1)
739 * @{ */
740 unsigned fDD : 1; /**< Descriptor Done. */
741 unsigned fEOP : 1; /**< End of packet. */
742 unsigned fIXSM : 1; /**< Ignore checksum indication. */
743 unsigned fVP : 1; /**< VLAN, matches VET. */
744 unsigned : 1;
745 unsigned fTCPCS : 1; /**< RCP Checksum calculated on the packet. */
746 unsigned fIPCS : 1; /**< IP Checksum calculated on the packet. */
747 unsigned fPIF : 1; /**< Passed in-exact filter */
748 /** @} */
749 /** @name Descriptor Errors field (3.2.3.2)
750 * (Only valid when fEOP and fDD are set.)
751 * @{ */
752 unsigned fCE : 1; /**< CRC or alignment error. */
753 unsigned : 4; /**< Reserved, varies with different models... */
754 unsigned fTCPE : 1; /**< TCP/UDP checksum error. */
755 unsigned fIPE : 1; /**< IP Checksum error. */
756 unsigned fRXE : 1; /**< RX Data error. */
757 /** @} */
758 /** @name Descriptor Special field (3.2.3.3)
759 * @{ */
760 unsigned u16Special : 16; /**< VLAN: Id, Canonical form, Priority. */
761 /** @} */
762};
763typedef struct E1kRxDStatus E1KRXDST;
764
765struct E1kRxDesc_st
766{
767 uint64_t u64BufAddr; /**< Address of data buffer */
768 uint16_t u16Length; /**< Length of data in buffer */
769 uint16_t u16Checksum; /**< Packet checksum */
770 E1KRXDST status;
771};
772typedef struct E1kRxDesc_st E1KRXDESC;
773AssertCompileSize(E1KRXDESC, 16);
774
775#define E1K_DTYP_LEGACY -1
776#define E1K_DTYP_CONTEXT 0
777#define E1K_DTYP_DATA 1
778#define E1K_DTYP_INVALID 2
779
780struct E1kTDLegacy
781{
782 uint64_t u64BufAddr; /**< Address of data buffer */
783 struct TDLCmd_st
784 {
785 unsigned u16Length : 16;
786 unsigned u8CSO : 8;
787 /* CMD field : 8 */
788 unsigned fEOP : 1;
789 unsigned fIFCS : 1;
790 unsigned fIC : 1;
791 unsigned fRS : 1;
792 unsigned fRPS : 1;
793 unsigned fDEXT : 1;
794 unsigned fVLE : 1;
795 unsigned fIDE : 1;
796 } cmd;
797 struct TDLDw3_st
798 {
799 /* STA field */
800 unsigned fDD : 1;
801 unsigned fEC : 1;
802 unsigned fLC : 1;
803 unsigned fTURSV : 1;
804 /* RSV field */
805 unsigned u4RSV : 4;
806 /* CSS field */
807 unsigned u8CSS : 8;
808 /* Special field*/
809 unsigned u16Special: 16;
810 } dw3;
811};
812
813/**
814 * TCP/IP Context Transmit Descriptor, section 3.3.6.
815 */
816struct E1kTDContext
817{
818 struct CheckSum_st
819 {
820 /** TSE: Header start. !TSE: Checksum start. */
821 unsigned u8CSS : 8;
822 /** Checksum offset - where to store it. */
823 unsigned u8CSO : 8;
824 /** Checksum ending (inclusive) offset, 0 = end of packet. */
825 unsigned u16CSE : 16;
826 } ip;
827 struct CheckSum_st tu;
828 struct TDCDw2_st
829 {
830 /** TSE: The total number of payload bytes for this context. Sans header. */
831 unsigned u20PAYLEN : 20;
832 /** The descriptor type - E1K_DTYP_CONTEXT (0). */
833 unsigned u4DTYP : 4;
834 /** TUCMD field, 8 bits
835 * @{ */
836 /** TSE: TCP (set) or UDP (clear). */
837 unsigned fTCP : 1;
838 /** TSE: IPv4 (set) or IPv6 (clear) - for finding the payload length field in
839 * the IP header. Does not affect the checksumming.
840 * @remarks 82544GC/EI interprets a cleared field differently. */
841 unsigned fIP : 1;
842 /** TSE: TCP segmentation enable. When clear the context describes */
843 unsigned fTSE : 1;
844 /** Report status (only applies to dw3.fDD for here). */
845 unsigned fRS : 1;
846 /** Reserved, MBZ. */
847 unsigned fRSV1 : 1;
848 /** Descriptor extension, must be set for this descriptor type. */
849 unsigned fDEXT : 1;
850 /** Reserved, MBZ. */
851 unsigned fRSV2 : 1;
852 /** Interrupt delay enable. */
853 unsigned fIDE : 1;
854 /** @} */
855 } dw2;
856 struct TDCDw3_st
857 {
858 /** Descriptor Done. */
859 unsigned fDD : 1;
860 /** Reserved, MBZ. */
861 unsigned u7RSV : 7;
862 /** TSO: The header (prototype) length (Ethernet[, VLAN tag], IP, TCP/UDP. */
863 unsigned u8HDRLEN : 8;
864 /** TSO: Maximum segment size. */
865 unsigned u16MSS : 16;
866 } dw3;
867};
868typedef struct E1kTDContext E1KTXCTX;
869
870/**
871 * TCP/IP Data Transmit Descriptor, section 3.3.7.
872 */
873struct E1kTDData
874{
875 uint64_t u64BufAddr; /**< Address of data buffer */
876 struct TDDCmd_st
877 {
878 /** The total length of data pointed to by this descriptor. */
879 unsigned u20DTALEN : 20;
880 /** The descriptor type - E1K_DTYP_DATA (1). */
881 unsigned u4DTYP : 4;
882 /** @name DCMD field, 8 bits (3.3.7.1).
883 * @{ */
884 /** End of packet. Note TSCTFC update. */
885 unsigned fEOP : 1;
886 /** Insert Ethernet FCS/CRC (requires fEOP to be set). */
887 unsigned fIFCS : 1;
888 /** Use the TSE context when set and the normal when clear. */
889 unsigned fTSE : 1;
890 /** Report status (dw3.STA). */
891 unsigned fRS : 1;
892 /** Reserved. 82544GC/EI defines this report packet set (RPS). */
893 unsigned fRPS : 1;
894 /** Descriptor extension, must be set for this descriptor type. */
895 unsigned fDEXT : 1;
896 /** VLAN enable, requires CTRL.VME, auto enables FCS/CRC.
897 * Insert dw3.SPECIAL after ethernet header. */
898 unsigned fVLE : 1;
899 /** Interrupt delay enable. */
900 unsigned fIDE : 1;
901 /** @} */
902 } cmd;
903 struct TDDDw3_st
904 {
905 /** @name STA field (3.3.7.2)
906 * @{ */
907 unsigned fDD : 1; /**< Descriptor done. */
908 unsigned fEC : 1; /**< Excess collision. */
909 unsigned fLC : 1; /**< Late collision. */
910 /** Reserved, except for the usual oddball (82544GC/EI) where it's called TU. */
911 unsigned fTURSV : 1;
912 /** @} */
913 unsigned u4RSV : 4; /**< Reserved field, MBZ. */
914 /** @name POPTS (Packet Option) field (3.3.7.3)
915 * @{ */
916 unsigned fIXSM : 1; /**< Insert IP checksum. */
917 unsigned fTXSM : 1; /**< Insert TCP/UDP checksum. */
918 unsigned u6RSV : 6; /**< Reserved, MBZ. */
919 /** @} */
920 /** @name SPECIAL field - VLAN tag to be inserted after ethernet header.
921 * Requires fEOP, fVLE and CTRL.VME to be set.
922 * @{ */
923 unsigned u16Special: 16; /**< VLAN: Id, Canonical form, Priority. */
924 /** @} */
925 } dw3;
926};
927typedef struct E1kTDData E1KTXDAT;
928
929union E1kTxDesc
930{
931 struct E1kTDLegacy legacy;
932 struct E1kTDContext context;
933 struct E1kTDData data;
934};
935typedef union E1kTxDesc E1KTXDESC;
936AssertCompileSize(E1KTXDESC, 16);
937
938#define RA_CTL_AS 0x0003
939#define RA_CTL_AV 0x8000
940
941union E1kRecAddr
942{
943 uint32_t au32[32];
944 struct RAArray
945 {
946 uint8_t addr[6];
947 uint16_t ctl;
948 } array[16];
949};
950typedef struct E1kRecAddr::RAArray E1KRAELEM;
951typedef union E1kRecAddr E1KRA;
952AssertCompileSize(E1KRA, 8*16);
953
954#define E1K_IP_RF UINT16_C(0x8000) /**< reserved fragment flag */
955#define E1K_IP_DF UINT16_C(0x4000) /**< dont fragment flag */
956#define E1K_IP_MF UINT16_C(0x2000) /**< more fragments flag */
957#define E1K_IP_OFFMASK UINT16_C(0x1fff) /**< mask for fragmenting bits */
958
959/** @todo use+extend RTNETIPV4 */
960struct E1kIpHeader
961{
962 /* type of service / version / header length */
963 uint16_t tos_ver_hl;
964 /* total length */
965 uint16_t total_len;
966 /* identification */
967 uint16_t ident;
968 /* fragment offset field */
969 uint16_t offset;
970 /* time to live / protocol*/
971 uint16_t ttl_proto;
972 /* checksum */
973 uint16_t chksum;
974 /* source IP address */
975 uint32_t src;
976 /* destination IP address */
977 uint32_t dest;
978};
979AssertCompileSize(struct E1kIpHeader, 20);
980
981#define E1K_TCP_FIN UINT16_C(0x01)
982#define E1K_TCP_SYN UINT16_C(0x02)
983#define E1K_TCP_RST UINT16_C(0x04)
984#define E1K_TCP_PSH UINT16_C(0x08)
985#define E1K_TCP_ACK UINT16_C(0x10)
986#define E1K_TCP_URG UINT16_C(0x20)
987#define E1K_TCP_ECE UINT16_C(0x40)
988#define E1K_TCP_CWR UINT16_C(0x80)
989#define E1K_TCP_FLAGS UINT16_C(0x3f)
990
991/** @todo use+extend RTNETTCP */
992struct E1kTcpHeader
993{
994 uint16_t src;
995 uint16_t dest;
996 uint32_t seqno;
997 uint32_t ackno;
998 uint16_t hdrlen_flags;
999 uint16_t wnd;
1000 uint16_t chksum;
1001 uint16_t urgp;
1002};
1003AssertCompileSize(struct E1kTcpHeader, 20);
1004
1005
1006#ifdef E1K_WITH_TXD_CACHE
1007/** The current Saved state version. */
1008# define E1K_SAVEDSTATE_VERSION 4
1009/** Saved state version for VirtualBox 4.2 with VLAN tag fields. */
1010# define E1K_SAVEDSTATE_VERSION_VBOX_42_VTAG 3
1011#else /* !E1K_WITH_TXD_CACHE */
1012/** The current Saved state version. */
1013# define E1K_SAVEDSTATE_VERSION 3
1014#endif /* !E1K_WITH_TXD_CACHE */
1015/** Saved state version for VirtualBox 4.1 and earlier.
1016 * These did not include VLAN tag fields. */
1017#define E1K_SAVEDSTATE_VERSION_VBOX_41 2
1018/** Saved state version for VirtualBox 3.0 and earlier.
1019 * This did not include the configuration part nor the E1kEEPROM. */
1020#define E1K_SAVEDSTATE_VERSION_VBOX_30 1
1021
1022/**
1023 * E1000 shared device state.
1024 *
1025 * This is shared between ring-0 and ring-3.
1026 */
1027typedef struct E1KSTATE
1028{
1029 char szPrf[8]; /**< Log prefix, e.g. E1000#1. */
1030
1031 /** Handle to PCI region \#0, the MMIO region. */
1032 IOMIOPORTHANDLE hMmioRegion;
1033 /** Handle to PCI region \#2, the I/O ports. */
1034 IOMIOPORTHANDLE hIoPorts;
1035
1036 /** Receive Interrupt Delay Timer. */
1037 TMTIMERHANDLE hRIDTimer;
1038 /** Receive Absolute Delay Timer. */
1039 TMTIMERHANDLE hRADTimer;
1040 /** Transmit Interrupt Delay Timer. */
1041 TMTIMERHANDLE hTIDTimer;
1042 /** Transmit Absolute Delay Timer. */
1043 TMTIMERHANDLE hTADTimer;
1044 /** Transmit Delay Timer. */
1045 TMTIMERHANDLE hTXDTimer;
1046 /** Late Interrupt Timer. */
1047 TMTIMERHANDLE hIntTimer;
1048 /** Link Up(/Restore) Timer. */
1049 TMTIMERHANDLE hLUTimer;
1050
1051 /** Transmit task. */
1052 PDMTASKHANDLE hTxTask;
1053
1054 /** Critical section - what is it protecting? */
1055 PDMCRITSECT cs;
1056 /** RX Critical section. */
1057 PDMCRITSECT csRx;
1058#ifdef E1K_WITH_TX_CS
1059 /** TX Critical section. */
1060 PDMCRITSECT csTx;
1061#endif /* E1K_WITH_TX_CS */
1062 /** MAC address obtained from the configuration. */
1063 RTMAC macConfigured;
1064 uint16_t u16Padding0;
1065 /** EMT: Last time the interrupt was acknowledged. */
1066 uint64_t u64AckedAt;
1067 /** All: Used for eliminating spurious interrupts. */
1068 bool fIntRaised;
1069 /** EMT: false if the cable is disconnected by the GUI. */
1070 bool fCableConnected;
1071 /** true if the device is attached to a driver. */
1072 bool fIsAttached;
1073 /** EMT: Compute Ethernet CRC for RX packets. */
1074 bool fEthernetCRC;
1075 /** All: throttle interrupts. */
1076 bool fItrEnabled;
1077 /** All: throttle RX interrupts. */
1078 bool fItrRxEnabled;
1079 /** All: Delay TX interrupts using TIDV/TADV. */
1080 bool fTidEnabled;
1081 bool afPadding[2];
1082 /** Link up delay (in milliseconds). */
1083 uint32_t cMsLinkUpDelay;
1084
1085 /** All: Device register storage. */
1086 uint32_t auRegs[E1K_NUM_OF_32BIT_REGS];
1087 /** TX/RX: Status LED. */
1088 PDMLED led;
1089 /** TX/RX: Number of packet being sent/received to show in debug log. */
1090 uint32_t u32PktNo;
1091
1092 /** EMT: Offset of the register to be read via IO. */
1093 uint32_t uSelectedReg;
1094 /** EMT: Multicast Table Array. */
1095 uint32_t auMTA[128];
1096 /** EMT: Receive Address registers. */
1097 E1KRA aRecAddr;
1098 /** EMT: VLAN filter table array. */
1099 uint32_t auVFTA[128];
1100 /** EMT: Receive buffer size. */
1101 uint16_t u16RxBSize;
1102 /** EMT: Locked state -- no state alteration possible. */
1103 bool fLocked;
1104 /** EMT: */
1105 bool fDelayInts;
1106 /** All: */
1107 bool fIntMaskUsed;
1108
1109 /** N/A: */
1110 bool volatile fMaybeOutOfSpace;
1111 /** EMT: Gets signalled when more RX descriptors become available. */
1112 SUPSEMEVENT hEventMoreRxDescAvail;
1113#ifdef E1K_WITH_RXD_CACHE
1114 /** RX: Fetched RX descriptors. */
1115 E1KRXDESC aRxDescriptors[E1K_RXD_CACHE_SIZE];
1116 //uint64_t aRxDescAddr[E1K_RXD_CACHE_SIZE];
1117 /** RX: Actual number of fetched RX descriptors. */
1118 uint32_t nRxDFetched;
1119 /** RX: Index in cache of RX descriptor being processed. */
1120 uint32_t iRxDCurrent;
1121#endif /* E1K_WITH_RXD_CACHE */
1122
1123 /** TX: Context used for TCP segmentation packets. */
1124 E1KTXCTX contextTSE;
1125 /** TX: Context used for ordinary packets. */
1126 E1KTXCTX contextNormal;
1127#ifdef E1K_WITH_TXD_CACHE
1128 /** TX: Fetched TX descriptors. */
1129 E1KTXDESC aTxDescriptors[E1K_TXD_CACHE_SIZE];
1130 /** TX: Validity of TX descriptors. Set by e1kLocateTxPacket, used by e1kXmitPacket. */
1131 bool afTxDValid[E1K_TXD_CACHE_SIZE];
1132 /** TX: Actual number of fetched TX descriptors. */
1133 uint8_t nTxDFetched;
1134 /** TX: Index in cache of TX descriptor being processed. */
1135 uint8_t iTxDCurrent;
1136 /** TX: Will this frame be sent as GSO. */
1137 bool fGSO;
1138 /** Alignment padding. */
1139 bool fReserved;
1140 /** TX: Number of bytes in next packet. */
1141 uint32_t cbTxAlloc;
1142
1143#endif /* E1K_WITH_TXD_CACHE */
1144 /** GSO context. u8Type is set to PDMNETWORKGSOTYPE_INVALID when not
1145 * applicable to the current TSE mode. */
1146 PDMNETWORKGSO GsoCtx;
1147 /** Scratch space for holding the loopback / fallback scatter / gather
1148 * descriptor. */
1149 union
1150 {
1151 PDMSCATTERGATHER Sg;
1152 uint8_t padding[8 * sizeof(RTUINTPTR)];
1153 } uTxFallback;
1154 /** TX: Transmit packet buffer use for TSE fallback and loopback. */
1155 uint8_t aTxPacketFallback[E1K_MAX_TX_PKT_SIZE];
1156 /** TX: Number of bytes assembled in TX packet buffer. */
1157 uint16_t u16TxPktLen;
1158 /** TX: False will force segmentation in e1000 instead of sending frames as GSO. */
1159 bool fGSOEnabled;
1160 /** TX: IP checksum has to be inserted if true. */
1161 bool fIPcsum;
1162 /** TX: TCP/UDP checksum has to be inserted if true. */
1163 bool fTCPcsum;
1164 /** TX: VLAN tag has to be inserted if true. */
1165 bool fVTag;
1166 /** TX: TCI part of VLAN tag to be inserted. */
1167 uint16_t u16VTagTCI;
1168 /** TX TSE fallback: Number of payload bytes remaining in TSE context. */
1169 uint32_t u32PayRemain;
1170 /** TX TSE fallback: Number of header bytes remaining in TSE context. */
1171 uint16_t u16HdrRemain;
1172 /** TX TSE fallback: Flags from template header. */
1173 uint16_t u16SavedFlags;
1174 /** TX TSE fallback: Partial checksum from template header. */
1175 uint32_t u32SavedCsum;
1176 /** ?: Emulated controller type. */
1177 E1KCHIP eChip;
1178
1179 /** EMT: Physical interface emulation. */
1180 PHY phy;
1181
1182#if 0
1183 /** Alignment padding. */
1184 uint8_t Alignment[HC_ARCH_BITS == 64 ? 8 : 4];
1185#endif
1186
1187 STAMCOUNTER StatReceiveBytes;
1188 STAMCOUNTER StatTransmitBytes;
1189#if defined(VBOX_WITH_STATISTICS)
1190 STAMPROFILEADV StatMMIOReadRZ;
1191 STAMPROFILEADV StatMMIOReadR3;
1192 STAMPROFILEADV StatMMIOWriteRZ;
1193 STAMPROFILEADV StatMMIOWriteR3;
1194 STAMPROFILEADV StatEEPROMRead;
1195 STAMPROFILEADV StatEEPROMWrite;
1196 STAMPROFILEADV StatIOReadRZ;
1197 STAMPROFILEADV StatIOReadR3;
1198 STAMPROFILEADV StatIOWriteRZ;
1199 STAMPROFILEADV StatIOWriteR3;
1200 STAMPROFILEADV StatLateIntTimer;
1201 STAMCOUNTER StatLateInts;
1202 STAMCOUNTER StatIntsRaised;
1203 STAMCOUNTER StatIntsPrevented;
1204 STAMPROFILEADV StatReceive;
1205 STAMPROFILEADV StatReceiveCRC;
1206 STAMPROFILEADV StatReceiveFilter;
1207 STAMPROFILEADV StatReceiveStore;
1208 STAMPROFILEADV StatTransmitRZ;
1209 STAMPROFILEADV StatTransmitR3;
1210 STAMPROFILE StatTransmitSendRZ;
1211 STAMPROFILE StatTransmitSendR3;
1212 STAMPROFILE StatRxOverflow;
1213 STAMCOUNTER StatRxOverflowWakeupRZ;
1214 STAMCOUNTER StatRxOverflowWakeupR3;
1215 STAMCOUNTER StatTxDescCtxNormal;
1216 STAMCOUNTER StatTxDescCtxTSE;
1217 STAMCOUNTER StatTxDescLegacy;
1218 STAMCOUNTER StatTxDescData;
1219 STAMCOUNTER StatTxDescTSEData;
1220 STAMCOUNTER StatTxPathFallback;
1221 STAMCOUNTER StatTxPathGSO;
1222 STAMCOUNTER StatTxPathRegular;
1223 STAMCOUNTER StatPHYAccesses;
1224 STAMCOUNTER aStatRegWrites[E1K_NUM_OF_REGS];
1225 STAMCOUNTER aStatRegReads[E1K_NUM_OF_REGS];
1226#endif /* VBOX_WITH_STATISTICS */
1227
1228#ifdef E1K_INT_STATS
1229 /* Internal stats */
1230 uint64_t u64ArmedAt;
1231 uint64_t uStatMaxTxDelay;
1232 uint32_t uStatInt;
1233 uint32_t uStatIntTry;
1234 uint32_t uStatIntLower;
1235 uint32_t uStatNoIntICR;
1236 int32_t iStatIntLost;
1237 int32_t iStatIntLostOne;
1238 uint32_t uStatIntIMS;
1239 uint32_t uStatIntSkip;
1240 uint32_t uStatIntLate;
1241 uint32_t uStatIntMasked;
1242 uint32_t uStatIntEarly;
1243 uint32_t uStatIntRx;
1244 uint32_t uStatIntTx;
1245 uint32_t uStatIntICS;
1246 uint32_t uStatIntRDTR;
1247 uint32_t uStatIntRXDMT0;
1248 uint32_t uStatIntTXQE;
1249 uint32_t uStatTxNoRS;
1250 uint32_t uStatTxIDE;
1251 uint32_t uStatTxDelayed;
1252 uint32_t uStatTxDelayExp;
1253 uint32_t uStatTAD;
1254 uint32_t uStatTID;
1255 uint32_t uStatRAD;
1256 uint32_t uStatRID;
1257 uint32_t uStatRxFrm;
1258 uint32_t uStatTxFrm;
1259 uint32_t uStatDescCtx;
1260 uint32_t uStatDescDat;
1261 uint32_t uStatDescLeg;
1262 uint32_t uStatTx1514;
1263 uint32_t uStatTx2962;
1264 uint32_t uStatTx4410;
1265 uint32_t uStatTx5858;
1266 uint32_t uStatTx7306;
1267 uint32_t uStatTx8754;
1268 uint32_t uStatTx16384;
1269 uint32_t uStatTx32768;
1270 uint32_t uStatTxLarge;
1271 uint32_t uStatAlign;
1272#endif /* E1K_INT_STATS */
1273} E1KSTATE;
1274/** Pointer to the E1000 device state. */
1275typedef E1KSTATE *PE1KSTATE;
1276
1277/**
1278 * E1000 ring-3 device state
1279 *
1280 * @implements PDMINETWORKDOWN
1281 * @implements PDMINETWORKCONFIG
1282 * @implements PDMILEDPORTS
1283 */
1284typedef struct E1KSTATER3
1285{
1286 PDMIBASE IBase;
1287 PDMINETWORKDOWN INetworkDown;
1288 PDMINETWORKCONFIG INetworkConfig;
1289 /** LED interface */
1290 PDMILEDPORTS ILeds;
1291 /** Attached network driver. */
1292 R3PTRTYPE(PPDMIBASE) pDrvBase;
1293 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
1294
1295 /** Pointer to the shared state. */
1296 R3PTRTYPE(PE1KSTATE) pShared;
1297
1298 /** Device instance. */
1299 PPDMDEVINSR3 pDevInsR3;
1300 /** Attached network driver. */
1301 PPDMINETWORKUPR3 pDrvR3;
1302 /** The scatter / gather buffer used for the current outgoing packet. */
1303 R3PTRTYPE(PPDMSCATTERGATHER) pTxSgR3;
1304
1305 /** EMT: EEPROM emulation */
1306 E1kEEPROM eeprom;
1307} E1KSTATER3;
1308/** Pointer to the E1000 ring-3 device state. */
1309typedef E1KSTATER3 *PE1KSTATER3;
1310
1311
1312/**
1313 * E1000 ring-0 device state
1314 */
1315typedef struct E1KSTATER0
1316{
1317 /** Device instance. */
1318 PPDMDEVINSR0 pDevInsR0;
1319 /** Attached network driver. */
1320 PPDMINETWORKUPR0 pDrvR0;
1321 /** The scatter / gather buffer used for the current outgoing packet - R0. */
1322 R0PTRTYPE(PPDMSCATTERGATHER) pTxSgR0;
1323} E1KSTATER0;
1324/** Pointer to the E1000 ring-0 device state. */
1325typedef E1KSTATER0 *PE1KSTATER0;
1326
1327
1328/**
1329 * E1000 raw-mode device state
1330 */
1331typedef struct E1KSTATERC
1332{
1333 /** Device instance. */
1334 PPDMDEVINSRC pDevInsRC;
1335 /** Attached network driver. */
1336 PPDMINETWORKUPRC pDrvRC;
1337 /** The scatter / gather buffer used for the current outgoing packet. */
1338 RCPTRTYPE(PPDMSCATTERGATHER) pTxSgRC;
1339} E1KSTATERC;
1340/** Pointer to the E1000 raw-mode device state. */
1341typedef E1KSTATERC *PE1KSTATERC;
1342
1343
1344/** @def PE1KSTATECC
1345 * Pointer to the instance data for the current context. */
1346#ifdef IN_RING3
1347typedef E1KSTATER3 E1KSTATECC;
1348typedef PE1KSTATER3 PE1KSTATECC;
1349#elif defined(IN_RING0)
1350typedef E1KSTATER0 E1KSTATECC;
1351typedef PE1KSTATER0 PE1KSTATECC;
1352#elif defined(IN_RC)
1353typedef E1KSTATERC E1KSTATECC;
1354typedef PE1KSTATERC PE1KSTATECC;
1355#else
1356# error "Not IN_RING3, IN_RING0 or IN_RC"
1357#endif
1358
1359
1360#ifndef VBOX_DEVICE_STRUCT_TESTCASE
1361
1362/* Forward declarations ******************************************************/
1363static int e1kXmitPending(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread);
1364
1365/**
1366 * E1000 register read handler.
1367 */
1368typedef int (FNE1KREGREAD)(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value);
1369/**
1370 * E1000 register write handler.
1371 */
1372typedef int (FNE1KREGWRITE)(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t u32Value);
1373
1374static FNE1KREGREAD e1kRegReadUnimplemented;
1375static FNE1KREGWRITE e1kRegWriteUnimplemented;
1376static FNE1KREGREAD e1kRegReadAutoClear;
1377static FNE1KREGREAD e1kRegReadDefault;
1378static FNE1KREGWRITE e1kRegWriteDefault;
1379#if 0 /* unused */
1380static FNE1KREGREAD e1kRegReadCTRL;
1381#endif
1382static FNE1KREGWRITE e1kRegWriteCTRL;
1383static FNE1KREGREAD e1kRegReadEECD;
1384static FNE1KREGWRITE e1kRegWriteEECD;
1385static FNE1KREGWRITE e1kRegWriteEERD;
1386static FNE1KREGWRITE e1kRegWriteMDIC;
1387static FNE1KREGREAD e1kRegReadICR;
1388static FNE1KREGWRITE e1kRegWriteICR;
1389static FNE1KREGREAD e1kRegReadICS;
1390static FNE1KREGWRITE e1kRegWriteICS;
1391static FNE1KREGWRITE e1kRegWriteIMS;
1392static FNE1KREGWRITE e1kRegWriteIMC;
1393static FNE1KREGWRITE e1kRegWriteRCTL;
1394static FNE1KREGWRITE e1kRegWritePBA;
1395static FNE1KREGWRITE e1kRegWriteRDT;
1396static FNE1KREGWRITE e1kRegWriteRDTR;
1397static FNE1KREGWRITE e1kRegWriteTDT;
1398static FNE1KREGREAD e1kRegReadMTA;
1399static FNE1KREGWRITE e1kRegWriteMTA;
1400static FNE1KREGREAD e1kRegReadRA;
1401static FNE1KREGWRITE e1kRegWriteRA;
1402static FNE1KREGREAD e1kRegReadVFTA;
1403static FNE1KREGWRITE e1kRegWriteVFTA;
1404
1405/**
1406 * Register map table.
1407 *
1408 * Override pfnRead and pfnWrite to get register-specific behavior.
1409 */
1410static const struct E1kRegMap_st
1411{
1412 /** Register offset in the register space. */
1413 uint32_t offset;
1414 /** Size in bytes. Registers of size > 4 are in fact tables. */
1415 uint32_t size;
1416 /** Readable bits. */
1417 uint32_t readable;
1418 /** Writable bits. */
1419 uint32_t writable;
1420 /** Read callback. */
1421 FNE1KREGREAD *pfnRead;
1422 /** Write callback. */
1423 FNE1KREGWRITE *pfnWrite;
1424 /** Abbreviated name. */
1425 const char *abbrev;
1426 /** Full name. */
1427 const char *name;
1428} g_aE1kRegMap[E1K_NUM_OF_REGS] =
1429{
1430 /* offset size read mask write mask read callback write callback abbrev full name */
1431 /*------- ------- ---------- ---------- ----------------------- ------------------------ ---------- ------------------------------*/
1432 { 0x00000, 0x00004, 0xDBF31BE9, 0xDBF31BE9, e1kRegReadDefault , e1kRegWriteCTRL , "CTRL" , "Device Control" },
1433 { 0x00008, 0x00004, 0x0000FDFF, 0x00000000, e1kRegReadDefault , e1kRegWriteUnimplemented, "STATUS" , "Device Status" },
1434 { 0x00010, 0x00004, 0x000027F0, 0x00000070, e1kRegReadEECD , e1kRegWriteEECD , "EECD" , "EEPROM/Flash Control/Data" },
1435 { 0x00014, 0x00004, 0xFFFFFF10, 0xFFFFFF00, e1kRegReadDefault , e1kRegWriteEERD , "EERD" , "EEPROM Read" },
1436 { 0x00018, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "CTRL_EXT", "Extended Device Control" },
1437 { 0x0001c, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FLA" , "Flash Access (N/A)" },
1438 { 0x00020, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteMDIC , "MDIC" , "MDI Control" },
1439 { 0x00028, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCAL" , "Flow Control Address Low" },
1440 { 0x0002c, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCAH" , "Flow Control Address High" },
1441 { 0x00030, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCT" , "Flow Control Type" },
1442 { 0x00038, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "VET" , "VLAN EtherType" },
1443 { 0x000c0, 0x00004, 0x0001F6DF, 0x0001F6DF, e1kRegReadICR , e1kRegWriteICR , "ICR" , "Interrupt Cause Read" },
1444 { 0x000c4, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "ITR" , "Interrupt Throttling" },
1445 { 0x000c8, 0x00004, 0x0001F6DF, 0xFFFFFFFF, e1kRegReadICS , e1kRegWriteICS , "ICS" , "Interrupt Cause Set" },
1446 { 0x000d0, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteIMS , "IMS" , "Interrupt Mask Set/Read" },
1447 { 0x000d8, 0x00004, 0x00000000, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteIMC , "IMC" , "Interrupt Mask Clear" },
1448 { 0x00100, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteRCTL , "RCTL" , "Receive Control" },
1449 { 0x00170, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCTTV" , "Flow Control Transmit Timer Value" },
1450 { 0x00178, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TXCW" , "Transmit Configuration Word (N/A)" },
1451 { 0x00180, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RXCW" , "Receive Configuration Word (N/A)" },
1452 { 0x00400, 0x00004, 0x017FFFFA, 0x017FFFFA, e1kRegReadDefault , e1kRegWriteDefault , "TCTL" , "Transmit Control" },
1453 { 0x00410, 0x00004, 0x3FFFFFFF, 0x3FFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TIPG" , "Transmit IPG" },
1454 { 0x00458, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "AIFS" , "Adaptive IFS Throttle - AIT" },
1455 { 0x00e00, 0x00004, 0xCFCFCFCF, 0xCFCFCFCF, e1kRegReadDefault , e1kRegWriteDefault , "LEDCTL" , "LED Control" },
1456 { 0x01000, 0x00004, 0xFFFF007F, 0x0000007F, e1kRegReadDefault , e1kRegWritePBA , "PBA" , "Packet Buffer Allocation" },
1457 { 0x02160, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCRTL" , "Flow Control Receive Threshold Low" },
1458 { 0x02168, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCRTH" , "Flow Control Receive Threshold High" },
1459 { 0x02410, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFH" , "Receive Data FIFO Head" },
1460 { 0x02418, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFT" , "Receive Data FIFO Tail" },
1461 { 0x02420, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFHS" , "Receive Data FIFO Head Saved Register" },
1462 { 0x02428, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFTS" , "Receive Data FIFO Tail Saved Register" },
1463 { 0x02430, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RDFPC" , "Receive Data FIFO Packet Count" },
1464 { 0x02800, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "RDBAL" , "Receive Descriptor Base Low" },
1465 { 0x02804, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "RDBAH" , "Receive Descriptor Base High" },
1466 { 0x02808, 0x00004, 0x000FFF80, 0x000FFF80, e1kRegReadDefault , e1kRegWriteDefault , "RDLEN" , "Receive Descriptor Length" },
1467 { 0x02810, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "RDH" , "Receive Descriptor Head" },
1468 { 0x02818, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteRDT , "RDT" , "Receive Descriptor Tail" },
1469 { 0x02820, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteRDTR , "RDTR" , "Receive Delay Timer" },
1470 { 0x02828, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RXDCTL" , "Receive Descriptor Control" },
1471 { 0x0282c, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "RADV" , "Receive Interrupt Absolute Delay Timer" },
1472 { 0x02c00, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RSRPD" , "Receive Small Packet Detect Interrupt" },
1473 { 0x03000, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TXDMAC" , "TX DMA Control (N/A)" },
1474 { 0x03410, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFH" , "Transmit Data FIFO Head" },
1475 { 0x03418, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFT" , "Transmit Data FIFO Tail" },
1476 { 0x03420, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFHS" , "Transmit Data FIFO Head Saved Register" },
1477 { 0x03428, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFTS" , "Transmit Data FIFO Tail Saved Register" },
1478 { 0x03430, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TDFPC" , "Transmit Data FIFO Packet Count" },
1479 { 0x03800, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TDBAL" , "Transmit Descriptor Base Low" },
1480 { 0x03804, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TDBAH" , "Transmit Descriptor Base High" },
1481 { 0x03808, 0x00004, 0x000FFF80, 0x000FFF80, e1kRegReadDefault , e1kRegWriteDefault , "TDLEN" , "Transmit Descriptor Length" },
1482 { 0x03810, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "TDH" , "Transmit Descriptor Head" },
1483 { 0x03818, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteTDT , "TDT" , "Transmit Descriptor Tail" },
1484 { 0x03820, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "TIDV" , "Transmit Interrupt Delay Value" },
1485 { 0x03828, 0x00004, 0xFF3F3F3F, 0xFF3F3F3F, e1kRegReadDefault , e1kRegWriteDefault , "TXDCTL" , "Transmit Descriptor Control" },
1486 { 0x0382c, 0x00004, 0x0000FFFF, 0x0000FFFF, e1kRegReadDefault , e1kRegWriteDefault , "TADV" , "Transmit Absolute Interrupt Delay Timer" },
1487 { 0x03830, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "TSPMT" , "TCP Segmentation Pad and Threshold" },
1488 { 0x04000, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "CRCERRS" , "CRC Error Count" },
1489 { 0x04004, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "ALGNERRC", "Alignment Error Count" },
1490 { 0x04008, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "SYMERRS" , "Symbol Error Count" },
1491 { 0x0400c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RXERRC" , "RX Error Count" },
1492 { 0x04010, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MPC" , "Missed Packets Count" },
1493 { 0x04014, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "SCC" , "Single Collision Count" },
1494 { 0x04018, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "ECOL" , "Excessive Collisions Count" },
1495 { 0x0401c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MCC" , "Multiple Collision Count" },
1496 { 0x04020, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "LATECOL" , "Late Collisions Count" },
1497 { 0x04028, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "COLC" , "Collision Count" },
1498 { 0x04030, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "DC" , "Defer Count" },
1499 { 0x04034, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "TNCRS" , "Transmit - No CRS" },
1500 { 0x04038, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "SEC" , "Sequence Error Count" },
1501 { 0x0403c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "CEXTERR" , "Carrier Extension Error Count" },
1502 { 0x04040, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RLEC" , "Receive Length Error Count" },
1503 { 0x04048, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XONRXC" , "XON Received Count" },
1504 { 0x0404c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XONTXC" , "XON Transmitted Count" },
1505 { 0x04050, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XOFFRXC" , "XOFF Received Count" },
1506 { 0x04054, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "XOFFTXC" , "XOFF Transmitted Count" },
1507 { 0x04058, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FCRUC" , "FC Received Unsupported Count" },
1508 { 0x0405c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC64" , "Packets Received (64 Bytes) Count" },
1509 { 0x04060, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC127" , "Packets Received (65-127 Bytes) Count" },
1510 { 0x04064, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC255" , "Packets Received (128-255 Bytes) Count" },
1511 { 0x04068, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC511" , "Packets Received (256-511 Bytes) Count" },
1512 { 0x0406c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC1023" , "Packets Received (512-1023 Bytes) Count" },
1513 { 0x04070, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PRC1522" , "Packets Received (1024-Max Bytes)" },
1514 { 0x04074, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GPRC" , "Good Packets Received Count" },
1515 { 0x04078, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "BPRC" , "Broadcast Packets Received Count" },
1516 { 0x0407c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "MPRC" , "Multicast Packets Received Count" },
1517 { 0x04080, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GPTC" , "Good Packets Transmitted Count" },
1518 { 0x04088, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GORCL" , "Good Octets Received Count (Low)" },
1519 { 0x0408c, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GORCH" , "Good Octets Received Count (Hi)" },
1520 { 0x04090, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GOTCL" , "Good Octets Transmitted Count (Low)" },
1521 { 0x04094, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "GOTCH" , "Good Octets Transmitted Count (Hi)" },
1522 { 0x040a0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RNBC" , "Receive No Buffers Count" },
1523 { 0x040a4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RUC" , "Receive Undersize Count" },
1524 { 0x040a8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RFC" , "Receive Fragment Count" },
1525 { 0x040ac, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "ROC" , "Receive Oversize Count" },
1526 { 0x040b0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "RJC" , "Receive Jabber Count" },
1527 { 0x040b4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MGTPRC" , "Management Packets Received Count" },
1528 { 0x040b8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MGTPDC" , "Management Packets Dropped Count" },
1529 { 0x040bc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "MGTPTC" , "Management Pkts Transmitted Count" },
1530 { 0x040c0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TORL" , "Total Octets Received (Lo)" },
1531 { 0x040c4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TORH" , "Total Octets Received (Hi)" },
1532 { 0x040c8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TOTL" , "Total Octets Transmitted (Lo)" },
1533 { 0x040cc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TOTH" , "Total Octets Transmitted (Hi)" },
1534 { 0x040d0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TPR" , "Total Packets Received" },
1535 { 0x040d4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TPT" , "Total Packets Transmitted" },
1536 { 0x040d8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC64" , "Packets Transmitted (64 Bytes) Count" },
1537 { 0x040dc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC127" , "Packets Transmitted (65-127 Bytes) Count" },
1538 { 0x040e0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC255" , "Packets Transmitted (128-255 Bytes) Count" },
1539 { 0x040e4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC511" , "Packets Transmitted (256-511 Bytes) Count" },
1540 { 0x040e8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC1023" , "Packets Transmitted (512-1023 Bytes) Count" },
1541 { 0x040ec, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "PTC1522" , "Packets Transmitted (1024 Bytes or Greater) Count" },
1542 { 0x040f0, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "MPTC" , "Multicast Packets Transmitted Count" },
1543 { 0x040f4, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "BPTC" , "Broadcast Packets Transmitted Count" },
1544 { 0x040f8, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TSCTC" , "TCP Segmentation Context Transmitted Count" },
1545 { 0x040fc, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadAutoClear , e1kRegWriteUnimplemented, "TSCTFC" , "TCP Segmentation Context Tx Fail Count" },
1546 { 0x05000, 0x00004, 0x000007FF, 0x000007FF, e1kRegReadDefault , e1kRegWriteDefault , "RXCSUM" , "Receive Checksum Control" },
1547 { 0x05800, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUC" , "Wakeup Control" },
1548 { 0x05808, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUFC" , "Wakeup Filter Control" },
1549 { 0x05810, 0x00004, 0xFFFFFFFF, 0x00000000, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUS" , "Wakeup Status" },
1550 { 0x05820, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadDefault , e1kRegWriteDefault , "MANC" , "Management Control" },
1551 { 0x05838, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "IPAV" , "IP Address Valid" },
1552 { 0x05900, 0x00004, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUPL" , "Wakeup Packet Length" },
1553 { 0x05200, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadMTA , e1kRegWriteMTA , "MTA" , "Multicast Table Array (n)" },
1554 { 0x05400, 0x00080, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadRA , e1kRegWriteRA , "RA" , "Receive Address (64-bit) (n)" },
1555 { 0x05600, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadVFTA , e1kRegWriteVFTA , "VFTA" , "VLAN Filter Table Array (n)" },
1556 { 0x05840, 0x0001c, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "IP4AT" , "IPv4 Address Table" },
1557 { 0x05880, 0x00010, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "IP6AT" , "IPv6 Address Table" },
1558 { 0x05a00, 0x00080, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "WUPM" , "Wakeup Packet Memory" },
1559 { 0x05f00, 0x0001c, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FFLT" , "Flexible Filter Length Table" },
1560 { 0x09000, 0x003fc, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FFMT" , "Flexible Filter Mask Table" },
1561 { 0x09800, 0x003fc, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "FFVT" , "Flexible Filter Value Table" },
1562 { 0x10000, 0x10000, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadUnimplemented, e1kRegWriteUnimplemented, "PBM" , "Packet Buffer Memory (n)" },
1563 { 0x00040, 0x00080, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadRA , e1kRegWriteRA , "RA82542" , "Receive Address (64-bit) (n) (82542)" },
1564 { 0x00200, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadMTA , e1kRegWriteMTA , "MTA82542", "Multicast Table Array (n) (82542)" },
1565 { 0x00600, 0x00200, 0xFFFFFFFF, 0xFFFFFFFF, e1kRegReadVFTA , e1kRegWriteVFTA , "VFTA82542", "VLAN Filter Table Array (n) (82542)" }
1566};
1567
1568#ifdef LOG_ENABLED
1569
1570/**
1571 * Convert U32 value to hex string. Masked bytes are replaced with dots.
1572 *
1573 * @remarks The mask has half-byte byte (not bit) granularity (e.g. 0000000F).
1574 *
1575 * @returns The buffer.
1576 *
1577 * @param u32 The word to convert into string.
1578 * @param mask Selects which bytes to convert.
1579 * @param buf Where to put the result.
1580 */
1581static char *e1kU32toHex(uint32_t u32, uint32_t mask, char *buf)
1582{
1583 for (char *ptr = buf + 7; ptr >= buf; --ptr, u32 >>=4, mask >>=4)
1584 {
1585 if (mask & 0xF)
1586 *ptr = (u32 & 0xF) + ((u32 & 0xF) > 9 ? '7' : '0');
1587 else
1588 *ptr = '.';
1589 }
1590 buf[8] = 0;
1591 return buf;
1592}
1593
1594/**
1595 * Returns timer name for debug purposes.
1596 *
1597 * @returns The timer name.
1598 *
1599 * @param pThis The device state structure.
1600 * @param hTimer The timer to name.
1601 */
1602DECLINLINE(const char *) e1kGetTimerName(PE1KSTATE pThis, TMTIMERHANDLE hTimer)
1603{
1604 if (hTimer == pThis->hTIDTimer)
1605 return "TID";
1606 if (hTimer == pThis->hTADTimer)
1607 return "TAD";
1608 if (hTimer == pThis->hRIDTimer)
1609 return "RID";
1610 if (hTimer == pThis->hRADTimer)
1611 return "RAD";
1612 if (hTimer == pThis->hIntTimer)
1613 return "Int";
1614 if (hTimer == pThis->hTXDTimer)
1615 return "TXD";
1616 if (hTimer == pThis->hLUTimer)
1617 return "LinkUp";
1618 return "unknown";
1619}
1620
1621#endif /* LOG_ENABLED */
1622
1623/**
1624 * Arm a timer.
1625 *
1626 * @param pDevIns The device instance.
1627 * @param pThis Pointer to the device state structure.
1628 * @param hTimer The timer to arm.
1629 * @param uExpireIn Expiration interval in microseconds.
1630 */
1631DECLINLINE(void) e1kArmTimer(PPDMDEVINS pDevIns, PE1KSTATE pThis, TMTIMERHANDLE hTimer, uint32_t uExpireIn)
1632{
1633 if (pThis->fLocked)
1634 return;
1635
1636 E1kLog2(("%s Arming %s timer to fire in %d usec...\n",
1637 pThis->szPrf, e1kGetTimerName(pThis, hTimer), uExpireIn));
1638 int rc = PDMDevHlpTimerSetMicro(pDevIns, hTimer, uExpireIn);
1639 AssertRC(rc);
1640}
1641
1642#ifdef IN_RING3
1643/**
1644 * Cancel a timer.
1645 *
1646 * @param pDevIns The device instance.
1647 * @param pThis Pointer to the device state structure.
1648 * @param pTimer Pointer to the timer.
1649 */
1650DECLINLINE(void) e1kCancelTimer(PPDMDEVINS pDevIns, PE1KSTATE pThis, TMTIMERHANDLE hTimer)
1651{
1652 E1kLog2(("%s Stopping %s timer...\n",
1653 pThis->szPrf, e1kGetTimerName(pThis, hTimer)));
1654 int rc = PDMDevHlpTimerStop(pDevIns, hTimer);
1655 if (RT_FAILURE(rc))
1656 E1kLog2(("%s e1kCancelTimer: TMTimerStop(%s) failed with %Rrc\n",
1657 pThis->szPrf, e1kGetTimerName(pThis, hTimer), rc));
1658 RT_NOREF_PV(pThis);
1659}
1660#endif /* IN_RING3 */
1661
1662
1663#define e1kCsEnter(ps, rcBusy) PDMDevHlpCritSectEnter(pDevIns, &(ps)->cs, (rcBusy))
1664#define e1kCsEnterReturn(ps, rcBusy) do { \
1665 int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->cs, (rcBusy)); \
1666 if (rcLock == VINF_SUCCESS) { /* likely */ } \
1667 else return rcLock; \
1668 } while (0)
1669#define e1kR3CsEnterAsserted(ps) do { \
1670 int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->cs, VERR_SEM_BUSY); \
1671 PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &(ps)->cs, rcLock); \
1672 } while (0)
1673#define e1kCsLeave(ps) PDMDevHlpCritSectLeave(pDevIns, &(ps)->cs)
1674
1675
1676#define e1kCsRxEnter(ps, rcBusy) PDMDevHlpCritSectEnter(pDevIns, &(ps)->csRx, (rcBusy))
1677#define e1kCsRxEnterReturn(ps) do { \
1678 int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->csRx, VERR_SEM_BUSY); \
1679 AssertRCReturn(rcLock, rcLock); \
1680 } while (0)
1681#define e1kR3CsRxEnterAsserted(ps) do { \
1682 int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->csRx, VERR_SEM_BUSY); \
1683 PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &(ps)->csRx, rcLock); \
1684 } while (0)
1685#define e1kCsRxLeave(ps) PDMDevHlpCritSectLeave(pDevIns, &(ps)->csRx)
1686#define e1kCsRxIsOwner(ps) PDMDevHlpCritSectIsOwner(pDevIns, &(ps)->csRx)
1687
1688
1689#ifndef E1K_WITH_TX_CS
1690# define e1kCsTxEnter(ps, rcBusy) VINF_SUCCESS
1691# define e1kR3CsTxEnterAsserted(ps) do { } while (0)
1692# define e1kCsTxLeave(ps) do { } while (0)
1693#else /* E1K_WITH_TX_CS */
1694# define e1kCsTxEnter(ps, rcBusy) PDMDevHlpCritSectEnter(pDevIns, &(ps)->csTx, (rcBusy))
1695# define e1kR3CsTxEnterAsserted(ps) do { \
1696 int const rcLock = PDMDevHlpCritSectEnter(pDevIns, &(ps)->csTx, VERR_SEM_BUSY); \
1697 PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &(ps)->csTx, rcLock); \
1698 } while (0)
1699# define e1kCsTxLeave(ps) PDMDevHlpCritSectLeave(pDevIns, &(ps)->csTx)
1700# define e1kCsTxIsOwner(ps) PDMDevHlpCritSectIsOwner(pDevIns, &(ps)->csTx)
1701#endif /* E1K_WITH_TX_CS */
1702
1703
1704#ifdef E1K_WITH_TXD_CACHE
1705/*
1706 * Transmit Descriptor Register Context
1707 */
1708struct E1kTxDContext
1709{
1710 uint32_t tdlen;
1711 uint32_t tdh;
1712 uint32_t tdt;
1713 uint8_t nextPacket;
1714};
1715typedef struct E1kTxDContext E1KTXDC, *PE1KTXDC;
1716
1717DECLINLINE(bool) e1kUpdateTxDContext(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pContext)
1718{
1719 Assert(e1kCsTxIsOwner(pThis));
1720 if (!e1kCsTxIsOwner(pThis))
1721 {
1722 memset(pContext, 0, sizeof(E1KTXDC));
1723 return false;
1724 }
1725 pContext->tdlen = TDLEN;
1726 pContext->tdh = TDH;
1727 pContext->tdt = TDT;
1728 uint32_t cTxRingSize = pContext->tdlen / sizeof(E1KTXDESC);
1729#ifdef DEBUG
1730 if (pContext->tdh >= cTxRingSize)
1731 {
1732 Log(("%s e1kUpdateTxDContext: will return false because TDH too big (%u >= %u)\n",
1733 pThis->szPrf, pContext->tdh, cTxRingSize));
1734 return VINF_SUCCESS;
1735 }
1736 if (pContext->tdt >= cTxRingSize)
1737 {
1738 Log(("%s e1kUpdateTxDContext: will return false because TDT too big (%u >= %u)\n",
1739 pThis->szPrf, pContext->tdt, cTxRingSize));
1740 return VINF_SUCCESS;
1741 }
1742#endif /* DEBUG */
1743 return pContext->tdh < cTxRingSize && pContext->tdt < cTxRingSize;
1744}
1745#endif /* E1K_WITH_TXD_CACHE */
1746#ifdef E1K_WITH_RXD_CACHE
1747/*
1748 * Receive Descriptor Register Context
1749 */
1750struct E1kRxDContext
1751{
1752 uint32_t rdlen;
1753 uint32_t rdh;
1754 uint32_t rdt;
1755};
1756typedef struct E1kRxDContext E1KRXDC, *PE1KRXDC;
1757
1758DECLINLINE(bool) e1kUpdateRxDContext(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pContext, const char *pcszCallee)
1759{
1760 Assert(e1kCsRxIsOwner(pThis));
1761 if (!e1kCsRxIsOwner(pThis))
1762 return false;
1763 pContext->rdlen = RDLEN;
1764 pContext->rdh = RDH;
1765 pContext->rdt = RDT;
1766 uint32_t cRxRingSize = pContext->rdlen / sizeof(E1KRXDESC);
1767 /*
1768 * Note that the checks for RDT are a bit different. Some guests, OS/2 for
1769 * example, intend to use all descriptors in RX ring, so they point RDT
1770 * right beyond the last descriptor in the ring. While this is not
1771 * acceptable for other registers, it works out fine for RDT.
1772 */
1773#ifdef DEBUG
1774 if (pContext->rdh >= cRxRingSize)
1775 {
1776 Log(("%s e1kUpdateRxDContext: called from %s, will return false because RDH too big (%u >= %u)\n",
1777 pThis->szPrf, pcszCallee, pContext->rdh, cRxRingSize));
1778 return VINF_SUCCESS;
1779 }
1780 if (pContext->rdt > cRxRingSize)
1781 {
1782 Log(("%s e1kUpdateRxDContext: called from %s, will return false because RDT too big (%u > %u)\n",
1783 pThis->szPrf, pcszCallee, pContext->rdt, cRxRingSize));
1784 return VINF_SUCCESS;
1785 }
1786#else /* !DEBUG */
1787 RT_NOREF(pcszCallee);
1788#endif /* !DEBUG */
1789 return pContext->rdh < cRxRingSize && pContext->rdt <= cRxRingSize; // && (RCTL & RCTL_EN);
1790}
1791#endif /* E1K_WITH_RXD_CACHE */
1792
1793/**
1794 * Wakeup the RX thread.
1795 */
1796static void e1kWakeupReceive(PPDMDEVINS pDevIns, PE1KSTATE pThis)
1797{
1798 if ( pThis->fMaybeOutOfSpace
1799 && pThis->hEventMoreRxDescAvail != NIL_SUPSEMEVENT)
1800 {
1801 STAM_COUNTER_INC(&pThis->CTX_SUFF_Z(StatRxOverflowWakeup));
1802 E1kLog(("%s Waking up Out-of-RX-space semaphore\n", pThis->szPrf));
1803 int rc = PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEventMoreRxDescAvail);
1804 AssertRC(rc);
1805 }
1806}
1807
1808#ifdef IN_RING3
1809
1810/**
1811 * Hardware reset. Revert all registers to initial values.
1812 *
1813 * @param pDevIns The device instance.
1814 * @param pThis The device state structure.
1815 * @param pThisCC The current context instance data.
1816 */
1817static void e1kR3HardReset(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
1818{
1819 E1kLog(("%s Hard reset triggered\n", pThis->szPrf));
1820 /* No interrupts should survive device reset, see @bugref(9556). */
1821 if (pThis->fIntRaised)
1822 {
1823 /* Lower(0) INTA(0) */
1824 PDMDevHlpPCISetIrq(pDevIns, 0, 0);
1825 pThis->fIntRaised = false;
1826 E1kLog(("%s e1kR3HardReset: Lowered IRQ: ICR=%08x\n", pThis->szPrf, ICR));
1827 }
1828 memset(pThis->auRegs, 0, sizeof(pThis->auRegs));
1829 memset(pThis->aRecAddr.au32, 0, sizeof(pThis->aRecAddr.au32));
1830# ifdef E1K_INIT_RA0
1831 memcpy(pThis->aRecAddr.au32, pThis->macConfigured.au8,
1832 sizeof(pThis->macConfigured.au8));
1833 pThis->aRecAddr.array[0].ctl |= RA_CTL_AV;
1834# endif /* E1K_INIT_RA0 */
1835 STATUS = 0x0081; /* SPEED=10b (1000 Mb/s), FD=1b (Full Duplex) */
1836 EECD = 0x0100; /* EE_PRES=1b (EEPROM present) */
1837 CTRL = 0x0a09; /* FRCSPD=1b SPEED=10b LRST=1b FD=1b */
1838 TSPMT = 0x01000400;/* TSMT=0400h TSPBP=0100h */
1839 Assert(GET_BITS(RCTL, BSIZE) == 0);
1840 pThis->u16RxBSize = 2048;
1841
1842 uint16_t u16LedCtl = 0x0602; /* LED0/LINK_UP#, LED2/LINK100# */
1843 pThisCC->eeprom.readWord(0x2F, &u16LedCtl); /* Read LEDCTL defaults from EEPROM */
1844 LEDCTL = 0x07008300 | (((uint32_t)u16LedCtl & 0xCF00) << 8) | (u16LedCtl & 0xCF); /* Only LED0 and LED2 defaults come from EEPROM */
1845
1846 /* Reset promiscuous mode */
1847 if (pThisCC->pDrvR3)
1848 pThisCC->pDrvR3->pfnSetPromiscuousMode(pThisCC->pDrvR3, false);
1849
1850# ifdef E1K_WITH_TXD_CACHE
1851 e1kR3CsTxEnterAsserted(pThis);
1852 pThis->nTxDFetched = 0;
1853 pThis->iTxDCurrent = 0;
1854 pThis->fGSO = false;
1855 pThis->cbTxAlloc = 0;
1856 e1kCsTxLeave(pThis);
1857# endif /* E1K_WITH_TXD_CACHE */
1858# ifdef E1K_WITH_RXD_CACHE
1859 e1kR3CsRxEnterAsserted(pThis);
1860 pThis->iRxDCurrent = pThis->nRxDFetched = 0;
1861 e1kCsRxLeave(pThis);
1862# endif /* E1K_WITH_RXD_CACHE */
1863# ifdef E1K_LSC_ON_RESET
1864 E1kLog(("%s Will trigger LSC in %d seconds...\n",
1865 pThis->szPrf, pThis->cMsLinkUpDelay / 1000));
1866 e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, pThis->cMsLinkUpDelay * 1000);
1867# endif /* E1K_LSC_ON_RESET */
1868}
1869
1870#endif /* IN_RING3 */
1871
1872/**
1873 * Compute Internet checksum.
1874 *
1875 * @remarks Refer to http://www.netfor2.com/checksum.html for short intro.
1876 *
1877 * @param pThis The device state structure.
1878 * @param cpPacket The packet.
1879 * @param cb The size of the packet.
1880 * @param pszText A string denoting direction of packet transfer.
1881 *
1882 * @return The 1's complement of the 1's complement sum.
1883 *
1884 * @thread E1000_TX
1885 */
1886static uint16_t e1kCSum16(const void *pvBuf, size_t cb)
1887{
1888 uint32_t csum = 0;
1889 uint16_t *pu16 = (uint16_t *)pvBuf;
1890
1891 while (cb > 1)
1892 {
1893 csum += *pu16++;
1894 cb -= 2;
1895 }
1896 if (cb)
1897 csum += *(uint8_t*)pu16;
1898 while (csum >> 16)
1899 csum = (csum >> 16) + (csum & 0xFFFF);
1900 Assert(csum < 65536);
1901 return (uint16_t)~csum;
1902}
1903
1904/**
1905 * Dump a packet to debug log.
1906 *
1907 * @param pDevIns The device instance.
1908 * @param pThis The device state structure.
1909 * @param cpPacket The packet.
1910 * @param cb The size of the packet.
1911 * @param pszText A string denoting direction of packet transfer.
1912 * @thread E1000_TX
1913 */
1914DECLINLINE(void) e1kPacketDump(PPDMDEVINS pDevIns, PE1KSTATE pThis, const uint8_t *cpPacket, size_t cb, const char *pszText)
1915{
1916#ifdef DEBUG
1917 if (RT_LIKELY(e1kCsEnter(pThis, VERR_SEM_BUSY) == VINF_SUCCESS))
1918 {
1919 Log4(("%s --- %s packet #%d: %RTmac => %RTmac (%d bytes) ---\n",
1920 pThis->szPrf, pszText, ++pThis->u32PktNo, cpPacket+6, cpPacket, cb));
1921 if (ntohs(*(uint16_t*)(cpPacket+12)) == 0x86DD)
1922 {
1923 Log4(("%s --- IPv6: %RTnaipv6 => %RTnaipv6\n",
1924 pThis->szPrf, cpPacket+14+8, cpPacket+14+24));
1925 if (*(cpPacket+14+6) == 0x6)
1926 Log4(("%s --- TCP: seq=%x ack=%x\n", pThis->szPrf,
1927 ntohl(*(uint32_t*)(cpPacket+14+40+4)), ntohl(*(uint32_t*)(cpPacket+14+40+8))));
1928 }
1929 else if (ntohs(*(uint16_t*)(cpPacket+12)) == 0x800)
1930 {
1931 Log4(("%s --- IPv4: %RTnaipv4 => %RTnaipv4\n",
1932 pThis->szPrf, *(uint32_t*)(cpPacket+14+12), *(uint32_t*)(cpPacket+14+16)));
1933 if (*(cpPacket+14+6) == 0x6)
1934 Log4(("%s --- TCP: seq=%x ack=%x\n", pThis->szPrf,
1935 ntohl(*(uint32_t*)(cpPacket+14+20+4)), ntohl(*(uint32_t*)(cpPacket+14+20+8))));
1936 }
1937 E1kLog3(("%.*Rhxd\n", cb, cpPacket));
1938 e1kCsLeave(pThis);
1939 }
1940#else
1941 if (RT_LIKELY(e1kCsEnter(pThis, VERR_SEM_BUSY) == VINF_SUCCESS))
1942 {
1943 if (ntohs(*(uint16_t*)(cpPacket+12)) == 0x86DD)
1944 E1kLogRel(("E1000: %s packet #%d, %RTmac => %RTmac, %RTnaipv6 => %RTnaipv6, seq=%x ack=%x\n",
1945 pszText, ++pThis->u32PktNo, cpPacket+6, cpPacket, cpPacket+14+8, cpPacket+14+24,
1946 ntohl(*(uint32_t*)(cpPacket+14+40+4)), ntohl(*(uint32_t*)(cpPacket+14+40+8))));
1947 else
1948 E1kLogRel(("E1000: %s packet #%d, %RTmac => %RTmac, %RTnaipv4 => %RTnaipv4, seq=%x ack=%x\n",
1949 pszText, ++pThis->u32PktNo, cpPacket+6, cpPacket,
1950 *(uint32_t*)(cpPacket+14+12), *(uint32_t*)(cpPacket+14+16),
1951 ntohl(*(uint32_t*)(cpPacket+14+20+4)), ntohl(*(uint32_t*)(cpPacket+14+20+8))));
1952 e1kCsLeave(pThis);
1953 }
1954 RT_NOREF2(cb, pszText);
1955#endif
1956}
1957
1958/**
1959 * Determine the type of transmit descriptor.
1960 *
1961 * @returns Descriptor type. See E1K_DTYP_XXX defines.
1962 *
1963 * @param pDesc Pointer to descriptor union.
1964 * @thread E1000_TX
1965 */
1966DECLINLINE(int) e1kGetDescType(E1KTXDESC *pDesc)
1967{
1968 if (pDesc->legacy.cmd.fDEXT)
1969 return pDesc->context.dw2.u4DTYP;
1970 return E1K_DTYP_LEGACY;
1971}
1972
1973
1974#ifdef E1K_WITH_RXD_CACHE
1975/**
1976 * Return the number of RX descriptor that belong to the hardware.
1977 *
1978 * @returns the number of available descriptors in RX ring.
1979 * @param pRxdc The receive descriptor register context.
1980 * @thread ???
1981 */
1982DECLINLINE(uint32_t) e1kGetRxLen(PE1KRXDC pRxdc)
1983{
1984 /**
1985 * Make sure RDT won't change during computation. EMT may modify RDT at
1986 * any moment.
1987 */
1988 uint32_t rdt = pRxdc->rdt;
1989 return (pRxdc->rdh > rdt ? pRxdc->rdlen/sizeof(E1KRXDESC) : 0) + rdt - pRxdc->rdh;
1990}
1991
1992DECLINLINE(unsigned) e1kRxDInCache(PE1KSTATE pThis)
1993{
1994 return pThis->nRxDFetched > pThis->iRxDCurrent ?
1995 pThis->nRxDFetched - pThis->iRxDCurrent : 0;
1996}
1997
1998DECLINLINE(unsigned) e1kRxDIsCacheEmpty(PE1KSTATE pThis)
1999{
2000 return pThis->iRxDCurrent >= pThis->nRxDFetched;
2001}
2002
2003/**
2004 * Load receive descriptors from guest memory. The caller needs to be in Rx
2005 * critical section.
2006 *
2007 * We need two physical reads in case the tail wrapped around the end of RX
2008 * descriptor ring.
2009 *
2010 * @returns the actual number of descriptors fetched.
2011 * @param pDevIns The device instance.
2012 * @param pThis The device state structure.
2013 * @thread EMT, RX
2014 */
2015DECLINLINE(unsigned) e1kRxDPrefetch(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pRxdc)
2016{
2017 E1kLog3(("%s e1kRxDPrefetch: RDH=%x RDT=%x RDLEN=%x "
2018 "iRxDCurrent=%x nRxDFetched=%x\n",
2019 pThis->szPrf, pRxdc->rdh, pRxdc->rdt, pRxdc->rdlen, pThis->iRxDCurrent, pThis->nRxDFetched));
2020 /* We've already loaded pThis->nRxDFetched descriptors past RDH. */
2021 unsigned nDescsAvailable = e1kGetRxLen(pRxdc) - e1kRxDInCache(pThis);
2022 unsigned nDescsToFetch = RT_MIN(nDescsAvailable, E1K_RXD_CACHE_SIZE - pThis->nRxDFetched);
2023 unsigned nDescsTotal = pRxdc->rdlen / sizeof(E1KRXDESC);
2024 Assert(nDescsTotal != 0);
2025 if (nDescsTotal == 0)
2026 return 0;
2027 unsigned nFirstNotLoaded = (pRxdc->rdh + e1kRxDInCache(pThis)) % nDescsTotal;
2028 unsigned nDescsInSingleRead = RT_MIN(nDescsToFetch, nDescsTotal - nFirstNotLoaded);
2029 E1kLog3(("%s e1kRxDPrefetch: nDescsAvailable=%u nDescsToFetch=%u "
2030 "nDescsTotal=%u nFirstNotLoaded=0x%x nDescsInSingleRead=%u\n",
2031 pThis->szPrf, nDescsAvailable, nDescsToFetch, nDescsTotal,
2032 nFirstNotLoaded, nDescsInSingleRead));
2033 if (nDescsToFetch == 0)
2034 return 0;
2035 E1KRXDESC* pFirstEmptyDesc = &pThis->aRxDescriptors[pThis->nRxDFetched];
2036 PDMDevHlpPCIPhysRead(pDevIns,
2037 ((uint64_t)RDBAH << 32) + RDBAL + nFirstNotLoaded * sizeof(E1KRXDESC),
2038 pFirstEmptyDesc, nDescsInSingleRead * sizeof(E1KRXDESC));
2039 // uint64_t addrBase = ((uint64_t)RDBAH << 32) + RDBAL;
2040 // unsigned i, j;
2041 // for (i = pThis->nRxDFetched; i < pThis->nRxDFetched + nDescsInSingleRead; ++i)
2042 // {
2043 // pThis->aRxDescAddr[i] = addrBase + (nFirstNotLoaded + i - pThis->nRxDFetched) * sizeof(E1KRXDESC);
2044 // E1kLog3(("%s aRxDescAddr[%d] = %p\n", pThis->szPrf, i, pThis->aRxDescAddr[i]));
2045 // }
2046 E1kLog3(("%s Fetched %u RX descriptors at %08x%08x(0x%x), RDLEN=%08x, RDH=%08x, RDT=%08x\n",
2047 pThis->szPrf, nDescsInSingleRead,
2048 RDBAH, RDBAL + pRxdc->rdh * sizeof(E1KRXDESC),
2049 nFirstNotLoaded, pRxdc->rdlen, pRxdc->rdh, pRxdc->rdt));
2050 if (nDescsToFetch > nDescsInSingleRead)
2051 {
2052 PDMDevHlpPCIPhysRead(pDevIns,
2053 ((uint64_t)RDBAH << 32) + RDBAL,
2054 pFirstEmptyDesc + nDescsInSingleRead,
2055 (nDescsToFetch - nDescsInSingleRead) * sizeof(E1KRXDESC));
2056 // Assert(i == pThis->nRxDFetched + nDescsInSingleRead);
2057 // for (j = 0; i < pThis->nRxDFetched + nDescsToFetch; ++i, ++j)
2058 // {
2059 // pThis->aRxDescAddr[i] = addrBase + j * sizeof(E1KRXDESC);
2060 // E1kLog3(("%s aRxDescAddr[%d] = %p\n", pThis->szPrf, i, pThis->aRxDescAddr[i]));
2061 // }
2062 E1kLog3(("%s Fetched %u RX descriptors at %08x%08x\n",
2063 pThis->szPrf, nDescsToFetch - nDescsInSingleRead,
2064 RDBAH, RDBAL));
2065 }
2066 pThis->nRxDFetched += nDescsToFetch;
2067 return nDescsToFetch;
2068}
2069
2070# ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
2071/**
2072 * Dump receive descriptor to debug log.
2073 *
2074 * @param pThis The device state structure.
2075 * @param pDesc Pointer to the descriptor.
2076 * @thread E1000_RX
2077 */
2078static void e1kPrintRDesc(PE1KSTATE pThis, E1KRXDESC *pDesc)
2079{
2080 RT_NOREF2(pThis, pDesc);
2081 E1kLog2(("%s <-- Receive Descriptor (%d bytes):\n", pThis->szPrf, pDesc->u16Length));
2082 E1kLog2((" Address=%16LX Length=%04X Csum=%04X\n",
2083 pDesc->u64BufAddr, pDesc->u16Length, pDesc->u16Checksum));
2084 E1kLog2((" STA: %s %s %s %s %s %s %s ERR: %s %s %s %s SPECIAL: %s VLAN=%03x PRI=%x\n",
2085 pDesc->status.fPIF ? "PIF" : "pif",
2086 pDesc->status.fIPCS ? "IPCS" : "ipcs",
2087 pDesc->status.fTCPCS ? "TCPCS" : "tcpcs",
2088 pDesc->status.fVP ? "VP" : "vp",
2089 pDesc->status.fIXSM ? "IXSM" : "ixsm",
2090 pDesc->status.fEOP ? "EOP" : "eop",
2091 pDesc->status.fDD ? "DD" : "dd",
2092 pDesc->status.fRXE ? "RXE" : "rxe",
2093 pDesc->status.fIPE ? "IPE" : "ipe",
2094 pDesc->status.fTCPE ? "TCPE" : "tcpe",
2095 pDesc->status.fCE ? "CE" : "ce",
2096 E1K_SPEC_CFI(pDesc->status.u16Special) ? "CFI" :"cfi",
2097 E1K_SPEC_VLAN(pDesc->status.u16Special),
2098 E1K_SPEC_PRI(pDesc->status.u16Special)));
2099}
2100# endif /* IN_RING3 */
2101#endif /* E1K_WITH_RXD_CACHE */
2102
2103/**
2104 * Dump transmit descriptor to debug log.
2105 *
2106 * @param pThis The device state structure.
2107 * @param pDesc Pointer to descriptor union.
2108 * @param pszDir A string denoting direction of descriptor transfer
2109 * @thread E1000_TX
2110 */
2111static void e1kPrintTDesc(PE1KSTATE pThis, E1KTXDESC *pDesc, const char *pszDir,
2112 unsigned uLevel = RTLOGGRPFLAGS_LEVEL_2)
2113{
2114 RT_NOREF4(pThis, pDesc, pszDir, uLevel);
2115
2116 /*
2117 * Unfortunately we cannot use our format handler here, we want R0 logging
2118 * as well.
2119 */
2120 switch (e1kGetDescType(pDesc))
2121 {
2122 case E1K_DTYP_CONTEXT:
2123 E1kLogX(uLevel, ("%s %s Context Transmit Descriptor %s\n",
2124 pThis->szPrf, pszDir, pszDir));
2125 E1kLogX(uLevel, (" IPCSS=%02X IPCSO=%02X IPCSE=%04X TUCSS=%02X TUCSO=%02X TUCSE=%04X\n",
2126 pDesc->context.ip.u8CSS, pDesc->context.ip.u8CSO, pDesc->context.ip.u16CSE,
2127 pDesc->context.tu.u8CSS, pDesc->context.tu.u8CSO, pDesc->context.tu.u16CSE));
2128 E1kLogX(uLevel, (" TUCMD:%s%s%s %s %s PAYLEN=%04x HDRLEN=%04x MSS=%04x STA: %s\n",
2129 pDesc->context.dw2.fIDE ? " IDE":"",
2130 pDesc->context.dw2.fRS ? " RS" :"",
2131 pDesc->context.dw2.fTSE ? " TSE":"",
2132 pDesc->context.dw2.fIP ? "IPv4":"IPv6",
2133 pDesc->context.dw2.fTCP ? "TCP":"UDP",
2134 pDesc->context.dw2.u20PAYLEN,
2135 pDesc->context.dw3.u8HDRLEN,
2136 pDesc->context.dw3.u16MSS,
2137 pDesc->context.dw3.fDD?"DD":""));
2138 break;
2139 case E1K_DTYP_DATA:
2140 E1kLogX(uLevel, ("%s %s Data Transmit Descriptor (%d bytes) %s\n",
2141 pThis->szPrf, pszDir, pDesc->data.cmd.u20DTALEN, pszDir));
2142 E1kLogX(uLevel, (" Address=%16LX DTALEN=%05X\n",
2143 pDesc->data.u64BufAddr,
2144 pDesc->data.cmd.u20DTALEN));
2145 E1kLogX(uLevel, (" DCMD:%s%s%s%s%s%s%s STA:%s%s%s POPTS:%s%s SPECIAL:%s VLAN=%03x PRI=%x\n",
2146 pDesc->data.cmd.fIDE ? " IDE" :"",
2147 pDesc->data.cmd.fVLE ? " VLE" :"",
2148 pDesc->data.cmd.fRPS ? " RPS" :"",
2149 pDesc->data.cmd.fRS ? " RS" :"",
2150 pDesc->data.cmd.fTSE ? " TSE" :"",
2151 pDesc->data.cmd.fIFCS? " IFCS":"",
2152 pDesc->data.cmd.fEOP ? " EOP" :"",
2153 pDesc->data.dw3.fDD ? " DD" :"",
2154 pDesc->data.dw3.fEC ? " EC" :"",
2155 pDesc->data.dw3.fLC ? " LC" :"",
2156 pDesc->data.dw3.fTXSM? " TXSM":"",
2157 pDesc->data.dw3.fIXSM? " IXSM":"",
2158 E1K_SPEC_CFI(pDesc->data.dw3.u16Special) ? "CFI" :"cfi",
2159 E1K_SPEC_VLAN(pDesc->data.dw3.u16Special),
2160 E1K_SPEC_PRI(pDesc->data.dw3.u16Special)));
2161 break;
2162 case E1K_DTYP_LEGACY:
2163 E1kLogX(uLevel, ("%s %s Legacy Transmit Descriptor (%d bytes) %s\n",
2164 pThis->szPrf, pszDir, pDesc->legacy.cmd.u16Length, pszDir));
2165 E1kLogX(uLevel, (" Address=%16LX DTALEN=%05X\n",
2166 pDesc->data.u64BufAddr,
2167 pDesc->legacy.cmd.u16Length));
2168 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",
2169 pDesc->legacy.cmd.fIDE ? " IDE" :"",
2170 pDesc->legacy.cmd.fVLE ? " VLE" :"",
2171 pDesc->legacy.cmd.fRPS ? " RPS" :"",
2172 pDesc->legacy.cmd.fRS ? " RS" :"",
2173 pDesc->legacy.cmd.fIC ? " IC" :"",
2174 pDesc->legacy.cmd.fIFCS? " IFCS":"",
2175 pDesc->legacy.cmd.fEOP ? " EOP" :"",
2176 pDesc->legacy.dw3.fDD ? " DD" :"",
2177 pDesc->legacy.dw3.fEC ? " EC" :"",
2178 pDesc->legacy.dw3.fLC ? " LC" :"",
2179 pDesc->legacy.cmd.u8CSO,
2180 pDesc->legacy.dw3.u8CSS,
2181 E1K_SPEC_CFI(pDesc->legacy.dw3.u16Special) ? "CFI" :"cfi",
2182 E1K_SPEC_VLAN(pDesc->legacy.dw3.u16Special),
2183 E1K_SPEC_PRI(pDesc->legacy.dw3.u16Special)));
2184 break;
2185 default:
2186 E1kLog(("%s %s Invalid Transmit Descriptor %s\n",
2187 pThis->szPrf, pszDir, pszDir));
2188 break;
2189 }
2190}
2191
2192/**
2193 * Raise an interrupt later.
2194 *
2195 * @param pThis The device state structure.
2196 */
2197DECLINLINE(void) e1kPostponeInterrupt(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint64_t nsDeadline)
2198{
2199 if (!PDMDevHlpTimerIsActive(pDevIns, pThis->hIntTimer))
2200 PDMDevHlpTimerSetNano(pDevIns, pThis->hIntTimer, nsDeadline);
2201}
2202
2203/**
2204 * Raise interrupt if not masked.
2205 *
2206 * @param pThis The device state structure.
2207 */
2208static int e1kRaiseInterrupt(PPDMDEVINS pDevIns, PE1KSTATE pThis, int rcBusy, uint32_t u32IntCause)
2209{
2210 /* Do NOT use e1kCsEnterReturn here as most callers doesn't check the
2211 status code. They'll pass a negative rcBusy. */
2212 int rc = e1kCsEnter(pThis, rcBusy);
2213 if (RT_LIKELY(rc == VINF_SUCCESS))
2214 { /* likely */ }
2215 else
2216 {
2217 PDM_CRITSECT_RELEASE_ASSERT_RC_DEV(pDevIns, &pThis->cs, rc);
2218 return rc;
2219 }
2220
2221 E1K_INC_ISTAT_CNT(pThis->uStatIntTry);
2222 ICR |= u32IntCause;
2223 if (ICR & IMS)
2224 {
2225 if (pThis->fIntRaised)
2226 {
2227 E1K_INC_ISTAT_CNT(pThis->uStatIntSkip);
2228 E1kLog2(("%s e1kRaiseInterrupt: Already raised, skipped. ICR&IMS=%08x\n",
2229 pThis->szPrf, ICR & IMS));
2230 }
2231 else
2232 {
2233 uint64_t tsNow = PDMDevHlpTimerGet(pDevIns, pThis->hIntTimer);
2234 if (!!ITR && tsNow - pThis->u64AckedAt < ITR * 256
2235 && pThis->fItrEnabled && (pThis->fItrRxEnabled || !(ICR & ICR_RXT0)))
2236 {
2237 E1K_INC_ISTAT_CNT(pThis->uStatIntEarly);
2238 E1kLog2(("%s e1kRaiseInterrupt: Too early to raise again: %d ns < %d ns.\n",
2239 pThis->szPrf, (uint32_t)(tsNow - pThis->u64AckedAt), ITR * 256));
2240 e1kPostponeInterrupt(pDevIns, pThis, ITR * 256);
2241 }
2242 else
2243 {
2244
2245 /* Since we are delivering the interrupt now
2246 * there is no need to do it later -- stop the timer.
2247 */
2248 PDMDevHlpTimerStop(pDevIns, pThis->hIntTimer);
2249 E1K_INC_ISTAT_CNT(pThis->uStatInt);
2250 STAM_COUNTER_INC(&pThis->StatIntsRaised);
2251 /* Got at least one unmasked interrupt cause */
2252 pThis->fIntRaised = true;
2253 /* Raise(1) INTA(0) */
2254 E1kLogRel(("E1000: irq RAISED icr&mask=0x%x, icr=0x%x\n", ICR & IMS, ICR));
2255 PDMDevHlpPCISetIrq(pDevIns, 0, 1);
2256 E1kLog(("%s e1kRaiseInterrupt: Raised. ICR&IMS=%08x\n",
2257 pThis->szPrf, ICR & IMS));
2258 }
2259 }
2260 }
2261 else
2262 {
2263 E1K_INC_ISTAT_CNT(pThis->uStatIntMasked);
2264 E1kLog2(("%s e1kRaiseInterrupt: Not raising, ICR=%08x, IMS=%08x\n",
2265 pThis->szPrf, ICR, IMS));
2266 }
2267 e1kCsLeave(pThis);
2268 return VINF_SUCCESS;
2269}
2270
2271/**
2272 * Compute the physical address of the descriptor.
2273 *
2274 * @returns the physical address of the descriptor.
2275 *
2276 * @param baseHigh High-order 32 bits of descriptor table address.
2277 * @param baseLow Low-order 32 bits of descriptor table address.
2278 * @param idxDesc The descriptor index in the table.
2279 */
2280DECLINLINE(RTGCPHYS) e1kDescAddr(uint32_t baseHigh, uint32_t baseLow, uint32_t idxDesc)
2281{
2282 AssertCompile(sizeof(E1KRXDESC) == sizeof(E1KTXDESC));
2283 return ((uint64_t)baseHigh << 32) + baseLow + idxDesc * sizeof(E1KRXDESC);
2284}
2285
2286#ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
2287/**
2288 * Advance the head pointer of the receive descriptor queue.
2289 *
2290 * @remarks RDH always points to the next available RX descriptor.
2291 *
2292 * @param pDevIns The device instance.
2293 * @param pThis The device state structure.
2294 */
2295DECLINLINE(void) e1kAdvanceRDH(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pRxdc)
2296{
2297 Assert(e1kCsRxIsOwner(pThis));
2298 //e1kR3CsEnterAsserted(pThis);
2299 if (++pRxdc->rdh * sizeof(E1KRXDESC) >= pRxdc->rdlen)
2300 pRxdc->rdh = 0;
2301 RDH = pRxdc->rdh; /* Sync the actual register and RXDC */
2302#ifdef E1K_WITH_RXD_CACHE
2303 /*
2304 * We need to fetch descriptors now as the guest may advance RDT all the way
2305 * to RDH as soon as we generate RXDMT0 interrupt. This is mostly to provide
2306 * compatibility with Phar Lap ETS, see @bugref(7346). Note that we do not
2307 * check if the receiver is enabled. It must be, otherwise we won't get here
2308 * in the first place.
2309 *
2310 * Note that we should have moved both RDH and iRxDCurrent by now.
2311 */
2312 if (e1kRxDIsCacheEmpty(pThis))
2313 {
2314 /* Cache is empty, reset it and check if we can fetch more. */
2315 pThis->iRxDCurrent = pThis->nRxDFetched = 0;
2316 E1kLog3(("%s e1kAdvanceRDH: Rx cache is empty, RDH=%x RDT=%x "
2317 "iRxDCurrent=%x nRxDFetched=%x\n",
2318 pThis->szPrf, pRxdc->rdh, pRxdc->rdt, pThis->iRxDCurrent, pThis->nRxDFetched));
2319 e1kRxDPrefetch(pDevIns, pThis, pRxdc);
2320 }
2321#endif /* E1K_WITH_RXD_CACHE */
2322 /*
2323 * Compute current receive queue length and fire RXDMT0 interrupt
2324 * if we are low on receive buffers
2325 */
2326 uint32_t uRQueueLen = pRxdc->rdh>pRxdc->rdt ? pRxdc->rdlen/sizeof(E1KRXDESC)-pRxdc->rdh+pRxdc->rdt : pRxdc->rdt-pRxdc->rdh;
2327 /*
2328 * The minimum threshold is controlled by RDMTS bits of RCTL:
2329 * 00 = 1/2 of RDLEN
2330 * 01 = 1/4 of RDLEN
2331 * 10 = 1/8 of RDLEN
2332 * 11 = reserved
2333 */
2334 uint32_t uMinRQThreshold = pRxdc->rdlen / sizeof(E1KRXDESC) / (2 << GET_BITS(RCTL, RDMTS));
2335 if (uRQueueLen <= uMinRQThreshold)
2336 {
2337 E1kLogRel(("E1000: low on RX descriptors, RDH=%x RDT=%x len=%x threshold=%x\n", pRxdc->rdh, pRxdc->rdt, uRQueueLen, uMinRQThreshold));
2338 E1kLog2(("%s Low on RX descriptors, RDH=%x RDT=%x len=%x threshold=%x, raise an interrupt\n",
2339 pThis->szPrf, pRxdc->rdh, pRxdc->rdt, uRQueueLen, uMinRQThreshold));
2340 E1K_INC_ISTAT_CNT(pThis->uStatIntRXDMT0);
2341 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_RXDMT0);
2342 }
2343 E1kLog2(("%s e1kAdvanceRDH: at exit RDH=%x RDT=%x len=%x\n",
2344 pThis->szPrf, pRxdc->rdh, pRxdc->rdt, uRQueueLen));
2345 //e1kCsLeave(pThis);
2346}
2347#endif /* IN_RING3 */
2348
2349#ifdef E1K_WITH_RXD_CACHE
2350
2351# ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
2352
2353/**
2354 * Obtain the next RX descriptor from RXD cache, fetching descriptors from the
2355 * RX ring if the cache is empty.
2356 *
2357 * Note that we cannot advance the cache pointer (iRxDCurrent) yet as it will
2358 * go out of sync with RDH which will cause trouble when EMT checks if the
2359 * cache is empty to do pre-fetch @bugref(6217).
2360 *
2361 * @param pDevIns The device instance.
2362 * @param pThis The device state structure.
2363 * @thread RX
2364 */
2365DECLINLINE(E1KRXDESC *) e1kRxDGet(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KRXDC pRxdc)
2366{
2367 Assert(e1kCsRxIsOwner(pThis));
2368 /* Check the cache first. */
2369 if (pThis->iRxDCurrent < pThis->nRxDFetched)
2370 return &pThis->aRxDescriptors[pThis->iRxDCurrent];
2371 /* Cache is empty, reset it and check if we can fetch more. */
2372 pThis->iRxDCurrent = pThis->nRxDFetched = 0;
2373 if (e1kRxDPrefetch(pDevIns, pThis, pRxdc))
2374 return &pThis->aRxDescriptors[pThis->iRxDCurrent];
2375 /* Out of Rx descriptors. */
2376 return NULL;
2377}
2378
2379
2380/**
2381 * Return the RX descriptor obtained with e1kRxDGet() and advance the cache
2382 * pointer. The descriptor gets written back to the RXD ring.
2383 *
2384 * @param pDevIns The device instance.
2385 * @param pThis The device state structure.
2386 * @param pDesc The descriptor being "returned" to the RX ring.
2387 * @thread RX
2388 */
2389DECLINLINE(void) e1kRxDPut(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KRXDESC* pDesc, PE1KRXDC pRxdc)
2390{
2391 Assert(e1kCsRxIsOwner(pThis));
2392 pThis->iRxDCurrent++;
2393 // Assert(pDesc >= pThis->aRxDescriptors);
2394 // Assert(pDesc < pThis->aRxDescriptors + E1K_RXD_CACHE_SIZE);
2395 // uint64_t addr = e1kDescAddr(RDBAH, RDBAL, RDH);
2396 // uint32_t rdh = RDH;
2397 // Assert(pThis->aRxDescAddr[pDesc - pThis->aRxDescriptors] == addr);
2398 PDMDevHlpPCIPhysWrite(pDevIns, e1kDescAddr(RDBAH, RDBAL, pRxdc->rdh), pDesc, sizeof(E1KRXDESC));
2399 /*
2400 * We need to print the descriptor before advancing RDH as it may fetch new
2401 * descriptors into the cache.
2402 */
2403 e1kPrintRDesc(pThis, pDesc);
2404 e1kAdvanceRDH(pDevIns, pThis, pRxdc);
2405}
2406
2407/**
2408 * Store a fragment of received packet at the specifed address.
2409 *
2410 * @param pDevIns The device instance.
2411 * @param pThis The device state structure.
2412 * @param pDesc The next available RX descriptor.
2413 * @param pvBuf The fragment.
2414 * @param cb The size of the fragment.
2415 */
2416static void e1kStoreRxFragment(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KRXDESC *pDesc, const void *pvBuf, size_t cb)
2417{
2418 STAM_PROFILE_ADV_START(&pThis->StatReceiveStore, a);
2419 E1kLog2(("%s e1kStoreRxFragment: store fragment of %04X at %016LX, EOP=%d\n",
2420 pThis->szPrf, cb, pDesc->u64BufAddr, pDesc->status.fEOP));
2421 PDMDevHlpPCIPhysWrite(pDevIns, pDesc->u64BufAddr, pvBuf, cb);
2422 pDesc->u16Length = (uint16_t)cb;
2423 Assert(pDesc->u16Length == cb);
2424 STAM_PROFILE_ADV_STOP(&pThis->StatReceiveStore, a);
2425 RT_NOREF(pThis);
2426}
2427
2428# endif /* IN_RING3 */
2429
2430#else /* !E1K_WITH_RXD_CACHE */
2431
2432/**
2433 * Store a fragment of received packet that fits into the next available RX
2434 * buffer.
2435 *
2436 * @remarks Trigger the RXT0 interrupt if it is the last fragment of the packet.
2437 *
2438 * @param pDevIns The device instance.
2439 * @param pThis The device state structure.
2440 * @param pDesc The next available RX descriptor.
2441 * @param pvBuf The fragment.
2442 * @param cb The size of the fragment.
2443 */
2444static void e1kStoreRxFragment(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KRXDESC *pDesc, const void *pvBuf, size_t cb)
2445{
2446 STAM_PROFILE_ADV_START(&pThis->StatReceiveStore, a);
2447 E1kLog2(("%s e1kStoreRxFragment: store fragment of %04X at %016LX, EOP=%d\n", pThis->szPrf, cb, pDesc->u64BufAddr, pDesc->status.fEOP));
2448 PDMDevHlpPCIPhysWrite(pDevIns, pDesc->u64BufAddr, pvBuf, cb);
2449 pDesc->u16Length = (uint16_t)cb; Assert(pDesc->u16Length == cb);
2450 /* Write back the descriptor */
2451 PDMDevHlpPCIPhysWrite(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), pDesc, sizeof(E1KRXDESC));
2452 e1kPrintRDesc(pThis, pDesc);
2453 E1kLogRel(("E1000: Wrote back RX desc, RDH=%x\n", RDH));
2454 /* Advance head */
2455 e1kAdvanceRDH(pDevIns, pThis);
2456 //E1kLog2(("%s e1kStoreRxFragment: EOP=%d RDTR=%08X RADV=%08X\n", pThis->szPrf, pDesc->fEOP, RDTR, RADV));
2457 if (pDesc->status.fEOP)
2458 {
2459 /* Complete packet has been stored -- it is time to let the guest know. */
2460#ifdef E1K_USE_RX_TIMERS
2461 if (RDTR)
2462 {
2463 /* Arm the timer to fire in RDTR usec (discard .024) */
2464 e1kArmTimer(pDevIns, pThis, pThis->hRIDTimer, RDTR);
2465 /* If absolute timer delay is enabled and the timer is not running yet, arm it. */
2466 if (RADV != 0 && !PDMDevHlpTimerIsActive(pDevIns, pThis->CTX_SUFF(pRADTimer)))
2467 e1kArmTimer(pThis, pThis->hRADTimer, RADV);
2468 }
2469 else
2470 {
2471#endif
2472 /* 0 delay means immediate interrupt */
2473 E1K_INC_ISTAT_CNT(pThis->uStatIntRx);
2474 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_RXT0);
2475#ifdef E1K_USE_RX_TIMERS
2476 }
2477#endif
2478 }
2479 STAM_PROFILE_ADV_STOP(&pThis->StatReceiveStore, a);
2480}
2481
2482#endif /* !E1K_WITH_RXD_CACHE */
2483
2484/**
2485 * Returns true if it is a broadcast packet.
2486 *
2487 * @returns true if destination address indicates broadcast.
2488 * @param pvBuf The ethernet packet.
2489 */
2490DECLINLINE(bool) e1kIsBroadcast(const void *pvBuf)
2491{
2492 static const uint8_t s_abBcastAddr[] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
2493 return memcmp(pvBuf, s_abBcastAddr, sizeof(s_abBcastAddr)) == 0;
2494}
2495
2496/**
2497 * Returns true if it is a multicast packet.
2498 *
2499 * @remarks returns true for broadcast packets as well.
2500 * @returns true if destination address indicates multicast.
2501 * @param pvBuf The ethernet packet.
2502 */
2503DECLINLINE(bool) e1kIsMulticast(const void *pvBuf)
2504{
2505 return (*(char*)pvBuf) & 1;
2506}
2507
2508#ifdef IN_RING3 /* currently only used in ring-3 due to stack space requirements of the caller */
2509/**
2510 * Set IXSM, IPCS and TCPCS flags according to the packet type.
2511 *
2512 * @remarks We emulate checksum offloading for major packets types only.
2513 *
2514 * @returns VBox status code.
2515 * @param pThis The device state structure.
2516 * @param pFrame The available data.
2517 * @param cb Number of bytes available in the buffer.
2518 * @param status Bit fields containing status info.
2519 */
2520static int e1kRxChecksumOffload(PE1KSTATE pThis, const uint8_t *pFrame, size_t cb, E1KRXDST *pStatus)
2521{
2522 /** @todo
2523 * It is not safe to bypass checksum verification for packets coming
2524 * from real wire. We currently unable to tell where packets are
2525 * coming from so we tell the driver to ignore our checksum flags
2526 * and do verification in software.
2527 */
2528# if 0
2529 uint16_t uEtherType = ntohs(*(uint16_t*)(pFrame + 12));
2530
2531 E1kLog2(("%s e1kRxChecksumOffload: EtherType=%x\n", pThis->szPrf, uEtherType));
2532
2533 switch (uEtherType)
2534 {
2535 case 0x800: /* IPv4 */
2536 {
2537 pStatus->fIXSM = false;
2538 pStatus->fIPCS = true;
2539 PRTNETIPV4 pIpHdr4 = (PRTNETIPV4)(pFrame + 14);
2540 /* TCP/UDP checksum offloading works with TCP and UDP only */
2541 pStatus->fTCPCS = pIpHdr4->ip_p == 6 || pIpHdr4->ip_p == 17;
2542 break;
2543 }
2544 case 0x86DD: /* IPv6 */
2545 pStatus->fIXSM = false;
2546 pStatus->fIPCS = false;
2547 pStatus->fTCPCS = true;
2548 break;
2549 default: /* ARP, VLAN, etc. */
2550 pStatus->fIXSM = true;
2551 break;
2552 }
2553# else
2554 pStatus->fIXSM = true;
2555 RT_NOREF_PV(pThis); RT_NOREF_PV(pFrame); RT_NOREF_PV(cb);
2556# endif
2557 return VINF_SUCCESS;
2558}
2559#endif /* IN_RING3 */
2560
2561/**
2562 * Pad and store received packet.
2563 *
2564 * @remarks Make sure that the packet appears to upper layer as one coming
2565 * from real Ethernet: pad it and insert FCS.
2566 *
2567 * @returns VBox status code.
2568 * @param pDevIns The device instance.
2569 * @param pThis The device state structure.
2570 * @param pvBuf The available data.
2571 * @param cb Number of bytes available in the buffer.
2572 * @param status Bit fields containing status info.
2573 */
2574static int e1kHandleRxPacket(PPDMDEVINS pDevIns, PE1KSTATE pThis, const void *pvBuf, size_t cb, E1KRXDST status)
2575{
2576#if defined(IN_RING3) /** @todo Remove this extra copying, it's gonna make us run out of kernel / hypervisor stack! */
2577 uint8_t rxPacket[E1K_MAX_RX_PKT_SIZE];
2578 uint8_t *ptr = rxPacket;
2579# ifdef E1K_WITH_RXD_CACHE
2580 E1KRXDC rxdc;
2581# endif /* E1K_WITH_RXD_CACHE */
2582
2583 e1kCsRxEnterReturn(pThis);
2584# ifdef E1K_WITH_RXD_CACHE
2585 if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kHandleRxPacket")))
2586 {
2587 e1kCsRxLeave(pThis);
2588 E1kLog(("%s e1kHandleRxPacket: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
2589 return VINF_SUCCESS;
2590 }
2591# endif /* E1K_WITH_RXD_CACHE */
2592
2593 if (cb > 70) /* unqualified guess */
2594 pThis->led.Asserted.s.fReading = pThis->led.Actual.s.fReading = 1;
2595
2596 Assert(cb <= E1K_MAX_RX_PKT_SIZE);
2597 Assert(cb > 16);
2598 size_t cbMax = ((RCTL & RCTL_LPE) ? E1K_MAX_RX_PKT_SIZE - 4 : 1518) - (status.fVP ? 0 : 4);
2599 E1kLog3(("%s Max RX packet size is %u\n", pThis->szPrf, cbMax));
2600 if (status.fVP)
2601 {
2602 /* VLAN packet -- strip VLAN tag in VLAN mode */
2603 if ((CTRL & CTRL_VME) && cb > 16)
2604 {
2605 uint16_t *u16Ptr = (uint16_t*)pvBuf;
2606 memcpy(rxPacket, pvBuf, 12); /* Copy src and dst addresses */
2607 status.u16Special = RT_BE2H_U16(u16Ptr[7]); /* Extract VLAN tag */
2608 memcpy(rxPacket + 12, (uint8_t*)pvBuf + 16, cb - 16); /* Copy the rest of the packet */
2609 cb -= 4;
2610 E1kLog3(("%s Stripped tag for VLAN %u (cb=%u)\n",
2611 pThis->szPrf, status.u16Special, cb));
2612 }
2613 else
2614 {
2615 status.fVP = false; /* Set VP only if we stripped the tag */
2616 memcpy(rxPacket, pvBuf, cb);
2617 }
2618 }
2619 else
2620 memcpy(rxPacket, pvBuf, cb);
2621 /* Pad short packets */
2622 if (cb < 60)
2623 {
2624 memset(rxPacket + cb, 0, 60 - cb);
2625 cb = 60;
2626 }
2627 if (!(RCTL & RCTL_SECRC) && cb <= cbMax)
2628 {
2629 STAM_PROFILE_ADV_START(&pThis->StatReceiveCRC, a);
2630 /*
2631 * Add FCS if CRC stripping is not enabled. Since the value of CRC
2632 * is ignored by most of drivers we may as well save us the trouble
2633 * of calculating it (see EthernetCRC CFGM parameter).
2634 */
2635 if (pThis->fEthernetCRC)
2636 *(uint32_t*)(rxPacket + cb) = RTCrc32(rxPacket, cb);
2637 cb += sizeof(uint32_t);
2638 STAM_PROFILE_ADV_STOP(&pThis->StatReceiveCRC, a);
2639 E1kLog3(("%s Added FCS (cb=%u)\n", pThis->szPrf, cb));
2640 }
2641 /* Compute checksum of complete packet */
2642 size_t cbCSumStart = RT_MIN(GET_BITS(RXCSUM, PCSS), cb);
2643 uint16_t checksum = e1kCSum16(rxPacket + cbCSumStart, cb - cbCSumStart);
2644 e1kRxChecksumOffload(pThis, rxPacket, cb, &status);
2645
2646 /* Update stats */
2647 E1K_INC_CNT32(GPRC);
2648 if (e1kIsBroadcast(pvBuf))
2649 E1K_INC_CNT32(BPRC);
2650 else if (e1kIsMulticast(pvBuf))
2651 E1K_INC_CNT32(MPRC);
2652 /* Update octet receive counter */
2653 E1K_ADD_CNT64(GORCL, GORCH, cb);
2654 STAM_REL_COUNTER_ADD(&pThis->StatReceiveBytes, cb);
2655 if (cb == 64)
2656 E1K_INC_CNT32(PRC64);
2657 else if (cb < 128)
2658 E1K_INC_CNT32(PRC127);
2659 else if (cb < 256)
2660 E1K_INC_CNT32(PRC255);
2661 else if (cb < 512)
2662 E1K_INC_CNT32(PRC511);
2663 else if (cb < 1024)
2664 E1K_INC_CNT32(PRC1023);
2665 else
2666 E1K_INC_CNT32(PRC1522);
2667
2668 E1K_INC_ISTAT_CNT(pThis->uStatRxFrm);
2669
2670# ifdef E1K_WITH_RXD_CACHE
2671 while (cb > 0)
2672 {
2673 E1KRXDESC *pDesc = e1kRxDGet(pDevIns, pThis, &rxdc);
2674
2675 if (pDesc == NULL)
2676 {
2677 E1kLog(("%s Out of receive buffers, dropping the packet "
2678 "(cb=%u, in_cache=%u, RDH=%x RDT=%x)\n",
2679 pThis->szPrf, cb, e1kRxDInCache(pThis), rxdc.rdh, rxdc.rdt));
2680 break;
2681 }
2682# else /* !E1K_WITH_RXD_CACHE */
2683 if (RDH == RDT)
2684 {
2685 E1kLog(("%s Out of receive buffers, dropping the packet\n",
2686 pThis->szPrf));
2687 }
2688 /* Store the packet to receive buffers */
2689 while (RDH != RDT)
2690 {
2691 /* Load the descriptor pointed by head */
2692 E1KRXDESC desc, *pDesc = &desc;
2693 PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), &desc, sizeof(desc));
2694# endif /* !E1K_WITH_RXD_CACHE */
2695 if (pDesc->u64BufAddr)
2696 {
2697 uint16_t u16RxBufferSize = pThis->u16RxBSize; /* see @bugref{9427} */
2698
2699 /* Update descriptor */
2700 pDesc->status = status;
2701 pDesc->u16Checksum = checksum;
2702 pDesc->status.fDD = true;
2703
2704 /*
2705 * We need to leave Rx critical section here or we risk deadlocking
2706 * with EMT in e1kRegWriteRDT when the write is to an unallocated
2707 * page or has an access handler associated with it.
2708 * Note that it is safe to leave the critical section here since
2709 * e1kRegWriteRDT() never modifies RDH. It never touches already
2710 * fetched RxD cache entries either.
2711 */
2712 if (cb > u16RxBufferSize)
2713 {
2714 pDesc->status.fEOP = false;
2715 e1kCsRxLeave(pThis);
2716 e1kStoreRxFragment(pDevIns, pThis, pDesc, ptr, u16RxBufferSize);
2717 e1kCsRxEnterReturn(pThis);
2718# ifdef E1K_WITH_RXD_CACHE
2719 if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kHandleRxPacket")))
2720 {
2721 e1kCsRxLeave(pThis);
2722 E1kLog(("%s e1kHandleRxPacket: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
2723 return VINF_SUCCESS;
2724 }
2725# endif /* E1K_WITH_RXD_CACHE */
2726 ptr += u16RxBufferSize;
2727 cb -= u16RxBufferSize;
2728 }
2729 else
2730 {
2731 pDesc->status.fEOP = true;
2732 e1kCsRxLeave(pThis);
2733 e1kStoreRxFragment(pDevIns, pThis, pDesc, ptr, cb);
2734# ifdef E1K_WITH_RXD_CACHE
2735 e1kCsRxEnterReturn(pThis);
2736 if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kHandleRxPacket")))
2737 {
2738 e1kCsRxLeave(pThis);
2739 E1kLog(("%s e1kHandleRxPacket: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
2740 return VINF_SUCCESS;
2741 }
2742 cb = 0;
2743# else /* !E1K_WITH_RXD_CACHE */
2744 pThis->led.Actual.s.fReading = 0;
2745 return VINF_SUCCESS;
2746# endif /* !E1K_WITH_RXD_CACHE */
2747 }
2748 /*
2749 * Note: RDH is advanced by e1kStoreRxFragment if E1K_WITH_RXD_CACHE
2750 * is not defined.
2751 */
2752 }
2753# ifdef E1K_WITH_RXD_CACHE
2754 /* Write back the descriptor. */
2755 pDesc->status.fDD = true;
2756 e1kRxDPut(pDevIns, pThis, pDesc, &rxdc);
2757# else /* !E1K_WITH_RXD_CACHE */
2758 else
2759 {
2760 /* Write back the descriptor. */
2761 pDesc->status.fDD = true;
2762 PDMDevHlpPCIPhysWrite(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), pDesc, sizeof(E1KRXDESC));
2763 e1kAdvanceRDH(pDevIns, pThis);
2764 }
2765# endif /* !E1K_WITH_RXD_CACHE */
2766 }
2767
2768 if (cb > 0)
2769 E1kLog(("%s Out of receive buffers, dropping %u bytes", pThis->szPrf, cb));
2770
2771 pThis->led.Actual.s.fReading = 0;
2772
2773 e1kCsRxLeave(pThis);
2774# ifdef E1K_WITH_RXD_CACHE
2775 /* Complete packet has been stored -- it is time to let the guest know. */
2776# ifdef E1K_USE_RX_TIMERS
2777 if (RDTR)
2778 {
2779 /* Arm the timer to fire in RDTR usec (discard .024) */
2780 e1kArmTimer(pThis, pThis->hRIDTimer, RDTR);
2781 /* If absolute timer delay is enabled and the timer is not running yet, arm it. */
2782 if (RADV != 0 && !PDMDevHlpTimerIsActive(pDevIns, pThis->hRADTimer))
2783 e1kArmTimer(pThis, pThis->hRADTimer, RADV);
2784 }
2785 else
2786 {
2787# endif /* E1K_USE_RX_TIMERS */
2788 /* 0 delay means immediate interrupt */
2789 E1K_INC_ISTAT_CNT(pThis->uStatIntRx);
2790 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_RXT0);
2791# ifdef E1K_USE_RX_TIMERS
2792 }
2793# endif /* E1K_USE_RX_TIMERS */
2794# endif /* E1K_WITH_RXD_CACHE */
2795
2796 return VINF_SUCCESS;
2797#else /* !IN_RING3 */
2798 RT_NOREF(pDevIns, pThis, pvBuf, cb, status);
2799 return VERR_INTERNAL_ERROR_2;
2800#endif /* !IN_RING3 */
2801}
2802
2803
2804#ifdef IN_RING3
2805/**
2806 * Bring the link up after the configured delay, 5 seconds by default.
2807 *
2808 * @param pDevIns The device instance.
2809 * @param pThis The device state structure.
2810 * @thread any
2811 */
2812DECLINLINE(void) e1kBringLinkUpDelayed(PPDMDEVINS pDevIns, PE1KSTATE pThis)
2813{
2814 E1kLog(("%s Will bring up the link in %d seconds...\n",
2815 pThis->szPrf, pThis->cMsLinkUpDelay / 1000));
2816 e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, pThis->cMsLinkUpDelay * 1000);
2817}
2818
2819/**
2820 * Bring up the link immediately.
2821 *
2822 * @param pDevIns The device instance.
2823 * @param pThis The device state structure.
2824 * @param pThisCC The current context instance data.
2825 */
2826DECLINLINE(void) e1kR3LinkUp(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
2827{
2828 E1kLog(("%s Link is up\n", pThis->szPrf));
2829 STATUS |= STATUS_LU;
2830 Phy::setLinkStatus(&pThis->phy, true);
2831 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_LSC);
2832 if (pThisCC->pDrvR3)
2833 pThisCC->pDrvR3->pfnNotifyLinkChanged(pThisCC->pDrvR3, PDMNETWORKLINKSTATE_UP);
2834 /* Trigger processing of pending TX descriptors (see @bugref{8942}). */
2835 PDMDevHlpTaskTrigger(pDevIns, pThis->hTxTask);
2836}
2837
2838/**
2839 * Bring down the link immediately.
2840 *
2841 * @param pDevIns The device instance.
2842 * @param pThis The device state structure.
2843 * @param pThisCC The current context instance data.
2844 */
2845DECLINLINE(void) e1kR3LinkDown(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
2846{
2847 E1kLog(("%s Link is down\n", pThis->szPrf));
2848 STATUS &= ~STATUS_LU;
2849#ifdef E1K_LSC_ON_RESET
2850 Phy::setLinkStatus(&pThis->phy, false);
2851#endif /* E1K_LSC_ON_RESET */
2852 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_LSC);
2853 if (pThisCC->pDrvR3)
2854 pThisCC->pDrvR3->pfnNotifyLinkChanged(pThisCC->pDrvR3, PDMNETWORKLINKSTATE_DOWN);
2855}
2856
2857/**
2858 * Bring down the link temporarily.
2859 *
2860 * @param pDevIns The device instance.
2861 * @param pThis The device state structure.
2862 * @param pThisCC The current context instance data.
2863 */
2864DECLINLINE(void) e1kR3LinkDownTemp(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC)
2865{
2866 E1kLog(("%s Link is down temporarily\n", pThis->szPrf));
2867 STATUS &= ~STATUS_LU;
2868 Phy::setLinkStatus(&pThis->phy, false);
2869 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_LSC);
2870 /*
2871 * Notifying the associated driver that the link went down (even temporarily)
2872 * seems to be the right thing, but it was not done before. This may cause
2873 * a regression if the driver does not expect the link to go down as a result
2874 * of sending PDMNETWORKLINKSTATE_DOWN_RESUME to this device. Earlier versions
2875 * of code notified the driver that the link was up! See @bugref{7057}.
2876 */
2877 if (pThisCC->pDrvR3)
2878 pThisCC->pDrvR3->pfnNotifyLinkChanged(pThisCC->pDrvR3, PDMNETWORKLINKSTATE_DOWN);
2879 e1kBringLinkUpDelayed(pDevIns, pThis);
2880}
2881#endif /* IN_RING3 */
2882
2883#if 0 /* unused */
2884/**
2885 * Read handler for Device Status register.
2886 *
2887 * Get the link status from PHY.
2888 *
2889 * @returns VBox status code.
2890 *
2891 * @param pThis The device state structure.
2892 * @param offset Register offset in memory-mapped frame.
2893 * @param index Register index in register array.
2894 * @param mask Used to implement partial reads (8 and 16-bit).
2895 */
2896static int e1kRegReadCTRL(PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
2897{
2898 E1kLog(("%s e1kRegReadCTRL: mdio dir=%s mdc dir=%s mdc=%d\n",
2899 pThis->szPrf, (CTRL & CTRL_MDIO_DIR)?"OUT":"IN ",
2900 (CTRL & CTRL_MDC_DIR)?"OUT":"IN ", !!(CTRL & CTRL_MDC)));
2901 if ((CTRL & CTRL_MDIO_DIR) == 0 && (CTRL & CTRL_MDC))
2902 {
2903 /* MDC is high and MDIO pin is used for input, read MDIO pin from PHY */
2904 if (Phy::readMDIO(&pThis->phy))
2905 *pu32Value = CTRL | CTRL_MDIO;
2906 else
2907 *pu32Value = CTRL & ~CTRL_MDIO;
2908 E1kLog(("%s e1kRegReadCTRL: Phy::readMDIO(%d)\n",
2909 pThis->szPrf, !!(*pu32Value & CTRL_MDIO)));
2910 }
2911 else
2912 {
2913 /* MDIO pin is used for output, ignore it */
2914 *pu32Value = CTRL;
2915 }
2916 return VINF_SUCCESS;
2917}
2918#endif /* unused */
2919
2920/**
2921 * A helper function to detect the link state to the other side of "the wire".
2922 *
2923 * When deciding to bring up the link we need to take into account both if the
2924 * cable is connected and if our device is actually connected to the outside
2925 * world. If no driver is attached we won't be able to allocate TX buffers,
2926 * which will prevent us from TX descriptor processing, which will result in
2927 * "TX unit hang" in the guest.
2928 *
2929 * @returns true if the device is connected to something.
2930 *
2931 * @param pDevIns The device instance.
2932 */
2933DECLINLINE(bool) e1kIsConnected(PPDMDEVINS pDevIns)
2934{
2935 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
2936 return pThis->fCableConnected && pThis->fIsAttached;
2937}
2938
2939/**
2940 * A callback used by PHY to indicate that the link needs to be updated due to
2941 * reset of PHY.
2942 *
2943 * @param pDevIns The device instance.
2944 * @thread any
2945 */
2946void e1kPhyLinkResetCallback(PPDMDEVINS pDevIns)
2947{
2948 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
2949
2950 /* Make sure we have cable connected and MAC can talk to PHY */
2951 if (e1kIsConnected(pDevIns) && (CTRL & CTRL_SLU))
2952 e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, E1K_INIT_LINKUP_DELAY_US);
2953 else
2954 Log(("%s PHY link reset callback ignored (cable %sconnected, driver %stached, CTRL_SLU=%u)\n", pThis->szPrf,
2955 pThis->fCableConnected ? "" : "dis", pThis->fIsAttached ? "at" : "de", CTRL & CTRL_SLU ? 1 : 0));
2956}
2957
2958/**
2959 * Write handler for Device Control register.
2960 *
2961 * Handles reset.
2962 *
2963 * @param pThis The device state structure.
2964 * @param offset Register offset in memory-mapped frame.
2965 * @param index Register index in register array.
2966 * @param value The value to store.
2967 * @param mask Used to implement partial writes (8 and 16-bit).
2968 * @thread EMT
2969 */
2970static int e1kRegWriteCTRL(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
2971{
2972 int rc = VINF_SUCCESS;
2973
2974 if (value & CTRL_RESET)
2975 { /* RST */
2976#ifndef IN_RING3
2977 return VINF_IOM_R3_MMIO_WRITE;
2978#else
2979 e1kR3HardReset(pDevIns, pThis, PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC));
2980#endif
2981 }
2982 else
2983 {
2984#ifdef E1K_LSC_ON_SLU
2985 /*
2986 * When the guest changes 'Set Link Up' bit from 0 to 1 we check if
2987 * the link is down and the cable is connected, and if they are we
2988 * bring the link up, see @bugref{8624}.
2989 */
2990 if ( (value & CTRL_SLU)
2991 && !(CTRL & CTRL_SLU)
2992 && pThis->fCableConnected
2993 && !(STATUS & STATUS_LU))
2994 {
2995 /* It should take about 2 seconds for the link to come up */
2996 e1kArmTimer(pDevIns, pThis, pThis->hLUTimer, E1K_INIT_LINKUP_DELAY_US);
2997 }
2998#else /* !E1K_LSC_ON_SLU */
2999 if ( (value & CTRL_SLU)
3000 && !(CTRL & CTRL_SLU)
3001 && e1kIsConnected(pDevIns)
3002 && !PDMDevHlpTimerIsActive(pDevIns, pThis->hLUTimer))
3003 {
3004 /* PXE does not use LSC interrupts, see @bugref{9113}. */
3005 STATUS |= STATUS_LU;
3006 }
3007#endif /* !E1K_LSC_ON_SLU */
3008 if ((value & CTRL_VME) != (CTRL & CTRL_VME))
3009 {
3010 E1kLog(("%s VLAN Mode %s\n", pThis->szPrf, (value & CTRL_VME) ? "Enabled" : "Disabled"));
3011 }
3012 Log7(("%s e1kRegWriteCTRL: mdio dir=%s mdc dir=%s mdc=%s mdio=%d\n",
3013 pThis->szPrf, (value & CTRL_MDIO_DIR)?"OUT":"IN ",
3014 (value & CTRL_MDC_DIR)?"OUT":"IN ", (value & CTRL_MDC)?"HIGH":"LOW ", !!(value & CTRL_MDIO)));
3015 if (value & CTRL_MDC)
3016 {
3017 if (value & CTRL_MDIO_DIR)
3018 {
3019 Log7(("%s e1kRegWriteCTRL: Phy::writeMDIO(%d)\n", pThis->szPrf, !!(value & CTRL_MDIO)));
3020 /* MDIO direction pin is set to output and MDC is high, write MDIO pin value to PHY */
3021 Phy::writeMDIO(&pThis->phy, !!(value & CTRL_MDIO), pDevIns);
3022 }
3023 else
3024 {
3025 if (Phy::readMDIO(&pThis->phy))
3026 value |= CTRL_MDIO;
3027 else
3028 value &= ~CTRL_MDIO;
3029 Log7(("%s e1kRegWriteCTRL: Phy::readMDIO(%d)\n", pThis->szPrf, !!(value & CTRL_MDIO)));
3030 }
3031 }
3032 rc = e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3033 }
3034
3035 return rc;
3036}
3037
3038/**
3039 * Write handler for EEPROM/Flash Control/Data register.
3040 *
3041 * Handles EEPROM access requests; forwards writes to EEPROM device if access has been granted.
3042 *
3043 * @param pThis The device state structure.
3044 * @param offset Register offset in memory-mapped frame.
3045 * @param index Register index in register array.
3046 * @param value The value to store.
3047 * @param mask Used to implement partial writes (8 and 16-bit).
3048 * @thread EMT
3049 */
3050static int e1kRegWriteEECD(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3051{
3052 RT_NOREF(pDevIns, offset, index);
3053#ifdef IN_RING3
3054 /* So far we are concerned with lower byte only */
3055 if ((EECD & EECD_EE_GNT) || pThis->eChip == E1K_CHIP_82543GC)
3056 {
3057 /* Access to EEPROM granted -- forward 4-wire bits to EEPROM device */
3058 /* Note: 82543GC does not need to request EEPROM access */
3059 STAM_PROFILE_ADV_START(&pThis->StatEEPROMWrite, a);
3060 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
3061 pThisCC->eeprom.write(value & EECD_EE_WIRES);
3062 STAM_PROFILE_ADV_STOP(&pThis->StatEEPROMWrite, a);
3063 }
3064 if (value & EECD_EE_REQ)
3065 EECD |= EECD_EE_REQ|EECD_EE_GNT;
3066 else
3067 EECD &= ~EECD_EE_GNT;
3068 //e1kRegWriteDefault(pThis, offset, index, value );
3069
3070 return VINF_SUCCESS;
3071#else /* !IN_RING3 */
3072 RT_NOREF(pThis, value);
3073 return VINF_IOM_R3_MMIO_WRITE;
3074#endif /* !IN_RING3 */
3075}
3076
3077/**
3078 * Read handler for EEPROM/Flash Control/Data register.
3079 *
3080 * Lower 4 bits come from EEPROM device if EEPROM access has been granted.
3081 *
3082 * @returns VBox status code.
3083 *
3084 * @param pThis The device state structure.
3085 * @param offset Register offset in memory-mapped frame.
3086 * @param index Register index in register array.
3087 * @param mask Used to implement partial reads (8 and 16-bit).
3088 * @thread EMT
3089 */
3090static int e1kRegReadEECD(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
3091{
3092#ifdef IN_RING3
3093 uint32_t value = 0; /* Get rid of false positive in parfait. */
3094 int rc = e1kRegReadDefault(pDevIns, pThis, offset, index, &value);
3095 if (RT_SUCCESS(rc))
3096 {
3097 if ((value & EECD_EE_GNT) || pThis->eChip == E1K_CHIP_82543GC)
3098 {
3099 /* Note: 82543GC does not need to request EEPROM access */
3100 /* Access to EEPROM granted -- get 4-wire bits to EEPROM device */
3101 STAM_PROFILE_ADV_START(&pThis->StatEEPROMRead, a);
3102 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
3103 value |= pThisCC->eeprom.read();
3104 STAM_PROFILE_ADV_STOP(&pThis->StatEEPROMRead, a);
3105 }
3106 *pu32Value = value;
3107 }
3108
3109 return rc;
3110#else /* !IN_RING3 */
3111 RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index); RT_NOREF_PV(pu32Value);
3112 return VINF_IOM_R3_MMIO_READ;
3113#endif /* !IN_RING3 */
3114}
3115
3116/**
3117 * Write handler for EEPROM Read register.
3118 *
3119 * Handles EEPROM word access requests, reads EEPROM and stores the result
3120 * into DATA field.
3121 *
3122 * @param pThis The device state structure.
3123 * @param offset Register offset in memory-mapped frame.
3124 * @param index Register index in register array.
3125 * @param value The value to store.
3126 * @param mask Used to implement partial writes (8 and 16-bit).
3127 * @thread EMT
3128 */
3129static int e1kRegWriteEERD(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3130{
3131#ifdef IN_RING3
3132 /* Make use of 'writable' and 'readable' masks. */
3133 e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3134 /* DONE and DATA are set only if read was triggered by START. */
3135 if (value & EERD_START)
3136 {
3137 STAM_PROFILE_ADV_START(&pThis->StatEEPROMRead, a);
3138 uint16_t tmp;
3139 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
3140 if (pThisCC->eeprom.readWord(GET_BITS_V(value, EERD, ADDR), &tmp))
3141 SET_BITS(EERD, DATA, tmp);
3142 EERD |= EERD_DONE;
3143 STAM_PROFILE_ADV_STOP(&pThis->StatEEPROMRead, a);
3144 }
3145
3146 return VINF_SUCCESS;
3147#else /* !IN_RING3 */
3148 RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index); RT_NOREF_PV(value);
3149 return VINF_IOM_R3_MMIO_WRITE;
3150#endif /* !IN_RING3 */
3151}
3152
3153
3154/**
3155 * Write handler for MDI Control register.
3156 *
3157 * Handles PHY read/write requests; forwards requests to internal PHY device.
3158 *
3159 * @param pThis The device state structure.
3160 * @param offset Register offset in memory-mapped frame.
3161 * @param index Register index in register array.
3162 * @param value The value to store.
3163 * @param mask Used to implement partial writes (8 and 16-bit).
3164 * @thread EMT
3165 */
3166static int e1kRegWriteMDIC(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3167{
3168 if (value & MDIC_INT_EN)
3169 {
3170 E1kLog(("%s ERROR! Interrupt at the end of an MDI cycle is not supported yet.\n",
3171 pThis->szPrf));
3172 }
3173 else if (value & MDIC_READY)
3174 {
3175 E1kLog(("%s ERROR! Ready bit is not reset by software during write operation.\n",
3176 pThis->szPrf));
3177 }
3178 else if (GET_BITS_V(value, MDIC, PHY) != 1)
3179 {
3180 E1kLog(("%s WARNING! Access to invalid PHY detected, phy=%d.\n",
3181 pThis->szPrf, GET_BITS_V(value, MDIC, PHY)));
3182 /*
3183 * Some drivers scan the MDIO bus for a PHY. We can work with these
3184 * drivers if we set MDIC_READY and MDIC_ERROR when there isn't a PHY
3185 * at the requested address, see @bugref{7346}.
3186 */
3187 MDIC = MDIC_READY | MDIC_ERROR;
3188 }
3189 else
3190 {
3191 /* Store the value */
3192 e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3193 STAM_COUNTER_INC(&pThis->StatPHYAccesses);
3194 /* Forward op to PHY */
3195 if (value & MDIC_OP_READ)
3196 SET_BITS(MDIC, DATA, Phy::readRegister(&pThis->phy, GET_BITS_V(value, MDIC, REG), pDevIns));
3197 else
3198 Phy::writeRegister(&pThis->phy, GET_BITS_V(value, MDIC, REG), value & MDIC_DATA_MASK, pDevIns);
3199 /* Let software know that we are done */
3200 MDIC |= MDIC_READY;
3201 }
3202
3203 return VINF_SUCCESS;
3204}
3205
3206/**
3207 * Write handler for Interrupt Cause Read register.
3208 *
3209 * Bits corresponding to 1s in 'value' will be cleared in ICR register.
3210 *
3211 * @param pThis The device state structure.
3212 * @param offset Register offset in memory-mapped frame.
3213 * @param index Register index in register array.
3214 * @param value The value to store.
3215 * @param mask Used to implement partial writes (8 and 16-bit).
3216 * @thread EMT
3217 */
3218static int e1kRegWriteICR(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3219{
3220 ICR &= ~value;
3221
3222 RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index);
3223 return VINF_SUCCESS;
3224}
3225
3226/**
3227 * Read handler for Interrupt Cause Read register.
3228 *
3229 * Reading this register acknowledges all interrupts.
3230 *
3231 * @returns VBox status code.
3232 *
3233 * @param pThis The device state structure.
3234 * @param offset Register offset in memory-mapped frame.
3235 * @param index Register index in register array.
3236 * @param mask Not used.
3237 * @thread EMT
3238 */
3239static int e1kRegReadICR(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
3240{
3241 e1kCsEnterReturn(pThis, VINF_IOM_R3_MMIO_READ);
3242
3243 uint32_t value = 0;
3244 int rc = e1kRegReadDefault(pDevIns, pThis, offset, index, &value);
3245 if (RT_SUCCESS(rc))
3246 {
3247 if (value)
3248 {
3249 if (!pThis->fIntRaised)
3250 E1K_INC_ISTAT_CNT(pThis->uStatNoIntICR);
3251 /*
3252 * Not clearing ICR causes QNX to hang as it reads ICR in a loop
3253 * with disabled interrupts.
3254 */
3255 //if (IMS)
3256 if (1)
3257 {
3258 /*
3259 * Interrupts were enabled -- we are supposedly at the very
3260 * beginning of interrupt handler
3261 */
3262 E1kLogRel(("E1000: irq lowered, icr=0x%x\n", ICR));
3263 E1kLog(("%s e1kRegReadICR: Lowered IRQ (%08x)\n", pThis->szPrf, ICR));
3264 /* Clear all pending interrupts */
3265 ICR = 0;
3266 pThis->fIntRaised = false;
3267 /* Lower(0) INTA(0) */
3268 PDMDevHlpPCISetIrq(pDevIns, 0, 0);
3269
3270 pThis->u64AckedAt = PDMDevHlpTimerGet(pDevIns, pThis->hIntTimer);
3271 if (pThis->fIntMaskUsed)
3272 pThis->fDelayInts = true;
3273 }
3274 else
3275 {
3276 /*
3277 * Interrupts are disabled -- in windows guests ICR read is done
3278 * just before re-enabling interrupts
3279 */
3280 E1kLog(("%s e1kRegReadICR: Suppressing auto-clear due to disabled interrupts (%08x)\n", pThis->szPrf, ICR));
3281 }
3282 }
3283 *pu32Value = value;
3284 }
3285 e1kCsLeave(pThis);
3286
3287 return rc;
3288}
3289
3290/**
3291 * Read handler for Interrupt Cause Set register.
3292 *
3293 * VxWorks driver uses this undocumented feature of real H/W to read ICR without acknowledging interrupts.
3294 *
3295 * @returns VBox status code.
3296 *
3297 * @param pThis The device state structure.
3298 * @param offset Register offset in memory-mapped frame.
3299 * @param index Register index in register array.
3300 * @param pu32Value Where to store the value of the register.
3301 * @thread EMT
3302 */
3303static int e1kRegReadICS(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
3304{
3305 RT_NOREF_PV(index);
3306 return e1kRegReadDefault(pDevIns, pThis, offset, ICR_IDX, pu32Value);
3307}
3308
3309/**
3310 * Write handler for Interrupt Cause Set register.
3311 *
3312 * Bits corresponding to 1s in 'value' will be set in ICR register.
3313 *
3314 * @param pThis The device state structure.
3315 * @param offset Register offset in memory-mapped frame.
3316 * @param index Register index in register array.
3317 * @param value The value to store.
3318 * @param mask Used to implement partial writes (8 and 16-bit).
3319 * @thread EMT
3320 */
3321static int e1kRegWriteICS(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3322{
3323 RT_NOREF_PV(offset); RT_NOREF_PV(index);
3324 E1K_INC_ISTAT_CNT(pThis->uStatIntICS);
3325 return e1kRaiseInterrupt(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE, value & g_aE1kRegMap[ICS_IDX].writable);
3326}
3327
3328/**
3329 * Write handler for Interrupt Mask Set register.
3330 *
3331 * Will trigger pending interrupts.
3332 *
3333 * @param pThis The device state structure.
3334 * @param offset Register offset in memory-mapped frame.
3335 * @param index Register index in register array.
3336 * @param value The value to store.
3337 * @param mask Used to implement partial writes (8 and 16-bit).
3338 * @thread EMT
3339 */
3340static int e1kRegWriteIMS(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3341{
3342 RT_NOREF_PV(offset); RT_NOREF_PV(index);
3343
3344 IMS |= value;
3345 E1kLogRel(("E1000: irq enabled, RDH=%x RDT=%x TDH=%x TDT=%x\n", RDH, RDT, TDH, TDT));
3346 E1kLog(("%s e1kRegWriteIMS: IRQ enabled\n", pThis->szPrf));
3347 /*
3348 * We cannot raise an interrupt here as it will occasionally cause an interrupt storm
3349 * in Windows guests (see @bugref{8624}, @bugref{5023}).
3350 */
3351 if ((ICR & IMS) && !pThis->fLocked)
3352 {
3353 E1K_INC_ISTAT_CNT(pThis->uStatIntIMS);
3354 e1kPostponeInterrupt(pDevIns, pThis, E1K_IMS_INT_DELAY_NS);
3355 }
3356
3357 return VINF_SUCCESS;
3358}
3359
3360/**
3361 * Write handler for Interrupt Mask Clear register.
3362 *
3363 * Bits corresponding to 1s in 'value' will be cleared in IMS register.
3364 *
3365 * @param pThis The device state structure.
3366 * @param offset Register offset in memory-mapped frame.
3367 * @param index Register index in register array.
3368 * @param value The value to store.
3369 * @param mask Used to implement partial writes (8 and 16-bit).
3370 * @thread EMT
3371 */
3372static int e1kRegWriteIMC(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3373{
3374 RT_NOREF_PV(offset); RT_NOREF_PV(index);
3375
3376 e1kCsEnterReturn(pThis, VINF_IOM_R3_MMIO_WRITE);
3377 if (pThis->fIntRaised)
3378 {
3379 /*
3380 * Technically we should reset fIntRaised in ICR read handler, but it will cause
3381 * Windows to freeze since it may receive an interrupt while still in the very beginning
3382 * of interrupt handler.
3383 */
3384 E1K_INC_ISTAT_CNT(pThis->uStatIntLower);
3385 STAM_COUNTER_INC(&pThis->StatIntsPrevented);
3386 E1kLogRel(("E1000: irq lowered (IMC), icr=0x%x\n", ICR));
3387 /* Lower(0) INTA(0) */
3388 PDMDevHlpPCISetIrq(pDevIns, 0, 0);
3389 pThis->fIntRaised = false;
3390 E1kLog(("%s e1kRegWriteIMC: Lowered IRQ: ICR=%08x\n", pThis->szPrf, ICR));
3391 }
3392 IMS &= ~value;
3393 E1kLog(("%s e1kRegWriteIMC: IRQ disabled\n", pThis->szPrf));
3394 e1kCsLeave(pThis);
3395
3396 return VINF_SUCCESS;
3397}
3398
3399/**
3400 * Write handler for Receive Control register.
3401 *
3402 * @param pThis The device state structure.
3403 * @param offset Register offset in memory-mapped frame.
3404 * @param index Register index in register array.
3405 * @param value The value to store.
3406 * @param mask Used to implement partial writes (8 and 16-bit).
3407 * @thread EMT
3408 */
3409static int e1kRegWriteRCTL(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3410{
3411 /* Update promiscuous mode */
3412 bool fBecomePromiscous = !!(value & (RCTL_UPE | RCTL_MPE));
3413 if (fBecomePromiscous != !!( RCTL & (RCTL_UPE | RCTL_MPE)))
3414 {
3415 /* Promiscuity has changed, pass the knowledge on. */
3416#ifndef IN_RING3
3417 return VINF_IOM_R3_MMIO_WRITE;
3418#else
3419 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
3420 if (pThisCC->pDrvR3)
3421 pThisCC->pDrvR3->pfnSetPromiscuousMode(pThisCC->pDrvR3, fBecomePromiscous);
3422#endif
3423 }
3424
3425 /* Adjust receive buffer size */
3426 unsigned cbRxBuf = 2048 >> GET_BITS_V(value, RCTL, BSIZE);
3427 if (value & RCTL_BSEX)
3428 cbRxBuf *= 16;
3429 if (cbRxBuf > E1K_MAX_RX_PKT_SIZE)
3430 cbRxBuf = E1K_MAX_RX_PKT_SIZE;
3431 if (cbRxBuf != pThis->u16RxBSize)
3432 E1kLog2(("%s e1kRegWriteRCTL: Setting receive buffer size to %d (old %d)\n",
3433 pThis->szPrf, cbRxBuf, pThis->u16RxBSize));
3434 Assert(cbRxBuf < 65536);
3435 pThis->u16RxBSize = (uint16_t)cbRxBuf;
3436
3437 /* Update the register */
3438 return e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3439}
3440
3441/**
3442 * Write handler for Packet Buffer Allocation register.
3443 *
3444 * TXA = 64 - RXA.
3445 *
3446 * @param pThis The device state structure.
3447 * @param offset Register offset in memory-mapped frame.
3448 * @param index Register index in register array.
3449 * @param value The value to store.
3450 * @param mask Used to implement partial writes (8 and 16-bit).
3451 * @thread EMT
3452 */
3453static int e1kRegWritePBA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3454{
3455 e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3456 PBA_st->txa = 64 - PBA_st->rxa;
3457
3458 return VINF_SUCCESS;
3459}
3460
3461/**
3462 * Write handler for Receive Descriptor Tail register.
3463 *
3464 * @remarks Write into RDT forces switch to HC and signal to
3465 * e1kR3NetworkDown_WaitReceiveAvail().
3466 *
3467 * @returns VBox status code.
3468 *
3469 * @param pThis The device state structure.
3470 * @param offset Register offset in memory-mapped frame.
3471 * @param index Register index in register array.
3472 * @param value The value to store.
3473 * @param mask Used to implement partial writes (8 and 16-bit).
3474 * @thread EMT
3475 */
3476static int e1kRegWriteRDT(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3477{
3478#ifndef IN_RING3
3479 /* XXX */
3480// return VINF_IOM_R3_MMIO_WRITE;
3481#endif
3482 int rc = e1kCsRxEnter(pThis, VINF_IOM_R3_MMIO_WRITE);
3483 if (RT_LIKELY(rc == VINF_SUCCESS))
3484 {
3485 E1kLog(("%s e1kRegWriteRDT\n", pThis->szPrf));
3486#ifndef E1K_WITH_RXD_CACHE
3487 /*
3488 * Some drivers advance RDT too far, so that it equals RDH. This
3489 * somehow manages to work with real hardware but not with this
3490 * emulated device. We can work with these drivers if we just
3491 * write 1 less when we see a driver writing RDT equal to RDH,
3492 * see @bugref{7346}.
3493 */
3494 if (value == RDH)
3495 {
3496 if (RDH == 0)
3497 value = (RDLEN / sizeof(E1KRXDESC)) - 1;
3498 else
3499 value = RDH - 1;
3500 }
3501#endif /* !E1K_WITH_RXD_CACHE */
3502 rc = e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3503#ifdef E1K_WITH_RXD_CACHE
3504 E1KRXDC rxdc;
3505 if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kRegWriteRDT")))
3506 {
3507 e1kCsRxLeave(pThis);
3508 E1kLog(("%s e1kRegWriteRDT: failed to update Rx context, returning VINF_SUCCESS\n", pThis->szPrf));
3509 return VINF_SUCCESS;
3510 }
3511 /*
3512 * We need to fetch descriptors now as RDT may go whole circle
3513 * before we attempt to store a received packet. For example,
3514 * Intel's DOS drivers use 2 (!) RX descriptors with the total ring
3515 * size being only 8 descriptors! Note that we fetch descriptors
3516 * only when the cache is empty to reduce the number of memory reads
3517 * in case of frequent RDT writes. Don't fetch anything when the
3518 * receiver is disabled either as RDH, RDT, RDLEN can be in some
3519 * messed up state.
3520 * Note that despite the cache may seem empty, meaning that there are
3521 * no more available descriptors in it, it may still be used by RX
3522 * thread which has not yet written the last descriptor back but has
3523 * temporarily released the RX lock in order to write the packet body
3524 * to descriptor's buffer. At this point we still going to do prefetch
3525 * but it won't actually fetch anything if there are no unused slots in
3526 * our "empty" cache (nRxDFetched==E1K_RXD_CACHE_SIZE). We must not
3527 * reset the cache here even if it appears empty. It will be reset at
3528 * a later point in e1kRxDGet().
3529 */
3530 if (e1kRxDIsCacheEmpty(pThis) && (RCTL & RCTL_EN))
3531 e1kRxDPrefetch(pDevIns, pThis, &rxdc);
3532#endif /* E1K_WITH_RXD_CACHE */
3533 e1kCsRxLeave(pThis);
3534 if (RT_SUCCESS(rc))
3535 {
3536 /* Signal that we have more receive descriptors available. */
3537 e1kWakeupReceive(pDevIns, pThis);
3538 }
3539 }
3540 return rc;
3541}
3542
3543/**
3544 * Write handler for Receive Delay Timer register.
3545 *
3546 * @param pThis The device state structure.
3547 * @param offset Register offset in memory-mapped frame.
3548 * @param index Register index in register array.
3549 * @param value The value to store.
3550 * @param mask Used to implement partial writes (8 and 16-bit).
3551 * @thread EMT
3552 */
3553static int e1kRegWriteRDTR(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
3554{
3555 e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
3556 if (value & RDTR_FPD)
3557 {
3558 /* Flush requested, cancel both timers and raise interrupt */
3559#ifdef E1K_USE_RX_TIMERS
3560 e1kCancelTimer(pDevIns, pThis, pThis->hRIDTimer);
3561 e1kCancelTimer(pDevIns, pThis, pThis->hRADTimer);
3562#endif
3563 E1K_INC_ISTAT_CNT(pThis->uStatIntRDTR);
3564 return e1kRaiseInterrupt(pDevIns, pThis, VINF_IOM_R3_MMIO_WRITE, ICR_RXT0);
3565 }
3566
3567 return VINF_SUCCESS;
3568}
3569
3570DECLINLINE(uint32_t) e1kGetTxLen(PE1KTXDC pTxdc)
3571{
3572 /**
3573 * Make sure TDT won't change during computation. EMT may modify TDT at
3574 * any moment.
3575 */
3576 uint32_t tdt = pTxdc->tdt;
3577 return (pTxdc->tdh > tdt ? pTxdc->tdlen/sizeof(E1KTXDESC) : 0) + tdt - pTxdc->tdh;
3578}
3579
3580#ifdef IN_RING3
3581
3582# ifdef E1K_TX_DELAY
3583/**
3584 * @callback_method_impl{FNTMTIMERDEV, Transmit Delay Timer handler.}
3585 */
3586static DECLCALLBACK(void) e1kR3TxDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3587{
3588 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3589 Assert(PDMDevHlpCritSectIsOwner(pDevIns, &pThis->csTx));
3590 RT_NOREF(hTimer);
3591
3592 E1K_INC_ISTAT_CNT(pThis->uStatTxDelayExp);
3593# ifdef E1K_INT_STATS
3594 uint64_t u64Elapsed = RTTimeNanoTS() - pThis->u64ArmedAt;
3595 if (u64Elapsed > pThis->uStatMaxTxDelay)
3596 pThis->uStatMaxTxDelay = u64Elapsed;
3597# endif
3598 int rc = e1kXmitPending(pDevIns, pThis, false /*fOnWorkerThread*/);
3599 AssertMsg(RT_SUCCESS(rc) || rc == VERR_TRY_AGAIN, ("%Rrc\n", rc));
3600}
3601# endif /* E1K_TX_DELAY */
3602
3603//# ifdef E1K_USE_TX_TIMERS
3604
3605/**
3606 * @callback_method_impl{FNTMTIMERDEV, Transmit Interrupt Delay Timer handler.}
3607 */
3608static DECLCALLBACK(void) e1kR3TxIntDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3609{
3610 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3611 Assert(hTimer == pThis->hTIDTimer); RT_NOREF(hTimer);
3612
3613 E1K_INC_ISTAT_CNT(pThis->uStatTID);
3614 /* Cancel absolute delay timer as we have already got attention */
3615# ifndef E1K_NO_TAD
3616 e1kCancelTimer(pDevIns, pThis, pThis->hTADTimer);
3617# endif
3618 e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_TXDW);
3619}
3620
3621/**
3622 * @callback_method_impl{FNTMTIMERDEV, Transmit Absolute Delay Timer handler.}
3623 */
3624static DECLCALLBACK(void) e1kR3TxAbsDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3625{
3626 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3627 Assert(hTimer == pThis->hTADTimer); RT_NOREF(hTimer);
3628
3629 E1K_INC_ISTAT_CNT(pThis->uStatTAD);
3630 /* Cancel interrupt delay timer as we have already got attention */
3631 e1kCancelTimer(pDevIns, pThis, pThis->hTIDTimer);
3632 e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_TXDW);
3633}
3634
3635//# endif /* E1K_USE_TX_TIMERS */
3636# ifdef E1K_USE_RX_TIMERS
3637
3638/**
3639 * @callback_method_impl{FNTMTIMERDEV, Receive Interrupt Delay Timer handler.}
3640 */
3641static DECLCALLBACK(void) e1kR3RxIntDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3642{
3643 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3644 Assert(hTimer == pThis->hRIDTimer); RT_NOREF(hTimer);
3645
3646 E1K_INC_ISTAT_CNT(pThis->uStatRID);
3647 /* Cancel absolute delay timer as we have already got attention */
3648 e1kCancelTimer(pDevIns, pThis, pThis->hRADTimer);
3649 e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_RXT0);
3650}
3651
3652/**
3653 * @callback_method_impl{FNTMTIMERDEV, Receive Absolute Delay Timer handler.}
3654 */
3655static DECLCALLBACK(void) e1kR3RxAbsDelayTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3656{
3657 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3658 Assert(hTimer == pThis->hRADTimer); RT_NOREF(hTimer);
3659
3660 E1K_INC_ISTAT_CNT(pThis->uStatRAD);
3661 /* Cancel interrupt delay timer as we have already got attention */
3662 e1kCancelTimer(pDevIns, pThis, pThis->hRIDTimer);
3663 e1kRaiseInterrupt(pDevIns, pThis, VERR_IGNORED, ICR_RXT0);
3664}
3665
3666# endif /* E1K_USE_RX_TIMERS */
3667
3668/**
3669 * @callback_method_impl{FNTMTIMERDEV, Late Interrupt Timer handler.}
3670 */
3671static DECLCALLBACK(void) e1kR3LateIntTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3672{
3673 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3674 Assert(hTimer == pThis->hIntTimer); RT_NOREF(hTimer);
3675 RT_NOREF(hTimer);
3676
3677 STAM_PROFILE_ADV_START(&pThis->StatLateIntTimer, a);
3678 STAM_COUNTER_INC(&pThis->StatLateInts);
3679 E1K_INC_ISTAT_CNT(pThis->uStatIntLate);
3680# if 0
3681 if (pThis->iStatIntLost > -100)
3682 pThis->iStatIntLost--;
3683# endif
3684 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, 0);
3685 STAM_PROFILE_ADV_STOP(&pThis->StatLateIntTimer, a);
3686}
3687
3688/**
3689 * @callback_method_impl{FNTMTIMERDEV, Link Up Timer handler.}
3690 */
3691static DECLCALLBACK(void) e1kR3LinkUpTimer(PPDMDEVINS pDevIns, TMTIMERHANDLE hTimer, void *pvUser)
3692{
3693 PE1KSTATE pThis = (PE1KSTATE)pvUser;
3694 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
3695 Assert(hTimer == pThis->hLUTimer); RT_NOREF(hTimer);
3696
3697 /*
3698 * This can happen if we set the link status to down when the Link up timer was
3699 * already armed (shortly after e1kR3LoadDone() or when the cable was disconnected
3700 * and connect+disconnect the cable very quick. Moreover, 82543GC triggers LSC
3701 * on reset even if the cable is unplugged (see @bugref{8942}).
3702 */
3703 if (e1kIsConnected(pDevIns))
3704 {
3705 /* 82543GC does not have an internal PHY */
3706 if (pThis->eChip == E1K_CHIP_82543GC || (CTRL & CTRL_SLU))
3707 e1kR3LinkUp(pDevIns, pThis, pThisCC);
3708 }
3709# ifdef E1K_LSC_ON_RESET
3710 else if (pThis->eChip == E1K_CHIP_82543GC)
3711 e1kR3LinkDown(pDevIns, pThis, pThisCC);
3712# endif /* E1K_LSC_ON_RESET */
3713}
3714
3715#endif /* IN_RING3 */
3716
3717/**
3718 * Sets up the GSO context according to the TSE new context descriptor.
3719 *
3720 * @param pGso The GSO context to setup.
3721 * @param pCtx The context descriptor.
3722 */
3723DECLINLINE(bool) e1kSetupGsoCtx(PPDMNETWORKGSO pGso, E1KTXCTX const *pCtx)
3724{
3725 pGso->u8Type = PDMNETWORKGSOTYPE_INVALID;
3726
3727 /*
3728 * See if the context descriptor describes something that could be TCP or
3729 * UDP over IPv[46].
3730 */
3731 /* Check the header ordering and spacing: 1. Ethernet, 2. IP, 3. TCP/UDP. */
3732 if (RT_UNLIKELY( pCtx->ip.u8CSS < sizeof(RTNETETHERHDR) ))
3733 {
3734 E1kLog(("e1kSetupGsoCtx: IPCSS=%#x\n", pCtx->ip.u8CSS));
3735 return false;
3736 }
3737 if (RT_UNLIKELY( pCtx->tu.u8CSS < (size_t)pCtx->ip.u8CSS + (pCtx->dw2.fIP ? RTNETIPV4_MIN_LEN : RTNETIPV6_MIN_LEN) ))
3738 {
3739 E1kLog(("e1kSetupGsoCtx: TUCSS=%#x\n", pCtx->tu.u8CSS));
3740 return false;
3741 }
3742 if (RT_UNLIKELY( pCtx->dw2.fTCP
3743 ? pCtx->dw3.u8HDRLEN < (size_t)pCtx->tu.u8CSS + RTNETTCP_MIN_LEN
3744 : pCtx->dw3.u8HDRLEN != (size_t)pCtx->tu.u8CSS + RTNETUDP_MIN_LEN ))
3745 {
3746 E1kLog(("e1kSetupGsoCtx: HDRLEN=%#x TCP=%d\n", pCtx->dw3.u8HDRLEN, pCtx->dw2.fTCP));
3747 return false;
3748 }
3749
3750 /* The end of the TCP/UDP checksum should stop at the end of the packet or at least after the headers. */
3751 if (RT_UNLIKELY( pCtx->tu.u16CSE > 0 && pCtx->tu.u16CSE <= pCtx->dw3.u8HDRLEN ))
3752 {
3753 E1kLog(("e1kSetupGsoCtx: TUCSE=%#x HDRLEN=%#x\n", pCtx->tu.u16CSE, pCtx->dw3.u8HDRLEN));
3754 return false;
3755 }
3756
3757 /* IPv4 checksum offset. */
3758 if (RT_UNLIKELY( pCtx->dw2.fIP && (size_t)pCtx->ip.u8CSO - pCtx->ip.u8CSS != RT_UOFFSETOF(RTNETIPV4, ip_sum) ))
3759 {
3760 E1kLog(("e1kSetupGsoCtx: IPCSO=%#x IPCSS=%#x\n", pCtx->ip.u8CSO, pCtx->ip.u8CSS));
3761 return false;
3762 }
3763
3764 /* TCP/UDP checksum offsets. */
3765 if (RT_UNLIKELY( (size_t)pCtx->tu.u8CSO - pCtx->tu.u8CSS
3766 != ( pCtx->dw2.fTCP
3767 ? RT_UOFFSETOF(RTNETTCP, th_sum)
3768 : RT_UOFFSETOF(RTNETUDP, uh_sum) ) ))
3769 {
3770 E1kLog(("e1kSetupGsoCtx: TUCSO=%#x TUCSS=%#x TCP=%d\n", pCtx->ip.u8CSO, pCtx->ip.u8CSS, pCtx->dw2.fTCP));
3771 return false;
3772 }
3773
3774 /*
3775 * Because of internal networking using a 16-bit size field for GSO context
3776 * plus frame, we have to make sure we don't exceed this.
3777 */
3778 if (RT_UNLIKELY( pCtx->dw3.u8HDRLEN + pCtx->dw2.u20PAYLEN > VBOX_MAX_GSO_SIZE ))
3779 {
3780 E1kLog(("e1kSetupGsoCtx: HDRLEN(=%#x) + PAYLEN(=%#x) = %#x, max is %#x\n",
3781 pCtx->dw3.u8HDRLEN, pCtx->dw2.u20PAYLEN, pCtx->dw3.u8HDRLEN + pCtx->dw2.u20PAYLEN, VBOX_MAX_GSO_SIZE));
3782 return false;
3783 }
3784
3785 /*
3786 * We're good for now - we'll do more checks when seeing the data.
3787 * So, figure the type of offloading and setup the context.
3788 */
3789 if (pCtx->dw2.fIP)
3790 {
3791 if (pCtx->dw2.fTCP)
3792 {
3793 pGso->u8Type = PDMNETWORKGSOTYPE_IPV4_TCP;
3794 pGso->cbHdrsSeg = pCtx->dw3.u8HDRLEN;
3795 }
3796 else
3797 {
3798 pGso->u8Type = PDMNETWORKGSOTYPE_IPV4_UDP;
3799 pGso->cbHdrsSeg = pCtx->tu.u8CSS; /* IP header only */
3800 }
3801 /** @todo Detect IPv4-IPv6 tunneling (need test setup since linux doesn't do
3802 * this yet it seems)... */
3803 }
3804 else
3805 {
3806 pGso->cbHdrsSeg = pCtx->dw3.u8HDRLEN; /** @todo IPv6 UFO */
3807 if (pCtx->dw2.fTCP)
3808 pGso->u8Type = PDMNETWORKGSOTYPE_IPV6_TCP;
3809 else
3810 pGso->u8Type = PDMNETWORKGSOTYPE_IPV6_UDP;
3811 }
3812 pGso->offHdr1 = pCtx->ip.u8CSS;
3813 pGso->offHdr2 = pCtx->tu.u8CSS;
3814 pGso->cbHdrsTotal = pCtx->dw3.u8HDRLEN;
3815 pGso->cbMaxSeg = pCtx->dw3.u16MSS + (pGso->u8Type == PDMNETWORKGSOTYPE_IPV4_UDP ? pGso->offHdr2 : 0);
3816 Assert(PDMNetGsoIsValid(pGso, sizeof(*pGso), pGso->cbMaxSeg * 5));
3817 E1kLog2(("e1kSetupGsoCtx: mss=%#x hdr=%#x hdrseg=%#x hdr1=%#x hdr2=%#x %s\n",
3818 pGso->cbMaxSeg, pGso->cbHdrsTotal, pGso->cbHdrsSeg, pGso->offHdr1, pGso->offHdr2, PDMNetGsoTypeName((PDMNETWORKGSOTYPE)pGso->u8Type) ));
3819 return PDMNetGsoIsValid(pGso, sizeof(*pGso), pGso->cbMaxSeg * 5);
3820}
3821
3822/**
3823 * Checks if we can use GSO processing for the current TSE frame.
3824 *
3825 * @param pThis The device state structure.
3826 * @param pGso The GSO context.
3827 * @param pData The first data descriptor of the frame.
3828 * @param pCtx The TSO context descriptor.
3829 */
3830DECLINLINE(bool) e1kCanDoGso(PE1KSTATE pThis, PCPDMNETWORKGSO pGso, E1KTXDAT const *pData, E1KTXCTX const *pCtx)
3831{
3832 if (!pData->cmd.fTSE)
3833 {
3834 E1kLog2(("e1kCanDoGso: !TSE\n"));
3835 return false;
3836 }
3837 if (pData->cmd.fVLE) /** @todo VLAN tagging. */
3838 {
3839 E1kLog(("e1kCanDoGso: VLE\n"));
3840 return false;
3841 }
3842 if (RT_UNLIKELY(!pThis->fGSOEnabled))
3843 {
3844 E1kLog3(("e1kCanDoGso: GSO disabled via CFGM\n"));
3845 return false;
3846 }
3847
3848 switch ((PDMNETWORKGSOTYPE)pGso->u8Type)
3849 {
3850 case PDMNETWORKGSOTYPE_IPV4_TCP:
3851 case PDMNETWORKGSOTYPE_IPV4_UDP:
3852 if (!pData->dw3.fIXSM)
3853 {
3854 E1kLog(("e1kCanDoGso: !IXSM (IPv4)\n"));
3855 return false;
3856 }
3857 if (!pData->dw3.fTXSM)
3858 {
3859 E1kLog(("e1kCanDoGso: !TXSM (IPv4)\n"));
3860 return false;
3861 }
3862 /** @todo what more check should we perform here? Ethernet frame type? */
3863 E1kLog2(("e1kCanDoGso: OK, IPv4\n"));
3864 return true;
3865
3866 case PDMNETWORKGSOTYPE_IPV6_TCP:
3867 case PDMNETWORKGSOTYPE_IPV6_UDP:
3868 if (pData->dw3.fIXSM && pCtx->ip.u8CSO)
3869 {
3870 E1kLog(("e1kCanDoGso: IXSM (IPv6)\n"));
3871 return false;
3872 }
3873 if (!pData->dw3.fTXSM)
3874 {
3875 E1kLog(("e1kCanDoGso: TXSM (IPv6)\n"));
3876 return false;
3877 }
3878 /** @todo what more check should we perform here? Ethernet frame type? */
3879 E1kLog2(("e1kCanDoGso: OK, IPv4\n"));
3880 return true;
3881
3882 default:
3883 Assert(pGso->u8Type == PDMNETWORKGSOTYPE_INVALID);
3884 E1kLog2(("e1kCanDoGso: e1kSetupGsoCtx failed\n"));
3885 return false;
3886 }
3887}
3888
3889/**
3890 * Frees the current xmit buffer.
3891 *
3892 * @param pThis The device state structure.
3893 */
3894static void e1kXmitFreeBuf(PE1KSTATE pThis, PE1KSTATECC pThisCC)
3895{
3896 PPDMSCATTERGATHER pSg = pThisCC->CTX_SUFF(pTxSg);
3897 if (pSg)
3898 {
3899 pThisCC->CTX_SUFF(pTxSg) = NULL;
3900
3901 if (pSg->pvAllocator != pThis)
3902 {
3903 PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
3904 if (pDrv)
3905 pDrv->pfnFreeBuf(pDrv, pSg);
3906 }
3907 else
3908 {
3909 /* loopback */
3910 AssertCompileMemberSize(E1KSTATE, uTxFallback.Sg, 8 * sizeof(size_t));
3911 Assert(pSg->fFlags == (PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_3));
3912 pSg->fFlags = 0;
3913 pSg->pvAllocator = NULL;
3914 }
3915 }
3916}
3917
3918#ifndef E1K_WITH_TXD_CACHE
3919/**
3920 * Allocates an xmit buffer.
3921 *
3922 * @returns See PDMINETWORKUP::pfnAllocBuf.
3923 * @param pThis The device state structure.
3924 * @param cbMin The minimum frame size.
3925 * @param fExactSize Whether cbMin is exact or if we have to max it
3926 * out to the max MTU size.
3927 * @param fGso Whether this is a GSO frame or not.
3928 */
3929DECLINLINE(int) e1kXmitAllocBuf(PE1KSTATE pThis, PE1KSTATECC pThisCC, size_t cbMin, bool fExactSize, bool fGso)
3930{
3931 /* Adjust cbMin if necessary. */
3932 if (!fExactSize)
3933 cbMin = RT_MAX(cbMin, E1K_MAX_TX_PKT_SIZE);
3934
3935 /* Deal with existing buffer (descriptor screw up, reset, etc). */
3936 if (RT_UNLIKELY(pThisCC->CTX_SUFF(pTxSg)))
3937 e1kXmitFreeBuf(pThis, pThisCC);
3938 Assert(pThisCC->CTX_SUFF(pTxSg) == NULL);
3939
3940 /*
3941 * Allocate the buffer.
3942 */
3943 PPDMSCATTERGATHER pSg;
3944 if (RT_LIKELY(GET_BITS(RCTL, LBM) != RCTL_LBM_TCVR))
3945 {
3946 PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
3947 if (RT_UNLIKELY(!pDrv))
3948 return VERR_NET_DOWN;
3949 int rc = pDrv->pfnAllocBuf(pDrv, cbMin, fGso ? &pThis->GsoCtx : NULL, &pSg);
3950 if (RT_FAILURE(rc))
3951 {
3952 /* Suspend TX as we are out of buffers atm */
3953 STATUS |= STATUS_TXOFF;
3954 return rc;
3955 }
3956 }
3957 else
3958 {
3959 /* Create a loopback using the fallback buffer and preallocated SG. */
3960 AssertCompileMemberSize(E1KSTATE, uTxFallback.Sg, 8 * sizeof(size_t));
3961 pSg = &pThis->uTxFallback.Sg;
3962 pSg->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_3;
3963 pSg->cbUsed = 0;
3964 pSg->cbAvailable = 0;
3965 pSg->pvAllocator = pThis;
3966 pSg->pvUser = NULL; /* No GSO here. */
3967 pSg->cSegs = 1;
3968 pSg->aSegs[0].pvSeg = pThis->aTxPacketFallback;
3969 pSg->aSegs[0].cbSeg = sizeof(pThis->aTxPacketFallback);
3970 }
3971
3972 pThisCC->CTX_SUFF(pTxSg) = pSg;
3973 return VINF_SUCCESS;
3974}
3975#else /* E1K_WITH_TXD_CACHE */
3976/**
3977 * Allocates an xmit buffer.
3978 *
3979 * @returns See PDMINETWORKUP::pfnAllocBuf.
3980 * @param pThis The device state structure.
3981 * @param cbMin The minimum frame size.
3982 * @param fExactSize Whether cbMin is exact or if we have to max it
3983 * out to the max MTU size.
3984 * @param fGso Whether this is a GSO frame or not.
3985 */
3986DECLINLINE(int) e1kXmitAllocBuf(PE1KSTATE pThis, PE1KSTATECC pThisCC, bool fGso)
3987{
3988 /* Deal with existing buffer (descriptor screw up, reset, etc). */
3989 if (RT_UNLIKELY(pThisCC->CTX_SUFF(pTxSg)))
3990 e1kXmitFreeBuf(pThis, pThisCC);
3991 Assert(pThisCC->CTX_SUFF(pTxSg) == NULL);
3992
3993 /*
3994 * Allocate the buffer.
3995 */
3996 PPDMSCATTERGATHER pSg;
3997 if (RT_LIKELY(GET_BITS(RCTL, LBM) != RCTL_LBM_TCVR))
3998 {
3999 if (pThis->cbTxAlloc == 0)
4000 {
4001 /* Zero packet, no need for the buffer */
4002 return VINF_SUCCESS;
4003 }
4004 if (fGso && pThis->GsoCtx.u8Type == PDMNETWORKGSOTYPE_INVALID)
4005 {
4006 E1kLog3(("Invalid GSO context, won't allocate this packet, cb=%u %s%s\n",
4007 pThis->cbTxAlloc, pThis->fVTag ? "VLAN " : "", pThis->fGSO ? "GSO " : ""));
4008 /* No valid GSO context is available, ignore this packet. */
4009 pThis->cbTxAlloc = 0;
4010 return VINF_SUCCESS;
4011 }
4012
4013 PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
4014 if (RT_UNLIKELY(!pDrv))
4015 return VERR_NET_DOWN;
4016 int rc = pDrv->pfnAllocBuf(pDrv, pThis->cbTxAlloc, fGso ? &pThis->GsoCtx : NULL, &pSg);
4017 if (RT_FAILURE(rc))
4018 {
4019 /* Suspend TX as we are out of buffers atm */
4020 STATUS |= STATUS_TXOFF;
4021 return rc;
4022 }
4023 E1kLog3(("%s Allocated buffer for TX packet: cb=%u %s%s\n",
4024 pThis->szPrf, pThis->cbTxAlloc,
4025 pThis->fVTag ? "VLAN " : "",
4026 pThis->fGSO ? "GSO " : ""));
4027 }
4028 else
4029 {
4030 /* Create a loopback using the fallback buffer and preallocated SG. */
4031 AssertCompileMemberSize(E1KSTATE, uTxFallback.Sg, 8 * sizeof(size_t));
4032 pSg = &pThis->uTxFallback.Sg;
4033 pSg->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_3;
4034 pSg->cbUsed = 0;
4035 pSg->cbAvailable = sizeof(pThis->aTxPacketFallback);
4036 pSg->pvAllocator = pThis;
4037 pSg->pvUser = NULL; /* No GSO here. */
4038 pSg->cSegs = 1;
4039 pSg->aSegs[0].pvSeg = pThis->aTxPacketFallback;
4040 pSg->aSegs[0].cbSeg = sizeof(pThis->aTxPacketFallback);
4041 }
4042 pThis->cbTxAlloc = 0;
4043
4044 pThisCC->CTX_SUFF(pTxSg) = pSg;
4045 return VINF_SUCCESS;
4046}
4047#endif /* E1K_WITH_TXD_CACHE */
4048
4049/**
4050 * Checks if it's a GSO buffer or not.
4051 *
4052 * @returns true / false.
4053 * @param pTxSg The scatter / gather buffer.
4054 */
4055DECLINLINE(bool) e1kXmitIsGsoBuf(PDMSCATTERGATHER const *pTxSg)
4056{
4057#if 0
4058 if (!pTxSg)
4059 E1kLog(("e1kXmitIsGsoBuf: pTxSG is NULL\n"));
4060 if (pTxSg && pTxSg->pvUser)
4061 E1kLog(("e1kXmitIsGsoBuf: pvUser is NULL\n"));
4062#endif
4063 return pTxSg && pTxSg->pvUser /* GSO indicator */;
4064}
4065
4066#ifndef E1K_WITH_TXD_CACHE
4067/**
4068 * Load transmit descriptor from guest memory.
4069 *
4070 * @param pDevIns The device instance.
4071 * @param pDesc Pointer to descriptor union.
4072 * @param addr Physical address in guest context.
4073 * @thread E1000_TX
4074 */
4075DECLINLINE(void) e1kLoadDesc(PPDMDEVINS pDevIns, E1KTXDESC *pDesc, RTGCPHYS addr)
4076{
4077 PDMDevHlpPCIPhysRead(pDevIns, addr, pDesc, sizeof(E1KTXDESC));
4078}
4079#else /* E1K_WITH_TXD_CACHE */
4080/**
4081 * Load transmit descriptors from guest memory.
4082 *
4083 * We need two physical reads in case the tail wrapped around the end of TX
4084 * descriptor ring.
4085 *
4086 * @returns the actual number of descriptors fetched.
4087 * @param pDevIns The device instance.
4088 * @param pThis The device state structure.
4089 * @thread E1000_TX
4090 */
4091DECLINLINE(unsigned) e1kTxDLoadMore(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pTxdc)
4092{
4093 Assert(pThis->iTxDCurrent == 0);
4094 /* We've already loaded pThis->nTxDFetched descriptors past TDH. */
4095 unsigned nDescsAvailable = e1kGetTxLen(pTxdc) - pThis->nTxDFetched;
4096 /* The following two lines ensure that pThis->nTxDFetched never overflows. */
4097 AssertCompile(E1K_TXD_CACHE_SIZE < (256 * sizeof(pThis->nTxDFetched)));
4098 unsigned nDescsToFetch = RT_MIN(nDescsAvailable, E1K_TXD_CACHE_SIZE - pThis->nTxDFetched);
4099 unsigned nDescsTotal = pTxdc->tdlen / sizeof(E1KTXDESC);
4100 Assert(nDescsTotal != 0);
4101 if (nDescsTotal == 0)
4102 return 0;
4103 unsigned nFirstNotLoaded = (pTxdc->tdh + pThis->nTxDFetched) % nDescsTotal;
4104 unsigned nDescsInSingleRead = RT_MIN(nDescsToFetch, nDescsTotal - nFirstNotLoaded);
4105 E1kLog3(("%s e1kTxDLoadMore: nDescsAvailable=%u nDescsToFetch=%u nDescsTotal=%u nFirstNotLoaded=0x%x nDescsInSingleRead=%u\n",
4106 pThis->szPrf, nDescsAvailable, nDescsToFetch, nDescsTotal,
4107 nFirstNotLoaded, nDescsInSingleRead));
4108 if (nDescsToFetch == 0)
4109 return 0;
4110 E1KTXDESC* pFirstEmptyDesc = &pThis->aTxDescriptors[pThis->nTxDFetched];
4111 PDMDevHlpPCIPhysRead(pDevIns,
4112 ((uint64_t)TDBAH << 32) + TDBAL + nFirstNotLoaded * sizeof(E1KTXDESC),
4113 pFirstEmptyDesc, nDescsInSingleRead * sizeof(E1KTXDESC));
4114 E1kLog3(("%s Fetched %u TX descriptors at %08x%08x(0x%x), TDLEN=%08x, TDH=%08x, TDT=%08x\n",
4115 pThis->szPrf, nDescsInSingleRead,
4116 TDBAH, TDBAL + pTxdc->tdh * sizeof(E1KTXDESC),
4117 nFirstNotLoaded, pTxdc->tdlen, pTxdc->tdh, pTxdc->tdt));
4118 if (nDescsToFetch > nDescsInSingleRead)
4119 {
4120 PDMDevHlpPCIPhysRead(pDevIns,
4121 ((uint64_t)TDBAH << 32) + TDBAL,
4122 pFirstEmptyDesc + nDescsInSingleRead,
4123 (nDescsToFetch - nDescsInSingleRead) * sizeof(E1KTXDESC));
4124 E1kLog3(("%s Fetched %u TX descriptors at %08x%08x\n",
4125 pThis->szPrf, nDescsToFetch - nDescsInSingleRead,
4126 TDBAH, TDBAL));
4127 }
4128 pThis->nTxDFetched += (uint8_t)nDescsToFetch;
4129 return nDescsToFetch;
4130}
4131
4132/**
4133 * Load transmit descriptors from guest memory only if there are no loaded
4134 * descriptors.
4135 *
4136 * @returns true if there are descriptors in cache.
4137 * @param pDevIns The device instance.
4138 * @param pThis The device state structure.
4139 * @thread E1000_TX
4140 */
4141DECLINLINE(bool) e1kTxDLazyLoad(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pTxdc)
4142{
4143 if (pThis->nTxDFetched == 0)
4144 return e1kTxDLoadMore(pDevIns, pThis, pTxdc) != 0;
4145 return true;
4146}
4147#endif /* E1K_WITH_TXD_CACHE */
4148
4149/**
4150 * Write back transmit descriptor to guest memory.
4151 *
4152 * @param pDevIns The device instance.
4153 * @param pThis The device state structure.
4154 * @param pDesc Pointer to descriptor union.
4155 * @param addr Physical address in guest context.
4156 * @thread E1000_TX
4157 */
4158DECLINLINE(void) e1kWriteBackDesc(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KTXDESC *pDesc, RTGCPHYS addr)
4159{
4160 /* Only the last half of the descriptor has to be written back. */
4161 e1kPrintTDesc(pThis, pDesc, "^^^");
4162 PDMDevHlpPCIPhysWrite(pDevIns, addr, pDesc, sizeof(E1KTXDESC));
4163}
4164
4165/**
4166 * Transmit complete frame.
4167 *
4168 * @remarks We skip the FCS since we're not responsible for sending anything to
4169 * a real ethernet wire.
4170 *
4171 * @param pDevIns The device instance.
4172 * @param pThis The device state structure.
4173 * @param pThisCC The current context instance data.
4174 * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
4175 * @thread E1000_TX
4176 */
4177static void e1kTransmitFrame(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, bool fOnWorkerThread)
4178{
4179 PPDMSCATTERGATHER pSg = pThisCC->CTX_SUFF(pTxSg);
4180 uint32_t cbFrame = pSg ? (uint32_t)pSg->cbUsed : 0;
4181 Assert(!pSg || pSg->cSegs == 1);
4182
4183 if (cbFrame < 14)
4184 {
4185 Log(("%s Ignoring invalid frame (%u bytes)\n", pThis->szPrf, cbFrame));
4186 return;
4187 }
4188 if (cbFrame > 70) /* unqualified guess */
4189 pThis->led.Asserted.s.fWriting = pThis->led.Actual.s.fWriting = 1;
4190
4191#ifdef E1K_INT_STATS
4192 if (cbFrame <= 1514)
4193 E1K_INC_ISTAT_CNT(pThis->uStatTx1514);
4194 else if (cbFrame <= 2962)
4195 E1K_INC_ISTAT_CNT(pThis->uStatTx2962);
4196 else if (cbFrame <= 4410)
4197 E1K_INC_ISTAT_CNT(pThis->uStatTx4410);
4198 else if (cbFrame <= 5858)
4199 E1K_INC_ISTAT_CNT(pThis->uStatTx5858);
4200 else if (cbFrame <= 7306)
4201 E1K_INC_ISTAT_CNT(pThis->uStatTx7306);
4202 else if (cbFrame <= 8754)
4203 E1K_INC_ISTAT_CNT(pThis->uStatTx8754);
4204 else if (cbFrame <= 16384)
4205 E1K_INC_ISTAT_CNT(pThis->uStatTx16384);
4206 else if (cbFrame <= 32768)
4207 E1K_INC_ISTAT_CNT(pThis->uStatTx32768);
4208 else
4209 E1K_INC_ISTAT_CNT(pThis->uStatTxLarge);
4210#endif /* E1K_INT_STATS */
4211
4212 /* Add VLAN tag */
4213 if (cbFrame > 12 && pThis->fVTag && pSg->cbUsed + 4 <= pSg->cbAvailable)
4214 {
4215 E1kLog3(("%s Inserting VLAN tag %08x\n",
4216 pThis->szPrf, RT_BE2H_U16((uint16_t)VET) | (RT_BE2H_U16(pThis->u16VTagTCI) << 16)));
4217 memmove((uint8_t*)pSg->aSegs[0].pvSeg + 16, (uint8_t*)pSg->aSegs[0].pvSeg + 12, cbFrame - 12);
4218 *((uint32_t*)pSg->aSegs[0].pvSeg + 3) = RT_BE2H_U16((uint16_t)VET) | (RT_BE2H_U16(pThis->u16VTagTCI) << 16);
4219 pSg->cbUsed += 4;
4220 cbFrame += 4;
4221 Assert(pSg->cbUsed == cbFrame);
4222 Assert(pSg->cbUsed <= pSg->cbAvailable);
4223 }
4224/* E1kLog2(("%s < < < Outgoing packet. Dump follows: > > >\n"
4225 "%.*Rhxd\n"
4226 "%s < < < < < < < < < < < < < End of dump > > > > > > > > > > > >\n",
4227 pThis->szPrf, cbFrame, pSg->aSegs[0].pvSeg, pThis->szPrf));*/
4228
4229 /* Update the stats */
4230 E1K_INC_CNT32(TPT);
4231 E1K_ADD_CNT64(TOTL, TOTH, cbFrame);
4232 E1K_INC_CNT32(GPTC);
4233 if (pSg && e1kIsBroadcast(pSg->aSegs[0].pvSeg))
4234 E1K_INC_CNT32(BPTC);
4235 else if (pSg && e1kIsMulticast(pSg->aSegs[0].pvSeg))
4236 E1K_INC_CNT32(MPTC);
4237 /* Update octet transmit counter */
4238 E1K_ADD_CNT64(GOTCL, GOTCH, cbFrame);
4239 if (pThisCC->CTX_SUFF(pDrv))
4240 STAM_REL_COUNTER_ADD(&pThis->StatTransmitBytes, cbFrame);
4241 if (cbFrame == 64)
4242 E1K_INC_CNT32(PTC64);
4243 else if (cbFrame < 128)
4244 E1K_INC_CNT32(PTC127);
4245 else if (cbFrame < 256)
4246 E1K_INC_CNT32(PTC255);
4247 else if (cbFrame < 512)
4248 E1K_INC_CNT32(PTC511);
4249 else if (cbFrame < 1024)
4250 E1K_INC_CNT32(PTC1023);
4251 else
4252 E1K_INC_CNT32(PTC1522);
4253
4254 E1K_INC_ISTAT_CNT(pThis->uStatTxFrm);
4255
4256 /*
4257 * Dump and send the packet.
4258 */
4259 int rc = VERR_NET_DOWN;
4260 if (pSg && pSg->pvAllocator != pThis)
4261 {
4262 e1kPacketDump(pDevIns, pThis, (uint8_t const *)pSg->aSegs[0].pvSeg, cbFrame, "--> Outgoing");
4263
4264 pThisCC->CTX_SUFF(pTxSg) = NULL;
4265 PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
4266 if (pDrv)
4267 {
4268 /* Release critical section to avoid deadlock in CanReceive */
4269 //e1kCsLeave(pThis);
4270 STAM_PROFILE_START(&pThis->CTX_SUFF_Z(StatTransmitSend), a);
4271 rc = pDrv->pfnSendBuf(pDrv, pSg, fOnWorkerThread);
4272 STAM_PROFILE_STOP(&pThis->CTX_SUFF_Z(StatTransmitSend), a);
4273 //e1kR3CsEnterAsserted(pThis);
4274 }
4275 }
4276 else if (pSg)
4277 {
4278 Assert(pSg->aSegs[0].pvSeg == pThis->aTxPacketFallback);
4279 e1kPacketDump(pDevIns, pThis, (uint8_t const *)pSg->aSegs[0].pvSeg, cbFrame, "--> Loopback");
4280
4281 /** @todo do we actually need to check that we're in loopback mode here? */
4282 if (GET_BITS(RCTL, LBM) == RCTL_LBM_TCVR)
4283 {
4284 E1KRXDST status;
4285 RT_ZERO(status);
4286 status.fPIF = true;
4287 e1kHandleRxPacket(pDevIns, pThis, pSg->aSegs[0].pvSeg, cbFrame, status);
4288 rc = VINF_SUCCESS;
4289 }
4290 e1kXmitFreeBuf(pThis, pThisCC);
4291 }
4292 else
4293 rc = VERR_NET_DOWN;
4294 if (RT_FAILURE(rc))
4295 {
4296 E1kLogRel(("E1000: ERROR! pfnSend returned %Rrc\n", rc));
4297 /** @todo handle VERR_NET_DOWN and VERR_NET_NO_BUFFER_SPACE. Signal error ? */
4298 }
4299
4300 pThis->led.Actual.s.fWriting = 0;
4301}
4302
4303/**
4304 * Compute and write internet checksum (e1kCSum16) at the specified offset.
4305 *
4306 * @param pThis The device state structure.
4307 * @param pPkt Pointer to the packet.
4308 * @param u16PktLen Total length of the packet.
4309 * @param cso Offset in packet to write checksum at.
4310 * @param css Offset in packet to start computing
4311 * checksum from.
4312 * @param cse Offset in packet to stop computing
4313 * checksum at.
4314 * @param fUdp Replace 0 checksum with all 1s.
4315 * @thread E1000_TX
4316 */
4317static void e1kInsertChecksum(PE1KSTATE pThis, uint8_t *pPkt, uint16_t u16PktLen, uint8_t cso, uint8_t css, uint16_t cse, bool fUdp = false)
4318{
4319 RT_NOREF1(pThis);
4320
4321 if (css >= u16PktLen)
4322 {
4323 E1kLog2(("%s css(%X) is greater than packet length-1(%X), checksum is not inserted\n",
4324 pThis->szPrf, cso, u16PktLen));
4325 return;
4326 }
4327
4328 if (cso >= u16PktLen - 1)
4329 {
4330 E1kLog2(("%s cso(%X) is greater than packet length-2(%X), checksum is not inserted\n",
4331 pThis->szPrf, cso, u16PktLen));
4332 return;
4333 }
4334
4335 if (cse == 0 || cse >= u16PktLen)
4336 cse = u16PktLen - 1;
4337 else if (cse < css)
4338 {
4339 E1kLog2(("%s css(%X) is greater than cse(%X), checksum is not inserted\n",
4340 pThis->szPrf, css, cse));
4341 return;
4342 }
4343
4344 uint16_t u16ChkSum = e1kCSum16(pPkt + css, cse - css + 1);
4345 if (fUdp && u16ChkSum == 0)
4346 u16ChkSum = ~u16ChkSum; /* 0 means no checksum computed in case of UDP (see @bugref{9883}) */
4347 E1kLog2(("%s Inserting csum: %04X at %02X, old value: %04X\n", pThis->szPrf,
4348 u16ChkSum, cso, *(uint16_t*)(pPkt + cso)));
4349 *(uint16_t*)(pPkt + cso) = u16ChkSum;
4350}
4351
4352/**
4353 * Add a part of descriptor's buffer to transmit frame.
4354 *
4355 * @remarks data.u64BufAddr is used unconditionally for both data
4356 * and legacy descriptors since it is identical to
4357 * legacy.u64BufAddr.
4358 *
4359 * @param pDevIns The device instance.
4360 * @param pThis The device state structure.
4361 * @param pDesc Pointer to the descriptor to transmit.
4362 * @param u16Len Length of buffer to the end of segment.
4363 * @param fSend Force packet sending.
4364 * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
4365 * @thread E1000_TX
4366 */
4367#ifndef E1K_WITH_TXD_CACHE
4368static void e1kFallbackAddSegment(PPDMDEVINS pDevIns, PE1KSTATE pThis, RTGCPHYS PhysAddr, uint16_t u16Len, bool fSend, bool fOnWorkerThread)
4369{
4370 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
4371 /* TCP header being transmitted */
4372 struct E1kTcpHeader *pTcpHdr = (struct E1kTcpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.tu.u8CSS);
4373 /* IP header being transmitted */
4374 struct E1kIpHeader *pIpHdr = (struct E1kIpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.ip.u8CSS);
4375
4376 E1kLog3(("%s e1kFallbackAddSegment: Length=%x, remaining payload=%x, header=%x, send=%RTbool\n",
4377 pThis->szPrf, u16Len, pThis->u32PayRemain, pThis->u16HdrRemain, fSend));
4378 Assert(pThis->u32PayRemain + pThis->u16HdrRemain > 0);
4379
4380 PDMDevHlpPCIPhysRead(pDevIns, PhysAddr, pThis->aTxPacketFallback + pThis->u16TxPktLen, u16Len);
4381 E1kLog3(("%s Dump of the segment:\n"
4382 "%.*Rhxd\n"
4383 "%s --- End of dump ---\n",
4384 pThis->szPrf, u16Len, pThis->aTxPacketFallback + pThis->u16TxPktLen, pThis->szPrf));
4385 pThis->u16TxPktLen += u16Len;
4386 E1kLog3(("%s e1kFallbackAddSegment: pThis->u16TxPktLen=%x\n",
4387 pThis->szPrf, pThis->u16TxPktLen));
4388 if (pThis->u16HdrRemain > 0)
4389 {
4390 /* The header was not complete, check if it is now */
4391 if (u16Len >= pThis->u16HdrRemain)
4392 {
4393 /* The rest is payload */
4394 u16Len -= pThis->u16HdrRemain;
4395 pThis->u16HdrRemain = 0;
4396 /* Save partial checksum and flags */
4397 pThis->u32SavedCsum = pTcpHdr->chksum;
4398 pThis->u16SavedFlags = pTcpHdr->hdrlen_flags;
4399 /* Clear FIN and PSH flags now and set them only in the last segment */
4400 pTcpHdr->hdrlen_flags &= ~htons(E1K_TCP_FIN | E1K_TCP_PSH);
4401 }
4402 else
4403 {
4404 /* Still not */
4405 pThis->u16HdrRemain -= u16Len;
4406 E1kLog3(("%s e1kFallbackAddSegment: Header is still incomplete, 0x%x bytes remain.\n",
4407 pThis->szPrf, pThis->u16HdrRemain));
4408 return;
4409 }
4410 }
4411
4412 pThis->u32PayRemain -= u16Len;
4413
4414 if (fSend)
4415 {
4416 /* Leave ethernet header intact */
4417 /* IP Total Length = payload + headers - ethernet header */
4418 pIpHdr->total_len = htons(pThis->u16TxPktLen - pThis->contextTSE.ip.u8CSS);
4419 E1kLog3(("%s e1kFallbackAddSegment: End of packet, pIpHdr->total_len=%x\n",
4420 pThis->szPrf, ntohs(pIpHdr->total_len)));
4421 /* Update IP Checksum */
4422 pIpHdr->chksum = 0;
4423 e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
4424 pThis->contextTSE.ip.u8CSO,
4425 pThis->contextTSE.ip.u8CSS,
4426 pThis->contextTSE.ip.u16CSE);
4427
4428 /* Update TCP flags */
4429 /* Restore original FIN and PSH flags for the last segment */
4430 if (pThis->u32PayRemain == 0)
4431 {
4432 pTcpHdr->hdrlen_flags = pThis->u16SavedFlags;
4433 E1K_INC_CNT32(TSCTC);
4434 }
4435 /* Add TCP length to partial pseudo header sum */
4436 uint32_t csum = pThis->u32SavedCsum
4437 + htons(pThis->u16TxPktLen - pThis->contextTSE.tu.u8CSS);
4438 while (csum >> 16)
4439 csum = (csum >> 16) + (csum & 0xFFFF);
4440 pTcpHdr->chksum = csum;
4441 /* Compute final checksum */
4442 e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
4443 pThis->contextTSE.tu.u8CSO,
4444 pThis->contextTSE.tu.u8CSS,
4445 pThis->contextTSE.tu.u16CSE);
4446
4447 /*
4448 * Transmit it. If we've use the SG already, allocate a new one before
4449 * we copy of the data.
4450 */
4451 PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
4452 if (!pTxSg)
4453 {
4454 e1kXmitAllocBuf(pThis, pThisCC, pThis->u16TxPktLen + (pThis->fVTag ? 4 : 0), true /*fExactSize*/, false /*fGso*/);
4455 pTxSg = pThisCC->CTX_SUFF(pTxSg);
4456 }
4457 if (pTxSg)
4458 {
4459 Assert(pThis->u16TxPktLen <= pThisCC->CTX_SUFF(pTxSg)->cbAvailable);
4460 Assert(pTxSg->cSegs == 1);
4461 if (pThis->CCCTX_SUFF(pTxSg)->aSegs[0].pvSeg != pThis->aTxPacketFallback)
4462 memcpy(pTxSg->aSegs[0].pvSeg, pThis->aTxPacketFallback, pThis->u16TxPktLen);
4463 pTxSg->cbUsed = pThis->u16TxPktLen;
4464 pTxSg->aSegs[0].cbSeg = pThis->u16TxPktLen;
4465 }
4466 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
4467
4468 /* Update Sequence Number */
4469 pTcpHdr->seqno = htonl(ntohl(pTcpHdr->seqno) + pThis->u16TxPktLen
4470 - pThis->contextTSE.dw3.u8HDRLEN);
4471 /* Increment IP identification */
4472 pIpHdr->ident = htons(ntohs(pIpHdr->ident) + 1);
4473 }
4474}
4475#else /* E1K_WITH_TXD_CACHE */
4476static int e1kFallbackAddSegment(PPDMDEVINS pDevIns, PE1KSTATE pThis, RTGCPHYS PhysAddr, uint16_t u16Len, bool fSend, bool fOnWorkerThread)
4477{
4478 int rc = VINF_SUCCESS;
4479 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
4480 /* TCP header being transmitted */
4481 struct E1kTcpHeader *pTcpHdr = (struct E1kTcpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.tu.u8CSS);
4482 /* IP header being transmitted */
4483 struct E1kIpHeader *pIpHdr = (struct E1kIpHeader *)(pThis->aTxPacketFallback + pThis->contextTSE.ip.u8CSS);
4484
4485 E1kLog3(("%s e1kFallbackAddSegment: Length=%x, remaining payload=%x, header=%x, send=%RTbool\n",
4486 pThis->szPrf, u16Len, pThis->u32PayRemain, pThis->u16HdrRemain, fSend));
4487 AssertReturn(pThis->u32PayRemain + pThis->u16HdrRemain > 0, VINF_SUCCESS);
4488
4489 if (pThis->u16TxPktLen + u16Len <= sizeof(pThis->aTxPacketFallback))
4490 PDMDevHlpPCIPhysRead(pDevIns, PhysAddr, pThis->aTxPacketFallback + pThis->u16TxPktLen, u16Len);
4491 else
4492 E1kLog(("%s e1kFallbackAddSegment: writing beyond aTxPacketFallback, u16TxPktLen=%d(0x%x) + u16Len=%d(0x%x) > %d\n",
4493 pThis->szPrf, pThis->u16TxPktLen, pThis->u16TxPktLen, u16Len, u16Len, sizeof(pThis->aTxPacketFallback)));
4494 E1kLog3(("%s Dump of the segment:\n"
4495 "%.*Rhxd\n"
4496 "%s --- End of dump ---\n",
4497 pThis->szPrf, u16Len, pThis->aTxPacketFallback + pThis->u16TxPktLen, pThis->szPrf));
4498 pThis->u16TxPktLen += u16Len;
4499 E1kLog3(("%s e1kFallbackAddSegment: pThis->u16TxPktLen=%x\n",
4500 pThis->szPrf, pThis->u16TxPktLen));
4501 if (pThis->u16HdrRemain > 0)
4502 {
4503 /* The header was not complete, check if it is now */
4504 if (u16Len >= pThis->u16HdrRemain)
4505 {
4506 /* The rest is payload */
4507 u16Len -= pThis->u16HdrRemain;
4508 pThis->u16HdrRemain = 0;
4509 /* Save partial checksum and flags */
4510 pThis->u32SavedCsum = pTcpHdr->chksum;
4511 pThis->u16SavedFlags = pTcpHdr->hdrlen_flags;
4512 /* Clear FIN and PSH flags now and set them only in the last segment */
4513 pTcpHdr->hdrlen_flags &= ~htons(E1K_TCP_FIN | E1K_TCP_PSH);
4514 }
4515 else
4516 {
4517 /* Still not */
4518 pThis->u16HdrRemain -= u16Len;
4519 E1kLog3(("%s e1kFallbackAddSegment: Header is still incomplete, 0x%x bytes remain.\n",
4520 pThis->szPrf, pThis->u16HdrRemain));
4521 return rc;
4522 }
4523 }
4524
4525 if (u16Len > pThis->u32PayRemain)
4526 pThis->u32PayRemain = 0;
4527 else
4528 pThis->u32PayRemain -= u16Len;
4529
4530 if (fSend)
4531 {
4532 /* Leave ethernet header intact */
4533 /* IP Total Length = payload + headers - ethernet header */
4534 pIpHdr->total_len = htons(pThis->u16TxPktLen - pThis->contextTSE.ip.u8CSS);
4535 E1kLog3(("%s e1kFallbackAddSegment: End of packet, pIpHdr->total_len=%x\n",
4536 pThis->szPrf, ntohs(pIpHdr->total_len)));
4537 /* Update IP Checksum */
4538 pIpHdr->chksum = 0;
4539 e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
4540 pThis->contextTSE.ip.u8CSO,
4541 pThis->contextTSE.ip.u8CSS,
4542 pThis->contextTSE.ip.u16CSE);
4543
4544 /* Update TCP flags */
4545 /* Restore original FIN and PSH flags for the last segment */
4546 if (pThis->u32PayRemain == 0)
4547 {
4548 pTcpHdr->hdrlen_flags = pThis->u16SavedFlags;
4549 E1K_INC_CNT32(TSCTC);
4550 }
4551 /* Add TCP length to partial pseudo header sum */
4552 uint32_t csum = pThis->u32SavedCsum
4553 + htons(pThis->u16TxPktLen - pThis->contextTSE.tu.u8CSS);
4554 while (csum >> 16)
4555 csum = (csum >> 16) + (csum & 0xFFFF);
4556 Assert(csum < 65536);
4557 pTcpHdr->chksum = (uint16_t)csum;
4558 /* Compute final checksum */
4559 e1kInsertChecksum(pThis, pThis->aTxPacketFallback, pThis->u16TxPktLen,
4560 pThis->contextTSE.tu.u8CSO,
4561 pThis->contextTSE.tu.u8CSS,
4562 pThis->contextTSE.tu.u16CSE);
4563
4564 /*
4565 * Transmit it.
4566 */
4567 PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
4568 if (pTxSg)
4569 {
4570 /* Make sure the packet fits into the allocated buffer */
4571 size_t cbCopy = RT_MIN(pThis->u16TxPktLen, pThisCC->CTX_SUFF(pTxSg)->cbAvailable);
4572#ifdef DEBUG
4573 if (pThis->u16TxPktLen > pTxSg->cbAvailable)
4574 E1kLog(("%s e1kFallbackAddSegment: truncating packet, u16TxPktLen=%d(0x%x) > cbAvailable=%d(0x%x)\n",
4575 pThis->szPrf, pThis->u16TxPktLen, pThis->u16TxPktLen, pTxSg->cbAvailable, pTxSg->cbAvailable));
4576#endif /* DEBUG */
4577 Assert(pTxSg->cSegs == 1);
4578 if (pTxSg->aSegs[0].pvSeg != pThis->aTxPacketFallback)
4579 memcpy(pTxSg->aSegs[0].pvSeg, pThis->aTxPacketFallback, cbCopy);
4580 pTxSg->cbUsed = cbCopy;
4581 pTxSg->aSegs[0].cbSeg = cbCopy;
4582 }
4583 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
4584
4585 /* Update Sequence Number */
4586 pTcpHdr->seqno = htonl(ntohl(pTcpHdr->seqno) + pThis->u16TxPktLen
4587 - pThis->contextTSE.dw3.u8HDRLEN);
4588 /* Increment IP identification */
4589 pIpHdr->ident = htons(ntohs(pIpHdr->ident) + 1);
4590
4591 /* Allocate new buffer for the next segment. */
4592 if (pThis->u32PayRemain)
4593 {
4594 pThis->cbTxAlloc = RT_MIN(pThis->u32PayRemain,
4595 pThis->contextTSE.dw3.u16MSS)
4596 + pThis->contextTSE.dw3.u8HDRLEN;
4597 /* Do not add VLAN tags to empty packets. */
4598 if (pThis->fVTag && pThis->cbTxAlloc > 0)
4599 pThis->cbTxAlloc += 4;
4600 rc = e1kXmitAllocBuf(pThis, pThisCC, false /* fGSO */);
4601 }
4602 }
4603
4604 return rc;
4605}
4606#endif /* E1K_WITH_TXD_CACHE */
4607
4608#ifndef E1K_WITH_TXD_CACHE
4609/**
4610 * TCP segmentation offloading fallback: Add descriptor's buffer to transmit
4611 * frame.
4612 *
4613 * We construct the frame in the fallback buffer first and the copy it to the SG
4614 * buffer before passing it down to the network driver code.
4615 *
4616 * @returns true if the frame should be transmitted, false if not.
4617 *
4618 * @param pThis The device state structure.
4619 * @param pDesc Pointer to the descriptor to transmit.
4620 * @param cbFragment Length of descriptor's buffer.
4621 * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
4622 * @thread E1000_TX
4623 */
4624static bool e1kFallbackAddToFrame(PE1KSTATE pThis, E1KTXDESC *pDesc, uint32_t cbFragment, bool fOnWorkerThread)
4625{
4626 PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
4627 Assert(e1kGetDescType(pDesc) == E1K_DTYP_DATA);
4628 Assert(pDesc->data.cmd.fTSE);
4629 Assert(!e1kXmitIsGsoBuf(pTxSg));
4630
4631 uint16_t u16MaxPktLen = pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw3.u16MSS;
4632 Assert(u16MaxPktLen != 0);
4633 Assert(u16MaxPktLen < E1K_MAX_TX_PKT_SIZE);
4634
4635 /*
4636 * Carve out segments.
4637 */
4638 do
4639 {
4640 /* Calculate how many bytes we have left in this TCP segment */
4641 uint32_t cb = u16MaxPktLen - pThis->u16TxPktLen;
4642 if (cb > cbFragment)
4643 {
4644 /* This descriptor fits completely into current segment */
4645 cb = cbFragment;
4646 e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, pDesc->data.cmd.fEOP /*fSend*/, fOnWorkerThread);
4647 }
4648 else
4649 {
4650 e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, true /*fSend*/, fOnWorkerThread);
4651 /*
4652 * Rewind the packet tail pointer to the beginning of payload,
4653 * so we continue writing right beyond the header.
4654 */
4655 pThis->u16TxPktLen = pThis->contextTSE.dw3.u8HDRLEN;
4656 }
4657
4658 pDesc->data.u64BufAddr += cb;
4659 cbFragment -= cb;
4660 } while (cbFragment > 0);
4661
4662 if (pDesc->data.cmd.fEOP)
4663 {
4664 /* End of packet, next segment will contain header. */
4665 if (pThis->u32PayRemain != 0)
4666 E1K_INC_CNT32(TSCTFC);
4667 pThis->u16TxPktLen = 0;
4668 e1kXmitFreeBuf(pThis, PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC));
4669 }
4670
4671 return false;
4672}
4673#else /* E1K_WITH_TXD_CACHE */
4674/**
4675 * TCP segmentation offloading fallback: Add descriptor's buffer to transmit
4676 * frame.
4677 *
4678 * We construct the frame in the fallback buffer first and the copy it to the SG
4679 * buffer before passing it down to the network driver code.
4680 *
4681 * @returns error code
4682 *
4683 * @param pDevIns The device instance.
4684 * @param pThis The device state structure.
4685 * @param pDesc Pointer to the descriptor to transmit.
4686 * @param cbFragment Length of descriptor's buffer.
4687 * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
4688 * @thread E1000_TX
4689 */
4690static int e1kFallbackAddToFrame(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KTXDESC *pDesc, bool fOnWorkerThread)
4691{
4692#ifdef VBOX_STRICT
4693 PPDMSCATTERGATHER pTxSg = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC)->CTX_SUFF(pTxSg);
4694 Assert(e1kGetDescType(pDesc) == E1K_DTYP_DATA);
4695 Assert(pDesc->data.cmd.fTSE);
4696 Assert(!e1kXmitIsGsoBuf(pTxSg));
4697#endif
4698
4699 uint16_t u16MaxPktLen = pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw3.u16MSS;
4700 /* We cannot produce empty packets, ignore all TX descriptors (see @bugref{9571}) */
4701 if (u16MaxPktLen == 0)
4702 return VINF_SUCCESS;
4703
4704 /*
4705 * Carve out segments.
4706 */
4707 int rc = VINF_SUCCESS;
4708 do
4709 {
4710 /* Calculate how many bytes we have left in this TCP segment */
4711 uint16_t cb = u16MaxPktLen - pThis->u16TxPktLen;
4712 if (cb > pDesc->data.cmd.u20DTALEN)
4713 {
4714 /* This descriptor fits completely into current segment */
4715 cb = (uint16_t)pDesc->data.cmd.u20DTALEN; /* u20DTALEN at this point is guarantied to fit into 16 bits. */
4716 rc = e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, pDesc->data.cmd.fEOP /*fSend*/, fOnWorkerThread);
4717 }
4718 else
4719 {
4720 rc = e1kFallbackAddSegment(pDevIns, pThis, pDesc->data.u64BufAddr, cb, true /*fSend*/, fOnWorkerThread);
4721 /*
4722 * Rewind the packet tail pointer to the beginning of payload,
4723 * so we continue writing right beyond the header.
4724 */
4725 pThis->u16TxPktLen = pThis->contextTSE.dw3.u8HDRLEN;
4726 }
4727
4728 pDesc->data.u64BufAddr += cb;
4729 pDesc->data.cmd.u20DTALEN -= cb;
4730 } while (pDesc->data.cmd.u20DTALEN > 0 && RT_SUCCESS(rc));
4731
4732 if (pDesc->data.cmd.fEOP)
4733 {
4734 /* End of packet, next segment will contain header. */
4735 if (pThis->u32PayRemain != 0)
4736 E1K_INC_CNT32(TSCTFC);
4737 pThis->u16TxPktLen = 0;
4738 e1kXmitFreeBuf(pThis, PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC));
4739 }
4740
4741 return VINF_SUCCESS; /// @todo consider rc;
4742}
4743#endif /* E1K_WITH_TXD_CACHE */
4744
4745
4746/**
4747 * Add descriptor's buffer to transmit frame.
4748 *
4749 * This deals with GSO and normal frames, e1kFallbackAddToFrame deals with the
4750 * TSE frames we cannot handle as GSO.
4751 *
4752 * @returns true on success, false on failure.
4753 *
4754 * @param pDevIns The device instance.
4755 * @param pThisCC The current context instance data.
4756 * @param pThis The device state structure.
4757 * @param PhysAddr The physical address of the descriptor buffer.
4758 * @param cbFragment Length of descriptor's buffer.
4759 * @thread E1000_TX
4760 */
4761static bool e1kAddToFrame(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, RTGCPHYS PhysAddr, uint32_t cbFragment)
4762{
4763 PPDMSCATTERGATHER pTxSg = pThisCC->CTX_SUFF(pTxSg);
4764 bool const fGso = e1kXmitIsGsoBuf(pTxSg);
4765 uint32_t const cbNewPkt = cbFragment + pThis->u16TxPktLen;
4766
4767 LogFlow(("%s e1kAddToFrame: ENTER cbFragment=%d u16TxPktLen=%d cbUsed=%d cbAvailable=%d fGSO=%s\n",
4768 pThis->szPrf, cbFragment, pThis->u16TxPktLen, pTxSg->cbUsed, pTxSg->cbAvailable,
4769 fGso ? "true" : "false"));
4770 PCPDMNETWORKGSO pGso = (PCPDMNETWORKGSO)pTxSg->pvUser;
4771 if (pGso)
4772 {
4773 if (RT_UNLIKELY(pGso->cbMaxSeg == 0))
4774 {
4775 E1kLog(("%s zero-sized fragments are not allowed\n", pThis->szPrf));
4776 return false;
4777 }
4778 if (RT_UNLIKELY(pGso->u8Type == PDMNETWORKGSOTYPE_IPV4_UDP))
4779 {
4780 E1kLog(("%s UDP fragmentation is no longer supported\n", pThis->szPrf));
4781 return false;
4782 }
4783 }
4784 if (RT_UNLIKELY( !fGso && cbNewPkt > E1K_MAX_TX_PKT_SIZE ))
4785 {
4786 E1kLog(("%s Transmit packet is too large: %u > %u(max)\n", pThis->szPrf, cbNewPkt, E1K_MAX_TX_PKT_SIZE));
4787 return false;
4788 }
4789 if (RT_UNLIKELY( cbNewPkt > pTxSg->cbAvailable ))
4790 {
4791 E1kLog(("%s Transmit packet is too large: %u > %u(max)\n", pThis->szPrf, cbNewPkt, pTxSg->cbAvailable));
4792 return false;
4793 }
4794
4795 if (RT_LIKELY(pTxSg))
4796 {
4797 Assert(pTxSg->cSegs == 1);
4798 if (pTxSg->cbUsed != pThis->u16TxPktLen)
4799 E1kLog(("%s e1kAddToFrame: pTxSg->cbUsed=%d(0x%x) != u16TxPktLen=%d(0x%x)\n",
4800 pThis->szPrf, pTxSg->cbUsed, pTxSg->cbUsed, pThis->u16TxPktLen, pThis->u16TxPktLen));
4801
4802 PDMDevHlpPCIPhysRead(pDevIns, PhysAddr, (uint8_t *)pTxSg->aSegs[0].pvSeg + pThis->u16TxPktLen, cbFragment);
4803
4804 pTxSg->cbUsed = cbNewPkt;
4805 }
4806 pThis->u16TxPktLen = cbNewPkt;
4807
4808 return true;
4809}
4810
4811
4812/**
4813 * Write the descriptor back to guest memory and notify the guest.
4814 *
4815 * @param pThis The device state structure.
4816 * @param pDesc Pointer to the descriptor have been transmitted.
4817 * @param addr Physical address of the descriptor in guest memory.
4818 * @thread E1000_TX
4819 */
4820static void e1kDescReport(PPDMDEVINS pDevIns, PE1KSTATE pThis, E1KTXDESC *pDesc, RTGCPHYS addr)
4821{
4822 /*
4823 * We fake descriptor write-back bursting. Descriptors are written back as they are
4824 * processed.
4825 */
4826 /* Let's pretend we process descriptors. Write back with DD set. */
4827 /*
4828 * Prior to r71586 we tried to accomodate the case when write-back bursts
4829 * are enabled without actually implementing bursting by writing back all
4830 * descriptors, even the ones that do not have RS set. This caused kernel
4831 * panics with Linux SMP kernels, as the e1000 driver tried to free up skb
4832 * associated with written back descriptor if it happened to be a context
4833 * descriptor since context descriptors do not have skb associated to them.
4834 * Starting from r71586 we write back only the descriptors with RS set,
4835 * which is a little bit different from what the real hardware does in
4836 * case there is a chain of data descritors where some of them have RS set
4837 * and others do not. It is very uncommon scenario imho.
4838 * We need to check RPS as well since some legacy drivers use it instead of
4839 * RS even with newer cards.
4840 */
4841 if (pDesc->legacy.cmd.fRS || pDesc->legacy.cmd.fRPS)
4842 {
4843 pDesc->legacy.dw3.fDD = 1; /* Descriptor Done */
4844 e1kWriteBackDesc(pDevIns, pThis, pDesc, addr);
4845 if (pDesc->legacy.cmd.fEOP)
4846 {
4847//#ifdef E1K_USE_TX_TIMERS
4848 if (pThis->fTidEnabled && pDesc->legacy.cmd.fIDE)
4849 {
4850 E1K_INC_ISTAT_CNT(pThis->uStatTxIDE);
4851 //if (pThis->fIntRaised)
4852 //{
4853 // /* Interrupt is already pending, no need for timers */
4854 // ICR |= ICR_TXDW;
4855 //}
4856 //else {
4857 /* Arm the timer to fire in TIVD usec (discard .024) */
4858 e1kArmTimer(pDevIns, pThis, pThis->hTIDTimer, TIDV);
4859# ifndef E1K_NO_TAD
4860 /* If absolute timer delay is enabled and the timer is not running yet, arm it. */
4861 E1kLog2(("%s Checking if TAD timer is running\n",
4862 pThis->szPrf));
4863 if (TADV != 0 && !PDMDevHlpTimerIsActive(pDevIns, pThis->hTADTimer))
4864 e1kArmTimer(pDevIns, pThis, pThis->hTADTimer, TADV);
4865# endif /* E1K_NO_TAD */
4866 }
4867 else
4868 {
4869 if (pThis->fTidEnabled)
4870 {
4871 E1kLog2(("%s No IDE set, cancel TAD timer and raise interrupt\n",
4872 pThis->szPrf));
4873 /* Cancel both timers if armed and fire immediately. */
4874# ifndef E1K_NO_TAD
4875 PDMDevHlpTimerStop(pDevIns, pThis->hTADTimer);
4876# endif
4877 PDMDevHlpTimerStop(pDevIns, pThis->hTIDTimer);
4878 }
4879//#endif /* E1K_USE_TX_TIMERS */
4880 E1K_INC_ISTAT_CNT(pThis->uStatIntTx);
4881 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXDW);
4882//#ifdef E1K_USE_TX_TIMERS
4883 }
4884//#endif /* E1K_USE_TX_TIMERS */
4885 }
4886 }
4887 else
4888 {
4889 E1K_INC_ISTAT_CNT(pThis->uStatTxNoRS);
4890 }
4891}
4892
4893#ifndef E1K_WITH_TXD_CACHE
4894
4895/**
4896 * Process Transmit Descriptor.
4897 *
4898 * E1000 supports three types of transmit descriptors:
4899 * - legacy data descriptors of older format (context-less).
4900 * - data the same as legacy but providing new offloading capabilities.
4901 * - context sets up the context for following data descriptors.
4902 *
4903 * @param pDevIns The device instance.
4904 * @param pThis The device state structure.
4905 * @param pThisCC The current context instance data.
4906 * @param pDesc Pointer to descriptor union.
4907 * @param addr Physical address of descriptor in guest memory.
4908 * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
4909 * @thread E1000_TX
4910 */
4911static int e1kXmitDesc(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, E1KTXDESC *pDesc,
4912 RTGCPHYS addr, bool fOnWorkerThread)
4913{
4914 int rc = VINF_SUCCESS;
4915 uint32_t cbVTag = 0;
4916
4917 e1kPrintTDesc(pThis, pDesc, "vvv");
4918
4919//#ifdef E1K_USE_TX_TIMERS
4920 if (pThis->fTidEnabled)
4921 e1kCancelTimer(pDevIns, pThis, pThis->hTIDTimer);
4922//#endif /* E1K_USE_TX_TIMERS */
4923
4924 switch (e1kGetDescType(pDesc))
4925 {
4926 case E1K_DTYP_CONTEXT:
4927 if (pDesc->context.dw2.fTSE)
4928 {
4929 pThis->contextTSE = pDesc->context;
4930 pThis->u32PayRemain = pDesc->context.dw2.u20PAYLEN;
4931 pThis->u16HdrRemain = pDesc->context.dw3.u8HDRLEN;
4932 e1kSetupGsoCtx(&pThis->GsoCtx, &pDesc->context);
4933 STAM_COUNTER_INC(&pThis->StatTxDescCtxTSE);
4934 }
4935 else
4936 {
4937 pThis->contextNormal = pDesc->context;
4938 STAM_COUNTER_INC(&pThis->StatTxDescCtxNormal);
4939 }
4940 E1kLog2(("%s %s context updated: IP CSS=%02X, IP CSO=%02X, IP CSE=%04X"
4941 ", TU CSS=%02X, TU CSO=%02X, TU CSE=%04X\n", pThis->szPrf,
4942 pDesc->context.dw2.fTSE ? "TSE" : "Normal",
4943 pDesc->context.ip.u8CSS,
4944 pDesc->context.ip.u8CSO,
4945 pDesc->context.ip.u16CSE,
4946 pDesc->context.tu.u8CSS,
4947 pDesc->context.tu.u8CSO,
4948 pDesc->context.tu.u16CSE));
4949 E1K_INC_ISTAT_CNT(pThis->uStatDescCtx);
4950 e1kDescReport(pThis, pDesc, addr);
4951 break;
4952
4953 case E1K_DTYP_DATA:
4954 {
4955 if (pDesc->data.cmd.u20DTALEN == 0 || pDesc->data.u64BufAddr == 0)
4956 {
4957 E1kLog2(("% Empty data descriptor, skipped.\n", pThis->szPrf));
4958 /** @todo Same as legacy when !TSE. See below. */
4959 break;
4960 }
4961 STAM_COUNTER_INC(pDesc->data.cmd.fTSE?
4962 &pThis->StatTxDescTSEData:
4963 &pThis->StatTxDescData);
4964 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
4965 E1K_INC_ISTAT_CNT(pThis->uStatDescDat);
4966
4967 /*
4968 * The last descriptor of non-TSE packet must contain VLE flag.
4969 * TSE packets have VLE flag in the first descriptor. The later
4970 * case is taken care of a bit later when cbVTag gets assigned.
4971 *
4972 * 1) pDesc->data.cmd.fEOP && !pDesc->data.cmd.fTSE
4973 */
4974 if (pDesc->data.cmd.fEOP && !pDesc->data.cmd.fTSE)
4975 {
4976 pThis->fVTag = pDesc->data.cmd.fVLE;
4977 pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
4978 }
4979 /*
4980 * First fragment: Allocate new buffer and save the IXSM and TXSM
4981 * packet options as these are only valid in the first fragment.
4982 */
4983 if (pThis->u16TxPktLen == 0)
4984 {
4985 pThis->fIPcsum = pDesc->data.dw3.fIXSM;
4986 pThis->fTCPcsum = pDesc->data.dw3.fTXSM;
4987 E1kLog2(("%s Saving checksum flags:%s%s; \n", pThis->szPrf,
4988 pThis->fIPcsum ? " IP" : "",
4989 pThis->fTCPcsum ? " TCP/UDP" : ""));
4990 if (pDesc->data.cmd.fTSE)
4991 {
4992 /* 2) pDesc->data.cmd.fTSE && pThis->u16TxPktLen == 0 */
4993 pThis->fVTag = pDesc->data.cmd.fVLE;
4994 pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
4995 cbVTag = pThis->fVTag ? 4 : 0;
4996 }
4997 else if (pDesc->data.cmd.fEOP)
4998 cbVTag = pDesc->data.cmd.fVLE ? 4 : 0;
4999 else
5000 cbVTag = 4;
5001 E1kLog3(("%s About to allocate TX buffer: cbVTag=%u\n", pThis->szPrf, cbVTag));
5002 if (e1kCanDoGso(pThis, &pThis->GsoCtx, &pDesc->data, &pThis->contextTSE))
5003 rc = e1kXmitAllocBuf(pThis, pThisCC, pThis->contextTSE.dw2.u20PAYLEN + pThis->contextTSE.dw3.u8HDRLEN + cbVTag,
5004 true /*fExactSize*/, true /*fGso*/);
5005 else if (pDesc->data.cmd.fTSE)
5006 rc = e1kXmitAllocBuf(pThis, pThisCC, , pThis->contextTSE.dw3.u16MSS + pThis->contextTSE.dw3.u8HDRLEN + cbVTag,
5007 pDesc->data.cmd.fTSE /*fExactSize*/, false /*fGso*/);
5008 else
5009 rc = e1kXmitAllocBuf(pThis, pThisCC, pDesc->data.cmd.u20DTALEN + cbVTag,
5010 pDesc->data.cmd.fEOP /*fExactSize*/, false /*fGso*/);
5011
5012 /**
5013 * @todo: Perhaps it is not that simple for GSO packets! We may
5014 * need to unwind some changes.
5015 */
5016 if (RT_FAILURE(rc))
5017 {
5018 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5019 break;
5020 }
5021 /** @todo Is there any way to indicating errors other than collisions? Like
5022 * VERR_NET_DOWN. */
5023 }
5024
5025 /*
5026 * Add the descriptor data to the frame. If the frame is complete,
5027 * transmit it and reset the u16TxPktLen field.
5028 */
5029 if (e1kXmitIsGsoBuf(pThisCC->CTX_SUFF(pTxSg)))
5030 {
5031 STAM_COUNTER_INC(&pThis->StatTxPathGSO);
5032 bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
5033 if (pDesc->data.cmd.fEOP)
5034 {
5035 if ( fRc
5036 && pThisCC->CTX_SUFF(pTxSg)
5037 && pThisCC->CTX_SUFF(pTxSg)->cbUsed == (size_t)pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN)
5038 {
5039 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5040 E1K_INC_CNT32(TSCTC);
5041 }
5042 else
5043 {
5044 if (fRc)
5045 E1kLog(("%s bad GSO/TSE %p or %u < %u\n" , pThis->szPrf,
5046 pThisCC->CTX_SUFF(pTxSg), pThisCC->CTX_SUFF(pTxSg) ? pThisCC->CTX_SUFF(pTxSg)->cbUsed : 0,
5047 pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN));
5048 e1kXmitFreeBuf(pThis);
5049 E1K_INC_CNT32(TSCTFC);
5050 }
5051 pThis->u16TxPktLen = 0;
5052 }
5053 }
5054 else if (!pDesc->data.cmd.fTSE)
5055 {
5056 STAM_COUNTER_INC(&pThis->StatTxPathRegular);
5057 bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
5058 if (pDesc->data.cmd.fEOP)
5059 {
5060 if (fRc && pThisCC->CTX_SUFF(pTxSg))
5061 {
5062 Assert(pThisCC->CTX_SUFF(pTxSg)->cSegs == 1);
5063 if (pThis->fIPcsum)
5064 e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
5065 pThis->contextNormal.ip.u8CSO,
5066 pThis->contextNormal.ip.u8CSS,
5067 pThis->contextNormal.ip.u16CSE);
5068 if (pThis->fTCPcsum)
5069 e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
5070 pThis->contextNormal.tu.u8CSO,
5071 pThis->contextNormal.tu.u8CSS,
5072 pThis->contextNormal.tu.u16CSE,
5073 !pThis->contextNormal.dw2.fTCP);
5074 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5075 }
5076 else
5077 e1kXmitFreeBuf(pThis);
5078 pThis->u16TxPktLen = 0;
5079 }
5080 }
5081 else
5082 {
5083 STAM_COUNTER_INC(&pThis->StatTxPathFallback);
5084 e1kFallbackAddToFrame(pDevIns, pThis, pDesc, pDesc->data.cmd.u20DTALEN, fOnWorkerThread);
5085 }
5086
5087 e1kDescReport(pThis, pDesc, addr);
5088 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5089 break;
5090 }
5091
5092 case E1K_DTYP_LEGACY:
5093 if (pDesc->legacy.cmd.u16Length == 0 || pDesc->legacy.u64BufAddr == 0)
5094 {
5095 E1kLog(("%s Empty legacy descriptor, skipped.\n", pThis->szPrf));
5096 /** @todo 3.3.3, Length/Buffer Address: RS set -> write DD when processing. */
5097 break;
5098 }
5099 STAM_COUNTER_INC(&pThis->StatTxDescLegacy);
5100 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
5101
5102 /* First fragment: allocate new buffer. */
5103 if (pThis->u16TxPktLen == 0)
5104 {
5105 if (pDesc->legacy.cmd.fEOP)
5106 cbVTag = pDesc->legacy.cmd.fVLE ? 4 : 0;
5107 else
5108 cbVTag = 4;
5109 E1kLog3(("%s About to allocate TX buffer: cbVTag=%u\n", pThis->szPrf, cbVTag));
5110 /** @todo reset status bits? */
5111 rc = e1kXmitAllocBuf(pThis, pThisCC, pDesc->legacy.cmd.u16Length + cbVTag, pDesc->legacy.cmd.fEOP, false /*fGso*/);
5112 if (RT_FAILURE(rc))
5113 {
5114 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5115 break;
5116 }
5117
5118 /** @todo Is there any way to indicating errors other than collisions? Like
5119 * VERR_NET_DOWN. */
5120 }
5121
5122 /* Add fragment to frame. */
5123 if (e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->legacy.cmd.u16Length))
5124 {
5125 E1K_INC_ISTAT_CNT(pThis->uStatDescLeg);
5126
5127 /* Last fragment: Transmit and reset the packet storage counter. */
5128 if (pDesc->legacy.cmd.fEOP)
5129 {
5130 pThis->fVTag = pDesc->legacy.cmd.fVLE;
5131 pThis->u16VTagTCI = pDesc->legacy.dw3.u16Special;
5132 /** @todo Offload processing goes here. */
5133 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5134 pThis->u16TxPktLen = 0;
5135 }
5136 }
5137 /* Last fragment + failure: free the buffer and reset the storage counter. */
5138 else if (pDesc->legacy.cmd.fEOP)
5139 {
5140 e1kXmitFreeBuf(pThis);
5141 pThis->u16TxPktLen = 0;
5142 }
5143
5144 e1kDescReport(pThis, pDesc, addr);
5145 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5146 break;
5147
5148 default:
5149 E1kLog(("%s ERROR Unsupported transmit descriptor type: 0x%04x\n",
5150 pThis->szPrf, e1kGetDescType(pDesc)));
5151 break;
5152 }
5153
5154 return rc;
5155}
5156
5157#else /* E1K_WITH_TXD_CACHE */
5158
5159/**
5160 * Process Transmit Descriptor.
5161 *
5162 * E1000 supports three types of transmit descriptors:
5163 * - legacy data descriptors of older format (context-less).
5164 * - data the same as legacy but providing new offloading capabilities.
5165 * - context sets up the context for following data descriptors.
5166 *
5167 * @param pDevIns The device instance.
5168 * @param pThis The device state structure.
5169 * @param pThisCC The current context instance data.
5170 * @param pDesc Pointer to descriptor union.
5171 * @param addr Physical address of descriptor in guest memory.
5172 * @param fOnWorkerThread Whether we're on a worker thread or an EMT.
5173 * @param cbPacketSize Size of the packet as previously computed.
5174 * @thread E1000_TX
5175 */
5176static int e1kXmitDesc(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KSTATECC pThisCC, E1KTXDESC *pDesc,
5177 RTGCPHYS addr, bool fOnWorkerThread)
5178{
5179 int rc = VINF_SUCCESS;
5180
5181 e1kPrintTDesc(pThis, pDesc, "vvv");
5182
5183//#ifdef E1K_USE_TX_TIMERS
5184 if (pThis->fTidEnabled)
5185 PDMDevHlpTimerStop(pDevIns, pThis->hTIDTimer);
5186//#endif /* E1K_USE_TX_TIMERS */
5187
5188 switch (e1kGetDescType(pDesc))
5189 {
5190 case E1K_DTYP_CONTEXT:
5191 /* The caller have already updated the context */
5192 E1K_INC_ISTAT_CNT(pThis->uStatDescCtx);
5193 e1kDescReport(pDevIns, pThis, pDesc, addr);
5194 break;
5195
5196 case E1K_DTYP_DATA:
5197 {
5198 STAM_COUNTER_INC(pDesc->data.cmd.fTSE?
5199 &pThis->StatTxDescTSEData:
5200 &pThis->StatTxDescData);
5201 E1K_INC_ISTAT_CNT(pThis->uStatDescDat);
5202 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
5203 if (pDesc->data.cmd.u20DTALEN == 0 || pDesc->data.u64BufAddr == 0)
5204 {
5205 E1kLog2(("%s Empty data descriptor, skipped.\n", pThis->szPrf));
5206 if (pDesc->data.cmd.fEOP)
5207 {
5208 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5209 pThis->u16TxPktLen = 0;
5210 }
5211 }
5212 else
5213 {
5214 /*
5215 * Add the descriptor data to the frame. If the frame is complete,
5216 * transmit it and reset the u16TxPktLen field.
5217 */
5218 if (e1kXmitIsGsoBuf(pThisCC->CTX_SUFF(pTxSg)))
5219 {
5220 STAM_COUNTER_INC(&pThis->StatTxPathGSO);
5221 bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
5222 if (pDesc->data.cmd.fEOP)
5223 {
5224 if ( fRc
5225 && pThisCC->CTX_SUFF(pTxSg)
5226 && pThisCC->CTX_SUFF(pTxSg)->cbUsed == (size_t)pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN)
5227 {
5228 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5229 E1K_INC_CNT32(TSCTC);
5230 }
5231 else
5232 {
5233 if (fRc)
5234 E1kLog(("%s bad GSO/TSE %p or %u < %u\n" , pThis->szPrf,
5235 pThisCC->CTX_SUFF(pTxSg), pThisCC->CTX_SUFF(pTxSg) ? pThisCC->CTX_SUFF(pTxSg)->cbUsed : 0,
5236 pThis->contextTSE.dw3.u8HDRLEN + pThis->contextTSE.dw2.u20PAYLEN));
5237 e1kXmitFreeBuf(pThis, pThisCC);
5238 E1K_INC_CNT32(TSCTFC);
5239 }
5240 pThis->u16TxPktLen = 0;
5241 }
5242 }
5243 else if (!pDesc->data.cmd.fTSE)
5244 {
5245 STAM_COUNTER_INC(&pThis->StatTxPathRegular);
5246 bool fRc = e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->data.cmd.u20DTALEN);
5247 if (pDesc->data.cmd.fEOP)
5248 {
5249 if (fRc && pThisCC->CTX_SUFF(pTxSg))
5250 {
5251 Assert(pThisCC->CTX_SUFF(pTxSg)->cSegs == 1);
5252 if (pThis->fIPcsum)
5253 e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
5254 pThis->contextNormal.ip.u8CSO,
5255 pThis->contextNormal.ip.u8CSS,
5256 pThis->contextNormal.ip.u16CSE);
5257 if (pThis->fTCPcsum)
5258 e1kInsertChecksum(pThis, (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg, pThis->u16TxPktLen,
5259 pThis->contextNormal.tu.u8CSO,
5260 pThis->contextNormal.tu.u8CSS,
5261 pThis->contextNormal.tu.u16CSE,
5262 !pThis->contextNormal.dw2.fTCP);
5263 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5264 }
5265 else
5266 e1kXmitFreeBuf(pThis, pThisCC);
5267 pThis->u16TxPktLen = 0;
5268 }
5269 }
5270 else
5271 {
5272 STAM_COUNTER_INC(&pThis->StatTxPathFallback);
5273 rc = e1kFallbackAddToFrame(pDevIns, pThis, pDesc, fOnWorkerThread);
5274 }
5275 }
5276 e1kDescReport(pDevIns, pThis, pDesc, addr);
5277 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5278 break;
5279 }
5280
5281 case E1K_DTYP_LEGACY:
5282 STAM_COUNTER_INC(&pThis->StatTxDescLegacy);
5283 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
5284 if (pDesc->legacy.cmd.u16Length == 0 || pDesc->legacy.u64BufAddr == 0)
5285 {
5286 E1kLog(("%s Empty legacy descriptor, skipped.\n", pThis->szPrf));
5287 if (pDesc->data.cmd.fEOP)
5288 {
5289 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5290 pThis->u16TxPktLen = 0;
5291 }
5292 }
5293 else
5294 {
5295 /* Add fragment to frame. */
5296 if (e1kAddToFrame(pDevIns, pThis, pThisCC, pDesc->data.u64BufAddr, pDesc->legacy.cmd.u16Length))
5297 {
5298 E1K_INC_ISTAT_CNT(pThis->uStatDescLeg);
5299
5300 /* Last fragment: Transmit and reset the packet storage counter. */
5301 if (pDesc->legacy.cmd.fEOP)
5302 {
5303 if (pDesc->legacy.cmd.fIC)
5304 {
5305 e1kInsertChecksum(pThis,
5306 (uint8_t *)pThisCC->CTX_SUFF(pTxSg)->aSegs[0].pvSeg,
5307 pThis->u16TxPktLen,
5308 pDesc->legacy.cmd.u8CSO,
5309 pDesc->legacy.dw3.u8CSS,
5310 0);
5311 }
5312 e1kTransmitFrame(pDevIns, pThis, pThisCC, fOnWorkerThread);
5313 pThis->u16TxPktLen = 0;
5314 }
5315 }
5316 /* Last fragment + failure: free the buffer and reset the storage counter. */
5317 else if (pDesc->legacy.cmd.fEOP)
5318 {
5319 e1kXmitFreeBuf(pThis, pThisCC);
5320 pThis->u16TxPktLen = 0;
5321 }
5322 }
5323 e1kDescReport(pDevIns, pThis, pDesc, addr);
5324 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5325 break;
5326
5327 default:
5328 E1kLog(("%s ERROR Unsupported transmit descriptor type: 0x%04x\n",
5329 pThis->szPrf, e1kGetDescType(pDesc)));
5330 break;
5331 }
5332
5333 return rc;
5334}
5335
5336DECLINLINE(bool) e1kUpdateTxContext(PE1KSTATE pThis, E1KTXDESC *pDesc)
5337{
5338 if (pDesc->context.dw2.fTSE)
5339 {
5340 if (!e1kSetupGsoCtx(&pThis->GsoCtx, &pDesc->context))
5341 {
5342 pThis->contextTSE.dw2.u4DTYP = E1K_DTYP_INVALID;
5343 return false;
5344 }
5345 pThis->contextTSE = pDesc->context;
5346 uint32_t cbMaxSegmentSize = pThis->contextTSE.dw3.u16MSS + pThis->contextTSE.dw3.u8HDRLEN + 4; /*VTAG*/
5347 if (RT_UNLIKELY(cbMaxSegmentSize > E1K_MAX_TX_PKT_SIZE))
5348 {
5349 pThis->contextTSE.dw3.u16MSS = E1K_MAX_TX_PKT_SIZE - pThis->contextTSE.dw3.u8HDRLEN - 4; /*VTAG*/
5350 LogRelMax(10, ("%s: Transmit packet is too large: %u > %u(max). Adjusted MSS to %u.\n",
5351 pThis->szPrf, cbMaxSegmentSize, E1K_MAX_TX_PKT_SIZE, pThis->contextTSE.dw3.u16MSS));
5352 }
5353 pThis->u32PayRemain = pThis->contextTSE.dw2.u20PAYLEN;
5354 pThis->u16HdrRemain = pThis->contextTSE.dw3.u8HDRLEN;
5355 e1kSetupGsoCtx(&pThis->GsoCtx, &pThis->contextTSE);
5356 STAM_COUNTER_INC(&pThis->StatTxDescCtxTSE);
5357 }
5358 else
5359 {
5360 pThis->contextNormal = pDesc->context;
5361 STAM_COUNTER_INC(&pThis->StatTxDescCtxNormal);
5362 }
5363 E1kLog2(("%s %s context updated: IP CSS=%02X, IP CSO=%02X, IP CSE=%04X"
5364 ", TU CSS=%02X, TU CSO=%02X, TU CSE=%04X\n", pThis->szPrf,
5365 pDesc->context.dw2.fTSE ? "TSE" : "Normal",
5366 pDesc->context.ip.u8CSS,
5367 pDesc->context.ip.u8CSO,
5368 pDesc->context.ip.u16CSE,
5369 pDesc->context.tu.u8CSS,
5370 pDesc->context.tu.u8CSO,
5371 pDesc->context.tu.u16CSE));
5372 return true; /* Consider returning false for invalid descriptors */
5373}
5374
5375enum E1kPacketType
5376{
5377 E1K_PACKET_NONE = 0,
5378 E1K_PACKET_LEGACY,
5379 E1K_PACKET_NORMAL,
5380 E1K_PACKET_TSE
5381};
5382
5383static int e1kLocateTxPacket(PE1KSTATE pThis, PE1KTXDC pTxdc)
5384{
5385 LogFlow(("%s e1kLocateTxPacket: ENTER cbTxAlloc=%d\n",
5386 pThis->szPrf, pThis->cbTxAlloc));
5387 /* Check if we have located the packet already. */
5388 if (pThis->cbTxAlloc)
5389 {
5390 LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d\n",
5391 pThis->szPrf, pThis->cbTxAlloc));
5392 return true;
5393 }
5394
5395 pThis->fGSO = false;
5396 pThis->fVTag = false;
5397 pThis->fIPcsum = false;
5398 pThis->fTCPcsum = false;
5399 pThis->u16TxPktLen = 0;
5400
5401 enum E1kPacketType packetType = E1K_PACKET_NONE;
5402 enum E1kPacketType expectedPacketType = E1K_PACKET_NONE;
5403 /*
5404 * Valid packets start with 1 or 0 context descriptors, followed by 1 or
5405 * more data descriptors of the same type: legacy, normal or TSE. Note
5406 * that legacy descriptors do not belong to neither normal nor segmentation
5407 * contexts rendering the sequence (context_descriptor, legacy_descriptor)
5408 * invalid, but the context descriptor will still be applied and the legacy
5409 * descriptor will be treated as the beginning of next packet.
5410 */
5411 bool fInvalidPacket = false;
5412 bool fTSE = false;
5413 uint32_t cbPacket = 0;
5414
5415 /* Since we process one packet at a time we will only mark current packet's descriptors as valid */
5416 memset(pThis->afTxDValid, 0, sizeof(pThis->afTxDValid));
5417 for (int i = pThis->iTxDCurrent; i < pThis->nTxDFetched; ++i)
5418 {
5419 E1KTXDESC *pDesc = &pThis->aTxDescriptors[i];
5420
5421 switch (e1kGetDescType(pDesc))
5422 {
5423 case E1K_DTYP_CONTEXT:
5424 /* There can be only one context per packet. Each context descriptor starts a new packet. */
5425 if (packetType != E1K_PACKET_NONE)
5426 {
5427 fInvalidPacket = true;
5428 break;
5429 }
5430 packetType = (pDesc->context.dw2.fTSE) ? E1K_PACKET_TSE : E1K_PACKET_NORMAL;
5431 if (cbPacket == 0)
5432 pThis->afTxDValid[i] = e1kUpdateTxContext(pThis, pDesc);
5433 else
5434 E1kLog(("%s e1kLocateTxPacket: ignoring a context descriptor in the middle of a packet, cbPacket=%d\n",
5435 pThis->szPrf, cbPacket));
5436 continue;
5437 case E1K_DTYP_LEGACY:
5438 if (packetType != E1K_PACKET_NONE && packetType != E1K_PACKET_LEGACY)
5439 {
5440 fInvalidPacket = true;
5441 break;
5442 }
5443 packetType = E1K_PACKET_LEGACY;
5444 /* Skip invalid descriptors. */
5445 if (cbPacket > 0 && (pThis->fGSO || fTSE))
5446 {
5447 E1kLog(("%s e1kLocateTxPacket: ignoring a legacy descriptor in the segmentation context, cbPacket=%d\n",
5448 pThis->szPrf, cbPacket));
5449 continue;
5450 }
5451 pThis->afTxDValid[i] = true; /* Passed all checks, process it */
5452
5453 /* Skip empty descriptors. */
5454 if (!pDesc->legacy.u64BufAddr || !pDesc->legacy.cmd.u16Length)
5455 break;
5456 cbPacket += pDesc->legacy.cmd.u16Length;
5457 pThis->fGSO = false;
5458 break;
5459 case E1K_DTYP_DATA:
5460 expectedPacketType = pDesc->data.cmd.fTSE ? E1K_PACKET_TSE : E1K_PACKET_NORMAL;
5461 if (packetType != E1K_PACKET_NONE && packetType != expectedPacketType)
5462 {
5463 fInvalidPacket = true;
5464 break;
5465 }
5466 /* Skip invalid descriptors. */
5467 if (pDesc->data.cmd.fTSE)
5468 {
5469 if (pThis->contextTSE.dw2.u4DTYP == E1K_DTYP_INVALID)
5470 {
5471 E1kLog(("%s e1kLocateTxPacket: ignoring TSE descriptor in invalid segmentation context, cbPacket=%d\n",
5472 pThis->szPrf, cbPacket));
5473 continue;
5474 }
5475 }
5476 else /* !TSE */
5477 {
5478 if (pThis->contextNormal.dw2.u4DTYP == E1K_DTYP_INVALID)
5479 {
5480 E1kLog(("%s e1kLocateTxPacket: ignoring non-TSE descriptor in invalid normal context, cbPacket=%d\n",
5481 pThis->szPrf, cbPacket));
5482 continue;
5483 }
5484 }
5485 if (cbPacket > 0 && (bool)pDesc->data.cmd.fTSE != fTSE)
5486 {
5487 E1kLog(("%s e1kLocateTxPacket: ignoring %sTSE descriptor in the %ssegmentation context, cbPacket=%d\n",
5488 pThis->szPrf, pDesc->data.cmd.fTSE ? "" : "non-", fTSE ? "" : "non-", cbPacket));
5489 continue;
5490 }
5491 pThis->afTxDValid[i] = true; /* Passed all checks, process it */
5492
5493 /* Skip empty descriptors. */
5494 if (!pDesc->data.u64BufAddr || !pDesc->data.cmd.u20DTALEN)
5495 break;
5496 if (cbPacket == 0)
5497 {
5498 /*
5499 * The first fragment: save IXSM and TXSM options
5500 * as these are only valid in the first fragment.
5501 */
5502 pThis->fIPcsum = pDesc->data.dw3.fIXSM;
5503 pThis->fTCPcsum = pDesc->data.dw3.fTXSM;
5504 fTSE = pDesc->data.cmd.fTSE;
5505 /*
5506 * TSE descriptors have VLE bit properly set in
5507 * the first fragment.
5508 */
5509 if (fTSE)
5510 {
5511 pThis->fVTag = pDesc->data.cmd.fVLE;
5512 pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
5513 }
5514 pThis->fGSO = e1kCanDoGso(pThis, &pThis->GsoCtx, &pDesc->data, &pThis->contextTSE);
5515 }
5516 cbPacket += pDesc->data.cmd.u20DTALEN;
5517 break;
5518 default:
5519 AssertMsgFailed(("Impossible descriptor type!"));
5520 continue;
5521 }
5522 if (fInvalidPacket)
5523 {
5524 for (int index = pThis->iTxDCurrent; index < i; ++index)
5525 pThis->afTxDValid[index] = false; /* Make sure all descriptors for this packet are skipped by processing */
5526 LogFlow(("%s e1kLocateTxPacket: marked %d descriptors as invalid\n", pThis->szPrf, i - pThis->iTxDCurrent));
5527 LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d cbPacket=%d%s%s\n",
5528 pThis->szPrf, pThis->cbTxAlloc, cbPacket,
5529 pThis->fGSO ? " GSO" : "", fTSE ? " TSE" : ""));
5530 pTxdc->nextPacket = i;
5531 return true;
5532 }
5533 if (pDesc->legacy.cmd.fEOP)
5534 {
5535 /*
5536 * Non-TSE descriptors have VLE bit properly set in
5537 * the last fragment.
5538 */
5539 if (!fTSE)
5540 {
5541 pThis->fVTag = pDesc->data.cmd.fVLE;
5542 pThis->u16VTagTCI = pDesc->data.dw3.u16Special;
5543 }
5544 /*
5545 * Compute the required buffer size. If we cannot do GSO but still
5546 * have to do segmentation we allocate the first segment only.
5547 */
5548 pThis->cbTxAlloc = (!fTSE || pThis->fGSO) ?
5549 cbPacket :
5550 RT_MIN(cbPacket, pThis->contextTSE.dw3.u16MSS + pThis->contextTSE.dw3.u8HDRLEN);
5551 /* Do not add VLAN tags to empty packets. */
5552 if (pThis->fVTag && pThis->cbTxAlloc > 0)
5553 pThis->cbTxAlloc += 4;
5554 LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d cbPacket=%d%s%s\n",
5555 pThis->szPrf, pThis->cbTxAlloc, cbPacket,
5556 pThis->fGSO ? " GSO" : "", fTSE ? " TSE" : ""));
5557 pTxdc->nextPacket = i + 1;
5558 return true;
5559 }
5560 }
5561
5562 if (cbPacket == 0 && pThis->nTxDFetched - pThis->iTxDCurrent > 0)
5563 {
5564 /* All descriptors were empty, we need to process them as a dummy packet */
5565 LogFlow(("%s e1kLocateTxPacket: RET true cbTxAlloc=%d, zero packet!\n",
5566 pThis->szPrf, pThis->cbTxAlloc));
5567 pTxdc->nextPacket = pThis->nTxDFetched;
5568 return true;
5569 }
5570 LogFlow(("%s e1kLocateTxPacket: RET false cbTxAlloc=%d cbPacket=%d\n",
5571 pThis->szPrf, pThis->cbTxAlloc, cbPacket));
5572 return false;
5573}
5574
5575static int e1kXmitPacket(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread, PE1KTXDC pTxdc)
5576{
5577 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
5578 int rc = VINF_SUCCESS;
5579
5580 LogFlow(("%s e1kXmitPacket: ENTER current=%d fetched=%d\n",
5581 pThis->szPrf, pThis->iTxDCurrent, pThis->nTxDFetched));
5582
5583 while (pThis->iTxDCurrent < pTxdc->nextPacket && pThis->iTxDCurrent < pThis->nTxDFetched)
5584 {
5585 E1KTXDESC *pDesc = &pThis->aTxDescriptors[pThis->iTxDCurrent];
5586 E1kLog3(("%s About to process new TX descriptor at %08x%08x, TDLEN=%08x, TDH=%08x, TDT=%08x\n",
5587 pThis->szPrf, TDBAH, TDBAL + pTxdc->tdh * sizeof(E1KTXDESC), pTxdc->tdlen, pTxdc->tdh, pTxdc->tdt));
5588 if (!pThis->afTxDValid[pThis->iTxDCurrent])
5589 {
5590 e1kPrintTDesc(pThis, pDesc, "vvv");
5591 E1kLog(("%s e1kXmitDesc: skipping bad descriptor ^^^\n", pThis->szPrf));
5592 e1kDescReport(pDevIns, pThis, pDesc, e1kDescAddr(TDBAH, TDBAL, pTxdc->tdh));
5593 rc = VINF_SUCCESS;
5594 }
5595 else
5596 rc = e1kXmitDesc(pDevIns, pThis, pThisCC, pDesc, e1kDescAddr(TDBAH, TDBAL, pTxdc->tdh), fOnWorkerThread);
5597 if (RT_FAILURE(rc))
5598 break;
5599 if (++pTxdc->tdh * sizeof(E1KTXDESC) >= pTxdc->tdlen)
5600 pTxdc->tdh = 0;
5601 TDH = pTxdc->tdh; /* Sync the actual register and TXDC */
5602 uint32_t uLowThreshold = GET_BITS(TXDCTL, LWTHRESH)*8;
5603 if (uLowThreshold != 0 && e1kGetTxLen(pTxdc) <= uLowThreshold)
5604 {
5605 E1kLog2(("%s Low on transmit descriptors, raise ICR.TXD_LOW, len=%x thresh=%x\n",
5606 pThis->szPrf, e1kGetTxLen(pTxdc), GET_BITS(TXDCTL, LWTHRESH)*8));
5607 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXD_LOW);
5608 }
5609 ++pThis->iTxDCurrent;
5610 if (e1kGetDescType(pDesc) != E1K_DTYP_CONTEXT && pDesc->legacy.cmd.fEOP)
5611 break;
5612 }
5613
5614 LogFlow(("%s e1kXmitPacket: RET %Rrc current=%d fetched=%d\n",
5615 pThis->szPrf, rc, pThis->iTxDCurrent, pThis->nTxDFetched));
5616 return rc;
5617}
5618
5619#endif /* E1K_WITH_TXD_CACHE */
5620#ifndef E1K_WITH_TXD_CACHE
5621
5622/**
5623 * Transmit pending descriptors.
5624 *
5625 * @returns VBox status code. VERR_TRY_AGAIN is returned if we're busy.
5626 *
5627 * @param pDevIns The device instance.
5628 * @param pThis The E1000 state.
5629 * @param fOnWorkerThread Whether we're on a worker thread or on an EMT.
5630 */
5631static int e1kXmitPending(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread)
5632{
5633 int rc = VINF_SUCCESS;
5634 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
5635
5636 /* Check if transmitter is enabled. */
5637 if (!(TCTL & TCTL_EN))
5638 return VINF_SUCCESS;
5639 /*
5640 * Grab the xmit lock of the driver as well as the E1K device state.
5641 */
5642 rc = e1kCsTxEnter(pThis, VERR_SEM_BUSY);
5643 if (RT_LIKELY(rc == VINF_SUCCESS))
5644 {
5645 PPDMINETWORKUP pDrv = pThis->CTX_SUFF(pDrv);
5646 if (pDrv)
5647 {
5648 rc = pDrv->pfnBeginXmit(pDrv, fOnWorkerThread);
5649 if (RT_FAILURE(rc))
5650 {
5651 e1kCsTxLeave(pThis);
5652 return rc;
5653 }
5654 }
5655 /*
5656 * Process all pending descriptors.
5657 * Note! Do not process descriptors in locked state
5658 */
5659 while (TDH != TDT && !pThis->fLocked)
5660 {
5661 E1KTXDESC desc;
5662 E1kLog3(("%s About to process new TX descriptor at %08x%08x, TDLEN=%08x, TDH=%08x, TDT=%08x\n",
5663 pThis->szPrf, TDBAH, TDBAL + TDH * sizeof(desc), TDLEN, TDH, TDT));
5664
5665 e1kLoadDesc(pDevIns, &desc, ((uint64_t)TDBAH << 32) + TDBAL + TDH * sizeof(desc));
5666 rc = e1kXmitDesc(pDevIns, pThis, pThisCC, &desc, e1kDescAddr(TDBAH, TDBAL, TDH), fOnWorkerThread);
5667 /* If we failed to transmit descriptor we will try it again later */
5668 if (RT_FAILURE(rc))
5669 break;
5670 if (++TDH * sizeof(desc) >= TDLEN)
5671 TDH = 0;
5672
5673 if (e1kGetTxLen(pThis) <= GET_BITS(TXDCTL, LWTHRESH)*8)
5674 {
5675 E1kLog2(("%s Low on transmit descriptors, raise ICR.TXD_LOW, len=%x thresh=%x\n",
5676 pThis->szPrf, e1kGetTxLen(pThis), GET_BITS(TXDCTL, LWTHRESH)*8));
5677 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXD_LOW);
5678 }
5679
5680 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5681 }
5682
5683 /// @todo uncomment: pThis->uStatIntTXQE++;
5684 /// @todo uncomment: e1kRaiseInterrupt(pDevIns, pThis, ICR_TXQE);
5685 /*
5686 * Release the lock.
5687 */
5688 if (pDrv)
5689 pDrv->pfnEndXmit(pDrv);
5690 e1kCsTxLeave(pThis);
5691 }
5692
5693 return rc;
5694}
5695
5696#else /* E1K_WITH_TXD_CACHE */
5697
5698static void e1kDumpTxDCache(PPDMDEVINS pDevIns, PE1KSTATE pThis, PE1KTXDC pTxdc)
5699{
5700 unsigned i, cDescs = pTxdc->tdlen / sizeof(E1KTXDESC);
5701 uint32_t tdh = pTxdc->tdh;
5702 LogRel(("E1000: -- Transmit Descriptors (%d total) --\n", cDescs));
5703 for (i = 0; i < cDescs; ++i)
5704 {
5705 E1KTXDESC desc;
5706 PDMDevHlpPCIPhysRead(pDevIns , e1kDescAddr(TDBAH, TDBAL, i), &desc, sizeof(desc));
5707 if (i == tdh)
5708 LogRel(("E1000: >>> "));
5709 LogRel(("E1000: %RGp: %R[e1ktxd]\n", e1kDescAddr(TDBAH, TDBAL, i), &desc));
5710 }
5711 LogRel(("E1000: -- Transmit Descriptors in Cache (at %d (TDH %d)/ fetched %d / max %d) --\n",
5712 pThis->iTxDCurrent, pTxdc->tdh, pThis->nTxDFetched, E1K_TXD_CACHE_SIZE));
5713 if (tdh > pThis->iTxDCurrent)
5714 tdh -= pThis->iTxDCurrent;
5715 else
5716 tdh = cDescs + tdh - pThis->iTxDCurrent;
5717 for (i = 0; i < pThis->nTxDFetched; ++i)
5718 {
5719 if (i == pThis->iTxDCurrent)
5720 LogRel(("E1000: >>> "));
5721 if (cDescs)
5722 LogRel(("E1000: %RGp: %R[e1ktxd]\n", e1kDescAddr(TDBAH, TDBAL, tdh++ % cDescs), &pThis->aTxDescriptors[i]));
5723 else
5724 LogRel(("E1000: <lost>: %R[e1ktxd]\n", &pThis->aTxDescriptors[i]));
5725 }
5726}
5727
5728/**
5729 * Transmit pending descriptors.
5730 *
5731 * @returns VBox status code. VERR_TRY_AGAIN is returned if we're busy.
5732 *
5733 * @param pDevIns The device instance.
5734 * @param pThis The E1000 state.
5735 * @param fOnWorkerThread Whether we're on a worker thread or on an EMT.
5736 */
5737static int e1kXmitPending(PPDMDEVINS pDevIns, PE1KSTATE pThis, bool fOnWorkerThread)
5738{
5739 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
5740 int rc = VINF_SUCCESS;
5741
5742 /* Check if transmitter is enabled. */
5743 if (!(TCTL & TCTL_EN))
5744 return VINF_SUCCESS;
5745 /*
5746 * Grab the xmit lock of the driver as well as the E1K device state.
5747 */
5748 PPDMINETWORKUP pDrv = pThisCC->CTX_SUFF(pDrv);
5749 if (pDrv)
5750 {
5751 rc = pDrv->pfnBeginXmit(pDrv, fOnWorkerThread);
5752 if (RT_FAILURE(rc))
5753 return rc;
5754 }
5755
5756 /*
5757 * Process all pending descriptors.
5758 * Note! Do not process descriptors in locked state
5759 */
5760 rc = e1kCsTxEnter(pThis, VERR_SEM_BUSY);
5761 if (RT_LIKELY(rc == VINF_SUCCESS && (TCTL & TCTL_EN)))
5762 {
5763 E1KTXDC txdc;
5764 bool fTxContextValid = e1kUpdateTxDContext(pDevIns, pThis, &txdc);
5765 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
5766 /*
5767 * fIncomplete is set whenever we try to fetch additional descriptors
5768 * for an incomplete packet. If fail to locate a complete packet on
5769 * the next iteration we need to reset the cache or we risk to get
5770 * stuck in this loop forever.
5771 */
5772 bool fIncomplete = false;
5773 while (fTxContextValid && !pThis->fLocked && e1kTxDLazyLoad(pDevIns, pThis, &txdc))
5774 {
5775 while (e1kLocateTxPacket(pThis, &txdc))
5776 {
5777 Log4(("%s e1kXmitPending: Located packet at %d. Next packet at %d\n",
5778 pThis->szPrf, pThis->iTxDCurrent, txdc.nextPacket));
5779 fIncomplete = false;
5780 /* Found a complete packet, allocate it. */
5781 rc = e1kXmitAllocBuf(pThis, pThisCC, pThis->fGSO);
5782 /* If we're out of bandwidth we'll come back later. */
5783 if (RT_FAILURE(rc))
5784 goto out;
5785 /* Copy the packet to allocated buffer and send it. */
5786 rc = e1kXmitPacket(pDevIns, pThis, fOnWorkerThread, &txdc);
5787 /* If we're out of bandwidth we'll come back later. */
5788 if (RT_FAILURE(rc))
5789 goto out;
5790 }
5791 uint8_t u8Remain = pThis->nTxDFetched - pThis->iTxDCurrent;
5792 if (RT_UNLIKELY(fIncomplete))
5793 {
5794 static bool fTxDCacheDumped = false;
5795 /*
5796 * The descriptor cache is full, but we were unable to find
5797 * a complete packet in it. Drop the cache and hope that
5798 * the guest driver can recover from network card error.
5799 */
5800 LogRel(("%s: No complete packets in%s TxD cache! "
5801 "Fetched=%d, current=%d, TX len=%d.\n",
5802 pThis->szPrf,
5803 u8Remain == E1K_TXD_CACHE_SIZE ? " full" : "",
5804 pThis->nTxDFetched, pThis->iTxDCurrent,
5805 e1kGetTxLen(&txdc)));
5806 if (!fTxDCacheDumped)
5807 {
5808 fTxDCacheDumped = true;
5809 e1kDumpTxDCache(pDevIns, pThis, &txdc);
5810 }
5811 pThis->iTxDCurrent = pThis->nTxDFetched = 0;
5812 /*
5813 * Returning an error at this point means Guru in R0
5814 * (see @bugref{6428}).
5815 */
5816# ifdef IN_RING3
5817 rc = VERR_NET_INCOMPLETE_TX_PACKET;
5818# else /* !IN_RING3 */
5819 rc = VINF_IOM_R3_MMIO_WRITE;
5820# endif /* !IN_RING3 */
5821 goto out;
5822 }
5823 if (u8Remain > 0)
5824 {
5825 Log4(("%s Incomplete packet at %d. Already fetched %d, "
5826 "%d more are available\n",
5827 pThis->szPrf, pThis->iTxDCurrent, u8Remain,
5828 e1kGetTxLen(&txdc) - u8Remain));
5829
5830 /*
5831 * A packet was partially fetched. Move incomplete packet to
5832 * the beginning of cache buffer, then load more descriptors.
5833 */
5834 memmove(pThis->aTxDescriptors,
5835 &pThis->aTxDescriptors[pThis->iTxDCurrent],
5836 u8Remain * sizeof(E1KTXDESC));
5837 pThis->iTxDCurrent = 0;
5838 pThis->nTxDFetched = u8Remain;
5839 e1kTxDLoadMore(pDevIns, pThis, &txdc);
5840 fIncomplete = true;
5841 }
5842 else
5843 pThis->nTxDFetched = 0;
5844 pThis->iTxDCurrent = 0;
5845 }
5846 if (!pThis->fLocked && GET_BITS(TXDCTL, LWTHRESH) == 0)
5847 {
5848 E1kLog2(("%s Out of transmit descriptors, raise ICR.TXD_LOW\n",
5849 pThis->szPrf));
5850 e1kRaiseInterrupt(pDevIns, pThis, VERR_SEM_BUSY, ICR_TXD_LOW);
5851 }
5852out:
5853 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
5854
5855 /// @todo uncomment: pThis->uStatIntTXQE++;
5856 /// @todo uncomment: e1kRaiseInterrupt(pDevIns, pThis, ICR_TXQE);
5857
5858 e1kCsTxLeave(pThis);
5859 }
5860
5861
5862 /*
5863 * Release the lock.
5864 */
5865 if (pDrv)
5866 pDrv->pfnEndXmit(pDrv);
5867 return rc;
5868}
5869
5870#endif /* E1K_WITH_TXD_CACHE */
5871#ifdef IN_RING3
5872
5873/**
5874 * @interface_method_impl{PDMINETWORKDOWN,pfnXmitPending}
5875 */
5876static DECLCALLBACK(void) e1kR3NetworkDown_XmitPending(PPDMINETWORKDOWN pInterface)
5877{
5878 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkDown);
5879 PE1KSTATE pThis = pThisCC->pShared;
5880 /* Resume suspended transmission */
5881 STATUS &= ~STATUS_TXOFF;
5882 e1kXmitPending(pThisCC->pDevInsR3, pThis, true /*fOnWorkerThread*/);
5883}
5884
5885/**
5886 * @callback_method_impl{FNPDMTASKDEV,
5887 * Executes e1kXmitPending at the behest of ring-0/raw-mode.}
5888 * @note Not executed on EMT.
5889 */
5890static DECLCALLBACK(void) e1kR3TxTaskCallback(PPDMDEVINS pDevIns, void *pvUser)
5891{
5892 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
5893 E1kLog2(("%s e1kR3TxTaskCallback:\n", pThis->szPrf));
5894
5895 int rc = e1kXmitPending(pDevIns, pThis, false /*fOnWorkerThread*/);
5896 AssertMsg(RT_SUCCESS(rc) || rc == VERR_TRY_AGAIN || rc == VERR_NET_DOWN, ("%Rrc\n", rc));
5897
5898 RT_NOREF(rc, pvUser);
5899}
5900
5901#endif /* IN_RING3 */
5902
5903/**
5904 * Write handler for Transmit Descriptor Tail register.
5905 *
5906 * @param pThis The device state structure.
5907 * @param offset Register offset in memory-mapped frame.
5908 * @param index Register index in register array.
5909 * @param value The value to store.
5910 * @param mask Used to implement partial writes (8 and 16-bit).
5911 * @thread EMT
5912 */
5913static int e1kRegWriteTDT(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
5914{
5915 int rc = e1kRegWriteDefault(pDevIns, pThis, offset, index, value);
5916
5917 /* All descriptors starting with head and not including tail belong to us. */
5918 /* Process them. */
5919 E1kLog2(("%s e1kRegWriteTDT: TDBAL=%08x, TDBAH=%08x, TDLEN=%08x, TDH=%08x, TDT=%08x\n",
5920 pThis->szPrf, TDBAL, TDBAH, TDLEN, TDH, TDT));
5921
5922 /* Compose a temporary TX context, breaking TX CS rule, for debugging purposes. */
5923 /* If we decide to transmit, the TX critical section will be entered later in e1kXmitPending(). */
5924 E1KTXDC txdc;
5925 txdc.tdlen = TDLEN;
5926 txdc.tdh = TDH;
5927 txdc.tdt = TDT;
5928 /* Ignore TDT writes when the link is down. */
5929 if (txdc.tdh != txdc.tdt && (STATUS & STATUS_LU))
5930 {
5931 Log5(("E1000: TDT write: TDH=%08x, TDT=%08x, %d descriptors to process\n", txdc.tdh, txdc.tdt, e1kGetTxLen(&txdc)));
5932 E1kLog(("%s e1kRegWriteTDT: %d descriptors to process\n",
5933 pThis->szPrf, e1kGetTxLen(&txdc)));
5934
5935 /* Transmit pending packets if possible, defer it if we cannot do it
5936 in the current context. */
5937#ifdef E1K_TX_DELAY
5938 rc = e1kCsTxEnter(pThis, VERR_SEM_BUSY);
5939 if (RT_LIKELY(rc == VINF_SUCCESS))
5940 {
5941 if (!PDMDevInsTimerIsActive(pDevIns, pThis->hTXDTimer))
5942 {
5943# ifdef E1K_INT_STATS
5944 pThis->u64ArmedAt = RTTimeNanoTS();
5945# endif
5946 e1kArmTimer(pDevIns, pThis, pThis->hTXDTimer, E1K_TX_DELAY);
5947 }
5948 E1K_INC_ISTAT_CNT(pThis->uStatTxDelayed);
5949 e1kCsTxLeave(pThis);
5950 return rc;
5951 }
5952 /* We failed to enter the TX critical section -- transmit as usual. */
5953#endif /* E1K_TX_DELAY */
5954#ifndef IN_RING3
5955 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
5956 if (!pThisCC->CTX_SUFF(pDrv))
5957 {
5958 PDMDevHlpTaskTrigger(pDevIns, pThis->hTxTask);
5959 rc = VINF_SUCCESS;
5960 }
5961 else
5962#endif
5963 {
5964 rc = e1kXmitPending(pDevIns, pThis, false /*fOnWorkerThread*/);
5965 if (rc == VERR_TRY_AGAIN)
5966 rc = VINF_SUCCESS;
5967#ifndef IN_RING3
5968 else if (rc == VERR_SEM_BUSY)
5969 rc = VINF_IOM_R3_MMIO_WRITE;
5970#endif
5971 AssertRC(rc);
5972 }
5973 }
5974
5975 return rc;
5976}
5977
5978/**
5979 * Write handler for Multicast Table Array registers.
5980 *
5981 * @param pThis The device state structure.
5982 * @param offset Register offset in memory-mapped frame.
5983 * @param index Register index in register array.
5984 * @param value The value to store.
5985 * @thread EMT
5986 */
5987static int e1kRegWriteMTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
5988{
5989 RT_NOREF_PV(pDevIns);
5990 AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->auMTA), VERR_DEV_IO_ERROR);
5991 pThis->auMTA[(offset - g_aE1kRegMap[index].offset) / sizeof(pThis->auMTA[0])] = value;
5992
5993 return VINF_SUCCESS;
5994}
5995
5996/**
5997 * Read handler for Multicast Table Array registers.
5998 *
5999 * @returns VBox status code.
6000 *
6001 * @param pThis The device state structure.
6002 * @param offset Register offset in memory-mapped frame.
6003 * @param index Register index in register array.
6004 * @thread EMT
6005 */
6006static int e1kRegReadMTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
6007{
6008 RT_NOREF_PV(pDevIns);
6009 AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->auMTA), VERR_DEV_IO_ERROR);
6010 *pu32Value = pThis->auMTA[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->auMTA[0])];
6011
6012 return VINF_SUCCESS;
6013}
6014
6015/**
6016 * Write handler for Receive Address registers.
6017 *
6018 * @param pThis The device state structure.
6019 * @param offset Register offset in memory-mapped frame.
6020 * @param index Register index in register array.
6021 * @param value The value to store.
6022 * @thread EMT
6023 */
6024static int e1kRegWriteRA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
6025{
6026 RT_NOREF_PV(pDevIns);
6027 AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->aRecAddr.au32), VERR_DEV_IO_ERROR);
6028 pThis->aRecAddr.au32[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->aRecAddr.au32[0])] = value;
6029
6030 return VINF_SUCCESS;
6031}
6032
6033/**
6034 * Read handler for Receive Address registers.
6035 *
6036 * @returns VBox status code.
6037 *
6038 * @param pThis The device state structure.
6039 * @param offset Register offset in memory-mapped frame.
6040 * @param index Register index in register array.
6041 * @thread EMT
6042 */
6043static int e1kRegReadRA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
6044{
6045 RT_NOREF_PV(pDevIns);
6046 AssertReturn(offset - g_aE1kRegMap[index].offset< sizeof(pThis->aRecAddr.au32), VERR_DEV_IO_ERROR);
6047 *pu32Value = pThis->aRecAddr.au32[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->aRecAddr.au32[0])];
6048
6049 return VINF_SUCCESS;
6050}
6051
6052/**
6053 * Write handler for VLAN Filter Table Array registers.
6054 *
6055 * @param pThis The device state structure.
6056 * @param offset Register offset in memory-mapped frame.
6057 * @param index Register index in register array.
6058 * @param value The value to store.
6059 * @thread EMT
6060 */
6061static int e1kRegWriteVFTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
6062{
6063 RT_NOREF_PV(pDevIns);
6064 AssertReturn(offset - g_aE1kRegMap[index].offset < sizeof(pThis->auVFTA), VINF_SUCCESS);
6065 pThis->auVFTA[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->auVFTA[0])] = value;
6066
6067 return VINF_SUCCESS;
6068}
6069
6070/**
6071 * Read handler for VLAN Filter Table Array registers.
6072 *
6073 * @returns VBox status code.
6074 *
6075 * @param pThis The device state structure.
6076 * @param offset Register offset in memory-mapped frame.
6077 * @param index Register index in register array.
6078 * @thread EMT
6079 */
6080static int e1kRegReadVFTA(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
6081{
6082 RT_NOREF_PV(pDevIns);
6083 AssertReturn(offset - g_aE1kRegMap[index].offset< sizeof(pThis->auVFTA), VERR_DEV_IO_ERROR);
6084 *pu32Value = pThis->auVFTA[(offset - g_aE1kRegMap[index].offset)/sizeof(pThis->auVFTA[0])];
6085
6086 return VINF_SUCCESS;
6087}
6088
6089/**
6090 * Read handler for unimplemented registers.
6091 *
6092 * Merely reports reads from unimplemented registers.
6093 *
6094 * @returns VBox status code.
6095 *
6096 * @param pThis The device state structure.
6097 * @param offset Register offset in memory-mapped frame.
6098 * @param index Register index in register array.
6099 * @thread EMT
6100 */
6101static int e1kRegReadUnimplemented(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
6102{
6103 RT_NOREF(pDevIns, pThis, offset, index);
6104 E1kLog(("%s At %08X read (00000000) attempt from unimplemented register %s (%s)\n",
6105 pThis->szPrf, offset, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6106 *pu32Value = 0;
6107
6108 return VINF_SUCCESS;
6109}
6110
6111/**
6112 * Default register read handler with automatic clear operation.
6113 *
6114 * Retrieves the value of register from register array in device state structure.
6115 * Then resets all bits.
6116 *
6117 * @remarks The 'mask' parameter is simply ignored as masking and shifting is
6118 * done in the caller.
6119 *
6120 * @returns VBox status code.
6121 *
6122 * @param pThis The device state structure.
6123 * @param offset Register offset in memory-mapped frame.
6124 * @param index Register index in register array.
6125 * @thread EMT
6126 */
6127static int e1kRegReadAutoClear(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
6128{
6129 AssertReturn(index < E1K_NUM_OF_32BIT_REGS, VERR_DEV_IO_ERROR);
6130 int rc = e1kRegReadDefault(pDevIns, pThis, offset, index, pu32Value);
6131 pThis->auRegs[index] = 0;
6132
6133 return rc;
6134}
6135
6136/**
6137 * Default register read handler.
6138 *
6139 * Retrieves the value of register from register array in device state structure.
6140 * Bits corresponding to 0s in 'readable' mask will always read as 0s.
6141 *
6142 * @remarks The 'mask' parameter is simply ignored as masking and shifting is
6143 * done in the caller.
6144 *
6145 * @returns VBox status code.
6146 *
6147 * @param pThis The device state structure.
6148 * @param offset Register offset in memory-mapped frame.
6149 * @param index Register index in register array.
6150 * @thread EMT
6151 */
6152static int e1kRegReadDefault(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t *pu32Value)
6153{
6154 RT_NOREF_PV(pDevIns); RT_NOREF_PV(offset);
6155
6156 AssertReturn(index < E1K_NUM_OF_32BIT_REGS, VERR_DEV_IO_ERROR);
6157 *pu32Value = pThis->auRegs[index] & g_aE1kRegMap[index].readable;
6158
6159 return VINF_SUCCESS;
6160}
6161
6162/**
6163 * Write handler for unimplemented registers.
6164 *
6165 * Merely reports writes to unimplemented registers.
6166 *
6167 * @param pThis The device state structure.
6168 * @param offset Register offset in memory-mapped frame.
6169 * @param index Register index in register array.
6170 * @param value The value to store.
6171 * @thread EMT
6172 */
6173
6174 static int e1kRegWriteUnimplemented(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
6175{
6176 RT_NOREF_PV(pDevIns); RT_NOREF_PV(pThis); RT_NOREF_PV(offset); RT_NOREF_PV(index); RT_NOREF_PV(value);
6177
6178 E1kLog(("%s At %08X write attempt (%08X) to unimplemented register %s (%s)\n",
6179 pThis->szPrf, offset, value, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6180
6181 return VINF_SUCCESS;
6182}
6183
6184/**
6185 * Default register write handler.
6186 *
6187 * Stores the value to the register array in device state structure. Only bits
6188 * corresponding to 1s both in 'writable' and 'mask' will be stored.
6189 *
6190 * @returns VBox status code.
6191 *
6192 * @param pThis The device state structure.
6193 * @param offset Register offset in memory-mapped frame.
6194 * @param index Register index in register array.
6195 * @param value The value to store.
6196 * @param mask Used to implement partial writes (8 and 16-bit).
6197 * @thread EMT
6198 */
6199
6200static int e1kRegWriteDefault(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offset, uint32_t index, uint32_t value)
6201{
6202 RT_NOREF(pDevIns, offset);
6203
6204 AssertReturn(index < E1K_NUM_OF_32BIT_REGS, VERR_DEV_IO_ERROR);
6205 pThis->auRegs[index] = (value & g_aE1kRegMap[index].writable)
6206 | (pThis->auRegs[index] & ~g_aE1kRegMap[index].writable);
6207
6208 return VINF_SUCCESS;
6209}
6210
6211/**
6212 * Search register table for matching register.
6213 *
6214 * @returns Index in the register table or -1 if not found.
6215 *
6216 * @param offReg Register offset in memory-mapped region.
6217 * @thread EMT
6218 */
6219static int e1kRegLookup(uint32_t offReg)
6220{
6221
6222#if 0
6223 int index;
6224
6225 for (index = 0; index < E1K_NUM_OF_REGS; index++)
6226 {
6227 if (g_aE1kRegMap[index].offset <= offReg && offReg < g_aE1kRegMap[index].offset + g_aE1kRegMap[index].size)
6228 {
6229 return index;
6230 }
6231 }
6232#else
6233 int iStart = 0;
6234 int iEnd = E1K_NUM_OF_BINARY_SEARCHABLE;
6235 for (;;)
6236 {
6237 int i = (iEnd - iStart) / 2 + iStart;
6238 uint32_t offCur = g_aE1kRegMap[i].offset;
6239 if (offReg < offCur)
6240 {
6241 if (i == iStart)
6242 break;
6243 iEnd = i;
6244 }
6245 else if (offReg >= offCur + g_aE1kRegMap[i].size)
6246 {
6247 i++;
6248 if (i == iEnd)
6249 break;
6250 iStart = i;
6251 }
6252 else
6253 return i;
6254 Assert(iEnd > iStart);
6255 }
6256
6257 for (unsigned i = E1K_NUM_OF_BINARY_SEARCHABLE; i < RT_ELEMENTS(g_aE1kRegMap); i++)
6258 if (offReg - g_aE1kRegMap[i].offset < g_aE1kRegMap[i].size)
6259 return (int)i;
6260
6261# ifdef VBOX_STRICT
6262 for (unsigned i = 0; i < RT_ELEMENTS(g_aE1kRegMap); i++)
6263 Assert(offReg - g_aE1kRegMap[i].offset >= g_aE1kRegMap[i].size);
6264# endif
6265
6266#endif
6267
6268 return -1;
6269}
6270
6271/**
6272 * Handle unaligned register read operation.
6273 *
6274 * Looks up and calls appropriate handler.
6275 *
6276 * @returns VBox status code.
6277 *
6278 * @param pDevIns The device instance.
6279 * @param pThis The device state structure.
6280 * @param offReg Register offset in memory-mapped frame.
6281 * @param pv Where to store the result.
6282 * @param cb Number of bytes to read.
6283 * @thread EMT
6284 * @remarks IOM takes care of unaligned and small reads via MMIO. For I/O port
6285 * accesses we have to take care of that ourselves.
6286 */
6287static int e1kRegReadUnaligned(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offReg, void *pv, uint32_t cb)
6288{
6289 uint32_t u32 = 0;
6290 uint32_t shift;
6291 int rc = VINF_SUCCESS;
6292 int index = e1kRegLookup(offReg);
6293#ifdef LOG_ENABLED
6294 char buf[9];
6295#endif
6296
6297 /*
6298 * From the spec:
6299 * For registers that should be accessed as 32-bit double words, partial writes (less than a 32-bit
6300 * double word) is ignored. Partial reads return all 32 bits of data regardless of the byte enables.
6301 */
6302
6303 /*
6304 * To be able to read bytes and short word we convert them to properly
6305 * shifted 32-bit words and masks. The idea is to keep register-specific
6306 * handlers simple. Most accesses will be 32-bit anyway.
6307 */
6308 uint32_t mask;
6309 switch (cb)
6310 {
6311 case 4: mask = 0xFFFFFFFF; break;
6312 case 2: mask = 0x0000FFFF; break;
6313 case 1: mask = 0x000000FF; break;
6314 default:
6315 return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "unsupported op size: offset=%#10x cb=%#10x\n", offReg, cb);
6316 }
6317 if (index >= 0)
6318 {
6319 RT_UNTRUSTED_VALIDATED_FENCE(); /* paranoia because of port I/O. */
6320 if (g_aE1kRegMap[index].readable)
6321 {
6322 /* Make the mask correspond to the bits we are about to read. */
6323 shift = (offReg - g_aE1kRegMap[index].offset) % sizeof(uint32_t) * 8;
6324 mask <<= shift;
6325 if (!mask)
6326 return PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "Zero mask: offset=%#10x cb=%#10x\n", offReg, cb);
6327 /*
6328 * Read it. Pass the mask so the handler knows what has to be read.
6329 * Mask out irrelevant bits.
6330 */
6331 //e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
6332 //pThis->fDelayInts = false;
6333 //pThis->iStatIntLost += pThis->iStatIntLostOne;
6334 //pThis->iStatIntLostOne = 0;
6335 rc = g_aE1kRegMap[index].pfnRead(pDevIns, pThis, offReg & 0xFFFFFFFC, (uint32_t)index, &u32);
6336 u32 &= mask;
6337 //e1kCsLeave(pThis);
6338 E1kLog2(("%s At %08X read %s from %s (%s)\n",
6339 pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf), g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6340 Log6(("%s At %08X read %s from %s (%s) [UNALIGNED]\n",
6341 pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf), g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6342 /* Shift back the result. */
6343 u32 >>= shift;
6344 }
6345 else
6346 E1kLog(("%s At %08X read (%s) attempt from write-only register %s (%s)\n",
6347 pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf), g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6348 if (IOM_SUCCESS(rc))
6349 STAM_COUNTER_INC(&pThis->aStatRegReads[index]);
6350 }
6351 else
6352 E1kLog(("%s At %08X read (%s) attempt from non-existing register\n",
6353 pThis->szPrf, offReg, e1kU32toHex(u32, mask, buf)));
6354
6355 memcpy(pv, &u32, cb);
6356 return rc;
6357}
6358
6359/**
6360 * Handle 4 byte aligned and sized read operation.
6361 *
6362 * Looks up and calls appropriate handler.
6363 *
6364 * @returns VBox status code.
6365 *
6366 * @param pDevIns The device instance.
6367 * @param pThis The device state structure.
6368 * @param offReg Register offset in memory-mapped frame.
6369 * @param pu32 Where to store the result.
6370 * @thread EMT
6371 */
6372static VBOXSTRICTRC e1kRegReadAlignedU32(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offReg, uint32_t *pu32)
6373{
6374 Assert(!(offReg & 3));
6375
6376 /*
6377 * Lookup the register and check that it's readable.
6378 */
6379 VBOXSTRICTRC rc = VINF_SUCCESS;
6380 int idxReg = e1kRegLookup(offReg);
6381 if (RT_LIKELY(idxReg >= 0))
6382 {
6383 RT_UNTRUSTED_VALIDATED_FENCE(); /* paranoia because of port I/O. */
6384 if (RT_UNLIKELY(g_aE1kRegMap[idxReg].readable))
6385 {
6386 /*
6387 * Read it. Pass the mask so the handler knows what has to be read.
6388 * Mask out irrelevant bits.
6389 */
6390 //e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
6391 //pThis->fDelayInts = false;
6392 //pThis->iStatIntLost += pThis->iStatIntLostOne;
6393 //pThis->iStatIntLostOne = 0;
6394 rc = g_aE1kRegMap[idxReg].pfnRead(pDevIns, pThis, offReg & 0xFFFFFFFC, (uint32_t)idxReg, pu32);
6395 //e1kCsLeave(pThis);
6396 Log6(("%s At %08X read %08X from %s (%s)\n",
6397 pThis->szPrf, offReg, *pu32, g_aE1kRegMap[idxReg].abbrev, g_aE1kRegMap[idxReg].name));
6398 if (IOM_SUCCESS(rc))
6399 STAM_COUNTER_INC(&pThis->aStatRegReads[idxReg]);
6400 }
6401 else
6402 E1kLog(("%s At %08X read attempt from non-readable register %s (%s)\n",
6403 pThis->szPrf, offReg, g_aE1kRegMap[idxReg].abbrev, g_aE1kRegMap[idxReg].name));
6404 }
6405 else
6406 E1kLog(("%s At %08X read attempt from non-existing register\n", pThis->szPrf, offReg));
6407 return rc;
6408}
6409
6410/**
6411 * Handle 4 byte sized and aligned register write operation.
6412 *
6413 * Looks up and calls appropriate handler.
6414 *
6415 * @returns VBox status code.
6416 *
6417 * @param pDevIns The device instance.
6418 * @param pThis The device state structure.
6419 * @param offReg Register offset in memory-mapped frame.
6420 * @param u32Value The value to write.
6421 * @thread EMT
6422 */
6423static VBOXSTRICTRC e1kRegWriteAlignedU32(PPDMDEVINS pDevIns, PE1KSTATE pThis, uint32_t offReg, uint32_t u32Value)
6424{
6425 VBOXSTRICTRC rc = VINF_SUCCESS;
6426 int index = e1kRegLookup(offReg);
6427 if (RT_LIKELY(index >= 0))
6428 {
6429 RT_UNTRUSTED_VALIDATED_FENCE(); /* paranoia because of port I/O. */
6430 if (RT_LIKELY(g_aE1kRegMap[index].writable))
6431 {
6432 /*
6433 * Write it. Pass the mask so the handler knows what has to be written.
6434 * Mask out irrelevant bits.
6435 */
6436 Log6(("%s At %08X write %08X to %s (%s)\n",
6437 pThis->szPrf, offReg, u32Value, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6438 //e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
6439 //pThis->fDelayInts = false;
6440 //pThis->iStatIntLost += pThis->iStatIntLostOne;
6441 //pThis->iStatIntLostOne = 0;
6442 rc = g_aE1kRegMap[index].pfnWrite(pDevIns, pThis, offReg, (uint32_t)index, u32Value);
6443 //e1kCsLeave(pThis);
6444 }
6445 else
6446 E1kLog(("%s At %08X write attempt (%08X) to read-only register %s (%s)\n",
6447 pThis->szPrf, offReg, u32Value, g_aE1kRegMap[index].abbrev, g_aE1kRegMap[index].name));
6448 if (IOM_SUCCESS(rc))
6449 STAM_COUNTER_INC(&pThis->aStatRegWrites[index]);
6450 }
6451 else
6452 E1kLog(("%s At %08X write attempt (%08X) to non-existing register\n",
6453 pThis->szPrf, offReg, u32Value));
6454 return rc;
6455}
6456
6457
6458/* -=-=-=-=- MMIO and I/O Port Callbacks -=-=-=-=- */
6459
6460/**
6461 * @callback_method_impl{FNIOMMMIONEWREAD}
6462 */
6463static DECLCALLBACK(VBOXSTRICTRC) e1kMMIORead(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void *pv, uint32_t cb)
6464{
6465 RT_NOREF2(pvUser, cb);
6466 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
6467 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatMMIORead), a);
6468
6469 Assert(off < E1K_MM_SIZE);
6470 Assert(cb == 4);
6471 Assert(!(off & 3));
6472
6473 VBOXSTRICTRC rcStrict = e1kRegReadAlignedU32(pDevIns, pThis, (uint32_t)off, (uint32_t *)pv);
6474
6475 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatMMIORead), a);
6476 return rcStrict;
6477}
6478
6479/**
6480 * @callback_method_impl{FNIOMMMIONEWWRITE}
6481 */
6482static DECLCALLBACK(VBOXSTRICTRC) e1kMMIOWrite(PPDMDEVINS pDevIns, void *pvUser, RTGCPHYS off, void const *pv, uint32_t cb)
6483{
6484 RT_NOREF2(pvUser, cb);
6485 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
6486 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatMMIOWrite), a);
6487
6488 Assert(off < E1K_MM_SIZE);
6489 Assert(cb == 4);
6490 Assert(!(off & 3));
6491
6492 VBOXSTRICTRC rcStrict = e1kRegWriteAlignedU32(pDevIns, pThis, (uint32_t)off, *(uint32_t const *)pv);
6493
6494 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatMMIOWrite), a);
6495 return rcStrict;
6496}
6497
6498/**
6499 * @callback_method_impl{FNIOMIOPORTNEWIN}
6500 */
6501static DECLCALLBACK(VBOXSTRICTRC) e1kIOPortIn(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t *pu32, unsigned cb)
6502{
6503 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
6504 VBOXSTRICTRC rc;
6505 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIORead), a);
6506 RT_NOREF_PV(pvUser);
6507
6508 if (RT_LIKELY(cb == 4))
6509 switch (offPort)
6510 {
6511 case 0x00: /* IOADDR */
6512 *pu32 = pThis->uSelectedReg;
6513 Log9(("%s e1kIOPortIn: IOADDR(0), selecting register %#010x, val=%#010x\n", pThis->szPrf, pThis->uSelectedReg, *pu32));
6514 rc = VINF_SUCCESS;
6515 break;
6516
6517 case 0x04: /* IODATA */
6518 if (!(pThis->uSelectedReg & 3))
6519 rc = e1kRegReadAlignedU32(pDevIns, pThis, pThis->uSelectedReg, pu32);
6520 else /** @todo r=bird: I wouldn't be surprised if this unaligned branch wasn't necessary. */
6521 rc = e1kRegReadUnaligned(pDevIns, pThis, pThis->uSelectedReg, pu32, cb);
6522 if (rc == VINF_IOM_R3_MMIO_READ)
6523 rc = VINF_IOM_R3_IOPORT_READ;
6524 Log9(("%s e1kIOPortIn: IODATA(4), reading from selected register %#010x, val=%#010x\n", pThis->szPrf, pThis->uSelectedReg, *pu32));
6525 break;
6526
6527 default:
6528 E1kLog(("%s e1kIOPortIn: invalid port %#010x\n", pThis->szPrf, offPort));
6529 /** @todo r=bird: Check what real hardware returns here. */
6530 //rc = VERR_IOM_IOPORT_UNUSED; /* Why not? */
6531 rc = VINF_IOM_MMIO_UNUSED_00; /* used to return VINF_SUCCESS and not touch *pu32, which amounted to this. */
6532 break;
6533 }
6534 else
6535 {
6536 E1kLog(("%s e1kIOPortIn: invalid op size: offPort=%RTiop cb=%08x", pThis->szPrf, offPort, cb));
6537 rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "%s e1kIOPortIn: invalid op size: offPort=%RTiop cb=%08x\n", pThis->szPrf, offPort, cb);
6538 *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.) */
6539 }
6540 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIORead), a);
6541 return rc;
6542}
6543
6544
6545/**
6546 * @callback_method_impl{FNIOMIOPORTNEWOUT}
6547 */
6548static DECLCALLBACK(VBOXSTRICTRC) e1kIOPortOut(PPDMDEVINS pDevIns, void *pvUser, RTIOPORT offPort, uint32_t u32, unsigned cb)
6549{
6550 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
6551 VBOXSTRICTRC rc;
6552 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIOWrite), a);
6553 RT_NOREF_PV(pvUser);
6554
6555 Log9(("%s e1kIOPortOut: offPort=%RTiop value=%08x\n", pThis->szPrf, offPort, u32));
6556 if (RT_LIKELY(cb == 4))
6557 {
6558 switch (offPort)
6559 {
6560 case 0x00: /* IOADDR */
6561 pThis->uSelectedReg = u32;
6562 Log9(("%s e1kIOPortOut: IOADDR(0), selected register %08x\n", pThis->szPrf, pThis->uSelectedReg));
6563 rc = VINF_SUCCESS;
6564 break;
6565
6566 case 0x04: /* IODATA */
6567 Log9(("%s e1kIOPortOut: IODATA(4), writing to selected register %#010x, value=%#010x\n", pThis->szPrf, pThis->uSelectedReg, u32));
6568 if (RT_LIKELY(!(pThis->uSelectedReg & 3)))
6569 {
6570 rc = e1kRegWriteAlignedU32(pDevIns, pThis, pThis->uSelectedReg, u32);
6571 if (rc == VINF_IOM_R3_MMIO_WRITE)
6572 rc = VINF_IOM_R3_IOPORT_WRITE;
6573 }
6574 else
6575 rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS,
6576 "Spec violation: misaligned offset: %#10x, ignored.\n", pThis->uSelectedReg);
6577 break;
6578
6579 default:
6580 E1kLog(("%s e1kIOPortOut: invalid port %#010x\n", pThis->szPrf, offPort));
6581 rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "invalid port %#010x\n", offPort);
6582 }
6583 }
6584 else
6585 {
6586 E1kLog(("%s e1kIOPortOut: invalid op size: offPort=%RTiop cb=%08x\n", pThis->szPrf, offPort, cb));
6587 rc = PDMDevHlpDBGFStop(pDevIns, RT_SRC_POS, "%s: invalid op size: offPort=%RTiop cb=%#x\n", pThis->szPrf, offPort, cb);
6588 }
6589
6590 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIOWrite), a);
6591 return rc;
6592}
6593
6594#ifdef IN_RING3
6595
6596/**
6597 * Dump complete device state to log.
6598 *
6599 * @param pThis Pointer to device state.
6600 */
6601static void e1kDumpState(PE1KSTATE pThis)
6602{
6603 RT_NOREF(pThis);
6604 for (int i = 0; i < E1K_NUM_OF_32BIT_REGS; ++i)
6605 E1kLog2(("%s: %8.8s = %08x\n", pThis->szPrf, g_aE1kRegMap[i].abbrev, pThis->auRegs[i]));
6606# ifdef E1K_INT_STATS
6607 LogRel(("%s: Interrupt attempts: %d\n", pThis->szPrf, pThis->uStatIntTry));
6608 LogRel(("%s: Interrupts raised : %d\n", pThis->szPrf, pThis->uStatInt));
6609 LogRel(("%s: Interrupts lowered: %d\n", pThis->szPrf, pThis->uStatIntLower));
6610 LogRel(("%s: ICR outside ISR : %d\n", pThis->szPrf, pThis->uStatNoIntICR));
6611 LogRel(("%s: IMS raised ints : %d\n", pThis->szPrf, pThis->uStatIntIMS));
6612 LogRel(("%s: Interrupts skipped: %d\n", pThis->szPrf, pThis->uStatIntSkip));
6613 LogRel(("%s: Masked interrupts : %d\n", pThis->szPrf, pThis->uStatIntMasked));
6614 LogRel(("%s: Early interrupts : %d\n", pThis->szPrf, pThis->uStatIntEarly));
6615 LogRel(("%s: Late interrupts : %d\n", pThis->szPrf, pThis->uStatIntLate));
6616 LogRel(("%s: Lost interrupts : %d\n", pThis->szPrf, pThis->iStatIntLost));
6617 LogRel(("%s: Interrupts by RX : %d\n", pThis->szPrf, pThis->uStatIntRx));
6618 LogRel(("%s: Interrupts by TX : %d\n", pThis->szPrf, pThis->uStatIntTx));
6619 LogRel(("%s: Interrupts by ICS : %d\n", pThis->szPrf, pThis->uStatIntICS));
6620 LogRel(("%s: Interrupts by RDTR: %d\n", pThis->szPrf, pThis->uStatIntRDTR));
6621 LogRel(("%s: Interrupts by RDMT: %d\n", pThis->szPrf, pThis->uStatIntRXDMT0));
6622 LogRel(("%s: Interrupts by TXQE: %d\n", pThis->szPrf, pThis->uStatIntTXQE));
6623 LogRel(("%s: TX int delay asked: %d\n", pThis->szPrf, pThis->uStatTxIDE));
6624 LogRel(("%s: TX delayed: %d\n", pThis->szPrf, pThis->uStatTxDelayed));
6625 LogRel(("%s: TX delay expired: %d\n", pThis->szPrf, pThis->uStatTxDelayExp));
6626 LogRel(("%s: TX no report asked: %d\n", pThis->szPrf, pThis->uStatTxNoRS));
6627 LogRel(("%s: TX abs timer expd : %d\n", pThis->szPrf, pThis->uStatTAD));
6628 LogRel(("%s: TX int timer expd : %d\n", pThis->szPrf, pThis->uStatTID));
6629 LogRel(("%s: RX abs timer expd : %d\n", pThis->szPrf, pThis->uStatRAD));
6630 LogRel(("%s: RX int timer expd : %d\n", pThis->szPrf, pThis->uStatRID));
6631 LogRel(("%s: TX CTX descriptors: %d\n", pThis->szPrf, pThis->uStatDescCtx));
6632 LogRel(("%s: TX DAT descriptors: %d\n", pThis->szPrf, pThis->uStatDescDat));
6633 LogRel(("%s: TX LEG descriptors: %d\n", pThis->szPrf, pThis->uStatDescLeg));
6634 LogRel(("%s: Received frames : %d\n", pThis->szPrf, pThis->uStatRxFrm));
6635 LogRel(("%s: Transmitted frames: %d\n", pThis->szPrf, pThis->uStatTxFrm));
6636 LogRel(("%s: TX frames up to 1514: %d\n", pThis->szPrf, pThis->uStatTx1514));
6637 LogRel(("%s: TX frames up to 2962: %d\n", pThis->szPrf, pThis->uStatTx2962));
6638 LogRel(("%s: TX frames up to 4410: %d\n", pThis->szPrf, pThis->uStatTx4410));
6639 LogRel(("%s: TX frames up to 5858: %d\n", pThis->szPrf, pThis->uStatTx5858));
6640 LogRel(("%s: TX frames up to 7306: %d\n", pThis->szPrf, pThis->uStatTx7306));
6641 LogRel(("%s: TX frames up to 8754: %d\n", pThis->szPrf, pThis->uStatTx8754));
6642 LogRel(("%s: TX frames up to 16384: %d\n", pThis->szPrf, pThis->uStatTx16384));
6643 LogRel(("%s: TX frames up to 32768: %d\n", pThis->szPrf, pThis->uStatTx32768));
6644 LogRel(("%s: Larger TX frames : %d\n", pThis->szPrf, pThis->uStatTxLarge));
6645 LogRel(("%s: Max TX Delay : %lld\n", pThis->szPrf, pThis->uStatMaxTxDelay));
6646# endif /* E1K_INT_STATS */
6647}
6648
6649
6650/* -=-=-=-=- PDMINETWORKDOWN -=-=-=-=- */
6651
6652/**
6653 * Check if the device can receive data now.
6654 * This must be called before the pfnRecieve() method is called.
6655 *
6656 * @returns VBox status code.
6657 * @retval VERR_NET_NO_BUFFER_SPACE if we cannot receive.
6658 * @param pDevIns The device instance.
6659 * @param pThis The instance data.
6660 * @thread EMT
6661 */
6662static int e1kR3CanReceive(PPDMDEVINS pDevIns, PE1KSTATE pThis)
6663{
6664# ifndef E1K_WITH_RXD_CACHE
6665 size_t cb;
6666
6667 e1kCsRxEnterReturn(pThis);
6668
6669 if (RT_UNLIKELY(RDLEN == sizeof(E1KRXDESC)))
6670 {
6671 E1KRXDESC desc;
6672 PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, RDH), &desc, sizeof(desc));
6673 if (desc.status.fDD)
6674 cb = 0;
6675 else
6676 cb = pThis->u16RxBSize;
6677 }
6678 else if (RDH < RDT)
6679 cb = (RDT - RDH) * pThis->u16RxBSize;
6680 else if (RDH > RDT)
6681 cb = (RDLEN / sizeof(E1KRXDESC) - RDH + RDT) * pThis->u16RxBSize;
6682 else
6683 {
6684 cb = 0;
6685 E1kLogRel(("E1000: OUT of RX descriptors!\n"));
6686 }
6687 E1kLog2(("%s e1kR3CanReceive: at exit RDH=%d RDT=%d RDLEN=%d u16RxBSize=%d cb=%lu\n",
6688 pThis->szPrf, RDH, RDT, RDLEN, pThis->u16RxBSize, cb));
6689
6690 e1kCsRxLeave(pThis);
6691 return cb > 0 ? VINF_SUCCESS : VERR_NET_NO_BUFFER_SPACE;
6692# else /* E1K_WITH_RXD_CACHE */
6693
6694 e1kCsRxEnterReturn(pThis);
6695
6696 E1KRXDC rxdc;
6697 if (RT_UNLIKELY(!e1kUpdateRxDContext(pDevIns, pThis, &rxdc, "e1kR3CanReceive")))
6698 {
6699 e1kCsRxLeave(pThis);
6700 E1kLog(("%s e1kR3CanReceive: failed to update Rx context, returning VERR_NET_NO_BUFFER_SPACE\n", pThis->szPrf));
6701 return VERR_NET_NO_BUFFER_SPACE;
6702 }
6703
6704 int rc = VINF_SUCCESS;
6705 if (RT_UNLIKELY(rxdc.rdlen == sizeof(E1KRXDESC)))
6706 {
6707 E1KRXDESC desc;
6708 PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, rxdc.rdh), &desc, sizeof(desc));
6709 if (desc.status.fDD)
6710 rc = VERR_NET_NO_BUFFER_SPACE;
6711 }
6712 else if (e1kRxDIsCacheEmpty(pThis) && rxdc.rdh == rxdc.rdt)
6713 {
6714 /* Cache is empty, so is the RX ring. */
6715 rc = VERR_NET_NO_BUFFER_SPACE;
6716 }
6717 E1kLog2(("%s e1kR3CanReceive: at exit in_cache=%d RDH=%d RDT=%d RDLEN=%d u16RxBSize=%d rc=%Rrc\n", pThis->szPrf,
6718 e1kRxDInCache(pThis), rxdc.rdh, rxdc.rdt, rxdc.rdlen, pThis->u16RxBSize, rc));
6719
6720 e1kCsRxLeave(pThis);
6721 return rc;
6722# endif /* E1K_WITH_RXD_CACHE */
6723}
6724
6725/**
6726 * @interface_method_impl{PDMINETWORKDOWN,pfnWaitReceiveAvail}
6727 */
6728static DECLCALLBACK(int) e1kR3NetworkDown_WaitReceiveAvail(PPDMINETWORKDOWN pInterface, RTMSINTERVAL cMillies)
6729{
6730 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkDown);
6731 PE1KSTATE pThis = pThisCC->pShared;
6732 PPDMDEVINS pDevIns = pThisCC->pDevInsR3;
6733
6734 int rc = e1kR3CanReceive(pDevIns, pThis);
6735 if (RT_SUCCESS(rc))
6736 return VINF_SUCCESS;
6737
6738 if (RT_UNLIKELY(cMillies == 0))
6739 return VERR_NET_NO_BUFFER_SPACE;
6740
6741 rc = VERR_INTERRUPTED;
6742 ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, true);
6743 STAM_PROFILE_START(&pThis->StatRxOverflow, a);
6744 VMSTATE enmVMState;
6745 while (RT_LIKELY( (enmVMState = PDMDevHlpVMState(pDevIns)) == VMSTATE_RUNNING
6746 || enmVMState == VMSTATE_RUNNING_LS))
6747 {
6748 int rc2 = e1kR3CanReceive(pDevIns, pThis);
6749 if (RT_SUCCESS(rc2))
6750 {
6751 rc = VINF_SUCCESS;
6752 break;
6753 }
6754 E1kLogRel(("E1000: e1kR3NetworkDown_WaitReceiveAvail: waiting cMillies=%u...\n", cMillies));
6755 E1kLog(("%s: e1kR3NetworkDown_WaitReceiveAvail: waiting cMillies=%u...\n", pThis->szPrf, cMillies));
6756 PDMDevHlpSUPSemEventWaitNoResume(pDevIns, pThis->hEventMoreRxDescAvail, cMillies);
6757 }
6758 STAM_PROFILE_STOP(&pThis->StatRxOverflow, a);
6759 ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, false);
6760
6761 return rc;
6762}
6763
6764
6765/**
6766 * Matches the packet addresses against Receive Address table. Looks for
6767 * exact matches only.
6768 *
6769 * @returns true if address matches.
6770 * @param pThis Pointer to the state structure.
6771 * @param pvBuf The ethernet packet.
6772 * @param cb Number of bytes available in the packet.
6773 * @thread EMT
6774 */
6775static bool e1kPerfectMatch(PE1KSTATE pThis, const void *pvBuf)
6776{
6777 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aRecAddr.array); i++)
6778 {
6779 E1KRAELEM* ra = pThis->aRecAddr.array + i;
6780
6781 /* Valid address? */
6782 if (ra->ctl & RA_CTL_AV)
6783 {
6784 Assert((ra->ctl & RA_CTL_AS) < 2);
6785 //unsigned char *pAddr = (unsigned char*)pvBuf + sizeof(ra->addr)*(ra->ctl & RA_CTL_AS);
6786 //E1kLog3(("%s Matching %02x:%02x:%02x:%02x:%02x:%02x against %02x:%02x:%02x:%02x:%02x:%02x...\n",
6787 // pThis->szPrf, pAddr[0], pAddr[1], pAddr[2], pAddr[3], pAddr[4], pAddr[5],
6788 // ra->addr[0], ra->addr[1], ra->addr[2], ra->addr[3], ra->addr[4], ra->addr[5]));
6789 /*
6790 * Address Select:
6791 * 00b = Destination address
6792 * 01b = Source address
6793 * 10b = Reserved
6794 * 11b = Reserved
6795 * Since ethernet header is (DA, SA, len) we can use address
6796 * select as index.
6797 */
6798 if (memcmp((char*)pvBuf + sizeof(ra->addr)*(ra->ctl & RA_CTL_AS),
6799 ra->addr, sizeof(ra->addr)) == 0)
6800 return true;
6801 }
6802 }
6803
6804 return false;
6805}
6806
6807/**
6808 * Matches the packet addresses against Multicast Table Array.
6809 *
6810 * @remarks This is imperfect match since it matches not exact address but
6811 * a subset of addresses.
6812 *
6813 * @returns true if address matches.
6814 * @param pThis Pointer to the state structure.
6815 * @param pvBuf The ethernet packet.
6816 * @param cb Number of bytes available in the packet.
6817 * @thread EMT
6818 */
6819static bool e1kImperfectMatch(PE1KSTATE pThis, const void *pvBuf)
6820{
6821 /* Get bits 32..47 of destination address */
6822 uint16_t u16Bit = ((uint16_t*)pvBuf)[2];
6823
6824 unsigned offset = GET_BITS(RCTL, MO);
6825 /*
6826 * offset means:
6827 * 00b = bits 36..47
6828 * 01b = bits 35..46
6829 * 10b = bits 34..45
6830 * 11b = bits 32..43
6831 */
6832 if (offset < 3)
6833 u16Bit = u16Bit >> (4 - offset);
6834 return ASMBitTest(pThis->auMTA, u16Bit & 0xFFF);
6835}
6836
6837/**
6838 * Determines if the packet is to be delivered to upper layer.
6839 *
6840 * The following filters supported:
6841 * - Exact Unicast/Multicast
6842 * - Promiscuous Unicast/Multicast
6843 * - Multicast
6844 * - VLAN
6845 *
6846 * @returns true if packet is intended for this node.
6847 * @param pThis Pointer to the state structure.
6848 * @param pvBuf The ethernet packet.
6849 * @param cb Number of bytes available in the packet.
6850 * @param pStatus Bit field to store status bits.
6851 * @thread EMT
6852 */
6853static bool e1kAddressFilter(PE1KSTATE pThis, const void *pvBuf, size_t cb, E1KRXDST *pStatus)
6854{
6855 Assert(cb > 14);
6856 /* Assume that we fail to pass exact filter. */
6857 pStatus->fPIF = false;
6858 pStatus->fVP = false;
6859 /* Discard oversized packets */
6860 if (cb > E1K_MAX_RX_PKT_SIZE)
6861 {
6862 E1kLog(("%s ERROR: Incoming packet is too big, cb=%d > max=%d\n",
6863 pThis->szPrf, cb, E1K_MAX_RX_PKT_SIZE));
6864 E1K_INC_CNT32(ROC);
6865 return false;
6866 }
6867 else if (!(RCTL & RCTL_LPE) && cb > 1522)
6868 {
6869 /* When long packet reception is disabled packets over 1522 are discarded */
6870 E1kLog(("%s Discarding incoming packet (LPE=0), cb=%d\n",
6871 pThis->szPrf, cb));
6872 E1K_INC_CNT32(ROC);
6873 return false;
6874 }
6875
6876 uint16_t *u16Ptr = (uint16_t*)pvBuf;
6877 /* Compare TPID with VLAN Ether Type */
6878 if (RT_BE2H_U16(u16Ptr[6]) == VET)
6879 {
6880 pStatus->fVP = true;
6881 /* Is VLAN filtering enabled? */
6882 if (RCTL & RCTL_VFE)
6883 {
6884 /* It is 802.1q packet indeed, let's filter by VID */
6885 if (RCTL & RCTL_CFIEN)
6886 {
6887 E1kLog3(("%s VLAN filter: VLAN=%d CFI=%d RCTL_CFI=%d\n", pThis->szPrf,
6888 E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7])),
6889 E1K_SPEC_CFI(RT_BE2H_U16(u16Ptr[7])),
6890 !!(RCTL & RCTL_CFI)));
6891 if (E1K_SPEC_CFI(RT_BE2H_U16(u16Ptr[7])) != !!(RCTL & RCTL_CFI))
6892 {
6893 E1kLog2(("%s Packet filter: CFIs do not match in packet and RCTL (%d!=%d)\n",
6894 pThis->szPrf, E1K_SPEC_CFI(RT_BE2H_U16(u16Ptr[7])), !!(RCTL & RCTL_CFI)));
6895 return false;
6896 }
6897 }
6898 else
6899 E1kLog3(("%s VLAN filter: VLAN=%d\n", pThis->szPrf,
6900 E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7]))));
6901 if (!ASMBitTest(pThis->auVFTA, E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7]))))
6902 {
6903 E1kLog2(("%s Packet filter: no VLAN match (id=%d)\n",
6904 pThis->szPrf, E1K_SPEC_VLAN(RT_BE2H_U16(u16Ptr[7]))));
6905 return false;
6906 }
6907 }
6908 }
6909 /* Broadcast filtering */
6910 if (e1kIsBroadcast(pvBuf) && (RCTL & RCTL_BAM))
6911 return true;
6912 E1kLog2(("%s Packet filter: not a broadcast\n", pThis->szPrf));
6913 if (e1kIsMulticast(pvBuf))
6914 {
6915 /* Is multicast promiscuous enabled? */
6916 if (RCTL & RCTL_MPE)
6917 return true;
6918 E1kLog2(("%s Packet filter: no promiscuous multicast\n", pThis->szPrf));
6919 /* Try perfect matches first */
6920 if (e1kPerfectMatch(pThis, pvBuf))
6921 {
6922 pStatus->fPIF = true;
6923 return true;
6924 }
6925 E1kLog2(("%s Packet filter: no perfect match\n", pThis->szPrf));
6926 if (e1kImperfectMatch(pThis, pvBuf))
6927 return true;
6928 E1kLog2(("%s Packet filter: no imperfect match\n", pThis->szPrf));
6929 }
6930 else {
6931 /* Is unicast promiscuous enabled? */
6932 if (RCTL & RCTL_UPE)
6933 return true;
6934 E1kLog2(("%s Packet filter: no promiscuous unicast\n", pThis->szPrf));
6935 if (e1kPerfectMatch(pThis, pvBuf))
6936 {
6937 pStatus->fPIF = true;
6938 return true;
6939 }
6940 E1kLog2(("%s Packet filter: no perfect match\n", pThis->szPrf));
6941 }
6942 E1kLog2(("%s Packet filter: packet discarded\n", pThis->szPrf));
6943 return false;
6944}
6945
6946/**
6947 * @interface_method_impl{PDMINETWORKDOWN,pfnReceive}
6948 */
6949static DECLCALLBACK(int) e1kR3NetworkDown_Receive(PPDMINETWORKDOWN pInterface, const void *pvBuf, size_t cb)
6950{
6951 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkDown);
6952 PE1KSTATE pThis = pThisCC->pShared;
6953 PPDMDEVINS pDevIns = pThisCC->pDevInsR3;
6954 int rc = VINF_SUCCESS;
6955
6956 /*
6957 * Drop packets if the VM is not running yet/anymore.
6958 */
6959 VMSTATE enmVMState = PDMDevHlpVMState(pDevIns);
6960 if ( enmVMState != VMSTATE_RUNNING
6961 && enmVMState != VMSTATE_RUNNING_LS)
6962 {
6963 E1kLog(("%s Dropping incoming packet as VM is not running.\n", pThis->szPrf));
6964 return VINF_SUCCESS;
6965 }
6966
6967 /* Discard incoming packets in locked state */
6968 if (!(RCTL & RCTL_EN) || pThis->fLocked || !(STATUS & STATUS_LU))
6969 {
6970 E1kLog(("%s Dropping incoming packet as receive operation is disabled.\n", pThis->szPrf));
6971 return VINF_SUCCESS;
6972 }
6973
6974 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
6975
6976 //e1kR3CsEnterAsserted(pThis);
6977
6978 e1kPacketDump(pDevIns, pThis, (const uint8_t*)pvBuf, cb, "<-- Incoming");
6979
6980 /* Update stats */
6981 e1kR3CsEnterAsserted(pThis);
6982 E1K_INC_CNT32(TPR);
6983 E1K_ADD_CNT64(TORL, TORH, cb < 64? 64 : cb);
6984 e1kCsLeave(pThis);
6985
6986 STAM_PROFILE_ADV_START(&pThis->StatReceiveFilter, a);
6987 E1KRXDST status;
6988 RT_ZERO(status);
6989 bool fPassed = e1kAddressFilter(pThis, pvBuf, cb, &status);
6990 STAM_PROFILE_ADV_STOP(&pThis->StatReceiveFilter, a);
6991 if (fPassed)
6992 {
6993 rc = e1kHandleRxPacket(pDevIns, pThis, pvBuf, cb, status);
6994 }
6995 //e1kCsLeave(pThis);
6996 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
6997
6998 return rc;
6999}
7000
7001
7002/* -=-=-=-=- PDMILEDPORTS -=-=-=-=- */
7003
7004/**
7005 * @interface_method_impl{PDMILEDPORTS,pfnQueryStatusLed}
7006 */
7007static DECLCALLBACK(int) e1kR3QueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
7008{
7009 if (iLUN == 0)
7010 {
7011 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, ILeds);
7012 *ppLed = &pThisCC->pShared->led;
7013 return VINF_SUCCESS;
7014 }
7015 return VERR_PDM_LUN_NOT_FOUND;
7016}
7017
7018
7019/* -=-=-=-=- PDMINETWORKCONFIG -=-=-=-=- */
7020
7021/**
7022 * @interface_method_impl{PDMINETWORKCONFIG,pfnGetMac}
7023 */
7024static DECLCALLBACK(int) e1kR3GetMac(PPDMINETWORKCONFIG pInterface, PRTMAC pMac)
7025{
7026 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkConfig);
7027 pThisCC->eeprom.getMac(pMac);
7028 return VINF_SUCCESS;
7029}
7030
7031/**
7032 * @interface_method_impl{PDMINETWORKCONFIG,pfnGetLinkState}
7033 */
7034static DECLCALLBACK(PDMNETWORKLINKSTATE) e1kR3GetLinkState(PPDMINETWORKCONFIG pInterface)
7035{
7036 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkConfig);
7037 PE1KSTATE pThis = pThisCC->pShared;
7038 if (STATUS & STATUS_LU)
7039 return PDMNETWORKLINKSTATE_UP;
7040 return PDMNETWORKLINKSTATE_DOWN;
7041}
7042
7043/**
7044 * @interface_method_impl{PDMINETWORKCONFIG,pfnSetLinkState}
7045 */
7046static DECLCALLBACK(int) e1kR3SetLinkState(PPDMINETWORKCONFIG pInterface, PDMNETWORKLINKSTATE enmState)
7047{
7048 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, INetworkConfig);
7049 PE1KSTATE pThis = pThisCC->pShared;
7050 PPDMDEVINS pDevIns = pThisCC->pDevInsR3;
7051
7052 E1kLog(("%s e1kR3SetLinkState: enmState=%d\n", pThis->szPrf, enmState));
7053 switch (enmState)
7054 {
7055 case PDMNETWORKLINKSTATE_UP:
7056 pThis->fCableConnected = true;
7057 /* If link was down, bring it up after a while. */
7058 if (!(STATUS & STATUS_LU))
7059 e1kBringLinkUpDelayed(pDevIns, pThis);
7060 break;
7061 case PDMNETWORKLINKSTATE_DOWN:
7062 pThis->fCableConnected = false;
7063 /* Always set the phy link state to down, regardless of the STATUS_LU bit.
7064 * We might have to set the link state before the driver initializes us. */
7065 Phy::setLinkStatus(&pThis->phy, false);
7066 /* If link was up, bring it down. */
7067 if (STATUS & STATUS_LU)
7068 e1kR3LinkDown(pDevIns, pThis, pThisCC);
7069 break;
7070 case PDMNETWORKLINKSTATE_DOWN_RESUME:
7071 /*
7072 * There is not much sense in bringing down the link if it has not come up yet.
7073 * If it is up though, we bring it down temporarely, then bring it up again.
7074 */
7075 if (STATUS & STATUS_LU)
7076 e1kR3LinkDownTemp(pDevIns, pThis, pThisCC);
7077 break;
7078 default:
7079 ;
7080 }
7081 return VINF_SUCCESS;
7082}
7083
7084
7085/* -=-=-=-=- PDMIBASE -=-=-=-=- */
7086
7087/**
7088 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
7089 */
7090static DECLCALLBACK(void *) e1kR3QueryInterface(struct PDMIBASE *pInterface, const char *pszIID)
7091{
7092 PE1KSTATECC pThisCC = RT_FROM_MEMBER(pInterface, E1KSTATECC, IBase);
7093 Assert(&pThisCC->IBase == pInterface);
7094
7095 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThisCC->IBase);
7096 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKDOWN, &pThisCC->INetworkDown);
7097 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKCONFIG, &pThisCC->INetworkConfig);
7098 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThisCC->ILeds);
7099 return NULL;
7100}
7101
7102
7103/* -=-=-=-=- Saved State -=-=-=-=- */
7104
7105/**
7106 * Saves the configuration.
7107 *
7108 * @param pThis The E1K state.
7109 * @param pSSM The handle to the saved state.
7110 */
7111static void e1kR3SaveConfig(PCPDMDEVHLPR3 pHlp, PE1KSTATE pThis, PSSMHANDLE pSSM)
7112{
7113 pHlp->pfnSSMPutMem(pSSM, &pThis->macConfigured, sizeof(pThis->macConfigured));
7114 pHlp->pfnSSMPutU32(pSSM, pThis->eChip);
7115}
7116
7117/**
7118 * @callback_method_impl{FNSSMDEVLIVEEXEC,Save basic configuration.}
7119 */
7120static DECLCALLBACK(int) e1kR3LiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
7121{
7122 RT_NOREF(uPass);
7123 e1kR3SaveConfig(pDevIns->pHlpR3, PDMDEVINS_2_DATA(pDevIns, PE1KSTATE), pSSM);
7124 return VINF_SSM_DONT_CALL_AGAIN;
7125}
7126
7127/**
7128 * @callback_method_impl{FNSSMDEVSAVEPREP,Synchronize.}
7129 */
7130static DECLCALLBACK(int) e1kR3SavePrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
7131{
7132 RT_NOREF(pSSM);
7133 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7134
7135 e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
7136 e1kCsLeave(pThis);
7137 return VINF_SUCCESS;
7138#if 0
7139 /* 1) Prevent all threads from modifying the state and memory */
7140 //pThis->fLocked = true;
7141 /* 2) Cancel all timers */
7142#ifdef E1K_TX_DELAY
7143 e1kCancelTimer(pThis, pThis->CTX_SUFF(pTXDTimer));
7144#endif /* E1K_TX_DELAY */
7145//#ifdef E1K_USE_TX_TIMERS
7146 if (pThis->fTidEnabled)
7147 {
7148 e1kCancelTimer(pThis, pThis->CTX_SUFF(pTIDTimer));
7149#ifndef E1K_NO_TAD
7150 e1kCancelTimer(pThis, pThis->CTX_SUFF(pTADTimer));
7151#endif /* E1K_NO_TAD */
7152 }
7153//#endif /* E1K_USE_TX_TIMERS */
7154#ifdef E1K_USE_RX_TIMERS
7155 e1kCancelTimer(pThis, pThis->CTX_SUFF(pRIDTimer));
7156 e1kCancelTimer(pThis, pThis->CTX_SUFF(pRADTimer));
7157#endif /* E1K_USE_RX_TIMERS */
7158 e1kCancelTimer(pThis, pThis->CTX_SUFF(pIntTimer));
7159 /* 3) Did I forget anything? */
7160 E1kLog(("%s Locked\n", pThis->szPrf));
7161 return VINF_SUCCESS;
7162#endif
7163}
7164
7165/**
7166 * @callback_method_impl{FNSSMDEVSAVEEXEC}
7167 */
7168static DECLCALLBACK(int) e1kR3SaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
7169{
7170 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7171 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7172 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
7173
7174 e1kR3SaveConfig(pHlp, pThis, pSSM);
7175 pThisCC->eeprom.save(pHlp, pSSM);
7176 e1kDumpState(pThis);
7177 pHlp->pfnSSMPutMem(pSSM, pThis->auRegs, sizeof(pThis->auRegs));
7178 pHlp->pfnSSMPutBool(pSSM, pThis->fIntRaised);
7179 Phy::saveState(pHlp, pSSM, &pThis->phy);
7180 pHlp->pfnSSMPutU32(pSSM, pThis->uSelectedReg);
7181 pHlp->pfnSSMPutMem(pSSM, pThis->auMTA, sizeof(pThis->auMTA));
7182 pHlp->pfnSSMPutMem(pSSM, &pThis->aRecAddr, sizeof(pThis->aRecAddr));
7183 pHlp->pfnSSMPutMem(pSSM, pThis->auVFTA, sizeof(pThis->auVFTA));
7184 pHlp->pfnSSMPutU64(pSSM, pThis->u64AckedAt);
7185 pHlp->pfnSSMPutU16(pSSM, pThis->u16RxBSize);
7186 //pHlp->pfnSSMPutBool(pSSM, pThis->fDelayInts);
7187 //pHlp->pfnSSMPutBool(pSSM, pThis->fIntMaskUsed);
7188 pHlp->pfnSSMPutU16(pSSM, pThis->u16TxPktLen);
7189/** @todo State wrt to the TSE buffer is incomplete, so little point in
7190 * saving this actually. */
7191 pHlp->pfnSSMPutMem(pSSM, pThis->aTxPacketFallback, pThis->u16TxPktLen);
7192 pHlp->pfnSSMPutBool(pSSM, pThis->fIPcsum);
7193 pHlp->pfnSSMPutBool(pSSM, pThis->fTCPcsum);
7194 pHlp->pfnSSMPutMem(pSSM, &pThis->contextTSE, sizeof(pThis->contextTSE));
7195 pHlp->pfnSSMPutMem(pSSM, &pThis->contextNormal, sizeof(pThis->contextNormal));
7196 pHlp->pfnSSMPutBool(pSSM, pThis->fVTag);
7197 pHlp->pfnSSMPutU16(pSSM, pThis->u16VTagTCI);
7198#ifdef E1K_WITH_TXD_CACHE
7199# if 0
7200 pHlp->pfnSSMPutU8(pSSM, pThis->nTxDFetched);
7201 pHlp->pfnSSMPutMem(pSSM, pThis->aTxDescriptors,
7202 pThis->nTxDFetched * sizeof(pThis->aTxDescriptors[0]));
7203# else
7204 /*
7205 * There is no point in storing TX descriptor cache entries as we can simply
7206 * fetch them again. Moreover, normally the cache is always empty when we
7207 * save the state. Store zero entries for compatibility.
7208 */
7209 pHlp->pfnSSMPutU8(pSSM, 0);
7210# endif
7211#endif /* E1K_WITH_TXD_CACHE */
7212/** @todo GSO requires some more state here. */
7213 E1kLog(("%s State has been saved\n", pThis->szPrf));
7214 return VINF_SUCCESS;
7215}
7216
7217#if 0
7218/**
7219 * @callback_method_impl{FNSSMDEVSAVEDONE}
7220 */
7221static DECLCALLBACK(int) e1kSaveDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
7222{
7223 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7224
7225 /* If VM is being powered off unlocking will result in assertions in PGM */
7226 if (PDMDevHlpGetVM(pDevIns)->enmVMState == VMSTATE_RUNNING)
7227 pThis->fLocked = false;
7228 else
7229 E1kLog(("%s VM is not running -- remain locked\n", pThis->szPrf));
7230 E1kLog(("%s Unlocked\n", pThis->szPrf));
7231 return VINF_SUCCESS;
7232}
7233#endif
7234
7235/**
7236 * @callback_method_impl{FNSSMDEVLOADPREP,Synchronize.}
7237 */
7238static DECLCALLBACK(int) e1kR3LoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
7239{
7240 RT_NOREF(pSSM);
7241 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7242
7243 e1kCsEnterReturn(pThis, VERR_SEM_BUSY);
7244 e1kCsLeave(pThis);
7245 return VINF_SUCCESS;
7246}
7247
7248/**
7249 * @callback_method_impl{FNSSMDEVLOADEXEC}
7250 */
7251static DECLCALLBACK(int) e1kR3LoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
7252{
7253 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7254 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7255 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
7256 int rc;
7257
7258 if ( uVersion != E1K_SAVEDSTATE_VERSION
7259#ifdef E1K_WITH_TXD_CACHE
7260 && uVersion != E1K_SAVEDSTATE_VERSION_VBOX_42_VTAG
7261#endif /* E1K_WITH_TXD_CACHE */
7262 && uVersion != E1K_SAVEDSTATE_VERSION_VBOX_41
7263 && uVersion != E1K_SAVEDSTATE_VERSION_VBOX_30)
7264 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
7265
7266 if ( uVersion > E1K_SAVEDSTATE_VERSION_VBOX_30
7267 || uPass != SSM_PASS_FINAL)
7268 {
7269 /* config checks */
7270 RTMAC macConfigured;
7271 rc = pHlp->pfnSSMGetMem(pSSM, &macConfigured, sizeof(macConfigured));
7272 AssertRCReturn(rc, rc);
7273 if ( memcmp(&macConfigured, &pThis->macConfigured, sizeof(macConfigured))
7274 && (uPass == 0 || !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)) )
7275 LogRel(("%s: The mac address differs: config=%RTmac saved=%RTmac\n", pThis->szPrf, &pThis->macConfigured, &macConfigured));
7276
7277 E1KCHIP eChip;
7278 rc = pHlp->pfnSSMGetU32(pSSM, &eChip);
7279 AssertRCReturn(rc, rc);
7280 if (eChip != pThis->eChip)
7281 return pHlp->pfnSSMSetCfgError(pSSM, RT_SRC_POS, N_("The chip type differs: config=%u saved=%u"), pThis->eChip, eChip);
7282 }
7283
7284 if (uPass == SSM_PASS_FINAL)
7285 {
7286 if (uVersion > E1K_SAVEDSTATE_VERSION_VBOX_30)
7287 {
7288 rc = pThisCC->eeprom.load(pHlp, pSSM);
7289 AssertRCReturn(rc, rc);
7290 }
7291 /* the state */
7292 pHlp->pfnSSMGetMem(pSSM, &pThis->auRegs, sizeof(pThis->auRegs));
7293 pHlp->pfnSSMGetBool(pSSM, &pThis->fIntRaised);
7294 /** @todo PHY could be made a separate device with its own versioning */
7295 Phy::loadState(pHlp, pSSM, &pThis->phy);
7296 pHlp->pfnSSMGetU32(pSSM, &pThis->uSelectedReg);
7297 pHlp->pfnSSMGetMem(pSSM, &pThis->auMTA, sizeof(pThis->auMTA));
7298 pHlp->pfnSSMGetMem(pSSM, &pThis->aRecAddr, sizeof(pThis->aRecAddr));
7299 pHlp->pfnSSMGetMem(pSSM, &pThis->auVFTA, sizeof(pThis->auVFTA));
7300 pHlp->pfnSSMGetU64(pSSM, &pThis->u64AckedAt);
7301 pHlp->pfnSSMGetU16(pSSM, &pThis->u16RxBSize);
7302 //pHlp->pfnSSMGetBool(pSSM, pThis->fDelayInts);
7303 //pHlp->pfnSSMGetBool(pSSM, pThis->fIntMaskUsed);
7304 rc = pHlp->pfnSSMGetU16(pSSM, &pThis->u16TxPktLen);
7305 AssertRCReturn(rc, rc);
7306 if (pThis->u16TxPktLen > sizeof(pThis->aTxPacketFallback))
7307 pThis->u16TxPktLen = sizeof(pThis->aTxPacketFallback);
7308 pHlp->pfnSSMGetMem(pSSM, &pThis->aTxPacketFallback[0], pThis->u16TxPktLen);
7309 pHlp->pfnSSMGetBool(pSSM, &pThis->fIPcsum);
7310 pHlp->pfnSSMGetBool(pSSM, &pThis->fTCPcsum);
7311 pHlp->pfnSSMGetMem(pSSM, &pThis->contextTSE, sizeof(pThis->contextTSE));
7312 rc = pHlp->pfnSSMGetMem(pSSM, &pThis->contextNormal, sizeof(pThis->contextNormal));
7313 AssertRCReturn(rc, rc);
7314 if (uVersion > E1K_SAVEDSTATE_VERSION_VBOX_41)
7315 {
7316 pHlp->pfnSSMGetBool(pSSM, &pThis->fVTag);
7317 rc = pHlp->pfnSSMGetU16(pSSM, &pThis->u16VTagTCI);
7318 AssertRCReturn(rc, rc);
7319 }
7320 else
7321 {
7322 pThis->fVTag = false;
7323 pThis->u16VTagTCI = 0;
7324 }
7325#ifdef E1K_WITH_TXD_CACHE
7326 if (uVersion > E1K_SAVEDSTATE_VERSION_VBOX_42_VTAG)
7327 {
7328 rc = pHlp->pfnSSMGetU8(pSSM, &pThis->nTxDFetched);
7329 AssertRCReturn(rc, rc);
7330 if (pThis->nTxDFetched)
7331 pHlp->pfnSSMGetMem(pSSM, pThis->aTxDescriptors,
7332 pThis->nTxDFetched * sizeof(pThis->aTxDescriptors[0]));
7333 }
7334 else
7335 pThis->nTxDFetched = 0;
7336 /**
7337 * @todo Perhaps we should not store TXD cache as the entries can be
7338 * simply fetched again from guest's memory. Or can't they?
7339 */
7340#endif /* E1K_WITH_TXD_CACHE */
7341#ifdef E1K_WITH_RXD_CACHE
7342 /*
7343 * There is no point in storing the RX descriptor cache in the saved
7344 * state, we just need to make sure it is empty.
7345 */
7346 pThis->iRxDCurrent = pThis->nRxDFetched = 0;
7347#endif /* E1K_WITH_RXD_CACHE */
7348 rc = pHlp->pfnSSMHandleGetStatus(pSSM);
7349 AssertRCReturn(rc, rc);
7350
7351 /* derived state */
7352 e1kSetupGsoCtx(&pThis->GsoCtx, &pThis->contextTSE);
7353
7354 E1kLog(("%s State has been restored\n", pThis->szPrf));
7355 e1kDumpState(pThis);
7356 }
7357 return VINF_SUCCESS;
7358}
7359
7360/**
7361 * @callback_method_impl{FNSSMDEVLOADDONE, Link status adjustments after loading.}
7362 */
7363static DECLCALLBACK(int) e1kR3LoadDone(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
7364{
7365 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7366 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7367 RT_NOREF(pSSM);
7368
7369 /* Update promiscuous mode */
7370 if (pThisCC->pDrvR3)
7371 pThisCC->pDrvR3->pfnSetPromiscuousMode(pThisCC->pDrvR3, !!(RCTL & (RCTL_UPE | RCTL_MPE)));
7372
7373 /*
7374 * Force the link down here, since PDMNETWORKLINKSTATE_DOWN_RESUME is never
7375 * passed to us. We go through all this stuff if the link was up and we
7376 * wasn't teleported.
7377 */
7378 if ( (STATUS & STATUS_LU)
7379 && !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)
7380 && pThis->cMsLinkUpDelay)
7381 {
7382 e1kR3LinkDownTemp(pDevIns, pThis, pThisCC);
7383 }
7384 return VINF_SUCCESS;
7385}
7386
7387
7388
7389/* -=-=-=-=- Debug Info + Log Types -=-=-=-=- */
7390
7391/**
7392 * @callback_method_impl{FNRTSTRFORMATTYPE}
7393 */
7394static DECLCALLBACK(size_t) e1kR3FmtRxDesc(PFNRTSTROUTPUT pfnOutput,
7395 void *pvArgOutput,
7396 const char *pszType,
7397 void const *pvValue,
7398 int cchWidth,
7399 int cchPrecision,
7400 unsigned fFlags,
7401 void *pvUser)
7402{
7403 RT_NOREF(cchWidth, cchPrecision, fFlags, pvUser);
7404 AssertReturn(strcmp(pszType, "e1krxd") == 0, 0);
7405 E1KRXDESC* pDesc = (E1KRXDESC*)pvValue;
7406 if (!pDesc)
7407 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "NULL_RXD");
7408
7409 size_t cbPrintf = 0;
7410 cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Address=%16LX Length=%04X Csum=%04X\n",
7411 pDesc->u64BufAddr, pDesc->u16Length, pDesc->u16Checksum);
7412 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",
7413 pDesc->status.fPIF ? "PIF" : "pif",
7414 pDesc->status.fIPCS ? "IPCS" : "ipcs",
7415 pDesc->status.fTCPCS ? "TCPCS" : "tcpcs",
7416 pDesc->status.fVP ? "VP" : "vp",
7417 pDesc->status.fIXSM ? "IXSM" : "ixsm",
7418 pDesc->status.fEOP ? "EOP" : "eop",
7419 pDesc->status.fDD ? "DD" : "dd",
7420 pDesc->status.fRXE ? "RXE" : "rxe",
7421 pDesc->status.fIPE ? "IPE" : "ipe",
7422 pDesc->status.fTCPE ? "TCPE" : "tcpe",
7423 pDesc->status.fCE ? "CE" : "ce",
7424 E1K_SPEC_CFI(pDesc->status.u16Special) ? "CFI" :"cfi",
7425 E1K_SPEC_VLAN(pDesc->status.u16Special),
7426 E1K_SPEC_PRI(pDesc->status.u16Special));
7427 return cbPrintf;
7428}
7429
7430/**
7431 * @callback_method_impl{FNRTSTRFORMATTYPE}
7432 */
7433static DECLCALLBACK(size_t) e1kR3FmtTxDesc(PFNRTSTROUTPUT pfnOutput,
7434 void *pvArgOutput,
7435 const char *pszType,
7436 void const *pvValue,
7437 int cchWidth,
7438 int cchPrecision,
7439 unsigned fFlags,
7440 void *pvUser)
7441{
7442 RT_NOREF(cchWidth, cchPrecision, fFlags, pvUser);
7443 AssertReturn(strcmp(pszType, "e1ktxd") == 0, 0);
7444 E1KTXDESC *pDesc = (E1KTXDESC*)pvValue;
7445 if (!pDesc)
7446 return RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "NULL_TXD");
7447
7448 size_t cbPrintf = 0;
7449 switch (e1kGetDescType(pDesc))
7450 {
7451 case E1K_DTYP_CONTEXT:
7452 cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Type=Context\n"
7453 " IPCSS=%02X IPCSO=%02X IPCSE=%04X TUCSS=%02X TUCSO=%02X TUCSE=%04X\n"
7454 " TUCMD:%s%s%s %s %s PAYLEN=%04x HDRLEN=%04x MSS=%04x STA: %s",
7455 pDesc->context.ip.u8CSS, pDesc->context.ip.u8CSO, pDesc->context.ip.u16CSE,
7456 pDesc->context.tu.u8CSS, pDesc->context.tu.u8CSO, pDesc->context.tu.u16CSE,
7457 pDesc->context.dw2.fIDE ? " IDE":"",
7458 pDesc->context.dw2.fRS ? " RS" :"",
7459 pDesc->context.dw2.fTSE ? " TSE":"",
7460 pDesc->context.dw2.fIP ? "IPv4":"IPv6",
7461 pDesc->context.dw2.fTCP ? "TCP":"UDP",
7462 pDesc->context.dw2.u20PAYLEN,
7463 pDesc->context.dw3.u8HDRLEN,
7464 pDesc->context.dw3.u16MSS,
7465 pDesc->context.dw3.fDD?"DD":"");
7466 break;
7467 case E1K_DTYP_DATA:
7468 cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Type=Data Address=%16LX DTALEN=%05X\n"
7469 " DCMD:%s%s%s%s%s%s%s STA:%s%s%s POPTS:%s%s SPECIAL:%s VLAN=%03x PRI=%x",
7470 pDesc->data.u64BufAddr,
7471 pDesc->data.cmd.u20DTALEN,
7472 pDesc->data.cmd.fIDE ? " IDE" :"",
7473 pDesc->data.cmd.fVLE ? " VLE" :"",
7474 pDesc->data.cmd.fRPS ? " RPS" :"",
7475 pDesc->data.cmd.fRS ? " RS" :"",
7476 pDesc->data.cmd.fTSE ? " TSE" :"",
7477 pDesc->data.cmd.fIFCS? " IFCS":"",
7478 pDesc->data.cmd.fEOP ? " EOP" :"",
7479 pDesc->data.dw3.fDD ? " DD" :"",
7480 pDesc->data.dw3.fEC ? " EC" :"",
7481 pDesc->data.dw3.fLC ? " LC" :"",
7482 pDesc->data.dw3.fTXSM? " TXSM":"",
7483 pDesc->data.dw3.fIXSM? " IXSM":"",
7484 E1K_SPEC_CFI(pDesc->data.dw3.u16Special) ? "CFI" :"cfi",
7485 E1K_SPEC_VLAN(pDesc->data.dw3.u16Special),
7486 E1K_SPEC_PRI(pDesc->data.dw3.u16Special));
7487 break;
7488 case E1K_DTYP_LEGACY:
7489 cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Type=Legacy Address=%16LX DTALEN=%05X\n"
7490 " CMD:%s%s%s%s%s%s%s STA:%s%s%s CSO=%02x CSS=%02x SPECIAL:%s VLAN=%03x PRI=%x",
7491 pDesc->data.u64BufAddr,
7492 pDesc->legacy.cmd.u16Length,
7493 pDesc->legacy.cmd.fIDE ? " IDE" :"",
7494 pDesc->legacy.cmd.fVLE ? " VLE" :"",
7495 pDesc->legacy.cmd.fRPS ? " RPS" :"",
7496 pDesc->legacy.cmd.fRS ? " RS" :"",
7497 pDesc->legacy.cmd.fIC ? " IC" :"",
7498 pDesc->legacy.cmd.fIFCS? " IFCS":"",
7499 pDesc->legacy.cmd.fEOP ? " EOP" :"",
7500 pDesc->legacy.dw3.fDD ? " DD" :"",
7501 pDesc->legacy.dw3.fEC ? " EC" :"",
7502 pDesc->legacy.dw3.fLC ? " LC" :"",
7503 pDesc->legacy.cmd.u8CSO,
7504 pDesc->legacy.dw3.u8CSS,
7505 E1K_SPEC_CFI(pDesc->legacy.dw3.u16Special) ? "CFI" :"cfi",
7506 E1K_SPEC_VLAN(pDesc->legacy.dw3.u16Special),
7507 E1K_SPEC_PRI(pDesc->legacy.dw3.u16Special));
7508 break;
7509 default:
7510 cbPrintf += RTStrFormat(pfnOutput, pvArgOutput, NULL, 0, "Invalid Transmit Descriptor");
7511 break;
7512 }
7513
7514 return cbPrintf;
7515}
7516
7517/** Initializes debug helpers (logging format types). */
7518static int e1kR3InitDebugHelpers(void)
7519{
7520 int rc = VINF_SUCCESS;
7521 static bool s_fHelpersRegistered = false;
7522 if (!s_fHelpersRegistered)
7523 {
7524 s_fHelpersRegistered = true;
7525 rc = RTStrFormatTypeRegister("e1krxd", e1kR3FmtRxDesc, NULL);
7526 AssertRCReturn(rc, rc);
7527 rc = RTStrFormatTypeRegister("e1ktxd", e1kR3FmtTxDesc, NULL);
7528 AssertRCReturn(rc, rc);
7529 }
7530 return rc;
7531}
7532
7533/**
7534 * Status info callback.
7535 *
7536 * @param pDevIns The device instance.
7537 * @param pHlp The output helpers.
7538 * @param pszArgs The arguments.
7539 */
7540static DECLCALLBACK(void) e1kR3Info(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
7541{
7542 RT_NOREF(pszArgs);
7543 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7544 unsigned i;
7545 // bool fRcvRing = false;
7546 // bool fXmtRing = false;
7547
7548 /*
7549 * Parse args.
7550 if (pszArgs)
7551 {
7552 fRcvRing = strstr(pszArgs, "verbose") || strstr(pszArgs, "rcv");
7553 fXmtRing = strstr(pszArgs, "verbose") || strstr(pszArgs, "xmt");
7554 }
7555 */
7556
7557 /*
7558 * Show info.
7559 */
7560 pHlp->pfnPrintf(pHlp, "E1000 #%d: port=%04x mmio=%RGp mac-cfg=%RTmac %s%s%s\n",
7561 pDevIns->iInstance,
7562 PDMDevHlpIoPortGetMappingAddress(pDevIns, pThis->hIoPorts),
7563 PDMDevHlpMmioGetMappingAddress(pDevIns, pThis->hMmioRegion),
7564 &pThis->macConfigured, g_aChips[pThis->eChip].pcszName,
7565 pDevIns->fRCEnabled ? " RC" : "", pDevIns->fR0Enabled ? " R0" : "");
7566
7567 e1kR3CsEnterAsserted(pThis); /* Not sure why but PCNet does it */
7568
7569 for (i = 0; i < E1K_NUM_OF_32BIT_REGS; ++i)
7570 pHlp->pfnPrintf(pHlp, "%8.8s = %08x\n", g_aE1kRegMap[i].abbrev, pThis->auRegs[i]);
7571
7572 for (i = 0; i < RT_ELEMENTS(pThis->aRecAddr.array); i++)
7573 {
7574 E1KRAELEM* ra = pThis->aRecAddr.array + i;
7575 if (ra->ctl & RA_CTL_AV)
7576 {
7577 const char *pcszTmp;
7578 switch (ra->ctl & RA_CTL_AS)
7579 {
7580 case 0: pcszTmp = "DST"; break;
7581 case 1: pcszTmp = "SRC"; break;
7582 default: pcszTmp = "reserved";
7583 }
7584 pHlp->pfnPrintf(pHlp, "RA%02d: %s %RTmac\n", i, pcszTmp, ra->addr);
7585 }
7586 }
7587 unsigned cDescs = RDLEN / sizeof(E1KRXDESC);
7588 uint32_t rdh = RDH;
7589 pHlp->pfnPrintf(pHlp, "\n-- Receive Descriptors (%d total) --\n", cDescs);
7590 for (i = 0; i < cDescs; ++i)
7591 {
7592 E1KRXDESC desc;
7593 PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(RDBAH, RDBAL, i),
7594 &desc, sizeof(desc));
7595 if (i == rdh)
7596 pHlp->pfnPrintf(pHlp, ">>> ");
7597 pHlp->pfnPrintf(pHlp, "%RGp: %R[e1krxd]\n", e1kDescAddr(RDBAH, RDBAL, i), &desc);
7598 }
7599#ifdef E1K_WITH_RXD_CACHE
7600 pHlp->pfnPrintf(pHlp, "\n-- Receive Descriptors in Cache (at %d (RDH %d)/ fetched %d / max %d) --\n",
7601 pThis->iRxDCurrent, RDH, pThis->nRxDFetched, E1K_RXD_CACHE_SIZE);
7602 if (rdh > pThis->iRxDCurrent)
7603 rdh -= pThis->iRxDCurrent;
7604 else
7605 rdh = cDescs + rdh - pThis->iRxDCurrent;
7606 for (i = 0; i < pThis->nRxDFetched; ++i)
7607 {
7608 if (i == pThis->iRxDCurrent)
7609 pHlp->pfnPrintf(pHlp, ">>> ");
7610 if (cDescs)
7611 pHlp->pfnPrintf(pHlp, "%RGp: %R[e1krxd]\n",
7612 e1kDescAddr(RDBAH, RDBAL, rdh++ % cDescs),
7613 &pThis->aRxDescriptors[i]);
7614 else
7615 pHlp->pfnPrintf(pHlp, "<lost>: %R[e1krxd]\n",
7616 &pThis->aRxDescriptors[i]);
7617 }
7618#endif /* E1K_WITH_RXD_CACHE */
7619
7620 cDescs = TDLEN / sizeof(E1KTXDESC);
7621 uint32_t tdh = TDH;
7622 pHlp->pfnPrintf(pHlp, "\n-- Transmit Descriptors (%d total) --\n", cDescs);
7623 for (i = 0; i < cDescs; ++i)
7624 {
7625 E1KTXDESC desc;
7626 PDMDevHlpPCIPhysRead(pDevIns, e1kDescAddr(TDBAH, TDBAL, i),
7627 &desc, sizeof(desc));
7628 if (i == tdh)
7629 pHlp->pfnPrintf(pHlp, ">>> ");
7630 pHlp->pfnPrintf(pHlp, "%RGp: %R[e1ktxd]\n", e1kDescAddr(TDBAH, TDBAL, i), &desc);
7631 }
7632#ifdef E1K_WITH_TXD_CACHE
7633 pHlp->pfnPrintf(pHlp, "\n-- Transmit Descriptors in Cache (at %d (TDH %d)/ fetched %d / max %d) --\n",
7634 pThis->iTxDCurrent, TDH, pThis->nTxDFetched, E1K_TXD_CACHE_SIZE);
7635 if (tdh > pThis->iTxDCurrent)
7636 tdh -= pThis->iTxDCurrent;
7637 else
7638 tdh = cDescs + tdh - pThis->iTxDCurrent;
7639 for (i = 0; i < pThis->nTxDFetched; ++i)
7640 {
7641 if (i == pThis->iTxDCurrent)
7642 pHlp->pfnPrintf(pHlp, ">>> ");
7643 if (cDescs)
7644 pHlp->pfnPrintf(pHlp, "%RGp: %R[e1ktxd]\n",
7645 e1kDescAddr(TDBAH, TDBAL, tdh++ % cDescs),
7646 &pThis->aTxDescriptors[i]);
7647 else
7648 pHlp->pfnPrintf(pHlp, "<lost>: %R[e1ktxd]\n",
7649 &pThis->aTxDescriptors[i]);
7650 }
7651#endif /* E1K_WITH_TXD_CACHE */
7652
7653
7654#ifdef E1K_INT_STATS
7655 pHlp->pfnPrintf(pHlp, "Interrupt attempts: %d\n", pThis->uStatIntTry);
7656 pHlp->pfnPrintf(pHlp, "Interrupts raised : %d\n", pThis->uStatInt);
7657 pHlp->pfnPrintf(pHlp, "Interrupts lowered: %d\n", pThis->uStatIntLower);
7658 pHlp->pfnPrintf(pHlp, "ICR outside ISR : %d\n", pThis->uStatNoIntICR);
7659 pHlp->pfnPrintf(pHlp, "IMS raised ints : %d\n", pThis->uStatIntIMS);
7660 pHlp->pfnPrintf(pHlp, "Interrupts skipped: %d\n", pThis->uStatIntSkip);
7661 pHlp->pfnPrintf(pHlp, "Masked interrupts : %d\n", pThis->uStatIntMasked);
7662 pHlp->pfnPrintf(pHlp, "Early interrupts : %d\n", pThis->uStatIntEarly);
7663 pHlp->pfnPrintf(pHlp, "Late interrupts : %d\n", pThis->uStatIntLate);
7664 pHlp->pfnPrintf(pHlp, "Lost interrupts : %d\n", pThis->iStatIntLost);
7665 pHlp->pfnPrintf(pHlp, "Interrupts by RX : %d\n", pThis->uStatIntRx);
7666 pHlp->pfnPrintf(pHlp, "Interrupts by TX : %d\n", pThis->uStatIntTx);
7667 pHlp->pfnPrintf(pHlp, "Interrupts by ICS : %d\n", pThis->uStatIntICS);
7668 pHlp->pfnPrintf(pHlp, "Interrupts by RDTR: %d\n", pThis->uStatIntRDTR);
7669 pHlp->pfnPrintf(pHlp, "Interrupts by RDMT: %d\n", pThis->uStatIntRXDMT0);
7670 pHlp->pfnPrintf(pHlp, "Interrupts by TXQE: %d\n", pThis->uStatIntTXQE);
7671 pHlp->pfnPrintf(pHlp, "TX int delay asked: %d\n", pThis->uStatTxIDE);
7672 pHlp->pfnPrintf(pHlp, "TX delayed: %d\n", pThis->uStatTxDelayed);
7673 pHlp->pfnPrintf(pHlp, "TX delayed expired: %d\n", pThis->uStatTxDelayExp);
7674 pHlp->pfnPrintf(pHlp, "TX no report asked: %d\n", pThis->uStatTxNoRS);
7675 pHlp->pfnPrintf(pHlp, "TX abs timer expd : %d\n", pThis->uStatTAD);
7676 pHlp->pfnPrintf(pHlp, "TX int timer expd : %d\n", pThis->uStatTID);
7677 pHlp->pfnPrintf(pHlp, "RX abs timer expd : %d\n", pThis->uStatRAD);
7678 pHlp->pfnPrintf(pHlp, "RX int timer expd : %d\n", pThis->uStatRID);
7679 pHlp->pfnPrintf(pHlp, "TX CTX descriptors: %d\n", pThis->uStatDescCtx);
7680 pHlp->pfnPrintf(pHlp, "TX DAT descriptors: %d\n", pThis->uStatDescDat);
7681 pHlp->pfnPrintf(pHlp, "TX LEG descriptors: %d\n", pThis->uStatDescLeg);
7682 pHlp->pfnPrintf(pHlp, "Received frames : %d\n", pThis->uStatRxFrm);
7683 pHlp->pfnPrintf(pHlp, "Transmitted frames: %d\n", pThis->uStatTxFrm);
7684 pHlp->pfnPrintf(pHlp, "TX frames up to 1514: %d\n", pThis->uStatTx1514);
7685 pHlp->pfnPrintf(pHlp, "TX frames up to 2962: %d\n", pThis->uStatTx2962);
7686 pHlp->pfnPrintf(pHlp, "TX frames up to 4410: %d\n", pThis->uStatTx4410);
7687 pHlp->pfnPrintf(pHlp, "TX frames up to 5858: %d\n", pThis->uStatTx5858);
7688 pHlp->pfnPrintf(pHlp, "TX frames up to 7306: %d\n", pThis->uStatTx7306);
7689 pHlp->pfnPrintf(pHlp, "TX frames up to 8754: %d\n", pThis->uStatTx8754);
7690 pHlp->pfnPrintf(pHlp, "TX frames up to 16384: %d\n", pThis->uStatTx16384);
7691 pHlp->pfnPrintf(pHlp, "TX frames up to 32768: %d\n", pThis->uStatTx32768);
7692 pHlp->pfnPrintf(pHlp, "Larger TX frames : %d\n", pThis->uStatTxLarge);
7693#endif /* E1K_INT_STATS */
7694
7695 e1kCsLeave(pThis);
7696}
7697
7698
7699
7700/* -=-=-=-=- PDMDEVREG -=-=-=-=- */
7701
7702/**
7703 * Detach notification.
7704 *
7705 * One port on the network card has been disconnected from the network.
7706 *
7707 * @param pDevIns The device instance.
7708 * @param iLUN The logical unit which is being detached.
7709 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
7710 */
7711static DECLCALLBACK(void) e1kR3Detach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
7712{
7713 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7714 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7715 Log(("%s e1kR3Detach:\n", pThis->szPrf));
7716 RT_NOREF(fFlags);
7717
7718 AssertLogRelReturnVoid(iLUN == 0);
7719
7720 e1kR3CsEnterAsserted(pThis);
7721
7722 /* Mark device as detached. */
7723 pThis->fIsAttached = false;
7724 /*
7725 * Zero some important members.
7726 */
7727 pThisCC->pDrvBase = NULL;
7728 pThisCC->pDrvR3 = NULL;
7729#if 0 /** @todo @bugref{9218} ring-0 driver stuff */
7730 pThisR0->pDrvR0 = NIL_RTR0PTR;
7731 pThisRC->pDrvRC = NIL_RTRCPTR;
7732#endif
7733
7734 PDMDevHlpCritSectLeave(pDevIns, &pThis->cs);
7735}
7736
7737/**
7738 * Attach the Network attachment.
7739 *
7740 * One port on the network card has been connected to a network.
7741 *
7742 * @returns VBox status code.
7743 * @param pDevIns The device instance.
7744 * @param iLUN The logical unit which is being attached.
7745 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
7746 *
7747 * @remarks This code path is not used during construction.
7748 */
7749static DECLCALLBACK(int) e1kR3Attach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
7750{
7751 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7752 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7753 LogFlow(("%s e1kR3Attach:\n", pThis->szPrf));
7754 RT_NOREF(fFlags);
7755
7756 AssertLogRelReturn(iLUN == 0, VERR_PDM_NO_SUCH_LUN);
7757
7758 e1kR3CsEnterAsserted(pThis);
7759
7760 /*
7761 * Attach the driver.
7762 */
7763 int rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->IBase, &pThisCC->pDrvBase, "Network Port");
7764 if (RT_SUCCESS(rc))
7765 {
7766 pThisCC->pDrvR3 = PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMINETWORKUP);
7767 AssertMsgStmt(pThisCC->pDrvR3, ("Failed to obtain the PDMINETWORKUP interface!\n"),
7768 rc = VERR_PDM_MISSING_INTERFACE_BELOW);
7769 if (RT_SUCCESS(rc))
7770 {
7771#if 0 /** @todo @bugref{9218} ring-0 driver stuff */
7772 pThisR0->pDrvR0 = PDMIBASER0_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASER0), PDMINETWORKUP);
7773 pThisRC->pDrvRC = PDMIBASERC_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASERC), PDMINETWORKUP);
7774#endif
7775 /* Mark device as attached. */
7776 pThis->fIsAttached = true;
7777 }
7778 }
7779 else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER
7780 || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME)
7781 {
7782 /* This should never happen because this function is not called
7783 * if there is no driver to attach! */
7784 Log(("%s No attached driver!\n", pThis->szPrf));
7785 }
7786
7787 /*
7788 * Temporary set the link down if it was up so that the guest will know
7789 * that we have change the configuration of the network card
7790 */
7791 if ((STATUS & STATUS_LU) && RT_SUCCESS(rc))
7792 e1kR3LinkDownTemp(pDevIns, pThis, pThisCC);
7793
7794 PDMDevHlpCritSectLeave(pDevIns, &pThis->cs);
7795 return rc;
7796}
7797
7798/**
7799 * @copydoc FNPDMDEVPOWEROFF
7800 */
7801static DECLCALLBACK(void) e1kR3PowerOff(PPDMDEVINS pDevIns)
7802{
7803 /* Poke thread waiting for buffer space. */
7804 e1kWakeupReceive(pDevIns, PDMDEVINS_2_DATA(pDevIns, PE1KSTATE));
7805}
7806
7807/**
7808 * @copydoc FNPDMDEVRESET
7809 */
7810static DECLCALLBACK(void) e1kR3Reset(PPDMDEVINS pDevIns)
7811{
7812 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7813 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7814#ifdef E1K_TX_DELAY
7815 e1kCancelTimer(pDevIns, pThis, pThis->hTXDTimer);
7816#endif /* E1K_TX_DELAY */
7817 e1kCancelTimer(pDevIns, pThis, pThis->hIntTimer);
7818 e1kCancelTimer(pDevIns, pThis, pThis->hLUTimer);
7819 e1kXmitFreeBuf(pThis, pThisCC);
7820 pThis->u16TxPktLen = 0;
7821 pThis->fIPcsum = false;
7822 pThis->fTCPcsum = false;
7823 pThis->fIntMaskUsed = false;
7824 pThis->fDelayInts = false;
7825 pThis->fLocked = false;
7826 pThis->u64AckedAt = 0;
7827 e1kR3HardReset(pDevIns, pThis, pThisCC);
7828}
7829
7830/**
7831 * @copydoc FNPDMDEVSUSPEND
7832 */
7833static DECLCALLBACK(void) e1kR3Suspend(PPDMDEVINS pDevIns)
7834{
7835 /* Poke thread waiting for buffer space. */
7836 e1kWakeupReceive(pDevIns, PDMDEVINS_2_DATA(pDevIns, PE1KSTATE));
7837}
7838
7839/**
7840 * Device relocation callback.
7841 *
7842 * When this callback is called the device instance data, and if the
7843 * device have a GC component, is being relocated, or/and the selectors
7844 * have been changed. The device must use the chance to perform the
7845 * necessary pointer relocations and data updates.
7846 *
7847 * Before the GC code is executed the first time, this function will be
7848 * called with a 0 delta so GC pointer calculations can be one in one place.
7849 *
7850 * @param pDevIns Pointer to the device instance.
7851 * @param offDelta The relocation delta relative to the old location.
7852 *
7853 * @remark A relocation CANNOT fail.
7854 */
7855static DECLCALLBACK(void) e1kR3Relocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
7856{
7857 PE1KSTATERC pThisRC = PDMINS_2_DATA_RC(pDevIns, PE1KSTATERC);
7858 if (pThisRC)
7859 pThisRC->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
7860 RT_NOREF(offDelta);
7861}
7862
7863/**
7864 * Destruct a device instance.
7865 *
7866 * We need to free non-VM resources only.
7867 *
7868 * @returns VBox status code.
7869 * @param pDevIns The device instance data.
7870 * @thread EMT
7871 */
7872static DECLCALLBACK(int) e1kR3Destruct(PPDMDEVINS pDevIns)
7873{
7874 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
7875 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7876
7877 e1kDumpState(pThis);
7878 E1kLog(("%s Destroying instance\n", pThis->szPrf));
7879 if (PDMDevHlpCritSectIsInitialized(pDevIns, &pThis->cs))
7880 {
7881 if (pThis->hEventMoreRxDescAvail != NIL_SUPSEMEVENT)
7882 {
7883 PDMDevHlpSUPSemEventSignal(pDevIns, pThis->hEventMoreRxDescAvail);
7884 RTThreadYield();
7885 PDMDevHlpSUPSemEventClose(pDevIns, pThis->hEventMoreRxDescAvail);
7886 pThis->hEventMoreRxDescAvail = NIL_SUPSEMEVENT;
7887 }
7888#ifdef E1K_WITH_TX_CS
7889 PDMDevHlpCritSectDelete(pDevIns, &pThis->csTx);
7890#endif /* E1K_WITH_TX_CS */
7891 PDMDevHlpCritSectDelete(pDevIns, &pThis->csRx);
7892 PDMDevHlpCritSectDelete(pDevIns, &pThis->cs);
7893 }
7894 return VINF_SUCCESS;
7895}
7896
7897
7898/**
7899 * Set PCI configuration space registers.
7900 *
7901 * @param pci Reference to PCI device structure.
7902 * @thread EMT
7903 */
7904static void e1kR3ConfigurePciDev(PPDMPCIDEV pPciDev, E1KCHIP eChip)
7905{
7906 Assert(eChip < RT_ELEMENTS(g_aChips));
7907 /* Configure PCI Device, assume 32-bit mode ******************************/
7908 PDMPciDevSetVendorId(pPciDev, g_aChips[eChip].uPCIVendorId);
7909 PDMPciDevSetDeviceId(pPciDev, g_aChips[eChip].uPCIDeviceId);
7910 PDMPciDevSetWord( pPciDev, VBOX_PCI_SUBSYSTEM_VENDOR_ID, g_aChips[eChip].uPCISubsystemVendorId);
7911 PDMPciDevSetWord( pPciDev, VBOX_PCI_SUBSYSTEM_ID, g_aChips[eChip].uPCISubsystemId);
7912
7913 PDMPciDevSetWord( pPciDev, VBOX_PCI_COMMAND, 0x0000);
7914 /* DEVSEL Timing (medium device), 66 MHz Capable, New capabilities */
7915 PDMPciDevSetWord( pPciDev, VBOX_PCI_STATUS,
7916 VBOX_PCI_STATUS_DEVSEL_MEDIUM | VBOX_PCI_STATUS_CAP_LIST | VBOX_PCI_STATUS_66MHZ);
7917 /* Stepping A2 */
7918 PDMPciDevSetByte( pPciDev, VBOX_PCI_REVISION_ID, 0x02);
7919 /* Ethernet adapter */
7920 PDMPciDevSetByte( pPciDev, VBOX_PCI_CLASS_PROG, 0x00);
7921 PDMPciDevSetWord( pPciDev, VBOX_PCI_CLASS_DEVICE, 0x0200);
7922 /* normal single function Ethernet controller */
7923 PDMPciDevSetByte( pPciDev, VBOX_PCI_HEADER_TYPE, 0x00);
7924 /* Memory Register Base Address */
7925 PDMPciDevSetDWord(pPciDev, VBOX_PCI_BASE_ADDRESS_0, 0x00000000);
7926 /* Memory Flash Base Address */
7927 PDMPciDevSetDWord(pPciDev, VBOX_PCI_BASE_ADDRESS_1, 0x00000000);
7928 /* IO Register Base Address */
7929 PDMPciDevSetDWord(pPciDev, VBOX_PCI_BASE_ADDRESS_2, 0x00000001);
7930 /* Expansion ROM Base Address */
7931 PDMPciDevSetDWord(pPciDev, VBOX_PCI_ROM_ADDRESS, 0x00000000);
7932 /* Capabilities Pointer */
7933 PDMPciDevSetByte( pPciDev, VBOX_PCI_CAPABILITY_LIST, 0xDC);
7934 /* Interrupt Pin: INTA# */
7935 PDMPciDevSetByte( pPciDev, VBOX_PCI_INTERRUPT_PIN, 0x01);
7936 /* Max_Lat/Min_Gnt: very high priority and time slice */
7937 PDMPciDevSetByte( pPciDev, VBOX_PCI_MIN_GNT, 0xFF);
7938 PDMPciDevSetByte( pPciDev, VBOX_PCI_MAX_LAT, 0x00);
7939
7940 /* PCI Power Management Registers ****************************************/
7941 /* Capability ID: PCI Power Management Registers */
7942 PDMPciDevSetByte( pPciDev, 0xDC, VBOX_PCI_CAP_ID_PM);
7943 /* Next Item Pointer: PCI-X */
7944 PDMPciDevSetByte( pPciDev, 0xDC + 1, 0xE4);
7945 /* Power Management Capabilities: PM disabled, DSI */
7946 PDMPciDevSetWord( pPciDev, 0xDC + 2,
7947 0x0002 | VBOX_PCI_PM_CAP_DSI);
7948 /* Power Management Control / Status Register: PM disabled */
7949 PDMPciDevSetWord( pPciDev, 0xDC + 4, 0x0000);
7950 /* PMCSR_BSE Bridge Support Extensions: Not supported */
7951 PDMPciDevSetByte( pPciDev, 0xDC + 6, 0x00);
7952 /* Data Register: PM disabled, always 0 */
7953 PDMPciDevSetByte( pPciDev, 0xDC + 7, 0x00);
7954
7955 /* PCI-X Configuration Registers *****************************************/
7956 /* Capability ID: PCI-X Configuration Registers */
7957 PDMPciDevSetByte( pPciDev, 0xE4, VBOX_PCI_CAP_ID_PCIX);
7958#ifdef E1K_WITH_MSI
7959 PDMPciDevSetByte( pPciDev, 0xE4 + 1, 0x80);
7960#else
7961 /* Next Item Pointer: None (Message Signalled Interrupts are disabled) */
7962 PDMPciDevSetByte( pPciDev, 0xE4 + 1, 0x00);
7963#endif
7964 /* PCI-X Command: Enable Relaxed Ordering */
7965 PDMPciDevSetWord( pPciDev, 0xE4 + 2, VBOX_PCI_X_CMD_ERO);
7966 /* PCI-X Status: 32-bit, 66MHz*/
7967 /** @todo is this value really correct? fff8 doesn't look like actual PCI address */
7968 PDMPciDevSetDWord(pPciDev, 0xE4 + 4, 0x0040FFF8);
7969}
7970
7971/**
7972 * @interface_method_impl{PDMDEVREG,pfnConstruct}
7973 */
7974static DECLCALLBACK(int) e1kR3Construct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
7975{
7976 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
7977 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
7978 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
7979 int rc;
7980
7981 /*
7982 * Initialize the instance data (state).
7983 * Note! Caller has initialized it to ZERO already.
7984 */
7985 RTStrPrintf(pThis->szPrf, sizeof(pThis->szPrf), "E1000#%d", iInstance);
7986 E1kLog(("%s Constructing new instance sizeof(E1KRXDESC)=%d\n", pThis->szPrf, sizeof(E1KRXDESC)));
7987 pThis->hEventMoreRxDescAvail = NIL_SUPSEMEVENT;
7988 pThis->u16TxPktLen = 0;
7989 pThis->fIPcsum = false;
7990 pThis->fTCPcsum = false;
7991 pThis->fIntMaskUsed = false;
7992 pThis->fDelayInts = false;
7993 pThis->fLocked = false;
7994 pThis->u64AckedAt = 0;
7995 pThis->led.u32Magic = PDMLED_MAGIC;
7996 pThis->u32PktNo = 1;
7997 pThis->fIsAttached = false;
7998
7999 pThisCC->pDevInsR3 = pDevIns;
8000 pThisCC->pShared = pThis;
8001
8002 /* Interfaces */
8003 pThisCC->IBase.pfnQueryInterface = e1kR3QueryInterface;
8004
8005 pThisCC->INetworkDown.pfnWaitReceiveAvail = e1kR3NetworkDown_WaitReceiveAvail;
8006 pThisCC->INetworkDown.pfnReceive = e1kR3NetworkDown_Receive;
8007 pThisCC->INetworkDown.pfnXmitPending = e1kR3NetworkDown_XmitPending;
8008
8009 pThisCC->ILeds.pfnQueryStatusLed = e1kR3QueryStatusLed;
8010
8011 pThisCC->INetworkConfig.pfnGetMac = e1kR3GetMac;
8012 pThisCC->INetworkConfig.pfnGetLinkState = e1kR3GetLinkState;
8013 pThisCC->INetworkConfig.pfnSetLinkState = e1kR3SetLinkState;
8014
8015 /*
8016 * Internal validations.
8017 */
8018 for (uint32_t iReg = 1; iReg < E1K_NUM_OF_BINARY_SEARCHABLE; iReg++)
8019 AssertLogRelMsgReturn( g_aE1kRegMap[iReg].offset > g_aE1kRegMap[iReg - 1].offset
8020 && g_aE1kRegMap[iReg].offset + g_aE1kRegMap[iReg].size
8021 >= g_aE1kRegMap[iReg - 1].offset + g_aE1kRegMap[iReg - 1].size,
8022 ("%s@%#xLB%#x vs %s@%#xLB%#x\n",
8023 g_aE1kRegMap[iReg].abbrev, g_aE1kRegMap[iReg].offset, g_aE1kRegMap[iReg].size,
8024 g_aE1kRegMap[iReg - 1].abbrev, g_aE1kRegMap[iReg - 1].offset, g_aE1kRegMap[iReg - 1].size),
8025 VERR_INTERNAL_ERROR_4);
8026
8027 /*
8028 * Validate configuration.
8029 */
8030 PDMDEV_VALIDATE_CONFIG_RETURN(pDevIns,
8031 "MAC|"
8032 "CableConnected|"
8033 "AdapterType|"
8034 "LineSpeed|"
8035 "ItrEnabled|"
8036 "ItrRxEnabled|"
8037 "EthernetCRC|"
8038 "GSOEnabled|"
8039 "LinkUpDelay|"
8040 "StatNo",
8041 "");
8042
8043 /** @todo LineSpeed unused! */
8044
8045 /*
8046 * Get config params
8047 */
8048 PCPDMDEVHLPR3 pHlp = pDevIns->pHlpR3;
8049 rc = pHlp->pfnCFGMQueryBytes(pCfg, "MAC", pThis->macConfigured.au8, sizeof(pThis->macConfigured.au8));
8050 if (RT_FAILURE(rc))
8051 return PDMDEV_SET_ERROR(pDevIns, rc,
8052 N_("Configuration error: Failed to get MAC address"));
8053 rc = pHlp->pfnCFGMQueryBool(pCfg, "CableConnected", &pThis->fCableConnected);
8054 if (RT_FAILURE(rc))
8055 return PDMDEV_SET_ERROR(pDevIns, rc,
8056 N_("Configuration error: Failed to get the value of 'CableConnected'"));
8057 rc = pHlp->pfnCFGMQueryU32(pCfg, "AdapterType", (uint32_t*)&pThis->eChip);
8058 if (RT_FAILURE(rc))
8059 return PDMDEV_SET_ERROR(pDevIns, rc,
8060 N_("Configuration error: Failed to get the value of 'AdapterType'"));
8061 Assert(pThis->eChip <= E1K_CHIP_82545EM);
8062
8063 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "EthernetCRC", &pThis->fEthernetCRC, true);
8064 if (RT_FAILURE(rc))
8065 return PDMDEV_SET_ERROR(pDevIns, rc,
8066 N_("Configuration error: Failed to get the value of 'EthernetCRC'"));
8067
8068 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "GSOEnabled", &pThis->fGSOEnabled, true);
8069 if (RT_FAILURE(rc))
8070 return PDMDEV_SET_ERROR(pDevIns, rc,
8071 N_("Configuration error: Failed to get the value of 'GSOEnabled'"));
8072
8073 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "ItrEnabled", &pThis->fItrEnabled, false);
8074 if (RT_FAILURE(rc))
8075 return PDMDEV_SET_ERROR(pDevIns, rc,
8076 N_("Configuration error: Failed to get the value of 'ItrEnabled'"));
8077
8078 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "ItrRxEnabled", &pThis->fItrRxEnabled, true);
8079 if (RT_FAILURE(rc))
8080 return PDMDEV_SET_ERROR(pDevIns, rc,
8081 N_("Configuration error: Failed to get the value of 'ItrRxEnabled'"));
8082
8083 rc = pHlp->pfnCFGMQueryBoolDef(pCfg, "TidEnabled", &pThis->fTidEnabled, false);
8084 if (RT_FAILURE(rc))
8085 return PDMDEV_SET_ERROR(pDevIns, rc,
8086 N_("Configuration error: Failed to get the value of 'TidEnabled'"));
8087
8088 /*
8089 * Increased the link up delay from 3 to 5 seconds to make sure a guest notices the link loss
8090 * and updates its network configuration when the link is restored. See @bugref{10114}.
8091 */
8092 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "LinkUpDelay", (uint32_t*)&pThis->cMsLinkUpDelay, 5000); /* ms */
8093 if (RT_FAILURE(rc))
8094 return PDMDEV_SET_ERROR(pDevIns, rc,
8095 N_("Configuration error: Failed to get the value of 'LinkUpDelay'"));
8096 Assert(pThis->cMsLinkUpDelay <= 300000); /* less than 5 minutes */
8097 if (pThis->cMsLinkUpDelay > 5000)
8098 LogRel(("%s: WARNING! Link up delay is set to %u seconds!\n", pThis->szPrf, pThis->cMsLinkUpDelay / 1000));
8099 else if (pThis->cMsLinkUpDelay == 0)
8100 LogRel(("%s: WARNING! Link up delay is disabled!\n", pThis->szPrf));
8101
8102 uint32_t uStatNo = (uint32_t)iInstance;
8103 rc = pHlp->pfnCFGMQueryU32Def(pCfg, "StatNo", &uStatNo, (uint32_t)iInstance);
8104 if (RT_FAILURE(rc))
8105 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Configuration error: Failed to get the \"StatNo\" value"));
8106
8107 LogRel(("%s: Chip=%s LinkUpDelay=%ums EthernetCRC=%s GSO=%s Itr=%s ItrRx=%s TID=%s R0=%s RC=%s\n", pThis->szPrf,
8108 g_aChips[pThis->eChip].pcszName, pThis->cMsLinkUpDelay,
8109 pThis->fEthernetCRC ? "on" : "off",
8110 pThis->fGSOEnabled ? "enabled" : "disabled",
8111 pThis->fItrEnabled ? "enabled" : "disabled",
8112 pThis->fItrRxEnabled ? "enabled" : "disabled",
8113 pThis->fTidEnabled ? "enabled" : "disabled",
8114 pDevIns->fR0Enabled ? "enabled" : "disabled",
8115 pDevIns->fRCEnabled ? "enabled" : "disabled"));
8116
8117 /*
8118 * Initialize sub-components and register everything with the VMM.
8119 */
8120
8121 /* Initialize the EEPROM. */
8122 pThisCC->eeprom.init(pThis->macConfigured);
8123
8124 /* Initialize internal PHY. */
8125 Phy::init(&pThis->phy, iInstance, pThis->eChip == E1K_CHIP_82543GC ? PHY_EPID_M881000 : PHY_EPID_M881011);
8126
8127 /* Initialize critical sections. We do our own locking. */
8128 rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
8129 AssertRCReturn(rc, rc);
8130
8131 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->cs, RT_SRC_POS, "E1000#%d", iInstance);
8132 AssertRCReturn(rc, rc);
8133 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->csRx, RT_SRC_POS, "E1000#%dRX", iInstance);
8134 AssertRCReturn(rc, rc);
8135#ifdef E1K_WITH_TX_CS
8136 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->csTx, RT_SRC_POS, "E1000#%dTX", iInstance);
8137 AssertRCReturn(rc, rc);
8138#endif
8139
8140 /* Saved state registration. */
8141 rc = PDMDevHlpSSMRegisterEx(pDevIns, E1K_SAVEDSTATE_VERSION, sizeof(E1KSTATE), NULL,
8142 NULL, e1kR3LiveExec, NULL,
8143 e1kR3SavePrep, e1kR3SaveExec, NULL,
8144 e1kR3LoadPrep, e1kR3LoadExec, e1kR3LoadDone);
8145 AssertRCReturn(rc, rc);
8146
8147 /* Set PCI config registers and register ourselves with the PCI bus. */
8148 PDMPCIDEV_ASSERT_VALID(pDevIns, pDevIns->apPciDevs[0]);
8149 e1kR3ConfigurePciDev(pDevIns->apPciDevs[0], pThis->eChip);
8150 rc = PDMDevHlpPCIRegister(pDevIns, pDevIns->apPciDevs[0]);
8151 AssertRCReturn(rc, rc);
8152
8153#ifdef E1K_WITH_MSI
8154 PDMMSIREG MsiReg;
8155 RT_ZERO(MsiReg);
8156 MsiReg.cMsiVectors = 1;
8157 MsiReg.iMsiCapOffset = 0x80;
8158 MsiReg.iMsiNextOffset = 0x0;
8159 MsiReg.fMsi64bit = false;
8160 rc = PDMDevHlpPCIRegisterMsi(pDevIns, &MsiReg);
8161 AssertRCReturn(rc, rc);
8162#endif
8163
8164 /*
8165 * Map our registers to memory space (region 0, see e1kR3ConfigurePciDev)
8166 * From the spec (regarding flags):
8167 * For registers that should be accessed as 32-bit double words,
8168 * partial writes (less than a 32-bit double word) is ignored.
8169 * Partial reads return all 32 bits of data regardless of the
8170 * byte enables.
8171 */
8172 rc = PDMDevHlpMmioCreateEx(pDevIns, E1K_MM_SIZE, IOMMMIO_FLAGS_READ_DWORD | IOMMMIO_FLAGS_WRITE_ONLY_DWORD,
8173 pDevIns->apPciDevs[0], 0 /*iPciRegion*/,
8174 e1kMMIOWrite, e1kMMIORead, NULL /*pfnFill*/, NULL /*pvUser*/, "E1000", &pThis->hMmioRegion);
8175 AssertRCReturn(rc, rc);
8176 rc = PDMDevHlpPCIIORegionRegisterMmio(pDevIns, 0, E1K_MM_SIZE, PCI_ADDRESS_SPACE_MEM, pThis->hMmioRegion, NULL);
8177 AssertRCReturn(rc, rc);
8178
8179 /* Map our registers to IO space (region 2, see e1kR3ConfigurePciDev) */
8180 static IOMIOPORTDESC const s_aExtDescs[] =
8181 {
8182 { "IOADDR", "IOADDR", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL },
8183 { "IODATA", "IODATA", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL }, { "unused", "unused", NULL, NULL },
8184 { NULL, NULL, NULL, NULL }
8185 };
8186 rc = PDMDevHlpIoPortCreate(pDevIns, E1K_IOPORT_SIZE, pDevIns->apPciDevs[0], 2 /*iPciRegion*/,
8187 e1kIOPortOut, e1kIOPortIn, NULL /*pvUser*/, "E1000", s_aExtDescs, &pThis->hIoPorts);
8188 AssertRCReturn(rc, rc);
8189 rc = PDMDevHlpPCIIORegionRegisterIo(pDevIns, 2, E1K_IOPORT_SIZE, pThis->hIoPorts);
8190 AssertRCReturn(rc, rc);
8191
8192 /* Create transmit queue */
8193 rc = PDMDevHlpTaskCreate(pDevIns, PDMTASK_F_RZ, "E1000-Xmit", e1kR3TxTaskCallback, NULL, &pThis->hTxTask);
8194 AssertRCReturn(rc, rc);
8195
8196#ifdef E1K_TX_DELAY
8197 /* Create Transmit Delay Timer */
8198 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3TxDelayTimer, pThis,
8199 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Xmit Delay", &pThis->hTXDTimer);
8200 AssertRCReturn(rc, rc);
8201 rc = PDMDevHlpTimerSetCritSect(pDevIns, pThis->hTXDTimer, &pThis->csTx);
8202 AssertRCReturn(rc, rc);
8203#endif /* E1K_TX_DELAY */
8204
8205//#ifdef E1K_USE_TX_TIMERS
8206 if (pThis->fTidEnabled)
8207 {
8208 /* Create Transmit Interrupt Delay Timer */
8209 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3TxIntDelayTimer, pThis,
8210 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Xmit IRQ Delay", &pThis->hTIDTimer);
8211 AssertRCReturn(rc, rc);
8212
8213# ifndef E1K_NO_TAD
8214 /* Create Transmit Absolute Delay Timer */
8215 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3TxAbsDelayTimer, pThis,
8216 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Xmit Abs Delay", &pThis->hTADTimer);
8217 AssertRCReturn(rc, rc);
8218# endif /* E1K_NO_TAD */
8219 }
8220//#endif /* E1K_USE_TX_TIMERS */
8221
8222#ifdef E1K_USE_RX_TIMERS
8223 /* Create Receive Interrupt Delay Timer */
8224 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3RxIntDelayTimer, pThis,
8225 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Recv IRQ Delay", &pThis->hRIDTimer);
8226 AssertRCReturn(rc, rc);
8227
8228 /* Create Receive Absolute Delay Timer */
8229 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3RxAbsDelayTimer, pThis,
8230 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Recv Abs Delay", &pThis->hRADTimer);
8231 AssertRCReturn(rc, rc);
8232#endif /* E1K_USE_RX_TIMERS */
8233
8234 /* Create Late Interrupt Timer */
8235 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3LateIntTimer, pThis,
8236 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Late IRQ", &pThis->hIntTimer);
8237 AssertRCReturn(rc, rc);
8238
8239 /* Create Link Up Timer */
8240 rc = PDMDevHlpTimerCreate(pDevIns, TMCLOCK_VIRTUAL, e1kR3LinkUpTimer, pThis,
8241 TMTIMER_FLAGS_NO_CRIT_SECT | TMTIMER_FLAGS_RING0, "E1000 Link Up", &pThis->hLUTimer);
8242 AssertRCReturn(rc, rc);
8243
8244 /* Register the info item */
8245 char szTmp[20];
8246 RTStrPrintf(szTmp, sizeof(szTmp), "e1k%d", iInstance);
8247 PDMDevHlpDBGFInfoRegister(pDevIns, szTmp, "E1000 info.", e1kR3Info);
8248
8249 /* Status driver */
8250 PPDMIBASE pBase;
8251 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThisCC->IBase, &pBase, "Status Port");
8252 if (RT_FAILURE(rc))
8253 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach the status LUN"));
8254 pThisCC->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
8255
8256 /* Network driver */
8257 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThisCC->IBase, &pThisCC->pDrvBase, "Network Port");
8258 if (RT_SUCCESS(rc))
8259 {
8260 pThisCC->pDrvR3 = PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMINETWORKUP);
8261 AssertMsgReturn(pThisCC->pDrvR3, ("Failed to obtain the PDMINETWORKUP interface!\n"), VERR_PDM_MISSING_INTERFACE_BELOW);
8262
8263#if 0 /** @todo @bugref{9218} ring-0 driver stuff */
8264 pThisR0->pDrvR0 = PDMIBASER0_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASER0), PDMINETWORKUP);
8265 pThisRC->pDrvRC = PDMIBASERC_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThisCC->pDrvBase, PDMIBASERC), PDMINETWORKUP);
8266#endif
8267 /* Mark device as attached. */
8268 pThis->fIsAttached = true;
8269 }
8270 else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER
8271 || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME)
8272 {
8273 /* No error! */
8274 E1kLog(("%s This adapter is not attached to any network!\n", pThis->szPrf));
8275 }
8276 else
8277 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Failed to attach the network LUN"));
8278
8279 rc = PDMDevHlpSUPSemEventCreate(pDevIns, &pThis->hEventMoreRxDescAvail);
8280 AssertRCReturn(rc, rc);
8281
8282 rc = e1kR3InitDebugHelpers();
8283 AssertRCReturn(rc, rc);
8284
8285 e1kR3HardReset(pDevIns, pThis, pThisCC);
8286
8287 /*
8288 * Register statistics.
8289 * The /Public/ bits are official and used by session info in the GUI.
8290 */
8291 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
8292 "Amount of data received", "/Public/NetAdapter/%u/BytesReceived", uStatNo);
8293 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES,
8294 "Amount of data transmitted", "/Public/NetAdapter/%u/BytesTransmitted", uStatNo);
8295 PDMDevHlpSTAMRegisterF(pDevIns, &pDevIns->iInstance, STAMTYPE_U32, STAMVISIBILITY_ALWAYS, STAMUNIT_NONE,
8296 "Device instance number", "/Public/NetAdapter/%u/%s", uStatNo, pDevIns->pReg->szName);
8297
8298 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, "ReceiveBytes", STAMUNIT_BYTES, "Amount of data received");
8299 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, "TransmitBytes", STAMUNIT_BYTES, "Amount of data transmitted");
8300
8301#if defined(VBOX_WITH_STATISTICS)
8302 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOReadRZ, STAMTYPE_PROFILE, "MMIO/ReadRZ", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO reads in RZ");
8303 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOReadR3, STAMTYPE_PROFILE, "MMIO/ReadR3", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO reads in R3");
8304 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOWriteRZ, STAMTYPE_PROFILE, "MMIO/WriteRZ", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO writes in RZ");
8305 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatMMIOWriteR3, STAMTYPE_PROFILE, "MMIO/WriteR3", STAMUNIT_TICKS_PER_CALL, "Profiling MMIO writes in R3");
8306 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatEEPROMRead, STAMTYPE_PROFILE, "EEPROM/Read", STAMUNIT_TICKS_PER_CALL, "Profiling EEPROM reads");
8307 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatEEPROMWrite, STAMTYPE_PROFILE, "EEPROM/Write", STAMUNIT_TICKS_PER_CALL, "Profiling EEPROM writes");
8308 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOReadRZ, STAMTYPE_PROFILE, "IO/ReadRZ", STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in RZ");
8309 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOReadR3, STAMTYPE_PROFILE, "IO/ReadR3", STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in R3");
8310 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOWriteRZ, STAMTYPE_PROFILE, "IO/WriteRZ", STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in RZ");
8311 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIOWriteR3, STAMTYPE_PROFILE, "IO/WriteR3", STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in R3");
8312 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatLateIntTimer, STAMTYPE_PROFILE, "LateInt/Timer", STAMUNIT_TICKS_PER_CALL, "Profiling late int timer");
8313 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatLateInts, STAMTYPE_COUNTER, "LateInt/Occured", STAMUNIT_OCCURENCES, "Number of late interrupts");
8314 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIntsRaised, STAMTYPE_COUNTER, "Interrupts/Raised", STAMUNIT_OCCURENCES, "Number of raised interrupts");
8315 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatIntsPrevented, STAMTYPE_COUNTER, "Interrupts/Prevented", STAMUNIT_OCCURENCES, "Number of prevented interrupts");
8316 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceive, STAMTYPE_PROFILE, "Receive/Total", STAMUNIT_TICKS_PER_CALL, "Profiling receive");
8317 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveCRC, STAMTYPE_PROFILE, "Receive/CRC", STAMUNIT_TICKS_PER_CALL, "Profiling receive checksumming");
8318 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveFilter, STAMTYPE_PROFILE, "Receive/Filter", STAMUNIT_TICKS_PER_CALL, "Profiling receive filtering");
8319 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatReceiveStore, STAMTYPE_PROFILE, "Receive/Store", STAMUNIT_TICKS_PER_CALL, "Profiling receive storing");
8320 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatRxOverflow, STAMTYPE_PROFILE, "RxOverflow", STAMUNIT_TICKS_PER_OCCURENCE, "Profiling RX overflows");
8321 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatRxOverflowWakeupRZ, STAMTYPE_COUNTER, "RxOverflowWakeupRZ", STAMUNIT_OCCURENCES, "Nr of RX overflow wakeups in RZ");
8322 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatRxOverflowWakeupR3, STAMTYPE_COUNTER, "RxOverflowWakeupR3", STAMUNIT_OCCURENCES, "Nr of RX overflow wakeups in R3");
8323 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitRZ, STAMTYPE_PROFILE, "Transmit/TotalRZ", STAMUNIT_TICKS_PER_CALL, "Profiling transmits in RZ");
8324 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitR3, STAMTYPE_PROFILE, "Transmit/TotalR3", STAMUNIT_TICKS_PER_CALL, "Profiling transmits in R3");
8325 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitSendRZ, STAMTYPE_PROFILE, "Transmit/SendRZ", STAMUNIT_TICKS_PER_CALL, "Profiling send transmit in RZ");
8326 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTransmitSendR3, STAMTYPE_PROFILE, "Transmit/SendR3", STAMUNIT_TICKS_PER_CALL, "Profiling send transmit in R3");
8327
8328 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescCtxNormal, STAMTYPE_COUNTER, "TxDesc/ContexNormal", STAMUNIT_OCCURENCES, "Number of normal context descriptors");
8329 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescCtxTSE, STAMTYPE_COUNTER, "TxDesc/ContextTSE", STAMUNIT_OCCURENCES, "Number of TSE context descriptors");
8330 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescData, STAMTYPE_COUNTER, "TxDesc/Data", STAMUNIT_OCCURENCES, "Number of TX data descriptors");
8331 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescLegacy, STAMTYPE_COUNTER, "TxDesc/Legacy", STAMUNIT_OCCURENCES, "Number of TX legacy descriptors");
8332 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxDescTSEData, STAMTYPE_COUNTER, "TxDesc/TSEData", STAMUNIT_OCCURENCES, "Number of TX TSE data descriptors");
8333 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxPathFallback, STAMTYPE_COUNTER, "TxPath/Fallback", STAMUNIT_OCCURENCES, "Fallback TSE descriptor path");
8334 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxPathGSO, STAMTYPE_COUNTER, "TxPath/GSO", STAMUNIT_OCCURENCES, "GSO TSE descriptor path");
8335 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatTxPathRegular, STAMTYPE_COUNTER, "TxPath/Normal", STAMUNIT_OCCURENCES, "Regular descriptor path");
8336 PDMDevHlpSTAMRegister(pDevIns, &pThis->StatPHYAccesses, STAMTYPE_COUNTER, "PHYAccesses", STAMUNIT_OCCURENCES, "Number of PHY accesses");
8337 for (unsigned iReg = 0; iReg < E1K_NUM_OF_REGS; iReg++)
8338 {
8339 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatRegReads[iReg], STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
8340 g_aE1kRegMap[iReg].name, "Regs/%s-Reads", g_aE1kRegMap[iReg].abbrev);
8341 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatRegWrites[iReg], STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES,
8342 g_aE1kRegMap[iReg].name, "Regs/%s-Writes", g_aE1kRegMap[iReg].abbrev);
8343 }
8344#endif /* VBOX_WITH_STATISTICS */
8345
8346#ifdef E1K_INT_STATS
8347 PDMDevHlpSTAMRegister(pDevIns, &pThis->u64ArmedAt, STAMTYPE_U64, "u64ArmedAt", STAMUNIT_NS, NULL);
8348 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatMaxTxDelay, STAMTYPE_U64, "uStatMaxTxDelay", STAMUNIT_NS, NULL);
8349 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatInt, STAMTYPE_U32, "uStatInt", STAMUNIT_NS, NULL);
8350 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntTry, STAMTYPE_U32, "uStatIntTry", STAMUNIT_NS, NULL);
8351 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntLower, STAMTYPE_U32, "uStatIntLower", STAMUNIT_NS, NULL);
8352 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatNoIntICR, STAMTYPE_U32, "uStatNoIntICR", STAMUNIT_NS, NULL);
8353 PDMDevHlpSTAMRegister(pDevIns, &pThis->iStatIntLost, STAMTYPE_U32, "iStatIntLost", STAMUNIT_NS, NULL);
8354 PDMDevHlpSTAMRegister(pDevIns, &pThis->iStatIntLostOne, STAMTYPE_U32, "iStatIntLostOne", STAMUNIT_NS, NULL);
8355 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntIMS, STAMTYPE_U32, "uStatIntIMS", STAMUNIT_NS, NULL);
8356 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntSkip, STAMTYPE_U32, "uStatIntSkip", STAMUNIT_NS, NULL);
8357 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntLate, STAMTYPE_U32, "uStatIntLate", STAMUNIT_NS, NULL);
8358 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntMasked, STAMTYPE_U32, "uStatIntMasked", STAMUNIT_NS, NULL);
8359 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntEarly, STAMTYPE_U32, "uStatIntEarly", STAMUNIT_NS, NULL);
8360 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntRx, STAMTYPE_U32, "uStatIntRx", STAMUNIT_NS, NULL);
8361 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntTx, STAMTYPE_U32, "uStatIntTx", STAMUNIT_NS, NULL);
8362 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntICS, STAMTYPE_U32, "uStatIntICS", STAMUNIT_NS, NULL);
8363 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntRDTR, STAMTYPE_U32, "uStatIntRDTR", STAMUNIT_NS, NULL);
8364 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntRXDMT0, STAMTYPE_U32, "uStatIntRXDMT0", STAMUNIT_NS, NULL);
8365 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatIntTXQE, STAMTYPE_U32, "uStatIntTXQE", STAMUNIT_NS, NULL);
8366 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxNoRS, STAMTYPE_U32, "uStatTxNoRS", STAMUNIT_NS, NULL);
8367 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxIDE, STAMTYPE_U32, "uStatTxIDE", STAMUNIT_NS, NULL);
8368 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxDelayed, STAMTYPE_U32, "uStatTxDelayed", STAMUNIT_NS, NULL);
8369 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxDelayExp, STAMTYPE_U32, "uStatTxDelayExp", STAMUNIT_NS, NULL);
8370 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTAD, STAMTYPE_U32, "uStatTAD", STAMUNIT_NS, NULL);
8371 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTID, STAMTYPE_U32, "uStatTID", STAMUNIT_NS, NULL);
8372 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatRAD, STAMTYPE_U32, "uStatRAD", STAMUNIT_NS, NULL);
8373 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatRID, STAMTYPE_U32, "uStatRID", STAMUNIT_NS, NULL);
8374 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatRxFrm, STAMTYPE_U32, "uStatRxFrm", STAMUNIT_NS, NULL);
8375 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxFrm, STAMTYPE_U32, "uStatTxFrm", STAMUNIT_NS, NULL);
8376 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatDescCtx, STAMTYPE_U32, "uStatDescCtx", STAMUNIT_NS, NULL);
8377 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatDescDat, STAMTYPE_U32, "uStatDescDat", STAMUNIT_NS, NULL);
8378 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatDescLeg, STAMTYPE_U32, "uStatDescLeg", STAMUNIT_NS, NULL);
8379 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx1514, STAMTYPE_U32, "uStatTx1514", STAMUNIT_NS, NULL);
8380 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx2962, STAMTYPE_U32, "uStatTx2962", STAMUNIT_NS, NULL);
8381 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx4410, STAMTYPE_U32, "uStatTx4410", STAMUNIT_NS, NULL);
8382 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx5858, STAMTYPE_U32, "uStatTx5858", STAMUNIT_NS, NULL);
8383 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx7306, STAMTYPE_U32, "uStatTx7306", STAMUNIT_NS, NULL);
8384 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx8754, STAMTYPE_U32, "uStatTx8754", STAMUNIT_NS, NULL);
8385 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx16384, STAMTYPE_U32, "uStatTx16384", STAMUNIT_NS, NULL);
8386 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTx32768, STAMTYPE_U32, "uStatTx32768", STAMUNIT_NS, NULL);
8387 PDMDevHlpSTAMRegister(pDevIns, &pThis->uStatTxLarge, STAMTYPE_U32, "uStatTxLarge", STAMUNIT_NS, NULL);
8388#endif /* E1K_INT_STATS */
8389
8390 return VINF_SUCCESS;
8391}
8392
8393#else /* !IN_RING3 */
8394
8395/**
8396 * @callback_method_impl{PDMDEVREGR0,pfnConstruct}
8397 */
8398static DECLCALLBACK(int) e1kRZConstruct(PPDMDEVINS pDevIns)
8399{
8400 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
8401 PE1KSTATE pThis = PDMDEVINS_2_DATA(pDevIns, PE1KSTATE);
8402 PE1KSTATECC pThisCC = PDMDEVINS_2_DATA_CC(pDevIns, PE1KSTATECC);
8403
8404 /* Initialize context specific state data: */
8405 pThisCC->CTX_SUFF(pDevIns) = pDevIns;
8406 /** @todo @bugref{9218} ring-0 driver stuff */
8407 pThisCC->CTX_SUFF(pDrv) = NULL;
8408 pThisCC->CTX_SUFF(pTxSg) = NULL;
8409
8410 /* Configure critical sections the same way: */
8411 int rc = PDMDevHlpSetDeviceCritSect(pDevIns, PDMDevHlpCritSectGetNop(pDevIns));
8412 AssertRCReturn(rc, rc);
8413
8414 /* Set up MMIO and I/O port callbacks for this context: */
8415 rc = PDMDevHlpMmioSetUpContext(pDevIns, pThis->hMmioRegion, e1kMMIOWrite, e1kMMIORead, NULL /*pvUser*/);
8416 AssertRCReturn(rc, rc);
8417
8418 rc = PDMDevHlpIoPortSetUpContext(pDevIns, pThis->hIoPorts, e1kIOPortOut, e1kIOPortIn, NULL /*pvUser*/);
8419 AssertRCReturn(rc, rc);
8420
8421 return VINF_SUCCESS;
8422}
8423
8424#endif /* !IN_RING3 */
8425
8426/**
8427 * The device registration structure.
8428 */
8429const PDMDEVREG g_DeviceE1000 =
8430{
8431 /* .u32version = */ PDM_DEVREG_VERSION,
8432 /* .uReserved0 = */ 0,
8433 /* .szName = */ "e1000",
8434 /* .fFlags = */ PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RZ | PDM_DEVREG_FLAGS_NEW_STYLE,
8435 /* .fClass = */ PDM_DEVREG_CLASS_NETWORK,
8436 /* .cMaxInstances = */ ~0U,
8437 /* .uSharedVersion = */ 42,
8438 /* .cbInstanceShared = */ sizeof(E1KSTATE),
8439 /* .cbInstanceCC = */ sizeof(E1KSTATECC),
8440 /* .cbInstanceRC = */ sizeof(E1KSTATERC),
8441 /* .cMaxPciDevices = */ 1,
8442 /* .cMaxMsixVectors = */ 0,
8443 /* .pszDescription = */ "Intel PRO/1000 MT Desktop Ethernet.",
8444#if defined(IN_RING3)
8445 /* .pszRCMod = */ "VBoxDDRC.rc",
8446 /* .pszR0Mod = */ "VBoxDDR0.r0",
8447 /* .pfnConstruct = */ e1kR3Construct,
8448 /* .pfnDestruct = */ e1kR3Destruct,
8449 /* .pfnRelocate = */ e1kR3Relocate,
8450 /* .pfnMemSetup = */ NULL,
8451 /* .pfnPowerOn = */ NULL,
8452 /* .pfnReset = */ e1kR3Reset,
8453 /* .pfnSuspend = */ e1kR3Suspend,
8454 /* .pfnResume = */ NULL,
8455 /* .pfnAttach = */ e1kR3Attach,
8456 /* .pfnDeatch = */ e1kR3Detach,
8457 /* .pfnQueryInterface = */ NULL,
8458 /* .pfnInitComplete = */ NULL,
8459 /* .pfnPowerOff = */ e1kR3PowerOff,
8460 /* .pfnSoftReset = */ NULL,
8461 /* .pfnReserved0 = */ NULL,
8462 /* .pfnReserved1 = */ NULL,
8463 /* .pfnReserved2 = */ NULL,
8464 /* .pfnReserved3 = */ NULL,
8465 /* .pfnReserved4 = */ NULL,
8466 /* .pfnReserved5 = */ NULL,
8467 /* .pfnReserved6 = */ NULL,
8468 /* .pfnReserved7 = */ NULL,
8469#elif defined(IN_RING0)
8470 /* .pfnEarlyConstruct = */ NULL,
8471 /* .pfnConstruct = */ e1kRZConstruct,
8472 /* .pfnDestruct = */ NULL,
8473 /* .pfnFinalDestruct = */ NULL,
8474 /* .pfnRequest = */ NULL,
8475 /* .pfnReserved0 = */ NULL,
8476 /* .pfnReserved1 = */ NULL,
8477 /* .pfnReserved2 = */ NULL,
8478 /* .pfnReserved3 = */ NULL,
8479 /* .pfnReserved4 = */ NULL,
8480 /* .pfnReserved5 = */ NULL,
8481 /* .pfnReserved6 = */ NULL,
8482 /* .pfnReserved7 = */ NULL,
8483#elif defined(IN_RC)
8484 /* .pfnConstruct = */ e1kRZConstruct,
8485 /* .pfnReserved0 = */ NULL,
8486 /* .pfnReserved1 = */ NULL,
8487 /* .pfnReserved2 = */ NULL,
8488 /* .pfnReserved3 = */ NULL,
8489 /* .pfnReserved4 = */ NULL,
8490 /* .pfnReserved5 = */ NULL,
8491 /* .pfnReserved6 = */ NULL,
8492 /* .pfnReserved7 = */ NULL,
8493#else
8494# error "Not in IN_RING3, IN_RING0 or IN_RC!"
8495#endif
8496 /* .u32VersionEnd = */ PDM_DEVREG_VERSION
8497};
8498
8499#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette