VirtualBox

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

Last change on this file since 51150 was 49813, checked in by vboxsync, 11 years ago

DevPCNet: register a dummy MMIO2 region to be compatible to old saved states but drop it later

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