VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DevPCNet.cpp@ 40392

Last change on this file since 40392 was 40282, checked in by vboxsync, 13 years ago

*: gcc-4.7: ~0 => ~0U in initializers (warning: narrowing conversion of -1' from int' to `unsigned int' inside { } is ill-formed in C++11 [-Wnarrowing])

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 205.5 KB
Line 
1/* $Id: DevPCNet.cpp 40282 2012-02-28 21:02:40Z vboxsync $ */
2/** @file
3 * DevPCNet - AMD PCnet-PCI II / PCnet-FAST III (Am79C970A / Am79C973) Ethernet Controller Emulation.
4 *
5 * This software was written to be compatible with the specifications:
6 * AMD Am79C970A PCnet-PCI II Ethernet Controller Data-Sheet
7 * AMD Publication# 19436 Rev:E Amendment/0 Issue Date: June 2000
8 * and
9 * todo
10 */
11
12/*
13 * Copyright (C) 2006-2011 Oracle Corporation
14 *
15 * This file is part of VirtualBox Open Source Edition (OSE), as
16 * available from http://www.virtualbox.org. This file is free software;
17 * you can redistribute it and/or modify it under the terms of the GNU
18 * General Public License (GPL) as published by the Free Software
19 * Foundation, in version 2 as it comes in the "COPYING" file of the
20 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
21 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
22 * --------------------------------------------------------------------
23 *
24 * This code is based on:
25 *
26 * AMD PC-Net II (Am79C970A) emulation
27 *
28 * Copyright (c) 2004 Antony T Curtis
29 *
30 * Permission is hereby granted, free of charge, to any person obtaining a copy
31 * of this software and associated documentation files (the "Software"), to deal
32 * in the Software without restriction, including without limitation the rights
33 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
34 * copies of the Software, and to permit persons to whom the Software is
35 * furnished to do so, subject to the following conditions:
36 *
37 * The above copyright notice and this permission notice shall be included in
38 * all copies or substantial portions of the Software.
39 *
40 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
41 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
42 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
43 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
44 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
45 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
46 * THE SOFTWARE.
47 */
48
49
50/*******************************************************************************
51* Header Files *
52*******************************************************************************/
53#define LOG_GROUP LOG_GROUP_DEV_PCNET
54#include <VBox/vmm/pdmdev.h>
55#include <VBox/vmm/pdmnetifs.h>
56#include <VBox/vmm/pgm.h>
57#include <VBox/DevPCNet.h>
58#include <iprt/asm.h>
59#include <iprt/assert.h>
60#include <iprt/critsect.h>
61#include <iprt/net.h>
62#include <iprt/string.h>
63#include <iprt/time.h>
64#ifdef IN_RING3
65# include <iprt/mem.h>
66# include <iprt/semaphore.h>
67# include <iprt/uuid.h>
68#endif
69
70#include "VBoxDD.h"
71
72/* Enable this to catch writes to the ring descriptors instead of using excessive polling */
73/* #define PCNET_NO_POLLING */
74
75/* Enable to handle frequent io reads in the guest context (recommended) */
76#define PCNET_GC_ENABLED
77
78#if defined(LOG_ENABLED)
79#define PCNET_DEBUG_IO
80#define PCNET_DEBUG_BCR
81#define PCNET_DEBUG_CSR
82#define PCNET_DEBUG_RMD
83#define PCNET_DEBUG_TMD
84#define PCNET_DEBUG_MATCH
85#define PCNET_DEBUG_MII
86#endif
87
88#define PCNET_IOPORT_SIZE 0x20
89#define PCNET_PNPMMIO_SIZE 0x20
90
91#define PCNET_SAVEDSTATE_VERSION 10
92
93#define BCR_MAX_RAP 50
94#define MII_MAX_REG 32
95#define CSR_MAX_REG 128
96
97/* Maximum number of times we report a link down to the guest (failure to send frame) */
98#define PCNET_MAX_LINKDOWN_REPORTED 3
99
100/* Maximum frame size we handle */
101#define MAX_FRAME 1536
102
103
104typedef struct PCNetState_st PCNetState;
105
106/**
107 * PCNET state.
108 *
109 * @extends PCIDEVICE
110 * @implements PDMIBASE
111 * @implements PDMINETWORKDOWN
112 * @implements PDMINETWORKCONFIG
113 * @implements PDMILEDPORTS
114 */
115struct PCNetState_st
116{
117 PCIDEVICE PciDev;
118
119 /** Pointer to the device instance - R3. */
120 PPDMDEVINSR3 pDevInsR3;
121 /** Transmit signaller - R3. */
122 R3PTRTYPE(PPDMQUEUE) pXmitQueueR3;
123 /** Receive signaller - R3. */
124 R3PTRTYPE(PPDMQUEUE) pCanRxQueueR3;
125 /** Pointer to the connector of the attached network driver - R3. */
126 PPDMINETWORKUPR3 pDrvR3;
127 /** Pointer to the attached network driver. */
128 R3PTRTYPE(PPDMIBASE) pDrvBase;
129 /** LUN\#0 + status LUN: The base interface. */
130 PDMIBASE IBase;
131 /** LUN\#0: The network port interface. */
132 PDMINETWORKDOWN INetworkDown;
133 /** LUN\#0: The network config port interface. */
134 PDMINETWORKCONFIG INetworkConfig;
135 /** The shared memory used for the private interface - R3. */
136 R3PTRTYPE(PPCNETGUESTSHAREDMEMORY) pSharedMMIOR3;
137 /** Software Interrupt timer - R3. */
138 PTMTIMERR3 pTimerSoftIntR3;
139#ifndef PCNET_NO_POLLING
140 /** Poll timer - R3. */
141 PTMTIMERR3 pTimerPollR3;
142#endif
143 /** Restore timer.
144 * This is used to disconnect and reconnect the link after a restore. */
145 PTMTIMERR3 pTimerRestore;
146
147 /** Pointer to the device instance - R0. */
148 PPDMDEVINSR0 pDevInsR0;
149 /** Receive signaller - R0. */
150 R0PTRTYPE(PPDMQUEUE) pCanRxQueueR0;
151 /** Transmit signaller - R0. */
152 R0PTRTYPE(PPDMQUEUE) pXmitQueueR0;
153 /** Pointer to the connector of the attached network driver - R0. */
154 PPDMINETWORKUPR0 pDrvR0;
155 /** The shared memory used for the private interface - R0. */
156 R0PTRTYPE(PPCNETGUESTSHAREDMEMORY) pSharedMMIOR0;
157 /** Software Interrupt timer - R0. */
158 PTMTIMERR0 pTimerSoftIntR0;
159#ifndef PCNET_NO_POLLING
160 /** Poll timer - R0. */
161 PTMTIMERR0 pTimerPollR0;
162#endif
163
164 /** Pointer to the device instance - RC. */
165 PPDMDEVINSRC pDevInsRC;
166 /** Receive signaller - RC. */
167 RCPTRTYPE(PPDMQUEUE) pCanRxQueueRC;
168 /** Transmit signaller - RC. */
169 RCPTRTYPE(PPDMQUEUE) pXmitQueueRC;
170 /** Pointer to the connector of the attached network driver - RC. */
171 PPDMINETWORKUPRC pDrvRC;
172 /** The shared memory used for the private interface - RC. */
173 RCPTRTYPE(PPCNETGUESTSHAREDMEMORY) pSharedMMIORC;
174 /** Software Interrupt timer - RC. */
175 PTMTIMERRC pTimerSoftIntRC;
176#ifndef PCNET_NO_POLLING
177 /** Poll timer - RC. */
178 PTMTIMERRC pTimerPollRC;
179#endif
180
181//#if HC_ARCH_BITS == 64
182 uint32_t Alignment1;
183//#endif
184
185 /** Register Address Pointer */
186 uint32_t u32RAP;
187 /** Internal interrupt service */
188 int32_t iISR;
189 /** ??? */
190 uint32_t u32Lnkst;
191 /** Address of the RX descriptor table (ring). Loaded at init. */
192 RTGCPHYS32 GCRDRA;
193 /** Address of the TX descriptor table (ring). Loaded at init. */
194 RTGCPHYS32 GCTDRA;
195 uint8_t aPROM[16];
196 uint16_t aCSR[CSR_MAX_REG];
197 uint16_t aBCR[BCR_MAX_RAP];
198 uint16_t aMII[MII_MAX_REG];
199 /** Holds the bits which were really seen by the guest. Relevant are bits
200 * 8..14 (IDON, TINT, RINT, MERR, MISS, CERR, BABL). We don't allow the
201 * guest to clear any of these bits (by writing a ONE) before a bit was
202 * seen by the guest. */
203 uint16_t u16CSR0LastSeenByGuest;
204 uint16_t Alignment2[HC_ARCH_BITS == 32 ? 2 : 2];
205 /** Last time we polled the queues */
206 uint64_t u64LastPoll;
207
208 /** The loopback transmit buffer (avoid stack allocations). */
209 uint8_t abLoopBuf[4096];
210 /** The recv buffer. */
211 uint8_t abRecvBuf[4096];
212
213 /** Unused / padding. */
214 uint32_t u32Unused;
215
216 /** Size of a RX/TX descriptor (8 or 16 bytes according to SWSTYLE */
217 int iLog2DescSize;
218 /** Bits 16..23 in 16-bit mode */
219 RTGCPHYS32 GCUpperPhys;
220
221 /** Base address of the MMIO region. */
222 RTGCPHYS32 MMIOBase;
223 /** Base port of the I/O space region. */
224 RTIOPORT IOPortBase;
225 /** If set the link is currently up. */
226 bool fLinkUp;
227 /** If set the link is temporarily down because of a saved state load. */
228 bool fLinkTempDown;
229
230 /** Number of times we've reported the link down. */
231 RTUINT cLinkDownReported;
232 /** The configured MAC address. */
233 RTMAC MacConfigured;
234 /** Alignment padding. */
235 uint8_t Alignment4[HC_ARCH_BITS == 64 ? 2 : 2];
236
237 /** The LED. */
238 PDMLED Led;
239 /** Status LUN: The LED ports. */
240 PDMILEDPORTS ILeds;
241 /** Partner of ILeds. */
242 R3PTRTYPE(PPDMILEDCONNECTORS) pLedsConnector;
243
244 /** Access critical section. */
245 PDMCRITSECT CritSect;
246 /** Event semaphore for blocking on receive. */
247 RTSEMEVENT hEventOutOfRxSpace;
248 /** We are waiting/about to start waiting for more receive buffers. */
249 bool volatile fMaybeOutOfSpace;
250 /** True if we signal the guest that RX packets are missing. */
251 bool fSignalRxMiss;
252 uint8_t Alignment5[HC_ARCH_BITS == 64 ? 2 : 6];
253
254#ifdef PCNET_NO_POLLING
255 RTGCPHYS32 TDRAPhysOld;
256 uint32_t cbTDRAOld;
257
258 RTGCPHYS32 RDRAPhysOld;
259 uint32_t cbRDRAOld;
260
261 DECLRCCALLBACKMEMBER(int, pfnEMInterpretInstructionRC, (PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, uint32_t *pcbSize));
262 DECLR0CALLBACKMEMBER(int, pfnEMInterpretInstructionR0, (PVM pVM, PCPUMCTXCORE pRegFrame, RTGCPTR pvFault, uint32_t *pcbSize));
263#endif
264
265 /** Error counter for bad receive descriptors. */
266 uint32_t uCntBadRMD;
267
268 /** True if host and guest admitted to use the private interface. */
269 bool fPrivIfEnabled;
270 bool fGCEnabled;
271 bool fR0Enabled;
272 bool fAm79C973;
273 uint32_t u32LinkSpeed;
274
275 STAMCOUNTER StatReceiveBytes;
276 STAMCOUNTER StatTransmitBytes;
277#ifdef VBOX_WITH_STATISTICS
278 STAMPROFILEADV StatMMIOReadRZ;
279 STAMPROFILEADV StatMMIOReadR3;
280 STAMPROFILEADV StatMMIOWriteRZ;
281 STAMPROFILEADV StatMMIOWriteR3;
282 STAMPROFILEADV StatAPROMRead;
283 STAMPROFILEADV StatAPROMWrite;
284 STAMPROFILEADV StatIOReadRZ;
285 STAMPROFILEADV StatIOReadR3;
286 STAMPROFILEADV StatIOWriteRZ;
287 STAMPROFILEADV StatIOWriteR3;
288 STAMPROFILEADV StatTimer;
289 STAMPROFILEADV StatReceive;
290 STAMPROFILEADV StatTransmitR3;
291 STAMPROFILEADV StatTransmitRZ;
292 STAMCOUNTER StatTransmitCase1;
293 STAMCOUNTER StatTransmitCase2;
294 STAMPROFILE StatTransmitSendR3;
295 STAMPROFILE StatTransmitSendRZ;
296 STAMPROFILEADV StatTdtePollRZ;
297 STAMPROFILEADV StatTdtePollR3;
298 STAMPROFILEADV StatTmdStoreRZ;
299 STAMPROFILEADV StatTmdStoreR3;
300 STAMPROFILEADV StatRdtePollR3;
301 STAMPROFILEADV StatRdtePollRZ;
302 STAMPROFILE StatRxOverflow;
303 STAMCOUNTER StatRxOverflowWakeup;
304 STAMCOUNTER aStatXmitFlush[16];
305 STAMCOUNTER aStatXmitChainCounts[16];
306 STAMCOUNTER StatXmitSkipCurrent;
307 STAMPROFILEADV StatInterrupt;
308 STAMPROFILEADV StatPollTimer;
309 STAMCOUNTER StatMIIReads;
310# ifdef PCNET_NO_POLLING
311 STAMCOUNTER StatRCVRingWrite;
312 STAMCOUNTER StatTXRingWrite;
313 STAMCOUNTER StatRingWriteR3;
314 STAMCOUNTER StatRingWriteR0;
315 STAMCOUNTER StatRingWriteRC;
316
317 STAMCOUNTER StatRingWriteFailedR3;
318 STAMCOUNTER StatRingWriteFailedR0;
319 STAMCOUNTER StatRingWriteFailedRC;
320
321 STAMCOUNTER StatRingWriteOutsideR3;
322 STAMCOUNTER StatRingWriteOutsideR0;
323 STAMCOUNTER StatRingWriteOutsideRC;
324# endif
325#endif /* VBOX_WITH_STATISTICS */
326};
327//AssertCompileMemberAlignment(PCNetState, StatReceiveBytes, 8);
328
329#define PCNETSTATE_2_DEVINS(pPCNet) ((pPCNet)->CTX_SUFF(pDevIns))
330#define PCIDEV_2_PCNETSTATE(pPciDev) ((PCNetState *)(pPciDev))
331#define PCNET_INST_NR (PCNETSTATE_2_DEVINS(pThis)->iInstance)
332
333/* BUS CONFIGURATION REGISTERS */
334#define BCR_MSRDA 0
335#define BCR_MSWRA 1
336#define BCR_MC 2
337#define BCR_RESERVED3 3
338#define BCR_LNKST 4
339#define BCR_LED1 5
340#define BCR_LED2 6
341#define BCR_LED3 7
342#define BCR_RESERVED8 8
343#define BCR_FDC 9
344/* 10 - 15 = reserved */
345#define BCR_IOBASEL 16 /* Reserved */
346#define BCR_IOBASEU 16 /* Reserved */
347#define BCR_BSBC 18
348#define BCR_EECAS 19
349#define BCR_SWS 20
350#define BCR_INTCON 21 /* Reserved */
351#define BCR_PLAT 22
352#define BCR_PCISVID 23
353#define BCR_PCISID 24
354#define BCR_SRAMSIZ 25
355#define BCR_SRAMB 26
356#define BCR_SRAMIC 27
357#define BCR_EBADDRL 28
358#define BCR_EBADDRU 29
359#define BCR_EBD 30
360#define BCR_STVAL 31
361#define BCR_MIICAS 32
362#define BCR_MIIADDR 33
363#define BCR_MIIMDR 34
364#define BCR_PCIVID 35
365#define BCR_PMC_A 36
366#define BCR_DATA0 37
367#define BCR_DATA1 38
368#define BCR_DATA2 39
369#define BCR_DATA3 40
370#define BCR_DATA4 41
371#define BCR_DATA5 42
372#define BCR_DATA6 43
373#define BCR_DATA7 44
374#define BCR_PMR1 45
375#define BCR_PMR2 46
376#define BCR_PMR3 47
377
378#define BCR_DWIO(S) !!((S)->aBCR[BCR_BSBC] & 0x0080)
379#define BCR_SSIZE32(S) !!((S)->aBCR[BCR_SWS ] & 0x0100)
380#define BCR_SWSTYLE(S) ((S)->aBCR[BCR_SWS ] & 0x00FF)
381
382#define CSR_INIT(S) !!((S)->aCSR[0] & 0x0001) /**< Init assertion */
383#define CSR_STRT(S) !!((S)->aCSR[0] & 0x0002) /**< Start assertion */
384#define CSR_STOP(S) !!((S)->aCSR[0] & 0x0004) /**< Stop assertion */
385#define CSR_TDMD(S) !!((S)->aCSR[0] & 0x0008) /**< Transmit demand. (perform xmit poll now (readable, settable, not clearable) */
386#define CSR_TXON(S) !!((S)->aCSR[0] & 0x0010) /**< Transmit on (readonly) */
387#define CSR_RXON(S) !!((S)->aCSR[0] & 0x0020) /**< Receive On */
388#define CSR_INEA(S) !!((S)->aCSR[0] & 0x0040) /**< Interrupt Enable */
389#define CSR_LAPPEN(S) !!((S)->aCSR[3] & 0x0020) /**< Look Ahead Packet Processing Enable */
390#define CSR_DXSUFLO(S) !!((S)->aCSR[3] & 0x0040) /**< Disable Transmit Stop on Underflow error */
391#define CSR_ASTRP_RCV(S) !!((S)->aCSR[4] & 0x0400) /**< Auto Strip Receive */
392#define CSR_DPOLL(S) !!((S)->aCSR[4] & 0x1000) /**< Disable Transmit Polling */
393#define CSR_SPND(S) !!((S)->aCSR[5] & 0x0001) /**< Suspend */
394#define CSR_LTINTEN(S) !!((S)->aCSR[5] & 0x4000) /**< Last Transmit Interrupt Enable */
395#define CSR_TOKINTD(S) !!((S)->aCSR[5] & 0x8000) /**< Transmit OK Interrupt Disable */
396
397#define CSR_STINT !!((S)->aCSR[7] & 0x0800) /**< Software Timer Interrupt */
398#define CSR_STINTE !!((S)->aCSR[7] & 0x0400) /**< Software Timer Interrupt Enable */
399
400#define CSR_DRX(S) !!((S)->aCSR[15] & 0x0001) /**< Disable Receiver */
401#define CSR_DTX(S) !!((S)->aCSR[15] & 0x0002) /**< Disable Transmit */
402#define CSR_LOOP(S) !!((S)->aCSR[15] & 0x0004) /**< Loopback Enable */
403#define CSR_DRCVPA(S) !!((S)->aCSR[15] & 0x2000) /**< Disable Receive Physical Address */
404#define CSR_DRCVBC(S) !!((S)->aCSR[15] & 0x4000) /**< Disable Receive Broadcast */
405#define CSR_PROM(S) !!((S)->aCSR[15] & 0x8000) /**< Promiscuous Mode */
406
407#if !defined(RT_ARCH_X86) && !defined(RT_ARCH_AMD64)
408#error fix macros (and more in this file) for big-endian machines
409#endif
410
411#define CSR_IADR(S) (*(uint32_t*)((S)->aCSR + 1)) /**< Initialization Block Address */
412#define CSR_CRBA(S) (*(uint32_t*)((S)->aCSR + 18)) /**< Current Receive Buffer Address */
413#define CSR_CXBA(S) (*(uint32_t*)((S)->aCSR + 20)) /**< Current Transmit Buffer Address */
414#define CSR_NRBA(S) (*(uint32_t*)((S)->aCSR + 22)) /**< Next Receive Buffer Address */
415#define CSR_BADR(S) (*(uint32_t*)((S)->aCSR + 24)) /**< Base Address of Receive Ring */
416#define CSR_NRDA(S) (*(uint32_t*)((S)->aCSR + 26)) /**< Next Receive Descriptor Address */
417#define CSR_CRDA(S) (*(uint32_t*)((S)->aCSR + 28)) /**< Current Receive Descriptor Address */
418#define CSR_BADX(S) (*(uint32_t*)((S)->aCSR + 30)) /**< Base Address of Transmit Descriptor */
419#define CSR_NXDA(S) (*(uint32_t*)((S)->aCSR + 32)) /**< Next Transmit Descriptor Address */
420#define CSR_CXDA(S) (*(uint32_t*)((S)->aCSR + 34)) /**< Current Transmit Descriptor Address */
421#define CSR_NNRD(S) (*(uint32_t*)((S)->aCSR + 36)) /**< Next Next Receive Descriptor Address */
422#define CSR_NNXD(S) (*(uint32_t*)((S)->aCSR + 38)) /**< Next Next Transmit Descriptor Address */
423#define CSR_CRBC(S) ((S)->aCSR[40]) /**< Current Receive Byte Count */
424#define CSR_CRST(S) ((S)->aCSR[41]) /**< Current Receive Status */
425#define CSR_CXBC(S) ((S)->aCSR[42]) /**< Current Transmit Byte Count */
426#define CSR_CXST(S) ((S)->aCSR[43]) /**< Current transmit status */
427#define CSR_NRBC(S) ((S)->aCSR[44]) /**< Next Receive Byte Count */
428#define CSR_NRST(S) ((S)->aCSR[45]) /**< Next Receive Status */
429#define CSR_POLL(S) ((S)->aCSR[46]) /**< Transmit Poll Time Counter */
430#define CSR_PINT(S) ((S)->aCSR[47]) /**< Transmit Polling Interval */
431#define CSR_PXDA(S) (*(uint32_t*)((S)->aCSR + 60)) /**< Previous Transmit Descriptor Address*/
432#define CSR_PXBC(S) ((S)->aCSR[62]) /**< Previous Transmit Byte Count */
433#define CSR_PXST(S) ((S)->aCSR[63]) /**< Previous Transmit Status */
434#define CSR_NXBA(S) (*(uint32_t*)((S)->aCSR + 64)) /**< Next Transmit Buffer Address */
435#define CSR_NXBC(S) ((S)->aCSR[66]) /**< Next Transmit Byte Count */
436#define CSR_NXST(S) ((S)->aCSR[67]) /**< Next Transmit Status */
437#define CSR_RCVRC(S) ((S)->aCSR[72]) /**< Receive Descriptor Ring Counter */
438#define CSR_XMTRC(S) ((S)->aCSR[74]) /**< Transmit Descriptor Ring Counter */
439#define CSR_RCVRL(S) ((S)->aCSR[76]) /**< Receive Descriptor Ring Length */
440#define CSR_XMTRL(S) ((S)->aCSR[78]) /**< Transmit Descriptor Ring Length */
441#define CSR_MISSC(S) ((S)->aCSR[112]) /**< Missed Frame Count */
442
443#define PHYSADDR(S,A) ((A) | (S)->GCUpperPhys)
444
445/* Version for the PCnet/FAST III 79C973 card */
446#define CSR_VERSION_LOW_79C973 0x5003 /* the lower two bits must be 11b for AMD */
447#define CSR_VERSION_LOW_79C970A 0x1003 /* the lower two bits must be 11b for AMD */
448#define CSR_VERSION_HIGH 0x0262
449
450/** @todo All structs: big endian? */
451
452struct INITBLK16
453{
454 uint16_t mode; /**< copied into csr15 */
455 uint16_t padr1; /**< MAC 0..15 */
456 uint16_t padr2; /**< MAC 16..32 */
457 uint16_t padr3; /**< MAC 33..47 */
458 uint16_t ladrf1; /**< logical address filter 0..15 */
459 uint16_t ladrf2; /**< logical address filter 16..31 */
460 uint16_t ladrf3; /**< logical address filter 32..47 */
461 uint16_t ladrf4; /**< logical address filter 48..63 */
462 uint32_t rdra:24; /**< address of receive descriptor ring */
463 uint32_t res1:5; /**< reserved */
464 uint32_t rlen:3; /**< number of receive descriptor ring entries */
465 uint32_t tdra:24; /**< address of transmit descriptor ring */
466 uint32_t res2:5; /**< reserved */
467 uint32_t tlen:3; /**< number of transmit descriptor ring entries */
468};
469AssertCompileSize(INITBLK16, 24);
470
471/** bird: I've changed the type for the bitfields. They should only be 16-bit all together.
472 * frank: I've changed the bitfiled types to uint32_t to prevent compiler warnings. */
473struct INITBLK32
474{
475 uint16_t mode; /**< copied into csr15 */
476 uint16_t res1:4; /**< reserved */
477 uint16_t rlen:4; /**< number of receive descriptor ring entries */
478 uint16_t res2:4; /**< reserved */
479 uint16_t tlen:4; /**< number of transmit descriptor ring entries */
480 uint16_t padr1; /**< MAC 0..15 */
481 uint16_t padr2; /**< MAC 16..31 */
482 uint16_t padr3; /**< MAC 32..47 */
483 uint16_t res3; /**< reserved */
484 uint16_t ladrf1; /**< logical address filter 0..15 */
485 uint16_t ladrf2; /**< logical address filter 16..31 */
486 uint16_t ladrf3; /**< logical address filter 32..47 */
487 uint16_t ladrf4; /**< logical address filter 48..63 */
488 uint32_t rdra; /**< address of receive descriptor ring */
489 uint32_t tdra; /**< address of transmit descriptor ring */
490};
491AssertCompileSize(INITBLK32, 28);
492
493/** Transmit Message Descriptor */
494typedef struct TMD
495{
496 struct
497 {
498 uint32_t tbadr; /**< transmit buffer address */
499 } tmd0;
500 struct
501 {
502 uint32_t bcnt:12; /**< buffer byte count (two's complement) */
503 uint32_t ones:4; /**< must be 1111b */
504 uint32_t res:7; /**< reserved */
505 uint32_t bpe:1; /**< bus parity error */
506 uint32_t enp:1; /**< end of packet */
507 uint32_t stp:1; /**< start of packet */
508 uint32_t def:1; /**< deferred */
509 uint32_t one:1; /**< exactly one retry was needed to transmit a frame */
510 uint32_t ltint:1; /**< suppress interrupts after successful transmission */
511 uint32_t nofcs:1; /**< when set, the state of DXMTFCS is ignored and
512 transmitter FCS generation is activated. */
513 uint32_t err:1; /**< error occurred */
514 uint32_t own:1; /**< 0=owned by guest driver, 1=owned by controller */
515 } tmd1;
516 struct
517 {
518 uint32_t trc:4; /**< transmit retry count */
519 uint32_t res:12; /**< reserved */
520 uint32_t tdr:10; /**< ??? */
521 uint32_t rtry:1; /**< retry error */
522 uint32_t lcar:1; /**< loss of carrier */
523 uint32_t lcol:1; /**< late collision */
524 uint32_t exdef:1; /**< excessive deferral */
525 uint32_t uflo:1; /**< underflow error */
526 uint32_t buff:1; /**< out of buffers (ENP not found) */
527 } tmd2;
528 struct
529 {
530 uint32_t res; /**< reserved for user defined space */
531 } tmd3;
532} TMD;
533AssertCompileSize(TMD, 16);
534
535/** Receive Message Descriptor */
536typedef struct RMD
537{
538 struct
539 {
540 uint32_t rbadr; /**< receive buffer address */
541 } rmd0;
542 struct
543 {
544 uint32_t bcnt:12; /**< buffer byte count (two's complement) */
545 uint32_t ones:4; /**< must be 1111b */
546 uint32_t res:4; /**< reserved */
547 uint32_t bam:1; /**< broadcast address match */
548 uint32_t lafm:1; /**< logical filter address match */
549 uint32_t pam:1; /**< physical address match */
550 uint32_t bpe:1; /**< bus parity error */
551 uint32_t enp:1; /**< end of packet */
552 uint32_t stp:1; /**< start of packet */
553 uint32_t buff:1; /**< buffer error */
554 uint32_t crc:1; /**< crc error on incoming frame */
555 uint32_t oflo:1; /**< overflow error (lost all or part of incoming frame) */
556 uint32_t fram:1; /**< frame error */
557 uint32_t err:1; /**< error occurred */
558 uint32_t own:1; /**< 0=owned by guest driver, 1=owned by controller */
559 } rmd1;
560 struct
561 {
562 uint32_t mcnt:12; /**< message byte count */
563 uint32_t zeros:4; /**< 0000b */
564 uint32_t rpc:8; /**< receive frame tag */
565 uint32_t rcc:8; /**< receive frame tag + reserved */
566 } rmd2;
567 struct
568 {
569 uint32_t res; /**< reserved for user defined space */
570 } rmd3;
571} RMD;
572AssertCompileSize(RMD, 16);
573
574
575#ifndef VBOX_DEVICE_STRUCT_TESTCASE
576/*******************************************************************************
577* Internal Functions *
578*******************************************************************************/
579#define PRINT_TMD(T) Log2(( \
580 "TMD0 : TBADR=%#010x\n" \
581 "TMD1 : OWN=%d, ERR=%d, FCS=%d, LTI=%d, " \
582 "ONE=%d, DEF=%d, STP=%d, ENP=%d,\n" \
583 " BPE=%d, BCNT=%d\n" \
584 "TMD2 : BUF=%d, UFL=%d, EXD=%d, LCO=%d, " \
585 "LCA=%d, RTR=%d,\n" \
586 " TDR=%d, TRC=%d\n", \
587 (T)->tmd0.tbadr, \
588 (T)->tmd1.own, (T)->tmd1.err, (T)->tmd1.nofcs, \
589 (T)->tmd1.ltint, (T)->tmd1.one, (T)->tmd1.def, \
590 (T)->tmd1.stp, (T)->tmd1.enp, (T)->tmd1.bpe, \
591 4096-(T)->tmd1.bcnt, \
592 (T)->tmd2.buff, (T)->tmd2.uflo, (T)->tmd2.exdef,\
593 (T)->tmd2.lcol, (T)->tmd2.lcar, (T)->tmd2.rtry, \
594 (T)->tmd2.tdr, (T)->tmd2.trc))
595
596#define PRINT_RMD(R) Log2(( \
597 "RMD0 : RBADR=%#010x\n" \
598 "RMD1 : OWN=%d, ERR=%d, FRAM=%d, OFLO=%d, " \
599 "CRC=%d, BUFF=%d, STP=%d, ENP=%d,\n " \
600 "BPE=%d, PAM=%d, LAFM=%d, BAM=%d, ONES=%d, BCNT=%d\n" \
601 "RMD2 : RCC=%d, RPC=%d, MCNT=%d, ZEROS=%d\n", \
602 (R)->rmd0.rbadr, \
603 (R)->rmd1.own, (R)->rmd1.err, (R)->rmd1.fram, \
604 (R)->rmd1.oflo, (R)->rmd1.crc, (R)->rmd1.buff, \
605 (R)->rmd1.stp, (R)->rmd1.enp, (R)->rmd1.bpe, \
606 (R)->rmd1.pam, (R)->rmd1.lafm, (R)->rmd1.bam, \
607 (R)->rmd1.ones, 4096-(R)->rmd1.bcnt, \
608 (R)->rmd2.rcc, (R)->rmd2.rpc, (R)->rmd2.mcnt, \
609 (R)->rmd2.zeros))
610
611static void pcnetPollTimerStart(PCNetState *pThis);
612static int pcnetXmitPending(PCNetState *pThis, bool fOnWorkerThread);
613
614
615
616/**
617 * Checks if the link is up.
618 * @returns true if the link is up.
619 * @returns false if the link is down.
620 */
621DECLINLINE(bool) pcnetIsLinkUp(PCNetState *pThis)
622{
623 return pThis->pDrvR3 && !pThis->fLinkTempDown && pThis->fLinkUp;
624}
625
626/**
627 * Load transmit message descriptor
628 * Make sure we read the own flag first.
629 *
630 * @param pThis adapter private data
631 * @param addr physical address of the descriptor
632 * @param fRetIfNotOwn return immediately after reading the own flag if we don't own the descriptor
633 * @return true if we own the descriptor, false otherwise
634 */
635DECLINLINE(bool) pcnetTmdLoad(PCNetState *pThis, TMD *tmd, RTGCPHYS32 addr, bool fRetIfNotOwn)
636{
637 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
638 uint8_t ownbyte;
639
640 if (pThis->fPrivIfEnabled)
641 {
642 /* RX/TX descriptors shared between host and guest => direct copy */
643 uint8_t *pv = (uint8_t*)pThis->CTX_SUFF(pSharedMMIO)
644 + (addr - pThis->GCTDRA)
645 + pThis->CTX_SUFF(pSharedMMIO)->V.V1.offTxDescriptors;
646 if (!(pv[7] & 0x80) && fRetIfNotOwn)
647 return false;
648 memcpy(tmd, pv, 16);
649 return true;
650 }
651 else if (RT_UNLIKELY(BCR_SWSTYLE(pThis) == 0))
652 {
653 uint16_t xda[4];
654
655 PDMDevHlpPhysRead(pDevIns, addr+3, &ownbyte, 1);
656 if (!(ownbyte & 0x80) && fRetIfNotOwn)
657 return false;
658 PDMDevHlpPhysRead(pDevIns, addr, (void*)&xda[0], sizeof(xda));
659 ((uint32_t *)tmd)[0] = (uint32_t)xda[0] | ((uint32_t)(xda[1] & 0x00ff) << 16);
660 ((uint32_t *)tmd)[1] = (uint32_t)xda[2] | ((uint32_t)(xda[1] & 0xff00) << 16);
661 ((uint32_t *)tmd)[2] = (uint32_t)xda[3] << 16;
662 ((uint32_t *)tmd)[3] = 0;
663 }
664 else if (RT_LIKELY(BCR_SWSTYLE(pThis) != 3))
665 {
666 PDMDevHlpPhysRead(pDevIns, addr+7, &ownbyte, 1);
667 if (!(ownbyte & 0x80) && fRetIfNotOwn)
668 return false;
669 PDMDevHlpPhysRead(pDevIns, addr, (void*)tmd, 16);
670 }
671 else
672 {
673 uint32_t xda[4];
674 PDMDevHlpPhysRead(pDevIns, addr+7, &ownbyte, 1);
675 if (!(ownbyte & 0x80) && fRetIfNotOwn)
676 return false;
677 PDMDevHlpPhysRead(pDevIns, addr, (void*)&xda[0], sizeof(xda));
678 ((uint32_t *)tmd)[0] = xda[2];
679 ((uint32_t *)tmd)[1] = xda[1];
680 ((uint32_t *)tmd)[2] = xda[0];
681 ((uint32_t *)tmd)[3] = xda[3];
682 }
683 /* Double check the own bit; guest drivers might be buggy and lock prefixes in the recompiler are ignored by other threads. */
684#ifdef DEBUG
685 if (tmd->tmd1.own == 1 && !(ownbyte & 0x80))
686 Log(("pcnetTmdLoad: own bit flipped while reading!!\n"));
687#endif
688 if (!(ownbyte & 0x80))
689 tmd->tmd1.own = 0;
690
691 return !!tmd->tmd1.own;
692}
693
694/**
695 * Store transmit message descriptor and hand it over to the host (the VM guest).
696 * Make sure that all data are transmitted before we clear the own flag.
697 */
698DECLINLINE(void) pcnetTmdStorePassHost(PCNetState *pThis, TMD *tmd, RTGCPHYS32 addr)
699{
700 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTmdStore), a);
701 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
702 if (pThis->fPrivIfEnabled)
703 {
704 /* RX/TX descriptors shared between host and guest => direct copy */
705 uint8_t *pv = (uint8_t*)pThis->CTX_SUFF(pSharedMMIO)
706 + (addr - pThis->GCTDRA)
707 + pThis->CTX_SUFF(pSharedMMIO)->V.V1.offTxDescriptors;
708 memcpy(pv, tmd, 16);
709 pv[7] &= ~0x80;
710 }
711 else if (RT_UNLIKELY(BCR_SWSTYLE(pThis) == 0))
712 {
713 uint16_t xda[4];
714 xda[0] = ((uint32_t *)tmd)[0] & 0xffff;
715 xda[1] = ((((uint32_t *)tmd)[0] >> 16) & 0xff) | ((((uint32_t *)tmd)[1]>>16) & 0xff00);
716 xda[2] = ((uint32_t *)tmd)[1] & 0xffff;
717 xda[3] = ((uint32_t *)tmd)[2] >> 16;
718 xda[1] |= 0x8000;
719 PDMDevHlpPhysWrite(pDevIns, addr, (void*)&xda[0], sizeof(xda));
720 xda[1] &= ~0x8000;
721 PDMDevHlpPhysWrite(pDevIns, addr+3, (uint8_t*)xda + 3, 1);
722 }
723 else if (RT_LIKELY(BCR_SWSTYLE(pThis) != 3))
724 {
725 ((uint32_t*)tmd)[1] |= 0x80000000;
726 PDMDevHlpPhysWrite(pDevIns, addr, (void*)tmd, 16);
727 ((uint32_t*)tmd)[1] &= ~0x80000000;
728 PDMDevHlpPhysWrite(pDevIns, addr+7, (uint8_t*)tmd + 7, 1);
729 }
730 else
731 {
732 uint32_t xda[4];
733 xda[0] = ((uint32_t *)tmd)[2];
734 xda[1] = ((uint32_t *)tmd)[1];
735 xda[2] = ((uint32_t *)tmd)[0];
736 xda[3] = ((uint32_t *)tmd)[3];
737 xda[1] |= 0x80000000;
738 PDMDevHlpPhysWrite(pDevIns, addr, (void*)&xda[0], sizeof(xda));
739 xda[1] &= ~0x80000000;
740 PDMDevHlpPhysWrite(pDevIns, addr+7, (uint8_t*)xda + 7, 1);
741 }
742 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTmdStore), a);
743}
744
745/**
746 * Load receive message descriptor
747 * Make sure we read the own flag first.
748 *
749 * @param pThis adapter private data
750 * @param addr physical address of the descriptor
751 * @param fRetIfNotOwn return immediately after reading the own flag if we don't own the descriptor
752 * @return true if we own the descriptor, false otherwise
753 */
754DECLINLINE(int) pcnetRmdLoad(PCNetState *pThis, RMD *rmd, RTGCPHYS32 addr, bool fRetIfNotOwn)
755{
756 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
757 uint8_t ownbyte;
758
759 if (pThis->fPrivIfEnabled)
760 {
761 /* RX/TX descriptors shared between host and guest => direct copy */
762 uint8_t *pb = (uint8_t*)pThis->CTX_SUFF(pSharedMMIO)
763 + (addr - pThis->GCRDRA)
764 + pThis->CTX_SUFF(pSharedMMIO)->V.V1.offRxDescriptors;
765 if (!(pb[7] & 0x80) && fRetIfNotOwn)
766 return false;
767 memcpy(rmd, pb, 16);
768 return true;
769 }
770 else if (RT_UNLIKELY(BCR_SWSTYLE(pThis) == 0))
771 {
772 uint16_t rda[4];
773 PDMDevHlpPhysRead(pDevIns, addr+3, &ownbyte, 1);
774 if (!(ownbyte & 0x80) && fRetIfNotOwn)
775 return false;
776 PDMDevHlpPhysRead(pDevIns, addr, (void*)&rda[0], sizeof(rda));
777 ((uint32_t *)rmd)[0] = (uint32_t)rda[0] | ((rda[1] & 0x00ff) << 16);
778 ((uint32_t *)rmd)[1] = (uint32_t)rda[2] | ((rda[1] & 0xff00) << 16);
779 ((uint32_t *)rmd)[2] = (uint32_t)rda[3];
780 ((uint32_t *)rmd)[3] = 0;
781 }
782 else if (RT_LIKELY(BCR_SWSTYLE(pThis) != 3))
783 {
784 PDMDevHlpPhysRead(pDevIns, addr+7, &ownbyte, 1);
785 if (!(ownbyte & 0x80) && fRetIfNotOwn)
786 return false;
787 PDMDevHlpPhysRead(pDevIns, addr, (void*)rmd, 16);
788 }
789 else
790 {
791 uint32_t rda[4];
792 PDMDevHlpPhysRead(pDevIns, addr+7, &ownbyte, 1);
793 if (!(ownbyte & 0x80) && fRetIfNotOwn)
794 return false;
795 PDMDevHlpPhysRead(pDevIns, addr, (void*)&rda[0], sizeof(rda));
796 ((uint32_t *)rmd)[0] = rda[2];
797 ((uint32_t *)rmd)[1] = rda[1];
798 ((uint32_t *)rmd)[2] = rda[0];
799 ((uint32_t *)rmd)[3] = rda[3];
800 }
801 /* Double check the own bit; guest drivers might be buggy and lock prefixes in the recompiler are ignored by other threads. */
802#ifdef DEBUG
803 if (rmd->rmd1.own == 1 && !(ownbyte & 0x80))
804 Log(("pcnetRmdLoad: own bit flipped while reading!!\n"));
805#endif
806 if (!(ownbyte & 0x80))
807 rmd->rmd1.own = 0;
808
809 return !!rmd->rmd1.own;
810}
811
812
813/**
814 * Store receive message descriptor and hand it over to the host (the VM guest).
815 * Make sure that all data are transmitted before we clear the own flag.
816 */
817DECLINLINE(void) pcnetRmdStorePassHost(PCNetState *pThis, RMD *rmd, RTGCPHYS32 addr)
818{
819 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
820 if (pThis->fPrivIfEnabled)
821 {
822 /* RX/TX descriptors shared between host and guest => direct copy */
823 uint8_t *pv = (uint8_t*)pThis->CTX_SUFF(pSharedMMIO)
824 + (addr - pThis->GCRDRA)
825 + pThis->CTX_SUFF(pSharedMMIO)->V.V1.offRxDescriptors;
826 memcpy(pv, rmd, 16);
827 pv[7] &= ~0x80;
828 }
829 else if (RT_UNLIKELY(BCR_SWSTYLE(pThis) == 0))
830 {
831 uint16_t rda[4];
832 rda[0] = ((uint32_t *)rmd)[0] & 0xffff;
833 rda[1] = ((((uint32_t *)rmd)[0]>>16) & 0xff) | ((((uint32_t *)rmd)[1]>>16) & 0xff00);
834 rda[2] = ((uint32_t *)rmd)[1] & 0xffff;
835 rda[3] = ((uint32_t *)rmd)[2] & 0xffff;
836 rda[1] |= 0x8000;
837 PDMDevHlpPhysWrite(pDevIns, addr, (void*)&rda[0], sizeof(rda));
838 rda[1] &= ~0x8000;
839 PDMDevHlpPhysWrite(pDevIns, addr+3, (uint8_t*)rda + 3, 1);
840 }
841 else if (RT_LIKELY(BCR_SWSTYLE(pThis) != 3))
842 {
843 ((uint32_t*)rmd)[1] |= 0x80000000;
844 PDMDevHlpPhysWrite(pDevIns, addr, (void*)rmd, 16);
845 ((uint32_t*)rmd)[1] &= ~0x80000000;
846 PDMDevHlpPhysWrite(pDevIns, addr+7, (uint8_t*)rmd + 7, 1);
847 }
848 else
849 {
850 uint32_t rda[4];
851 rda[0] = ((uint32_t *)rmd)[2];
852 rda[1] = ((uint32_t *)rmd)[1];
853 rda[2] = ((uint32_t *)rmd)[0];
854 rda[3] = ((uint32_t *)rmd)[3];
855 rda[1] |= 0x80000000;
856 PDMDevHlpPhysWrite(pDevIns, addr, (void*)&rda[0], sizeof(rda));
857 rda[1] &= ~0x80000000;
858 PDMDevHlpPhysWrite(pDevIns, addr+7, (uint8_t*)rda + 7, 1);
859 }
860}
861
862#ifdef IN_RING3
863/**
864 * Read+Write a TX/RX descriptor to prevent PDMDevHlpPhysWrite() allocating
865 * pages later when we shouldn't schedule to EMT. Temporarily hack.
866 */
867static void pcnetDescTouch(PCNetState *pThis, RTGCPHYS32 addr)
868{
869 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
870
871 if (!pThis->fPrivIfEnabled)
872 {
873 uint8_t aBuf[16];
874 size_t cbDesc;
875 if (RT_UNLIKELY(BCR_SWSTYLE(pThis) == 0))
876 cbDesc = 8;
877 else
878 cbDesc = 16;
879 PDMDevHlpPhysRead(pDevIns, addr, aBuf, cbDesc);
880 PDMDevHlpPhysWrite(pDevIns, addr, aBuf, cbDesc);
881 }
882}
883#endif /* IN_RING3 */
884
885/** Checks if it's a bad (as in invalid) RMD.*/
886#define IS_RMD_BAD(rmd) ((rmd).rmd1.ones != 15 || (rmd).rmd2.zeros != 0)
887
888/** The network card is the owner of the RDTE/TDTE, actually it is this driver */
889#define CARD_IS_OWNER(desc) (((desc) & 0x8000))
890
891/** The host is the owner of the RDTE/TDTE -- actually the VM guest. */
892#define HOST_IS_OWNER(desc) (!((desc) & 0x8000))
893
894#ifndef ETHER_IS_MULTICAST /* Net/Open BSD macro it seems */
895#define ETHER_IS_MULTICAST(a) ((*(uint8_t *)(a)) & 1)
896#endif
897
898#define ETHER_ADDR_LEN ETH_ALEN
899#define ETH_ALEN 6
900#pragma pack(1)
901struct ether_header /** @todo Use RTNETETHERHDR */
902{
903 uint8_t ether_dhost[ETH_ALEN]; /**< destination ethernet address */
904 uint8_t ether_shost[ETH_ALEN]; /**< source ethernet address */
905 uint16_t ether_type; /**< packet type ID field */
906};
907#pragma pack()
908
909#define PRINT_PKTHDR(BUF) do { \
910 struct ether_header *hdr = (struct ether_header *)(BUF); \
911 Log(("#%d packet dhost=%02x:%02x:%02x:%02x:%02x:%02x, " \
912 "shost=%02x:%02x:%02x:%02x:%02x:%02x, " \
913 "type=%#06x (bcast=%d)\n", PCNET_INST_NR, \
914 hdr->ether_dhost[0],hdr->ether_dhost[1],hdr->ether_dhost[2], \
915 hdr->ether_dhost[3],hdr->ether_dhost[4],hdr->ether_dhost[5], \
916 hdr->ether_shost[0],hdr->ether_shost[1],hdr->ether_shost[2], \
917 hdr->ether_shost[3],hdr->ether_shost[4],hdr->ether_shost[5], \
918 htons(hdr->ether_type), \
919 !!ETHER_IS_MULTICAST(hdr->ether_dhost))); \
920} while (0)
921
922
923#ifdef IN_RING3
924/**
925 * Initialize the shared memory for the private guest interface.
926 *
927 * @note Changing this layout will break SSM for guests using the private guest interface!
928 */
929static void pcnetInitSharedMemory(PCNetState *pThis)
930{
931 /* Clear the entire block for pcnetReset usage. */
932 memset(pThis->pSharedMMIOR3, 0, PCNET_GUEST_SHARED_MEMORY_SIZE);
933
934 pThis->pSharedMMIOR3->u32Version = PCNET_GUEST_INTERFACE_VERSION;
935 uint32_t off = 2048; /* Leave some space for more fields within the header */
936
937 /*
938 * The Descriptor arrays.
939 */
940 pThis->pSharedMMIOR3->V.V1.offTxDescriptors = off;
941 off = RT_ALIGN(off + PCNET_GUEST_TX_DESCRIPTOR_SIZE * PCNET_GUEST_MAX_TX_DESCRIPTORS, 32);
942
943 pThis->pSharedMMIOR3->V.V1.offRxDescriptors = off;
944 off = RT_ALIGN(off + PCNET_GUEST_RX_DESCRIPTOR_SIZE * PCNET_GUEST_MAX_RX_DESCRIPTORS, 32);
945
946 /* Make sure all the descriptors are mapped into HMA space (and later ring-0). The 8192
947 bytes limit is hardcoded in the PDMDevHlpMMHyperMapMMIO2 call down in pcnetConstruct. */
948 AssertRelease(off <= 8192);
949
950 /*
951 * The buffer arrays.
952 */
953#if 0
954 /* Don't allocate TX buffers since Windows guests cannot use it */
955 pThis->pSharedMMIOR3->V.V1.offTxBuffers = off;
956 off = RT_ALIGN(off + PCNET_GUEST_NIC_BUFFER_SIZE * PCNET_GUEST_MAX_TX_DESCRIPTORS, 32);
957#endif
958
959 pThis->pSharedMMIOR3->V.V1.offRxBuffers = off;
960 pThis->pSharedMMIOR3->fFlags = PCNET_GUEST_FLAGS_ADMIT_HOST;
961 off = RT_ALIGN(off + PCNET_GUEST_NIC_BUFFER_SIZE * PCNET_GUEST_MAX_RX_DESCRIPTORS, 32);
962 AssertRelease(off <= PCNET_GUEST_SHARED_MEMORY_SIZE);
963
964 /* Update the header with the final size. */
965 pThis->pSharedMMIOR3->cbUsed = off;
966}
967#endif /* IN_RING3 */
968
969#define MULTICAST_FILTER_LEN 8
970
971DECLINLINE(uint32_t) lnc_mchash(const uint8_t *ether_addr)
972{
973#define LNC_POLYNOMIAL 0xEDB88320UL
974 uint32_t crc = 0xFFFFFFFF;
975 int idx, bit;
976 uint8_t data;
977
978 for (idx = 0; idx < ETHER_ADDR_LEN; idx++)
979 {
980 for (data = *ether_addr++, bit = 0; bit < MULTICAST_FILTER_LEN; bit++)
981 {
982 crc = (crc >> 1) ^ (((crc ^ data) & 1) ? LNC_POLYNOMIAL : 0);
983 data >>= 1;
984 }
985 }
986 return crc;
987#undef LNC_POLYNOMIAL
988}
989
990#define CRC(crc, ch) (crc = (crc >> 8) ^ crctab[(crc ^ (ch)) & 0xff])
991
992/* generated using the AUTODIN II polynomial
993 * x^32 + x^26 + x^23 + x^22 + x^16 +
994 * x^12 + x^11 + x^10 + x^8 + x^7 + x^5 + x^4 + x^2 + x^1 + 1
995 */
996static const uint32_t crctab[256] =
997{
998 0x00000000, 0x77073096, 0xee0e612c, 0x990951ba,
999 0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
1000 0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
1001 0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
1002 0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de,
1003 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
1004 0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec,
1005 0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
1006 0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
1007 0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
1008 0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940,
1009 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
1010 0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116,
1011 0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
1012 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
1013 0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
1014 0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a,
1015 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
1016 0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818,
1017 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
1018 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
1019 0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
1020 0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c,
1021 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
1022 0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2,
1023 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
1024 0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
1025 0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
1026 0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086,
1027 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
1028 0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4,
1029 0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
1030 0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
1031 0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
1032 0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8,
1033 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
1034 0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe,
1035 0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
1036 0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
1037 0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
1038 0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252,
1039 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
1040 0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60,
1041 0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
1042 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
1043 0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
1044 0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04,
1045 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
1046 0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a,
1047 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
1048 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
1049 0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
1050 0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e,
1051 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
1052 0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c,
1053 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
1054 0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
1055 0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
1056 0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0,
1057 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
1058 0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6,
1059 0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
1060 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
1061 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d,
1062};
1063
1064DECLINLINE(int) padr_match(PCNetState *pThis, const uint8_t *buf, size_t size)
1065{
1066 struct ether_header *hdr = (struct ether_header *)buf;
1067 int result;
1068#if (defined(RT_ARCH_X86) || defined(RT_ARCH_AMD64)) && !defined(PCNET_DEBUG_MATCH)
1069 result = !CSR_DRCVPA(pThis) && !memcmp(hdr->ether_dhost, pThis->aCSR + 12, 6);
1070#else
1071 uint8_t padr[6];
1072 padr[0] = pThis->aCSR[12] & 0xff;
1073 padr[1] = pThis->aCSR[12] >> 8;
1074 padr[2] = pThis->aCSR[13] & 0xff;
1075 padr[3] = pThis->aCSR[13] >> 8;
1076 padr[4] = pThis->aCSR[14] & 0xff;
1077 padr[5] = pThis->aCSR[14] >> 8;
1078 result = !CSR_DRCVPA(pThis) && !memcmp(hdr->ether_dhost, padr, 6);
1079#endif
1080
1081#ifdef PCNET_DEBUG_MATCH
1082 Log(("#%d packet dhost=%02x:%02x:%02x:%02x:%02x:%02x, "
1083 "padr=%02x:%02x:%02x:%02x:%02x:%02x => %d\n", PCNET_INST_NR,
1084 hdr->ether_dhost[0],hdr->ether_dhost[1],hdr->ether_dhost[2],
1085 hdr->ether_dhost[3],hdr->ether_dhost[4],hdr->ether_dhost[5],
1086 padr[0],padr[1],padr[2],padr[3],padr[4],padr[5], result));
1087#endif
1088 return result;
1089}
1090
1091DECLINLINE(int) padr_bcast(PCNetState *pThis, const uint8_t *buf, size_t size)
1092{
1093 static uint8_t aBCAST[6] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
1094 struct ether_header *hdr = (struct ether_header *)buf;
1095 int result = !CSR_DRCVBC(pThis) && !memcmp(hdr->ether_dhost, aBCAST, 6);
1096#ifdef PCNET_DEBUG_MATCH
1097 Log(("#%d padr_bcast result=%d\n", PCNET_INST_NR, result));
1098#endif
1099 return result;
1100}
1101
1102static int ladr_match(PCNetState *pThis, const uint8_t *buf, size_t size)
1103{
1104 struct ether_header *hdr = (struct ether_header *)buf;
1105 if (RT_UNLIKELY(hdr->ether_dhost[0] & 0x01) && ((uint64_t *)&pThis->aCSR[8])[0] != 0LL)
1106 {
1107 int index;
1108#if defined(RT_ARCH_X86) || defined(RT_ARCH_AMD64)
1109 index = lnc_mchash(hdr->ether_dhost) >> 26;
1110 return ((uint8_t*)(pThis->aCSR + 8))[index >> 3] & (1 << (index & 7));
1111#else
1112 uint8_t ladr[8];
1113 ladr[0] = pThis->aCSR[8] & 0xff;
1114 ladr[1] = pThis->aCSR[8] >> 8;
1115 ladr[2] = pThis->aCSR[9] & 0xff;
1116 ladr[3] = pThis->aCSR[9] >> 8;
1117 ladr[4] = pThis->aCSR[10] & 0xff;
1118 ladr[5] = pThis->aCSR[10] >> 8;
1119 ladr[6] = pThis->aCSR[11] & 0xff;
1120 ladr[7] = pThis->aCSR[11] >> 8;
1121 index = lnc_mchash(hdr->ether_dhost) >> 26;
1122 return (ladr[index >> 3] & (1 << (index & 7)));
1123#endif
1124 }
1125 return 0;
1126}
1127
1128
1129/**
1130 * Get the receive descriptor ring address with a given index.
1131 */
1132DECLINLINE(RTGCPHYS32) pcnetRdraAddr(PCNetState *pThis, int idx)
1133{
1134 return pThis->GCRDRA + ((CSR_RCVRL(pThis) - idx) << pThis->iLog2DescSize);
1135}
1136
1137/**
1138 * Get the transmit descriptor ring address with a given index.
1139 */
1140DECLINLINE(RTGCPHYS32) pcnetTdraAddr(PCNetState *pThis, int idx)
1141{
1142 return pThis->GCTDRA + ((CSR_XMTRL(pThis) - idx) << pThis->iLog2DescSize);
1143}
1144
1145RT_C_DECLS_BEGIN
1146#ifndef IN_RING3
1147DECLEXPORT(int) pcnetHandleRingWrite(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame,
1148 RTGCPTR pvFault, RTGCPHYS GCPhysFault, void *pvUser);
1149#endif
1150RT_C_DECLS_END
1151
1152#undef htonl
1153#define htonl(x) ASMByteSwapU32(x)
1154#undef htons
1155#define htons(x) ( (((x) & 0xff00) >> 8) | (((x) & 0x00ff) << 8) )
1156
1157static void pcnetPollRxTx(PCNetState *pThis);
1158static void pcnetPollTimer(PCNetState *pThis);
1159static void pcnetUpdateIrq(PCNetState *pThis);
1160static uint32_t pcnetBCRReadU16(PCNetState *pThis, uint32_t u32RAP);
1161static int pcnetBCRWriteU16(PCNetState *pThis, uint32_t u32RAP, uint32_t val);
1162
1163
1164#ifdef PCNET_NO_POLLING
1165# ifndef IN_RING3
1166
1167/**
1168 * #PF Virtual Handler callback for Guest write access to the ring descriptor page(pThis)
1169 *
1170 * @return VBox status code (appropriate for trap handling and GC return).
1171 * @param pVM VM Handle.
1172 * @param uErrorCode CPU Error code.
1173 * @param pRegFrame Trap register frame.
1174 * @param pvFault The fault address (cr2).
1175 * @param GCPhysFault The GC physical address corresponding to pvFault.
1176 * @param pvUser User argument.
1177 */
1178DECLEXPORT(int) pcnetHandleRingWrite(PVM pVM, RTGCUINT uErrorCode, PCPUMCTXCORE pRegFrame,
1179 RTGCPTR pvFault, RTGCPHYS GCPhysFault, void *pvUser)
1180{
1181 PCNetState *pThis = (PCNetState *)pvUser;
1182
1183 Log(("#%d pcnetHandleRingWriteGC: write to %#010x\n", PCNET_INST_NR, GCPhysFault));
1184
1185 uint32_t cb;
1186 int rc = CTXALLSUFF(pThis->pfnEMInterpretInstruction)(pVM, pRegFrame, pvFault, &cb);
1187 if (RT_SUCCESS(rc) && cb)
1188 {
1189 if ( (GCPhysFault >= pThis->GCTDRA && GCPhysFault + cb < pcnetTdraAddr(pThis, 0))
1190#ifdef PCNET_MONITOR_RECEIVE_RING
1191 || (GCPhysFault >= pThis->GCRDRA && GCPhysFault + cb < pcnetRdraAddr(pThis, 0))
1192#endif
1193 )
1194 {
1195 uint32_t offsetTDRA = (GCPhysFault - pThis->GCTDRA);
1196
1197 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
1198 if (RT_SUCCESS(rc))
1199 {
1200 STAM_COUNTER_INC(&CTXALLSUFF(pThis->StatRingWrite)); ;
1201
1202 /* Check if we can do something now */
1203 pcnetPollRxTx(pThis);
1204 pcnetUpdateIrq(pThis);
1205
1206 PDMCritSectLeave(&pThis->CritSect);
1207 return VINF_SUCCESS;
1208 }
1209 }
1210 else
1211 {
1212 STAM_COUNTER_INC(&CTXALLSUFF(pThis->StatRingWriteOutside)); ;
1213 return VINF_SUCCESS; /* outside of the ring range */
1214 }
1215 }
1216 STAM_COUNTER_INC(&CTXALLSUFF(pThis->StatRingWriteFailed)); ;
1217 return VINF_IOM_R3_MMIO_WRITE; /* handle in ring3 */
1218}
1219
1220# else /* IN_RING3 */
1221
1222/**
1223 * #PF Handler callback for physical access handler ranges (MMIO among others) in HC.
1224 *
1225 * The handler can not raise any faults, it's mainly for monitoring write access
1226 * to certain pages.
1227 *
1228 * @returns VINF_SUCCESS if the handler have carried out the operation.
1229 * @returns VINF_PGM_HANDLER_DO_DEFAULT if the caller should carry out the access operation.
1230 * @param pVM VM Handle.
1231 * @param GCPhys The physical address the guest is writing to.
1232 * @param pvPhys The HC mapping of that address.
1233 * @param pvBuf What the guest is reading/writing.
1234 * @param cbBuf How much it's reading/writing.
1235 * @param enmAccessType The access type.
1236 * @param pvUser User argument.
1237 */
1238static DECLCALLBACK(int) pcnetHandleRingWrite(PVM pVM, RTGCPHYS GCPhys, void *pvPhys, void *pvBuf,
1239 size_t cbBuf, PGMACCESSTYPE enmAccessType, void *pvUser)
1240{
1241 PPDMDEVINS pDevIns = (PPDMDEVINS)pvUser;
1242 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
1243
1244 Log(("#%d pcnetHandleRingWrite: write to %#010x\n", PCNET_INST_NR, GCPhys));
1245#ifdef VBOX_WITH_STATISTICS
1246 STAM_COUNTER_INC(&CTXSUFF(pThis->StatRingWrite));
1247 if (GCPhys >= pThis->GCRDRA && GCPhys < pcnetRdraAddr(pThis, 0))
1248 STAM_COUNTER_INC(&pThis->StatRCVRingWrite);
1249 else if (GCPhys >= pThis->GCTDRA && GCPhys < pcnetTdraAddr(pThis, 0))
1250 STAM_COUNTER_INC(&pThis->StatTXRingWrite);
1251#endif
1252 /* Perform the actual write */
1253 memcpy((char *)pvPhys, pvBuf, cbBuf);
1254
1255 /* Writes done by our code don't require polling of course */
1256 if (PDMCritSectIsOwner(&pThis->CritSect) == false)
1257 {
1258 if ( (GCPhys >= pThis->GCTDRA && GCPhys + cbBuf < pcnetTdraAddr(pThis, 0))
1259#ifdef PCNET_MONITOR_RECEIVE_RING
1260 || (GCPhys >= pThis->GCRDRA && GCPhys + cbBuf < pcnetRdraAddr(pThis, 0))
1261#endif
1262 )
1263 {
1264 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
1265 AssertReleaseRC(rc);
1266 /* Check if we can do something now */
1267 pcnetPollRxTx(pThis);
1268 pcnetUpdateIrq(pThis);
1269 PDMCritSectLeave(&pThis->CritSect);
1270 }
1271 }
1272 return VINF_SUCCESS;
1273}
1274# endif /* !IN_RING3 */
1275#endif /* PCNET_NO_POLLING */
1276
1277static void pcnetSoftReset(PCNetState *pThis)
1278{
1279 Log(("#%d pcnetSoftReset:\n", PCNET_INST_NR));
1280
1281 pThis->u32Lnkst = 0x40;
1282 pThis->GCRDRA = 0;
1283 pThis->GCTDRA = 0;
1284 pThis->u32RAP = 0;
1285
1286 pThis->aCSR[0] = 0x0004;
1287 pThis->aCSR[3] = 0x0000;
1288 pThis->aCSR[4] = 0x0115;
1289 pThis->aCSR[5] = 0x0000;
1290 pThis->aCSR[6] = 0x0000;
1291 pThis->aCSR[8] = 0;
1292 pThis->aCSR[9] = 0;
1293 pThis->aCSR[10] = 0;
1294 pThis->aCSR[11] = 0;
1295 pThis->aCSR[12] = RT_LE2H_U16(((uint16_t *)&pThis->aPROM[0])[0]);
1296 pThis->aCSR[13] = RT_LE2H_U16(((uint16_t *)&pThis->aPROM[0])[1]);
1297 pThis->aCSR[14] = RT_LE2H_U16(((uint16_t *)&pThis->aPROM[0])[2]);
1298 pThis->aCSR[15] &= 0x21c4;
1299 CSR_RCVRC(pThis) = 1;
1300 CSR_XMTRC(pThis) = 1;
1301 CSR_RCVRL(pThis) = 1;
1302 CSR_XMTRL(pThis) = 1;
1303 pThis->aCSR[80] = 0x1410;
1304 pThis->aCSR[88] = pThis->fAm79C973 ? CSR_VERSION_LOW_79C973 : CSR_VERSION_LOW_79C970A;
1305 pThis->aCSR[89] = CSR_VERSION_HIGH;
1306 pThis->aCSR[94] = 0x0000;
1307 pThis->aCSR[100] = 0x0200;
1308 pThis->aCSR[103] = 0x0105;
1309 pThis->aCSR[103] = 0x0105;
1310 CSR_MISSC(pThis) = 0;
1311 pThis->aCSR[114] = 0x0000;
1312 pThis->aCSR[122] = 0x0000;
1313 pThis->aCSR[124] = 0x0000;
1314}
1315
1316/**
1317 * Check if we have to send an interrupt to the guest. An interrupt can occur on
1318 * - csr0 (written quite often)
1319 * - csr4 (only written by pcnetSoftReset(), pcnetStop() or by the guest driver)
1320 * - csr5 (only written by pcnetSoftReset(), pcnetStop or by the driver guest)
1321 */
1322static void pcnetUpdateIrq(PCNetState *pThis)
1323{
1324 register int iISR = 0;
1325 register uint16_t csr0 = pThis->aCSR[0];
1326
1327 csr0 &= ~0x0080; /* clear INTR */
1328
1329 STAM_PROFILE_ADV_START(&pThis->StatInterrupt, a);
1330
1331 /* Linux guests set csr4=0x0915
1332 * W2k guests set csr3=0x4940 (disable BABL, MERR, IDON, DXSUFLO */
1333
1334#if 1
1335 if ( ( (csr0 & ~pThis->aCSR[3]) & 0x5f00)
1336 || (((pThis->aCSR[4]>>1) & ~pThis->aCSR[4]) & 0x0115)
1337 || (((pThis->aCSR[5]>>1) & pThis->aCSR[5]) & 0x0048))
1338#else
1339 if ( ( !(pThis->aCSR[3] & 0x4000) && !!(csr0 & 0x4000)) /* BABL */
1340 ||( !(pThis->aCSR[3] & 0x1000) && !!(csr0 & 0x1000)) /* MISS */
1341 ||( !(pThis->aCSR[3] & 0x0100) && !!(csr0 & 0x0100)) /* IDON */
1342 ||( !(pThis->aCSR[3] & 0x0200) && !!(csr0 & 0x0200)) /* TINT */
1343 ||( !(pThis->aCSR[3] & 0x0400) && !!(csr0 & 0x0400)) /* RINT */
1344 ||( !(pThis->aCSR[3] & 0x0800) && !!(csr0 & 0x0800)) /* MERR */
1345 ||( !(pThis->aCSR[4] & 0x0001) && !!(pThis->aCSR[4] & 0x0002)) /* JAB */
1346 ||( !(pThis->aCSR[4] & 0x0004) && !!(pThis->aCSR[4] & 0x0008)) /* TXSTRT */
1347 ||( !(pThis->aCSR[4] & 0x0010) && !!(pThis->aCSR[4] & 0x0020)) /* RCVO */
1348 ||( !(pThis->aCSR[4] & 0x0100) && !!(pThis->aCSR[4] & 0x0200)) /* MFCO */
1349 ||(!!(pThis->aCSR[5] & 0x0040) && !!(pThis->aCSR[5] & 0x0080)) /* EXDINT */
1350 ||(!!(pThis->aCSR[5] & 0x0008) && !!(pThis->aCSR[5] & 0x0010)) /* MPINT */)
1351#endif
1352 {
1353 iISR = !!(csr0 & 0x0040); /* CSR_INEA */
1354 csr0 |= 0x0080; /* set INTR */
1355 }
1356
1357#ifdef VBOX
1358 if (pThis->aCSR[4] & 0x0080) /* UINTCMD */
1359 {
1360 pThis->aCSR[4] &= ~0x0080; /* clear UINTCMD */
1361 pThis->aCSR[4] |= 0x0040; /* set UINT */
1362 Log(("#%d user int\n", PCNET_INST_NR));
1363 }
1364 if (pThis->aCSR[4] & csr0 & 0x0040 /* CSR_INEA */)
1365 {
1366 csr0 |= 0x0080; /* set INTR */
1367 iISR = 1;
1368 }
1369#else /* !VBOX */
1370 if (!!(pThis->aCSR[4] & 0x0080) && CSR_INEA(pThis)) /* UINTCMD */
1371 {
1372 pThis->aCSR[4] &= ~0x0080;
1373 pThis->aCSR[4] |= 0x0040; /* set UINT */
1374 csr0 |= 0x0080; /* set INTR */
1375 iISR = 1;
1376 Log(("#%d user int\n", PCNET_INST_NR));
1377 }
1378#endif /* !VBOX */
1379
1380#if 1
1381 if (((pThis->aCSR[5]>>1) & pThis->aCSR[5]) & 0x0500)
1382#else
1383 if ( (!!(pThis->aCSR[5] & 0x0400) && !!(pThis->aCSR[5] & 0x0800)) /* SINT */
1384 ||(!!(pThis->aCSR[5] & 0x0100) && !!(pThis->aCSR[5] & 0x0200)) /* SLPINT */)
1385#endif
1386 {
1387 iISR = 1;
1388 csr0 |= 0x0080; /* INTR */
1389 }
1390
1391 if ((pThis->aCSR[7] & 0x0C00) == 0x0C00) /* STINT + STINTE */
1392 iISR = 1;
1393
1394 pThis->aCSR[0] = csr0;
1395
1396 Log2(("#%d set irq iISR=%d\n", PCNET_INST_NR, iISR));
1397
1398 /* normal path is to _not_ change the IRQ status */
1399 if (RT_UNLIKELY(iISR != pThis->iISR))
1400 {
1401 Log(("#%d INTA=%d\n", PCNET_INST_NR, iISR));
1402 PDMDevHlpPCISetIrq(PCNETSTATE_2_DEVINS(pThis), 0, iISR);
1403 pThis->iISR = iISR;
1404 }
1405 STAM_PROFILE_ADV_STOP(&pThis->StatInterrupt, a);
1406}
1407
1408/**
1409 * Enable/disable the private guest interface.
1410 */
1411static void pcnetEnablePrivateIf(PCNetState *pThis)
1412{
1413 bool fPrivIfEnabled = pThis->pSharedMMIOR3
1414 && !!(pThis->CTX_SUFF(pSharedMMIO)->fFlags & PCNET_GUEST_FLAGS_ADMIT_GUEST);
1415 if (fPrivIfEnabled != pThis->fPrivIfEnabled)
1416 {
1417 pThis->fPrivIfEnabled = fPrivIfEnabled;
1418 LogRel(("PCNet#%d: %s private interface\n", PCNET_INST_NR, fPrivIfEnabled ? "Enabling" : "Disabling"));
1419 }
1420}
1421
1422#ifdef IN_RING3
1423#ifdef PCNET_NO_POLLING
1424static void pcnetUpdateRingHandlers(PCNetState *pThis)
1425{
1426 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
1427 int rc;
1428
1429 Log(("pcnetUpdateRingHandlers TD %RX32 size %#x -> %RX32 ?size? %#x\n", pThis->TDRAPhysOld, pThis->cbTDRAOld, pThis->GCTDRA, pcnetTdraAddr(pThis, 0)));
1430 Log(("pcnetUpdateRingHandlers RX %RX32 size %#x -> %RX32 ?size? %#x\n", pThis->RDRAPhysOld, pThis->cbRDRAOld, pThis->GCRDRA, pcnetRdraAddr(pThis, 0)));
1431
1432 /** @todo unregister order not correct! */
1433
1434#ifdef PCNET_MONITOR_RECEIVE_RING
1435 if (pThis->GCRDRA != pThis->RDRAPhysOld || CSR_RCVRL(pThis) != pThis->cbRDRAOld)
1436 {
1437 if (pThis->RDRAPhysOld != 0)
1438 PGMHandlerPhysicalDeregister(PDMDevHlpGetVM(pDevIns),
1439 pThis->RDRAPhysOld & ~PAGE_OFFSET_MASK);
1440
1441 rc = PGMR3HandlerPhysicalRegister(PDMDevHlpGetVM(pDevIns),
1442 PGMPHYSHANDLERTYPE_PHYSICAL_WRITE,
1443 pThis->GCRDRA & ~PAGE_OFFSET_MASK,
1444 RT_ALIGN(pcnetRdraAddr(pThis, 0), PAGE_SIZE) - 1,
1445 pcnetHandleRingWrite, pDevIns,
1446 g_DevicePCNet.szR0Mod, "pcnetHandleRingWrite",
1447 pThis->pDevInsHC->pvInstanceDataHC,
1448 g_DevicePCNet.szRCMod, "pcnetHandleRingWrite",
1449 pThis->pDevInsHC->pvInstanceDataRC,
1450 "PCNet receive ring write access handler");
1451 AssertRC(rc);
1452
1453 pThis->RDRAPhysOld = pThis->GCRDRA;
1454 pThis->cbRDRAOld = pcnetRdraAddr(pThis, 0);
1455 }
1456#endif /* PCNET_MONITOR_RECEIVE_RING */
1457
1458#ifdef PCNET_MONITOR_RECEIVE_RING
1459 /* 3 possibilities:
1460 * 1) TDRA on different physical page as RDRA
1461 * 2) TDRA completely on same physical page as RDRA
1462 * 3) TDRA & RDRA overlap partly with different physical pages
1463 */
1464 RTGCPHYS32 RDRAPageStart = pThis->GCRDRA & ~PAGE_OFFSET_MASK;
1465 RTGCPHYS32 RDRAPageEnd = (pcnetRdraAddr(pThis, 0) - 1) & ~PAGE_OFFSET_MASK;
1466 RTGCPHYS32 TDRAPageStart = pThis->GCTDRA & ~PAGE_OFFSET_MASK;
1467 RTGCPHYS32 TDRAPageEnd = (pcnetTdraAddr(pThis, 0) - 1) & ~PAGE_OFFSET_MASK;
1468
1469 if ( RDRAPageStart > TDRAPageEnd
1470 || TDRAPageStart > RDRAPageEnd)
1471 {
1472#endif /* PCNET_MONITOR_RECEIVE_RING */
1473 /* 1) */
1474 if (pThis->GCTDRA != pThis->TDRAPhysOld || CSR_XMTRL(pThis) != pThis->cbTDRAOld)
1475 {
1476 if (pThis->TDRAPhysOld != 0)
1477 PGMHandlerPhysicalDeregister(PDMDevHlpGetVM(pDevIns),
1478 pThis->TDRAPhysOld & ~PAGE_OFFSET_MASK);
1479
1480 rc = PGMR3HandlerPhysicalRegister(PDMDevHlpGetVM(pDevIns),
1481 PGMPHYSHANDLERTYPE_PHYSICAL_WRITE,
1482 pThis->GCTDRA & ~PAGE_OFFSET_MASK,
1483 RT_ALIGN(pcnetTdraAddr(pThis, 0), PAGE_SIZE) - 1,
1484 pcnetHandleRingWrite, pDevIns,
1485 g_DevicePCNet.szR0Mod, "pcnetHandleRingWrite",
1486 pThis->pDevInsHC->pvInstanceDataHC,
1487 g_DevicePCNet.szRCMod, "pcnetHandleRingWrite",
1488 pThis->pDevInsHC->pvInstanceDataRC,
1489 "PCNet transmit ring write access handler");
1490 AssertRC(rc);
1491
1492 pThis->TDRAPhysOld = pThis->GCTDRA;
1493 pThis->cbTDRAOld = pcnetTdraAddr(pThis, 0);
1494 }
1495#ifdef PCNET_MONITOR_RECEIVE_RING
1496 }
1497 else
1498 if ( RDRAPageStart != TDRAPageStart
1499 && ( TDRAPageStart == RDRAPageEnd
1500 || TDRAPageEnd == RDRAPageStart
1501 )
1502 )
1503 {
1504 /* 3) */
1505 AssertFailed();
1506 }
1507 /* else 2) */
1508#endif
1509}
1510#endif /* PCNET_NO_POLLING */
1511
1512static void pcnetInit(PCNetState *pThis)
1513{
1514 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
1515 Log(("#%d pcnetInit: init_addr=%#010x\n", PCNET_INST_NR, PHYSADDR(pThis, CSR_IADR(pThis))));
1516
1517 /** @todo Documentation says that RCVRL and XMTRL are stored as two's complement!
1518 * Software is allowed to write these registers directly. */
1519#define PCNET_INIT() do { \
1520 PDMDevHlpPhysRead(pDevIns, PHYSADDR(pThis, CSR_IADR(pThis)), \
1521 (uint8_t *)&initblk, sizeof(initblk)); \
1522 pThis->aCSR[15] = RT_LE2H_U16(initblk.mode); \
1523 CSR_RCVRL(pThis) = (initblk.rlen < 9) ? (1 << initblk.rlen) : 512; \
1524 CSR_XMTRL(pThis) = (initblk.tlen < 9) ? (1 << initblk.tlen) : 512; \
1525 pThis->aCSR[ 6] = (initblk.tlen << 12) | (initblk.rlen << 8); \
1526 pThis->aCSR[ 8] = RT_LE2H_U16(initblk.ladrf1); \
1527 pThis->aCSR[ 9] = RT_LE2H_U16(initblk.ladrf2); \
1528 pThis->aCSR[10] = RT_LE2H_U16(initblk.ladrf3); \
1529 pThis->aCSR[11] = RT_LE2H_U16(initblk.ladrf4); \
1530 pThis->aCSR[12] = RT_LE2H_U16(initblk.padr1); \
1531 pThis->aCSR[13] = RT_LE2H_U16(initblk.padr2); \
1532 pThis->aCSR[14] = RT_LE2H_U16(initblk.padr3); \
1533 pThis->GCRDRA = PHYSADDR(pThis, initblk.rdra); \
1534 pThis->GCTDRA = PHYSADDR(pThis, initblk.tdra); \
1535} while (0)
1536
1537 pcnetEnablePrivateIf(pThis);
1538
1539 if (BCR_SSIZE32(pThis))
1540 {
1541 struct INITBLK32 initblk;
1542 pThis->GCUpperPhys = 0;
1543 PCNET_INIT();
1544 Log(("#%d initblk.rlen=%#04x, initblk.tlen=%#04x\n",
1545 PCNET_INST_NR, initblk.rlen, initblk.tlen));
1546 }
1547 else
1548 {
1549 struct INITBLK16 initblk;
1550 pThis->GCUpperPhys = (0xff00 & (uint32_t)pThis->aCSR[2]) << 16;
1551 PCNET_INIT();
1552 Log(("#%d initblk.rlen=%#04x, initblk.tlen=%#04x\n",
1553 PCNET_INST_NR, initblk.rlen, initblk.tlen));
1554 }
1555
1556#undef PCNET_INIT
1557
1558 size_t cbRxBuffers = 0;
1559 for (int i = CSR_RCVRL(pThis); i >= 1; i--)
1560 {
1561 RMD rmd;
1562 RTGCPHYS32 rdaddr = PHYSADDR(pThis, pcnetRdraAddr(pThis, i));
1563
1564 pcnetDescTouch(pThis, rdaddr);
1565 /* At this time it is not guaranteed that the buffers are already initialized. */
1566 if (pcnetRmdLoad(pThis, &rmd, rdaddr, false))
1567 {
1568 uint32_t cbBuf = 4096U-rmd.rmd1.bcnt;
1569 cbRxBuffers += cbBuf;
1570 }
1571 }
1572
1573 for (int i = CSR_XMTRL(pThis); i >= 1; i--)
1574 {
1575 RTGCPHYS32 tdaddr = PHYSADDR(pThis, pcnetTdraAddr(pThis, i));
1576
1577 pcnetDescTouch(pThis, tdaddr);
1578 }
1579
1580 /*
1581 * Heuristics: The Solaris pcn driver allocates too few RX buffers (128 buffers of a
1582 * size of 128 bytes are 16KB in summary) leading to frequent RX buffer overflows. In
1583 * that case we don't signal RX overflows through the CSR0_MISS flag as the driver
1584 * re-initializes the device on every miss. Other guests use at least 32 buffers of
1585 * usually 1536 bytes and should therefore not run into condition. If they are still
1586 * short in RX buffers we notify this condition.
1587 */
1588 pThis->fSignalRxMiss = (cbRxBuffers == 0 || cbRxBuffers >= 32*_1K);
1589
1590 if (pThis->pDrvR3)
1591 pThis->pDrvR3->pfnSetPromiscuousMode(pThis->pDrvR3, CSR_PROM(pThis));
1592
1593 CSR_RCVRC(pThis) = CSR_RCVRL(pThis);
1594 CSR_XMTRC(pThis) = CSR_XMTRL(pThis);
1595
1596#ifdef PCNET_NO_POLLING
1597 pcnetUpdateRingHandlers(pThis);
1598#endif
1599
1600 /* Reset cached RX and TX states */
1601 CSR_CRST(pThis) = CSR_CRBC(pThis) = CSR_NRST(pThis) = CSR_NRBC(pThis) = 0;
1602 CSR_CXST(pThis) = CSR_CXBC(pThis) = CSR_NXST(pThis) = CSR_NXBC(pThis) = 0;
1603
1604 LogRel(("PCNet#%d: Init: ss32=%d GCRDRA=%#010x[%d] GCTDRA=%#010x[%d]%s\n",
1605 PCNET_INST_NR, BCR_SSIZE32(pThis),
1606 pThis->GCRDRA, CSR_RCVRL(pThis), pThis->GCTDRA, CSR_XMTRL(pThis),
1607 !pThis->fSignalRxMiss ? " (CSR0_MISS disabled)" : ""));
1608
1609 pThis->aCSR[0] |= 0x0101; /* Initialization done */
1610 pThis->aCSR[0] &= ~0x0004; /* clear STOP bit */
1611}
1612#endif /* IN_RING3 */
1613
1614/**
1615 * Start RX/TX operation.
1616 */
1617static void pcnetStart(PCNetState *pThis)
1618{
1619 Log(("#%d pcnetStart:\n", PCNET_INST_NR));
1620 if (!CSR_DTX(pThis))
1621 pThis->aCSR[0] |= 0x0010; /* set TXON */
1622 if (!CSR_DRX(pThis))
1623 pThis->aCSR[0] |= 0x0020; /* set RXON */
1624 pcnetEnablePrivateIf(pThis);
1625 pThis->aCSR[0] &= ~0x0004; /* clear STOP bit */
1626 pThis->aCSR[0] |= 0x0002; /* STRT */
1627 pcnetPollTimerStart(pThis); /* start timer if it was stopped */
1628}
1629
1630/**
1631 * Stop RX/TX operation.
1632 */
1633static void pcnetStop(PCNetState *pThis)
1634{
1635 Log(("#%d pcnetStop:\n", PCNET_INST_NR));
1636 pThis->aCSR[0] &= ~0x7feb;
1637 pThis->aCSR[0] |= 0x0014;
1638 pThis->aCSR[4] &= ~0x02c2;
1639 pThis->aCSR[5] &= ~0x0011;
1640 pcnetEnablePrivateIf(pThis);
1641 pcnetPollTimer(pThis);
1642}
1643
1644#ifdef IN_RING3
1645static DECLCALLBACK(void) pcnetWakeupReceive(PPDMDEVINS pDevIns)
1646{
1647 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
1648 STAM_COUNTER_INC(&pThis->StatRxOverflowWakeup);
1649 if (pThis->hEventOutOfRxSpace != NIL_RTSEMEVENT)
1650 RTSemEventSignal(pThis->hEventOutOfRxSpace);
1651}
1652
1653static DECLCALLBACK(bool) pcnetCanRxQueueConsumer(PPDMDEVINS pDevIns, PPDMQUEUEITEMCORE pItem)
1654{
1655 pcnetWakeupReceive(pDevIns);
1656 return true;
1657}
1658#endif /* IN_RING3 */
1659
1660
1661/**
1662 * Poll Receive Descriptor Table Entry and cache the results in the appropriate registers.
1663 * Note: Once a descriptor belongs to the network card (this driver), it cannot be changed
1664 * by the host (the guest driver) anymore. Well, it could but the results are undefined by
1665 * definition.
1666 * @param fSkipCurrent if true, don't scan the current RDTE.
1667 */
1668static void pcnetRdtePoll(PCNetState *pThis, bool fSkipCurrent=false)
1669{
1670 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatRdtePoll), a);
1671 /* assume lack of a next receive descriptor */
1672 CSR_NRST(pThis) = 0;
1673
1674 if (RT_LIKELY(pThis->GCRDRA))
1675 {
1676 /*
1677 * The current receive message descriptor.
1678 */
1679 RMD rmd;
1680 int i = CSR_RCVRC(pThis);
1681 RTGCPHYS32 addr;
1682
1683 if (i < 1)
1684 i = CSR_RCVRL(pThis);
1685
1686 if (!fSkipCurrent)
1687 {
1688 addr = pcnetRdraAddr(pThis, i);
1689 CSR_CRDA(pThis) = CSR_CRBA(pThis) = 0;
1690 CSR_CRBC(pThis) = CSR_CRST(pThis) = 0;
1691 if (!pcnetRmdLoad(pThis, &rmd, PHYSADDR(pThis, addr), true))
1692 {
1693 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatRdtePoll), a);
1694 return;
1695 }
1696 if (RT_LIKELY(!IS_RMD_BAD(rmd)))
1697 {
1698 CSR_CRDA(pThis) = addr; /* Receive Descriptor Address */
1699 CSR_CRBA(pThis) = rmd.rmd0.rbadr; /* Receive Buffer Address */
1700 CSR_CRBC(pThis) = rmd.rmd1.bcnt; /* Receive Byte Count */
1701 CSR_CRST(pThis) = ((uint32_t *)&rmd)[1] >> 16; /* Receive Status */
1702 if (pThis->fMaybeOutOfSpace)
1703 {
1704#ifdef IN_RING3
1705 pcnetWakeupReceive(PCNETSTATE_2_DEVINS(pThis));
1706#else
1707 PPDMQUEUEITEMCORE pItem = PDMQueueAlloc(pThis->CTX_SUFF(pCanRxQueue));
1708 if (pItem)
1709 PDMQueueInsert(pThis->CTX_SUFF(pCanRxQueue), pItem);
1710#endif
1711 }
1712 }
1713 else
1714 {
1715 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatRdtePoll), a);
1716 /* This is not problematic since we don't own the descriptor
1717 * We actually do own it, otherwise pcnetRmdLoad would have returned false.
1718 * Don't flood the release log with errors.
1719 */
1720 if (++pThis->uCntBadRMD < 50)
1721 LogRel(("PCNet#%d: BAD RMD ENTRIES AT %#010x (i=%d)\n",
1722 PCNET_INST_NR, addr, i));
1723 return;
1724 }
1725 }
1726
1727 /*
1728 * The next descriptor.
1729 */
1730 if (--i < 1)
1731 i = CSR_RCVRL(pThis);
1732 addr = pcnetRdraAddr(pThis, i);
1733 CSR_NRDA(pThis) = CSR_NRBA(pThis) = 0;
1734 CSR_NRBC(pThis) = 0;
1735 if (!pcnetRmdLoad(pThis, &rmd, PHYSADDR(pThis, addr), true))
1736 {
1737 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatRdtePoll), a);
1738 return;
1739 }
1740 if (RT_LIKELY(!IS_RMD_BAD(rmd)))
1741 {
1742 CSR_NRDA(pThis) = addr; /* Receive Descriptor Address */
1743 CSR_NRBA(pThis) = rmd.rmd0.rbadr; /* Receive Buffer Address */
1744 CSR_NRBC(pThis) = rmd.rmd1.bcnt; /* Receive Byte Count */
1745 CSR_NRST(pThis) = ((uint32_t *)&rmd)[1] >> 16; /* Receive Status */
1746 }
1747 else
1748 {
1749 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatRdtePoll), a);
1750 /* This is not problematic since we don't own the descriptor
1751 * We actually do own it, otherwise pcnetRmdLoad would have returned false.
1752 * Don't flood the release log with errors.
1753 */
1754 if (++pThis->uCntBadRMD < 50)
1755 LogRel(("PCNet#%d: BAD RMD ENTRIES + AT %#010x (i=%d)\n",
1756 PCNET_INST_NR, addr, i));
1757 return;
1758 }
1759
1760 /**
1761 * @todo NNRD
1762 */
1763 }
1764 else
1765 {
1766 CSR_CRDA(pThis) = CSR_CRBA(pThis) = CSR_NRDA(pThis) = CSR_NRBA(pThis) = 0;
1767 CSR_CRBC(pThis) = CSR_NRBC(pThis) = CSR_CRST(pThis) = 0;
1768 }
1769 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatRdtePoll), a);
1770}
1771
1772/**
1773 * Poll Transmit Descriptor Table Entry
1774 * @return true if transmit descriptors available
1775 */
1776static int pcnetTdtePoll(PCNetState *pThis, TMD *tmd)
1777{
1778 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTdtePoll), a);
1779 if (RT_LIKELY(pThis->GCTDRA))
1780 {
1781 RTGCPHYS32 cxda = pcnetTdraAddr(pThis, CSR_XMTRC(pThis));
1782
1783 if (!pcnetTmdLoad(pThis, tmd, PHYSADDR(pThis, cxda), true))
1784 {
1785 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTdtePoll), a);
1786 return 0;
1787 }
1788
1789 if (RT_UNLIKELY(tmd->tmd1.ones != 15))
1790 {
1791 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTdtePoll), a);
1792 LogRel(("PCNet#%d: BAD TMD XDA=%#010x\n",
1793 PCNET_INST_NR, PHYSADDR(pThis, cxda)));
1794 return 0;
1795 }
1796
1797 /* previous xmit descriptor */
1798 CSR_PXDA(pThis) = CSR_CXDA(pThis);
1799 CSR_PXBC(pThis) = CSR_CXBC(pThis);
1800 CSR_PXST(pThis) = CSR_CXST(pThis);
1801
1802 /* set current transmit descriptor. */
1803 CSR_CXDA(pThis) = cxda;
1804 CSR_CXBC(pThis) = tmd->tmd1.bcnt;
1805 CSR_CXST(pThis) = ((uint32_t *)tmd)[1] >> 16;
1806 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTdtePoll), a);
1807 return CARD_IS_OWNER(CSR_CXST(pThis));
1808 }
1809 else
1810 {
1811 /** @todo consistency with previous receive descriptor */
1812 CSR_CXDA(pThis) = 0;
1813 CSR_CXBC(pThis) = CSR_CXST(pThis) = 0;
1814 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTdtePoll), a);
1815 return 0;
1816 }
1817}
1818
1819
1820/**
1821 * Write data into guest receive buffers.
1822 */
1823static void pcnetReceiveNoSync(PCNetState *pThis, const uint8_t *buf, size_t cbToRecv)
1824{
1825 PPDMDEVINS pDevIns = PCNETSTATE_2_DEVINS(pThis);
1826 int is_padr = 0, is_bcast = 0, is_ladr = 0;
1827 unsigned iRxDesc;
1828 int cbPacket;
1829
1830 if (RT_UNLIKELY(CSR_DRX(pThis) || CSR_STOP(pThis) || CSR_SPND(pThis) || !cbToRecv))
1831 return;
1832
1833 /*
1834 * Drop packets if the VM is not running yet/anymore.
1835 */
1836 VMSTATE enmVMState = PDMDevHlpVMState(pDevIns);
1837 if ( enmVMState != VMSTATE_RUNNING
1838 && enmVMState != VMSTATE_RUNNING_LS)
1839 return;
1840
1841 /*
1842 * Drop packets if the cable is not connected
1843 */
1844 if (!pcnetIsLinkUp(pThis))
1845 return;
1846
1847 Log(("#%d pcnetReceiveNoSync: size=%d\n", PCNET_INST_NR, cbToRecv));
1848
1849 /*
1850 * Perform address matching.
1851 */
1852 if ( CSR_PROM(pThis)
1853 || (is_padr = padr_match(pThis, buf, cbToRecv))
1854 || (is_bcast = padr_bcast(pThis, buf, cbToRecv))
1855 || (is_ladr = ladr_match(pThis, buf, cbToRecv)))
1856 {
1857 if (HOST_IS_OWNER(CSR_CRST(pThis)))
1858 pcnetRdtePoll(pThis);
1859 if (RT_UNLIKELY(HOST_IS_OWNER(CSR_CRST(pThis))))
1860 {
1861 /* Not owned by controller. This should not be possible as
1862 * we already called pcnetCanReceive(). */
1863 LogRel(("PCNet#%d: no buffer: RCVRC=%d\n",
1864 PCNET_INST_NR, CSR_RCVRC(pThis)));
1865 /* Dump the status of all RX descriptors */
1866 const unsigned cb = 1 << pThis->iLog2DescSize;
1867 RTGCPHYS32 GCPhys = pThis->GCRDRA;
1868 iRxDesc = CSR_RCVRL(pThis);
1869 while (iRxDesc-- > 0)
1870 {
1871 RMD rmd;
1872 pcnetRmdLoad(pThis, &rmd, PHYSADDR(pThis, GCPhys), false);
1873 LogRel((" %#010x\n", rmd.rmd1));
1874 GCPhys += cb;
1875 }
1876 pThis->aCSR[0] |= 0x1000; /* Set MISS flag */
1877 CSR_MISSC(pThis)++;
1878 }
1879 else
1880 {
1881 uint8_t *src = &pThis->abRecvBuf[8];
1882 RTGCPHYS32 crda = CSR_CRDA(pThis);
1883 RTGCPHYS32 next_crda;
1884 RMD rmd, next_rmd;
1885
1886 memcpy(src, buf, cbToRecv);
1887 if (!CSR_ASTRP_RCV(pThis))
1888 {
1889 uint32_t fcs = ~0;
1890 uint8_t *p = src;
1891
1892 while (cbToRecv < 60)
1893 src[cbToRecv++] = 0;
1894 while (p != &src[cbToRecv])
1895 CRC(fcs, *p++);
1896 ((uint32_t *)&src[cbToRecv])[0] = htonl(fcs);
1897 /* FCS at end of packet */
1898 }
1899 cbToRecv += 4;
1900 cbPacket = (int)cbToRecv; Assert((size_t)cbPacket == cbToRecv);
1901
1902#ifdef PCNET_DEBUG_MATCH
1903 PRINT_PKTHDR(buf);
1904#endif
1905
1906 pcnetRmdLoad(pThis, &rmd, PHYSADDR(pThis, crda), false);
1907 /*if (!CSR_LAPPEN(pThis))*/
1908 rmd.rmd1.stp = 1;
1909
1910 size_t cbBuf = RT_MIN(4096 - (size_t)rmd.rmd1.bcnt, cbToRecv);
1911 RTGCPHYS32 rbadr = PHYSADDR(pThis, rmd.rmd0.rbadr);
1912
1913 /* save the old value to check if it was changed as long as we didn't
1914 * hold the critical section */
1915 iRxDesc = CSR_RCVRC(pThis);
1916
1917 /* We have to leave the critical section here or we risk deadlocking
1918 * with EMT when the write is to an unallocated page or has an access
1919 * handler associated with it.
1920 *
1921 * This shouldn't be a problem because:
1922 * - any modification to the RX descriptor by the driver is
1923 * forbidden as long as it is owned by the device
1924 * - we don't cache any register state beyond this point
1925 */
1926 PDMCritSectLeave(&pThis->CritSect);
1927 PDMDevHlpPhysWrite(pDevIns, rbadr, src, cbBuf);
1928 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
1929 AssertReleaseRC(rc);
1930
1931 /* RX disabled in the meantime? If so, abort RX. */
1932 if (RT_UNLIKELY(CSR_DRX(pThis) || CSR_STOP(pThis) || CSR_SPND(pThis)))
1933 return;
1934
1935 /* Was the register modified in the meantime? If so, don't touch the
1936 * register but still update the RX descriptor. */
1937 if (RT_LIKELY(iRxDesc == CSR_RCVRC(pThis)))
1938 {
1939 if (iRxDesc-- < 2)
1940 iRxDesc = CSR_RCVRL(pThis);
1941 CSR_RCVRC(pThis) = iRxDesc;
1942 }
1943 else
1944 iRxDesc = CSR_RCVRC(pThis);
1945
1946 src += cbBuf;
1947 cbToRecv -= cbBuf;
1948
1949 while (cbToRecv > 0)
1950 {
1951 /* Read the entire next descriptor as we're likely to need it. */
1952 next_crda = pcnetRdraAddr(pThis, iRxDesc);
1953
1954 /* Check next descriptor's own bit. If we don't own it, we have
1955 * to quit and write error status into the last descriptor we own.
1956 */
1957 if (!pcnetRmdLoad(pThis, &next_rmd, PHYSADDR(pThis, next_crda), true))
1958 break;
1959
1960 /* Write back current descriptor, clear the own bit. */
1961 pcnetRmdStorePassHost(pThis, &rmd, PHYSADDR(pThis, crda));
1962
1963 /* Switch to the next descriptor */
1964 crda = next_crda;
1965 rmd = next_rmd;
1966
1967 cbBuf = RT_MIN(4096 - (size_t)rmd.rmd1.bcnt, cbToRecv);
1968 RTGCPHYS32 rbadr2 = PHYSADDR(pThis, rmd.rmd0.rbadr);
1969
1970 /* We have to leave the critical section here or we risk deadlocking
1971 * with EMT when the write is to an unallocated page or has an access
1972 * handler associated with it. See above for additional comments. */
1973 PDMCritSectLeave(&pThis->CritSect);
1974 PDMDevHlpPhysWrite(pDevIns, rbadr2, src, cbBuf);
1975 rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
1976 AssertReleaseRC(rc);
1977
1978 /* RX disabled in the meantime? If so, abort RX. */
1979 if (RT_UNLIKELY(CSR_DRX(pThis) || CSR_STOP(pThis) || CSR_SPND(pThis)))
1980 return;
1981
1982 /* Was the register modified in the meantime? If so, don't touch the
1983 * register but still update the RX descriptor. */
1984 if (RT_LIKELY(iRxDesc == CSR_RCVRC(pThis)))
1985 {
1986 if (iRxDesc-- < 2)
1987 iRxDesc = CSR_RCVRL(pThis);
1988 CSR_RCVRC(pThis) = iRxDesc;
1989 }
1990 else
1991 iRxDesc = CSR_RCVRC(pThis);
1992
1993 src += cbBuf;
1994 cbToRecv -= cbBuf;
1995 }
1996
1997 if (RT_LIKELY(cbToRecv == 0))
1998 {
1999 rmd.rmd1.enp = 1;
2000 rmd.rmd1.pam = !CSR_PROM(pThis) && is_padr;
2001 rmd.rmd1.lafm = !CSR_PROM(pThis) && is_ladr;
2002 rmd.rmd1.bam = !CSR_PROM(pThis) && is_bcast;
2003 rmd.rmd2.mcnt = cbPacket;
2004
2005 STAM_REL_COUNTER_ADD(&pThis->StatReceiveBytes, cbPacket);
2006 }
2007 else
2008 {
2009 Log(("#%d: Overflow by %ubytes\n", PCNET_INST_NR, cbToRecv));
2010 rmd.rmd1.oflo = 1;
2011 rmd.rmd1.buff = 1;
2012 rmd.rmd1.err = 1;
2013 }
2014
2015 /* write back, clear the own bit */
2016 pcnetRmdStorePassHost(pThis, &rmd, PHYSADDR(pThis, crda));
2017
2018 pThis->aCSR[0] |= 0x0400;
2019
2020 Log(("#%d RCVRC=%d CRDA=%#010x\n", PCNET_INST_NR,
2021 CSR_RCVRC(pThis), PHYSADDR(pThis, CSR_CRDA(pThis))));
2022#ifdef PCNET_DEBUG_RMD
2023 PRINT_RMD(&rmd);
2024#endif
2025
2026 /* guest driver is owner: force repoll of current and next RDTEs */
2027 CSR_CRST(pThis) = 0;
2028 }
2029 }
2030
2031 /* see description of TXDPOLL:
2032 * ``transmit polling will take place following receive activities'' */
2033 pcnetPollRxTx(pThis);
2034 pcnetUpdateIrq(pThis);
2035}
2036
2037
2038/**
2039 * Transmit queue consumer
2040 * This is just a very simple way of delaying sending to R3.
2041 *
2042 * @returns Success indicator.
2043 * If false the item will not be removed and the flushing will stop.
2044 * @param pDevIns The device instance.
2045 * @param pItem The item to consume. Upon return this item will be freed.
2046 */
2047static DECLCALLBACK(bool) pcnetXmitQueueConsumer(PPDMDEVINS pDevIns, PPDMQUEUEITEMCORE pItem)
2048{
2049 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
2050 NOREF(pItem);
2051
2052 /*
2053 * Transmit as much as we can.
2054 */
2055 pcnetXmitPending(pThis, true /*fOnWorkerThread*/);
2056
2057 return true;
2058}
2059
2060
2061/**
2062 * Allocates a scatter/gather buffer for a transfer.
2063 *
2064 * @returns See PPDMINETWORKUP::pfnAllocBuf.
2065 * @param pThis The device instance.
2066 * @param cbMin The minimum buffer size.
2067 * @param fLoopback Set if we're in loopback mode.
2068 * @param pSgLoop Pointer to stack storage for the loopback SG.
2069 * @param ppSgBuf Where to return the SG buffer descriptor on success.
2070 * Always set.
2071 */
2072DECLINLINE(int) pcnetXmitAllocBuf(PCNetState *pThis, size_t cbMin, bool fLoopback,
2073 PPDMSCATTERGATHER pSgLoop, PPPDMSCATTERGATHER ppSgBuf)
2074{
2075 int rc;
2076
2077 if (RT_UNLIKELY(fLoopback)) /* hope that loopback mode is rare */
2078 {
2079 pSgLoop->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1;
2080 pSgLoop->cbUsed = 0;
2081 pSgLoop->cbAvailable = sizeof(pThis->abLoopBuf);
2082 pSgLoop->pvAllocator = pThis;
2083 pSgLoop->pvUser = NULL;
2084 pSgLoop->cSegs = 1;
2085 pSgLoop->aSegs[0].cbSeg = sizeof(pThis->abLoopBuf);
2086 pSgLoop->aSegs[0].pvSeg = pThis->abLoopBuf;
2087 *ppSgBuf = pSgLoop;
2088 rc = VINF_SUCCESS;
2089 }
2090 else
2091 {
2092 PPDMINETWORKUP pDrv = pThis->CTX_SUFF(pDrv);
2093 if (RT_LIKELY(pDrv))
2094 {
2095 rc = pDrv->pfnAllocBuf(pDrv, cbMin, NULL /*pGso*/, ppSgBuf);
2096 AssertMsg(rc == VINF_SUCCESS || rc == VERR_TRY_AGAIN || rc == VERR_NET_DOWN || rc == VERR_NO_MEMORY, ("%Rrc\n", rc));
2097 if (RT_FAILURE(rc))
2098 *ppSgBuf = NULL;
2099 }
2100 else
2101 {
2102 rc = VERR_NET_DOWN;
2103 *ppSgBuf = NULL;
2104 }
2105 }
2106 return rc;
2107}
2108
2109
2110/**
2111 * Frees an unsent buffer.
2112 *
2113 * @param pThis The device instance.
2114 * @param fLoopback Set if we're in loopback mode.
2115 * @param pSgBuf The SG to free. Can be NULL.
2116 */
2117DECLINLINE(void) pcnetXmitFreeBuf(PCNetState *pThis, bool fLoopback, PPDMSCATTERGATHER pSgBuf)
2118{
2119 if (pSgBuf)
2120 {
2121 if (RT_UNLIKELY(fLoopback))
2122 pSgBuf->pvAllocator = NULL;
2123 else
2124 {
2125 PPDMINETWORKUP pDrv = pThis->CTX_SUFF(pDrv);
2126 if (RT_LIKELY(pDrv))
2127 pDrv->pfnFreeBuf(pDrv, pSgBuf);
2128 }
2129 }
2130}
2131
2132
2133/**
2134 * Sends the scatter/gather buffer.
2135 *
2136 * Wrapper around PDMINETWORKUP::pfnSendBuf, so check it out for the fine print.
2137 *
2138 * @returns See PDMINETWORKUP::pfnSendBuf.
2139 * @param pThis The device instance.
2140 * @param fLoopback Set if we're in loopback mode.
2141 * @param pSgBuf The SG to send.
2142 * @param fOnWorkerThread Set if we're being called on a work thread. Clear
2143 * if an EMT.
2144 */
2145DECLINLINE(int) pcnetXmitSendBuf(PCNetState *pThis, bool fLoopback, PPDMSCATTERGATHER pSgBuf, bool fOnWorkerThread)
2146{
2147 int rc;
2148 STAM_REL_COUNTER_ADD(&pThis->StatTransmitBytes, pSgBuf->cbUsed);
2149 if (RT_UNLIKELY(fLoopback)) /* hope that loopback mode is rare */
2150 {
2151 Assert(pSgBuf->pvAllocator == (void *)pThis);
2152 pThis->Led.Asserted.s.fReading = pThis->Led.Actual.s.fReading = 1;
2153 if (HOST_IS_OWNER(CSR_CRST(pThis)))
2154 pcnetRdtePoll(pThis);
2155
2156 pcnetReceiveNoSync(pThis, pThis->abLoopBuf, pSgBuf->cbUsed);
2157 pThis->Led.Actual.s.fReading = 0;
2158 rc = VINF_SUCCESS;
2159 }
2160 else
2161 {
2162 /** @todo We used to leave the critsect here, not sure if that's necessary any
2163 * longer. If we could avoid that we could cache a bit more info in
2164 * the loop and make it part of the driver<->device contract, saving
2165 * critsect mess down in DrvIntNet. */
2166 STAM_PROFILE_START(&pThis->CTX_SUFF_Z(StatTransmitSend), a);
2167 if (pSgBuf->cbUsed > 70) /* unqualified guess */
2168 pThis->Led.Asserted.s.fWriting = pThis->Led.Actual.s.fWriting = 1;
2169
2170 PPDMINETWORKUP pDrv = pThis->CTX_SUFF(pDrv);
2171 if (RT_LIKELY(pDrv))
2172 {
2173 rc = pDrv->pfnSendBuf(pDrv, pSgBuf, fOnWorkerThread);
2174 AssertMsg(rc == VINF_SUCCESS || rc == VERR_NET_DOWN || rc == VERR_NET_NO_BUFFER_SPACE, ("%Rrc\n", rc));
2175 }
2176 else
2177 rc = VERR_NET_DOWN;
2178
2179 pThis->Led.Actual.s.fWriting = 0;
2180 STAM_PROFILE_STOP(&pThis->CTX_SUFF_Z(StatTransmitSend), a);
2181 }
2182 return rc;
2183}
2184
2185
2186/**
2187 * pcnetXmitRead1st worker that handles the unlikely + slower segmented code
2188 * path.
2189 */
2190static void pcnetXmitRead1stSlow(PCNetState *pThis, RTGCPHYS32 GCPhysFrame, unsigned cbFrame,
2191 PPDMSCATTERGATHER pSgBuf)
2192{
2193 AssertFailed(); /* This path is not supposed to be taken atm */
2194
2195 pSgBuf->cbUsed = cbFrame;
2196 for (uint32_t iSeg = 0; ; iSeg++)
2197 {
2198 Assert(iSeg < pSgBuf->cSegs);
2199 uint32_t cbRead = (uint32_t)RT_MIN(cbFrame, pSgBuf->aSegs[iSeg].cbSeg);
2200 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysFrame, pSgBuf->aSegs[iSeg].pvSeg, cbRead);
2201 cbFrame -= cbRead;
2202 if (!cbFrame)
2203 return;
2204 GCPhysFrame += cbRead;
2205 }
2206}
2207
2208
2209/**
2210 * pcnetXmitSgReadMore worker that handles the unlikely + slower segmented code
2211 * path.
2212 */
2213static void pcnetXmitReadMoreSlow(PCNetState *pThis, RTGCPHYS32 GCPhysFrame, unsigned cbFrame,
2214 PPDMSCATTERGATHER pSgBuf)
2215{
2216 AssertFailed(); /* This path is not supposed to be taken atm */
2217
2218 /* Find the segment which we'll put the next byte into. */
2219 size_t off = pSgBuf->cbUsed;
2220 size_t offSeg = 0;
2221 uint32_t iSeg = 0;
2222 while (offSeg + pSgBuf->aSegs[iSeg].cbSeg <= off)
2223 {
2224 offSeg += pSgBuf->aSegs[iSeg].cbSeg;
2225 iSeg++;
2226 Assert(iSeg < pSgBuf->cSegs);
2227 }
2228
2229 /* Commit before we start copying so we can decrement cbFrame. */
2230 pSgBuf->cbUsed = off + cbFrame;
2231
2232 /* Deal with the first segment if we at an offset into it. */
2233 if (off != offSeg)
2234 {
2235 size_t offIntoSeg = off - offSeg;
2236 uint32_t cbRead = (uint32_t)RT_MIN(pSgBuf->aSegs[iSeg].cbSeg - offIntoSeg, cbFrame);
2237 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysFrame,
2238 (uint8_t *)pSgBuf->aSegs[iSeg].pvSeg + offIntoSeg, cbRead);
2239 cbFrame -= cbRead;
2240 if (!cbFrame)
2241 return;
2242 GCPhysFrame += cbRead;
2243 iSeg++;
2244 }
2245
2246 /* For the remainder, we've got whole segments. */
2247 for (;; iSeg++)
2248 {
2249 Assert(iSeg < pSgBuf->cSegs);
2250
2251 uint32_t cbRead = (uint32_t)RT_MIN(pSgBuf->aSegs[iSeg].cbSeg, cbFrame);
2252 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysFrame, pSgBuf->aSegs[iSeg].pvSeg, cbRead);
2253 cbFrame -= cbRead;
2254 if (!cbFrame)
2255 return;
2256 GCPhysFrame += cbFrame;
2257 }
2258}
2259
2260
2261/**
2262 * Reads the first part of a frame into the scatter gather buffer.
2263 */
2264DECLINLINE(void) pcnetXmitRead1st(PCNetState *pThis, RTGCPHYS32 GCPhysFrame, const unsigned cbFrame,
2265 PPDMSCATTERGATHER pSgBuf)
2266{
2267 Assert(PDMCritSectIsOwner(&pThis->CritSect));
2268 Assert(pSgBuf->cbAvailable >= cbFrame);
2269
2270 if (RT_LIKELY(pSgBuf->aSegs[0].cbSeg >= cbFrame)) /* justification: all drivers returns a single segment atm. */
2271 {
2272 pSgBuf->cbUsed = cbFrame;
2273 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysFrame, pSgBuf->aSegs[0].pvSeg, cbFrame);
2274 }
2275 else
2276 pcnetXmitRead1stSlow(pThis, GCPhysFrame, cbFrame, pSgBuf);
2277}
2278
2279/**
2280 * Reads more into the current frame.
2281 */
2282DECLINLINE(void) pcnetXmitReadMore(PCNetState *pThis, RTGCPHYS32 GCPhysFrame, const unsigned cbFrame,
2283 PPDMSCATTERGATHER pSgBuf)
2284{
2285 size_t off = pSgBuf->cbUsed;
2286 Assert(pSgBuf->cbAvailable >= cbFrame + off);
2287
2288 if (RT_LIKELY(pSgBuf->aSegs[0].cbSeg >= cbFrame + off))
2289 {
2290 pSgBuf->cbUsed = cbFrame + off;
2291 PDMDevHlpPhysRead(pThis->CTX_SUFF(pDevIns), GCPhysFrame,
2292 (uint8_t *)pSgBuf->aSegs[0].pvSeg + off, cbFrame);
2293 }
2294 else
2295 pcnetXmitReadMoreSlow(pThis, GCPhysFrame, cbFrame, pSgBuf);
2296}
2297
2298
2299/**
2300 * Fails a TMD with a link down error.
2301 */
2302static void pcnetXmitFailTMDLinkDown(PCNetState *pThis, TMD *pTmd)
2303{
2304 /* make carrier error - hope this is correct. */
2305 pThis->cLinkDownReported++;
2306 pTmd->tmd2.lcar = pTmd->tmd1.err = 1;
2307 pThis->aCSR[0] |= RT_BIT(15) | RT_BIT(13); /* ERR | CERR */
2308 pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1;
2309 Log(("#%d pcnetTransmit: Signaling send error. swstyle=%#x\n",
2310 PCNET_INST_NR, pThis->aBCR[BCR_SWS]));
2311}
2312
2313/**
2314 * Fails a TMD with a generic error.
2315 */
2316static void pcnetXmitFailTMDGeneric(PCNetState *pThis, TMD *pTmd)
2317{
2318 /* make carrier error - hope this is correct. */
2319 pTmd->tmd2.lcar = pTmd->tmd1.err = 1;
2320 pThis->aCSR[0] |= RT_BIT(15) | RT_BIT(13); /* ERR | CERR */
2321 pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1;
2322 Log(("#%d pcnetTransmit: Signaling send error. swstyle=%#x\n",
2323 PCNET_INST_NR, pThis->aBCR[BCR_SWS]));
2324}
2325
2326
2327/**
2328 * Try to transmit frames
2329 */
2330static void pcnetTransmit(PCNetState *pThis)
2331{
2332 if (RT_UNLIKELY(!CSR_TXON(pThis)))
2333 {
2334 pThis->aCSR[0] &= ~0x0008; /* Clear TDMD */
2335 return;
2336 }
2337
2338 /*
2339 * Check the current transmit descriptors.
2340 */
2341 TMD tmd;
2342 if (!pcnetTdtePoll(pThis, &tmd))
2343 return;
2344
2345 /*
2346 * Clear TDMD.
2347 */
2348 pThis->aCSR[0] &= ~0x0008;
2349
2350 /*
2351 * Transmit pending packets if possible, defer it if we cannot do it
2352 * in the current context.
2353 */
2354#if defined(IN_RING0) || defined(IN_RC)
2355 if (!pThis->CTX_SUFF(pDrv))
2356 {
2357 PPDMQUEUEITEMCORE pItem = PDMQueueAlloc(pThis->CTX_SUFF(pXmitQueue));
2358 if (RT_UNLIKELY(pItem))
2359 PDMQueueInsert(pThis->CTX_SUFF(pXmitQueue), pItem);
2360 }
2361 else
2362#endif
2363 {
2364 int rc = pcnetXmitPending(pThis, false /*fOnWorkerThread*/);
2365 if (rc == VERR_TRY_AGAIN)
2366 rc = VINF_SUCCESS;
2367 AssertRC(rc);
2368 }
2369}
2370
2371
2372/**
2373 * Actually try transmit frames.
2374 *
2375 * @threads TX or EMT.
2376 */
2377static int pcnetAsyncTransmit(PCNetState *pThis, bool fOnWorkerThread)
2378{
2379 Assert(PDMCritSectIsOwner(&pThis->CritSect));
2380
2381 /*
2382 * Just cleared transmit demand if the transmitter is off.
2383 */
2384 if (RT_UNLIKELY(!CSR_TXON(pThis)))
2385 {
2386 pThis->aCSR[0] &= ~0x0008; /* Clear TDMD */
2387 return VINF_SUCCESS;
2388 }
2389
2390 /*
2391 * Iterate the transmit descriptors.
2392 */
2393 int rc;
2394 unsigned cFlushIrq = 0;
2395 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatTransmit), a);
2396 do
2397 {
2398#ifdef VBOX_WITH_STATISTICS
2399 unsigned cBuffers = 1;
2400#endif
2401 TMD tmd;
2402 if (!pcnetTdtePoll(pThis, &tmd))
2403 break;
2404
2405 /* Don't continue sending packets when the link is down. */
2406 if (RT_UNLIKELY( !pcnetIsLinkUp(pThis)
2407 && pThis->cLinkDownReported > PCNET_MAX_LINKDOWN_REPORTED)
2408 )
2409 break;
2410
2411#ifdef PCNET_DEBUG_TMD
2412 Log2(("#%d TMDLOAD %#010x\n", PCNET_INST_NR, PHYSADDR(pThis, CSR_CXDA(pThis))));
2413 PRINT_TMD(&tmd);
2414#endif
2415 bool const fLoopback = CSR_LOOP(pThis);
2416 PDMSCATTERGATHER SgLoop;
2417 PPDMSCATTERGATHER pSgBuf;
2418
2419 /*
2420 * The typical case - a complete packet.
2421 */
2422 if (tmd.tmd1.stp && tmd.tmd1.enp)
2423 {
2424 const unsigned cb = 4096 - tmd.tmd1.bcnt;
2425 Log(("#%d pcnetAsyncTransmit: stp&enp: cb=%d xmtrc=%#x\n", PCNET_INST_NR, cb, CSR_XMTRC(pThis)));
2426 STAM_COUNTER_INC(&pThis->StatTransmitCase1);
2427
2428 if (RT_LIKELY(pcnetIsLinkUp(pThis) || fLoopback))
2429 {
2430 /* From the manual: ``A zero length buffer is acceptable as
2431 * long as it is not the last buffer in a chain (STP = 0 and
2432 * ENP = 1).'' That means that the first buffer might have a
2433 * zero length if it is not the last one in the chain. */
2434 if (RT_LIKELY(cb <= MAX_FRAME))
2435 {
2436 rc = pcnetXmitAllocBuf(pThis, cb, fLoopback, &SgLoop, &pSgBuf);
2437 if (RT_SUCCESS(rc))
2438 {
2439 pcnetXmitRead1st(pThis, PHYSADDR(pThis, tmd.tmd0.tbadr), cb, pSgBuf);
2440 rc = pcnetXmitSendBuf(pThis, fLoopback, pSgBuf, fOnWorkerThread);
2441 }
2442 else if (rc == VERR_TRY_AGAIN)
2443 {
2444 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
2445 return VINF_SUCCESS;
2446 }
2447 if (RT_FAILURE(rc))
2448 pcnetXmitFailTMDLinkDown(pThis, &tmd);
2449 }
2450 else if (cb == 4096)
2451 {
2452 /* The Windows NT4 pcnet driver sometimes marks the first
2453 * unused descriptor as owned by us. Ignore that (by
2454 * passing it back). Do not update the ring counter in this
2455 * case (otherwise that driver becomes even more confused,
2456 * which causes transmit to stall for about 10 seconds).
2457 * This is just a workaround, not a final solution. */
2458 /* r=frank: IMHO this is the correct implementation. The
2459 * manual says: ``If the OWN bit is set and the buffer
2460 * length is 0, the OWN bit will be cleared. In the C-LANCE
2461 * the buffer length of 0 is interpreted as a 4096-byte
2462 * buffer.'' */
2463 LogRel(("PCNet#%d: pcnetAsyncTransmit: illegal 4kb frame -> ignoring\n", PCNET_INST_NR));
2464 pcnetTmdStorePassHost(pThis, &tmd, PHYSADDR(pThis, CSR_CXDA(pThis)));
2465 break;
2466 }
2467 else
2468 {
2469 /* Signal error, as this violates the Ethernet specs. */
2470 /** @todo check if the correct error is generated. */
2471 LogRel(("PCNet#%d: pcnetAsyncTransmit: illegal 4kb frame -> signalling error\n", PCNET_INST_NR));
2472
2473 pcnetXmitFailTMDGeneric(pThis, &tmd);
2474 }
2475 }
2476 else
2477 pcnetXmitFailTMDLinkDown(pThis, &tmd);
2478
2479 /* Write back the TMD and pass it to the host (clear own bit). */
2480 pcnetTmdStorePassHost(pThis, &tmd, PHYSADDR(pThis, CSR_CXDA(pThis)));
2481
2482 /* advance the ring counter register */
2483 if (CSR_XMTRC(pThis) < 2)
2484 CSR_XMTRC(pThis) = CSR_XMTRL(pThis);
2485 else
2486 CSR_XMTRC(pThis)--;
2487 }
2488 else if (tmd.tmd1.stp)
2489 {
2490 STAM_COUNTER_INC(&pThis->StatTransmitCase2);
2491
2492 /*
2493 * Read TMDs until end-of-packet or tdte poll fails (underflow).
2494 *
2495 * We allocate a maximum sized buffer here since we do not wish to
2496 * waste time finding out how much space we actually need even if
2497 * we could reliably do that on SMP guests.
2498 */
2499 unsigned cb = 4096 - tmd.tmd1.bcnt;
2500 rc = pcnetXmitAllocBuf(pThis, RT_MAX(MAX_FRAME, cb), fLoopback, &SgLoop, &pSgBuf);
2501 if (rc == VERR_TRY_AGAIN)
2502 {
2503 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
2504 return VINF_SUCCESS;
2505 }
2506
2507 bool fDropFrame = RT_FAILURE(rc);
2508 if (!fDropFrame)
2509 pcnetXmitRead1st(pThis, PHYSADDR(pThis, tmd.tmd0.tbadr), cb, pSgBuf);
2510
2511 for (;;)
2512 {
2513 /*
2514 * Advance the ring counter register and check the next tmd.
2515 */
2516#ifdef LOG_ENABLED
2517 const uint32_t iStart = CSR_XMTRC(pThis);
2518#endif
2519 const uint32_t GCPhysPrevTmd = PHYSADDR(pThis, CSR_CXDA(pThis));
2520 if (CSR_XMTRC(pThis) < 2)
2521 CSR_XMTRC(pThis) = CSR_XMTRL(pThis);
2522 else
2523 CSR_XMTRC(pThis)--;
2524
2525 TMD dummy;
2526 if (!pcnetTdtePoll(pThis, &dummy))
2527 {
2528 /*
2529 * Underflow!
2530 */
2531 tmd.tmd2.buff = tmd.tmd2.uflo = tmd.tmd1.err = 1;
2532 pThis->aCSR[0] |= 0x0200; /* set TINT */
2533 /* Don't allow the guest to clear TINT before reading it */
2534 pThis->u16CSR0LastSeenByGuest &= ~0x0200;
2535 if (!CSR_DXSUFLO(pThis)) /* stop on xmit underflow */
2536 pThis->aCSR[0] &= ~0x0010; /* clear TXON */
2537 pcnetTmdStorePassHost(pThis, &tmd, GCPhysPrevTmd);
2538 AssertMsgFailed(("pcnetAsyncTransmit: Underflow!!!\n"));
2539 pcnetXmitFreeBuf(pThis, fLoopback, pSgBuf);
2540 break;
2541 }
2542
2543 /* release & save the previous tmd, pass it to the host */
2544 pcnetTmdStorePassHost(pThis, &tmd, GCPhysPrevTmd);
2545
2546 /*
2547 * The next tmd.
2548 */
2549#ifdef VBOX_WITH_STATISTICS
2550 cBuffers++;
2551#endif
2552 pcnetTmdLoad(pThis, &tmd, PHYSADDR(pThis, CSR_CXDA(pThis)), false);
2553 cb = 4096 - tmd.tmd1.bcnt;
2554 if ( !fDropFrame
2555 && pSgBuf->cbUsed + cb <= MAX_FRAME) /** @todo this used to be ... + cb < MAX_FRAME. */
2556 pcnetXmitReadMore(pThis, PHYSADDR(pThis, tmd.tmd0.tbadr), cb, pSgBuf);
2557 else
2558 {
2559 AssertMsg(fDropFrame, ("pcnetAsyncTransmit: Frame is too big!!! %d bytes\n", pSgBuf->cbUsed + cb));
2560 fDropFrame = true;
2561 }
2562
2563 /*
2564 * Done already?
2565 */
2566 if (tmd.tmd1.enp)
2567 {
2568 Log(("#%d pcnetAsyncTransmit: stp: cb=%d xmtrc=%#x-%#x\n", PCNET_INST_NR,
2569 pSgBuf ? pSgBuf->cbUsed : 0, iStart, CSR_XMTRC(pThis)));
2570 if (!fDropFrame && (pcnetIsLinkUp(pThis) || fLoopback))
2571 {
2572 rc = pcnetXmitSendBuf(pThis, fLoopback, pSgBuf, fOnWorkerThread);
2573 fDropFrame = RT_FAILURE(rc);
2574 }
2575 else
2576 pcnetXmitFreeBuf(pThis, fLoopback, pSgBuf);
2577 if (fDropFrame)
2578 pcnetXmitFailTMDLinkDown(pThis, &tmd);
2579
2580 /* Write back the TMD, pass it to the host */
2581 pcnetTmdStorePassHost(pThis, &tmd, PHYSADDR(pThis, CSR_CXDA(pThis)));
2582
2583 /* advance the ring counter register */
2584 if (CSR_XMTRC(pThis) < 2)
2585 CSR_XMTRC(pThis) = CSR_XMTRL(pThis);
2586 else
2587 CSR_XMTRC(pThis)--;
2588 break;
2589 }
2590 } /* the loop */
2591 }
2592 else
2593 {
2594 /*
2595 * We underflowed in a previous transfer, or the driver is giving us shit.
2596 * Simply stop the transmitting for now.
2597 */
2598 /** @todo according to the specs we're supposed to clear the own bit and move on to the next one. */
2599 Log(("#%d pcnetAsyncTransmit: guest is giving us shit!\n", PCNET_INST_NR));
2600 break;
2601 }
2602 /* Update TDMD, TXSTRT and TINT. */
2603 pThis->aCSR[0] &= ~0x0008; /* clear TDMD */
2604
2605 pThis->aCSR[4] |= 0x0008; /* set TXSTRT */
2606 if ( !CSR_TOKINTD(pThis) /* Transmit OK Interrupt Disable, no infl. on errors. */
2607 || (CSR_LTINTEN(pThis) && tmd.tmd1.ltint)
2608 || tmd.tmd1.err)
2609 {
2610 cFlushIrq++;
2611 }
2612
2613 /** @todo should we continue after an error (tmd.tmd1.err) or not? */
2614
2615 STAM_COUNTER_INC(&pThis->aStatXmitChainCounts[RT_MIN(cBuffers,
2616 RT_ELEMENTS(pThis->aStatXmitChainCounts)) - 1]);
2617 } while (CSR_TXON(pThis)); /* transfer on */
2618
2619 if (cFlushIrq)
2620 {
2621 STAM_COUNTER_INC(&pThis->aStatXmitFlush[RT_MIN(cFlushIrq, RT_ELEMENTS(pThis->aStatXmitFlush)) - 1]);
2622 /* The WinXP PCnet driver has apparently a bug: It sets CSR0.TDMD _before_
2623 * it clears CSR0.TINT. This can lead to a race where the driver clears
2624 * CSR0.TINT right after it was set by the device. The driver waits until
2625 * CSR0.TINT is set again but this will never happen. So prevent clearing
2626 * this bit as long as the driver didn't read it. xtracker #5288. */
2627 pThis->aCSR[0] |= 0x0200; /* set TINT */
2628 /* Don't allow the guest to clear TINT before reading it */
2629 pThis->u16CSR0LastSeenByGuest &= ~0x0200;
2630 pcnetUpdateIrq(pThis);
2631 }
2632
2633 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatTransmit), a);
2634
2635 return VINF_SUCCESS;
2636}
2637
2638
2639/**
2640 * Transmit pending descriptors.
2641 *
2642 * @returns VBox status code. VERR_TRY_AGAIN is returned if we're busy.
2643 *
2644 * @param pThis The PCNet instance data.
2645 * @param fOnWorkerThread Whether we're on a worker thread or on an EMT.
2646 */
2647static int pcnetXmitPending(PCNetState *pThis, bool fOnWorkerThread)
2648{
2649 int rc = VINF_SUCCESS;
2650
2651 /*
2652 * Grab the xmit lock of the driver as well as the E1K device state.
2653 */
2654 PPDMINETWORKUP pDrv = pThis->CTX_SUFF(pDrv);
2655 if (pDrv)
2656 {
2657 rc = pDrv->pfnBeginXmit(pDrv, false /*fOnWorkerThread*/);
2658 if (RT_FAILURE(rc))
2659 return rc;
2660 }
2661 rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
2662 if (RT_SUCCESS(rc))
2663 {
2664 /** @todo check if we're supposed to suspend now. */
2665 /*
2666 * Do the transmitting.
2667 */
2668 int rc2 = pcnetAsyncTransmit(pThis, false /*fOnWorkerThread*/);
2669 AssertReleaseRC(rc2);
2670
2671 /*
2672 * Release the locks.
2673 */
2674 PDMCritSectLeave(&pThis->CritSect);
2675 }
2676 else
2677 AssertLogRelRC(rc);
2678 if (pDrv)
2679 pDrv->pfnEndXmit(pDrv);
2680
2681 return rc;
2682}
2683
2684
2685/**
2686 * Poll for changes in RX and TX descriptor rings.
2687 */
2688static void pcnetPollRxTx(PCNetState *pThis)
2689{
2690 if (CSR_RXON(pThis))
2691 {
2692 /*
2693 * The second case is important for pcnetWaitReceiveAvail(): If CSR_CRST(pThis) was
2694 * true but pcnetCanReceive() returned false for some other reason we need to check
2695 * _now_ if we have to wakeup pcnetWaitReceiveAvail().
2696 */
2697 if ( HOST_IS_OWNER(CSR_CRST(pThis)) /* only poll RDTEs if none available or ... */
2698 || pThis->fMaybeOutOfSpace) /* ... for waking up pcnetWaitReceiveAvail() */
2699 pcnetRdtePoll(pThis);
2700 }
2701
2702 if (CSR_TDMD(pThis) || (CSR_TXON(pThis) && !CSR_DPOLL(pThis)))
2703 pcnetTransmit(pThis);
2704}
2705
2706
2707/**
2708 * Start the poller timer.
2709 * Poll timer interval is fixed to 500Hz. Don't stop it.
2710 * @thread EMT, TAP.
2711 */
2712static void pcnetPollTimerStart(PCNetState *pThis)
2713{
2714 TMTimerSetMillies(pThis->CTX_SUFF(pTimerPoll), 2);
2715}
2716
2717
2718/**
2719 * Update the poller timer.
2720 * @thread EMT.
2721 */
2722static void pcnetPollTimer(PCNetState *pThis)
2723{
2724 STAM_PROFILE_ADV_START(&pThis->StatPollTimer, a);
2725
2726#ifdef LOG_ENABLED
2727 TMD dummy;
2728 if (CSR_STOP(pThis) || CSR_SPND(pThis))
2729 Log2(("#%d pcnetPollTimer time=%#010llx CSR_STOP=%d CSR_SPND=%d\n",
2730 PCNET_INST_NR, RTTimeMilliTS(), CSR_STOP(pThis), CSR_SPND(pThis)));
2731 else
2732 Log2(("#%d pcnetPollTimer time=%#010llx TDMD=%d TXON=%d POLL=%d TDTE=%d TDRA=%#x\n",
2733 PCNET_INST_NR, RTTimeMilliTS(), CSR_TDMD(pThis), CSR_TXON(pThis),
2734 !CSR_DPOLL(pThis), pcnetTdtePoll(pThis, &dummy), pThis->GCTDRA));
2735 Log2(("#%d pcnetPollTimer: CSR_CXDA=%#x CSR_XMTRL=%d CSR_XMTRC=%d\n",
2736 PCNET_INST_NR, CSR_CXDA(pThis), CSR_XMTRL(pThis), CSR_XMTRC(pThis)));
2737#endif
2738#ifdef PCNET_DEBUG_TMD
2739 if (CSR_CXDA(pThis))
2740 {
2741 TMD tmd;
2742 pcnetTmdLoad(pThis, &tmd, PHYSADDR(pThis, CSR_CXDA(pThis)), false);
2743 Log2(("#%d pcnetPollTimer: TMDLOAD %#010x\n", PCNET_INST_NR, PHYSADDR(pThis, CSR_CXDA(pThis))));
2744 PRINT_TMD(&tmd);
2745 }
2746#endif
2747 if (CSR_TDMD(pThis))
2748 pcnetTransmit(pThis);
2749
2750 pcnetUpdateIrq(pThis);
2751
2752 /* If the receive thread is waiting for new descriptors, poll TX/RX even if polling
2753 * disabled. We wouldn't need to poll for new TX descriptors in that case but it will
2754 * not hurt as waiting for RX descriptors should happen very seldom */
2755 if (RT_LIKELY( !CSR_STOP(pThis)
2756 && !CSR_SPND(pThis)
2757 && ( !CSR_DPOLL(pThis)
2758 || pThis->fMaybeOutOfSpace)))
2759 {
2760 /* We ensure that we poll at least every 2ms (500Hz) but not more often than
2761 * 5000 times per second. This way we completely prevent the overhead from
2762 * heavy reprogramming the timer which turned out to be very CPU-intensive.
2763 * The drawback is that csr46 and csr47 are not updated properly anymore
2764 * but so far I have not seen any guest depending on these values. The 2ms
2765 * interval is the default polling interval of the PCNet card (65536/33MHz). */
2766#ifdef PCNET_NO_POLLING
2767 pcnetPollRxTx(pThis);
2768#else
2769 uint64_t u64Now = TMTimerGet(pThis->CTX_SUFF(pTimerPoll));
2770 if (RT_UNLIKELY(u64Now - pThis->u64LastPoll > 200000))
2771 {
2772 pThis->u64LastPoll = u64Now;
2773 pcnetPollRxTx(pThis);
2774 }
2775 if (!TMTimerIsActive(pThis->CTX_SUFF(pTimerPoll)))
2776 pcnetPollTimerStart(pThis);
2777#endif
2778 }
2779 STAM_PROFILE_ADV_STOP(&pThis->StatPollTimer, a);
2780}
2781
2782
2783static int pcnetCSRWriteU16(PCNetState *pThis, uint32_t u32RAP, uint32_t val)
2784{
2785 int rc = VINF_SUCCESS;
2786#ifdef PCNET_DEBUG_CSR
2787 Log(("#%d pcnetCSRWriteU16: rap=%d val=%#06x\n", PCNET_INST_NR, u32RAP, val));
2788#endif
2789 switch (u32RAP)
2790 {
2791 case 0:
2792 {
2793 uint16_t csr0 = pThis->aCSR[0];
2794 /* Clear any interrupt flags.
2795 * Don't clear an interrupt flag which was not seen by the guest yet. */
2796 csr0 &= ~(val & 0x7f00 & pThis->u16CSR0LastSeenByGuest);
2797 csr0 = (csr0 & ~0x0040) | (val & 0x0048);
2798 val = (val & 0x007f) | (csr0 & 0x7f00);
2799
2800 /* Iff STOP, STRT and INIT are set, clear STRT and INIT */
2801 if ((val & 7) == 7)
2802 val &= ~3;
2803
2804 Log(("#%d CSR0: old=%#06x new=%#06x\n", PCNET_INST_NR, pThis->aCSR[0], csr0));
2805
2806#ifndef IN_RING3
2807 if (!(csr0 & 0x0001/*init*/) && (val & 1))
2808 {
2809 Log(("#%d pcnetCSRWriteU16: pcnetInit requested => HC\n", PCNET_INST_NR));
2810 return VINF_IOM_R3_IOPORT_WRITE;
2811 }
2812#endif
2813 pThis->aCSR[0] = csr0;
2814
2815 if (!CSR_STOP(pThis) && (val & 4))
2816 pcnetStop(pThis);
2817
2818#ifdef IN_RING3
2819 if (!CSR_INIT(pThis) && (val & 1))
2820 pcnetInit(pThis);
2821#endif
2822
2823 if (!CSR_STRT(pThis) && (val & 2))
2824 pcnetStart(pThis);
2825
2826 if (CSR_TDMD(pThis))
2827 pcnetTransmit(pThis);
2828
2829 return rc;
2830 }
2831 case 1: /* IADRL */
2832 case 2: /* IADRH */
2833 case 8: /* LADRF 0..15 */
2834 case 9: /* LADRF 16..31 */
2835 case 10: /* LADRF 32..47 */
2836 case 11: /* LADRF 48..63 */
2837 case 12: /* PADR 0..15 */
2838 case 13: /* PADR 16..31 */
2839 case 14: /* PADR 32..47 */
2840 case 18: /* CRBAL */
2841 case 19: /* CRBAU */
2842 case 20: /* CXBAL */
2843 case 21: /* CXBAU */
2844 case 22: /* NRBAL */
2845 case 23: /* NRBAU */
2846 case 26: /* NRDAL */
2847 case 27: /* NRDAU */
2848 case 28: /* CRDAL */
2849 case 29: /* CRDAU */
2850 case 32: /* NXDAL */
2851 case 33: /* NXDAU */
2852 case 34: /* CXDAL */
2853 case 35: /* CXDAU */
2854 case 36: /* NNRDL */
2855 case 37: /* NNRDU */
2856 case 38: /* NNXDL */
2857 case 39: /* NNXDU */
2858 case 40: /* CRBCL */
2859 case 41: /* CRBCU */
2860 case 42: /* CXBCL */
2861 case 43: /* CXBCU */
2862 case 44: /* NRBCL */
2863 case 45: /* NRBCU */
2864 case 46: /* POLL */
2865 case 47: /* POLLINT */
2866 case 72: /* RCVRC */
2867 case 74: /* XMTRC */
2868 case 112: /* MISSC */
2869 if (CSR_STOP(pThis) || CSR_SPND(pThis))
2870 break;
2871 case 3: /* Interrupt Mask and Deferral Control */
2872 break;
2873 case 4: /* Test and Features Control */
2874 pThis->aCSR[4] &= ~(val & 0x026a);
2875 val &= ~0x026a;
2876 val |= pThis->aCSR[4] & 0x026a;
2877 break;
2878 case 5: /* Extended Control and Interrupt 1 */
2879 pThis->aCSR[5] &= ~(val & 0x0a90);
2880 val &= ~0x0a90;
2881 val |= pThis->aCSR[5] & 0x0a90;
2882 break;
2883 case 7: /* Extended Control and Interrupt 2 */
2884 {
2885 uint16_t csr7 = pThis->aCSR[7];
2886 csr7 &= ~0x0400 ;
2887 csr7 &= ~(val & 0x0800);
2888 csr7 |= (val & 0x0400);
2889 pThis->aCSR[7] = csr7;
2890 return rc;
2891 }
2892 case 15: /* Mode */
2893 if ((pThis->aCSR[15] & 0x8000) != (uint16_t)(val & 0x8000) && pThis->pDrvR3)
2894 {
2895 Log(("#%d: promiscuous mode changed to %d\n", PCNET_INST_NR, !!(val & 0x8000)));
2896#ifndef IN_RING3
2897 return VINF_IOM_R3_IOPORT_WRITE;
2898#else
2899 /* check for promiscuous mode change */
2900 if (pThis->pDrvR3)
2901 pThis->pDrvR3->pfnSetPromiscuousMode(pThis->pDrvR3, !!(val & 0x8000));
2902#endif
2903 }
2904 break;
2905 case 16: /* IADRL */
2906 return pcnetCSRWriteU16(pThis, 1, val);
2907 case 17: /* IADRH */
2908 return pcnetCSRWriteU16(pThis, 2, val);
2909
2910 /*
2911 * 24 and 25 are the Base Address of Receive Descriptor.
2912 * We combine and mirror these in GCRDRA.
2913 */
2914 case 24: /* BADRL */
2915 case 25: /* BADRU */
2916 if (!CSR_STOP(pThis) && !CSR_SPND(pThis))
2917 {
2918 Log(("#%d: WRITE CSR%d, %#06x !!\n", PCNET_INST_NR, u32RAP, val));
2919 return rc;
2920 }
2921 if (u32RAP == 24)
2922 pThis->GCRDRA = (pThis->GCRDRA & 0xffff0000) | (val & 0x0000ffff);
2923 else
2924 pThis->GCRDRA = (pThis->GCRDRA & 0x0000ffff) | ((val & 0x0000ffff) << 16);
2925 Log(("#%d: WRITE CSR%d, %#06x => GCRDRA=%08x (alt init)\n", PCNET_INST_NR, u32RAP, val, pThis->GCRDRA));
2926 break;
2927
2928 /*
2929 * 30 & 31 are the Base Address of Transmit Descriptor.
2930 * We combine and mirrorthese in GCTDRA.
2931 */
2932 case 30: /* BADXL */
2933 case 31: /* BADXU */
2934 if (!CSR_STOP(pThis) && !CSR_SPND(pThis))
2935 {
2936 Log(("#%d: WRITE CSR%d, %#06x !!\n", PCNET_INST_NR, u32RAP, val));
2937 return rc;
2938 }
2939 if (u32RAP == 30)
2940 pThis->GCTDRA = (pThis->GCTDRA & 0xffff0000) | (val & 0x0000ffff);
2941 else
2942 pThis->GCTDRA = (pThis->GCTDRA & 0x0000ffff) | ((val & 0x0000ffff) << 16);
2943 Log(("#%d: WRITE CSR%d, %#06x => GCTDRA=%08x (alt init)\n", PCNET_INST_NR, u32RAP, val, pThis->GCTDRA));
2944 break;
2945
2946 case 58: /* Software Style */
2947 rc = pcnetBCRWriteU16(pThis, BCR_SWS, val);
2948 break;
2949
2950 /*
2951 * Registers 76 and 78 aren't stored correctly (see todos), but I'm don't dare
2952 * try fix that right now. So, as a quick hack for 'alt init' I'll just correct them here.
2953 */
2954 case 76: /* RCVRL */ /** @todo call pcnetUpdateRingHandlers */
2955 /** @todo receive ring length is stored in two's complement! */
2956 case 78: /* XMTRL */ /** @todo call pcnetUpdateRingHandlers */
2957 /** @todo transmit ring length is stored in two's complement! */
2958 if (!CSR_STOP(pThis) && !CSR_SPND(pThis))
2959 {
2960 Log(("#%d: WRITE CSR%d, %#06x !!\n", PCNET_INST_NR, u32RAP, val));
2961 return rc;
2962 }
2963 Log(("#%d: WRITE CSR%d, %#06x (hacked %#06x) (alt init)\n", PCNET_INST_NR,
2964 u32RAP, val, 1 + ~(uint16_t)val));
2965 val = 1 + ~(uint16_t)val;
2966
2967 /*
2968 * HACK ALERT! Set the counter registers too.
2969 */
2970 pThis->aCSR[u32RAP - 4] = val;
2971 break;
2972
2973 default:
2974 return rc;
2975 }
2976 pThis->aCSR[u32RAP] = val;
2977 return rc;
2978}
2979
2980/**
2981 * Encode a 32-bit link speed into a custom 16-bit floating-point value
2982 */
2983static uint32_t pcnetLinkSpd(uint32_t speed)
2984{
2985 unsigned exp = 0;
2986
2987 while (speed & 0xFFFFE000)
2988 {
2989 speed /= 10;
2990 ++exp;
2991 }
2992 return (exp << 13) | speed;
2993}
2994
2995static uint32_t pcnetCSRReadU16(PCNetState *pThis, uint32_t u32RAP)
2996{
2997 uint32_t val;
2998 switch (u32RAP)
2999 {
3000 case 0:
3001 pcnetUpdateIrq(pThis);
3002 val = pThis->aCSR[0];
3003 val |= (val & 0x7800) ? 0x8000 : 0;
3004 pThis->u16CSR0LastSeenByGuest = val;
3005 break;
3006 case 16:
3007 return pcnetCSRReadU16(pThis, 1);
3008 case 17:
3009 return pcnetCSRReadU16(pThis, 2);
3010 case 58:
3011 return pcnetBCRReadU16(pThis, BCR_SWS);
3012 case 68: /* Custom register to pass link speed to driver */
3013 return pcnetLinkSpd(pThis->u32LinkSpeed);
3014 case 88:
3015 val = pThis->aCSR[89];
3016 val <<= 16;
3017 val |= pThis->aCSR[88];
3018 break;
3019 default:
3020 val = pThis->aCSR[u32RAP];
3021 }
3022#ifdef PCNET_DEBUG_CSR
3023 Log(("#%d pcnetCSRReadU16: rap=%d val=%#06x\n", PCNET_INST_NR, u32RAP, val));
3024#endif
3025 return val;
3026}
3027
3028static int pcnetBCRWriteU16(PCNetState *pThis, uint32_t u32RAP, uint32_t val)
3029{
3030 int rc = VINF_SUCCESS;
3031 u32RAP &= 0x7f;
3032#ifdef PCNET_DEBUG_BCR
3033 Log2(("#%d pcnetBCRWriteU16: rap=%d val=%#06x\n", PCNET_INST_NR, u32RAP, val));
3034#endif
3035 switch (u32RAP)
3036 {
3037 case BCR_SWS:
3038 if (!(CSR_STOP(pThis) || CSR_SPND(pThis)))
3039 return rc;
3040 val &= ~0x0300;
3041 switch (val & 0x00ff)
3042 {
3043 default:
3044 Log(("#%d Bad SWSTYLE=%#04x\n", PCNET_INST_NR, val & 0xff));
3045 // fall through
3046 case 0:
3047 val |= 0x0200; /* 16 bit */
3048 pThis->iLog2DescSize = 3;
3049 pThis->GCUpperPhys = (0xff00 & (uint32_t)pThis->aCSR[2]) << 16;
3050 break;
3051 case 1:
3052 val |= 0x0100; /* 32 bit */
3053 pThis->iLog2DescSize = 4;
3054 pThis->GCUpperPhys = 0;
3055 break;
3056 case 2:
3057 case 3:
3058 val |= 0x0300; /* 32 bit */
3059 pThis->iLog2DescSize = 4;
3060 pThis->GCUpperPhys = 0;
3061 break;
3062 }
3063 Log(("#%d BCR_SWS=%#06x\n", PCNET_INST_NR, val));
3064 pThis->aCSR[58] = val;
3065 /* fall through */
3066 case BCR_LNKST:
3067 case BCR_LED1:
3068 case BCR_LED2:
3069 case BCR_LED3:
3070 case BCR_MC:
3071 case BCR_FDC:
3072 case BCR_BSBC:
3073 case BCR_EECAS:
3074 case BCR_PLAT:
3075 case BCR_MIICAS:
3076 case BCR_MIIADDR:
3077 pThis->aBCR[u32RAP] = val;
3078 break;
3079
3080 case BCR_STVAL:
3081 val &= 0xffff;
3082 pThis->aBCR[BCR_STVAL] = val;
3083 if (pThis->fAm79C973)
3084 TMTimerSetNano(pThis->CTX_SUFF(pTimerSoftInt), 12800U * val);
3085 break;
3086
3087 case BCR_MIIMDR:
3088 pThis->aMII[pThis->aBCR[BCR_MIIADDR] & 0x1f] = val;
3089#ifdef PCNET_DEBUG_MII
3090 Log(("#%d pcnet: mii write %d <- %#x\n", PCNET_INST_NR, pThis->aBCR[BCR_MIIADDR] & 0x1f, val));
3091#endif
3092 break;
3093
3094 default:
3095 break;
3096 }
3097 return rc;
3098}
3099
3100static uint32_t pcnetMIIReadU16(PCNetState *pThis, uint32_t miiaddr)
3101{
3102 uint32_t val;
3103 bool autoneg, duplex, fast;
3104 STAM_COUNTER_INC(&pThis->StatMIIReads);
3105
3106 autoneg = (pThis->aBCR[BCR_MIICAS] & 0x20) != 0;
3107 duplex = (pThis->aBCR[BCR_MIICAS] & 0x10) != 0;
3108 fast = (pThis->aBCR[BCR_MIICAS] & 0x08) != 0;
3109
3110 switch (miiaddr)
3111 {
3112 case 0:
3113 /* MII basic mode control register. */
3114 val = 0;
3115 if (autoneg)
3116 val |= 0x1000; /* Enable auto negotiation. */
3117 if (fast)
3118 val |= 0x2000; /* 100 Mbps */
3119 if (duplex) /* Full duplex forced */
3120 val |= 0x0100; /* Full duplex */
3121 break;
3122
3123 case 1:
3124 /* MII basic mode status register. */
3125 val = 0x7800 /* Can do 100mbps FD/HD and 10mbps FD/HD. */
3126 | 0x0040 /* Mgmt frame preamble not required. */
3127 | 0x0020 /* Auto-negotiation complete. */
3128 | 0x0008 /* Able to do auto-negotiation. */
3129 | 0x0004 /* Link up. */
3130 | 0x0001; /* Extended Capability, i.e. registers 4+ valid. */
3131 if (!pThis->fLinkUp || pThis->fLinkTempDown) {
3132 val &= ~(0x0020 | 0x0004);
3133 pThis->cLinkDownReported++;
3134 }
3135 if (!autoneg) {
3136 /* Auto-negotiation disabled. */
3137 val &= ~(0x0020 | 0x0008);
3138 if (duplex)
3139 /* Full duplex forced. */
3140 val &= ~0x2800;
3141 else
3142 /* Half duplex forced. */
3143 val &= ~0x5000;
3144
3145 if (fast)
3146 /* 100 Mbps forced */
3147 val &= ~0x1800;
3148 else
3149 /* 10 Mbps forced */
3150 val &= ~0x6000;
3151 }
3152 break;
3153
3154 case 2:
3155 /* PHY identifier 1. */
3156 val = 0x22; /* Am79C874 PHY */
3157 break;
3158
3159 case 3:
3160 /* PHY identifier 2. */
3161 val = 0x561b; /* Am79C874 PHY */
3162 break;
3163
3164 case 4:
3165 /* Advertisement control register. */
3166 val = 0x01e0 /* Try 100mbps FD/HD and 10mbps FD/HD. */
3167#if 0
3168 // Advertising flow control is a) not the default, and b) confuses
3169 // the link speed detection routine in Windows PCnet driver
3170 | 0x0400 /* Try flow control. */
3171#endif
3172 | 0x0001; /* CSMA selector. */
3173 break;
3174
3175 case 5:
3176 /* Link partner ability register. */
3177 if (pThis->fLinkUp && !pThis->fLinkTempDown)
3178 val = 0x8000 /* Next page bit. */
3179 | 0x4000 /* Link partner acked us. */
3180 | 0x0400 /* Can do flow control. */
3181 | 0x01e0 /* Can do 100mbps FD/HD and 10mbps FD/HD. */
3182 | 0x0001; /* Use CSMA selector. */
3183 else
3184 {
3185 val = 0;
3186 pThis->cLinkDownReported++;
3187 }
3188 break;
3189
3190 case 6:
3191 /* Auto negotiation expansion register. */
3192 if (pThis->fLinkUp && !pThis->fLinkTempDown)
3193 val = 0x0008 /* Link partner supports npage. */
3194 | 0x0004 /* Enable npage words. */
3195 | 0x0001; /* Can do N-way auto-negotiation. */
3196 else
3197 {
3198 val = 0;
3199 pThis->cLinkDownReported++;
3200 }
3201 break;
3202
3203 default:
3204 val = 0;
3205 break;
3206 }
3207
3208#ifdef PCNET_DEBUG_MII
3209 Log(("#%d pcnet: mii read %d -> %#x\n", PCNET_INST_NR, miiaddr, val));
3210#endif
3211 return val;
3212}
3213
3214static uint32_t pcnetBCRReadU16(PCNetState *pThis, uint32_t u32RAP)
3215{
3216 uint32_t val;
3217 u32RAP &= 0x7f;
3218 switch (u32RAP)
3219 {
3220 case BCR_LNKST:
3221 case BCR_LED1:
3222 case BCR_LED2:
3223 case BCR_LED3:
3224 val = pThis->aBCR[u32RAP] & ~0x8000;
3225 /* Clear LNKSTE if we're not connected or if we've just loaded a VM state. */
3226 if (!pThis->pDrvR3 || pThis->fLinkTempDown || !pThis->fLinkUp)
3227 {
3228 if (u32RAP == 4)
3229 pThis->cLinkDownReported++;
3230 val &= ~0x40;
3231 }
3232 val |= (val & 0x017f & pThis->u32Lnkst) ? 0x8000 : 0;
3233 break;
3234
3235 case BCR_MIIMDR:
3236 if (pThis->fAm79C973 && (pThis->aBCR[BCR_MIIADDR] >> 5 & 0x1f) == 0)
3237 {
3238 uint32_t miiaddr = pThis->aBCR[BCR_MIIADDR] & 0x1f;
3239 val = pcnetMIIReadU16(pThis, miiaddr);
3240 }
3241 else
3242 val = 0xffff;
3243 break;
3244
3245 default:
3246 val = u32RAP < BCR_MAX_RAP ? pThis->aBCR[u32RAP] : 0;
3247 break;
3248 }
3249#ifdef PCNET_DEBUG_BCR
3250 Log2(("#%d pcnetBCRReadU16: rap=%d val=%#06x\n", PCNET_INST_NR, u32RAP, val));
3251#endif
3252 return val;
3253}
3254
3255#ifdef IN_RING3 /* move down */
3256static void pcnetHardReset(PCNetState *pThis)
3257{
3258 int i;
3259 uint16_t checksum;
3260
3261 /* Initialize the PROM */
3262 Assert(sizeof(pThis->MacConfigured) == 6);
3263 memcpy(pThis->aPROM, &pThis->MacConfigured, sizeof(pThis->MacConfigured));
3264 pThis->aPROM[ 8] = 0x00;
3265 pThis->aPROM[ 9] = 0x11;
3266 pThis->aPROM[12] = pThis->aPROM[13] = 0x00;
3267 pThis->aPROM[14] = pThis->aPROM[15] = 0x57;
3268
3269 for (i = 0, checksum = 0; i < 16; i++)
3270 checksum += pThis->aPROM[i];
3271 *(uint16_t *)&pThis->aPROM[12] = RT_H2LE_U16(checksum);
3272
3273 pThis->aBCR[BCR_MSRDA] = 0x0005;
3274 pThis->aBCR[BCR_MSWRA] = 0x0005;
3275 pThis->aBCR[BCR_MC ] = 0x0002;
3276 pThis->aBCR[BCR_LNKST] = 0x00c0;
3277 pThis->aBCR[BCR_LED1 ] = 0x0084;
3278 pThis->aBCR[BCR_LED2 ] = 0x0088;
3279 pThis->aBCR[BCR_LED3 ] = 0x0090;
3280 pThis->aBCR[BCR_FDC ] = 0x0000;
3281 pThis->aBCR[BCR_BSBC ] = 0x9001;
3282 pThis->aBCR[BCR_EECAS] = 0x0002;
3283 pThis->aBCR[BCR_STVAL] = 0xffff;
3284 pThis->aCSR[58 ] = /* CSR58 is an alias for BCR20 */
3285 pThis->aBCR[BCR_SWS ] = 0x0200;
3286 pThis->iLog2DescSize = 3;
3287 pThis->aBCR[BCR_PLAT ] = 0xff06;
3288 pThis->aBCR[BCR_MIIADDR ] = 0; /* Internal PHY on Am79C973 would be (0x1e << 5) */
3289 pThis->aBCR[BCR_PCIVID] = PCIDevGetVendorId(&pThis->PciDev);
3290 pThis->aBCR[BCR_PCISID] = PCIDevGetSubSystemId(&pThis->PciDev);
3291 pThis->aBCR[BCR_PCISVID] = PCIDevGetSubSystemVendorId(&pThis->PciDev);
3292
3293 /* Reset the error counter. */
3294 pThis->uCntBadRMD = 0;
3295
3296 pcnetSoftReset(pThis);
3297}
3298#endif /* IN_RING3 */
3299
3300static void pcnetAPROMWriteU8(PCNetState *pThis, uint32_t addr, uint32_t val)
3301{
3302 addr &= 0x0f;
3303 val &= 0xff;
3304 Log(("#%d pcnetAPROMWriteU8: addr=%#010x val=%#04x\n", PCNET_INST_NR, addr, val));
3305 /* Check APROMWE bit to enable write access */
3306 if (pcnetBCRReadU16(pThis, 2) & 0x80)
3307 pThis->aPROM[addr] = val;
3308}
3309
3310static uint32_t pcnetAPROMReadU8(PCNetState *pThis, uint32_t addr)
3311{
3312 uint32_t val = pThis->aPROM[addr &= 0x0f];
3313 Log(("#%d pcnetAPROMReadU8: addr=%#010x val=%#04x\n", PCNET_INST_NR, addr, val));
3314 return val;
3315}
3316
3317static int pcnetIoportWriteU8(PCNetState *pThis, uint32_t addr, uint32_t val)
3318{
3319 int rc = VINF_SUCCESS;
3320
3321#ifdef PCNET_DEBUG_IO
3322 Log2(("#%d pcnetIoportWriteU8: addr=%#010x val=%#06x\n", PCNET_INST_NR,
3323 addr, val));
3324#endif
3325 if (RT_LIKELY(!BCR_DWIO(pThis)))
3326 {
3327 switch (addr & 0x0f)
3328 {
3329 case 0x04: /* RESET */
3330 break;
3331 }
3332 }
3333 else
3334 Log(("#%d pcnetIoportWriteU8: addr=%#010x val=%#06x BCR_DWIO !!\n", PCNET_INST_NR, addr, val));
3335
3336 return rc;
3337}
3338
3339static uint32_t pcnetIoportReadU8(PCNetState *pThis, uint32_t addr, int *pRC)
3340{
3341 uint32_t val = ~0U;
3342
3343 *pRC = VINF_SUCCESS;
3344
3345 if (RT_LIKELY(!BCR_DWIO(pThis)))
3346 {
3347 switch (addr & 0x0f)
3348 {
3349 case 0x04: /* RESET */
3350 pcnetSoftReset(pThis);
3351 val = 0;
3352 break;
3353 }
3354 }
3355 else
3356 Log(("#%d pcnetIoportReadU8: addr=%#010x val=%#06x BCR_DWIO !!\n", PCNET_INST_NR, addr, val & 0xff));
3357
3358 pcnetUpdateIrq(pThis);
3359
3360#ifdef PCNET_DEBUG_IO
3361 Log2(("#%d pcnetIoportReadU8: addr=%#010x val=%#06x\n", PCNET_INST_NR, addr, val & 0xff));
3362#endif
3363 return val;
3364}
3365
3366static int pcnetIoportWriteU16(PCNetState *pThis, uint32_t addr, uint32_t val)
3367{
3368 int rc = VINF_SUCCESS;
3369
3370#ifdef PCNET_DEBUG_IO
3371 Log2(("#%d pcnetIoportWriteU16: addr=%#010x val=%#06x\n", PCNET_INST_NR,
3372 addr, val));
3373#endif
3374 if (RT_LIKELY(!BCR_DWIO(pThis)))
3375 {
3376 switch (addr & 0x0f)
3377 {
3378 case 0x00: /* RDP */
3379 pcnetPollTimer(pThis);
3380 rc = pcnetCSRWriteU16(pThis, pThis->u32RAP, val);
3381 pcnetUpdateIrq(pThis);
3382 break;
3383 case 0x02: /* RAP */
3384 pThis->u32RAP = val & 0x7f;
3385 break;
3386 case 0x06: /* BDP */
3387 rc = pcnetBCRWriteU16(pThis, pThis->u32RAP, val);
3388 break;
3389 }
3390 }
3391 else
3392 Log(("#%d pcnetIoportWriteU16: addr=%#010x val=%#06x BCR_DWIO !!\n", PCNET_INST_NR, addr, val));
3393
3394 return rc;
3395}
3396
3397static uint32_t pcnetIoportReadU16(PCNetState *pThis, uint32_t addr, int *pRC)
3398{
3399 uint32_t val = ~0U;
3400
3401 *pRC = VINF_SUCCESS;
3402
3403 if (RT_LIKELY(!BCR_DWIO(pThis)))
3404 {
3405 switch (addr & 0x0f)
3406 {
3407 case 0x00: /* RDP */
3408 /** @note if we're not polling, then the guest will tell us when to poll by setting TDMD in CSR0 */
3409 /** Polling is then useless here and possibly expensive. */
3410 if (!CSR_DPOLL(pThis))
3411 pcnetPollTimer(pThis);
3412
3413 val = pcnetCSRReadU16(pThis, pThis->u32RAP);
3414 if (pThis->u32RAP == 0) // pcnetUpdateIrq() already called by pcnetCSRReadU16()
3415 goto skip_update_irq;
3416 break;
3417 case 0x02: /* RAP */
3418 val = pThis->u32RAP;
3419 goto skip_update_irq;
3420 case 0x04: /* RESET */
3421 pcnetSoftReset(pThis);
3422 val = 0;
3423 break;
3424 case 0x06: /* BDP */
3425 val = pcnetBCRReadU16(pThis, pThis->u32RAP);
3426 break;
3427 }
3428 }
3429 else
3430 Log(("#%d pcnetIoportReadU16: addr=%#010x val=%#06x BCR_DWIO !!\n", PCNET_INST_NR, addr, val & 0xffff));
3431
3432 pcnetUpdateIrq(pThis);
3433
3434skip_update_irq:
3435#ifdef PCNET_DEBUG_IO
3436 Log2(("#%d pcnetIoportReadU16: addr=%#010x val=%#06x\n", PCNET_INST_NR, addr, val & 0xffff));
3437#endif
3438 return val;
3439}
3440
3441static int pcnetIoportWriteU32(PCNetState *pThis, uint32_t addr, uint32_t val)
3442{
3443 int rc = VINF_SUCCESS;
3444
3445#ifdef PCNET_DEBUG_IO
3446 Log2(("#%d pcnetIoportWriteU32: addr=%#010x val=%#010x\n", PCNET_INST_NR,
3447 addr, val));
3448#endif
3449 if (RT_LIKELY(BCR_DWIO(pThis)))
3450 {
3451 switch (addr & 0x0f)
3452 {
3453 case 0x00: /* RDP */
3454 pcnetPollTimer(pThis);
3455 rc = pcnetCSRWriteU16(pThis, pThis->u32RAP, val & 0xffff);
3456 pcnetUpdateIrq(pThis);
3457 break;
3458 case 0x04: /* RAP */
3459 pThis->u32RAP = val & 0x7f;
3460 break;
3461 case 0x0c: /* BDP */
3462 rc = pcnetBCRWriteU16(pThis, pThis->u32RAP, val & 0xffff);
3463 break;
3464 }
3465 }
3466 else if ((addr & 0x0f) == 0)
3467 {
3468 /* switch device to dword I/O mode */
3469 pcnetBCRWriteU16(pThis, BCR_BSBC, pcnetBCRReadU16(pThis, BCR_BSBC) | 0x0080);
3470#ifdef PCNET_DEBUG_IO
3471 Log2(("device switched into dword i/o mode\n"));
3472#endif
3473 }
3474 else
3475 Log(("#%d pcnetIoportWriteU32: addr=%#010x val=%#010x !BCR_DWIO !!\n", PCNET_INST_NR, addr, val));
3476
3477 return rc;
3478}
3479
3480static uint32_t pcnetIoportReadU32(PCNetState *pThis, uint32_t addr, int *pRC)
3481{
3482 uint32_t val = ~0U;
3483
3484 *pRC = VINF_SUCCESS;
3485
3486 if (RT_LIKELY(BCR_DWIO(pThis)))
3487 {
3488 switch (addr & 0x0f)
3489 {
3490 case 0x00: /* RDP */
3491 /** @note if we're not polling, then the guest will tell us when to poll by setting TDMD in CSR0 */
3492 /** Polling is then useless here and possibly expensive. */
3493 if (!CSR_DPOLL(pThis))
3494 pcnetPollTimer(pThis);
3495
3496 val = pcnetCSRReadU16(pThis, pThis->u32RAP);
3497 if (pThis->u32RAP == 0) // pcnetUpdateIrq() already called by pcnetCSRReadU16()
3498 goto skip_update_irq;
3499 break;
3500 case 0x04: /* RAP */
3501 val = pThis->u32RAP;
3502 goto skip_update_irq;
3503 case 0x08: /* RESET */
3504 pcnetSoftReset(pThis);
3505 val = 0;
3506 break;
3507 case 0x0c: /* BDP */
3508 val = pcnetBCRReadU16(pThis, pThis->u32RAP);
3509 break;
3510 }
3511 }
3512 else
3513 Log(("#%d pcnetIoportReadU32: addr=%#010x val=%#010x !BCR_DWIO !!\n", PCNET_INST_NR, addr, val));
3514 pcnetUpdateIrq(pThis);
3515
3516skip_update_irq:
3517#ifdef PCNET_DEBUG_IO
3518 Log2(("#%d pcnetIoportReadU32: addr=%#010x val=%#010x\n", PCNET_INST_NR, addr, val));
3519#endif
3520 return val;
3521}
3522
3523static void pcnetMMIOWriteU8(PCNetState *pThis, RTGCPHYS addr, uint32_t val)
3524{
3525#ifdef PCNET_DEBUG_IO
3526 Log2(("#%d pcnetMMIOWriteU8: addr=%#010x val=%#04x\n", PCNET_INST_NR, addr, val));
3527#endif
3528 if (!(addr & 0x10))
3529 pcnetAPROMWriteU8(pThis, addr, val);
3530}
3531
3532static uint32_t pcnetMMIOReadU8(PCNetState *pThis, RTGCPHYS addr)
3533{
3534 uint32_t val = ~0U;
3535 if (!(addr & 0x10))
3536 val = pcnetAPROMReadU8(pThis, addr);
3537#ifdef PCNET_DEBUG_IO
3538 Log2(("#%d pcnetMMIOReadU8: addr=%#010x val=%#04x\n", PCNET_INST_NR, addr, val & 0xff));
3539#endif
3540 return val;
3541}
3542
3543static void pcnetMMIOWriteU16(PCNetState *pThis, RTGCPHYS addr, uint32_t val)
3544{
3545#ifdef PCNET_DEBUG_IO
3546 Log2(("#%d pcnetMMIOWriteU16: addr=%#010x val=%#06x\n", PCNET_INST_NR, addr, val));
3547#endif
3548 if (addr & 0x10)
3549 pcnetIoportWriteU16(pThis, addr & 0x0f, val);
3550 else
3551 {
3552 pcnetAPROMWriteU8(pThis, addr, val );
3553 pcnetAPROMWriteU8(pThis, addr+1, val >> 8);
3554 }
3555}
3556
3557static uint32_t pcnetMMIOReadU16(PCNetState *pThis, RTGCPHYS addr)
3558{
3559 uint32_t val = ~0U;
3560 int rc;
3561
3562 if (addr & 0x10)
3563 val = pcnetIoportReadU16(pThis, addr & 0x0f, &rc);
3564 else
3565 {
3566 val = pcnetAPROMReadU8(pThis, addr+1);
3567 val <<= 8;
3568 val |= pcnetAPROMReadU8(pThis, addr);
3569 }
3570#ifdef PCNET_DEBUG_IO
3571 Log2(("#%d pcnetMMIOReadU16: addr=%#010x val = %#06x\n", PCNET_INST_NR, addr, val & 0xffff));
3572#endif
3573 return val;
3574}
3575
3576static void pcnetMMIOWriteU32(PCNetState *pThis, RTGCPHYS addr, uint32_t val)
3577{
3578#ifdef PCNET_DEBUG_IO
3579 Log2(("#%d pcnetMMIOWriteU32: addr=%#010x val=%#010x\n", PCNET_INST_NR, addr, val));
3580#endif
3581 if (addr & 0x10)
3582 pcnetIoportWriteU32(pThis, addr & 0x0f, val);
3583 else
3584 {
3585 pcnetAPROMWriteU8(pThis, addr, val );
3586 pcnetAPROMWriteU8(pThis, addr+1, val >> 8);
3587 pcnetAPROMWriteU8(pThis, addr+2, val >> 16);
3588 pcnetAPROMWriteU8(pThis, addr+3, val >> 24);
3589 }
3590}
3591
3592static uint32_t pcnetMMIOReadU32(PCNetState *pThis, RTGCPHYS addr)
3593{
3594 uint32_t val;
3595 int rc;
3596
3597 if (addr & 0x10)
3598 val = pcnetIoportReadU32(pThis, addr & 0x0f, &rc);
3599 else
3600 {
3601 val = pcnetAPROMReadU8(pThis, addr+3);
3602 val <<= 8;
3603 val |= pcnetAPROMReadU8(pThis, addr+2);
3604 val <<= 8;
3605 val |= pcnetAPROMReadU8(pThis, addr+1);
3606 val <<= 8;
3607 val |= pcnetAPROMReadU8(pThis, addr );
3608 }
3609#ifdef PCNET_DEBUG_IO
3610 Log2(("#%d pcnetMMIOReadU32: addr=%#010x val=%#010x\n", PCNET_INST_NR, addr, val));
3611#endif
3612 return val;
3613}
3614
3615
3616/**
3617 * Port I/O Handler for IN operations.
3618 *
3619 * @returns VBox status code.
3620 *
3621 * @param pDevIns The device instance.
3622 * @param pvUser User argument.
3623 * @param Port Port number used for the IN operation.
3624 * @param pu32 Where to store the result.
3625 * @param cb Number of bytes read.
3626 */
3627PDMBOTHCBDECL(int) pcnetIOPortAPromRead(PPDMDEVINS pDevIns, void *pvUser,
3628 RTIOPORT Port, uint32_t *pu32, unsigned cb)
3629{
3630 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
3631 int rc;
3632
3633 STAM_PROFILE_ADV_START(&pThis->StatAPROMRead, a);
3634 rc = PDMCritSectEnter(&pThis->CritSect, VINF_IOM_R3_IOPORT_WRITE);
3635 if (rc == VINF_SUCCESS)
3636 {
3637
3638 /* FreeBSD is accessing in dwords. */
3639 if (cb == 1)
3640 *pu32 = pcnetAPROMReadU8(pThis, Port);
3641 else if (cb == 2 && !BCR_DWIO(pThis))
3642 *pu32 = pcnetAPROMReadU8(pThis, Port)
3643 | (pcnetAPROMReadU8(pThis, Port + 1) << 8);
3644 else if (cb == 4 && BCR_DWIO(pThis))
3645 *pu32 = pcnetAPROMReadU8(pThis, Port)
3646 | (pcnetAPROMReadU8(pThis, Port + 1) << 8)
3647 | (pcnetAPROMReadU8(pThis, Port + 2) << 16)
3648 | (pcnetAPROMReadU8(pThis, Port + 3) << 24);
3649 else
3650 {
3651 Log(("#%d pcnetIOPortAPromRead: Port=%RTiop cb=%d BCR_DWIO !!\n", PCNET_INST_NR, Port, cb));
3652 rc = VERR_IOM_IOPORT_UNUSED;
3653 }
3654 PDMCritSectLeave(&pThis->CritSect);
3655 }
3656 STAM_PROFILE_ADV_STOP(&pThis->StatAPROMRead, a);
3657 LogFlow(("#%d pcnetIOPortAPromRead: Port=%RTiop *pu32=%#RX32 cb=%d rc=%Rrc\n", PCNET_INST_NR, Port, *pu32, cb, rc));
3658 return rc;
3659}
3660
3661
3662/**
3663 * Port I/O Handler for OUT operations.
3664 *
3665 * @returns VBox status code.
3666 *
3667 * @param pDevIns The device instance.
3668 * @param pvUser User argument.
3669 * @param Port Port number used for the IN operation.
3670 * @param u32 The value to output.
3671 * @param cb The value size in bytes.
3672 */
3673PDMBOTHCBDECL(int) pcnetIOPortAPromWrite(PPDMDEVINS pDevIns, void *pvUser,
3674 RTIOPORT Port, uint32_t u32, unsigned cb)
3675{
3676 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
3677 int rc;
3678
3679 if (cb == 1)
3680 {
3681 STAM_PROFILE_ADV_START(&pThis->StatAPROMWrite, a);
3682 rc = PDMCritSectEnter(&pThis->CritSect, VINF_IOM_R3_IOPORT_WRITE);
3683 if (RT_LIKELY(rc == VINF_SUCCESS))
3684 {
3685 pcnetAPROMWriteU8(pThis, Port, u32);
3686 PDMCritSectLeave(&pThis->CritSect);
3687 }
3688 STAM_PROFILE_ADV_STOP(&pThis->StatAPROMWrite, a);
3689 }
3690 else
3691 {
3692 AssertMsgFailed(("Port=%#x cb=%d u32=%#x\n", Port, cb, u32));
3693 rc = VINF_SUCCESS;
3694 }
3695 LogFlow(("#%d pcnetIOPortAPromWrite: Port=%RTiop u32=%#RX32 cb=%d rc=%Rrc\n", PCNET_INST_NR, Port, u32, cb, rc));
3696#ifdef LOG_ENABLED
3697 if (rc == VINF_IOM_R3_IOPORT_WRITE)
3698 LogFlow(("#%d => HC\n", PCNET_INST_NR));
3699#endif
3700 return rc;
3701}
3702
3703
3704/**
3705 * Port I/O Handler for IN operations.
3706 *
3707 * @returns VBox status code.
3708 *
3709 * @param pDevIns The device instance.
3710 * @param pvUser User argument.
3711 * @param Port Port number used for the IN operation.
3712 * @param pu32 Where to store the result.
3713 * @param cb Number of bytes read.
3714 */
3715PDMBOTHCBDECL(int) pcnetIOPortRead(PPDMDEVINS pDevIns, void *pvUser,
3716 RTIOPORT Port, uint32_t *pu32, unsigned cb)
3717{
3718 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
3719 int rc = VINF_SUCCESS;
3720
3721 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIORead), a);
3722 rc = PDMCritSectEnter(&pThis->CritSect, VINF_IOM_R3_IOPORT_READ);
3723 if (RT_LIKELY(rc == VINF_SUCCESS))
3724 {
3725 switch (cb)
3726 {
3727 case 1: *pu32 = pcnetIoportReadU8(pThis, Port, &rc); break;
3728 case 2: *pu32 = pcnetIoportReadU16(pThis, Port, &rc); break;
3729 case 4: *pu32 = pcnetIoportReadU32(pThis, Port, &rc); break;
3730 default:
3731 rc = PDMDevHlpDBGFStop(pThis->CTX_SUFF(pDevIns), RT_SRC_POS,
3732 "pcnetIOPortRead: unsupported op size: offset=%#10x cb=%u\n",
3733 Port, cb);
3734 }
3735 PDMCritSectLeave(&pThis->CritSect);
3736 }
3737 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIORead), a);
3738 Log2(("#%d pcnetIOPortRead: Port=%RTiop *pu32=%#RX32 cb=%d rc=%Rrc\n", PCNET_INST_NR, Port, *pu32, cb, rc));
3739#ifdef LOG_ENABLED
3740 if (rc == VINF_IOM_R3_IOPORT_READ)
3741 LogFlow(("#%d pcnetIOPortRead/critsect failed in GC => HC\n", PCNET_INST_NR));
3742#endif
3743 return rc;
3744}
3745
3746
3747/**
3748 * Port I/O Handler for OUT operations.
3749 *
3750 * @returns VBox status code.
3751 *
3752 * @param pDevIns The device instance.
3753 * @param pvUser User argument.
3754 * @param Port Port number used for the IN operation.
3755 * @param u32 The value to output.
3756 * @param cb The value size in bytes.
3757 */
3758PDMBOTHCBDECL(int) pcnetIOPortWrite(PPDMDEVINS pDevIns, void *pvUser,
3759 RTIOPORT Port, uint32_t u32, unsigned cb)
3760{
3761 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
3762 int rc = VINF_SUCCESS;
3763
3764 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatIOWrite), a);
3765 rc = PDMCritSectEnter(&pThis->CritSect, VINF_IOM_R3_IOPORT_WRITE);
3766 if (RT_LIKELY(rc == VINF_SUCCESS))
3767 {
3768 switch (cb)
3769 {
3770 case 1: rc = pcnetIoportWriteU8(pThis, Port, u32); break;
3771 case 2: rc = pcnetIoportWriteU16(pThis, Port, u32); break;
3772 case 4: rc = pcnetIoportWriteU32(pThis, Port, u32); break;
3773 default:
3774 rc = PDMDevHlpDBGFStop(pThis->CTX_SUFF(pDevIns), RT_SRC_POS,
3775 "pcnetIOPortWrite: unsupported op size: offset=%#10x cb=%u\n",
3776 Port, cb);
3777 }
3778 PDMCritSectLeave(&pThis->CritSect);
3779 }
3780 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatIOWrite), a);
3781 Log2(("#%d pcnetIOPortWrite: Port=%RTiop u32=%#RX32 cb=%d rc=%Rrc\n", PCNET_INST_NR, Port, u32, cb, rc));
3782#ifdef LOG_ENABLED
3783 if (rc == VINF_IOM_R3_IOPORT_WRITE)
3784 LogFlow(("#%d pcnetIOPortWrite/critsect failed in GC => HC\n", PCNET_INST_NR));
3785#endif
3786 return rc;
3787}
3788
3789
3790/**
3791 * Memory mapped I/O Handler for read operations.
3792 *
3793 * @returns VBox status code.
3794 *
3795 * @param pDevIns The device instance.
3796 * @param pvUser User argument.
3797 * @param GCPhysAddr Physical address (in GC) where the read starts.
3798 * @param pv Where to store the result.
3799 * @param cb Number of bytes read.
3800 */
3801PDMBOTHCBDECL(int) pcnetMMIORead(PPDMDEVINS pDevIns, void *pvUser,
3802 RTGCPHYS GCPhysAddr, void *pv, unsigned cb)
3803{
3804 PCNetState *pThis = (PCNetState *)pvUser;
3805 int rc = VINF_SUCCESS;
3806
3807 /*
3808 * We have to check the range, because we're page aligning the MMIO stuff presently.
3809 */
3810 if (GCPhysAddr - pThis->MMIOBase < PCNET_PNPMMIO_SIZE)
3811 {
3812 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatMMIORead), a);
3813 rc = PDMCritSectEnter(&pThis->CritSect, VINF_IOM_R3_MMIO_READ);
3814 if (RT_LIKELY(rc == VINF_SUCCESS))
3815 {
3816 switch (cb)
3817 {
3818 case 1: *(uint8_t *)pv = pcnetMMIOReadU8 (pThis, GCPhysAddr); break;
3819 case 2: *(uint16_t *)pv = pcnetMMIOReadU16(pThis, GCPhysAddr); break;
3820 case 4: *(uint32_t *)pv = pcnetMMIOReadU32(pThis, GCPhysAddr); break;
3821 default:
3822 rc = PDMDevHlpDBGFStop(pThis->CTX_SUFF(pDevIns), RT_SRC_POS,
3823 "pcnetMMIORead: unsupported op size: address=%RGp cb=%u\n",
3824 GCPhysAddr, cb);
3825 }
3826 PDMCritSectLeave(&pThis->CritSect);
3827 }
3828 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatMMIORead), a);
3829 }
3830 else
3831 memset(pv, 0, cb);
3832
3833 LogFlow(("#%d pcnetMMIORead: pvUser=%p:{%.*Rhxs} cb=%d GCPhysAddr=%RGp rc=%Rrc\n",
3834 PCNET_INST_NR, pv, cb, pv, cb, GCPhysAddr, rc));
3835#ifdef LOG_ENABLED
3836 if (rc == VINF_IOM_R3_MMIO_READ)
3837 LogFlow(("#%d => HC\n", PCNET_INST_NR));
3838#endif
3839 return rc;
3840}
3841
3842
3843/**
3844 * Port I/O Handler for write operations.
3845 *
3846 * @returns VBox status code.
3847 *
3848 * @param pDevIns The device instance.
3849 * @param pvUser User argument.
3850 * @param GCPhysAddr Physical address (in GC) where the read starts.
3851 * @param pv Where to fetch the result.
3852 * @param cb Number of bytes to write.
3853 */
3854PDMBOTHCBDECL(int) pcnetMMIOWrite(PPDMDEVINS pDevIns, void *pvUser,
3855 RTGCPHYS GCPhysAddr, void const *pv, unsigned cb)
3856{
3857 PCNetState *pThis = (PCNetState *)pvUser;
3858 int rc = VINF_SUCCESS;
3859
3860 /*
3861 * We have to check the range, because we're page aligning the MMIO stuff presently.
3862 */
3863 if (GCPhysAddr - pThis->MMIOBase < PCNET_PNPMMIO_SIZE)
3864 {
3865 STAM_PROFILE_ADV_START(&pThis->CTX_SUFF_Z(StatMMIOWrite), a);
3866 rc = PDMCritSectEnter(&pThis->CritSect, VINF_IOM_R3_MMIO_WRITE);
3867 if (RT_LIKELY(rc == VINF_SUCCESS))
3868 {
3869 switch (cb)
3870 {
3871 case 1: pcnetMMIOWriteU8 (pThis, GCPhysAddr, *(uint8_t *)pv); break;
3872 case 2: pcnetMMIOWriteU16(pThis, GCPhysAddr, *(uint16_t *)pv); break;
3873 case 4: pcnetMMIOWriteU32(pThis, GCPhysAddr, *(uint32_t *)pv); break;
3874 default:
3875 rc = PDMDevHlpDBGFStop(pThis->CTX_SUFF(pDevIns), RT_SRC_POS,
3876 "pcnetMMIOWrite: unsupported op size: address=%RGp cb=%u\n",
3877 GCPhysAddr, cb);
3878 }
3879 PDMCritSectLeave(&pThis->CritSect);
3880 }
3881 // else rc == VINF_IOM_R3_MMIO_WRITE => handle in ring3
3882
3883 STAM_PROFILE_ADV_STOP(&pThis->CTX_SUFF_Z(StatMMIOWrite), a);
3884 }
3885 LogFlow(("#%d pcnetMMIOWrite: pvUser=%p:{%.*Rhxs} cb=%d GCPhysAddr=%RGp rc=%Rrc\n",
3886 PCNET_INST_NR, pv, cb, pv, cb, GCPhysAddr, rc));
3887#ifdef LOG_ENABLED
3888 if (rc == VINF_IOM_R3_MMIO_WRITE)
3889 LogFlow(("#%d => HC\n", PCNET_INST_NR));
3890#endif
3891 return rc;
3892}
3893
3894
3895#ifdef IN_RING3
3896/**
3897 * Device timer callback function.
3898 *
3899 * @param pDevIns Device instance of the device which registered the timer.
3900 * @param pTimer The timer handle.
3901 * @thread EMT
3902 */
3903static DECLCALLBACK(void) pcnetTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
3904{
3905 PCNetState *pThis = (PCNetState *)pvUser;
3906 Assert(PDMCritSectIsOwner(&pThis->CritSect));
3907
3908 STAM_PROFILE_ADV_START(&pThis->StatTimer, a);
3909 pcnetPollTimer(pThis);
3910 STAM_PROFILE_ADV_STOP(&pThis->StatTimer, a);
3911}
3912
3913
3914/**
3915 * Software interrupt timer callback function.
3916 *
3917 * @param pDevIns Device instance of the device which registered the timer.
3918 * @param pTimer The timer handle.
3919 * @thread EMT
3920 */
3921static DECLCALLBACK(void) pcnetTimerSoftInt(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
3922{
3923 PCNetState *pThis = (PCNetState *)pvUser;
3924 Assert(PDMCritSectIsOwner(&pThis->CritSect));
3925
3926 pThis->aCSR[7] |= 0x0800; /* STINT */
3927 pcnetUpdateIrq(pThis);
3928 TMTimerSetNano(pThis->CTX_SUFF(pTimerSoftInt), 12800U * (pThis->aBCR[BCR_STVAL] & 0xffff));
3929}
3930
3931
3932/**
3933 * Restore timer callback.
3934 *
3935 * This is only called when we restore a saved state and temporarily
3936 * disconnected the network link to inform the guest that network connections
3937 * should be considered lost.
3938 *
3939 * @param pDevIns Device instance of the device which registered the timer.
3940 * @param pTimer The timer handle.
3941 */
3942static DECLCALLBACK(void) pcnetTimerRestore(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
3943{
3944 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
3945 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
3946 AssertReleaseRC(rc);
3947
3948 rc = VERR_GENERAL_FAILURE;
3949 if (pThis->cLinkDownReported <= PCNET_MAX_LINKDOWN_REPORTED)
3950 rc = TMTimerSetMillies(pThis->pTimerRestore, 1500);
3951 if (RT_FAILURE(rc))
3952 {
3953 pThis->fLinkTempDown = false;
3954 if (pThis->fLinkUp)
3955 {
3956 LogRel(("PCNet#%d: The link is back up again after the restore.\n",
3957 pDevIns->iInstance));
3958 Log(("#%d pcnetTimerRestore: Clearing ERR and CERR after load. cLinkDownReported=%d\n",
3959 pDevIns->iInstance, pThis->cLinkDownReported));
3960 pThis->aCSR[0] &= ~(RT_BIT(15) | RT_BIT(13)); /* ERR | CERR - probably not 100% correct either... */
3961 pThis->Led.Actual.s.fError = 0;
3962 }
3963 }
3964 else
3965 Log(("#%d pcnetTimerRestore: cLinkDownReported=%d, wait another 1500ms...\n",
3966 pDevIns->iInstance, pThis->cLinkDownReported));
3967
3968 PDMCritSectLeave(&pThis->CritSect);
3969}
3970
3971/**
3972 * Callback function for mapping an PCI I/O region.
3973 *
3974 * @return VBox status code.
3975 * @param pPciDev Pointer to PCI device. Use pPciDev->pDevIns to get the device instance.
3976 * @param iRegion The region number.
3977 * @param GCPhysAddress Physical address of the region. If iType is PCI_ADDRESS_SPACE_IO, this is an
3978 * I/O port, else it's a physical address.
3979 * This address is *NOT* relative to pci_mem_base like earlier!
3980 * @param cb Region size.
3981 * @param enmType One of the PCI_ADDRESS_SPACE_* values.
3982 */
3983static DECLCALLBACK(int) pcnetIOPortMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
3984 RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
3985{
3986 int rc;
3987 PPDMDEVINS pDevIns = pPciDev->pDevIns;
3988 RTIOPORT Port = (RTIOPORT)GCPhysAddress;
3989 PCNetState *pThis = PCIDEV_2_PCNETSTATE(pPciDev);
3990
3991 Assert(enmType == PCI_ADDRESS_SPACE_IO);
3992 Assert(cb >= 0x20);
3993
3994 rc = PDMDevHlpIOPortRegister(pDevIns, Port, 0x10, 0, pcnetIOPortAPromWrite,
3995 pcnetIOPortAPromRead, NULL, NULL, "PCNet ARPOM");
3996 if (RT_FAILURE(rc))
3997 return rc;
3998 rc = PDMDevHlpIOPortRegister(pDevIns, Port + 0x10, 0x10, 0, pcnetIOPortWrite,
3999 pcnetIOPortRead, NULL, NULL, "PCNet");
4000 if (RT_FAILURE(rc))
4001 return rc;
4002
4003 if (pThis->fGCEnabled)
4004 {
4005 rc = PDMDevHlpIOPortRegisterRC(pDevIns, Port, 0x10, 0, "pcnetIOPortAPromWrite",
4006 "pcnetIOPortAPromRead", NULL, NULL, "PCNet aprom");
4007 if (RT_FAILURE(rc))
4008 return rc;
4009 rc = PDMDevHlpIOPortRegisterRC(pDevIns, Port + 0x10, 0x10, 0, "pcnetIOPortWrite",
4010 "pcnetIOPortRead", NULL, NULL, "PCNet");
4011 if (RT_FAILURE(rc))
4012 return rc;
4013 }
4014 if (pThis->fR0Enabled)
4015 {
4016 rc = PDMDevHlpIOPortRegisterR0(pDevIns, Port, 0x10, 0, "pcnetIOPortAPromWrite",
4017 "pcnetIOPortAPromRead", NULL, NULL, "PCNet aprom");
4018 if (RT_FAILURE(rc))
4019 return rc;
4020 rc = PDMDevHlpIOPortRegisterR0(pDevIns, Port + 0x10, 0x10, 0, "pcnetIOPortWrite",
4021 "pcnetIOPortRead", NULL, NULL, "PCNet");
4022 if (RT_FAILURE(rc))
4023 return rc;
4024 }
4025
4026 pThis->IOPortBase = Port;
4027 return VINF_SUCCESS;
4028}
4029
4030
4031/**
4032 * Callback function for mapping the MMIO region.
4033 *
4034 * @return VBox status code.
4035 * @param pPciDev Pointer to PCI device. Use pPciDev->pDevIns to get the device instance.
4036 * @param iRegion The region number.
4037 * @param GCPhysAddress Physical address of the region. If iType is PCI_ADDRESS_SPACE_IO, this is an
4038 * I/O port, else it's a physical address.
4039 * This address is *NOT* relative to pci_mem_base like earlier!
4040 * @param cb Region size.
4041 * @param enmType One of the PCI_ADDRESS_SPACE_* values.
4042 */
4043static DECLCALLBACK(int) pcnetMMIOMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
4044 RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
4045{
4046 PCNetState *pThis = PCIDEV_2_PCNETSTATE(pPciDev);
4047 int rc;
4048
4049 Assert(enmType == PCI_ADDRESS_SPACE_MEM);
4050 Assert(cb >= PCNET_PNPMMIO_SIZE);
4051
4052 /* We use the assigned size here, because we currently only support page aligned MMIO ranges. */
4053 rc = PDMDevHlpMMIORegister(pPciDev->pDevIns, GCPhysAddress, cb, pThis,
4054 IOMMMIO_FLAGS_READ_PASSTHRU | IOMMMIO_FLAGS_WRITE_PASSTHRU,
4055 pcnetMMIOWrite, pcnetMMIORead, "PCNet");
4056 if (RT_FAILURE(rc))
4057 return rc;
4058 pThis->MMIOBase = GCPhysAddress;
4059 return rc;
4060}
4061
4062
4063/**
4064 * Callback function for mapping the MMIO region.
4065 *
4066 * @return VBox status code.
4067 * @param pPciDev Pointer to PCI device. Use pPciDev->pDevIns to get the device instance.
4068 * @param iRegion The region number.
4069 * @param GCPhysAddress Physical address of the region. If iType is PCI_ADDRESS_SPACE_IO, this is an
4070 * I/O port, else it's a physical address.
4071 * This address is *NOT* relative to pci_mem_base like earlier!
4072 * @param cb Region size.
4073 * @param enmType One of the PCI_ADDRESS_SPACE_* values.
4074 */
4075static DECLCALLBACK(int) pcnetMMIOSharedMap(PPCIDEVICE pPciDev, /*unsigned*/ int iRegion,
4076 RTGCPHYS GCPhysAddress, uint32_t cb, PCIADDRESSSPACE enmType)
4077{
4078 if (GCPhysAddress != NIL_RTGCPHYS)
4079 return PDMDevHlpMMIO2Map(pPciDev->pDevIns, iRegion, GCPhysAddress);
4080
4081 /* nothing to clean up */
4082 return VINF_SUCCESS;
4083}
4084
4085
4086/**
4087 * PCNET status info callback.
4088 *
4089 * @param pDevIns The device instance.
4090 * @param pHlp The output helpers.
4091 * @param pszArgs The arguments.
4092 */
4093static DECLCALLBACK(void) pcnetInfo(PPDMDEVINS pDevIns, PCDBGFINFOHLP pHlp, const char *pszArgs)
4094{
4095 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4096 bool fRcvRing = false;
4097 bool fXmtRing = false;
4098
4099 /*
4100 * Parse args.
4101 */
4102 if (pszArgs)
4103 {
4104 fRcvRing = strstr(pszArgs, "verbose") || strstr(pszArgs, "rcv");
4105 fXmtRing = strstr(pszArgs, "verbose") || strstr(pszArgs, "xmt");
4106 }
4107
4108 /*
4109 * Show info.
4110 */
4111 pHlp->pfnPrintf(pHlp,
4112 "pcnet #%d: port=%RTiop mmio=%RX32 mac-cfg=%RTmac %s\n",
4113 pDevIns->iInstance,
4114 pThis->IOPortBase, pThis->MMIOBase, &pThis->MacConfigured,
4115 pThis->fAm79C973 ? "Am79C973" : "Am79C970A", pThis->fGCEnabled ? " GC" : "", pThis->fR0Enabled ? " R0" : "");
4116
4117 PDMCritSectEnter(&pThis->CritSect, VERR_INTERNAL_ERROR); /* Take it here so we know why we're hanging... */
4118
4119 pHlp->pfnPrintf(pHlp,
4120 "CSR0=%#06x:\n",
4121 pThis->aCSR[0]);
4122
4123 pHlp->pfnPrintf(pHlp,
4124 "CSR1=%#06x:\n",
4125 pThis->aCSR[1]);
4126
4127 pHlp->pfnPrintf(pHlp,
4128 "CSR2=%#06x:\n",
4129 pThis->aCSR[2]);
4130
4131 pHlp->pfnPrintf(pHlp,
4132 "CSR3=%#06x: BSWP=%d EMBA=%d DXMT2PD=%d LAPPEN=%d DXSUFLO=%d IDONM=%d TINTM=%d RINTM=%d MERRM=%d MISSM=%d BABLM=%d\n",
4133 pThis->aCSR[3],
4134 !!(pThis->aCSR[3] & RT_BIT(2)), !!(pThis->aCSR[3] & RT_BIT(3)), !!(pThis->aCSR[3] & RT_BIT(4)), CSR_LAPPEN(pThis),
4135 CSR_DXSUFLO(pThis), !!(pThis->aCSR[3] & RT_BIT(8)), !!(pThis->aCSR[3] & RT_BIT(9)), !!(pThis->aCSR[3] & RT_BIT(10)),
4136 !!(pThis->aCSR[3] & RT_BIT(11)), !!(pThis->aCSR[3] & RT_BIT(12)), !!(pThis->aCSR[3] & RT_BIT(14)));
4137
4138 pHlp->pfnPrintf(pHlp,
4139 "CSR4=%#06x: JABM=%d JAB=%d TXSTRM=%d TXSTRT=%d RCVCOOM=%d RCVCCO=%d UINT=%d UINTCMD=%d\n"
4140 " MFCOM=%d MFCO=%d ASTRP_RCV=%d APAD_XMT=%d DPOLL=%d TIMER=%d EMAPLUS=%d EN124=%d\n",
4141 pThis->aCSR[4],
4142 !!(pThis->aCSR[4] & RT_BIT( 0)), !!(pThis->aCSR[4] & RT_BIT( 1)), !!(pThis->aCSR[4] & RT_BIT( 2)), !!(pThis->aCSR[4] & RT_BIT( 3)),
4143 !!(pThis->aCSR[4] & RT_BIT( 4)), !!(pThis->aCSR[4] & RT_BIT( 5)), !!(pThis->aCSR[4] & RT_BIT( 6)), !!(pThis->aCSR[4] & RT_BIT( 7)),
4144 !!(pThis->aCSR[4] & RT_BIT( 8)), !!(pThis->aCSR[4] & RT_BIT( 9)), !!(pThis->aCSR[4] & RT_BIT(10)), !!(pThis->aCSR[4] & RT_BIT(11)),
4145 !!(pThis->aCSR[4] & RT_BIT(12)), !!(pThis->aCSR[4] & RT_BIT(13)), !!(pThis->aCSR[4] & RT_BIT(14)), !!(pThis->aCSR[4] & RT_BIT(15)));
4146
4147 pHlp->pfnPrintf(pHlp,
4148 "CSR5=%#06x:\n",
4149 pThis->aCSR[5]);
4150
4151 pHlp->pfnPrintf(pHlp,
4152 "CSR6=%#06x: RLEN=%#x* TLEN=%#x* [* encoded]\n",
4153 pThis->aCSR[6],
4154 (pThis->aCSR[6] >> 8) & 0xf, (pThis->aCSR[6] >> 12) & 0xf);
4155
4156 pHlp->pfnPrintf(pHlp,
4157 "CSR8..11=%#06x,%#06x,%#06x,%#06x: LADRF=%#018llx\n",
4158 pThis->aCSR[8], pThis->aCSR[9], pThis->aCSR[10], pThis->aCSR[11],
4159 (uint64_t)(pThis->aCSR[ 8] & 0xffff)
4160 | (uint64_t)(pThis->aCSR[ 9] & 0xffff) << 16
4161 | (uint64_t)(pThis->aCSR[10] & 0xffff) << 32
4162 | (uint64_t)(pThis->aCSR[11] & 0xffff) << 48);
4163
4164 pHlp->pfnPrintf(pHlp,
4165 "CSR12..14=%#06x,%#06x,%#06x: PADR=%02x:%02x:%02x:%02x:%02x:%02x (Current MAC Address)\n",
4166 pThis->aCSR[12], pThis->aCSR[13], pThis->aCSR[14],
4167 pThis->aCSR[12] & 0xff,
4168 (pThis->aCSR[12] >> 8) & 0xff,
4169 pThis->aCSR[13] & 0xff,
4170 (pThis->aCSR[13] >> 8) & 0xff,
4171 pThis->aCSR[14] & 0xff,
4172 (pThis->aCSR[14] >> 8) & 0xff);
4173
4174 pHlp->pfnPrintf(pHlp,
4175 "CSR15=%#06x: DXR=%d DTX=%d LOOP=%d DXMTFCS=%d FCOLL=%d DRTY=%d INTL=%d PORTSEL=%d LTR=%d\n"
4176 " MENDECL=%d DAPC=%d DLNKTST=%d DRCVPV=%d DRCVBC=%d PROM=%d\n",
4177 pThis->aCSR[15],
4178 !!(pThis->aCSR[15] & RT_BIT( 0)), !!(pThis->aCSR[15] & RT_BIT( 1)), !!(pThis->aCSR[15] & RT_BIT( 2)), !!(pThis->aCSR[15] & RT_BIT( 3)),
4179 !!(pThis->aCSR[15] & RT_BIT( 4)), !!(pThis->aCSR[15] & RT_BIT( 5)), !!(pThis->aCSR[15] & RT_BIT( 6)), (pThis->aCSR[15] >> 7) & 3,
4180 !!(pThis->aCSR[15] & RT_BIT( 9)), !!(pThis->aCSR[15] & RT_BIT(10)), !!(pThis->aCSR[15] & RT_BIT(11)),
4181 !!(pThis->aCSR[15] & RT_BIT(12)), !!(pThis->aCSR[15] & RT_BIT(13)), !!(pThis->aCSR[15] & RT_BIT(14)), !!(pThis->aCSR[15] & RT_BIT(15)));
4182
4183 pHlp->pfnPrintf(pHlp,
4184 "CSR46=%#06x: POLL=%#06x (Poll Time Counter)\n",
4185 pThis->aCSR[46], pThis->aCSR[46] & 0xffff);
4186
4187 pHlp->pfnPrintf(pHlp,
4188 "CSR47=%#06x: POLLINT=%#06x (Poll Time Interval)\n",
4189 pThis->aCSR[47], pThis->aCSR[47] & 0xffff);
4190
4191 pHlp->pfnPrintf(pHlp,
4192 "CSR58=%#06x: SWSTYLE=%d %s SSIZE32=%d CSRPCNET=%d APERRENT=%d\n",
4193 pThis->aCSR[58],
4194 pThis->aCSR[58] & 0x7f,
4195 (pThis->aCSR[58] & 0x7f) == 0 ? "C-LANCE / PCnet-ISA"
4196 : (pThis->aCSR[58] & 0x7f) == 1 ? "ILACC"
4197 : (pThis->aCSR[58] & 0x7f) == 2 ? "PCNet-PCI II"
4198 : (pThis->aCSR[58] & 0x7f) == 3 ? "PCNet-PCI II controller"
4199 : "!!reserved!!",
4200 !!(pThis->aCSR[58] & RT_BIT(8)), !!(pThis->aCSR[58] & RT_BIT(9)), !!(pThis->aCSR[58] & RT_BIT(10)));
4201
4202 pHlp->pfnPrintf(pHlp,
4203 "CSR112=%04RX32: MFC=%04x (Missed receive Frame Count)\n",
4204 pThis->aCSR[112], pThis->aCSR[112] & 0xffff);
4205
4206 pHlp->pfnPrintf(pHlp,
4207 "CSR122=%04RX32: RCVALGN=%04x (Receive Frame Align)\n",
4208 pThis->aCSR[122], !!(pThis->aCSR[122] & RT_BIT(0)));
4209
4210 pHlp->pfnPrintf(pHlp,
4211 "CSR124=%04RX32: RPA=%04x (Runt Packet Accept)\n",
4212 pThis->aCSR[122], !!(pThis->aCSR[122] & RT_BIT(3)));
4213
4214
4215 /*
4216 * Dump the receive ring.
4217 */
4218 pHlp->pfnPrintf(pHlp,
4219 "RCVRL=%04x RCVRC=%04x GCRDRA=%RX32 \n"
4220 "CRDA=%08RX32 CRBA=%08RX32 CRBC=%03x CRST=%04x\n"
4221 "NRDA=%08RX32 NRBA=%08RX32 NRBC=%03x NRST=%04x\n"
4222 "NNRDA=%08RX32\n"
4223 ,
4224 CSR_RCVRL(pThis), CSR_RCVRC(pThis), pThis->GCRDRA,
4225 CSR_CRDA(pThis), CSR_CRBA(pThis), CSR_CRBC(pThis), CSR_CRST(pThis),
4226 CSR_NRDA(pThis), CSR_NRBA(pThis), CSR_NRBC(pThis), CSR_NRST(pThis),
4227 CSR_NNRD(pThis));
4228 if (fRcvRing)
4229 {
4230 const unsigned cb = 1 << pThis->iLog2DescSize;
4231 RTGCPHYS32 GCPhys = pThis->GCRDRA;
4232 unsigned i = CSR_RCVRL(pThis);
4233 while (i-- > 0)
4234 {
4235 RMD rmd;
4236 pcnetRmdLoad(pThis, &rmd, PHYSADDR(pThis, GCPhys), false);
4237 pHlp->pfnPrintf(pHlp,
4238 "%04x %RX32:%c%c RBADR=%08RX32 BCNT=%03x MCNT=%03x "
4239 "OWN=%d ERR=%d FRAM=%d OFLO=%d CRC=%d BUFF=%d STP=%d ENP=%d BPE=%d "
4240 "PAM=%d LAFM=%d BAM=%d RCC=%02x RPC=%02x ONES=%#x ZEROS=%d\n",
4241 i, GCPhys, i + 1 == CSR_RCVRC(pThis) ? '*' : ' ', GCPhys == CSR_CRDA(pThis) ? '*' : ' ',
4242 rmd.rmd0.rbadr, 4096 - rmd.rmd1.bcnt, rmd.rmd2.mcnt,
4243 rmd.rmd1.own, rmd.rmd1.err, rmd.rmd1.fram, rmd.rmd1.oflo, rmd.rmd1.crc, rmd.rmd1.buff,
4244 rmd.rmd1.stp, rmd.rmd1.enp, rmd.rmd1.bpe,
4245 rmd.rmd1.pam, rmd.rmd1.lafm, rmd.rmd1.bam, rmd.rmd2.rcc, rmd.rmd2.rpc,
4246 rmd.rmd1.ones, rmd.rmd2.zeros);
4247
4248 GCPhys += cb;
4249 }
4250 }
4251
4252 /*
4253 * Dump the transmit ring.
4254 */
4255 pHlp->pfnPrintf(pHlp,
4256 "XMTRL=%04x XMTRC=%04x GCTDRA=%08RX32 BADX=%08RX32\n"
4257 "PXDA=%08RX32 PXBC=%03x PXST=%04x\n"
4258 "CXDA=%08RX32 CXBA=%08RX32 CXBC=%03x CXST=%04x\n"
4259 "NXDA=%08RX32 NXBA=%08RX32 NXBC=%03x NXST=%04x\n"
4260 "NNXDA=%08RX32\n"
4261 ,
4262 CSR_XMTRL(pThis), CSR_XMTRC(pThis),
4263 pThis->GCTDRA, CSR_BADX(pThis),
4264 CSR_PXDA(pThis), CSR_PXBC(pThis), CSR_PXST(pThis),
4265 CSR_CXDA(pThis), CSR_CXBA(pThis), CSR_CXBC(pThis), CSR_CXST(pThis),
4266 CSR_NXDA(pThis), CSR_NXBA(pThis), CSR_NXBC(pThis), CSR_NXST(pThis),
4267 CSR_NNXD(pThis));
4268 if (fXmtRing)
4269 {
4270 const unsigned cb = 1 << pThis->iLog2DescSize;
4271 RTGCPHYS32 GCPhys = pThis->GCTDRA;
4272 unsigned i = CSR_XMTRL(pThis);
4273 while (i-- > 0)
4274 {
4275 TMD tmd;
4276 pcnetTmdLoad(pThis, &tmd, PHYSADDR(pThis, GCPhys), false);
4277 pHlp->pfnPrintf(pHlp,
4278 "%04x %RX32:%c%c TBADR=%08RX32 BCNT=%03x OWN=%d "
4279 "ERR=%d NOFCS=%d LTINT=%d ONE=%d DEF=%d STP=%d ENP=%d BPE=%d "
4280 "BUFF=%d UFLO=%d EXDEF=%d LCOL=%d LCAR=%d RTRY=%d TDR=%03x TRC=%#x ONES=%#x\n"
4281 ,
4282 i, GCPhys, i + 1 == CSR_XMTRC(pThis) ? '*' : ' ', GCPhys == CSR_CXDA(pThis) ? '*' : ' ',
4283 tmd.tmd0.tbadr, 4096 - tmd.tmd1.bcnt,
4284 tmd.tmd2.tdr,
4285 tmd.tmd2.trc,
4286 tmd.tmd1.own,
4287 tmd.tmd1.err,
4288 tmd.tmd1.nofcs,
4289 tmd.tmd1.ltint,
4290 tmd.tmd1.one,
4291 tmd.tmd1.def,
4292 tmd.tmd1.stp,
4293 tmd.tmd1.enp,
4294 tmd.tmd1.bpe,
4295 tmd.tmd2.buff,
4296 tmd.tmd2.uflo,
4297 tmd.tmd2.exdef,
4298 tmd.tmd2.lcol,
4299 tmd.tmd2.lcar,
4300 tmd.tmd2.rtry,
4301 tmd.tmd2.tdr,
4302 tmd.tmd2.trc,
4303 tmd.tmd1.ones);
4304
4305 GCPhys += cb;
4306 }
4307 }
4308
4309 PDMCritSectLeave(&pThis->CritSect);
4310}
4311
4312
4313/**
4314 * Takes down the link temporarily if it's current status is up.
4315 *
4316 * This is used during restore and when replumbing the network link.
4317 *
4318 * The temporary link outage is supposed to indicate to the OS that all network
4319 * connections have been lost and that it for instance is appropriate to
4320 * renegotiate any DHCP lease.
4321 *
4322 * @param pThis The PCNet instance data.
4323 */
4324static void pcnetTempLinkDown(PCNetState *pThis)
4325{
4326 if (pThis->fLinkUp)
4327 {
4328 pThis->fLinkTempDown = true;
4329 pThis->cLinkDownReported = 0;
4330 pThis->aCSR[0] |= RT_BIT(15) | RT_BIT(13); /* ERR | CERR (this is probably wrong) */
4331 pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1;
4332 int rc = TMTimerSetMillies(pThis->pTimerRestore, 5000);
4333 AssertRC(rc);
4334 }
4335}
4336
4337
4338/**
4339 * Saves the configuration.
4340 *
4341 * @param pThis The PCNet instance data.
4342 * @param pSSM The saved state handle.
4343 */
4344static void pcnetSaveConfig(PCNetState *pThis, PSSMHANDLE pSSM)
4345{
4346 SSMR3PutMem(pSSM, &pThis->MacConfigured, sizeof(pThis->MacConfigured));
4347 SSMR3PutBool(pSSM, pThis->fAm79C973); /* >= If version 0.8 */
4348 SSMR3PutU32(pSSM, pThis->u32LinkSpeed);
4349}
4350
4351
4352/**
4353 * Live Save, pass 0.
4354 *
4355 * @returns VBox status code.
4356 * @param pDevIns The device instance.
4357 * @param pSSM The saved state handle.
4358 * @param uPass The pass number.
4359 */
4360static DECLCALLBACK(int) pcnetLiveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uPass)
4361{
4362 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4363 pcnetSaveConfig(pThis, pSSM);
4364 return VINF_SSM_DONT_CALL_AGAIN;
4365}
4366
4367
4368/**
4369 * Serializes the receive thread, it may be working inside the critsect.
4370 *
4371 * @returns VBox status code.
4372 * @param pDevIns The device instance.
4373 * @param pSSM The saved state handle.
4374 */
4375static DECLCALLBACK(int) pcnetSavePrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4376{
4377 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4378
4379 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4380 AssertRC(rc);
4381 PDMCritSectLeave(&pThis->CritSect);
4382
4383 return VINF_SUCCESS;
4384}
4385
4386
4387/**
4388 * Saves a state of the PC-Net II device.
4389 *
4390 * @returns VBox status code.
4391 * @param pDevIns The device instance.
4392 * @param pSSM The saved state handle.
4393 */
4394static DECLCALLBACK(int) pcnetSaveExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4395{
4396 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4397
4398 SSMR3PutBool(pSSM, pThis->fLinkUp);
4399 SSMR3PutU32(pSSM, pThis->u32RAP);
4400 SSMR3PutS32(pSSM, pThis->iISR);
4401 SSMR3PutU32(pSSM, pThis->u32Lnkst);
4402 SSMR3PutBool(pSSM, pThis->fPrivIfEnabled); /* >= If version 0.9 */
4403 SSMR3PutBool(pSSM, pThis->fSignalRxMiss); /* >= If version 0.10 */
4404 SSMR3PutGCPhys32(pSSM, pThis->GCRDRA);
4405 SSMR3PutGCPhys32(pSSM, pThis->GCTDRA);
4406 SSMR3PutMem(pSSM, pThis->aPROM, sizeof(pThis->aPROM));
4407 SSMR3PutMem(pSSM, pThis->aCSR, sizeof(pThis->aCSR));
4408 SSMR3PutMem(pSSM, pThis->aBCR, sizeof(pThis->aBCR));
4409 SSMR3PutMem(pSSM, pThis->aMII, sizeof(pThis->aMII));
4410 SSMR3PutU16(pSSM, pThis->u16CSR0LastSeenByGuest);
4411 SSMR3PutU64(pSSM, pThis->u64LastPoll);
4412 pcnetSaveConfig(pThis, pSSM);
4413
4414 int rc = VINF_SUCCESS;
4415#ifndef PCNET_NO_POLLING
4416 rc = TMR3TimerSave(pThis->CTX_SUFF(pTimerPoll), pSSM);
4417 if (RT_FAILURE(rc))
4418 return rc;
4419#endif
4420 if (pThis->fAm79C973)
4421 rc = TMR3TimerSave(pThis->CTX_SUFF(pTimerSoftInt), pSSM);
4422 return rc;
4423}
4424
4425
4426/**
4427 * Serializes the receive thread, it may be working inside the critsect.
4428 *
4429 * @returns VBox status code.
4430 * @param pDevIns The device instance.
4431 * @param pSSM The saved state handle.
4432 */
4433static DECLCALLBACK(int) pcnetLoadPrep(PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
4434{
4435 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4436
4437 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4438 AssertRC(rc);
4439 PDMCritSectLeave(&pThis->CritSect);
4440
4441 return VINF_SUCCESS;
4442}
4443
4444
4445/**
4446 * Loads a saved PC-Net II device state.
4447 *
4448 * @returns VBox status code.
4449 * @param pDevIns The device instance.
4450 * @param pSSM The handle to the saved state.
4451 * @param uVersion The data unit version number.
4452 * @param uPass The data pass.
4453 */
4454static DECLCALLBACK(int) pcnetLoadExec(PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
4455{
4456 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4457
4458 if ( SSM_VERSION_MAJOR_CHANGED(uVersion, PCNET_SAVEDSTATE_VERSION)
4459 || SSM_VERSION_MINOR(uVersion) < 7)
4460 return VERR_SSM_UNSUPPORTED_DATA_UNIT_VERSION;
4461
4462 if (uPass == SSM_PASS_FINAL)
4463 {
4464 /* restore data */
4465 SSMR3GetBool(pSSM, &pThis->fLinkUp);
4466 SSMR3GetU32(pSSM, &pThis->u32RAP);
4467 SSMR3GetS32(pSSM, &pThis->iISR);
4468 SSMR3GetU32(pSSM, &pThis->u32Lnkst);
4469 if ( SSM_VERSION_MAJOR(uVersion) > 0
4470 || SSM_VERSION_MINOR(uVersion) >= 9)
4471 {
4472 SSMR3GetBool(pSSM, &pThis->fPrivIfEnabled);
4473 if (pThis->fPrivIfEnabled)
4474 LogRel(("PCNet#%d: Enabling private interface\n", PCNET_INST_NR));
4475 }
4476 if ( SSM_VERSION_MAJOR(uVersion) > 0
4477 || SSM_VERSION_MINOR(uVersion) >= 10)
4478 {
4479 SSMR3GetBool(pSSM, &pThis->fSignalRxMiss);
4480 }
4481 SSMR3GetGCPhys32(pSSM, &pThis->GCRDRA);
4482 SSMR3GetGCPhys32(pSSM, &pThis->GCTDRA);
4483 SSMR3GetMem(pSSM, &pThis->aPROM, sizeof(pThis->aPROM));
4484 SSMR3GetMem(pSSM, &pThis->aCSR, sizeof(pThis->aCSR));
4485 SSMR3GetMem(pSSM, &pThis->aBCR, sizeof(pThis->aBCR));
4486 SSMR3GetMem(pSSM, &pThis->aMII, sizeof(pThis->aMII));
4487 SSMR3GetU16(pSSM, &pThis->u16CSR0LastSeenByGuest);
4488 SSMR3GetU64(pSSM, &pThis->u64LastPoll);
4489 }
4490
4491 /* check config */
4492 RTMAC Mac;
4493 int rc = SSMR3GetMem(pSSM, &Mac, sizeof(Mac));
4494 AssertRCReturn(rc, rc);
4495 if ( memcmp(&Mac, &pThis->MacConfigured, sizeof(Mac))
4496 && (uPass == 0 || !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)) )
4497 LogRel(("PCNet#%u: The mac address differs: config=%RTmac saved=%RTmac\n", PCNET_INST_NR, &pThis->MacConfigured, &Mac));
4498
4499 bool fAm79C973;
4500 rc = SSMR3GetBool(pSSM, &fAm79C973);
4501 AssertRCReturn(rc, rc);
4502 if (pThis->fAm79C973 != fAm79C973)
4503 return SSMR3SetCfgError(pSSM, RT_SRC_POS, N_("The fAm79C973 flag differs: config=%RTbool saved=%RTbool"), pThis->fAm79C973, fAm79C973);
4504
4505 uint32_t u32LinkSpeed;
4506 rc = SSMR3GetU32(pSSM, &u32LinkSpeed);
4507 AssertRCReturn(rc, rc);
4508 if ( pThis->u32LinkSpeed != u32LinkSpeed
4509 && (uPass == 0 || !PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns)) )
4510 LogRel(("PCNet#%u: The mac link speed differs: config=%u saved=%u\n", PCNET_INST_NR, pThis->u32LinkSpeed, u32LinkSpeed));
4511
4512 if (uPass == SSM_PASS_FINAL)
4513 {
4514 /* restore timers and stuff */
4515#ifndef PCNET_NO_POLLING
4516 TMR3TimerLoad(pThis->CTX_SUFF(pTimerPoll), pSSM);
4517#endif
4518 if (pThis->fAm79C973)
4519 {
4520 if ( SSM_VERSION_MAJOR(uVersion) > 0
4521 || SSM_VERSION_MINOR(uVersion) >= 8)
4522 TMR3TimerLoad(pThis->CTX_SUFF(pTimerSoftInt), pSSM);
4523 }
4524
4525 pThis->iLog2DescSize = BCR_SWSTYLE(pThis)
4526 ? 4
4527 : 3;
4528 pThis->GCUpperPhys = BCR_SSIZE32(pThis)
4529 ? 0
4530 : (0xff00 & (uint32_t)pThis->aCSR[2]) << 16;
4531
4532 /* update promiscuous mode. */
4533 if (pThis->pDrvR3)
4534 pThis->pDrvR3->pfnSetPromiscuousMode(pThis->pDrvR3, CSR_PROM(pThis));
4535
4536#ifdef PCNET_NO_POLLING
4537 /* Enable physical monitoring again (!) */
4538 pcnetUpdateRingHandlers(pThis);
4539#endif
4540 /* Indicate link down to the guest OS that all network connections have
4541 been lost, unless we've been teleported here. */
4542 if (!PDMDevHlpVMTeleportedAndNotFullyResumedYet(pDevIns))
4543 pcnetTempLinkDown(pThis);
4544 }
4545
4546 return VINF_SUCCESS;
4547}
4548
4549
4550/**
4551 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
4552 */
4553static DECLCALLBACK(void *) pcnetQueryInterface(struct PDMIBASE *pInterface, const char *pszIID)
4554{
4555 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, IBase);
4556 Assert(&pThis->IBase == pInterface);
4557 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
4558 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKDOWN, &pThis->INetworkDown);
4559 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKCONFIG, &pThis->INetworkConfig);
4560 PDMIBASE_RETURN_INTERFACE(pszIID, PDMILEDPORTS, &pThis->ILeds);
4561 return NULL;
4562}
4563
4564
4565
4566/**
4567 * Check if the device/driver can receive data now.
4568 * This must be called before the pfnRecieve() method is called.
4569 *
4570 * @returns VBox status code.
4571 * @param pInterface Pointer to the interface structure containing the called function pointer.
4572 */
4573static int pcnetCanReceive(PCNetState *pThis)
4574{
4575 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4576 AssertReleaseRC(rc);
4577
4578 rc = VERR_NET_NO_BUFFER_SPACE;
4579
4580 if (RT_LIKELY(!CSR_DRX(pThis) && !CSR_STOP(pThis) && !CSR_SPND(pThis)))
4581 {
4582 if (HOST_IS_OWNER(CSR_CRST(pThis)) && pThis->GCRDRA)
4583 pcnetRdtePoll(pThis);
4584
4585 if (RT_UNLIKELY(HOST_IS_OWNER(CSR_CRST(pThis))))
4586 {
4587 /** @todo Notify the guest _now_. Will potentially increase the interrupt load */
4588 if (pThis->fSignalRxMiss)
4589 pThis->aCSR[0] |= 0x1000; /* Set MISS flag */
4590 }
4591 else
4592 rc = VINF_SUCCESS;
4593 }
4594
4595 PDMCritSectLeave(&pThis->CritSect);
4596 return rc;
4597}
4598
4599
4600/**
4601 * @interface_method_impl{PDMINETWORKDOWN,pfnWaitReceiveAvail}
4602 */
4603static DECLCALLBACK(int) pcnetNetworkDown_WaitReceiveAvail(PPDMINETWORKDOWN pInterface, RTMSINTERVAL cMillies)
4604{
4605 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, INetworkDown);
4606
4607 int rc = pcnetCanReceive(pThis);
4608 if (RT_SUCCESS(rc))
4609 return VINF_SUCCESS;
4610 if (RT_UNLIKELY(cMillies == 0))
4611 return VERR_NET_NO_BUFFER_SPACE;
4612
4613 rc = VERR_INTERRUPTED;
4614 ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, true);
4615 STAM_PROFILE_START(&pThis->StatRxOverflow, a);
4616 VMSTATE enmVMState;
4617 while (RT_LIKELY( (enmVMState = PDMDevHlpVMState(pThis->CTX_SUFF(pDevIns))) == VMSTATE_RUNNING
4618 || enmVMState == VMSTATE_RUNNING_LS))
4619 {
4620 int rc2 = pcnetCanReceive(pThis);
4621 if (RT_SUCCESS(rc2))
4622 {
4623 rc = VINF_SUCCESS;
4624 break;
4625 }
4626 LogFlow(("pcnetNetworkDown_WaitReceiveAvail: waiting cMillies=%u...\n", cMillies));
4627 /* Start the poll timer once which will remain active as long fMaybeOutOfSpace
4628 * is true -- even if (transmit) polling is disabled (CSR_DPOLL). */
4629 rc2 = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4630 AssertReleaseRC(rc2);
4631 pcnetPollTimerStart(pThis);
4632 PDMCritSectLeave(&pThis->CritSect);
4633 RTSemEventWait(pThis->hEventOutOfRxSpace, cMillies);
4634 }
4635 STAM_PROFILE_STOP(&pThis->StatRxOverflow, a);
4636 ASMAtomicXchgBool(&pThis->fMaybeOutOfSpace, false);
4637
4638 return rc;
4639}
4640
4641
4642/**
4643 * @interface_method_impl{PDMINETWORKDOWN,pfnReceive}
4644 */
4645static DECLCALLBACK(int) pcnetNetworkDown_Receive(PPDMINETWORKDOWN pInterface, const void *pvBuf, size_t cb)
4646{
4647 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, INetworkDown);
4648 int rc;
4649
4650 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
4651 rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4652 AssertReleaseRC(rc);
4653
4654 /*
4655 * Check for the max ethernet frame size, taking the IEEE 802.1Q (VLAN) tag into
4656 * account. Note that we are *not* expecting the CRC Checksum.
4657 * Ethernet frames consists of a 14-byte header [+ 4-byte vlan tag] + a 1500-byte body.
4658 */
4659 if (RT_LIKELY( cb <= 1514
4660 || ( cb <= 1518
4661 && ((PCRTNETETHERHDR)pvBuf)->EtherType == RT_H2BE_U16_C(RTNET_ETHERTYPE_VLAN))))
4662 {
4663 if (cb > 70) /* unqualified guess */
4664 pThis->Led.Asserted.s.fReading = pThis->Led.Actual.s.fReading = 1;
4665 pcnetReceiveNoSync(pThis, (const uint8_t *)pvBuf, cb);
4666 pThis->Led.Actual.s.fReading = 0;
4667 }
4668#ifdef LOG_ENABLED
4669 else
4670 {
4671 static bool s_fFirstBigFrameLoss = true;
4672 unsigned cbMaxFrame = ((PCRTNETETHERHDR)pvBuf)->EtherType == RT_H2BE_U16_C(RTNET_ETHERTYPE_VLAN)
4673 ? 1518 : 1514;
4674 if (s_fFirstBigFrameLoss)
4675 {
4676 s_fFirstBigFrameLoss = false;
4677 Log(("PCNet#%d: Received giant frame %zu, max %u. (Further giants will be reported at level5.)\n",
4678 PCNET_INST_NR, cb, cbMaxFrame));
4679 }
4680 else
4681 Log5(("PCNet#%d: Received giant frame %zu bytes, max %u.\n",
4682 PCNET_INST_NR, cb, cbMaxFrame));
4683 }
4684#endif /* LOG_ENABLED */
4685
4686 PDMCritSectLeave(&pThis->CritSect);
4687 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
4688
4689 return VINF_SUCCESS;
4690}
4691
4692
4693/**
4694 * @interface_method_impl{PDMINETWORKDOWN,pfnXmitPending}
4695 */
4696static DECLCALLBACK(void) pcnetNetworkDown_XmitPending(PPDMINETWORKDOWN pInterface)
4697{
4698 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, INetworkDown);
4699 pcnetXmitPending(pThis, true /*fOnWorkerThread*/);
4700}
4701
4702
4703
4704/**
4705 * Gets the current Media Access Control (MAC) address.
4706 *
4707 * @returns VBox status code.
4708 * @param pInterface Pointer to the interface structure containing the called function pointer.
4709 * @param pMac Where to store the MAC address.
4710 * @thread EMT
4711 */
4712static DECLCALLBACK(int) pcnetGetMac(PPDMINETWORKCONFIG pInterface, PRTMAC pMac)
4713{
4714 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, INetworkConfig);
4715 memcpy(pMac, pThis->aPROM, sizeof(*pMac));
4716 return VINF_SUCCESS;
4717}
4718
4719
4720/**
4721 * Gets the new link state.
4722 *
4723 * @returns The current link state.
4724 * @param pInterface Pointer to the interface structure containing the called function pointer.
4725 * @thread EMT
4726 */
4727static DECLCALLBACK(PDMNETWORKLINKSTATE) pcnetGetLinkState(PPDMINETWORKCONFIG pInterface)
4728{
4729 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, INetworkConfig);
4730 if (pThis->fLinkUp && !pThis->fLinkTempDown)
4731 return PDMNETWORKLINKSTATE_UP;
4732 if (!pThis->fLinkUp)
4733 return PDMNETWORKLINKSTATE_DOWN;
4734 if (pThis->fLinkTempDown)
4735 return PDMNETWORKLINKSTATE_DOWN_RESUME;
4736 AssertMsgFailed(("Invalid link state!\n"));
4737 return PDMNETWORKLINKSTATE_INVALID;
4738}
4739
4740
4741/**
4742 * Sets the new link state.
4743 *
4744 * @returns VBox status code.
4745 * @param pInterface Pointer to the interface structure containing the called function pointer.
4746 * @param enmState The new link state
4747 * @thread EMT
4748 */
4749static DECLCALLBACK(int) pcnetSetLinkState(PPDMINETWORKCONFIG pInterface, PDMNETWORKLINKSTATE enmState)
4750{
4751 PCNetState *pThis = RT_FROM_MEMBER(pInterface, PCNetState, INetworkConfig);
4752 bool fLinkUp;
4753 if ( enmState != PDMNETWORKLINKSTATE_DOWN
4754 && enmState != PDMNETWORKLINKSTATE_UP)
4755 {
4756 AssertMsgFailed(("Invalid parameter enmState=%d\n", enmState));
4757 return VERR_INVALID_PARAMETER;
4758 }
4759
4760 /* has the state changed? */
4761 fLinkUp = enmState == PDMNETWORKLINKSTATE_UP;
4762 if (pThis->fLinkUp != fLinkUp)
4763 {
4764 pThis->fLinkUp = fLinkUp;
4765 if (fLinkUp)
4766 {
4767 /* connect with a delay of 5 seconds */
4768 pThis->fLinkTempDown = true;
4769 pThis->cLinkDownReported = 0;
4770 pThis->aCSR[0] |= RT_BIT(15) | RT_BIT(13); /* ERR | CERR (this is probably wrong) */
4771 pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1;
4772 int rc = TMTimerSetMillies(pThis->pTimerRestore, 5000);
4773 AssertRC(rc);
4774 }
4775 else
4776 {
4777 /* disconnect */
4778 pThis->cLinkDownReported = 0;
4779 pThis->aCSR[0] |= RT_BIT(15) | RT_BIT(13); /* ERR | CERR (this is probably wrong) */
4780 pThis->Led.Asserted.s.fError = pThis->Led.Actual.s.fError = 1;
4781 }
4782 Assert(!PDMCritSectIsOwner(&pThis->CritSect));
4783 if (pThis->pDrvR3)
4784 pThis->pDrvR3->pfnNotifyLinkChanged(pThis->pDrvR3, enmState);
4785 }
4786 return VINF_SUCCESS;
4787}
4788
4789
4790/**
4791 * Gets the pointer to the status LED of a unit.
4792 *
4793 * @returns VBox status code.
4794 * @param pInterface Pointer to the interface structure containing the called function pointer.
4795 * @param iLUN The unit which status LED we desire.
4796 * @param ppLed Where to store the LED pointer.
4797 */
4798static DECLCALLBACK(int) pcnetQueryStatusLed(PPDMILEDPORTS pInterface, unsigned iLUN, PPDMLED *ppLed)
4799{
4800 PCNetState *pThis = (PCNetState *)( (uintptr_t)pInterface - RT_OFFSETOF(PCNetState, ILeds) );
4801 if (iLUN == 0)
4802 {
4803 *ppLed = &pThis->Led;
4804 return VINF_SUCCESS;
4805 }
4806 return VERR_PDM_LUN_NOT_FOUND;
4807}
4808
4809
4810/**
4811 * @copydoc FNPDMDEVPOWEROFF
4812 */
4813static DECLCALLBACK(void) pcnetPowerOff(PPDMDEVINS pDevIns)
4814{
4815 /* Poke thread waiting for buffer space. */
4816 pcnetWakeupReceive(pDevIns);
4817}
4818
4819
4820/**
4821 * Detach notification.
4822 *
4823 * One port on the network card has been disconnected from the network.
4824 *
4825 * @param pDevIns The device instance.
4826 * @param iLUN The logical unit which is being detached.
4827 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
4828 */
4829static DECLCALLBACK(void) pcnetDetach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
4830{
4831 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4832 Log(("#%d pcnetDetach:\n", PCNET_INST_NR));
4833
4834 AssertLogRelReturnVoid(iLUN == 0);
4835
4836 PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4837
4838 /** @todo: r=pritesh still need to check if i missed
4839 * to clean something in this function
4840 */
4841
4842 /*
4843 * Zero some important members.
4844 */
4845 pThis->pDrvBase = NULL;
4846 pThis->pDrvR3 = NULL;
4847 pThis->pDrvR0 = NIL_RTR0PTR;
4848 pThis->pDrvRC = NIL_RTRCPTR;
4849
4850 PDMCritSectLeave(&pThis->CritSect);
4851}
4852
4853
4854/**
4855 * Attach the Network attachment.
4856 *
4857 * One port on the network card has been connected to a network.
4858 *
4859 * @returns VBox status code.
4860 * @param pDevIns The device instance.
4861 * @param iLUN The logical unit which is being attached.
4862 * @param fFlags Flags, combination of the PDMDEVATT_FLAGS_* \#defines.
4863 *
4864 * @remarks This code path is not used during construction.
4865 */
4866static DECLCALLBACK(int) pcnetAttach(PPDMDEVINS pDevIns, unsigned iLUN, uint32_t fFlags)
4867{
4868 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4869 LogFlow(("#%d pcnetAttach:\n", PCNET_INST_NR));
4870
4871 AssertLogRelReturn(iLUN == 0, VERR_PDM_NO_SUCH_LUN);
4872
4873 PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
4874
4875 /*
4876 * Attach the driver.
4877 */
4878 int rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThis->IBase, &pThis->pDrvBase, "Network Port");
4879 if (RT_SUCCESS(rc))
4880 {
4881 if (rc == VINF_NAT_DNS)
4882 {
4883#ifdef RT_OS_LINUX
4884 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "NoDNSforNAT",
4885 N_("A Domain Name Server (DNS) for NAT networking could not be determined. Please check your /etc/resolv.conf for <tt>nameserver</tt> entries. Either add one manually (<i>man resolv.conf</i>) or ensure that your host is correctly connected to an ISP. If you ignore this warning the guest will not be able to perform nameserver lookups and it will probably observe delays if trying so"));
4886#else
4887 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "NoDNSforNAT",
4888 N_("A Domain Name Server (DNS) for NAT networking could not be determined. Ensure that your host is correctly connected to an ISP. If you ignore this warning the guest will not be able to perform nameserver lookups and it will probably observe delays if trying so"));
4889#endif
4890 }
4891 pThis->pDrvR3 = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMINETWORKUP);
4892 AssertMsgStmt(pThis->pDrvR3, ("Failed to obtain the PDMINETWORKUP interface!\n"),
4893 rc = VERR_PDM_MISSING_INTERFACE_BELOW);
4894 pThis->pDrvR0 = PDMIBASER0_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBASER0), PDMINETWORKUP);
4895 pThis->pDrvRC = PDMIBASERC_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBASERC), PDMINETWORKUP);
4896 }
4897 else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER
4898 || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME)
4899 {
4900 /* This should never happen because this function is not called
4901 * if there is no driver to attach! */
4902 Log(("#%d No attached driver!\n", PCNET_INST_NR));
4903 }
4904
4905 /*
4906 * Temporary set the link down if it was up so that the guest
4907 * will know that we have change the configuration of the
4908 * network card
4909 */
4910 if (RT_SUCCESS(rc))
4911 pcnetTempLinkDown(pThis);
4912
4913 PDMCritSectLeave(&pThis->CritSect);
4914 return rc;
4915
4916}
4917
4918
4919/**
4920 * @copydoc FNPDMDEVSUSPEND
4921 */
4922static DECLCALLBACK(void) pcnetSuspend(PPDMDEVINS pDevIns)
4923{
4924 /* Poke thread waiting for buffer space. */
4925 pcnetWakeupReceive(pDevIns);
4926}
4927
4928
4929/**
4930 * @copydoc FNPDMDEVRESET
4931 */
4932static DECLCALLBACK(void) pcnetReset(PPDMDEVINS pDevIns)
4933{
4934 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4935 if (pThis->fLinkTempDown)
4936 {
4937 pThis->cLinkDownReported = 0x10000;
4938 TMTimerStop(pThis->pTimerRestore);
4939 pcnetTimerRestore(pDevIns, pThis->pTimerRestore, pThis);
4940 }
4941 if (pThis->pSharedMMIOR3)
4942 pcnetInitSharedMemory(pThis);
4943
4944 /** @todo How to flush the queues? */
4945 pcnetHardReset(pThis);
4946}
4947
4948
4949/**
4950 * @copydoc FNPDMDEVRELOCATE
4951 */
4952static DECLCALLBACK(void) pcnetRelocate(PPDMDEVINS pDevIns, RTGCINTPTR offDelta)
4953{
4954 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4955 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
4956 pThis->pXmitQueueRC = PDMQueueRCPtr(pThis->pXmitQueueR3);
4957 pThis->pCanRxQueueRC = PDMQueueRCPtr(pThis->pCanRxQueueR3);
4958 if (pThis->pSharedMMIOR3)
4959 pThis->pSharedMMIORC += offDelta;
4960#ifdef PCNET_NO_POLLING
4961 pThis->pfnEMInterpretInstructionRC += offDelta;
4962#else
4963 pThis->pTimerPollRC = TMTimerRCPtr(pThis->pTimerPollR3);
4964#endif
4965 if (pThis->fAm79C973)
4966 pThis->pTimerSoftIntRC = TMTimerRCPtr(pThis->pTimerSoftIntR3);
4967}
4968
4969
4970/**
4971 * Destruct a device instance.
4972 *
4973 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
4974 * resources can be freed correctly.
4975 *
4976 * @returns VBox status.
4977 * @param pDevIns The device instance data.
4978 */
4979static DECLCALLBACK(int) pcnetDestruct(PPDMDEVINS pDevIns)
4980{
4981 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
4982 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
4983
4984 if (PDMCritSectIsInitialized(&pThis->CritSect))
4985 {
4986 RTSemEventSignal(pThis->hEventOutOfRxSpace);
4987 RTSemEventDestroy(pThis->hEventOutOfRxSpace);
4988 pThis->hEventOutOfRxSpace = NIL_RTSEMEVENT;
4989 PDMR3CritSectDelete(&pThis->CritSect);
4990 }
4991 return VINF_SUCCESS;
4992}
4993
4994
4995/**
4996 * @interface_method_impl{PDMDEVREG,pfnConstruct}
4997 */
4998static DECLCALLBACK(int) pcnetConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
4999{
5000 PCNetState *pThis = PDMINS_2_DATA(pDevIns, PCNetState *);
5001 PPDMIBASE pBase;
5002 char szTmp[128];
5003 int rc;
5004
5005 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
5006 Assert(RT_ELEMENTS(pThis->aBCR) == BCR_MAX_RAP);
5007 Assert(RT_ELEMENTS(pThis->aMII) == MII_MAX_REG);
5008 Assert(sizeof(pThis->abLoopBuf) == RT_ALIGN_Z(sizeof(pThis->abLoopBuf), 16));
5009
5010 /*
5011 * Init what's required to make the destructor safe.
5012 */
5013 pThis->hEventOutOfRxSpace = NIL_RTSEMEVENT;
5014
5015 /*
5016 * Validate configuration.
5017 */
5018 if (!CFGMR3AreValuesValid(pCfg, "MAC\0" "CableConnected\0" "Am79C973\0" "LineSpeed\0" "GCEnabled\0" "R0Enabled\0" "PrivIfEnabled\0"))
5019 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
5020 N_("Invalid configuration for pcnet device"));
5021
5022 /*
5023 * Read the configuration.
5024 */
5025 rc = CFGMR3QueryBytes(pCfg, "MAC", &pThis->MacConfigured, sizeof(pThis->MacConfigured));
5026 if (RT_FAILURE(rc))
5027 return PDMDEV_SET_ERROR(pDevIns, rc,
5028 N_("Configuration error: Failed to get the \"MAC\" value"));
5029 rc = CFGMR3QueryBoolDef(pCfg, "CableConnected", &pThis->fLinkUp, true);
5030 if (RT_FAILURE(rc))
5031 return PDMDEV_SET_ERROR(pDevIns, rc,
5032 N_("Configuration error: Failed to get the \"CableConnected\" value"));
5033
5034 rc = CFGMR3QueryBoolDef(pCfg, "Am79C973", &pThis->fAm79C973, false);
5035 if (RT_FAILURE(rc))
5036 return PDMDEV_SET_ERROR(pDevIns, rc,
5037 N_("Configuration error: Failed to get the \"Am79C973\" value"));
5038
5039 rc = CFGMR3QueryU32Def(pCfg, "LineSpeed", &pThis->u32LinkSpeed, 1000000); /* 1GBit/s (in kbps units)*/
5040 if (RT_FAILURE(rc))
5041 return PDMDEV_SET_ERROR(pDevIns, rc,
5042 N_("Configuration error: Failed to get the \"LineSpeed\" value"));
5043
5044#ifdef PCNET_GC_ENABLED
5045 rc = CFGMR3QueryBoolDef(pCfg, "GCEnabled", &pThis->fGCEnabled, true);
5046 if (RT_FAILURE(rc))
5047 return PDMDEV_SET_ERROR(pDevIns, rc,
5048 N_("Configuration error: Failed to get the \"GCEnabled\" value"));
5049
5050 rc = CFGMR3QueryBoolDef(pCfg, "R0Enabled", &pThis->fR0Enabled, true);
5051 if (RT_FAILURE(rc))
5052 return PDMDEV_SET_ERROR(pDevIns, rc,
5053 N_("Configuration error: Failed to get the \"R0Enabled\" value"));
5054
5055#else /* !PCNET_GC_ENABLED */
5056 pThis->fGCEnabled = false;
5057 pThis->fR0Enabled = false;
5058#endif /* !PCNET_GC_ENABLED */
5059
5060
5061 /*
5062 * Initialize data (most of it anyway).
5063 */
5064 pThis->pDevInsR3 = pDevIns;
5065 pThis->pDevInsR0 = PDMDEVINS_2_R0PTR(pDevIns);
5066 pThis->pDevInsRC = PDMDEVINS_2_RCPTR(pDevIns);
5067 pThis->Led.u32Magic = PDMLED_MAGIC;
5068 /* IBase */
5069 pThis->IBase.pfnQueryInterface = pcnetQueryInterface;
5070 /* INeworkPort */
5071 pThis->INetworkDown.pfnWaitReceiveAvail = pcnetNetworkDown_WaitReceiveAvail;
5072 pThis->INetworkDown.pfnReceive = pcnetNetworkDown_Receive;
5073 pThis->INetworkDown.pfnXmitPending = pcnetNetworkDown_XmitPending;
5074 /* INetworkConfig */
5075 pThis->INetworkConfig.pfnGetMac = pcnetGetMac;
5076 pThis->INetworkConfig.pfnGetLinkState = pcnetGetLinkState;
5077 pThis->INetworkConfig.pfnSetLinkState = pcnetSetLinkState;
5078 /* ILeds */
5079 pThis->ILeds.pfnQueryStatusLed = pcnetQueryStatusLed;
5080
5081 /* PCI Device */
5082 PCIDevSetVendorId(&pThis->PciDev, 0x1022);
5083 PCIDevSetDeviceId(&pThis->PciDev, 0x2000);
5084 pThis->PciDev.config[0x04] = 0x07; /* command */
5085 pThis->PciDev.config[0x05] = 0x00;
5086 pThis->PciDev.config[0x06] = 0x80; /* status */
5087 pThis->PciDev.config[0x07] = 0x02;
5088 pThis->PciDev.config[0x08] = pThis->fAm79C973 ? 0x40 : 0x10; /* revision */
5089 pThis->PciDev.config[0x09] = 0x00;
5090 pThis->PciDev.config[0x0a] = 0x00; /* ethernet network controller */
5091 pThis->PciDev.config[0x0b] = 0x02;
5092 pThis->PciDev.config[0x0e] = 0x00; /* header_type */
5093
5094 pThis->PciDev.config[0x10] = 0x01; /* IO Base */
5095 pThis->PciDev.config[0x11] = 0x00;
5096 pThis->PciDev.config[0x12] = 0x00;
5097 pThis->PciDev.config[0x13] = 0x00;
5098 pThis->PciDev.config[0x14] = 0x00; /* MMIO Base */
5099 pThis->PciDev.config[0x15] = 0x00;
5100 pThis->PciDev.config[0x16] = 0x00;
5101 pThis->PciDev.config[0x17] = 0x00;
5102
5103 /* subsystem and subvendor IDs */
5104 pThis->PciDev.config[0x2c] = 0x22; /* subsystem vendor id */
5105 pThis->PciDev.config[0x2d] = 0x10;
5106 pThis->PciDev.config[0x2e] = 0x00; /* subsystem id */
5107 pThis->PciDev.config[0x2f] = 0x20;
5108 pThis->PciDev.config[0x3d] = 1; /* interrupt pin 0 */
5109 pThis->PciDev.config[0x3e] = 0x06;
5110 pThis->PciDev.config[0x3f] = 0xff;
5111
5112 /*
5113 * Register the PCI device, its I/O regions, the timer and the saved state item.
5114 */
5115 rc = PDMDevHlpPCIRegister(pDevIns, &pThis->PciDev);
5116 if (RT_FAILURE(rc))
5117 return rc;
5118 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 0, PCNET_IOPORT_SIZE,
5119 PCI_ADDRESS_SPACE_IO, pcnetIOPortMap);
5120 if (RT_FAILURE(rc))
5121 return rc;
5122 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 1, PCNET_PNPMMIO_SIZE,
5123 PCI_ADDRESS_SPACE_MEM, pcnetMMIOMap);
5124 if (RT_FAILURE(rc))
5125 return rc;
5126
5127 bool fPrivIfEnabled;
5128 rc = CFGMR3QueryBool(pCfg, "PrivIfEnabled", &fPrivIfEnabled);
5129 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
5130 fPrivIfEnabled = true;
5131 else if (RT_FAILURE(rc))
5132 return PDMDEV_SET_ERROR(pDevIns, rc,
5133 N_("Configuration error: Failed to get the \"PrivIfEnabled\" value"));
5134
5135 if (fPrivIfEnabled)
5136 {
5137 /*
5138 * Initialize shared memory between host and guest for descriptors and RX buffers. Most guests
5139 * should not care if there is an additional PCI resource but just in case we made this configurable.
5140 */
5141 rc = PDMDevHlpMMIO2Register(pDevIns, 2, PCNET_GUEST_SHARED_MEMORY_SIZE, 0, (void **)&pThis->pSharedMMIOR3, "PCNetShMem");
5142 if (RT_FAILURE(rc))
5143 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5144 N_("Failed to allocate %u bytes of memory for the PCNet device"), PCNET_GUEST_SHARED_MEMORY_SIZE);
5145 rc = PDMDevHlpMMHyperMapMMIO2(pDevIns, 2, 0, 8192, "PCNetShMem", &pThis->pSharedMMIORC);
5146 if (RT_FAILURE(rc))
5147 return PDMDevHlpVMSetError(pDevIns, rc, RT_SRC_POS,
5148 N_("Failed to map 8192 bytes of memory for the PCNet device into the hyper memory"));
5149 pThis->pSharedMMIOR0 = (uintptr_t)pThis->pSharedMMIOR3; /** @todo #1865: Map MMIO2 into ring-0. */
5150
5151 pcnetInitSharedMemory(pThis);
5152 rc = PDMDevHlpPCIIORegionRegister(pDevIns, 2, PCNET_GUEST_SHARED_MEMORY_SIZE,
5153 PCI_ADDRESS_SPACE_MEM, pcnetMMIOSharedMap);
5154 if (RT_FAILURE(rc))
5155 return rc;
5156 }
5157
5158 /*
5159 * Initialize critical section.
5160 * This must be done before register the critsect with the timer code, and also before
5161 * attaching drivers or anything else that may call us back.
5162 */
5163 rc = PDMDevHlpCritSectInit(pDevIns, &pThis->CritSect, RT_SRC_POS, "PCNet#%u", iInstance);
5164 if (RT_FAILURE(rc))
5165 return rc;
5166
5167 rc = RTSemEventCreate(&pThis->hEventOutOfRxSpace);
5168 AssertRC(rc);
5169
5170#ifdef PCNET_NO_POLLING
5171 /*
5172 * Resolve the R0 and RC handlers.
5173 */
5174 rc = PDMR3LdrGetSymbolR0Lazy(PDMDevHlpGetVM(pDevIns), NULL, NULL, "EMInterpretInstruction", &pThis->pfnEMInterpretInstructionR0);
5175 if (RT_SUCCESS(rc))
5176 rc = PDMR3LdrGetSymbolRCLazy(PDMDevHlpGetVM(pDevIns), NULL, NULL, "EMInterpretInstruction", (RTGCPTR *)&pThis->pfnEMInterpretInstructionRC);
5177 AssertLogRelMsgRCReturn(rc, ("PDMR3LdrGetSymbolRCLazy(EMInterpretInstruction) -> %Rrc\n", rc), rc);
5178#else
5179 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, pcnetTimer, pThis,
5180 TMTIMER_FLAGS_NO_CRIT_SECT, "PCNet Poll Timer", &pThis->pTimerPollR3);
5181 if (RT_FAILURE(rc))
5182 return rc;
5183 pThis->pTimerPollR0 = TMTimerR0Ptr(pThis->pTimerPollR3);
5184 pThis->pTimerPollRC = TMTimerRCPtr(pThis->pTimerPollR3);
5185 TMR3TimerSetCritSect(pThis->pTimerPollR3, &pThis->CritSect);
5186#endif
5187 if (pThis->fAm79C973)
5188 {
5189 /* Software Interrupt timer */
5190 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, pcnetTimerSoftInt, pThis, /** @todo r=bird: the locking here looks bogus now with SMP... */
5191 TMTIMER_FLAGS_NO_CRIT_SECT, "PCNet SoftInt Timer", &pThis->pTimerSoftIntR3);
5192 if (RT_FAILURE(rc))
5193 return rc;
5194 pThis->pTimerSoftIntR0 = TMTimerR0Ptr(pThis->pTimerSoftIntR3);
5195 pThis->pTimerSoftIntRC = TMTimerRCPtr(pThis->pTimerSoftIntR3);
5196 TMR3TimerSetCritSect(pThis->pTimerSoftIntR3, &pThis->CritSect);
5197 }
5198 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, pcnetTimerRestore, pThis,
5199 TMTIMER_FLAGS_NO_CRIT_SECT, "PCNet Restore Timer", &pThis->pTimerRestore);
5200 if (RT_FAILURE(rc))
5201 return rc;
5202
5203 rc = PDMDevHlpSSMRegisterEx(pDevIns, PCNET_SAVEDSTATE_VERSION, sizeof(*pThis), NULL,
5204 NULL, pcnetLiveExec, NULL,
5205 pcnetSavePrep, pcnetSaveExec, NULL,
5206 pcnetLoadPrep, pcnetLoadExec, NULL);
5207 if (RT_FAILURE(rc))
5208 return rc;
5209
5210 /*
5211 * Create the transmit queue.
5212 */
5213 rc = PDMDevHlpQueueCreate(pDevIns, sizeof(PDMQUEUEITEMCORE), 1, 0,
5214 pcnetXmitQueueConsumer, true, "PCNet-Xmit", &pThis->pXmitQueueR3);
5215 if (RT_FAILURE(rc))
5216 return rc;
5217 pThis->pXmitQueueR0 = PDMQueueR0Ptr(pThis->pXmitQueueR3);
5218 pThis->pXmitQueueRC = PDMQueueRCPtr(pThis->pXmitQueueR3);
5219
5220 /*
5221 * Create the RX notifier signaller.
5222 */
5223 rc = PDMDevHlpQueueCreate(pDevIns, sizeof(PDMQUEUEITEMCORE), 1, 0,
5224 pcnetCanRxQueueConsumer, true, "PCNet-Rcv", &pThis->pCanRxQueueR3);
5225 if (RT_FAILURE(rc))
5226 return rc;
5227 pThis->pCanRxQueueR0 = PDMQueueR0Ptr(pThis->pCanRxQueueR3);
5228 pThis->pCanRxQueueRC = PDMQueueRCPtr(pThis->pCanRxQueueR3);
5229
5230 /*
5231 * Register the info item.
5232 */
5233 RTStrPrintf(szTmp, sizeof(szTmp), "pcnet%d", pDevIns->iInstance);
5234 PDMDevHlpDBGFInfoRegister(pDevIns, szTmp, "PCNET info.", pcnetInfo);
5235
5236 /*
5237 * Attach status driver (optional).
5238 */
5239 rc = PDMDevHlpDriverAttach(pDevIns, PDM_STATUS_LUN, &pThis->IBase, &pBase, "Status Port");
5240 if (RT_SUCCESS(rc))
5241 pThis->pLedsConnector = PDMIBASE_QUERY_INTERFACE(pBase, PDMILEDCONNECTORS);
5242 else if ( rc != VERR_PDM_NO_ATTACHED_DRIVER
5243 && rc != VERR_PDM_CFG_MISSING_DRIVER_NAME)
5244 {
5245 AssertMsgFailed(("Failed to attach to status driver. rc=%Rrc\n", rc));
5246 return rc;
5247 }
5248
5249 /*
5250 * Attach driver.
5251 */
5252 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThis->IBase, &pThis->pDrvBase, "Network Port");
5253 if (RT_SUCCESS(rc))
5254 {
5255 if (rc == VINF_NAT_DNS)
5256 {
5257#ifdef RT_OS_LINUX
5258 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "NoDNSforNAT",
5259 N_("A Domain Name Server (DNS) for NAT networking could not be determined. Please check your /etc/resolv.conf for <tt>nameserver</tt> entries. Either add one manually (<i>man resolv.conf</i>) or ensure that your host is correctly connected to an ISP. If you ignore this warning the guest will not be able to perform nameserver lookups and it will probably observe delays if trying so"));
5260#else
5261 PDMDevHlpVMSetRuntimeError(pDevIns, 0 /*fFlags*/, "NoDNSforNAT",
5262 N_("A Domain Name Server (DNS) for NAT networking could not be determined. Ensure that your host is correctly connected to an ISP. If you ignore this warning the guest will not be able to perform nameserver lookups and it will probably observe delays if trying so"));
5263#endif
5264 }
5265 pThis->pDrvR3 = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMINETWORKUP);
5266 AssertMsgReturn(pThis->pDrvR3, ("Failed to obtain the PDMINETWORKUP interface!\n"),
5267 VERR_PDM_MISSING_INTERFACE_BELOW);
5268 pThis->pDrvR0 = PDMIBASER0_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBASER0), PDMINETWORKUP);
5269 pThis->pDrvRC = PDMIBASERC_QUERY_INTERFACE(PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMIBASERC), PDMINETWORKUP);
5270 }
5271 else if ( rc == VERR_PDM_NO_ATTACHED_DRIVER
5272 || rc == VERR_PDM_CFG_MISSING_DRIVER_NAME)
5273 {
5274 /* No error! */
5275 Log(("No attached driver!\n"));
5276 }
5277 else
5278 return rc;
5279
5280 /*
5281 * Reset the device state. (Do after attaching.)
5282 */
5283 pcnetHardReset(pThis);
5284
5285#ifdef VBOX_WITH_STATISTICS
5286 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatMMIOReadRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling MMIO reads in RZ", "/Devices/PCNet%d/MMIO/ReadRZ", iInstance);
5287 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatMMIOReadR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling MMIO reads in R3", "/Devices/PCNet%d/MMIO/ReadR3", iInstance);
5288 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatMMIOWriteRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling MMIO writes in RZ", "/Devices/PCNet%d/MMIO/WriteRZ", iInstance);
5289 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatMMIOWriteR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling MMIO writes in R3", "/Devices/PCNet%d/MMIO/WriteR3", iInstance);
5290 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatAPROMRead, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling APROM reads", "/Devices/PCNet%d/IO/APROMRead", iInstance);
5291 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatAPROMWrite, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling APROM writes", "/Devices/PCNet%d/IO/APROMWrite", iInstance);
5292 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOReadRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in RZ", "/Devices/PCNet%d/IO/ReadRZ", iInstance);
5293 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOReadR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO reads in R3", "/Devices/PCNet%d/IO/ReadR3", iInstance);
5294 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOWriteRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in RZ", "/Devices/PCNet%d/IO/WriteRZ", iInstance);
5295 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatIOWriteR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling IO writes in R3", "/Devices/PCNet%d/IO/WriteR3", iInstance);
5296 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTimer, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling Timer", "/Devices/PCNet%d/Timer", iInstance);
5297 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceive, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling receive", "/Devices/PCNet%d/Receive", iInstance);
5298 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRxOverflow, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_OCCURENCE, "Profiling RX overflows", "/Devices/PCNet%d/RxOverflow", iInstance);
5299 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRxOverflowWakeup, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_OCCURENCE, "Nr of RX overflow wakeups", "/Devices/PCNet%d/RxOverflowWakeup", iInstance);
5300#endif
5301 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatReceiveBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Amount of data received", "/Devices/PCNet%d/ReceiveBytes", iInstance);
5302#ifdef VBOX_WITH_STATISTICS
5303 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitCase1, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Single descriptor transmit", "/Devices/PCNet%d/Transmit/Case1", iInstance);
5304 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitCase2, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Multi descriptor transmit", "/Devices/PCNet%d/Transmit/Case2", iInstance);
5305 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling transmits in RZ", "/Devices/PCNet%d/Transmit/TotalRZ", iInstance);
5306 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling transmits in R3", "/Devices/PCNet%d/Transmit/TotalR3", iInstance);
5307#endif
5308 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Amount of data transmitted", "/Devices/PCNet%d/TransmitBytes", iInstance);
5309#ifdef VBOX_WITH_STATISTICS
5310 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitSendRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet send transmit in RZ","/Devices/PCNet%d/Transmit/SendRZ", iInstance);
5311 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTransmitSendR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet send transmit in R3","/Devices/PCNet%d/Transmit/SendR3", iInstance);
5312 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTdtePollRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet TdtePoll in RZ", "/Devices/PCNet%d/TdtePollRZ", iInstance);
5313 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTdtePollR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet TdtePoll in R3", "/Devices/PCNet%d/TdtePollR3", iInstance);
5314 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRdtePollRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet RdtePoll in RZ", "/Devices/PCNet%d/RdtePollRZ", iInstance);
5315 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRdtePollR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet RdtePoll in R3", "/Devices/PCNet%d/RdtePollR3", iInstance);
5316
5317 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTmdStoreRZ, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet TmdStore in RZ", "/Devices/PCNet%d/TmdStoreRZ", iInstance);
5318 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTmdStoreR3, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling PCNet TmdStore in R3", "/Devices/PCNet%d/TmdStoreR3", iInstance);
5319
5320 unsigned i;
5321 for (i = 0; i < RT_ELEMENTS(pThis->aStatXmitFlush) - 1; i++)
5322 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatXmitFlush[i], STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_OCCURENCES, "", "/Devices/PCNet%d/XmitFlushIrq/%d", iInstance, i + 1);
5323 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatXmitFlush[i], STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_OCCURENCES, "", "/Devices/PCNet%d/XmitFlushIrq/%d+", iInstance, i + 1);
5324
5325 for (i = 0; i < RT_ELEMENTS(pThis->aStatXmitChainCounts) - 1; i++)
5326 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatXmitChainCounts[i], STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_OCCURENCES, "", "/Devices/PCNet%d/XmitChainCounts/%d", iInstance, i + 1);
5327 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->aStatXmitChainCounts[i], STAMTYPE_COUNTER, STAMVISIBILITY_USED, STAMUNIT_OCCURENCES, "", "/Devices/PCNet%d/XmitChainCounts/%d+", iInstance, i + 1);
5328
5329 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatXmitSkipCurrent, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "", "/Devices/PCNet%d/Xmit/Skipped", iInstance, i + 1);
5330
5331 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatInterrupt, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling interrupt checks", "/Devices/PCNet%d/UpdateIRQ", iInstance);
5332 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatPollTimer, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling poll timer", "/Devices/PCNet%d/PollTimer", iInstance);
5333 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatMIIReads, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of MII reads", "/Devices/PCNet%d/MIIReads", iInstance);
5334# ifdef PCNET_NO_POLLING
5335 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRCVRingWrite, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of receive ring writes", "/Devices/PCNet%d/Ring/RCVWrites", iInstance);
5336 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatTXRingWrite, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of transmit ring writes", "/Devices/PCNet%d/Ring/TXWrites", iInstance);
5337 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteR3, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of monitored ring page writes", "/Devices/PCNet%d/Ring/R3/Writes", iInstance);
5338 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteR0, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of monitored ring page writes", "/Devices/PCNet%d/Ring/R0/Writes", iInstance);
5339 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteRC, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of monitored ring page writes", "/Devices/PCNet%d/Ring/RC/Writes", iInstance);
5340 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteFailedR3, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of failed ring page writes", "/Devices/PCNet%d/Ring/R3/Failed", iInstance);
5341 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteFailedR0, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of failed ring page writes", "/Devices/PCNet%d/Ring/R0/Failed", iInstance);
5342 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteFailedRC, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of failed ring page writes", "/Devices/PCNet%d/Ring/RC/Failed", iInstance);
5343 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteOutsideR3, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of monitored writes outside ring","/Devices/PCNet%d/Ring/R3/Outside", iInstance);
5344 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteOutsideR0, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of monitored writes outside ring","/Devices/PCNet%d/Ring/R0/Outside", iInstance);
5345 PDMDevHlpSTAMRegisterF(pDevIns, &pThis->StatRingWriteOutsideRC, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Nr of monitored writes outside ring","/Devices/PCNet%d/Ring/RC/Outside", iInstance);
5346# endif /* PCNET_NO_POLLING */
5347#endif
5348
5349 return VINF_SUCCESS;
5350}
5351
5352
5353/**
5354 * The device registration structure.
5355 */
5356const PDMDEVREG g_DevicePCNet =
5357{
5358 /* u32Version */
5359 PDM_DEVREG_VERSION,
5360 /* szName */
5361 "pcnet",
5362 /* szRCMod */
5363#ifdef PCNET_GC_ENABLED
5364 "VBoxDDGC.gc",
5365 "VBoxDDR0.r0",
5366#else
5367 "",
5368 "",
5369#endif
5370 /* pszDescription */
5371 "AMD PC-Net II Ethernet controller.\n",
5372 /* fFlags */
5373#ifdef PCNET_GC_ENABLED
5374 PDM_DEVREG_FLAGS_DEFAULT_BITS | PDM_DEVREG_FLAGS_RC | PDM_DEVREG_FLAGS_R0,
5375#else
5376 PDM_DEVREG_FLAGS_DEFAULT_BITS,
5377#endif
5378 /* fClass */
5379 PDM_DEVREG_CLASS_NETWORK,
5380 /* cMaxInstances */
5381 ~0U,
5382 /* cbInstance */
5383 sizeof(PCNetState),
5384 /* pfnConstruct */
5385 pcnetConstruct,
5386 /* pfnDestruct */
5387 pcnetDestruct,
5388 /* pfnRelocate */
5389 pcnetRelocate,
5390 /* pfnIOCtl */
5391 NULL,
5392 /* pfnPowerOn */
5393 NULL,
5394 /* pfnReset */
5395 pcnetReset,
5396 /* pfnSuspend */
5397 pcnetSuspend,
5398 /* pfnResume */
5399 NULL,
5400 /* pfnAttach */
5401 pcnetAttach,
5402 /* pfnDetach */
5403 pcnetDetach,
5404 /* pfnQueryInterface. */
5405 NULL,
5406 /* pfnInitComplete. */
5407 NULL,
5408 /* pfnPowerOff. */
5409 pcnetPowerOff,
5410 /* pfnSoftReset */
5411 NULL,
5412 /* u32VersionEnd */
5413 PDM_DEVREG_VERSION
5414};
5415
5416#endif /* IN_RING3 */
5417#endif /* !VBOX_DEVICE_STRUCT_TESTCASE */
5418
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