VirtualBox

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

Last change on this file since 10757 was 10746, checked in by vboxsync, 16 years ago

Added pSession argument to all the intnet request packets.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.2 KB
Line 
1/* $Id: DrvIntNet.cpp 10746 2008-07-18 12:43:53Z 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/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_DRV_INTNET
27#include <VBox/pdmdrv.h>
28#include <VBox/cfgm.h>
29#include <VBox/intnet.h>
30#include <VBox/vmm.h>
31#include <VBox/err.h>
32
33#include <VBox/log.h>
34#include <iprt/asm.h>
35#include <iprt/assert.h>
36#include <iprt/thread.h>
37#include <iprt/semaphore.h>
38#include <iprt/string.h>
39#include <iprt/time.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 /** Pointer to the driver instance. */
72 PPDMDRVINS pDrvIns;
73 /** Interface handle. */
74 INTNETIFHANDLE hIf;
75 /** Pointer to the communication buffer. */
76 PINTNETBUF pBuf;
77 /** The thread state. */
78 ASYNCSTATE volatile enmState;
79 /** Reader thread. */
80 RTTHREAD Thread;
81 /** Event semaphore the Thread waits on while the VM is suspended. */
82 RTSEMEVENT EventSuspended;
83 /** Set if the link is down.
84 * When the link is down all incoming packets will be dropped. */
85 bool volatile fLinkDown;
86 /** Set if data transmission should start immediately and deactivate
87 * as late as possible. */
88 bool fActivateEarlyDeactivateLate;
89
90#ifdef VBOX_WITH_STATISTICS
91 /** Profiling packet transmit runs. */
92 STAMPROFILE StatTransmit;
93 /** Profiling packet receive runs. */
94 STAMPROFILEADV StatReceive;
95#endif /* VBOX_WITH_STATISTICS */
96
97#ifdef LOG_ENABLED
98 /** The nano ts of the last transfer. */
99 uint64_t u64LastTransferTS;
100 /** The nano ts of the last receive. */
101 uint64_t u64LastReceiveTS;
102#endif
103 /** The network name. */
104 char szNetwork[INTNET_MAX_NETWORK_NAME];
105} DRVINTNET, *PDRVINTNET;
106
107
108/** Converts a pointer to DRVINTNET::INetworkConnector to a PDRVINTNET. */
109#define PDMINETWORKCONNECTOR_2_DRVINTNET(pInterface) ( (PDRVINTNET)((uintptr_t)pInterface - RT_OFFSETOF(DRVINTNET, INetworkConnector)) )
110
111
112/**
113 * Writes a frame packet to the buffer.
114 *
115 * @returns VBox status code.
116 * @param pBuf The buffer.
117 * @param pRingBuf The ring buffer to read from.
118 * @param pvFrame The frame to write.
119 * @param cbFrame The size of the frame.
120 * @remark This is the same as INTNETRingWriteFrame
121 */
122static int drvIntNetRingWriteFrame(PINTNETBUF pBuf, PINTNETRINGBUF pRingBuf, const void *pvFrame, uint32_t cbFrame)
123{
124 /*
125 * Validate input.
126 */
127 Assert(pBuf);
128 Assert(pRingBuf);
129 Assert(pvFrame);
130 Assert(cbFrame >= sizeof(PDMMAC) * 2);
131 uint32_t offWrite = pRingBuf->offWrite;
132 Assert(offWrite == RT_ALIGN_32(offWrite, sizeof(INTNETHDR)));
133 uint32_t offRead = pRingBuf->offRead;
134 Assert(offRead == RT_ALIGN_32(offRead, sizeof(INTNETHDR)));
135
136 const uint32_t cb = RT_ALIGN_32(cbFrame, sizeof(INTNETHDR));
137 if (offRead <= offWrite)
138 {
139 /*
140 * Try fit it all before the end of the buffer.
141 */
142 if (pRingBuf->offEnd - offWrite >= cb + sizeof(INTNETHDR))
143 {
144 PINTNETHDR pHdr = (PINTNETHDR)((uint8_t *)pBuf + offWrite);
145 pHdr->u16Type = INTNETHDR_TYPE_FRAME;
146 pHdr->cbFrame = cbFrame;
147 pHdr->offFrame = sizeof(INTNETHDR);
148
149 memcpy(pHdr + 1, pvFrame, cbFrame);
150
151 offWrite += cb + sizeof(INTNETHDR);
152 Assert(offWrite <= pRingBuf->offEnd && offWrite >= pRingBuf->offStart);
153 if (offWrite >= pRingBuf->offEnd)
154 offWrite = pRingBuf->offStart;
155 Log2(("WriteFrame: offWrite: %#x -> %#x (1)\n", pRingBuf->offWrite, offWrite));
156 ASMAtomicXchgU32(&pRingBuf->offWrite, offWrite);
157 return VINF_SUCCESS;
158 }
159
160 /*
161 * Try fit the frame at the start of the buffer.
162 * (The header fits before the end of the buffer because of alignment.)
163 */
164 AssertMsg(pRingBuf->offEnd - offWrite >= sizeof(INTNETHDR), ("offEnd=%x offWrite=%x\n", pRingBuf->offEnd, offWrite));
165 if (offRead - pRingBuf->offStart > cb) /* not >= ! */
166 {
167 PINTNETHDR pHdr = (PINTNETHDR)((uint8_t *)pBuf + offWrite);
168 void *pvFrameOut = (PINTNETHDR)((uint8_t *)pBuf + pRingBuf->offStart);
169 pHdr->u16Type = INTNETHDR_TYPE_FRAME;
170 pHdr->cbFrame = cbFrame;
171 pHdr->offFrame = (intptr_t)pvFrameOut - (intptr_t)pHdr;
172
173 memcpy(pvFrameOut, pvFrame, cbFrame);
174
175 offWrite = pRingBuf->offStart + cb;
176 ASMAtomicXchgU32(&pRingBuf->offWrite, offWrite);
177 Log2(("WriteFrame: offWrite: %#x -> %#x (2)\n", pRingBuf->offWrite, offWrite));
178 return VINF_SUCCESS;
179 }
180 }
181 /*
182 * The reader is ahead of the writer, try fit it into that space.
183 */
184 else if (offRead - offWrite > cb + sizeof(INTNETHDR)) /* not >= ! */
185 {
186 PINTNETHDR pHdr = (PINTNETHDR)((uint8_t *)pBuf + offWrite);
187 pHdr->u16Type = INTNETHDR_TYPE_FRAME;
188 pHdr->cbFrame = cbFrame;
189 pHdr->offFrame = sizeof(INTNETHDR);
190
191 memcpy(pHdr + 1, pvFrame, cbFrame);
192
193 offWrite += cb + sizeof(INTNETHDR);
194 ASMAtomicXchgU32(&pRingBuf->offWrite, offWrite);
195 Log2(("WriteFrame: offWrite: %#x -> %#x (3)\n", pRingBuf->offWrite, offWrite));
196 return VINF_SUCCESS;
197 }
198
199 /* (it didn't fit) */
200 /** @todo stats */
201 return VERR_BUFFER_OVERFLOW;
202}
203
204
205/**
206 * Send data to the network.
207 *
208 * @returns VBox status code.
209 * @param pInterface Pointer to the interface structure containing the called function pointer.
210 * @param pvBuf Data to send.
211 * @param cb Number of bytes to send.
212 * @thread EMT
213 */
214static DECLCALLBACK(int) drvIntNetSend(PPDMINETWORKCONNECTOR pInterface, const void *pvBuf, size_t cb)
215{
216 PDRVINTNET pThis = PDMINETWORKCONNECTOR_2_DRVINTNET(pInterface);
217 STAM_PROFILE_START(&pThis->StatTransmit, a);
218
219#ifdef LOG_ENABLED
220 uint64_t u64Now = RTTimeProgramNanoTS();
221 LogFlow(("drvIntNetSend: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
222 cb, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
223 pThis->u64LastTransferTS = u64Now;
224 Log2(("drvIntNetSend: pvBuf=%p cb=%#x\n"
225 "%.*Vhxd\n",
226 pvBuf, cb, cb, pvBuf));
227#endif
228
229 /*
230 * Add the frame to the send buffer and push it onto the network.
231 */
232 int rc = drvIntNetRingWriteFrame(pThis->pBuf, &pThis->pBuf->Send, pvBuf, cb);
233 if ( rc == VERR_BUFFER_OVERFLOW
234 && pThis->pBuf->cbSend < cb)
235 {
236 INTNETIFSENDREQ SendReq;
237 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
238 SendReq.Hdr.cbReq = sizeof(SendReq);
239 SendReq.pSession = NULL;
240 SendReq.hIf = pThis->hIf;
241 pThis->pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pThis->pDrvIns, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
242
243 rc = drvIntNetRingWriteFrame(pThis->pBuf, &pThis->pBuf->Send, pvBuf, cb);
244 }
245
246 if (RT_SUCCESS(rc))
247 {
248 INTNETIFSENDREQ SendReq;
249 SendReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
250 SendReq.Hdr.cbReq = sizeof(SendReq);
251 SendReq.pSession = NULL;
252 SendReq.hIf = pThis->hIf;
253 rc = pThis->pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pThis->pDrvIns, VMMR0_DO_INTNET_IF_SEND, &SendReq, sizeof(SendReq));
254 }
255
256 STAM_PROFILE_STOP(&pThis->StatTransmit, a);
257 AssertRC(rc);
258 return rc;
259}
260
261
262/**
263 * Set promiscuous mode.
264 *
265 * This is called when the promiscuous mode is set. This means that there doesn't have
266 * to be a mode change when it's called.
267 *
268 * @param pInterface Pointer to the interface structure containing the called function pointer.
269 * @param fPromiscuous Set if the adaptor is now in promiscuous mode. Clear if it is not.
270 * @thread EMT
271 */
272static DECLCALLBACK(void) drvIntNetSetPromiscuousMode(PPDMINETWORKCONNECTOR pInterface, bool fPromiscuous)
273{
274 PDRVINTNET pThis = PDMINETWORKCONNECTOR_2_DRVINTNET(pInterface);
275 INTNETIFSETPROMISCUOUSMODEREQ Req;
276 Req.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
277 Req.Hdr.cbReq = sizeof(Req);
278 Req.pSession = NULL;
279 Req.hIf = pThis->hIf;
280 Req.fPromiscuous = fPromiscuous;
281 int rc = pThis->pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pThis->pDrvIns, VMMR0_DO_INTNET_IF_SET_PROMISCUOUS_MODE, &Req, sizeof(Req));
282 LogFlow(("drvIntNetSetPromiscuousMode: fPromiscuous=%RTbool\n", fPromiscuous));
283 AssertRC(rc);
284}
285
286
287/**
288 * Notification on link status changes.
289 *
290 * @param pInterface Pointer to the interface structure containing the called function pointer.
291 * @param enmLinkState The new link state.
292 * @thread EMT
293 */
294static DECLCALLBACK(void) drvIntNetNotifyLinkChanged(PPDMINETWORKCONNECTOR pInterface, PDMNETWORKLINKSTATE enmLinkState)
295{
296 PDRVINTNET pThis = PDMINETWORKCONNECTOR_2_DRVINTNET(pInterface);
297 bool fLinkDown;
298 switch (enmLinkState)
299 {
300 case PDMNETWORKLINKSTATE_DOWN:
301 case PDMNETWORKLINKSTATE_DOWN_RESUME:
302 fLinkDown = true;
303 break;
304 default:
305 AssertMsgFailed(("enmLinkState=%d\n", enmLinkState));
306 case PDMNETWORKLINKSTATE_UP:
307 fLinkDown = false;
308 break;
309 }
310 LogFlow(("drvIntNetNotifyLinkChanged: enmLinkState=%d %d->%d\n", enmLinkState, pThis->fLinkDown, fLinkDown));
311 ASMAtomicXchgSize(&pThis->fLinkDown, fLinkDown);
312}
313
314
315/**
316 * Wait for space to become available up the driver/device chain.
317 *
318 * @returns VINF_SUCCESS if space is available.
319 * @returns VERR_STATE_CHANGED if the state changed.
320 * @returns VBox status code on other errors.
321 * @param pThis Pointer to the instance data.
322 */
323static int drvIntNetAsyncIoWaitForSpace(PDRVINTNET pThis)
324{
325 LogFlow(("drvIntNetAsyncIoWaitForSpace:\n"));
326 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
327 int rc = pThis->pPort->pfnWaitReceiveAvail(pThis->pPort, RT_INDEFINITE_WAIT);
328 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
329 LogFlow(("drvIntNetAsyncIoWaitForSpace: returns %Vrc\n", rc));
330 return rc;
331}
332
333
334/**
335 * Executes async I/O (RUNNING mode).
336 *
337 * @returns VERR_STATE_CHANGED if the state changed.
338 * @returns Appropriate VBox status code (error) on fatal error.
339 * @param pThis The driver instance data.
340 */
341static int drvIntNetAsyncIoRun(PDRVINTNET pThis)
342{
343 PPDMDRVINS pDrvIns = pThis->pDrvIns;
344 LogFlow(("drvIntNetAsyncIoRun: pThis=%p\n", pThis));
345
346 /*
347 * The running loop - processing received data and waiting for more to arrive.
348 */
349 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
350 PINTNETBUF pBuf = pThis->pBuf;
351 PINTNETRINGBUF pRingBuf = &pThis->pBuf->Recv;
352 for (;;)
353 {
354 /*
355 * Process the receive buffer.
356 */
357 while (INTNETRingGetReadable(pRingBuf) > 0)
358 {
359 /*
360 * Check the state and then inspect the packet.
361 */
362 if (pThis->enmState != ASYNCSTATE_RUNNING)
363 {
364 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
365 LogFlow(("drvIntNetAsyncIoRun: returns VERR_STATE_CHANGED (state changed - #0)\n"));
366 return VERR_STATE_CHANGED;
367 }
368
369 PINTNETHDR pHdr = (PINTNETHDR)((uintptr_t)pBuf + pRingBuf->offRead);
370 Log2(("pHdr=%p offRead=%#x: %.8Rhxs\n", pHdr, pRingBuf->offRead, pHdr));
371 if ( pHdr->u16Type == INTNETHDR_TYPE_FRAME
372 && !pThis->fLinkDown)
373 {
374 /*
375 * Check if there is room for the frame and pass it up.
376 */
377 size_t cbFrame = pHdr->cbFrame;
378 int rc = pThis->pPort->pfnWaitReceiveAvail(pThis->pPort, 0);
379 if (rc == VINF_SUCCESS)
380 {
381#ifdef LOG_ENABLED
382 uint64_t u64Now = RTTimeProgramNanoTS();
383 LogFlow(("drvIntNetAsyncIoRun: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
384 cbFrame, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
385 pThis->u64LastReceiveTS = u64Now;
386 Log2(("drvIntNetAsyncIoRun: cbFrame=%#x\n"
387 "%.*Vhxd\n",
388 cbFrame, cbFrame, INTNETHdrGetFramePtr(pHdr, pBuf)));
389#endif
390 int rc = pThis->pPort->pfnReceive(pThis->pPort, INTNETHdrGetFramePtr(pHdr, pBuf), cbFrame);
391 AssertRC(rc);
392
393 /* skip to the next frame. */
394 INTNETRingSkipFrame(pBuf, pRingBuf);
395 }
396 else
397 {
398 /*
399 * Wait for sufficient space to become available and then retry.
400 */
401 rc = drvIntNetAsyncIoWaitForSpace(pThis);
402 if (VBOX_FAILURE(rc))
403 {
404 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
405 LogFlow(("drvIntNetAsyncIoRun: returns %Vrc (wait-for-space)\n", rc));
406 return rc;
407 }
408 }
409 }
410 else
411 {
412 /*
413 * Link down or unknown frame - skip to the next frame.
414 */
415 AssertMsg(pHdr->u16Type == INTNETHDR_TYPE_FRAME, ("Unknown frame type %RX16! offRead=%#x\n",
416 pHdr->u16Type, pRingBuf->offRead));
417 INTNETRingSkipFrame(pBuf, pRingBuf);
418 }
419 } /* while more received data */
420
421 /*
422 * Wait for data, checking the state before we block.
423 */
424 if (pThis->enmState != ASYNCSTATE_RUNNING)
425 {
426 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
427 LogFlow(("drvIntNetAsyncIoRun: returns VINF_SUCCESS (state changed - #1)\n"));
428 return VERR_STATE_CHANGED;
429 }
430 INTNETIFWAITREQ WaitReq;
431 WaitReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
432 WaitReq.Hdr.cbReq = sizeof(WaitReq);
433 WaitReq.pSession = NULL;
434 WaitReq.hIf = pThis->hIf;
435 WaitReq.cMillies = 30000; /* 30s - don't wait forever, timeout now and then. */
436 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
437 int rc = pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_WAIT, &WaitReq, sizeof(WaitReq));
438 if ( VBOX_FAILURE(rc)
439 && rc != VERR_TIMEOUT
440 && rc != VERR_INTERRUPTED)
441 {
442 LogFlow(("drvIntNetAsyncIoRun: returns %Vrc\n", rc));
443 return rc;
444 }
445 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
446 }
447}
448
449
450/**
451 * Asynchronous I/O thread for handling receive.
452 *
453 * @returns VINF_SUCCESS (ignored).
454 * @param ThreadSelf Thread handle.
455 * @param pvUser Pointer to a DRVINTNET structure.
456 */
457static DECLCALLBACK(int) drvIntNetAsyncIoThread(RTTHREAD ThreadSelf, void *pvUser)
458{
459 PDRVINTNET pThis = (PDRVINTNET)pvUser;
460 LogFlow(("drvIntNetAsyncIoThread: pThis=%p\n", pThis));
461 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
462
463 /*
464 * The main loop - acting on state.
465 */
466 for (;;)
467 {
468 ASYNCSTATE enmState = pThis->enmState;
469 switch (enmState)
470 {
471 case ASYNCSTATE_SUSPENDED:
472 {
473 int rc = RTSemEventWait(pThis->EventSuspended, 30000);
474 if ( VBOX_FAILURE(rc)
475 && rc != VERR_TIMEOUT)
476 {
477 LogFlow(("drvIntNetAsyncIoThread: returns %Vrc\n", rc));
478 return rc;
479 }
480 break;
481 }
482
483 case ASYNCSTATE_RUNNING:
484 {
485 int rc = drvIntNetAsyncIoRun(pThis);
486 if ( rc != VERR_STATE_CHANGED
487 && VBOX_FAILURE(rc))
488 {
489 LogFlow(("drvIntNetAsyncIoThread: returns %Vrc\n", rc));
490 return rc;
491 }
492 break;
493 }
494
495 default:
496 AssertMsgFailed(("Invalid state %d\n", enmState));
497 case ASYNCSTATE_TERMINATE:
498 LogFlow(("drvIntNetAsyncIoThread: returns VINF_SUCCESS\n"));
499 return VINF_SUCCESS;
500 }
501 }
502}
503
504
505/**
506 * Queries an interface to the driver.
507 *
508 * @returns Pointer to interface.
509 * @returns NULL if the interface was not supported by the driver.
510 * @param pInterface Pointer to this interface structure.
511 * @param enmInterface The requested interface identification.
512 * @thread Any thread.
513 */
514static DECLCALLBACK(void *) drvIntNetQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
515{
516 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
517 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
518 switch (enmInterface)
519 {
520 case PDMINTERFACE_BASE:
521 return &pDrvIns->IBase;
522 case PDMINTERFACE_NETWORK_CONNECTOR:
523 return &pThis->INetworkConnector;
524 default:
525 return NULL;
526 }
527}
528
529
530/**
531 * Power Off notification.
532 *
533 * @param pDrvIns The driver instance.
534 */
535static DECLCALLBACK(void) drvIntNetPowerOff(PPDMDRVINS pDrvIns)
536{
537 LogFlow(("drvIntNetPowerOff\n"));
538 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
539 if (!pThis->fActivateEarlyDeactivateLate)
540 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_SUSPENDED);
541}
542
543
544/**
545 * Resume notification.
546 *
547 * @param pDrvIns The driver instance.
548 */
549static DECLCALLBACK(void) drvIntNetResume(PPDMDRVINS pDrvIns)
550{
551 LogFlow(("drvIntNetPowerResume\n"));
552 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
553 if (!pThis->fActivateEarlyDeactivateLate)
554 {
555 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_RUNNING);
556 RTSemEventSignal(pThis->EventSuspended);
557 }
558}
559
560
561/**
562 * Suspend notification.
563 *
564 * @param pDrvIns The driver instance.
565 */
566static DECLCALLBACK(void) drvIntNetSuspend(PPDMDRVINS pDrvIns)
567{
568 LogFlow(("drvIntNetPowerSuspend\n"));
569 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
570 if (!pThis->fActivateEarlyDeactivateLate)
571 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_SUSPENDED);
572}
573
574
575/**
576 * Power On notification.
577 *
578 * @param pDrvIns The driver instance.
579 */
580static DECLCALLBACK(void) drvIntNetPowerOn(PPDMDRVINS pDrvIns)
581{
582 LogFlow(("drvIntNetPowerOn\n"));
583 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
584 if (!pThis->fActivateEarlyDeactivateLate)
585 {
586 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_RUNNING);
587 RTSemEventSignal(pThis->EventSuspended);
588 }
589}
590
591
592/**
593 * Destruct a driver instance.
594 *
595 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
596 * resources can be freed correctly.
597 *
598 * @param pDrvIns The driver instance data.
599 */
600static DECLCALLBACK(void) drvIntNetDestruct(PPDMDRVINS pDrvIns)
601{
602 LogFlow(("drvIntNetDestruct\n"));
603 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
604
605 /*
606 * Indicate to the thread that it's time to quit.
607 */
608 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_TERMINATE);
609 ASMAtomicXchgSize(&pThis->fLinkDown, true);
610 RTSEMEVENT EventSuspended = pThis->EventSuspended;
611 pThis->EventSuspended = NIL_RTSEMEVENT;
612
613 /*
614 * Close the interface
615 */
616 if (pThis->hIf != INTNET_HANDLE_INVALID)
617 {
618 INTNETIFCLOSEREQ CloseReq;
619 CloseReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
620 CloseReq.Hdr.cbReq = sizeof(CloseReq);
621 CloseReq.pSession = NULL;
622 CloseReq.hIf = pThis->hIf;
623 pThis->hIf = INTNET_HANDLE_INVALID;
624 int rc = pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_CLOSE, &CloseReq, sizeof(CloseReq));
625 AssertRC(rc);
626 }
627
628 /*
629 * Wait for the thread to terminate.
630 */
631 if (pThis->Thread != NIL_RTTHREAD)
632 {
633 if (EventSuspended != NIL_RTSEMEVENT)
634 RTSemEventSignal(EventSuspended);
635 int rc = RTThreadWait(pThis->Thread, 5000, NULL);
636 AssertRC(rc);
637 pThis->Thread = NIL_RTTHREAD;
638 }
639
640 /*
641 * Destroy the semaphores.
642 */
643 if (EventSuspended != NIL_RTSEMEVENT)
644 RTSemEventDestroy(EventSuspended);
645}
646
647
648/**
649 * Construct a TAP network transport driver instance.
650 *
651 * @returns VBox status.
652 * @param pDrvIns The driver instance data.
653 * If the registration structure is needed, pDrvIns->pDrvReg points to it.
654 * @param pCfgHandle Configuration node handle for the driver. Use this to obtain the configuration
655 * of the driver instance. It's also found in pDrvIns->pCfgHandle, but like
656 * iInstance it's expected to be used a bit in this function.
657 */
658static DECLCALLBACK(int) drvIntNetConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfgHandle)
659{
660 PDRVINTNET pThis = PDMINS2DATA(pDrvIns, PDRVINTNET);
661
662 /*
663 * Init the static parts.
664 */
665 pThis->pDrvIns = pDrvIns;
666 pThis->hIf = INTNET_HANDLE_INVALID;
667 pThis->Thread = NIL_RTTHREAD;
668 pThis->EventSuspended = NIL_RTSEMEVENT;
669 pThis->enmState = ASYNCSTATE_SUSPENDED;
670 pThis->fActivateEarlyDeactivateLate = false;
671 /* IBase */
672 pDrvIns->IBase.pfnQueryInterface = drvIntNetQueryInterface;
673 /* INetwork */
674 pThis->INetworkConnector.pfnSend = drvIntNetSend;
675 pThis->INetworkConnector.pfnSetPromiscuousMode = drvIntNetSetPromiscuousMode;
676 pThis->INetworkConnector.pfnNotifyLinkChanged = drvIntNetNotifyLinkChanged;
677
678 /*
679 * Validate the config.
680 */
681 if (!CFGMR3AreValuesValid(pCfgHandle, "Network\0ReceiveBufferSize\0SendBufferSize\0RestrictAccess\0IsService\0"))
682 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
683
684 /*
685 * Check that no-one is attached to us.
686 */
687 int rc = pDrvIns->pDrvHlp->pfnAttach(pDrvIns, NULL);
688 if (rc != VERR_PDM_NO_ATTACHED_DRIVER)
689 {
690 AssertMsgFailed(("Configuration error: Cannot attach drivers to the TAP driver!\n"));
691 return VERR_PDM_DRVINS_NO_ATTACH;
692 }
693
694 /*
695 * Query the network port interface.
696 */
697 pThis->pPort = (PPDMINETWORKPORT)pDrvIns->pUpBase->pfnQueryInterface(pDrvIns->pUpBase, PDMINTERFACE_NETWORK_PORT);
698 if (!pThis->pPort)
699 {
700 AssertMsgFailed(("Configuration error: the above device/driver didn't export the network port interface!\n"));
701 return VERR_PDM_MISSING_INTERFACE_ABOVE;
702 }
703
704 /*
705 * Read the configuration.
706 */
707 INTNETOPENREQ OpenReq;
708 memset(&OpenReq, 0, sizeof(OpenReq));
709 OpenReq.Hdr.cbReq = sizeof(OpenReq);
710 OpenReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
711 OpenReq.pSession = NULL;
712
713 /** @cfgm{Network, string}
714 * The name of the internal network to connect to.
715 */
716 rc = CFGMR3QueryString(pCfgHandle, "Network", OpenReq.szNetwork, sizeof(OpenReq.szNetwork));
717 if (VBOX_FAILURE(rc))
718 return PDMDRV_SET_ERROR(pDrvIns, rc,
719 N_("Configuration error: Failed to get the \"Network\" value"));
720 strcpy(pThis->szNetwork, OpenReq.szNetwork);
721
722 /** @cfgm{TrunkType, uint32_t, kIntNetTrunkType_None}
723 * The trunk connection type see INTNETTRUNKTYPE.
724 */
725 uint32_t u32TrunkType;
726 rc = CFGMR3QueryU32(pCfgHandle, "TrunkType", &u32TrunkType);
727 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
728 u32TrunkType = kIntNetTrunkType_None;
729 else if (VBOX_FAILURE(rc))
730 return PDMDRV_SET_ERROR(pDrvIns, rc,
731 N_("Configuration error: Failed to get the \"TrunkType\" value"));
732 OpenReq.enmTrunkType = (INTNETTRUNKTYPE)u32TrunkType;
733
734 /** @cfgm{Trunk, string, ""}
735 * The name of the trunk connection.
736 */
737 rc = CFGMR3QueryString(pCfgHandle, "Trunk", OpenReq.szTrunk, sizeof(OpenReq.szTrunk));
738 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
739 OpenReq.szTrunk[0] = '\0';
740 else if (VBOX_FAILURE(rc))
741 return PDMDRV_SET_ERROR(pDrvIns, rc,
742 N_("Configuration error: Failed to get the \"Trunk\" value"));
743
744 /** @cfgm{RestrictAccess, boolean, true}
745 * Whether to restrict the access to the network or if it should be public. Everyone on
746 * the computer can connect to a public network. Don't change this.
747 */
748 bool fRestrictAccess;
749 rc = CFGMR3QueryBool(pCfgHandle, "RestrictAccess", &fRestrictAccess);
750 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
751 fRestrictAccess = true;
752 else if (VBOX_FAILURE(rc))
753 return PDMDRV_SET_ERROR(pDrvIns, rc,
754 N_("Configuration error: Failed to get the \"RestrictAccess\" value"));
755 OpenReq.fFlags = fRestrictAccess ? 0 : INTNET_OPEN_FLAGS_PUBLIC;
756
757 /** @cfgm{ReceiveBufferSize, uint32_t, 256 KB}
758 * The size of the receive buffer.
759 */
760 rc = CFGMR3QueryU32(pCfgHandle, "ReceiveBufferSize", &OpenReq.cbRecv);
761 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
762 OpenReq.cbRecv = _256K;
763 else if (VBOX_FAILURE(rc))
764 return PDMDRV_SET_ERROR(pDrvIns, rc,
765 N_("Configuration error: Failed to get the \"ReceiveBufferSize\" value"));
766
767 /** @cfgm{SendBufferSize, uint32_t, 4 KB}
768 * The size of the send (transmit) buffer.
769 */
770 rc = CFGMR3QueryU32(pCfgHandle, "SendBufferSize", &OpenReq.cbSend);
771 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
772 OpenReq.cbSend = _4K;
773 else if (VBOX_FAILURE(rc))
774 return PDMDRV_SET_ERROR(pDrvIns, rc,
775 N_("Configuration error: Failed to get the \"SendBufferSize\" value"));
776 if (OpenReq.cbSend < 16)
777 return PDMDRV_SET_ERROR(pDrvIns, rc,
778 N_("Configuration error: The \"SendBufferSize\" value is too small"));
779 if (OpenReq.cbSend < 1536*2 + 4)
780 LogRel(("DrvIntNet: Warning! SendBufferSize=%u, Recommended minimum size %u butes.\n", OpenReq.cbSend, 1536*2 + 4));
781
782 /** @cfgm{IsService, boolean, true}
783 * This alterns the way the thread is suspended and resumed. When it's being used by
784 * a service such as LWIP/iSCSI it shouldn't suspend immediately like for a NIC.
785 */
786 rc = CFGMR3QueryBool(pCfgHandle, "IsService", &pThis->fActivateEarlyDeactivateLate);
787 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
788 pThis->fActivateEarlyDeactivateLate = false;
789 else if (VBOX_FAILURE(rc))
790 return PDMDRV_SET_ERROR(pDrvIns, rc,
791 N_("Configuration error: Failed to get the \"IsService\" value"));
792
793 LogRel(("IntNet#%u: szNetwork={%s} enmTrunkType=%d szTrunk={%s} fFlags=%#x cbRecv=%u cbSend=%u\n",
794 pDrvIns->iInstance, OpenReq.szNetwork, OpenReq.enmTrunkType, OpenReq.szTrunk, OpenReq.fFlags,
795 OpenReq.cbRecv, OpenReq.cbSend));
796
797 /*
798 * Create the event semaphores
799 */
800 rc = RTSemEventCreate(&pThis->EventSuspended);
801 if (VBOX_FAILURE(rc))
802 return rc;
803
804 /*
805 * Create the interface.
806 */
807 OpenReq.hIf = INTNET_HANDLE_INVALID;
808 rc = pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_OPEN, &OpenReq, sizeof(OpenReq));
809 if (VBOX_FAILURE(rc))
810 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
811 N_("Failed to open/create the internal network '%s'"), pThis->szNetwork);
812 AssertRelease(OpenReq.hIf != INTNET_HANDLE_INVALID);
813 pThis->hIf = OpenReq.hIf;
814 Log(("IntNet%d: hIf=%RX32 '%s'\n", pDrvIns->iInstance, pThis->hIf, pThis->szNetwork));
815
816 /*
817 * Get default buffer.
818 */
819 INTNETIFGETRING3BUFFERREQ GetRing3BufferReq;
820 GetRing3BufferReq.Hdr.u32Magic = SUPVMMR0REQHDR_MAGIC;
821 GetRing3BufferReq.Hdr.cbReq = sizeof(GetRing3BufferReq);
822 GetRing3BufferReq.pSession = NULL;
823 GetRing3BufferReq.hIf = pThis->hIf;
824 GetRing3BufferReq.pRing3Buf = NULL;
825 rc = pDrvIns->pDrvHlp->pfnSUPCallVMMR0Ex(pDrvIns, VMMR0_DO_INTNET_IF_GET_RING3_BUFFER, &GetRing3BufferReq, sizeof(GetRing3BufferReq));
826 if (VBOX_FAILURE(rc))
827 return PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
828 N_("Failed to get ring-3 buffer for the newly created interface to '%s'"), pThis->szNetwork);
829 AssertRelease(VALID_PTR(GetRing3BufferReq.pRing3Buf));
830 pThis->pBuf = GetRing3BufferReq.pRing3Buf;
831
832 /*
833 * Create the async I/O thread.
834 * Note! Using a PDM thread here doesn't fit with the IsService=true operation.
835 */
836 rc = RTThreadCreate(&pThis->Thread, drvIntNetAsyncIoThread, pThis, _128K, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "INTNET");
837 if (VBOX_FAILURE(rc))
838 {
839 AssertRC(rc);
840 return rc;
841 }
842
843 char szStatName[64];
844 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Bytes/Received", pDrvIns->iInstance);
845 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cbStatRecv, STAMTYPE_COUNTER, szStatName, STAMUNIT_BYTES, "Number of received bytes.");
846 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Bytes/Sent", pDrvIns->iInstance);
847 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cbStatSend, STAMTYPE_COUNTER, szStatName, STAMUNIT_BYTES, "Number of sent bytes.");
848 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Packets/Received", pDrvIns->iInstance);
849 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cStatRecvs, STAMTYPE_COUNTER, szStatName, STAMUNIT_OCCURENCES, "Number of received packets.");
850 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Packets/Sent", pDrvIns->iInstance);
851 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cStatSends, STAMTYPE_COUNTER, szStatName, STAMUNIT_OCCURENCES, "Number of sent packets.");
852 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Packets/Lost", pDrvIns->iInstance);
853 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cStatLost, STAMTYPE_COUNTER, szStatName, STAMUNIT_OCCURENCES, "Number of lost packets.");
854 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/YieldOk", pDrvIns->iInstance);
855 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cStatYieldsOk, STAMTYPE_COUNTER, szStatName, STAMUNIT_OCCURENCES, "Number of times yielding fixed an overflow.");
856 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/YieldNok", pDrvIns->iInstance);
857 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->pBuf->cStatYieldsNok, STAMTYPE_COUNTER, szStatName, STAMUNIT_OCCURENCES, "Number of times yielding didn't help fix an overflow.");
858
859#ifdef VBOX_WITH_STATISTICS
860 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Receive", pDrvIns->iInstance);
861 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->StatReceive, STAMTYPE_PROFILE, szStatName, STAMUNIT_TICKS_PER_CALL, "Profiling packet receive runs.");
862 RTStrPrintf(szStatName, sizeof(szStatName), "/Net/IntNet%d/Transmit", pDrvIns->iInstance);
863 pDrvIns->pDrvHlp->pfnSTAMRegister(pDrvIns, &pThis->StatTransmit, STAMTYPE_PROFILE, szStatName, STAMUNIT_TICKS_PER_CALL, "Profiling packet transmit runs.");
864#endif
865
866 /*
867 * Activate data transmission as early as possible
868 */
869 if (pThis->fActivateEarlyDeactivateLate)
870 {
871 ASMAtomicXchgSize(&pThis->enmState, ASYNCSTATE_RUNNING);
872 RTSemEventSignal(pThis->EventSuspended);
873 }
874
875 return rc;
876}
877
878
879/**
880 * Internal networking transport driver registration record.
881 */
882const PDMDRVREG g_DrvIntNet =
883{
884 /* u32Version */
885 PDM_DRVREG_VERSION,
886 /* szDriverName */
887 "IntNet",
888 /* pszDescription */
889 "Internal Networking Transport Driver",
890 /* fFlags */
891 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
892 /* fClass. */
893 PDM_DRVREG_CLASS_NETWORK,
894 /* cMaxInstances */
895 ~0,
896 /* cbInstance */
897 sizeof(DRVINTNET),
898 /* pfnConstruct */
899 drvIntNetConstruct,
900 /* pfnDestruct */
901 drvIntNetDestruct,
902 /* pfnIOCtl */
903 NULL,
904 /* pfnPowerOn */
905 drvIntNetPowerOn,
906 /* pfnReset */
907 NULL,
908 /* pfnSuspend */
909 drvIntNetSuspend,
910 /* pfnResume */
911 drvIntNetResume,
912 /* pfnDetach */
913 NULL,
914 /* pfnPowerOff */
915 drvIntNetPowerOff
916};
917
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