VirtualBox

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

Last change on this file since 23194 was 22277, checked in by vboxsync, 15 years ago

PDMDRVREG change (big changeset).

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