VirtualBox

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

Last change on this file since 28440 was 28339, checked in by vboxsync, 15 years ago

alignment fix.

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