VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DrvVDE.cpp@ 30309

Last change on this file since 30309 was 29461, checked in by vboxsync, 15 years ago

Devices/Network/VDE: applied "[vbox-dev] [PATCH] VirtualBox. vde_close was missing..." (07/05/2010, Renzo Davoli)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 22.8 KB
Line 
1/* $Id: DrvVDE.cpp 29461 2010-05-14 11:10:33Z vboxsync $ */
2/** @file
3 * VDE network transport driver.
4 */
5
6/*
7 * Contributed by Renzo Davoli. VirtualSquare. University of Bologna, 2010
8 * Copyright (C) 2006-2010 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_DRV_TUN
23#include <VBox/log.h>
24#include <VBox/pdmdrv.h>
25#include <VBox/pdmnetifs.h>
26#include <VBox/pdmnetinline.h>
27#include <VBox/VDEPlug.h>
28
29#include <iprt/asm.h>
30#include <iprt/assert.h>
31#include <iprt/ctype.h>
32#include <iprt/file.h>
33#include <iprt/mem.h>
34#include <iprt/param.h>
35#include <iprt/path.h>
36#include <iprt/semaphore.h>
37#include <iprt/string.h>
38#include <iprt/thread.h>
39#include <iprt/uuid.h>
40
41#include <sys/ioctl.h>
42#include <sys/poll.h>
43#include <sys/fcntl.h>
44#include <errno.h>
45#include <unistd.h>
46
47#include "Builtins.h"
48
49
50/*******************************************************************************
51* Structures and Typedefs *
52*******************************************************************************/
53/**
54 * VDE driver instance data.
55 *
56 * @implements PDMINETWORKUP
57 */
58typedef struct DRVVDE
59{
60 /** The network interface. */
61 PDMINETWORKUP INetworkUp;
62 /** The network interface. */
63 PPDMINETWORKDOWN pIAboveNet;
64 /** Pointer to the driver instance. */
65 PPDMDRVINS pDrvIns;
66 /** VDE device file handle. */
67 RTFILE FileDevice;
68 /** The configured VDE device name. */
69 char *pszDeviceName;
70 /** The write end of the control pipe. */
71 RTFILE PipeWrite;
72 /** The read end of the control pipe. */
73 RTFILE PipeRead;
74 /** Reader thread. */
75 PPDMTHREAD pThread;
76 /** The connection to the VDE switch */
77 VDECONN *vdeconn;
78
79 /** @todo The transmit thread. */
80 /** Transmit lock used by drvTAPNetworkUp_BeginXmit. */
81 RTCRITSECT XmitLock;
82
83#ifdef VBOX_WITH_STATISTICS
84 /** Number of sent packets. */
85 STAMCOUNTER StatPktSent;
86 /** Number of sent bytes. */
87 STAMCOUNTER StatPktSentBytes;
88 /** Number of received packets. */
89 STAMCOUNTER StatPktRecv;
90 /** Number of received bytes. */
91 STAMCOUNTER StatPktRecvBytes;
92 /** Profiling packet transmit runs. */
93 STAMPROFILE StatTransmit;
94 /** Profiling packet receive runs. */
95 STAMPROFILEADV StatReceive;
96#endif /* VBOX_WITH_STATISTICS */
97
98#ifdef LOG_ENABLED
99 /** The nano ts of the last transfer. */
100 uint64_t u64LastTransferTS;
101 /** The nano ts of the last receive. */
102 uint64_t u64LastReceiveTS;
103#endif
104} DRVVDE, *PDRVVDE;
105
106
107/** Converts a pointer to VDE::INetworkUp to a PRDVVDE. */
108#define PDMINETWORKUP_2_DRVVDE(pInterface) ( (PDRVVDE)((uintptr_t)pInterface - RT_OFFSETOF(DRVVDE, INetworkUp)) )
109
110
111/*******************************************************************************
112* Internal Functions *
113*******************************************************************************/
114
115
116
117/**
118 * @interface_method_impl{PDMINETWORKUP,pfnBeginXmit}
119 */
120static DECLCALLBACK(int) drvVDENetworkUp_BeginXmit(PPDMINETWORKUP pInterface, bool fOnWorkerThread)
121{
122 PDRVVDE pThis = PDMINETWORKUP_2_DRVVDE(pInterface);
123 int rc = RTCritSectTryEnter(&pThis->XmitLock);
124 if (RT_FAILURE(rc))
125 {
126 /** @todo XMIT thread */
127 rc = VERR_TRY_AGAIN;
128 }
129 return rc;
130}
131
132
133/**
134 * @interface_method_impl{PDMINETWORKUP,pfnAllocBuf}
135 */
136static DECLCALLBACK(int) drvVDENetworkUp_AllocBuf(PPDMINETWORKUP pInterface, size_t cbMin,
137 PCPDMNETWORKGSO pGso, PPPDMSCATTERGATHER ppSgBuf)
138{
139 PDRVVDE pThis = PDMINETWORKUP_2_DRVVDE(pInterface);
140 Assert(RTCritSectIsOwner(&pThis->XmitLock));
141
142 /*
143 * Allocate a scatter / gather buffer descriptor that is immediately
144 * followed by the buffer space of its single segment. The GSO context
145 * comes after that again.
146 */
147 PPDMSCATTERGATHER pSgBuf = (PPDMSCATTERGATHER)RTMemAlloc( RT_ALIGN_Z(sizeof(*pSgBuf), 16)
148 + RT_ALIGN_Z(cbMin, 16)
149 + (pGso ? RT_ALIGN_Z(sizeof(*pGso), 16) : 0));
150 if (!pSgBuf)
151 return VERR_NO_MEMORY;
152
153 /*
154 * Initialize the S/G buffer and return.
155 */
156 pSgBuf->fFlags = PDMSCATTERGATHER_FLAGS_MAGIC | PDMSCATTERGATHER_FLAGS_OWNER_1;
157 pSgBuf->cbUsed = 0;
158 pSgBuf->cbAvailable = RT_ALIGN_Z(cbMin, 16);
159 pSgBuf->pvAllocator = NULL;
160 if (!pGso)
161 pSgBuf->pvUser = NULL;
162 else
163 {
164 pSgBuf->pvUser = (uint8_t *)(pSgBuf + 1) + pSgBuf->cbAvailable;
165 *(PPDMNETWORKGSO)pSgBuf->pvUser = *pGso;
166 }
167 pSgBuf->cSegs = 1;
168 pSgBuf->aSegs[0].cbSeg = pSgBuf->cbAvailable;
169 pSgBuf->aSegs[0].pvSeg = pSgBuf + 1;
170
171#if 0 /* poison */
172 memset(pSgBuf->aSegs[0].pvSeg, 'F', pSgBuf->aSegs[0].cbSeg);
173#endif
174 *ppSgBuf = pSgBuf;
175 return VINF_SUCCESS;
176}
177
178
179/**
180 * @interface_method_impl{PDMINETWORKUP,pfnFreeBuf}
181 */
182static DECLCALLBACK(int) drvVDENetworkUp_FreeBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf)
183{
184 PDRVVDE pThis = PDMINETWORKUP_2_DRVVDE(pInterface);
185 Assert(RTCritSectIsOwner(&pThis->XmitLock));
186 if (pSgBuf)
187 {
188 Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
189 pSgBuf->fFlags = 0;
190 RTMemFree(pSgBuf);
191 }
192 return VINF_SUCCESS;
193}
194
195
196/**
197 * @interface_method_impl{PDMINETWORKUP,pfnSendBuf}
198 */
199static DECLCALLBACK(int) drvVDENetworkUp_SendBuf(PPDMINETWORKUP pInterface, PPDMSCATTERGATHER pSgBuf, bool fOnWorkerThread)
200{
201 PDRVVDE pThis = PDMINETWORKUP_2_DRVVDE(pInterface);
202 STAM_COUNTER_INC(&pThis->StatPktSent);
203 STAM_COUNTER_ADD(&pThis->StatPktSentBytes, pSgBuf->cbUsed);
204 STAM_PROFILE_START(&pThis->StatTransmit, a);
205
206 AssertPtr(pSgBuf);
207 Assert((pSgBuf->fFlags & PDMSCATTERGATHER_FLAGS_MAGIC_MASK) == PDMSCATTERGATHER_FLAGS_MAGIC);
208 Assert(RTCritSectIsOwner(&pThis->XmitLock));
209
210 int rc;
211 if (!pSgBuf->pvUser)
212 {
213#ifdef LOG_ENABLED
214 uint64_t u64Now = RTTimeProgramNanoTS();
215 LogFlow(("drvVDESend: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
216 pSgBuf->cbUsed, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
217 pThis->u64LastTransferTS = u64Now;
218#endif
219 Log2(("drvVDESend: pSgBuf->aSegs[0].pvSeg=%p pSgBuf->cbUsed=%#x\n"
220 "%.*Rhxd\n",
221 pSgBuf->aSegs[0].pvSeg, pSgBuf->cbUsed, pSgBuf->cbUsed, pSgBuf->aSegs[0].pvSeg));
222
223 ssize_t cbSent;
224 cbSent = vde_send(pThis->vdeconn, pSgBuf->aSegs[0].pvSeg, pSgBuf->cbUsed, 0);
225 rc = cbSent < 0 ? RTErrConvertFromErrno(-cbSent) : VINF_SUCCESS;
226 }
227 else
228 {
229 uint8_t abHdrScratch[256];
230 uint8_t const *pbFrame = (uint8_t const *)pSgBuf->aSegs[0].pvSeg;
231 PCPDMNETWORKGSO pGso = (PCPDMNETWORKGSO)pSgBuf->pvUser;
232 uint32_t const cSegs = PDMNetGsoCalcSegmentCount(pGso, pSgBuf->cbUsed); Assert(cSegs > 1);
233 rc = 0;
234 for (size_t iSeg = 0; iSeg < cSegs; iSeg++)
235 {
236 uint32_t cbSegFrame;
237 void *pvSegFrame = PDMNetGsoCarveSegmentQD(pGso, (uint8_t *)pbFrame, pSgBuf->cbUsed, abHdrScratch,
238 iSeg, cSegs, &cbSegFrame);
239 ssize_t cbSent;
240 cbSent = vde_send(pThis->vdeconn, pvSegFrame, cbSegFrame, 0);
241 rc = cbSent < 0 ? RTErrConvertFromErrno(-cbSent) : VINF_SUCCESS;
242 if (RT_FAILURE(rc))
243 break;
244 }
245 }
246
247 pSgBuf->fFlags = 0;
248 RTMemFree(pSgBuf);
249
250 STAM_PROFILE_STOP(&pThis->StatTransmit, a);
251 AssertRC(rc);
252 if (RT_FAILURE(rc))
253 rc = rc == VERR_NO_MEMORY ? VERR_NET_NO_BUFFER_SPACE : VERR_NET_DOWN;
254 return rc;
255}
256
257
258/**
259 * @interface_method_impl{PDMINETWORKUP,pfnEndXmit}
260 */
261static DECLCALLBACK(void) drvVDENetworkUp_EndXmit(PPDMINETWORKUP pInterface)
262{
263 PDRVVDE pThis = PDMINETWORKUP_2_DRVVDE(pInterface);
264 RTCritSectLeave(&pThis->XmitLock);
265}
266
267
268/**
269 * @interface_method_impl{PDMINETWORKUP,pfnSetPromiscuousMode}
270 */
271static DECLCALLBACK(void) drvVDENetworkUp_SetPromiscuousMode(PPDMINETWORKUP pInterface, bool fPromiscuous)
272{
273 LogFlow(("drvVDESetPromiscuousMode: fPromiscuous=%d\n", fPromiscuous));
274 /* nothing to do */
275}
276
277
278/**
279 * Notification on link status changes.
280 *
281 * @param pInterface Pointer to the interface structure containing the called function pointer.
282 * @param enmLinkState The new link state.
283 * @thread EMT
284 */
285static DECLCALLBACK(void) drvVDENetworkUp_NotifyLinkChanged(PPDMINETWORKUP pInterface, PDMNETWORKLINKSTATE enmLinkState)
286{
287 LogFlow(("drvNATNetworkUp_NotifyLinkChanged: enmLinkState=%d\n", enmLinkState));
288 /** @todo take action on link down and up. Stop the polling and such like. */
289}
290
291
292/**
293 * Asynchronous I/O thread for handling receive.
294 *
295 * @returns VINF_SUCCESS (ignored).
296 * @param Thread Thread handle.
297 * @param pvUser Pointer to a DRVVDE structure.
298 */
299static DECLCALLBACK(int) drvVDEAsyncIoThread(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
300{
301 PDRVVDE pThis = PDMINS_2_DATA(pDrvIns, PDRVVDE);
302 LogFlow(("drvVDEAsyncIoThread: pThis=%p\n", pThis));
303
304 if (pThread->enmState == PDMTHREADSTATE_INITIALIZING)
305 return VINF_SUCCESS;
306
307 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
308
309 /*
310 * Polling loop.
311 */
312 while (pThread->enmState == PDMTHREADSTATE_RUNNING)
313 {
314 /*
315 * Wait for something to become available.
316 */
317 struct pollfd aFDs[2];
318 aFDs[0].fd = vde_datafd(pThis->vdeconn);
319 aFDs[0].events = POLLIN | POLLPRI;
320 aFDs[0].revents = 0;
321 aFDs[1].fd = pThis->PipeRead;
322 aFDs[1].events = POLLIN | POLLPRI | POLLERR | POLLHUP;
323 aFDs[1].revents = 0;
324 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
325 errno=0;
326 int rc = poll(&aFDs[0], RT_ELEMENTS(aFDs), -1 /* infinite */);
327
328 /* this might have changed in the meantime */
329 if (pThread->enmState != PDMTHREADSTATE_RUNNING)
330 break;
331
332 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
333 if ( rc > 0
334 && (aFDs[0].revents & (POLLIN | POLLPRI))
335 && !aFDs[1].revents)
336 {
337 /*
338 * Read the frame.
339 */
340 char achBuf[16384];
341 ssize_t cbRead = 0;
342 cbRead = vde_recv(pThis->vdeconn, achBuf, sizeof(achBuf), 0);
343 rc = cbRead < 0 ? RTErrConvertFromErrno(-cbRead) : VINF_SUCCESS;
344 if (RT_SUCCESS(rc))
345 {
346 /*
347 * Wait for the device to have space for this frame.
348 * Most guests use frame-sized receive buffers, hence non-zero cbMax
349 * automatically means there is enough room for entire frame. Some
350 * guests (eg. Solaris) use large chains of small receive buffers
351 * (each 128 or so bytes large). We will still start receiving as soon
352 * as cbMax is non-zero because:
353 * - it would be quite expensive for pfnCanReceive to accurately
354 * determine free receive buffer space
355 * - if we were waiting for enough free buffers, there is a risk
356 * of deadlocking because the guest could be waiting for a receive
357 * overflow error to allocate more receive buffers
358 */
359 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
360 int rc1 = pThis->pIAboveNet->pfnWaitReceiveAvail(pThis->pIAboveNet, RT_INDEFINITE_WAIT);
361 STAM_PROFILE_ADV_START(&pThis->StatReceive, a);
362
363 /*
364 * A return code != VINF_SUCCESS means that we were woken up during a VM
365 * state transistion. Drop the packet and wait for the next one.
366 */
367 if (RT_FAILURE(rc1))
368 continue;
369
370 /*
371 * Pass the data up.
372 */
373#ifdef LOG_ENABLED
374 uint64_t u64Now = RTTimeProgramNanoTS();
375 LogFlow(("drvVDEAsyncIoThread: %-4d bytes at %llu ns deltas: r=%llu t=%llu\n",
376 cbRead, u64Now, u64Now - pThis->u64LastReceiveTS, u64Now - pThis->u64LastTransferTS));
377 pThis->u64LastReceiveTS = u64Now;
378#endif
379 Log2(("drvVDEAsyncIoThread: cbRead=%#x\n" "%.*Rhxd\n", cbRead, cbRead, achBuf));
380 STAM_COUNTER_INC(&pThis->StatPktRecv);
381 STAM_COUNTER_ADD(&pThis->StatPktRecvBytes, cbRead);
382 rc1 = pThis->pIAboveNet->pfnReceive(pThis->pIAboveNet, achBuf, cbRead);
383 AssertRC(rc1);
384 }
385 else
386 {
387 LogFlow(("drvVDEAsyncIoThread: RTFileRead -> %Rrc\n", rc));
388 if (rc == VERR_INVALID_HANDLE)
389 break;
390 RTThreadYield();
391 }
392 }
393 else if ( rc > 0
394 && aFDs[1].revents)
395 {
396 LogFlow(("drvVDEAsyncIoThread: Control message: enmState=%d revents=%#x\n", pThread->enmState, aFDs[1].revents));
397 if (aFDs[1].revents & (POLLHUP | POLLERR | POLLNVAL))
398 break;
399
400 /* drain the pipe */
401 char ch;
402 size_t cbRead;
403 RTFileRead(pThis->PipeRead, &ch, 1, &cbRead);
404 }
405 else
406 {
407 /*
408 * poll() failed for some reason. Yield to avoid eating too much CPU.
409 *
410 * EINTR errors have been seen frequently. They should be harmless, even
411 * if they are not supposed to occur in our setup.
412 */
413 if (errno == EINTR)
414 Log(("rc=%d revents=%#x,%#x errno=%p %s\n", rc, aFDs[0].revents, aFDs[1].revents, errno, strerror(errno)));
415 else
416 AssertMsgFailed(("rc=%d revents=%#x,%#x errno=%p %s\n", rc, aFDs[0].revents, aFDs[1].revents, errno, strerror(errno)));
417 RTThreadYield();
418 }
419 }
420
421
422 LogFlow(("drvVDEAsyncIoThread: returns %Rrc\n", VINF_SUCCESS));
423 STAM_PROFILE_ADV_STOP(&pThis->StatReceive, a);
424 return VINF_SUCCESS;
425}
426
427
428/**
429 * Unblock the send thread so it can respond to a state change.
430 *
431 * @returns VBox status code.
432 * @param pDevIns The pcnet device instance.
433 * @param pThread The send thread.
434 */
435static DECLCALLBACK(int) drvVDEAsyncIoWakeup(PPDMDRVINS pDrvIns, PPDMTHREAD pThread)
436{
437 PDRVVDE pThis = PDMINS_2_DATA(pDrvIns, PDRVVDE);
438
439 int rc = RTFileWrite(pThis->PipeWrite, "", 1, NULL);
440 AssertRC(rc);
441
442 return VINF_SUCCESS;
443}
444
445
446/* -=-=-=-=- PDMIBASE -=-=-=-=- */
447
448/**
449 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
450 */
451static DECLCALLBACK(void *) drvVDEQueryInterface(PPDMIBASE pInterface, const char *pszIID)
452{
453 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
454 PDRVVDE pThis = PDMINS_2_DATA(pDrvIns, PDRVVDE);
455
456 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
457 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKUP, &pThis->INetworkUp);
458 return NULL;
459}
460
461/* -=-=-=-=- PDMDRVREG -=-=-=-=- */
462
463/**
464 * Destruct a driver instance.
465 *
466 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
467 * resources can be freed correctly.
468 *
469 * @param pDrvIns The driver instance data.
470 */
471static DECLCALLBACK(void) drvVDEDestruct(PPDMDRVINS pDrvIns)
472{
473 LogFlow(("drvVDEDestruct\n"));
474 PDRVVDE pThis = PDMINS_2_DATA(pDrvIns, PDRVVDE);
475 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
476
477 /*
478 * Terminate the control pipe.
479 */
480 if (pThis->PipeWrite != NIL_RTFILE)
481 {
482 int rc = RTFileClose(pThis->PipeWrite);
483 AssertRC(rc);
484 pThis->PipeWrite = NIL_RTFILE;
485 }
486 if (pThis->PipeRead != NIL_RTFILE)
487 {
488 int rc = RTFileClose(pThis->PipeRead);
489 AssertRC(rc);
490 pThis->PipeRead = NIL_RTFILE;
491 }
492
493 MMR3HeapFree(pThis->pszDeviceName);
494
495 /*
496 * Kill the xmit lock.
497 */
498 if (RTCritSectIsInitialized(&pThis->XmitLock))
499 RTCritSectDelete(&pThis->XmitLock);
500
501 vde_close(pThis->vdeconn);
502#ifdef VBOX_WITH_STATISTICS
503 /*
504 * Deregister statistics.
505 */
506 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatPktSent);
507 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatPktSentBytes);
508 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatPktRecv);
509 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatPktRecvBytes);
510 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatTransmit);
511 PDMDrvHlpSTAMDeregister(pDrvIns, &pThis->StatReceive);
512#endif /* VBOX_WITH_STATISTICS */
513}
514
515
516/**
517 * Construct a VDE network transport driver instance.
518 *
519 * @copydoc FNPDMDRVCONSTRUCT
520 */
521static DECLCALLBACK(int) drvVDEConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
522{
523 PDRVVDE pThis = PDMINS_2_DATA(pDrvIns, PDRVVDE);
524 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
525
526 /*
527 * Init the static parts.
528 */
529 pThis->pDrvIns = pDrvIns;
530 pThis->FileDevice = NIL_RTFILE;
531 pThis->pszDeviceName = NULL;
532 pThis->PipeRead = NIL_RTFILE;
533 pThis->PipeWrite = NIL_RTFILE;
534
535 /* IBase */
536 pDrvIns->IBase.pfnQueryInterface = drvVDEQueryInterface;
537 /* INetwork */
538 pThis->INetworkUp.pfnBeginXmit = drvVDENetworkUp_BeginXmit;
539 pThis->INetworkUp.pfnAllocBuf = drvVDENetworkUp_AllocBuf;
540 pThis->INetworkUp.pfnFreeBuf = drvVDENetworkUp_FreeBuf;
541 pThis->INetworkUp.pfnSendBuf = drvVDENetworkUp_SendBuf;
542 pThis->INetworkUp.pfnEndXmit = drvVDENetworkUp_EndXmit;
543 pThis->INetworkUp.pfnSetPromiscuousMode = drvVDENetworkUp_SetPromiscuousMode;
544 pThis->INetworkUp.pfnNotifyLinkChanged = drvVDENetworkUp_NotifyLinkChanged;
545
546#ifdef VBOX_WITH_STATISTICS
547 /*
548 * Statistics.
549 */
550 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatPktSent, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of sent packets.", "/Drivers/VDE%d/Packets/Sent", pDrvIns->iInstance);
551 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatPktSentBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Number of sent bytes.", "/Drivers/VDE%d/Bytes/Sent", pDrvIns->iInstance);
552 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatPktRecv, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_OCCURENCES, "Number of received packets.", "/Drivers/VDE%d/Packets/Received", pDrvIns->iInstance);
553 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatPktRecvBytes, STAMTYPE_COUNTER, STAMVISIBILITY_ALWAYS, STAMUNIT_BYTES, "Number of received bytes.", "/Drivers/VDE%d/Bytes/Received", pDrvIns->iInstance);
554 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatTransmit, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling packet transmit runs.", "/Drivers/VDE%d/Transmit", pDrvIns->iInstance);
555 PDMDrvHlpSTAMRegisterF(pDrvIns, &pThis->StatReceive, STAMTYPE_PROFILE, STAMVISIBILITY_ALWAYS, STAMUNIT_TICKS_PER_CALL, "Profiling packet receive runs.", "/Drivers/VDE%d/Receive", pDrvIns->iInstance);
556#endif /* VBOX_WITH_STATISTICS */
557
558 /*
559 * Validate the config.
560 */
561 if (!CFGMR3AreValuesValid(pCfg, "Network"))
562 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES, "");
563
564 /*
565 * Check that no-one is attached to us.
566 */
567 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
568 ("Configuration error: Not possible to attach anything to this driver!\n"),
569 VERR_PDM_DRVINS_NO_ATTACH);
570
571 /*
572 * Query the network port interface.
573 */
574 pThis->pIAboveNet = PDMIBASE_QUERY_INTERFACE(pDrvIns->pUpBase, PDMINETWORKDOWN);
575 if (!pThis->pIAboveNet)
576 return PDMDRV_SET_ERROR(pDrvIns, VERR_PDM_MISSING_INTERFACE_ABOVE,
577 N_("Configuration error: The above device/driver didn't export the network port interface"));
578
579 /*
580 * Read the configuration.
581 */
582 int rc;
583 char szNetwork[RTPATH_MAX];
584 rc = CFGMR3QueryString(pCfg, "Network", szNetwork, sizeof(szNetwork));
585 if (RT_FAILURE(rc))
586 *szNetwork=0;
587
588 if (RT_FAILURE(DrvVDELoadVDEPlug()))
589 return PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_HIF_OPEN_FAILED, RT_SRC_POS,
590 N_("VDEplug library: not found"));
591 pThis->vdeconn = vde_open(szNetwork, "VirtualBOX", NULL);
592 if (pThis->vdeconn == NULL)
593 return PDMDrvHlpVMSetError(pThis->pDrvIns, VERR_PDM_HIF_OPEN_FAILED, RT_SRC_POS,
594 N_("Failed to connect to the VDE SWITCH"));
595
596 /*
597 * Create the transmit lock.
598 */
599 rc = RTCritSectInit(&pThis->XmitLock);
600 AssertRCReturn(rc, rc);
601
602 /*
603 * Create the control pipe.
604 */
605 int fds[2];
606 if (pipe(&fds[0]) != 0) /** @todo RTPipeCreate() or something... */
607 {
608 rc = RTErrConvertFromErrno(errno);
609 AssertRC(rc);
610 return rc;
611 }
612 pThis->PipeRead = fds[0];
613 pThis->PipeWrite = fds[1];
614
615 /*
616 * Create the async I/O thread.
617 */
618 rc = PDMDrvHlpThreadCreate(pDrvIns, &pThis->pThread, pThis, drvVDEAsyncIoThread, drvVDEAsyncIoWakeup, 128 * _1K, RTTHREADTYPE_IO, "VDE");
619 AssertRCReturn(rc, rc);
620
621 return rc;
622}
623
624
625/**
626 * VDE network transport driver registration record.
627 */
628const PDMDRVREG g_DrvVDE =
629{
630 /* u32Version */
631 PDM_DRVREG_VERSION,
632 /* szName */
633 "VDE",
634 /* szRCMod */
635 "",
636 /* szR0Mod */
637 "",
638 /* pszDescription */
639 "VDE Network Transport Driver",
640 /* fFlags */
641 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
642 /* fClass. */
643 PDM_DRVREG_CLASS_NETWORK,
644 /* cMaxInstances */
645 ~0,
646 /* cbInstance */
647 sizeof(DRVVDE),
648 /* pfnConstruct */
649 drvVDEConstruct,
650 /* pfnDestruct */
651 drvVDEDestruct,
652 /* pfnRelocate */
653 NULL,
654 /* pfnIOCtl */
655 NULL,
656 /* pfnPowerOn */
657 NULL,
658 /* pfnReset */
659 NULL,
660 /* pfnSuspend */
661 NULL, /** @todo Do power on, suspend and resume handlers! */
662 /* pfnResume */
663 NULL,
664 /* pfnAttach */
665 NULL,
666 /* pfnDetach */
667 NULL,
668 /* pfnPowerOff */
669 NULL,
670 /* pfnSoftReset */
671 NULL,
672 /* u32EndVersion */
673 PDM_DRVREG_VERSION
674};
675
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