VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DrvIntNet.cpp@ 29603

Last change on this file since 29603 was 29598, checked in by vboxsync, 15 years ago

warning

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 62.5 KB
Line 
1/* $Id: DrvIntNet.cpp 29598 2010-05-18 08:32:49Z vboxsync $ */
2/** @file
3 * DrvIntNet - Internal network transport driver.
4 */
5
6/*
7 * Copyright (C) 2006-2010 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_DRV_INTNET
22#include <VBox/pdmdrv.h>
23#include <VBox/pdmnetinline.h>
24#include <VBox/pdmnetifs.h>
25#include <VBox/cfgm.h>
26#include <VBox/intnet.h>
27#include <VBox/intnetinline.h>
28#include <VBox/vmm.h>
29#include <VBox/sup.h>
30#include <VBox/err.h>
31
32#include <VBox/param.h>
33#include <VBox/log.h>
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/ctype.h>
37#include <iprt/memcache.h>
38#include <iprt/net.h>
39#include <iprt/semaphore.h>
40#include <iprt/string.h>
41#include <iprt/time.h>
42#include <iprt/thread.h>
43#include <iprt/uuid.h>
44
45#include "../Builtins.h"
46
47
48/*******************************************************************************
49* Defined Constants And Macros *
50*******************************************************************************/
51/** Enables the ring-0 part. */
52#define VBOX_WITH_DRVINTNET_IN_R0
53
54
55/*******************************************************************************
56* Structures and Typedefs *
57*******************************************************************************/
58/**
59 * The state of the asynchronous thread.
60 */
61typedef enum RECVSTATE
62{
63 /** The thread is suspended. */
64 RECVSTATE_SUSPENDED = 1,
65 /** The thread is running. */
66 RECVSTATE_RUNNING,
67 /** The thread must (/has) terminate. */
68 RECVSTATE_TERMINATE,
69 /** The usual 32-bit type blowup. */
70 RECVSTATE_32BIT_HACK = 0x7fffffff
71} RECVSTATE;
72
73/**
74 * Internal networking driver instance data.
75 *
76 * @implements PDMINETWORKUP
77 */
78typedef struct DRVINTNET
79{
80 /** The network interface. */
81 PDMINETWORKUP INetworkUpR3;
82 /** The network interface. */
83 R3PTRTYPE(PPDMINETWORKDOWN) pIAboveNet;
84 /** The network config interface.
85 * Can (in theory at least) be NULL. */
86 R3PTRTYPE(PPDMINETWORKCONFIG) pIAboveConfigR3;
87 /** Pointer to the driver instance (ring-3). */
88 PPDMDRVINSR3 pDrvInsR3;
89 /** Pointer to the communication buffer (ring-3). */
90 R3PTRTYPE(PINTNETBUF) pBufR3;
91 /** Ring-3 base interface for the ring-0 context. */
92 PDMIBASER0 IBaseR0;
93 /** Ring-3 base interface for the raw-mode context. */
94 PDMIBASERC IBaseRC;
95 RTR3PTR R3PtrAlignment;
96
97 /** The network interface for the ring-0 context. */
98 PDMINETWORKUPR0 INetworkUpR0;
99 /** Pointer to the driver instance (ring-0). */
100 PPDMDRVINSR0 pDrvInsR0;
101 /** Pointer to the communication buffer (ring-0). */
102 R0PTRTYPE(PINTNETBUF) pBufR0;
103
104 /** The network interface for the raw-mode context. */
105 PDMINETWORKUPRC INetworkUpRC;
106 /** Pointer to the driver instance. */
107 PPDMDRVINSRC pDrvInsRC;
108 RTRCPTR RCPtrAlignment;
109
110 /** The transmit lock. */
111 PDMCRITSECT XmitLock;
112 /** Interface handle. */
113 INTNETIFHANDLE hIf;
114 /** The receive thread state. */
115 RECVSTATE volatile enmRecvState;
116 /** The receive thread. */
117 RTTHREAD hRecvThread;
118 /** The event semaphore that the receive thread waits on. */
119 RTSEMEVENT hRecvEvt;
120 /** The transmit thread. */
121 PPDMTHREAD pXmitThread;
122 /** The event semaphore that the transmit thread waits on. */
123 SUPSEMEVENT hXmitEvt;
124 /** The support driver session handle. */
125 PSUPDRVSESSION pSupDrvSession;
126 /** Scatter/gather descriptor cache. */
127 RTMEMCACHE hSgCache;
128 /** Set if the link is down.
129 * When the link is down all incoming packets will be dropped. */
130 bool volatile fLinkDown;
131 /** Set when the xmit thread has been signalled. (atomic) */
132 bool volatile fXmitSignalled;
133 /** Set if the transmit thread the one busy transmitting. */
134 bool volatile fXmitOnXmitThread;
135 /** The xmit thread should process the ring ASAP. */
136 bool fXmitProcessRing;
137 /** Set if data transmission should start immediately and deactivate
138 * as late as possible. */
139 bool fActivateEarlyDeactivateLate;
140 /** Padding. */
141 bool afReserved[HC_ARCH_BITS == 64 ? 3 : 3];
142 /** Scratch space for holding the ring-0 scatter / gather descriptor.
143 * The PDMSCATTERGATHER::fFlags member is used to indicate whether it is in
144 * use or not. Always accessed while owning the XmitLock. */
145 union
146 {
147 PDMSCATTERGATHER Sg;
148 uint8_t padding[8 * sizeof(RTUINTPTR)];
149 } u;
150 /** The network name. */
151 char szNetwork[INTNET_MAX_NETWORK_NAME];
152
153 /** Number of GSO packets sent. */
154 STAMCOUNTER StatSentGso;
155 /** Number of GSO packets recevied. */
156 STAMCOUNTER StatReceivedGso;
157 /** Number of packets send from ring-0. */
158 STAMCOUNTER StatSentR0;
159 /** The number of times we've had to wake up the xmit thread to contine the
160 * ring-0 job. */
161 STAMCOUNTER StatXmitWakeupR0;
162 /** The number of times we've had to wake up the xmit thread to contine the
163 * ring-3 job. */
164 STAMCOUNTER StatXmitWakeupR3;
165 /** The times the xmit thread has been told to process the ring. */
166 STAMCOUNTER StatXmitProcessRing;
167#ifdef VBOX_WITH_STATISTICS
168 /** Profiling packet transmit runs. */
169 STAMPROFILE StatTransmit;
170 /** Profiling packet receive runs. */
171 STAMPROFILEADV StatReceive;
172#endif /* VBOX_WITH_STATISTICS */
173#ifdef LOG_ENABLED
174 /** The nano ts of the last transfer. */
175 uint64_t u64LastTransferTS;
176 /** The nano ts of the last receive. */
177 uint64_t u64LastReceiveTS;
178#endif
179} DRVINTNET;
180AssertCompileMemberAlignment(DRVINTNET, XmitLock, 8);
181AssertCompileMemberAlignment(DRVINTNET, StatSentGso, 8);
182/** Pointer to instance data of the internal networking driver. */
183typedef DRVINTNET *PDRVINTNET;
184
185
186#ifdef IN_RING3
187
188
189/**
190 * Updates the MAC address on the kernel side.
191 *
192 * @returns VBox status code.
193 * @param pThis The driver instance.
194 */
195static int drvR3IntNetUpdateMacAddress(PDRVINTNET pThis)
196{
197 if (!pThis->pIAboveConfigR3)
198 return VINF_SUCCESS;
199
200 INTNETIFSETMACADDRESSREQ SetMacAddressReq;
201 SetMacAddressReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
202 SetMacAddressReq.Hdr.cbReq = sizeof(SetMacAddressReq);
203 SetMacAddressReq.pSession = NIL_RTR0PTR;
204 SetMacAddressReq.hIf = pThis->hIf;
205 int rc = pThis->pIAboveConfigR3->pfnGetMac(pThis->pIAboveConfigR3, &SetMacAddressReq.Mac);
206 if (RT_SUCCESS(rc))
207 rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SET_MAC_ADDRESS,
208 &SetMacAddressReq, sizeof(SetMacAddressReq));
209
210 Log(("drvR3IntNetUpdateMacAddress: %.*Rhxs rc=%Rrc\n", sizeof(SetMacAddressReq.Mac), &SetMacAddressReq.Mac, rc));
211 return rc;
212}
213
214
215/**
216 * Sets the kernel interface active or inactive.
217 *
218 * Worker for poweron, poweroff, suspend and resume.
219 *
220 * @returns VBox status code.
221 * @param pThis The driver instance.
222 * @param fActive The new state.
223 */
224static int drvR3IntNetSetActive(PDRVINTNET pThis, bool fActive)
225{
226 if (!pThis->pIAboveConfigR3)
227 return VINF_SUCCESS;
228
229 INTNETIFSETACTIVEREQ SetActiveReq;
230 SetActiveReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
231 SetActiveReq.Hdr.cbReq = sizeof(SetActiveReq);
232 SetActiveReq.pSession = NIL_RTR0PTR;
233 SetActiveReq.hIf = pThis->hIf;
234 SetActiveReq.fActive = fActive;
235 int rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SET_ACTIVE,
236 &SetActiveReq, sizeof(SetActiveReq));
237
238 Log(("drvR3IntNetSetActive: fActive=%d rc=%Rrc\n", fActive, rc));
239 AssertRC(rc);
240 return rc;
241}
242
243#endif /* IN_RING3 */
244
245/* -=-=-=-=- PDMINETWORKUP -=-=-=-=- */
246
247/**
248 * Helper for signalling the xmit thread.
249 *
250 * @returns VERR_TRY_AGAIN (convenience).
251 * @param pThis The instance data..
252 */
253DECLINLINE(int) drvIntNetSignalXmit(PDRVINTNET pThis)
254{
255 /// @todo if (!ASMAtomicXchgBool(&pThis->fXmitSignalled, true)) - needs careful optimizing.
256 {
257 int rc = SUPSemEventSignal(pThis->pSupDrvSession, pThis->hXmitEvt);
258 AssertRC(rc);
259 STAM_REL_COUNTER_INC(&pThis->CTX_SUFF(StatXmitWakeup));
260 }
261 return VERR_TRY_AGAIN;
262}
263
264
265/**
266 * Helper for processing the ring-0 consumer side of the xmit ring.
267 *
268 * The caller MUST own the xmit lock.
269 *
270 * @returns Status code from IntNetR0IfSend, except for VERR_TRY_AGAIN.
271 * @param pThis The instance data..
272 */
273DECLINLINE(int) drvIntNetProcessXmit(PDRVINTNET pThis)
274{
275 Assert(PDMCritSectIsOwner(&pThis->XmitLock));
276
277#ifdef IN_RING3
278 INTNETIFSENDREQ SendReq;
279 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
280 SendReq.Hdr.cbReq = sizeof(SendReq);
281 SendReq.pSession = NIL_RTR0PTR;
282 SendReq.hIf = pThis->hIf;
283 int rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
284#else
285 int rc = IntNetR0IfSend(pThis->hIf, pThis->pSupDrvSession);
286 if (rc == VERR_TRY_AGAIN)
287 {
288 ASMAtomicUoWriteBool(&pThis->fXmitProcessRing, true);
289 drvIntNetSignalXmit(pThis);
290 rc = VINF_SUCCESS;
291 }
292#endif
293 return rc;
294}
295
296
297
298/**
299 * @interface_method_impl{PDMINETWORKUP,pfnBeginXmit}
300 */
301PDMBOTHCBDECL(int) drvIntNetUp_BeginXmit(PPDMINETWORKUP pInterface, bool fOnWorkerThread)
302{
303 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
304#ifndef IN_RING3
305 Assert(!fOnWorkerThread);
306#endif
307
308 int rc = PDMCritSectTryEnter(&pThis->XmitLock);
309 if (RT_SUCCESS(rc))
310 {
311 if (fOnWorkerThread)
312 {
313 ASMAtomicUoWriteBool(&pThis->fXmitOnXmitThread, true);
314 ASMAtomicWriteBool(&pThis->fXmitSignalled, false);
315 }
316 }
317 else if (rc == VERR_SEM_BUSY)
318 {
319 /** @todo Does this actually make sense if the other dude is an EMT and so
320 * forth? I seriously think this is ring-0 only...
321 * We might end up waking up the xmit thread unnecessarily here, even when in
322 * ring-0... This needs some more thought and opitmizations when the ring-0 bits
323 * are working. */
324#ifdef IN_RING3
325 if ( !fOnWorkerThread
326 /*&& !ASMAtomicUoReadBool(&pThis->fXmitOnXmitThread)
327 && ASMAtomicCmpXchgBool(&pThis->fXmitSignalled, true, false)*/)
328 {
329 rc = SUPSemEventSignal(pThis->pSupDrvSession, pThis->hXmitEvt);
330 AssertRC(rc);
331 }
332 rc = VERR_TRY_AGAIN;
333#else /* IN_RING0 */
334 rc = drvIntNetSignalXmit(pThis);
335#endif /* IN_RING0 */
336 }
337 return rc;
338}
339
340
341/**
342 * @interface_method_impl{PDMINETWORKUP,pfnAllocBuf}
343 */
344PDMBOTHCBDECL(int) drvIntNetUp_AllocBuf(PPDMINETWORKUP pInterface, size_t cbMin,
345 PCPDMNETWORKGSO pGso, PPPDMSCATTERGATHER ppSgBuf)
346{
347 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
348 int rc = VINF_SUCCESS;
349 Assert(cbMin < UINT32_MAX / 2);
350 Assert(PDMCritSectIsOwner(&pThis->XmitLock));
351
352 /*
353 * Allocate a S/G descriptor.
354 * This shouldn't normally fail as the NICs usually won't allocate more
355 * than one buffer at a time and the SG gets freed on sending.
356 */
357#ifdef IN_RING3
358 PPDMSCATTERGATHER pSgBuf = (PPDMSCATTERGATHER)RTMemCacheAlloc(pThis->hSgCache);
359 if (!pSgBuf)
360 return VERR_NO_MEMORY;
361#else
362 PPDMSCATTERGATHER pSgBuf = &pThis->u.Sg;
363 if (RT_UNLIKELY(pSgBuf->fFlags != 0))
364 return drvIntNetSignalXmit(pThis);
365#endif
366
367 /*
368 * Allocate room in the ring buffer.
369 *
370 * In ring-3 we may have to process the xmit ring before there is
371 * sufficient buffer space since we might've stacked up a few frames to the
372 * trunk while in ring-0. (There is not point of doing this in ring-0.)
373 */
374 PINTNETHDR pHdr = NULL; /* gcc silliness */
375 if (pGso)
376 rc = IntNetRingAllocateGsoFrame(&pThis->CTX_SUFF(pBuf)->Send, (uint32_t)cbMin, pGso,
377 &pHdr, &pSgBuf->aSegs[0].pvSeg);
378 else
379 rc = IntNetRingAllocateFrame(&pThis->CTX_SUFF(pBuf)->Send, (uint32_t)cbMin,
380 &pHdr, &pSgBuf->aSegs[0].pvSeg);
381#ifdef IN_RING3
382 if ( RT_FAILURE(rc)
383 && pThis->CTX_SUFF(pBuf)->cbSend >= cbMin * 2 + sizeof(INTNETHDR))
384 {
385 drvIntNetProcessXmit(pThis);
386 if (pGso)
387 rc = IntNetRingAllocateGsoFrame(&pThis->CTX_SUFF(pBuf)->Send, (uint32_t)cbMin, pGso,
388 &pHdr, &pSgBuf->aSegs[0].pvSeg);
389 else
390 rc = IntNetRingAllocateFrame(&pThis->CTX_SUFF(pBuf)->Send, (uint32_t)cbMin,
391 &pHdr, &pSgBuf->aSegs[0].pvSeg);
392 }
393#endif
394 if (RT_SUCCESS(rc))
395 {
396 /*
397 * Set up the S/G descriptor and return successfully.
398 */
399 pSgBuf->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1;
400 pSgBuf->cbUsed = 0;
401 pSgBuf->cbAvailable = cbMin;
402 pSgBuf->pvAllocator = pHdr;
403 pSgBuf->pvUser = pGso ? (PPDMNETWORKGSO)pSgBuf->aSegs[0].pvSeg - 1 : NULL;
404 pSgBuf->cSegs = 1;
405 pSgBuf->aSegs[0].cbSeg = cbMin;
406
407 *ppSgBuf = pSgBuf;
408 return VINF_SUCCESS;
409 }
410
411#ifdef IN_RING3
412 /*
413 * If the above fails, then we're really out of space. There are nobody
414 * competing with us here because of the xmit lock.
415 */
416 rc = VERR_NO_MEMORY;
417 RTMemCacheFree(pThis->hSgCache, pSgBuf);
418
419#else /* IN_RING0 */
420 /*
421 * If the request is reasonable, kick the xmit thread and tell it to
422 * process the xmit ring ASAP.
423 */
424 if (pThis->CTX_SUFF(pBuf)->cbSend >= cbMin * 2 + sizeof(INTNETHDR))
425 {
426 pThis->fXmitProcessRing = true;
427 rc = drvIntNetSignalXmit(pThis);
428 }
429 else
430 rc = VERR_NO_MEMORY;
431 pSgBuf->fFlags = 0;
432#endif /* IN_RING0 */
433 return rc;
434}
435
436
437/**
438 * @interface_method_impl{PDMINETWORKUP,pfnFreeBuf}
439 */
440PDMBOTHCBDECL(int) drvIntNetUp_FreeBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf)
441{
442 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
443 PINTNETHDR pHdr = (PINTNETHDR)pSgBuf->pvAllocator;
444#ifdef IN_RING0
445 Assert(pSgBuf == &pThis->u.Sg);
446#endif
447 Assert(pSgBuf->fFlags == (PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1));
448 Assert(pSgBuf->cbUsed <= pSgBuf->cbAvailable);
449 Assert( pHdr->u16Type == INTNETHDR_TYPE_FRAME
450 || pHdr->u16Type == INTNETHDR_TYPE_GSO);
451 Assert(PDMCritSectIsOwner(&pThis->XmitLock));
452
453 /** @todo LATER: try unalloc the frame. */
454 pHdr->u16Type = INTNETHDR_TYPE_PADDING;
455 IntNetRingCommitFrame(&pThis->CTX_SUFF(pBuf)->Send, pHdr);
456
457#ifdef IN_RING3
458 RTMemCacheFree(pThis->hSgCache, pSgBuf);
459#else
460 pSgBuf->fFlags = 0;
461#endif
462 return VINF_SUCCESS;
463}
464
465
466/**
467 * @interface_method_impl{PDMINETWORKUP,pfnSendBuf}
468 */
469PDMBOTHCBDECL(int) drvIntNetUp_SendBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf, bool fOnWorkerThread)
470{
471 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
472 STAM_PROFILE_START(&pThis->StatTransmit, a);
473
474 AssertPtr(pSgBuf);
475 Assert(pSgBuf->fFlags == (PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1));
476 Assert(pSgBuf->cbUsed <= pSgBuf->cbAvailable);
477 Assert(PDMCritSectIsOwner(&pThis->XmitLock));
478
479 if (pSgBuf->pvUser)
480 STAM_COUNTER_INC(&pThis->StatSentGso);
481
482 /*
483 * Commit the frame and push it thru the switch.
484 */
485 PINTNETHDR pHdr = (PINTNETHDR)pSgBuf->pvAllocator;
486 IntNetRingCommitFrameEx(&pThis->CTX_SUFF(pBuf)->Send, pHdr, pSgBuf->cbUsed);
487 int rc = drvIntNetProcessXmit(pThis);
488 STAM_PROFILE_STOP(&pThis->StatTransmit, a);
489
490 /*
491 * Free the descriptor and return.
492 */
493#ifdef IN_RING3
494 RTMemCacheFree(pThis->hSgCache, pSgBuf);
495#else
496 STAM_REL_COUNTER_INC(&pThis->StatSentR0);
497 pSgBuf->fFlags = 0;
498#endif
499 return rc;
500}
501
502
503/**
504 * @interface_method_impl{PDMINETWORKUP,pfnEndXmit}
505 */
506PDMBOTHCBDECL(void) drvIntNetUp_EndXmit(PPDMINETWORKUP pInterface)
507{
508 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
509 ASMAtomicUoWriteBool(&pThis->fXmitOnXmitThread, false);
510 PDMCritSectLeave(&pThis->XmitLock);
511}
512
513
514/**
515 * @interface_method_impl{PDMINETWORKUP,pfnSetPromiscuousMode}
516 */
517PDMBOTHCBDECL(void) drvIntNetUp_SetPromiscuousMode(PPDMINETWORKUP pInterface, bool fPromiscuous)
518{
519 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
520
521#ifdef IN_RING3
522 INTNETIFSETPROMISCUOUSMODEREQ Req;
523 Req.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
524 Req.Hdr.cbReq = sizeof(Req);
525 Req.pSession = NIL_RTR0PTR;
526 Req.hIf = pThis->hIf;
527 Req.fPromiscuous = fPromiscuous;
528 int rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SET_PROMISCUOUS_MODE, &Req, sizeof(Req));
529#else /* IN_RING0 */
530 int rc = IntNetR0IfSetPromiscuousMode(pThis->hIf, pThis->pSupDrvSession, fPromiscuous);
531#endif /* IN_RING0 */
532
533 LogFlow(("drvIntNetUp_SetPromiscuousMode: fPromiscuous=%RTbool\n", fPromiscuous));
534 AssertRC(rc);
535}
536
537#ifdef IN_RING3
538
539/**
540 * @interface_method_impl{PDMINETWORKUP,pfnNotifyLinkChanged}
541 */
542static DECLCALLBACK(void) drvR3IntNetUp_NotifyLinkChanged(PPDMINETWORKUP pInterface, PDMNETWORKLINKSTATE enmLinkState)
543{
544 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, CTX_SUFF(INetworkUp));
545 bool fLinkDown;
546 switch (enmLinkState)
547 {
548 case PDMNETWORKLINKSTATE_DOWN:
549 case PDMNETWORKLINKSTATE_DOWN_RESUME:
550 fLinkDown = true;
551 break;
552 default:
553 AssertMsgFailed(("enmLinkState=%d\n", enmLinkState));
554 case PDMNETWORKLINKSTATE_UP:
555 fLinkDown = false;
556 break;
557 }
558 LogFlow(("drvR3IntNetUp_NotifyLinkChanged: enmLinkState=%d %d->%d\n", enmLinkState, pThis->fLinkDown, fLinkDown));
559 ASMAtomicXchgSize(&pThis->fLinkDown, fLinkDown);
560}
561
562
563/* -=-=-=-=- Transmit Thread -=-=-=-=- */
564
565/**
566 * Async I/O thread for defered packet transmission.
567 *
568 * @returns VBox status code. Returning failure will naturally terminate the thread.
569 * @param pDrvIns The internal networking driver instance.
570 * @param pThread The thread.
571 */
572static DECLCALLBACK(int) drvR3IntNetXmitThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
573{
574 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
575
576 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
577 {
578 /*
579 * Transmit any pending packets.
580 */
581 /** @todo Optimize this. We shouldn't call pfnXmitPending unless asked for.
582 * Also there is no need to call drvIntNetProcessXmit if we also
583 * called pfnXmitPending and send one or more frames. */
584 if (ASMAtomicXchgBool(&pThis->fXmitProcessRing, false))
585 {
586 STAM_REL_COUNTER_INC(&pThis->StatXmitProcessRing);
587 PDMCritSectEnter(&pThis->XmitLock, VERR_IGNORED);
588 drvIntNetProcessXmit(pThis);
589 PDMCritSectLeave(&pThis->XmitLock);
590 }
591
592 pThis->pIAboveNet->pfnXmitPending(pThis->pIAboveNet);
593
594 if (ASMAtomicXchgBool(&pThis->fXmitProcessRing, false))
595 {
596 STAM_REL_COUNTER_INC(&pThis->StatXmitProcessRing);
597 PDMCritSectEnter(&pThis->XmitLock, VERR_IGNORED);
598 drvIntNetProcessXmit(pThis);
599 PDMCritSectLeave(&pThis->XmitLock);
600 }
601
602 /*
603 * Block until we've got something to send or is supposed
604 * to leave the running state.
605 */
606 int rc = SUPSemEventWaitNoResume(pThis->pSupDrvSession, pThis->hXmitEvt, RT_INDEFINITE_WAIT);
607 AssertLogRelMsgReturn(RT_SUCCESS(rc) || rc == VERR_INTERRUPTED, ("%Rrc\n", rc), rc);
608 if (RT_UNLIKELY(pThread->enmState != PDMTHREADSTATE_RUNNING))
609 break;
610
611 }
612
613 /* The thread is being initialized, suspended or terminated. */
614 return VINF_SUCCESS;
615}
616
617
618/**
619 * @copydoc FNPDMTHREADWAKEUPDRV
620 */
621static DECLCALLBACK(int) drvR3IntNetXmitWakeUp(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
622{
623 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
624 return SUPSemEventSignal(pThis->pSupDrvSession, pThis->hXmitEvt);
625}
626
627
628/* -=-=-=-=- Receive Thread -=-=-=-=- */
629
630/**
631 * Wait for space to become available up the driver/device chain.
632 *
633 * @returns VINF_SUCCESS if space is available.
634 * @returns VERR_STATE_CHANGED if the state changed.
635 * @returns VBox status code on other errors.
636 * @param pThis Pointer to the instance data.
637 */
638static int drvR3IntNetRecvWaitForSpace(PDRVINTNET pThis)
639{
640 LogFlow(("drvR3IntNetRecvWaitForSpace:\n"));
641 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
642 int rc = pThis->pIAboveNet->pfnWaitReceiveAvail(pThis->pIAboveNet, RT_INDEFINITE_WAIT);
643 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
644 LogFlow(("drvR3IntNetRecvWaitForSpace: returns %Rrc\n", rc));
645 return rc;
646}
647
648
649/**
650 * Executes async I/O (RUNNING mode).
651 *
652 * @returns VERR_STATE_CHANGED if the state changed.
653 * @returns Appropriate VBox status code (error) on fatal error.
654 * @param pThis The driver instance data.
655 */
656static int drvR3IntNetRecvRun(PDRVINTNET pThis)
657{
658 PPDMDRVINS pDrvIns = pThis->pDrvInsR3;
659 LogFlow(("drvR3IntNetRecvRun: pThis=%p\n", pThis));
660
661 /*
662 * The running loop - processing received data and waiting for more to arrive.
663 */
664 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
665 PINTNETBUF pBuf = pThis->CTX_SUFF(pBuf);
666 PINTNETRINGBUF pRingBuf = &pBuf->Recv;
667 for (;;)
668 {
669 /*
670 * Process the receive buffer.
671 */
672 PINTNETHDR pHdr;
673 while ((pHdr = IntNetRingGetNextFrameToRead(pRingBuf)) != NULL)
674 {
675 /*
676 * Check the state and then inspect the packet.
677 */
678 if (pThis->enmRecvState != RECVSTATE_RUNNING)
679 {
680 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
681 LogFlow(("drvR3IntNetRecvRun: returns VERR_STATE_CHANGED (state changed - #0)\n"));
682 return VERR_STATE_CHANGED;
683 }
684
685 Log2(("pHdr=%p offRead=%#x: %.8Rhxs\n", pHdr, pRingBuf->offReadX, pHdr));
686 uint16_t u16Type = pHdr->u16Type;
687 if ( ( u16Type == INTNETHDR_TYPE_FRAME
688 || u16Type == INTNETHDR_TYPE_GSO)
689 && !pThis->fLinkDown)
690 {
691 /*
692 * Check if there is room for the frame and pass it up.
693 */
694 size_t cbFrame = pHdr->cbFrame;
695 int rc = pThis->pIAboveNet->pfnWaitReceiveAvail(pThis->pIAboveNet, 0);
696 if (rc == VINF_SUCCESS)
697 {
698 if (u16Type == INTNETHDR_TYPE_FRAME)
699 {
700 /*
701 * Normal frame.
702 */
703#ifdef LOG_ENABLED
704 if (LogIsEnabled())
705 {
706 uint64_t u64Now = RTTimeProgramNanoTS();
707 LogFlow(("drvR3IntNetRecvRun: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
708 cbFrame, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
709 pThis->u64LastReceiveTS = u64Now;
710 Log2(("drvR3IntNetRecvRun: cbFrame=%#x\n"
711 "%.*Rhxd\n",
712 cbFrame, cbFrame, IntNetHdrGetFramePtr(pHdr, pBuf)));
713 }
714#endif
715 rc = pThis->pIAboveNet->pfnReceive(pThis->pIAboveNet, IntNetHdrGetFramePtr(pHdr, pBuf), cbFrame);
716 AssertRC(rc);
717
718 /* skip to the next frame. */
719 IntNetRingSkipFrame(pRingBuf);
720 }
721 else
722 {
723 /*
724 * Generic segment offload frame (INTNETHDR_TYPE_GSO).
725 *
726 * This is where we do the offloading since we don't
727 * emulate any NICs with large receive offload (LRO).
728 */
729 STAM_COUNTER_INC(&pThis->StatReceivedGso);
730 PCPDMNETWORKGSO pGso = IntNetHdrGetGsoContext(pHdr, pBuf);
731 if (PDMNetGsoIsValid(pGso, cbFrame, cbFrame - sizeof(PDMNETWORKGSO)))
732 {
733 cbFrame -= sizeof(PDMNETWORKGSO);
734
735 uint8_t abHdrScratch[256];
736 uint32_t const cSegs = PDMNetGsoCalcSegmentCount(pGso, cbFrame);
737#ifdef LOG_ENABLED
738 if (LogIsEnabled())
739 {
740 uint64_t u64Now = RTTimeProgramNanoTS();
741 LogFlow(("drvR3IntNetRecvRun: %-4d bytes at %llu ns deltas: r=%llu t=%llu; GSO - %u segs\n",
742 cbFrame, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS, cSegs));
743 pThis->u64LastReceiveTS = u64Now;
744 Log2(("drvR3IntNetRecvRun: cbFrame=%#x type=%d cbHdrs=%#x Hdr1=%#x Hdr2=%#x MMS=%#x\n"
745 "%.*Rhxd\n",
746 cbFrame, pGso->u8Type, pGso->cbHdrs, pGso->offHdr1, pGso->offHdr2, pGso->cbMaxSeg,
747 cbFrame - sizeof(*pGso), pGso + 1));
748 }
749#endif
750 for (size_t iSeg = 0; iSeg < cSegs; iSeg++)
751 {
752 uint32_t cbSegFrame;
753 void *pvSegFrame = PDMNetGsoCarveSegmentQD(pGso, (uint8_t *)(pGso + 1), cbFrame, abHdrScratch,
754 iSeg, cSegs, &cbSegFrame);
755 rc = drvR3IntNetRecvWaitForSpace(pThis);
756 if (RT_FAILURE(rc))
757 {
758 Log(("drvR3IntNetRecvRun: drvR3IntNetRecvWaitForSpace -> %Rrc; iSeg=%u cSegs=%u\n", iSeg, cSegs));
759 break; /* we drop the rest. */
760 }
761 rc = pThis->pIAboveNet->pfnReceive(pThis->pIAboveNet, pvSegFrame, cbSegFrame);
762 AssertRC(rc);
763 }
764 }
765 else
766 {
767 AssertMsgFailed(("cbFrame=%#x type=%d cbHdrs=%#x Hdr1=%#x Hdr2=%#x MMS=%#x\n",
768 cbFrame, pGso->u8Type, pGso->cbHdrs, pGso->offHdr1, pGso->offHdr2, pGso->cbMaxSeg));
769 STAM_REL_COUNTER_INC(&pBuf->cStatBadFrames);
770 }
771
772 IntNetRingSkipFrame(pRingBuf);
773 }
774 }
775 else
776 {
777 /*
778 * Wait for sufficient space to become available and then retry.
779 */
780 rc = drvR3IntNetRecvWaitForSpace(pThis);
781 if (RT_FAILURE(rc))
782 {
783 if (rc == VERR_INTERRUPTED)
784 {
785 /*
786 * NIC is going down, likely because the VM is being reset. Skip the frame.
787 */
788 AssertMsg(IntNetIsValidFrameType(pHdr->u16Type), ("Unknown frame type %RX16! offRead=%#x\n", pHdr->u16Type, pRingBuf->offReadX));
789 IntNetRingSkipFrame(pRingBuf);
790 }
791 else
792 {
793 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
794 LogFlow(("drvR3IntNetRecvRun: returns %Rrc (wait-for-space)\n", rc));
795 return rc;
796 }
797 }
798 }
799 }
800 else
801 {
802 /*
803 * Link down or unknown frame - skip to the next frame.
804 */
805 AssertMsg(IntNetIsValidFrameType(pHdr->u16Type), ("Unknown frame type %RX16! offRead=%#x\n", pHdr->u16Type, pRingBuf->offReadX));
806 IntNetRingSkipFrame(pRingBuf);
807 STAM_REL_COUNTER_INC(&pBuf->cStatBadFrames);
808 }
809 } /* while more received data */
810
811 /*
812 * Wait for data, checking the state before we block.
813 */
814 if (pThis->enmRecvState != RECVSTATE_RUNNING)
815 {
816 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
817 LogFlow(("drvR3IntNetRecvRun: returns VINF_SUCCESS (state changed - #1)\n"));
818 return VERR_STATE_CHANGED;
819 }
820 INTNETIFWAITREQ WaitReq;
821 WaitReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
822 WaitReq.Hdr.cbReq = sizeof(WaitReq);
823 WaitReq.pSession = NIL_RTR0PTR;
824 WaitReq.hIf = pThis->hIf;
825 WaitReq.cMillies = 30000; /* 30s - don't wait forever, timeout now and then. */
826 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
827 int rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_WAIT, &WaitReq, sizeof(WaitReq));
828 if ( RT_FAILURE(rc)
829 && rc != VERR_TIMEOUT
830 && rc != VERR_INTERRUPTED)
831 {
832 LogFlow(("drvR3IntNetRecvRun: returns %Rrc\n", rc));
833 return rc;
834 }
835 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
836 }
837}
838
839
840/**
841 * Asynchronous I/O thread for handling receive.
842 *
843 * @returns VINF_SUCCESS (ignored).
844 * @param ThreadSelf Thread handle.
845 * @param pvUser Pointer to a DRVINTNET structure.
846 */
847static DECLCALLBACK(int) drvR3IntNetRecvThread(RTTHREAD ThreadSelf, void *pvUser)
848{
849 PDRVINTNET pThis = (PDRVINTNET)pvUser;
850 LogFlow(("drvR3IntNetRecvThread: pThis=%p\n", pThis));
851 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
852
853 /*
854 * The main loop - acting on state.
855 */
856 for (;;)
857 {
858 RECVSTATE enmRecvState = pThis->enmRecvState;
859 switch (enmRecvState)
860 {
861 case RECVSTATE_SUSPENDED:
862 {
863 int rc = RTSemEventWait(pThis->hRecvEvt, 30000);
864 if ( RT_FAILURE(rc)
865 && rc != VERR_TIMEOUT)
866 {
867 LogFlow(("drvR3IntNetRecvThread: returns %Rrc\n", rc));
868 return rc;
869 }
870 break;
871 }
872
873 case RECVSTATE_RUNNING:
874 {
875 int rc = drvR3IntNetRecvRun(pThis);
876 if ( rc != VERR_STATE_CHANGED
877 && RT_FAILURE(rc))
878 {
879 LogFlow(("drvR3IntNetRecvThread: returns %Rrc\n", rc));
880 return rc;
881 }
882 break;
883 }
884
885 default:
886 AssertMsgFailed(("Invalid state %d\n", enmRecvState));
887 case RECVSTATE_TERMINATE:
888 LogFlow(("drvR3IntNetRecvThread: returns VINF_SUCCESS\n"));
889 return VINF_SUCCESS;
890 }
891 }
892}
893
894
895/* -=-=-=-=- PDMIBASERC -=-=-=-=- */
896
897/**
898 * @interface_method_impl{PDMIBASERC,pfnQueryInterface}
899 */
900static DECLCALLBACK(RTRCPTR) drvR3IntNetIBaseRC_QueryInterface(PPDMIBASERC pInterface, const char *pszIID)
901{
902 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, IBaseRC);
903
904#if 0
905 PDMIBASERC_RETURN_INTERFACE(pThis->pDrvInsR3, pszIID, PDMINETWORKUP, &pThis->INetworkUpRC);
906#endif
907 return NIL_RTRCPTR;
908}
909
910
911/* -=-=-=-=- PDMIBASER0 -=-=-=-=- */
912
913/**
914 * @interface_method_impl{PDMIBASER0,pfnQueryInterface}
915 */
916static DECLCALLBACK(RTR0PTR) drvR3IntNetIBaseR0_QueryInterface(PPDMIBASER0 pInterface, const char *pszIID)
917{
918 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, IBaseR0);
919#ifdef VBOX_WITH_DRVINTNET_IN_R0
920 PDMIBASER0_RETURN_INTERFACE(pThis->pDrvInsR3, pszIID, PDMINETWORKUP, &pThis->INetworkUpR0);
921#endif
922 return NIL_RTR0PTR;
923}
924
925
926/* -=-=-=-=- PDMIBASE -=-=-=-=- */
927
928/**
929 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
930 */
931static DECLCALLBACK(void *) drvR3IntNetIBase_QueryInterface(PPDMIBASE pInterface, const char *pszIID)
932{
933 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
934 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
935
936 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
937 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASER0, &pThis->IBaseR0);
938 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASERC, &pThis->IBaseRC);
939 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKUP, &pThis->INetworkUpR3);
940 return NULL;
941}
942
943
944/* -=-=-=-=- PDMDRVREG -=-=-=-=- */
945
946/**
947 * Power Off notification.
948 *
949 * @param pDrvIns The driver instance.
950 */
951static DECLCALLBACK(void) drvR3IntNetPowerOff(PPDMDRVINS pDrvIns)
952{
953 LogFlow(("drvR3IntNetPowerOff\n"));
954 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
955 if (!pThis->fActivateEarlyDeactivateLate)
956 {
957 ASMAtomicXchgSize(&pThis->enmRecvState, RECVSTATE_SUSPENDED);
958 drvR3IntNetSetActive(pThis, false /* fActive */);
959 }
960}
961
962
963/**
964 * drvR3IntNetResume helper.
965 */
966static int drvR3IntNetResumeSend(PDRVINTNET pThis, const void *pvBuf, size_t cb)
967{
968 /*
969 * Add the frame to the send buffer and push it onto the network.
970 */
971 int rc = IntNetRingWriteFrame(&pThis->pBufR3->Send, pvBuf, (uint32_t)cb);
972 if ( rc == VERR_BUFFER_OVERFLOW
973 && pThis->pBufR3->cbSend < cb)
974 {
975 INTNETIFSENDREQ SendReq;
976 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
977 SendReq.Hdr.cbReq = sizeof(SendReq);
978 SendReq.pSession = NIL_RTR0PTR;
979 SendReq.hIf = pThis->hIf;
980 PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
981
982 rc = IntNetRingWriteFrame(&pThis->pBufR3->Send, pvBuf, (uint32_t)cb);
983 }
984
985 if (RT_SUCCESS(rc))
986 {
987 INTNETIFSENDREQ SendReq;
988 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
989 SendReq.Hdr.cbReq = sizeof(SendReq);
990 SendReq.pSession = NIL_RTR0PTR;
991 SendReq.hIf = pThis->hIf;
992 rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
993 }
994
995 AssertRC(rc);
996 return rc;
997}
998
999
1000/**
1001 * Resume notification.
1002 *
1003 * @param pDrvIns The driver instance.
1004 */
1005static DECLCALLBACK(void) drvR3IntNetResume(PPDMDRVINS pDrvIns)
1006{
1007 LogFlow(("drvR3IntNetPowerResume\n"));
1008 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
1009 if (!pThis->fActivateEarlyDeactivateLate)
1010 {
1011 ASMAtomicXchgSize(&pThis->enmRecvState, RECVSTATE_RUNNING);
1012 RTSemEventSignal(pThis->hRecvEvt);
1013 drvR3IntNetUpdateMacAddress(pThis); /* (could be a state restore) */
1014 drvR3IntNetSetActive(pThis, true /* fActive */);
1015 }
1016 if ( PDMDrvHlpVMTeleportedAndNotFullyResumedYet(pDrvIns)
1017 && pThis->pIAboveConfigR3)
1018 {
1019 /*
1020 * We've just been teleported and need to drop a hint to the switch
1021 * since we're likely to have changed to a different port. We just
1022 * push out some ethernet frame that doesn't mean anything to anyone.
1023 * For this purpose ethertype 0x801e was chosen since it was registered
1024 * to Sun (dunno what it is/was used for though).
1025 */
1026 union
1027 {
1028 RTNETETHERHDR Hdr;
1029 uint8_t ab[128];
1030 } Frame;
1031 RT_ZERO(Frame);
1032 Frame.Hdr.DstMac.au16[0] = 0xffff;
1033 Frame.Hdr.DstMac.au16[1] = 0xffff;
1034 Frame.Hdr.DstMac.au16[2] = 0xffff;
1035 Frame.Hdr.EtherType = RT_H2BE_U16_C(0x801e);
1036 int rc = pThis->pIAboveConfigR3->pfnGetMac(pThis->pIAboveConfigR3, &Frame.Hdr.SrcMac);
1037 if (RT_SUCCESS(rc))
1038 rc = drvR3IntNetResumeSend(pThis, &Frame, sizeof(Frame));
1039 if (RT_FAILURE(rc))
1040 LogRel(("IntNet#%u: Sending dummy frame failed: %Rrc\n", pDrvIns->iInstance, rc));
1041 }
1042}
1043
1044
1045/**
1046 * Suspend notification.
1047 *
1048 * @param pDrvIns The driver instance.
1049 */
1050static DECLCALLBACK(void) drvR3IntNetSuspend(PPDMDRVINS pDrvIns)
1051{
1052 LogFlow(("drvR3IntNetPowerSuspend\n"));
1053 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
1054 if (!pThis->fActivateEarlyDeactivateLate)
1055 {
1056 ASMAtomicXchgSize(&pThis->enmRecvState, RECVSTATE_SUSPENDED);
1057 drvR3IntNetSetActive(pThis, false /* fActive */);
1058 }
1059}
1060
1061
1062/**
1063 * Power On notification.
1064 *
1065 * @param pDrvIns The driver instance.
1066 */
1067static DECLCALLBACK(void) drvR3IntNetPowerOn(PPDMDRVINS pDrvIns)
1068{
1069 LogFlow(("drvR3IntNetPowerOn\n"));
1070 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
1071 if (!pThis->fActivateEarlyDeactivateLate)
1072 {
1073 ASMAtomicXchgSize(&pThis->enmRecvState, RECVSTATE_RUNNING);
1074 RTSemEventSignal(pThis->hRecvEvt);
1075 drvR3IntNetUpdateMacAddress(pThis);
1076 drvR3IntNetSetActive(pThis, true /* fActive */);
1077 }
1078}
1079
1080
1081/**
1082 * @interface_method_impl{PDMDRVREG,pfnRelocate}
1083 */
1084static DECLCALLBACK(void) drvR3IntNetRelocate(PPDMDRVINS pDrvIns, RTGCINTPTR offDelta)
1085{
1086 /* nothing to do here yet */
1087}
1088
1089
1090/**
1091 * Destruct a driver instance.
1092 *
1093 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
1094 * resources can be freed correctly.
1095 *
1096 * @param pDrvIns The driver instance data.
1097 */
1098static DECLCALLBACK(void) drvR3IntNetDestruct(PPDMDRVINS pDrvIns)
1099{
1100 LogFlow(("drvR3IntNetDestruct\n"));
1101 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
1102 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
1103
1104 /*
1105 * Indicate to the thread that it's time to quit.
1106 */
1107 ASMAtomicXchgSize(&pThis->enmRecvState, RECVSTATE_TERMINATE);
1108 ASMAtomicXchgSize(&pThis->fLinkDown, true);
1109 RTSEMEVENT hRecvEvt = pThis->hRecvEvt;
1110 pThis->hRecvEvt = NIL_RTSEMEVENT;
1111
1112 /*
1113 * Close the interface
1114 */
1115 if (pThis->hIf != INTNET_HANDLE_INVALID)
1116 {
1117 INTNETIFCLOSEREQ CloseReq;
1118 CloseReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
1119 CloseReq.Hdr.cbReq = sizeof(CloseReq);
1120 CloseReq.pSession = NIL_RTR0PTR;
1121 CloseReq.hIf = pThis->hIf;
1122 pThis->hIf = INTNET_HANDLE_INVALID;
1123 int rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_CLOSE, &CloseReq, sizeof(CloseReq));
1124 AssertRC(rc);
1125 }
1126
1127 /*
1128 * Wait for the thread to terminate.
1129 */
1130 if (hRecvEvt != NIL_RTSEMEVENT)
1131 RTSemEventSignal(hRecvEvt);
1132
1133 if (pThis->pXmitThread)
1134 {
1135 int rc = PDMR3ThreadDestroy(pThis->pXmitThread, NULL);
1136 AssertRC(rc);
1137 pThis->pXmitThread = NULL;
1138 }
1139
1140 if (pThis->hRecvThread != NIL_RTTHREAD)
1141 {
1142 int rc = RTThreadWait(pThis->hRecvThread, 5000, NULL);
1143 AssertRC(rc);
1144 pThis->hRecvThread = NIL_RTTHREAD;
1145 }
1146
1147
1148 /*
1149 * Destroy the semaphores, S/G cache and xmit lock.
1150 */
1151 if (hRecvEvt != NIL_RTSEMEVENT)
1152 RTSemEventDestroy(hRecvEvt);
1153
1154 if (pThis->hXmitEvt != NIL_SUPSEMEVENT)
1155 {
1156 SUPSemEventClose(pThis->pSupDrvSession, pThis->hXmitEvt);
1157 pThis->hXmitEvt = NIL_SUPSEMEVENT;
1158 }
1159
1160 RTMemCacheDestroy(pThis->hSgCache);
1161 pThis->hSgCache = NIL_RTMEMCACHE;
1162
1163 if (PDMCritSectIsInitialized(&pThis->XmitLock))
1164 PDMR3CritSectDelete(&pThis->XmitLock);
1165
1166 if (pThis->pBufR3)
1167 {
1168 /*
1169 * Deregister statistics in case we're being detached.
1170 */
1171 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Recv.cStatFrames);
1172 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Recv.cbStatWritten);
1173 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Recv.cOverflows);
1174 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Send.cStatFrames);
1175 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Send.cbStatWritten);
1176 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Send.cOverflows);
1177 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatYieldsOk);
1178 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatYieldsNok);
1179 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatLost);
1180 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatBadFrames);
1181 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReceivedGso);
1182 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatSentGso);
1183#ifdef VBOX_WITH_STATISTICS
1184 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReceive);
1185 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatTransmit);
1186#endif
1187 }
1188}
1189
1190
1191/**
1192 * Construct a TAP network transport driver instance.
1193 *
1194 * @copydoc FNPDMDRVCONSTRUCT
1195 */
1196static DECLCALLBACK(int) drvR3IntNetConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
1197{
1198 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
1199 bool f;
1200 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
1201
1202 /*
1203 * Init the static parts.
1204 */
1205 pThis->pDrvInsR3 = pDrvIns;
1206 pThis->hIf = INTNET_HANDLE_INVALID;
1207 pThis->hRecvThread = NIL_RTTHREAD;
1208 pThis->hRecvEvt = NIL_RTSEMEVENT;
1209 pThis->pXmitThread = NULL;
1210 pThis->hXmitEvt = NIL_SUPSEMEVENT;
1211 pThis->pSupDrvSession = PDMDrvHlpGetSupDrvSession(pDrvIns);
1212 pThis->hSgCache = NIL_RTMEMCACHE;
1213 pThis->enmRecvState = RECVSTATE_SUSPENDED;
1214 pThis->fActivateEarlyDeactivateLate = false;
1215 /* IBase* */
1216 pDrvIns->IBase.pfnQueryInterface = drvR3IntNetIBase_QueryInterface;
1217 pThis->IBaseR0.pfnQueryInterface = drvR3IntNetIBaseR0_QueryInterface;
1218 pThis->IBaseRC.pfnQueryInterface = drvR3IntNetIBaseRC_QueryInterface;
1219 /* INetworkUp */
1220 pThis->INetworkUpR3.pfnBeginXmit = drvIntNetUp_BeginXmit;
1221 pThis->INetworkUpR3.pfnAllocBuf = drvIntNetUp_AllocBuf;
1222 pThis->INetworkUpR3.pfnFreeBuf = drvIntNetUp_FreeBuf;
1223 pThis->INetworkUpR3.pfnSendBuf = drvIntNetUp_SendBuf;
1224 pThis->INetworkUpR3.pfnEndXmit = drvIntNetUp_EndXmit;
1225 pThis->INetworkUpR3.pfnSetPromiscuousMode = drvIntNetUp_SetPromiscuousMode;
1226 pThis->INetworkUpR3.pfnNotifyLinkChanged = drvR3IntNetUp_NotifyLinkChanged;
1227
1228 /*
1229 * Validate the config.
1230 */
1231 if (!CFGMR3AreValuesValid(pCfg,
1232 "Network\0"
1233 "Trunk\0"
1234 "TrunkType\0"
1235 "ReceiveBufferSize\0"
1236 "SendBufferSize\0"
1237 "RestrictAccess\0"
1238 "SharedMacOnWire\0"
1239 "IgnoreAllPromisc\0"
1240 "QuietlyIgnoreAllPromisc\0"
1241 "IgnoreClientPromisc\0"
1242 "QuietlyIgnoreClientPromisc\0"
1243 "IgnoreTrunkWirePromisc\0"
1244 "QuietlyIgnoreTrunkWirePromisc\0"
1245 "IgnoreTrunkHostPromisc\0"
1246 "QuietlyIgnoreTrunkHostPromisc\0"
1247 "IsService\0"))
1248 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
1249
1250 /*
1251 * Check that no-one is attached to us.
1252 */
1253 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
1254 ("Configuration error: Not possible to attach anything to this driver!\n"),
1255 VERR_PDM_DRVINS_NO_ATTACH);
1256
1257 /*
1258 * Query the network port interface.
1259 */
1260 pThis->pIAboveNet = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMINETWORKDOWN);
1261 if (!pThis->pIAboveNet)
1262 {
1263 AssertMsgFailed(("Configuration error: the above device/driver didn't export the network port interface!\n"));
1264 return VERR_PDM_MISSING_INTERFACE_ABOVE;
1265 }
1266 pThis->pIAboveConfigR3 = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMINETWORKCONFIG);
1267
1268 /*
1269 * Read the configuration.
1270 */
1271 INTNETOPENREQ OpenReq;
1272 memset(&OpenReq, 0, sizeof(OpenReq));
1273 OpenReq.Hdr.cbReq = sizeof(OpenReq);
1274 OpenReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
1275 OpenReq.pSession = NIL_RTR0PTR;
1276
1277 /** @cfgm{Network, string}
1278 * The name of the internal network to connect to.
1279 */
1280 int rc = CFGMR3QueryString(pCfg, "Network", OpenReq.szNetwork, sizeof(OpenReq.szNetwork));
1281 if (RT_FAILURE(rc))
1282 return PDMDRV_SET_ERROR(pDrvIns, rc,
1283 N_("Configuration error: Failed to get the \"Network\" value"));
1284 strcpy(pThis->szNetwork, OpenReq.szNetwork);
1285
1286 /** @cfgm{TrunkType, uint32_t, kIntNetTrunkType_None}
1287 * The trunk connection type see INTNETTRUNKTYPE.
1288 */
1289 uint32_t u32TrunkType;
1290 rc = CFGMR3QueryU32(pCfg, "TrunkType", &u32TrunkType);
1291 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1292 u32TrunkType = kIntNetTrunkType_None;
1293 else if (RT_FAILURE(rc))
1294 return PDMDRV_SET_ERROR(pDrvIns, rc,
1295 N_("Configuration error: Failed to get the \"TrunkType\" value"));
1296 OpenReq.enmTrunkType = (INTNETTRUNKTYPE)u32TrunkType;
1297
1298 /** @cfgm{Trunk, string, ""}
1299 * The name of the trunk connection.
1300 */
1301 rc = CFGMR3QueryString(pCfg, "Trunk", OpenReq.szTrunk, sizeof(OpenReq.szTrunk));
1302 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1303 OpenReq.szTrunk[0] = '\0';
1304 else if (RT_FAILURE(rc))
1305 return PDMDRV_SET_ERROR(pDrvIns, rc,
1306 N_("Configuration error: Failed to get the \"Trunk\" value"));
1307
1308 /** @cfgm{RestrictAccess, boolean, true}
1309 * Whether to restrict the access to the network or if it should be public. Everyone on
1310 * the computer can connect to a public network. Don't change this.
1311 */
1312 bool fRestrictAccess;
1313 rc = CFGMR3QueryBool(pCfg, "RestrictAccess", &fRestrictAccess);
1314 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1315 fRestrictAccess = true;
1316 else if (RT_FAILURE(rc))
1317 return PDMDRV_SET_ERROR(pDrvIns, rc,
1318 N_("Configuration error: Failed to get the \"RestrictAccess\" value"));
1319 OpenReq.fFlags = fRestrictAccess ? 0 : INTNET_OPEN_FLAGS_PUBLIC;
1320
1321 /** @cfgm{IgnoreAllPromisc, boolean, false}
1322 * When set all request for operating any interface or trunk in promiscuous
1323 * mode will be ignored. */
1324 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreAllPromisc", &f, false);
1325 if (RT_FAILURE(rc))
1326 return PDMDRV_SET_ERROR(pDrvIns, rc,
1327 N_("Configuration error: Failed to get the \"IgnoreAllPromisc\" value"));
1328 if (f)
1329 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC;
1330
1331 /** @cfgm{QuietlyIgnoreAllPromisc, boolean, false}
1332 * When set all request for operating any interface or trunk in promiscuous
1333 * mode will be ignored. This differs from IgnoreAllPromisc in that clients
1334 * won't get VERR_INTNET_INCOMPATIBLE_FLAGS. */
1335 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreAllPromisc", &f, false);
1336 if (RT_FAILURE(rc))
1337 return PDMDRV_SET_ERROR(pDrvIns, rc,
1338 N_("Configuration error: Failed to get the \"QuietlyIgnoreAllPromisc\" value"));
1339 if (f)
1340 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC;
1341
1342 /** @cfgm{IgnoreClientPromisc, boolean, false}
1343 * When set all request for operating any non-trunk interface in promiscuous
1344 * mode will be ignored. */
1345 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreClientPromisc", &f, false);
1346 if (RT_FAILURE(rc))
1347 return PDMDRV_SET_ERROR(pDrvIns, rc,
1348 N_("Configuration error: Failed to get the \"IgnoreClientPromisc\" value"));
1349 if (f)
1350 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC; /** @todo add special flag for this. */
1351
1352 /** @cfgm{QuietlyIgnoreClientPromisc, boolean, false}
1353 * When set all request for operating any non-trunk interface promiscuous mode
1354 * will be ignored. This differs from IgnoreClientPromisc in that clients won't
1355 * get VERR_INTNET_INCOMPATIBLE_FLAGS. */
1356 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreClientPromisc", &f, false);
1357 if (RT_FAILURE(rc))
1358 return PDMDRV_SET_ERROR(pDrvIns, rc,
1359 N_("Configuration error: Failed to get the \"QuietlyIgnoreClientPromisc\" value"));
1360 if (f)
1361 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC; /** @todo add special flag for this. */
1362
1363 /** @cfgm{IgnoreTrunkWirePromisc, boolean, false}
1364 * When set all request for operating the trunk-wire connection in promiscuous
1365 * mode will be ignored. */
1366 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreTrunkWirePromisc", &f, false);
1367 if (RT_FAILURE(rc))
1368 return PDMDRV_SET_ERROR(pDrvIns, rc,
1369 N_("Configuration error: Failed to get the \"IgnoreTrunkWirePromisc\" value"));
1370 if (f)
1371 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC_TRUNK_WIRE;
1372
1373 /** @cfgm{QuietlyIgnoreTrunkWirePromisc, boolean, false}
1374 * When set all request for operating any trunk-wire connection promiscuous mode
1375 * will be ignored. This differs from IgnoreTrunkWirePromisc in that clients
1376 * won't get VERR_INTNET_INCOMPATIBLE_FLAGS. */
1377 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreTrunkWirePromisc", &f, false);
1378 if (RT_FAILURE(rc))
1379 return PDMDRV_SET_ERROR(pDrvIns, rc,
1380 N_("Configuration error: Failed to get the \"QuietlyIgnoreTrunkWirePromisc\" value"));
1381 if (f)
1382 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC_TRUNK_WIRE;
1383
1384 /** @cfgm{IgnoreTrunkHostPromisc, boolean, false}
1385 * When set all request for operating the trunk-host connection in promiscuous
1386 * mode will be ignored. */
1387 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreTrunkHostPromisc", &f, false);
1388 if (RT_FAILURE(rc))
1389 return PDMDRV_SET_ERROR(pDrvIns, rc,
1390 N_("Configuration error: Failed to get the \"IgnoreTrunkHostPromisc\" value"));
1391 if (f)
1392 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC_TRUNK_HOST;
1393
1394 /** @cfgm{QuietlyIgnoreTrunkHostPromisc, boolean, false}
1395 * When set all request for operating any trunk-host connection promiscuous mode
1396 * will be ignored. This differs from IgnoreTrunkHostPromisc in that clients
1397 * won't get VERR_INTNET_INCOMPATIBLE_FLAGS. */
1398 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreTrunkHostPromisc", &f, false);
1399 if (RT_FAILURE(rc))
1400 return PDMDRV_SET_ERROR(pDrvIns, rc,
1401 N_("Configuration error: Failed to get the \"QuietlyIgnoreTrunkHostPromisc\" value"));
1402 if (f)
1403 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC_TRUNK_HOST;
1404
1405 /** @todo flags for not sending to the host and for setting the trunk-wire
1406 * connection in promiscuous mode. */
1407
1408
1409 /** @cfgm{SharedMacOnWire, boolean, false}
1410 * Whether to shared the MAC address of the host interface when using the wire. When
1411 * attaching to a wireless NIC this option is usally a requirement.
1412 */
1413 bool fSharedMacOnWire;
1414 rc = CFGMR3QueryBoolDef(pCfg, "SharedMacOnWire", &fSharedMacOnWire, false);
1415 if (RT_FAILURE(rc))
1416 return PDMDRV_SET_ERROR(pDrvIns, rc,
1417 N_("Configuration error: Failed to get the \"SharedMacOnWire\" value"));
1418 if (fSharedMacOnWire)
1419 OpenReq.fFlags |= INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE;
1420
1421 /** @cfgm{ReceiveBufferSize, uint32_t, 318 KB}
1422 * The size of the receive buffer.
1423 */
1424 rc = CFGMR3QueryU32(pCfg, "ReceiveBufferSize", &OpenReq.cbRecv);
1425 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1426 OpenReq.cbRecv = 318 * _1K ;
1427 else if (RT_FAILURE(rc))
1428 return PDMDRV_SET_ERROR(pDrvIns, rc,
1429 N_("Configuration error: Failed to get the \"ReceiveBufferSize\" value"));
1430
1431 /** @cfgm{SendBufferSize, uint32_t, 196 KB}
1432 * The size of the send (transmit) buffer.
1433 * This should be more than twice the size of the larges frame size because
1434 * the ring buffer is very simple and doesn't support splitting up frames
1435 * nor inserting padding. So, if this is too close to the frame size the
1436 * header will fragment the buffer such that the frame won't fit on either
1437 * side of it and the code will get very upset about it all.
1438 */
1439 rc = CFGMR3QueryU32(pCfg, "SendBufferSize", &OpenReq.cbSend);
1440 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1441 OpenReq.cbSend = RT_ALIGN_Z(VBOX_MAX_GSO_SIZE * 3, _1K);
1442 else if (RT_FAILURE(rc))
1443 return PDMDRV_SET_ERROR(pDrvIns, rc,
1444 N_("Configuration error: Failed to get the \"SendBufferSize\" value"));
1445 if (OpenReq.cbSend < 128)
1446 return PDMDRV_SET_ERROR(pDrvIns, rc,
1447 N_("Configuration error: The \"SendBufferSize\" value is too small"));
1448 if (OpenReq.cbSend < VBOX_MAX_GSO_SIZE * 3)
1449 LogRel(("DrvIntNet: Warning! SendBufferSize=%u, Recommended minimum size %u butes.\n", OpenReq.cbSend, VBOX_MAX_GSO_SIZE * 4));
1450
1451 /** @cfgm{IsService, boolean, true}
1452 * This alterns the way the thread is suspended and resumed. When it's being used by
1453 * a service such as LWIP/iSCSI it shouldn't suspend immediately like for a NIC.
1454 */
1455 rc = CFGMR3QueryBool(pCfg, "IsService", &pThis->fActivateEarlyDeactivateLate);
1456 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1457 pThis->fActivateEarlyDeactivateLate = false;
1458 else if (RT_FAILURE(rc))
1459 return PDMDRV_SET_ERROR(pDrvIns, rc,
1460 N_("Configuration error: Failed to get the \"IsService\" value"));
1461
1462 LogRel(("IntNet#%u: szNetwork={%s} enmTrunkType=%d szTrunk={%s} fFlags=%#x cbRecv=%u cbSend=%u\n",
1463 pDrvIns->iInstance, OpenReq.szNetwork, OpenReq.enmTrunkType, OpenReq.szTrunk, OpenReq.fFlags,
1464 OpenReq.cbRecv, OpenReq.cbSend));
1465
1466#ifdef RT_OS_DARWIN
1467 /* Temporary hack: attach to a network with the name 'if=en0' and you're hitting the wire. */
1468 if ( !OpenReq.szTrunk[0]
1469 && OpenReq.enmTrunkType == kIntNetTrunkType_None
1470 && !strncmp(pThis->szNetwork, "if=en", sizeof("if=en") - 1)
1471 && RT_C_IS_DIGIT(pThis->szNetwork[sizeof("if=en") - 1])
1472 && !pThis->szNetwork[sizeof("if=en")])
1473 {
1474 OpenReq.enmTrunkType = kIntNetTrunkType_NetFlt;
1475 strcpy(OpenReq.szTrunk, &pThis->szNetwork[sizeof("if=") - 1]);
1476 }
1477 /* Temporary hack: attach to a network with the name 'wif=en0' and you're on the air. */
1478 if ( !OpenReq.szTrunk[0]
1479 && OpenReq.enmTrunkType == kIntNetTrunkType_None
1480 && !strncmp(pThis->szNetwork, "wif=en", sizeof("wif=en") - 1)
1481 && RT_C_IS_DIGIT(pThis->szNetwork[sizeof("wif=en") - 1])
1482 && !pThis->szNetwork[sizeof("wif=en")])
1483 {
1484 OpenReq.enmTrunkType = kIntNetTrunkType_NetFlt;
1485 OpenReq.fFlags |= INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE;
1486 strcpy(OpenReq.szTrunk, &pThis->szNetwork[sizeof("wif=") - 1]);
1487 }
1488#endif /* DARWIN */
1489
1490 /*
1491 * Create the event semaphore, S/G cache and xmit critsect.
1492 */
1493 rc = RTSemEventCreate(&pThis->hRecvEvt);
1494 if (RT_FAILURE(rc))
1495 return rc;
1496 rc = RTMemCacheCreate(&pThis->hSgCache, sizeof(PDMSCATTERGATHER), 0, UINT32_MAX, NULL, NULL, pThis, 0);
1497 if (RT_FAILURE(rc))
1498 return rc;
1499 rc = PDMDrvHlpCritSectInit(pDrvIns, &pThis->XmitLock, RT_SRC_POS, "IntNetXmit");
1500 if (RT_FAILURE(rc))
1501 return rc;
1502
1503 /*
1504 * Create the interface.
1505 */
1506 OpenReq.hIf = INTNET_HANDLE_INVALID;
1507 rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_OPEN, &OpenReq, sizeof(OpenReq));
1508 if (RT_FAILURE(rc))
1509 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1510 N_("Failed to open/create the internal network '%s'"), pThis->szNetwork);
1511 AssertRelease(OpenReq.hIf != INTNET_HANDLE_INVALID);
1512 pThis->hIf = OpenReq.hIf;
1513 Log(("IntNet%d: hIf=%RX32 '%s'\n", pDrvIns->iInstance, pThis->hIf, pThis->szNetwork));
1514
1515 /*
1516 * Get default buffer.
1517 */
1518 INTNETIFGETBUFFERPTRSREQ GetBufferPtrsReq;
1519 GetBufferPtrsReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
1520 GetBufferPtrsReq.Hdr.cbReq = sizeof(GetBufferPtrsReq);
1521 GetBufferPtrsReq.pSession = NIL_RTR0PTR;
1522 GetBufferPtrsReq.hIf = pThis->hIf;
1523 GetBufferPtrsReq.pRing3Buf = NULL;
1524 GetBufferPtrsReq.pRing0Buf = NIL_RTR0PTR;
1525 rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_GET_BUFFER_PTRS, &GetBufferPtrsReq, sizeof(GetBufferPtrsReq));
1526 if (RT_FAILURE(rc))
1527 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1528 N_("Failed to get ring-3 buffer for the newly created interface to '%s'"), pThis->szNetwork);
1529 AssertRelease(VALID_PTR(GetBufferPtrsReq.pRing3Buf));
1530 pThis->pBufR3 = GetBufferPtrsReq.pRing3Buf;
1531 pThis->pBufR0 = GetBufferPtrsReq.pRing0Buf;
1532
1533 /*
1534 * Register statistics.
1535 */
1536 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->Recv.cbStatWritten, "Bytes/Received", STAMUNIT_BYTES, "Number of received bytes.");
1537 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->Send.cbStatWritten, "Bytes/Sent", STAMUNIT_BYTES, "Number of sent bytes.");
1538 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->Recv.cOverflows, "Overflows/Recv", STAMUNIT_COUNT, "Number overflows.");
1539 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->Send.cOverflows, "Overflows/Sent", STAMUNIT_COUNT, "Number overflows.");
1540 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->Recv.cStatFrames, "Packets/Received", STAMUNIT_COUNT, "Number of received packets.");
1541 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->Send.cStatFrames, "Packets/Sent", STAMUNIT_COUNT, "Number of sent packets.");
1542 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->StatReceivedGso, "Packets/Received-Gso", STAMUNIT_COUNT, "The GSO portion of the received packets.");
1543 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->StatSentGso, "Packets/Sent-Gso", STAMUNIT_COUNT, "The GSO portion of the sent packets.");
1544 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->StatSentR0, "Packets/Sent-R0", STAMUNIT_COUNT, "The ring-0 portion of the sent packets.");
1545
1546 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->cStatLost, "Packets/Lost", STAMUNIT_COUNT, "Number of lost packets.");
1547 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->cStatYieldsNok, "YieldOk", STAMUNIT_COUNT, "Number of times yielding helped fix an overflow.");
1548 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->cStatYieldsOk, "YieldNok", STAMUNIT_COUNT, "Number of times yielding didn't help fix an overflow.");
1549 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->pBufR3->cStatBadFrames, "BadFrames", STAMUNIT_COUNT, "Number of bad frames seed by the consumers.");
1550#ifdef VBOX_WITH_STATISTICS
1551 PDMDrvHlpSTAMRegProfileAdv(pDrvIns, &pThis->StatReceive, "Receive", STAMUNIT_TICKS_PER_CALL, "Profiling packet receive runs.");
1552 PDMDrvHlpSTAMRegProfile(pDrvIns, &pThis->StatTransmit, "Transmit", STAMUNIT_TICKS_PER_CALL, "Profiling packet transmit runs.");
1553#endif
1554 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->StatXmitWakeupR0, "XmitWakeup-R0", STAMUNIT_COUNT, "Xmit thread wakeups from ring-0.");
1555 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->StatXmitWakeupR3, "XmitWakeup-R3", STAMUNIT_COUNT, "Xmit thread wakeups from ring-3.");
1556 PDMDrvHlpSTAMRegCounter(pDrvIns, &pThis->StatXmitProcessRing, "XmitProcessRing", STAMUNIT_COUNT, "Time xmit thread was told to process the ring.");
1557
1558 /*
1559 * Create the async I/O threads.
1560 * Note! Using a PDM thread here doesn't fit with the IsService=true operation.
1561 */
1562 rc = RTThreadCreate(&pThis->hRecvThread, drvR3IntNetRecvThread, pThis, 0,
1563 RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "INTNET-RECV");
1564 if (RT_FAILURE(rc))
1565 {
1566 AssertRC(rc);
1567 return rc;
1568 }
1569
1570 rc = SUPSemEventCreate(pThis->pSupDrvSession, &pThis->hXmitEvt);
1571 AssertRCReturn(rc, rc);
1572
1573 rc = PDMDrvHlpThreadCreate(pDrvIns, &pThis->pXmitThread, pThis,
1574 drvR3IntNetXmitThread, drvR3IntNetXmitWakeUp, 0, RTTHREADTYPE_IO, "INTNET-XMIT");
1575 AssertRCReturn(rc, rc);
1576
1577#ifdef VBOX_WITH_DRVINTNET_IN_R0
1578 /*
1579 * Resolve the ring-0 context interface addresses.
1580 */
1581 rc = pDrvIns->pHlpR3->pfnLdrGetR0InterfaceSymbols(pDrvIns, &pThis->INetworkUpR0, sizeof(pThis->INetworkUpR0),
1582 "drvIntNetUp_", PDMINETWORKUP_SYM_LIST);
1583 AssertLogRelRCReturn(rc, rc);
1584#endif
1585
1586 /*
1587 * Activate data transmission as early as possible
1588 */
1589 if (pThis->fActivateEarlyDeactivateLate)
1590 {
1591 ASMAtomicXchgSize(&pThis->enmRecvState, RECVSTATE_RUNNING);
1592 RTSemEventSignal(pThis->hRecvEvt);
1593
1594 drvR3IntNetUpdateMacAddress(pThis);
1595 drvR3IntNetSetActive(pThis, true /* fActive */);
1596 }
1597
1598 return rc;
1599}
1600
1601
1602
1603/**
1604 * Internal networking transport driver registration record.
1605 */
1606const PDMDRVREG g_DrvIntNet =
1607{
1608 /* u32Version */
1609 PDM_DRVREG_VERSION,
1610 /* szName */
1611 "IntNet",
1612 /* szRCMod */
1613 "VBoxDDGC.rc",
1614 /* szR0Mod */
1615 "VBoxDDR0.r0",
1616 /* pszDescription */
1617 "Internal Networking Transport Driver",
1618 /* fFlags */
1619#ifdef VBOX_WITH_DRVINTNET_IN_R0
1620 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DRVREG_FLAGS_R0,
1621#else
1622 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1623#endif
1624 /* fClass. */
1625 PDM_DRVREG_CLASS_NETWORK,
1626 /* cMaxInstances */
1627 ~0,
1628 /* cbInstance */
1629 sizeof(DRVINTNET),
1630 /* pfnConstruct */
1631 drvR3IntNetConstruct,
1632 /* pfnDestruct */
1633 drvR3IntNetDestruct,
1634 /* pfnRelocate */
1635 drvR3IntNetRelocate,
1636 /* pfnIOCtl */
1637 NULL,
1638 /* pfnPowerOn */
1639 drvR3IntNetPowerOn,
1640 /* pfnReset */
1641 NULL,
1642 /* pfnSuspend */
1643 drvR3IntNetSuspend,
1644 /* pfnResume */
1645 drvR3IntNetResume,
1646 /* pfnAttach */
1647 NULL,
1648 /* pfnDetach */
1649 NULL,
1650 /* pfnPowerOff */
1651 drvR3IntNetPowerOff,
1652 /* pfnSoftReset */
1653 NULL,
1654 /* u32EndVersion */
1655 PDM_DRVREG_VERSION
1656};
1657
1658#endif /* IN_RING3 */
1659
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