VirtualBox

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

Last change on this file since 27867 was 27840, checked in by vboxsync, 15 years ago

DrvIntNet,++: Implemented the missing buffer methods, added padding frame.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 48.0 KB
Line 
1/* $Id: DrvIntNet.cpp 27840 2010-03-30 21:00:51Z 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/pdmnetifs.h>
28#include <VBox/cfgm.h>
29#include <VBox/intnet.h>
30#include <VBox/intnetinline.h>
31#include <VBox/vmm.h>
32#include <VBox/err.h>
33
34#include <VBox/log.h>
35#include <iprt/asm.h>
36#include <iprt/assert.h>
37#include <iprt/ctype.h>
38#include <iprt/memcache.h>
39#include <iprt/net.h>
40#include <iprt/semaphore.h>
41#include <iprt/string.h>
42#include <iprt/time.h>
43#include <iprt/thread.h>
44#include <iprt/uuid.h>
45
46#include "../Builtins.h"
47
48
49/*******************************************************************************
50* Structures and Typedefs *
51*******************************************************************************/
52/**
53 * The state of the asynchronous thread.
54 */
55typedef enum ASYNCSTATE
56{
57 /** The thread is suspended. */
58 ASYNCSTATE_SUSPENDED = 1,
59 /** The thread is running. */
60 ASYNCSTATE_RUNNING,
61 /** The thread must (/has) terminate. */
62 ASYNCSTATE_TERMINATE,
63 /** The usual 32-bit type blowup. */
64 ASYNCSTATE_32BIT_HACK = 0x7fffffff
65} ASYNCSTATE;
66
67/**
68 * Internal networking driver instance data.
69 *
70 * @implements PDMINETWORKUP
71 */
72typedef struct DRVINTNET
73{
74 /** The network interface. */
75 PDMINETWORKUP INetworkUpR3;
76 /** The network interface. */
77 R3PTRTYPE(PPDMINETWORKDOWN) pIAboveNet;
78 /** The network config interface.
79 * Can (in theory at least) be NULL. */
80 R3PTRTYPE(PPDMINETWORKCONFIG) pIAboveConfigR3;
81 /** Pointer to the driver instance. */
82 PPDMDRVINSR3 pDrvInsR3;
83 /** Pointer to the communication buffer. */
84 R3PTRTYPE(PINTNETBUF) pBufR3;
85 /** Interface handle. */
86 INTNETIFHANDLE hIf;
87
88 /** The thread state. */
89 ASYNCSTATE volatile enmState;
90 /** Reader thread. */
91 RTTHREAD Thread;
92 /** Event semaphore the Thread waits on while the VM is suspended. */
93 RTSEMEVENT hEvtSuspended;
94 /** Scatter/gather descriptor cache. */
95 RTMEMCACHE hSgCache;
96 /** Set if the link is down.
97 * When the link is down all incoming packets will be dropped. */
98 bool volatile fLinkDown;
99 /** Set if data transmission should start immediately and deactivate
100 * as late as possible. */
101 bool fActivateEarlyDeactivateLate;
102 /** Padding. */
103 bool afReserved[2];
104 /** The network name. */
105 char szNetwork[INTNET_MAX_NETWORK_NAME];
106
107 /** Base interface for ring-0. */
108 PDMIBASER0 IBaseR0;
109 /** Base interface for ring-0. */
110 PDMIBASERC IBaseRC;
111
112#ifdef LOG_ENABLED
113 /** The nano ts of the last transfer. */
114 uint64_t u64LastTransferTS;
115 /** The nano ts of the last receive. */
116 uint64_t u64LastReceiveTS;
117#endif
118#ifdef VBOX_WITH_STATISTICS
119 /** Profiling packet transmit runs. */
120 STAMPROFILE StatTransmit;
121 /** Profiling packet receive runs. */
122 STAMPROFILEADV StatReceive;
123#endif /* VBOX_WITH_STATISTICS */
124} DRVINTNET;
125/** Pointer to instance data of the internal networking driver. */
126typedef DRVINTNET *PDRVINTNET;
127
128
129#ifdef IN_RING3
130
131/* -=-=-=-=- PDMINETWORKUP -=-=-=-=- */
132
133
134/**
135 * Updates the MAC address on the kernel side.
136 *
137 * @returns VBox status code.
138 * @param pThis The driver instance.
139 */
140static int drvR3IntNetUpdateMacAddress(PDRVINTNET pThis)
141{
142 if (!pThis->pIAboveConfigR3)
143 return VINF_SUCCESS;
144
145 INTNETIFSETMACADDRESSREQ SetMacAddressReq;
146 SetMacAddressReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
147 SetMacAddressReq.Hdr.cbReq = sizeof(SetMacAddressReq);
148 SetMacAddressReq.pSession = NIL_RTR0PTR;
149 SetMacAddressReq.hIf = pThis->hIf;
150 int rc = pThis->pIAboveConfigR3->pfnGetMac(pThis->pIAboveConfigR3, &SetMacAddressReq.Mac);
151 if (RT_SUCCESS(rc))
152 rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SET_MAC_ADDRESS,
153 &SetMacAddressReq, sizeof(SetMacAddressReq));
154
155 Log(("drvR3IntNetUpdateMacAddress: %.*Rhxs rc=%Rrc\n", sizeof(SetMacAddressReq.Mac), &SetMacAddressReq.Mac, rc));
156 return rc;
157}
158
159
160/**
161 * Sets the kernel interface active or inactive.
162 *
163 * Worker for poweron, poweroff, suspend and resume.
164 *
165 * @returns VBox status code.
166 * @param pThis The driver instance.
167 * @param fActive The new state.
168 */
169static int drvR3IntNetSetActive(PDRVINTNET pThis, bool fActive)
170{
171 if (!pThis->pIAboveConfigR3)
172 return VINF_SUCCESS;
173
174 INTNETIFSETACTIVEREQ SetActiveReq;
175 SetActiveReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
176 SetActiveReq.Hdr.cbReq = sizeof(SetActiveReq);
177 SetActiveReq.pSession = NIL_RTR0PTR;
178 SetActiveReq.hIf = pThis->hIf;
179 SetActiveReq.fActive = fActive;
180 int rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SET_ACTIVE,
181 &SetActiveReq, sizeof(SetActiveReq));
182
183 Log(("drvR3IntNetSetActive: fActive=%d rc=%Rrc\n", fActive, rc));
184 AssertRC(rc);
185 return rc;
186}
187
188
189/**
190 * @interface_method_impl{PDMINETWORKUP,pfnAllocBuf}
191 */
192static DECLCALLBACK(int) drvR3IntNetUp_AllocBuf(PPDMINETWORKUP pInterface, size_t cbMin, PPPDMSCATTERGATHER ppSgBuf)
193{
194 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, INetworkUpR3);
195 int rc = VINF_SUCCESS;
196 Assert(cbMin < UINT32_MAX / 2);
197
198 /*
199 * Allocate a S/G buffer.
200 */
201 PPDMSCATTERGATHER pSgBuf = (PPDMSCATTERGATHER)RTMemCacheAlloc(pThis->hSgCache);
202 if (pSgBuf)
203 {
204 PINTNETHDR pHdr;
205 rc = INTNETRingAllocateFrame(&pThis->pBufR3->Send, (uint32_t)cbMin,
206 &pHdr, &pSgBuf->aSegs[0].pvSeg);
207#if 1 /** @todo implement VERR_TRY_AGAIN once this moves to EMT. */
208 if ( RT_FAILURE(rc)
209 && pThis->pBufR3->cbSend >= cbMin * 2 + sizeof(INTNETHDR))
210 {
211 INTNETIFSENDREQ SendReq;
212 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
213 SendReq.Hdr.cbReq = sizeof(SendReq);
214 SendReq.pSession = NIL_RTR0PTR;
215 SendReq.hIf = pThis->hIf;
216 PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
217
218 rc = INTNETRingAllocateFrame(&pThis->pBufR3->Send, (uint32_t)cbMin,
219 &pHdr, &pSgBuf->aSegs[0].pvSeg);
220 }
221#endif
222 if (RT_SUCCESS(rc))
223 {
224 pSgBuf->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1;
225 pSgBuf->cbUsed = 0;
226 pSgBuf->cbAvailable = cbMin;
227 pSgBuf->pvUser = pHdr;
228 pSgBuf->cSegs = 1;
229 pSgBuf->aSegs[0].cbSeg = cbMin;
230
231 *ppSgBuf = pSgBuf;
232 return VINF_SUCCESS;
233 }
234
235 /*
236 * Arm the try again stuff.
237 */
238/** @todo if (pThis->pBufR3->cbSend >= cbMin * 2 + sizeof(INTNETHDR))
239 rc = VERR_TRY_AGAIN;
240 else */
241 rc = VERR_NO_MEMORY;
242
243 RTMemCacheFree(pThis->hSgCache, pSgBuf);
244 }
245 else
246 rc = VERR_NO_MEMORY;
247 /** @todo implement VERR_TRY_AGAIN */
248 return rc;
249}
250
251/**
252 * @interface_method_impl{PDMINETWORKUP,pfnFreeBuf}
253 */
254static DECLCALLBACK(int) drvR3IntNetUp_FreeBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf)
255{
256 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, INetworkUpR3);
257 PINTNETHDR pHdr = (PINTNETHDR)pSgBuf->pvUser;
258 Assert(pSgBuf->fFlags == (PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1));
259 Assert(pSgBuf->cbUsed <= pSgBuf->cbAvailable);
260 Assert(pHdr->u16Type == INTNETHDR_TYPE_FRAME);
261
262 /** @todo LATER: try unalloc the frame. */
263 pHdr->u16Type = INTNETHDR_TYPE_PADDING;
264 INTNETRingCommitFrame(&pThis->pBufR3->Send, pHdr);
265
266 RTMemCacheFree(pThis->hSgCache, pSgBuf);
267 return VINF_SUCCESS;
268}
269
270/**
271 * @interface_method_impl{PDMINETWORKUP,pfnSendBuf}
272 */
273static DECLCALLBACK(int) drvR3IntNetUp_SendBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf, bool fOnWorkerThread)
274{
275 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, INetworkUpR3);
276 STAM_PROFILE_START(&pThis->StatTransmit, a);
277
278 AssertPtr(pSgBuf);
279 Assert(pSgBuf->fFlags == (PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1));
280 Assert(pSgBuf->cbUsed <= pSgBuf->cbAvailable);
281
282 /*
283 * Commit the frame and push it thru the switch.
284 */
285 PINTNETHDR pHdr = (PINTNETHDR)pSgBuf->pvUser;
286 INTNETRingCommitFrameEx(&pThis->pBufR3->Send, pHdr, pSgBuf->cbUsed);
287
288 INTNETIFSENDREQ SendReq;
289 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
290 SendReq.Hdr.cbReq = sizeof(SendReq);
291 SendReq.pSession = NIL_RTR0PTR;
292 SendReq.hIf = pThis->hIf;
293 PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
294
295 RTMemCacheFree(pThis->hSgCache, pSgBuf);
296
297 STAM_PROFILE_STOP(&pThis->StatTransmit, a);
298 return VINF_SUCCESS;
299}
300
301/**
302 * @interface_method_impl{PDMINETWORKUP,pfnSendDeprecated}
303 */
304static DECLCALLBACK(int) drvR3IntNetUp_SendDeprecated(PPDMINETWORKUP pInterface, const void *pvBuf, size_t cb)
305{
306 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, INetworkUpR3);
307 STAM_PROFILE_START(&pThis->StatTransmit, a);
308
309#ifdef LOG_ENABLED
310 uint64_t u64Now = RTTimeProgramNanoTS();
311 LogFlow(("drvR3IntNetSend: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
312 cb, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
313 pThis->u64LastTransferTS = u64Now;
314 Log2(("drvR3IntNetSend: pvBuf=%p cb=%#x\n"
315 "%.*Rhxd\n",
316 pvBuf, cb, cb, pvBuf));
317#endif
318
319 /*
320 * Add the frame to the send buffer and push it onto the network.
321 */
322 int rc = INTNETRingWriteFrame(&pThis->pBufR3->Send, pvBuf, (uint32_t)cb);
323 if ( rc == VERR_BUFFER_OVERFLOW
324 && pThis->pBufR3->cbSend < cb)
325 {
326 INTNETIFSENDREQ SendReq;
327 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
328 SendReq.Hdr.cbReq = sizeof(SendReq);
329 SendReq.pSession = NIL_RTR0PTR;
330 SendReq.hIf = pThis->hIf;
331 PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
332
333 rc = INTNETRingWriteFrame(&pThis->pBufR3->Send, pvBuf, (uint32_t)cb);
334 }
335
336 if (RT_SUCCESS(rc))
337 {
338 INTNETIFSENDREQ SendReq;
339 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
340 SendReq.Hdr.cbReq = sizeof(SendReq);
341 SendReq.pSession = NIL_RTR0PTR;
342 SendReq.hIf = pThis->hIf;
343 rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
344 }
345
346 STAM_PROFILE_STOP(&pThis->StatTransmit, a);
347 AssertRC(rc);
348 return rc;
349}
350
351
352/**
353 * @interface_method_impl{PDMINETWORKUP,pfnSetPromiscuousMode}
354 */
355static DECLCALLBACK(void) drvR3IntNetUp_SetPromiscuousMode(PPDMINETWORKUP pInterface, bool fPromiscuous)
356{
357 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, INetworkUpR3);
358 INTNETIFSETPROMISCUOUSMODEREQ Req;
359 Req.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
360 Req.Hdr.cbReq = sizeof(Req);
361 Req.pSession = NIL_RTR0PTR;
362 Req.hIf = pThis->hIf;
363 Req.fPromiscuous = fPromiscuous;
364 int rc = PDMDrvHlpSUPCallVMMR0Ex(pThis->pDrvInsR3, VMMR0_DO_INTNET_IF_SET_PROMISCUOUS_MODE, &Req, sizeof(Req));
365 LogFlow(("drvR3IntNetUp_SetPromiscuousMode: fPromiscuous=%RTbool\n", fPromiscuous));
366 AssertRC(rc);
367}
368
369
370/**
371 * @interface_method_impl{PDMINETWORKUP,pfnNotifyLinkChanged}
372 */
373static DECLCALLBACK(void) drvR3IntNetUp_NotifyLinkChanged(PPDMINETWORKUP pInterface, PDMNETWORKLINKSTATE enmLinkState)
374{
375 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, INetworkUpR3);
376 bool fLinkDown;
377 switch (enmLinkState)
378 {
379 case PDMNETWORKLINKSTATE_DOWN:
380 case PDMNETWORKLINKSTATE_DOWN_RESUME:
381 fLinkDown = true;
382 break;
383 default:
384 AssertMsgFailed(("enmLinkState=%d\n", enmLinkState));
385 case PDMNETWORKLINKSTATE_UP:
386 fLinkDown = false;
387 break;
388 }
389 LogFlow(("drvR3IntNetUp_NotifyLinkChanged: enmLinkState=%d %d->%d\n", enmLinkState, pThis->fLinkDown, fLinkDown));
390 ASMAtomicXchgSize(&pThis->fLinkDown, fLinkDown);
391}
392
393
394/**
395 * Wait for space to become available up the driver/device chain.
396 *
397 * @returns VINF_SUCCESS if space is available.
398 * @returns VERR_STATE_CHANGED if the state changed.
399 * @returns VBox status code on other errors.
400 * @param pThis Pointer to the instance data.
401 */
402static int drvR3IntNetAsyncIoWaitForSpace(PDRVINTNET pThis)
403{
404 LogFlow(("drvR3IntNetAsyncIoWaitForSpace:\n"));
405 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
406 int rc = pThis->pIAboveNet->pfnWaitReceiveAvail(pThis->pIAboveNet, RT_INDEFINITE_WAIT);
407 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
408 LogFlow(("drvR3IntNetAsyncIoWaitForSpace: returns %Rrc\n", rc));
409 return rc;
410}
411
412
413/**
414 * Executes async I/O (RUNNING mode).
415 *
416 * @returns VERR_STATE_CHANGED if the state changed.
417 * @returns Appropriate VBox status code (error) on fatal error.
418 * @param pThis The driver instance data.
419 */
420static int drvR3IntNetAsyncIoRun(PDRVINTNET pThis)
421{
422 PPDMDRVINS pDrvIns = pThis->pDrvInsR3;
423 LogFlow(("drvR3IntNetAsyncIoRun: pThis=%p\n", pThis));
424
425 /*
426 * The running loop - processing received data and waiting for more to arrive.
427 */
428 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
429 PINTNETBUF pBuf = pThis->pBufR3;
430 PINTNETRINGBUF pRingBuf = &pBuf->Recv;
431 for (;;)
432 {
433 /*
434 * Process the receive buffer.
435 */
436 PINTNETHDR pHdr;
437 while ((pHdr = INTNETRingGetNextFrameToRead(pRingBuf)) != NULL)
438 {
439 /*
440 * Check the state and then inspect the packet.
441 */
442 if (pThis->enmState != ASYNCSTATE_RUNNING)
443 {
444 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
445 LogFlow(("drvR3IntNetAsyncIoRun: returns VERR_STATE_CHANGED (state changed - #0)\n"));
446 return VERR_STATE_CHANGED;
447 }
448
449 Log2(("pHdr=%p offRead=%#x: %.8Rhxs\n", pHdr, pRingBuf->offReadX, pHdr));
450 if ( pHdr->u16Type == INTNETHDR_TYPE_FRAME
451 && !pThis->fLinkDown)
452 {
453 /*
454 * Check if there is room for the frame and pass it up.
455 */
456 size_t cbFrame = pHdr->cbFrame;
457 int rc = pThis->pIAboveNet->pfnWaitReceiveAvail(pThis->pIAboveNet, 0);
458 if (rc == VINF_SUCCESS)
459 {
460#ifdef LOG_ENABLED
461 uint64_t u64Now = RTTimeProgramNanoTS();
462 LogFlow(("drvR3IntNetAsyncIoRun: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
463 cbFrame, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
464 pThis->u64LastReceiveTS = u64Now;
465 Log2(("drvR3IntNetAsyncIoRun: cbFrame=%#x\n"
466 "%.*Rhxd\n",
467 cbFrame, cbFrame, INTNETHdrGetFramePtr(pHdr, pBuf)));
468#endif
469 rc = pThis->pIAboveNet->pfnReceive(pThis->pIAboveNet, INTNETHdrGetFramePtr(pHdr, pBuf), cbFrame);
470 AssertRC(rc);
471
472 /* skip to the next frame. */
473 INTNETRingSkipFrame(pRingBuf);
474 }
475 else
476 {
477 /*
478 * Wait for sufficient space to become available and then retry.
479 */
480 rc = drvR3IntNetAsyncIoWaitForSpace(pThis);
481 if (RT_FAILURE(rc))
482 {
483 if (rc == VERR_INTERRUPTED)
484 {
485 /*
486 * NIC is going down, likely because the VM is being reset. Skip the frame.
487 */
488 AssertMsg( pHdr->u16Type == INTNETHDR_TYPE_FRAME
489 || pHdr->u16Type == INTNETHDR_TYPE_PADDING,
490 ("Unknown frame type %RX16! offRead=%#x\n", pHdr->u16Type, pRingBuf->offReadX));
491 INTNETRingSkipFrame(pRingBuf);
492 }
493 else
494 {
495 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
496 LogFlow(("drvR3IntNetAsyncIoRun: returns %Rrc (wait-for-space)\n", rc));
497 return rc;
498 }
499 }
500 }
501 }
502 else
503 {
504 /*
505 * Link down or unknown frame - skip to the next frame.
506 */
507 AssertMsg( pHdr->u16Type == INTNETHDR_TYPE_FRAME
508 || pHdr->u16Type == INTNETHDR_TYPE_PADDING,
509 ("Unknown frame type %RX16! offRead=%#x\n", pHdr->u16Type, pRingBuf->offReadX));
510 INTNETRingSkipFrame(pRingBuf);
511 STAM_REL_COUNTER_INC(&pBuf->cStatBadFrames);
512 }
513 } /* while more received data */
514
515 /*
516 * Wait for data, checking the state before we block.
517 */
518 if (pThis->enmState != ASYNCSTATE_RUNNING)
519 {
520 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
521 LogFlow(("drvR3IntNetAsyncIoRun: returns VINF_SUCCESS (state changed - #1)\n"));
522 return VERR_STATE_CHANGED;
523 }
524 INTNETIFWAITREQ WaitReq;
525 WaitReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
526 WaitReq.Hdr.cbReq = sizeof(WaitReq);
527 WaitReq.pSession = NIL_RTR0PTR;
528 WaitReq.hIf = pThis->hIf;
529 WaitReq.cMillies = 30000; /* 30s - don't wait forever, timeout now and then. */
530 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
531 int rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_WAIT, &WaitReq, sizeof(WaitReq));
532 if ( RT_FAILURE(rc)
533 && rc != VERR_TIMEOUT
534 && rc != VERR_INTERRUPTED)
535 {
536 LogFlow(("drvR3IntNetAsyncIoRun: returns %Rrc\n", rc));
537 return rc;
538 }
539 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
540 }
541}
542
543
544/**
545 * Asynchronous I/O thread for handling receive.
546 *
547 * @returns VINF_SUCCESS (ignored).
548 * @param ThreadSelf Thread handle.
549 * @param pvUser Pointer to a DRVINTNET structure.
550 */
551static DECLCALLBACK(int) drvR3IntNetAsyncIoThread(RTTHREAD ThreadSelf, void *pvUser)
552{
553 PDRVINTNET pThis = (PDRVINTNET)pvUser;
554 LogFlow(("drvR3IntNetAsyncIoThread: pThis=%p\n", pThis));
555 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
556
557 /*
558 * The main loop - acting on state.
559 */
560 for (;;)
561 {
562 ASYNCSTATE enmState = pThis->enmState;
563 switch (enmState)
564 {
565 case ASYNCSTATE_SUSPENDED:
566 {
567 int rc = RTSemEventWait(pThis->hEvtSuspended, 30000);
568 if ( RT_FAILURE(rc)
569 && rc != VERR_TIMEOUT)
570 {
571 LogFlow(("drvR3IntNetAsyncIoThread: returns %Rrc\n", rc));
572 return rc;
573 }
574 break;
575 }
576
577 case ASYNCSTATE_RUNNING:
578 {
579 int rc = drvR3IntNetAsyncIoRun(pThis);
580 if ( rc != VERR_STATE_CHANGED
581 && RT_FAILURE(rc))
582 {
583 LogFlow(("drvR3IntNetAsyncIoThread: returns %Rrc\n", rc));
584 return rc;
585 }
586 break;
587 }
588
589 default:
590 AssertMsgFailed(("Invalid state %d\n", enmState));
591 case ASYNCSTATE_TERMINATE:
592 LogFlow(("drvR3IntNetAsyncIoThread: returns VINF_SUCCESS\n"));
593 return VINF_SUCCESS;
594 }
595 }
596}
597
598/* -=-=-=-=- PDMIBASERC -=-=-=-=- */
599
600/**
601 * @interface_method_impl{PDMIBASERC,pfnQueryInterface}
602 */
603static DECLCALLBACK(RTRCPTR) drvR3IntNetIBaseRC_QueryInterface(PPDMIBASERC pInterface, const char *pszIID)
604{
605 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, IBaseRC);
606
607 PDMIBASERC_RETURN_INTERFACE(pThis->pDrvInsR3, pszIID, PDMIBASERC, &pThis->IBaseRC);
608 return NIL_RTRCPTR;
609}
610
611/* -=-=-=-=- PDMIBASER0 -=-=-=-=- */
612
613/**
614 * @interface_method_impl{PDMIBASER0,pfnQueryInterface}
615 */
616static DECLCALLBACK(RTR0PTR) drvR3IntNetIBaseR0_QueryInterface(PPDMIBASER0 pInterface, const char *pszIID)
617{
618 PDRVINTNET pThis = RT_FROM_MEMBER(pInterface, DRVINTNET, IBaseR0);
619
620 PDMIBASER0_RETURN_INTERFACE(pThis->pDrvInsR3, pszIID, PDMIBASER0, &pThis->IBaseR0);
621 return NIL_RTR0PTR;
622}
623
624/* -=-=-=-=- PDMIBASE -=-=-=-=- */
625
626/**
627 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
628 */
629static DECLCALLBACK(void *) drvR3IntNetIBase_QueryInterface(PPDMIBASE pInterface, const char *pszIID)
630{
631 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
632 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
633
634 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
635 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASER0, &pThis->IBaseR0);
636 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASERC, &pThis->IBaseRC);
637 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKUP, &pThis->INetworkUpR3);
638 return NULL;
639}
640
641/* -=-=-=-=- PDMDRVREG -=-=-=-=- */
642
643/**
644 * Power Off notification.
645 *
646 * @param pDrvIns The driver instance.
647 */
648static DECLCALLBACK(void) drvR3IntNetPowerOff(PPDMDRVINS pDrvIns)
649{
650 LogFlow(("drvR3IntNetPowerOff\n"));
651 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
652 if (!pThis->fActivateEarlyDeactivateLate)
653 {
654 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_SUSPENDED);
655 drvR3IntNetSetActive(pThis, false /* fActive */);
656 }
657}
658
659
660/**
661 * Resume notification.
662 *
663 * @param pDrvIns The driver instance.
664 */
665static DECLCALLBACK(void) drvR3IntNetResume(PPDMDRVINS pDrvIns)
666{
667 LogFlow(("drvR3IntNetPowerResume\n"));
668 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
669 if (!pThis->fActivateEarlyDeactivateLate)
670 {
671 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_RUNNING);
672 RTSemEventSignal(pThis->hEvtSuspended);
673 drvR3IntNetUpdateMacAddress(pThis); /* (could be a state restore) */
674 drvR3IntNetSetActive(pThis, true /* fActive */);
675 }
676 if ( PDMDrvHlpVMTeleportedAndNotFullyResumedYet(pDrvIns)
677 && pThis->pIAboveConfigR3)
678 {
679 /*
680 * We've just been teleported and need to drop a hint to the switch
681 * since we're likely to have changed to a different port. We just
682 * push out some ethernet frame that doesn't mean anything to anyone.
683 * For this purpose ethertype 0x801e was chosen since it was registered
684 * to Sun (dunno what it is/was used for though).
685 */
686 union
687 {
688 RTNETETHERHDR Hdr;
689 uint8_t ab[128];
690 } Frame;
691 RT_ZERO(Frame);
692 Frame.Hdr.DstMac.au16[0] = 0xffff;
693 Frame.Hdr.DstMac.au16[1] = 0xffff;
694 Frame.Hdr.DstMac.au16[2] = 0xffff;
695 Frame.Hdr.EtherType = RT_H2BE_U16(0x801e);
696 int rc = pThis->pIAboveConfigR3->pfnGetMac(pThis->pIAboveConfigR3, &Frame.Hdr.SrcMac);
697 if (RT_SUCCESS(rc))
698 rc = drvR3IntNetUp_SendDeprecated(&pThis->INetworkUpR3, &Frame, sizeof(Frame));
699 if (RT_FAILURE(rc))
700 LogRel(("IntNet#%u: Sending dummy frame failed: %Rrc\n", pDrvIns->iInstance, rc));
701 }
702}
703
704
705/**
706 * Suspend notification.
707 *
708 * @param pDrvIns The driver instance.
709 */
710static DECLCALLBACK(void) drvR3IntNetSuspend(PPDMDRVINS pDrvIns)
711{
712 LogFlow(("drvR3IntNetPowerSuspend\n"));
713 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
714 if (!pThis->fActivateEarlyDeactivateLate)
715 {
716 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_SUSPENDED);
717 drvR3IntNetSetActive(pThis, false /* fActive */);
718 }
719}
720
721
722/**
723 * Power On notification.
724 *
725 * @param pDrvIns The driver instance.
726 */
727static DECLCALLBACK(void) drvR3IntNetPowerOn(PPDMDRVINS pDrvIns)
728{
729 LogFlow(("drvR3IntNetPowerOn\n"));
730 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
731 if (!pThis->fActivateEarlyDeactivateLate)
732 {
733 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_RUNNING);
734 RTSemEventSignal(pThis->hEvtSuspended);
735 drvR3IntNetUpdateMacAddress(pThis);
736 drvR3IntNetSetActive(pThis, true /* fActive */);
737 }
738}
739
740
741/**
742 * Destruct a driver instance.
743 *
744 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
745 * resources can be freed correctly.
746 *
747 * @param pDrvIns The driver instance data.
748 */
749static DECLCALLBACK(void) drvR3IntNetDestruct(PPDMDRVINS pDrvIns)
750{
751 LogFlow(("drvR3IntNetDestruct\n"));
752 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
753 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
754
755 /*
756 * Indicate to the thread that it's time to quit.
757 */
758 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_TERMINATE);
759 ASMAtomicXchgSize(&pThis->fLinkDown, true);
760 RTSEMEVENT hEvtSuspended = pThis->hEvtSuspended;
761 pThis->hEvtSuspended = NIL_RTSEMEVENT;
762
763 /*
764 * Close the interface
765 */
766 if (pThis->hIf != INTNET_HANDLE_INVALID)
767 {
768 INTNETIFCLOSEREQ CloseReq;
769 CloseReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
770 CloseReq.Hdr.cbReq = sizeof(CloseReq);
771 CloseReq.pSession = NIL_RTR0PTR;
772 CloseReq.hIf = pThis->hIf;
773 pThis->hIf = INTNET_HANDLE_INVALID;
774 int rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_CLOSE, &CloseReq, sizeof(CloseReq));
775 AssertRC(rc);
776 }
777
778 /*
779 * Wait for the thread to terminate.
780 */
781 if (pThis->Thread != NIL_RTTHREAD)
782 {
783 if (hEvtSuspended != NIL_RTSEMEVENT)
784 RTSemEventSignal(hEvtSuspended);
785 int rc = RTThreadWait(pThis->Thread, 5000, NULL);
786 AssertRC(rc);
787 pThis->Thread = NIL_RTTHREAD;
788 }
789
790 /*
791 * Destroy the semaphore and S/G cache.
792 */
793 if (hEvtSuspended != NIL_RTSEMEVENT)
794 RTSemEventDestroy(hEvtSuspended);
795
796 RTMemCacheDestroy(pThis->hSgCache);
797 pThis->hSgCache = NIL_RTMEMCACHE;
798
799 if (pThis->pBufR3)
800 {
801 /*
802 * Deregister statistics in case we're being detached.
803 */
804 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Recv.cStatFrames);
805 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Recv.cbStatWritten);
806 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Recv.cOverflows);
807 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Send.cStatFrames);
808 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Send.cbStatWritten);
809 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->Send.cOverflows);
810 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatYieldsOk);
811 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatYieldsNok);
812 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatLost);
813 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->pBufR3->cStatBadFrames);
814#ifdef VBOX_WITH_STATISTICS
815 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReceive);
816 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatTransmit);
817#endif
818 }
819}
820
821
822/**
823 * Construct a TAP network transport driver instance.
824 *
825 * @copydoc FNPDMDRVCONSTRUCT
826 */
827static DECLCALLBACK(int) drvR3IntNetConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
828{
829 PDRVINTNET pThis = PDMINS_2_DATA(pDrvIns, PDRVINTNET);
830 bool f;
831 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
832
833 /*
834 * Init the static parts.
835 */
836 pThis->pDrvInsR3 = pDrvIns;
837 pThis->hIf = INTNET_HANDLE_INVALID;
838 pThis->Thread = NIL_RTTHREAD;
839 pThis->hEvtSuspended = NIL_RTSEMEVENT;
840 pThis->hSgCache = NIL_RTMEMCACHE;
841 pThis->enmState = ASYNCSTATE_SUSPENDED;
842 pThis->fActivateEarlyDeactivateLate = false;
843 /* IBase* */
844 pDrvIns->IBase.pfnQueryInterface = drvR3IntNetIBase_QueryInterface;
845 pThis->IBaseR0.pfnQueryInterface = drvR3IntNetIBaseR0_QueryInterface;
846 pThis->IBaseRC.pfnQueryInterface = drvR3IntNetIBaseRC_QueryInterface;
847 /* INetworkUp */
848 pThis->INetworkUpR3.pfnAllocBuf = drvR3IntNetUp_AllocBuf;
849 pThis->INetworkUpR3.pfnFreeBuf = drvR3IntNetUp_FreeBuf;
850 pThis->INetworkUpR3.pfnSendBuf = drvR3IntNetUp_SendBuf;
851 pThis->INetworkUpR3.pfnSendDeprecated = drvR3IntNetUp_SendDeprecated;
852 pThis->INetworkUpR3.pfnSetPromiscuousMode = drvR3IntNetUp_SetPromiscuousMode;
853 pThis->INetworkUpR3.pfnNotifyLinkChanged = drvR3IntNetUp_NotifyLinkChanged;
854
855 /*
856 * Validate the config.
857 */
858 if (!CFGMR3AreValuesValid(pCfg,
859 "Network\0"
860 "Trunk\0"
861 "TrunkType\0"
862 "ReceiveBufferSize\0"
863 "SendBufferSize\0"
864 "RestrictAccess\0"
865 "SharedMacOnWire\0"
866 "IgnoreAllPromisc\0"
867 "QuietlyIgnoreAllPromisc\0"
868 "IgnoreClientPromisc\0"
869 "QuietlyIgnoreClientPromisc\0"
870 "IgnoreTrunkWirePromisc\0"
871 "QuietlyIgnoreTrunkWirePromisc\0"
872 "IgnoreTrunkHostPromisc\0"
873 "QuietlyIgnoreTrunkHostPromisc\0"
874 "IsService\0"))
875 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
876
877 /*
878 * Check that no-one is attached to us.
879 */
880 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
881 ("Configuration error: Not possible to attach anything to this driver!\n"),
882 VERR_PDM_DRVINS_NO_ATTACH);
883
884 /*
885 * Query the network port interface.
886 */
887 pThis->pIAboveNet = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMINETWORKDOWN);
888 if (!pThis->pIAboveNet)
889 {
890 AssertMsgFailed(("Configuration error: the above device/driver didn't export the network port interface!\n"));
891 return VERR_PDM_MISSING_INTERFACE_ABOVE;
892 }
893 pThis->pIAboveConfigR3 = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMINETWORKCONFIG);
894
895 /*
896 * Read the configuration.
897 */
898 INTNETOPENREQ OpenReq;
899 memset(&OpenReq, 0, sizeof(OpenReq));
900 OpenReq.Hdr.cbReq = sizeof(OpenReq);
901 OpenReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
902 OpenReq.pSession = NIL_RTR0PTR;
903
904 /** @cfgm{Network, string}
905 * The name of the internal network to connect to.
906 */
907 int rc = CFGMR3QueryString(pCfg, "Network", OpenReq.szNetwork, sizeof(OpenReq.szNetwork));
908 if (RT_FAILURE(rc))
909 return PDMDRV_SET_ERROR(pDrvIns, rc,
910 N_("Configuration error: Failed to get the \"Network\" value"));
911 strcpy(pThis->szNetwork, OpenReq.szNetwork);
912
913 /** @cfgm{TrunkType, uint32_t, kIntNetTrunkType_None}
914 * The trunk connection type see INTNETTRUNKTYPE.
915 */
916 uint32_t u32TrunkType;
917 rc = CFGMR3QueryU32(pCfg, "TrunkType", &u32TrunkType);
918 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
919 u32TrunkType = kIntNetTrunkType_None;
920 else if (RT_FAILURE(rc))
921 return PDMDRV_SET_ERROR(pDrvIns, rc,
922 N_("Configuration error: Failed to get the \"TrunkType\" value"));
923 OpenReq.enmTrunkType = (INTNETTRUNKTYPE)u32TrunkType;
924
925 /** @cfgm{Trunk, string, ""}
926 * The name of the trunk connection.
927 */
928 rc = CFGMR3QueryString(pCfg, "Trunk", OpenReq.szTrunk, sizeof(OpenReq.szTrunk));
929 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
930 OpenReq.szTrunk[0] = '\0';
931 else if (RT_FAILURE(rc))
932 return PDMDRV_SET_ERROR(pDrvIns, rc,
933 N_("Configuration error: Failed to get the \"Trunk\" value"));
934
935 /** @cfgm{RestrictAccess, boolean, true}
936 * Whether to restrict the access to the network or if it should be public. Everyone on
937 * the computer can connect to a public network. Don't change this.
938 */
939 bool fRestrictAccess;
940 rc = CFGMR3QueryBool(pCfg, "RestrictAccess", &fRestrictAccess);
941 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
942 fRestrictAccess = true;
943 else if (RT_FAILURE(rc))
944 return PDMDRV_SET_ERROR(pDrvIns, rc,
945 N_("Configuration error: Failed to get the \"RestrictAccess\" value"));
946 OpenReq.fFlags = fRestrictAccess ? 0 : INTNET_OPEN_FLAGS_PUBLIC;
947
948 /** @cfgm{IgnoreAllPromisc, boolean, false}
949 * When set all request for operating any interface or trunk in promiscuous
950 * mode will be ignored. */
951 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreAllPromisc", &f, false);
952 if (RT_FAILURE(rc))
953 return PDMDRV_SET_ERROR(pDrvIns, rc,
954 N_("Configuration error: Failed to get the \"IgnoreAllPromisc\" value"));
955 if (f)
956 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC;
957
958 /** @cfgm{QuietlyIgnoreAllPromisc, boolean, false}
959 * When set all request for operating any interface or trunk in promiscuous
960 * mode will be ignored. This differs from IgnoreAllPromisc in that clients
961 * won't get VERR_INTNET_INCOMPATIBLE_FLAGS. */
962 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreAllPromisc", &f, false);
963 if (RT_FAILURE(rc))
964 return PDMDRV_SET_ERROR(pDrvIns, rc,
965 N_("Configuration error: Failed to get the \"QuietlyIgnoreAllPromisc\" value"));
966 if (f)
967 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC;
968
969 /** @cfgm{IgnoreClientPromisc, boolean, false}
970 * When set all request for operating any non-trunk interface in promiscuous
971 * mode will be ignored. */
972 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreClientPromisc", &f, false);
973 if (RT_FAILURE(rc))
974 return PDMDRV_SET_ERROR(pDrvIns, rc,
975 N_("Configuration error: Failed to get the \"IgnoreClientPromisc\" value"));
976 if (f)
977 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC; /** @todo add special flag for this. */
978
979 /** @cfgm{QuietlyIgnoreClientPromisc, boolean, false}
980 * When set all request for operating any non-trunk interface promiscuous mode
981 * will be ignored. This differs from IgnoreClientPromisc in that clients won't
982 * get VERR_INTNET_INCOMPATIBLE_FLAGS. */
983 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreClientPromisc", &f, false);
984 if (RT_FAILURE(rc))
985 return PDMDRV_SET_ERROR(pDrvIns, rc,
986 N_("Configuration error: Failed to get the \"QuietlyIgnoreClientPromisc\" value"));
987 if (f)
988 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC; /** @todo add special flag for this. */
989
990 /** @cfgm{IgnoreTrunkWirePromisc, boolean, false}
991 * When set all request for operating the trunk-wire connection in promiscuous
992 * mode will be ignored. */
993 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreTrunkWirePromisc", &f, false);
994 if (RT_FAILURE(rc))
995 return PDMDRV_SET_ERROR(pDrvIns, rc,
996 N_("Configuration error: Failed to get the \"IgnoreTrunkWirePromisc\" value"));
997 if (f)
998 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC_TRUNK_WIRE;
999
1000 /** @cfgm{QuietlyIgnoreTrunkWirePromisc, boolean, false}
1001 * When set all request for operating any trunk-wire connection promiscuous mode
1002 * will be ignored. This differs from IgnoreTrunkWirePromisc in that clients
1003 * won't get VERR_INTNET_INCOMPATIBLE_FLAGS. */
1004 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreTrunkWirePromisc", &f, false);
1005 if (RT_FAILURE(rc))
1006 return PDMDRV_SET_ERROR(pDrvIns, rc,
1007 N_("Configuration error: Failed to get the \"QuietlyIgnoreTrunkWirePromisc\" value"));
1008 if (f)
1009 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC_TRUNK_WIRE;
1010
1011 /** @cfgm{IgnoreTrunkHostPromisc, boolean, false}
1012 * When set all request for operating the trunk-host connection in promiscuous
1013 * mode will be ignored. */
1014 rc = CFGMR3QueryBoolDef(pCfg, "IgnoreTrunkHostPromisc", &f, false);
1015 if (RT_FAILURE(rc))
1016 return PDMDRV_SET_ERROR(pDrvIns, rc,
1017 N_("Configuration error: Failed to get the \"IgnoreTrunkHostPromisc\" value"));
1018 if (f)
1019 OpenReq.fFlags |= INTNET_OPEN_FLAGS_IGNORE_PROMISC_TRUNK_HOST;
1020
1021 /** @cfgm{QuietlyIgnoreTrunkHostPromisc, boolean, false}
1022 * When set all request for operating any trunk-host connection promiscuous mode
1023 * will be ignored. This differs from IgnoreTrunkHostPromisc in that clients
1024 * won't get VERR_INTNET_INCOMPATIBLE_FLAGS. */
1025 rc = CFGMR3QueryBoolDef(pCfg, "QuietlyIgnoreTrunkHostPromisc", &f, false);
1026 if (RT_FAILURE(rc))
1027 return PDMDRV_SET_ERROR(pDrvIns, rc,
1028 N_("Configuration error: Failed to get the \"QuietlyIgnoreTrunkHostPromisc\" value"));
1029 if (f)
1030 OpenReq.fFlags |= INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC_TRUNK_HOST;
1031
1032 /** @todo flags for not sending to the host and for setting the trunk-wire
1033 * connection in promiscuous mode. */
1034
1035
1036 /** @cfgm{SharedMacOnWire, boolean, false}
1037 * Whether to shared the MAC address of the host interface when using the wire. When
1038 * attaching to a wireless NIC this option is usally a requirement.
1039 */
1040 bool fSharedMacOnWire;
1041 rc = CFGMR3QueryBoolDef(pCfg, "SharedMacOnWire", &fSharedMacOnWire, false);
1042 if (RT_FAILURE(rc))
1043 return PDMDRV_SET_ERROR(pDrvIns, rc,
1044 N_("Configuration error: Failed to get the \"SharedMacOnWire\" value"));
1045 if (fSharedMacOnWire)
1046 OpenReq.fFlags |= INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE;
1047
1048 /** @cfgm{ReceiveBufferSize, uint32_t, 218 KB}
1049 * The size of the receive buffer.
1050 */
1051 rc = CFGMR3QueryU32(pCfg, "ReceiveBufferSize", &OpenReq.cbRecv);
1052 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1053 OpenReq.cbRecv = 218 * _1K ;
1054 else if (RT_FAILURE(rc))
1055 return PDMDRV_SET_ERROR(pDrvIns, rc,
1056 N_("Configuration error: Failed to get the \"ReceiveBufferSize\" value"));
1057
1058 /** @cfgm{SendBufferSize, uint32_t, 36 KB}
1059 * The size of the send (transmit) buffer.
1060 * This should be more than twice the size of the larges frame size because
1061 * the ring buffer is very simple and doesn't support splitting up frames
1062 * nor inserting padding. So, if this is too close to the frame size the
1063 * header will fragment the buffer such that the frame won't fit on either
1064 * side of it and the code will get very upset about it all.
1065 */
1066 rc = CFGMR3QueryU32(pCfg, "SendBufferSize", &OpenReq.cbSend);
1067 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1068 OpenReq.cbSend = 36*_1K;
1069 else if (RT_FAILURE(rc))
1070 return PDMDRV_SET_ERROR(pDrvIns, rc,
1071 N_("Configuration error: Failed to get the \"SendBufferSize\" value"));
1072 if (OpenReq.cbSend < 32)
1073 return PDMDRV_SET_ERROR(pDrvIns, rc,
1074 N_("Configuration error: The \"SendBufferSize\" value is too small"));
1075 if (OpenReq.cbSend < 16384*2 + 64)
1076 LogRel(("DrvIntNet: Warning! SendBufferSize=%u, Recommended minimum size %u butes.\n", OpenReq.cbSend, 16384*2 + 64));
1077
1078 /** @cfgm{IsService, boolean, true}
1079 * This alterns the way the thread is suspended and resumed. When it's being used by
1080 * a service such as LWIP/iSCSI it shouldn't suspend immediately like for a NIC.
1081 */
1082 rc = CFGMR3QueryBool(pCfg, "IsService", &pThis->fActivateEarlyDeactivateLate);
1083 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
1084 pThis->fActivateEarlyDeactivateLate = false;
1085 else if (RT_FAILURE(rc))
1086 return PDMDRV_SET_ERROR(pDrvIns, rc,
1087 N_("Configuration error: Failed to get the \"IsService\" value"));
1088
1089 LogRel(("IntNet#%u: szNetwork={%s} enmTrunkType=%d szTrunk={%s} fFlags=%#x cbRecv=%u cbSend=%u\n",
1090 pDrvIns->iInstance, OpenReq.szNetwork, OpenReq.enmTrunkType, OpenReq.szTrunk, OpenReq.fFlags,
1091 OpenReq.cbRecv, OpenReq.cbSend));
1092
1093#ifdef RT_OS_DARWIN
1094 /* Temporary hack: attach to a network with the name 'if=en0' and you're hitting the wire. */
1095 if ( !OpenReq.szTrunk[0]
1096 && OpenReq.enmTrunkType == kIntNetTrunkType_None
1097 && !strncmp(pThis->szNetwork, "if=en", sizeof("if=en") - 1)
1098 && RT_C_IS_DIGIT(pThis->szNetwork[sizeof("if=en") - 1])
1099 && !pThis->szNetwork[sizeof("if=en")])
1100 {
1101 OpenReq.enmTrunkType = kIntNetTrunkType_NetFlt;
1102 strcpy(OpenReq.szTrunk, &pThis->szNetwork[sizeof("if=") - 1]);
1103 }
1104 /* Temporary hack: attach to a network with the name 'wif=en0' and you're on the air. */
1105 if ( !OpenReq.szTrunk[0]
1106 && OpenReq.enmTrunkType == kIntNetTrunkType_None
1107 && !strncmp(pThis->szNetwork, "wif=en", sizeof("wif=en") - 1)
1108 && RT_C_IS_DIGIT(pThis->szNetwork[sizeof("wif=en") - 1])
1109 && !pThis->szNetwork[sizeof("wif=en")])
1110 {
1111 OpenReq.enmTrunkType = kIntNetTrunkType_NetFlt;
1112 OpenReq.fFlags |= INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE;
1113 strcpy(OpenReq.szTrunk, &pThis->szNetwork[sizeof("wif=") - 1]);
1114 }
1115#endif /* DARWIN */
1116
1117 /*
1118 * Create the event semaphore and S/G cache.
1119 */
1120 rc = RTSemEventCreate(&pThis->hEvtSuspended);
1121 if (RT_FAILURE(rc))
1122 return rc;
1123 rc = RTMemCacheCreate(&pThis->hSgCache, sizeof(PDMSCATTERGATHER), 0, UINT32_MAX, NULL, NULL, pThis, 0);
1124 if (RT_FAILURE(rc))
1125 return rc;
1126
1127 /*
1128 * Create the interface.
1129 */
1130 OpenReq.hIf = INTNET_HANDLE_INVALID;
1131 rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_OPEN, &OpenReq, sizeof(OpenReq));
1132 if (RT_FAILURE(rc))
1133 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1134 N_("Failed to open/create the internal network '%s'"), pThis->szNetwork);
1135 AssertRelease(OpenReq.hIf != INTNET_HANDLE_INVALID);
1136 pThis->hIf = OpenReq.hIf;
1137 Log(("IntNet%d: hIf=%RX32 '%s'\n", pDrvIns->iInstance, pThis->hIf, pThis->szNetwork));
1138
1139 /*
1140 * Get default buffer.
1141 */
1142 INTNETIFGETRING3BUFFERREQ GetRing3BufferReq;
1143 GetRing3BufferReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
1144 GetRing3BufferReq.Hdr.cbReq = sizeof(GetRing3BufferReq);
1145 GetRing3BufferReq.pSession = NIL_RTR0PTR;
1146 GetRing3BufferReq.hIf = pThis->hIf;
1147 GetRing3BufferReq.pRing3Buf = NULL;
1148 rc = PDMDrvHlpSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_GET_RING3_BUFFER, &GetRing3BufferReq, sizeof(GetRing3BufferReq));
1149 if (RT_FAILURE(rc))
1150 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1151 N_("Failed to get ring-3 buffer for the newly created interface to '%s'"), pThis->szNetwork);
1152 AssertRelease(VALID_PTR(GetRing3BufferReq.pRing3Buf));
1153 pThis->pBufR3 = GetRing3BufferReq.pRing3Buf;
1154
1155 /*
1156 * Create the async I/O thread.
1157 * Note! Using a PDM thread here doesn't fit with the IsService=true operation.
1158 */
1159 rc = RTThreadCreate(&pThis->Thread, drvR3IntNetAsyncIoThread, pThis, _128K, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "INTNET");
1160 if (RT_FAILURE(rc))
1161 {
1162 AssertRC(rc);
1163 return rc;
1164 }
1165 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->Recv.cbStatWritten, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Number of received bytes.", "/Net/IntNet%d/Bytes/Received", pDrvIns->iInstance);
1166 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->Send.cbStatWritten, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Number of sent bytes.", "/Net/IntNet%d/Bytes/Sent", pDrvIns->iInstance);
1167 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->Recv.cOverflows, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number overflows.", "/Net/IntNet%d/Overflows/Recv", pDrvIns->iInstance);
1168 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->Send.cOverflows, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number overflows.", "/Net/IntNet%d/Overflows/Sent", pDrvIns->iInstance);
1169 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->Recv.cStatFrames, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of received packets.", "/Net/IntNet%d/Packets/Received", pDrvIns->iInstance);
1170 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->Send.cStatFrames, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of sent packets.", "/Net/IntNet%d/Packets/Sent", pDrvIns->iInstance);
1171 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->cStatLost, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of lost packets.", "/Net/IntNet%d/Packets/Lost", pDrvIns->iInstance);
1172 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->cStatYieldsNok, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of times yielding helped fix an overflow.", "/Net/IntNet%d/YieldOk", pDrvIns->iInstance);
1173 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->cStatYieldsOk, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of times yielding didn't help fix an overflow.", "/Net/IntNet%d/YieldNok", pDrvIns->iInstance);
1174 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->pBufR3->cStatBadFrames, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_COUNT, "Number of bad frames seed by the consumers.", "/Net/IntNet%d/BadFrames", pDrvIns->iInstance);
1175#ifdef VBOX_WITH_STATISTICS
1176 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReceive, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling packet receive runs.", "/Net/IntNet%d/Receive", pDrvIns->iInstance);
1177 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatTransmit, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling packet transmit runs.", "/Net/IntNet%d/Transmit", pDrvIns->iInstance);
1178#endif
1179
1180 /*
1181 * Activate data transmission as early as possible
1182 */
1183 if (pThis->fActivateEarlyDeactivateLate)
1184 {
1185 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_RUNNING);
1186 RTSemEventSignal(pThis->hEvtSuspended);
1187 drvR3IntNetUpdateMacAddress(pThis);
1188 drvR3IntNetSetActive(pThis, true /* fActive */);
1189 }
1190
1191 return rc;
1192}
1193
1194
1195/**
1196 * Internal networking transport driver registration record.
1197 */
1198const PDMDRVREG g_DrvIntNet =
1199{
1200 /* u32Version */
1201 PDM_DRVREG_VERSION,
1202 /* szName */
1203 "IntNet",
1204 /* szRCMod */
1205 "VBoxDD",
1206 /* szR0Mod */
1207 "VBoxDD",
1208 /* pszDescription */
1209 "Internal Networking Transport Driver",
1210 /* fFlags */
1211#ifdef VBOX_WITH_R0_AND_RC_DRIVERS
1212 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT | PDM_DRVREG_FLAGS_R0 | PDM_DRVREG_FLAGS_RC,
1213#else
1214 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1215#endif
1216 /* fClass. */
1217 PDM_DRVREG_CLASS_NETWORK,
1218 /* cMaxInstances */
1219 ~0,
1220 /* cbInstance */
1221 sizeof(DRVINTNET),
1222 /* pfnConstruct */
1223 drvR3IntNetConstruct,
1224 /* pfnDestruct */
1225 drvR3IntNetDestruct,
1226 /* pfnRelocate */
1227 NULL,
1228 /* pfnIOCtl */
1229 NULL,
1230 /* pfnPowerOn */
1231 drvR3IntNetPowerOn,
1232 /* pfnReset */
1233 NULL,
1234 /* pfnSuspend */
1235 drvR3IntNetSuspend,
1236 /* pfnResume */
1237 drvR3IntNetResume,
1238 /* pfnAttach */
1239 NULL,
1240 /* pfnDetach */
1241 NULL,
1242 /* pfnPowerOff */
1243 drvR3IntNetPowerOff,
1244 /* pfnSoftReset */
1245 NULL,
1246 /* u32EndVersion */
1247 PDM_DRVREG_VERSION
1248};
1249
1250#endif /* IN_RING3 */
1251
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