VirtualBox

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

Last change on this file since 57009 was 57006, checked in by vboxsync, 9 years ago

VMM,*: Annotated format strings in the VMM APIs and dealt with the fallout.

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