VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/DevINIP.cpp@ 47420

Last change on this file since 47420 was 46573, checked in by vboxsync, 12 years ago

LWIP/LWIP-new/NAT: adding/removing interfaces should be done on tcpip thread.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 29.0 KB
Line 
1/* $Id: DevINIP.cpp 46573 2013-06-15 12:19:21Z vboxsync $ */
2/** @file
3 * DevINIP - Internal Network IP stack device/service.
4 */
5
6/*
7 * Copyright (C) 2007-2013 Oracle Corporation
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
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_DEV_INIP
23#include <iprt/cdefs.h> /* include early to allow RT_C_DECLS_BEGIN hack */
24#include <iprt/mem.h> /* include anything of ours that the lwip headers use. */
25#include <iprt/semaphore.h>
26#include <iprt/thread.h>
27#include <iprt/alloca.h>
28/* All lwip header files are not C++ safe. So hack around this. */
29RT_C_DECLS_BEGIN
30#include "lwip/sys.h"
31#include "lwip/stats.h"
32#include "lwip/mem.h"
33#include "lwip/memp.h"
34#include "lwip/pbuf.h"
35#include "lwip/netif.h"
36#ifndef VBOX_WITH_NEW_LWIP
37# include "ipv4/lwip/ip.h"
38#else
39# include "lwip/api.h"
40# include "lwip/tcp_impl.h"
41# include "ipv6/lwip/ethip6.h"
42#endif
43#include "lwip/udp.h"
44#include "lwip/tcp.h"
45#include "lwip/tcpip.h"
46#include "lwip/sockets.h"
47#include "netif/etharp.h"
48RT_C_DECLS_END
49#include <VBox/vmm/pdmdev.h>
50#include <VBox/vmm/pdmnetifs.h>
51#include <VBox/vmm/tm.h>
52#include <iprt/assert.h>
53#include <iprt/string.h>
54#include <iprt/uuid.h>
55
56#include "VBoxDD.h"
57
58#ifdef VBOX_WITH_NEW_LWIP
59# include "VBoxLwipCore.h"
60#endif
61
62/*******************************************************************************
63* Macros and Defines *
64*******************************************************************************/
65
66/** Maximum frame size this device can handle. */
67#define DEVINIP_MAX_FRAME 1514
68
69
70/*******************************************************************************
71* Structures and Typedefs *
72*******************************************************************************/
73
74/**
75 * Internal Network IP stack device instance data.
76 *
77 * @implements PDMIBASE
78 * @implements PDMINETWORKDOWN
79 */
80typedef struct DEVINTNETIP
81{
82 /** The base interface for LUN\#0. */
83 PDMIBASE IBase;
84 /** The network port this device provides (LUN\#0). */
85 PDMINETWORKDOWN INetworkDown;
86 /** The network configuration port this device provides (LUN\#0). */
87 PDMINETWORKCONFIG INetworkConfig;
88 /** The base interface of the network driver below us. */
89 PPDMIBASE pDrvBase;
90 /** The connector of the network driver below us. */
91 PPDMINETWORKUP pDrv;
92 /** Pointer to the device instance. */
93 PPDMDEVINSR3 pDevIns;
94 /** MAC address. */
95 RTMAC MAC;
96 /** Static IP address of the interface. */
97 char *pszIP;
98 /** Netmask of the interface. */
99 char *pszNetmask;
100 /** Gateway for the interface. */
101 char *pszGateway;
102 /** lwIP network interface description. */
103 struct netif IntNetIF;
104 /** lwIP ARP timer. */
105 PTMTIMERR3 ARPTimer;
106 /** lwIP TCP fast timer. */
107 PTMTIMERR3 TCPFastTimer;
108 /** lwIP TCP slow timer. */
109 PTMTIMERR3 TCPSlowTimer;
110 /** lwIP semaphore to coordinate TCPIP init/terminate. */
111 sys_sem_t LWIPTcpInitSem;
112 /** hack: get linking right. remove this eventually, once the device
113 * provides a proper interface to all IP stack functions. */
114 const void *pLinkHack;
115 /** Flag whether the link is up. */
116 bool fLnkUp;
117#ifndef VBOX_WITH_NEW_LWIP
118 /**
119 * This hack-flag for spliting initialization logic in devINIPTcpipInitDone,
120 * this is the only place when during initialization we can be called from TCPIP
121 * thread.
122 * This callback used for Initialization and Finalization with old lwip.
123 */
124 bool fTermination;
125#endif
126 /**
127 * In callback we're getting status of interface adding operation (TCPIP thread),
128 * but we need inform constructing routine whether it was success or not(EMT thread).
129 */
130 bool rcInitialization;
131} DEVINTNETIP, *PDEVINTNETIP;
132
133
134/*******************************************************************************
135* Global Variables *
136*******************************************************************************/
137
138/**
139 * Pointer to the (only) instance data in this device.
140 */
141static PDEVINTNETIP g_pDevINIPData = NULL;
142
143/*
144 * really ugly hack to avoid linking problems on unix style platforms
145 * using .a libraries for now.
146 */
147static const PFNRT g_pDevINILinkHack[] =
148{
149 (PFNRT)lwip_socket,
150 (PFNRT)lwip_close,
151 (PFNRT)lwip_setsockopt,
152 (PFNRT)lwip_recv,
153 (PFNRT)lwip_send,
154 (PFNRT)lwip_select
155};
156
157
158/*******************************************************************************
159* Internal Functions *
160*******************************************************************************/
161#ifndef VBOX_WITH_NEW_LWIP
162static DECLCALLBACK(void) devINIPARPTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer);
163static DECLCALLBACK(void) devINIPTCPFastTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer);
164static DECLCALLBACK(void) devINIPTCPSlowTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer);
165#endif
166static DECLCALLBACK(err_t) devINIPOutput(struct netif *netif, struct pbuf *p, struct ip_addr *ipaddr);
167static DECLCALLBACK(err_t) devINIPOutputRaw(struct netif *netif, struct pbuf *p);
168static DECLCALLBACK(err_t) devINIPInterface(struct netif *netif);
169
170
171#ifndef VBOX_WITH_NEW_LWIP
172/**
173 * ARP cache timeout handling for lwIP.
174 *
175 * @param pDevIns Device instance.
176 * @param pTimer Pointer to timer.
177 */
178static DECLCALLBACK(void) devINIPARPTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
179{
180 PDEVINTNETIP pThis = (PDEVINTNETIP)pvUser;
181 LogFlow(("%s: pDevIns=%p pTimer=%p\n", __FUNCTION__, pDevIns, pTimer));
182 lwip_etharp_tmr();
183 TMTimerSetMillies(pThis->ARPTimer, ARP_TMR_INTERVAL);
184 LogFlow(("%s: return\n", __FUNCTION__));
185}
186
187/**
188 * TCP fast timer handling for lwIP.
189 *
190 * @param pDevIns Device instance.
191 * @param pTimer Pointer to timer.
192 */
193static DECLCALLBACK(void) devINIPTCPFastTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
194{
195 PDEVINTNETIP pThis = (PDEVINTNETIP)pvUser;
196 LogFlow(("%s: pDevIns=%p pTimer=%p\n", __FUNCTION__, pDevIns, pTimer));
197 lwip_tcp_fasttmr();
198 TMTimerSetMillies(pThis->TCPFastTimer, TCP_FAST_INTERVAL);
199 LogFlow(("%s: return\n", __FUNCTION__));
200}
201
202/**
203 * TCP slow timer handling for lwIP.
204 *
205 * @param pDevIns Device instance.
206 * @param pTimer Pointer to timer.
207 */
208static DECLCALLBACK(void) devINIPTCPSlowTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
209{
210 PDEVINTNETIP pThis = (PDEVINTNETIP)pvUser;
211 LogFlow(("%s: pDevIns=%p pTimer=%p\n", __FUNCTION__, pDevIns, pTimer));
212 lwip_tcp_slowtmr();
213 TMTimerSetMillies(pThis->TCPSlowTimer, TCP_SLOW_INTERVAL);
214 LogFlow(("%s: return\n", __FUNCTION__));
215}
216#endif /* VBOX_WITH_NEW_LWIP */
217
218/**
219 * Output a TCP/IP packet on the interface. Uses the generic lwIP ARP
220 * code to resolve the address and call the link-level packet function.
221 *
222 * @returns lwIP error code
223 * @param netif Interface on which to send IP packet.
224 * @param p Packet data.
225 * @param ipaddr Destination IP address.
226 */
227static DECLCALLBACK(err_t) devINIPOutput(struct netif *netif, struct pbuf *p, struct ip_addr *ipaddr)
228{
229 err_t lrc;
230 LogFlow(("%s: netif=%p p=%p ipaddr=%#04x\n", __FUNCTION__, netif, p,
231 ipaddr->addr));
232#ifndef VBOX_WITH_NEW_LWIP
233 lrc = lwip_etharp_output(netif, ipaddr, p);
234#else
235 lrc = lwip_etharp_output(netif, p, ipaddr);
236#endif
237 LogFlow(("%s: return %d\n", __FUNCTION__, lrc));
238 return lrc;
239}
240
241/**
242 * Output a raw packet on the interface.
243 *
244 * @returns lwIP error code
245 * @param netif Interface on which to send frame.
246 * @param p Frame data.
247 */
248static DECLCALLBACK(err_t) devINIPOutputRaw(struct netif *netif, struct pbuf *p)
249{
250 int rc = VINF_SUCCESS;
251
252 LogFlow(("%s: netif=%p p=%p\n", __FUNCTION__, netif, p));
253 Assert(g_pDevINIPData);
254 Assert(g_pDevINIPData->pDrv);
255
256 /* Silently ignore packets being sent while lwIP isn't set up. */
257 if (g_pDevINIPData)
258 {
259 PPDMSCATTERGATHER pSgBuf;
260
261 rc = g_pDevINIPData->pDrv->pfnBeginXmit(g_pDevINIPData->pDrv, true /* fOnWorkerThread */);
262 if (RT_FAILURE(rc))
263 return ERR_IF;
264
265 rc = g_pDevINIPData->pDrv->pfnAllocBuf(g_pDevINIPData->pDrv, DEVINIP_MAX_FRAME, NULL /*pGso*/, &pSgBuf);
266 if (RT_SUCCESS(rc))
267 {
268#if ETH_PAD_SIZE
269 lwip_pbuf_header(p, -ETH_PAD_SIZE); /* drop the padding word */
270#endif
271
272 uint8_t *pbBuf = pSgBuf ? (uint8_t *)pSgBuf->aSegs[0].pvSeg : NULL;
273 size_t cbBuf = 0;
274 for (struct pbuf *q = p; q != NULL; q = q->next)
275 {
276 if (cbBuf + q->len <= DEVINIP_MAX_FRAME)
277 {
278 if (RT_LIKELY(pbBuf))
279 {
280 memcpy(pbBuf, q->payload, q->len);
281 pbBuf += q->len;
282 }
283 cbBuf += q->len;
284 }
285 else
286 {
287 LogRel(("INIP: exceeded frame size\n"));
288 break;
289 }
290 }
291 if (cbBuf)
292 {
293 pSgBuf->cbUsed = cbBuf;
294 rc = g_pDevINIPData->pDrv->pfnSendBuf(g_pDevINIPData->pDrv, pSgBuf, true /* fOnWorkerThread */);
295 }
296 else
297 rc = g_pDevINIPData->pDrv->pfnFreeBuf(g_pDevINIPData->pDrv, pSgBuf);
298
299#if ETH_PAD_SIZE
300 lwip_pbuf_header(p, ETH_PAD_SIZE); /* reclaim the padding word */
301#endif
302 }
303
304 g_pDevINIPData->pDrv->pfnEndXmit(g_pDevINIPData->pDrv);
305 }
306
307 err_t lrc = ERR_OK;
308 if (RT_FAILURE(rc))
309 lrc = ERR_IF;
310 LogFlow(("%s: return %d (vbox: %Rrc)\n", __FUNCTION__, rc, lrc));
311 return lrc;
312}
313
314/**
315 * Implements the ethernet interface backend initialization for lwIP.
316 *
317 * @returns lwIP error code
318 * @param netif Interface to configure.
319 */
320static DECLCALLBACK(err_t) devINIPInterface(struct netif *netif)
321{
322 LogFlow(("%s: netif=%p\n", __FUNCTION__, netif));
323 Assert(g_pDevINIPData != NULL);
324 netif->state = g_pDevINIPData;
325 netif->hwaddr_len = sizeof(g_pDevINIPData->MAC);
326 memcpy(netif->hwaddr, &g_pDevINIPData->MAC, sizeof(g_pDevINIPData->MAC));
327 netif->mtu = DEVINIP_MAX_FRAME;
328 netif->flags = NETIF_FLAG_BROADCAST;
329#ifdef VBOX_WITH_NEW_LWIP
330 /** @todo why explicit ARP routing required for 1.2.0 case? */
331 netif->flags |= NETIF_FLAG_ETHARP;
332 netif->flags |= NETIF_FLAG_ETHERNET;
333 /* Note! We always assign link-local IPv6 address */
334 netif_create_ip6_linklocal_address(netif, 0);
335 netif_ip6_addr_set_state(netif, 0, IP6_ADDR_VALID);
336 netif->output_ip6 = ethip6_output;
337 netif->ip6_autoconfig_enabled=1;
338 LogFunc(("netif: ipv6:%RTnaipv6\n", &netif->ip6_addr[0].addr[0]));
339 netif->output = lwip_etharp_output;
340
341 lwip_etharp_init();
342#else
343 netif->output = devINIPOutput;
344
345 lwip_etharp_init();
346 TMTimerSetMillies(g_pDevINIPData->ARPTimer, ARP_TMR_INTERVAL);
347 #endif
348 netif->linkoutput = devINIPOutputRaw;
349
350 LogFlow(("%s: success\n", __FUNCTION__));
351 return ERR_OK;
352}
353
354/**
355 * Parses CFGM parameters related to network connection
356 */
357static DECLCALLBACK(int) devINIPNetworkConfiguration(PPDMDEVINS pDevIns, PDEVINTNETIP pThis, PCFGMNODE pCfg)
358{
359 int rc = VINF_SUCCESS;
360 rc = CFGMR3QueryStringAlloc(pCfg, "IP", &pThis->pszIP);
361 if (RT_FAILURE(rc))
362 {
363 PDMDEV_SET_ERROR(pDevIns, rc,
364 N_("Configuration error: Failed to get the \"IP\" value"));
365 /* @todo: perhaps we should panic if IPv4 address isn't specify, with assumtion that
366 * ISCSI target specified in IPv6 form.
367 */
368 return rc;
369 }
370
371 rc = CFGMR3QueryStringAlloc(pCfg, "Netmask", &pThis->pszNetmask);
372 if (RT_FAILURE(rc))
373 {
374 PDMDEV_SET_ERROR(pDevIns, rc,
375 N_("Configuration error: Failed to get the \"Netmask\" value"));
376 return rc;
377 }
378 rc = CFGMR3QueryStringAlloc(pCfg, "Gateway", &pThis->pszGateway);
379 if ( RT_FAILURE(rc)
380 && rc != VERR_CFGM_VALUE_NOT_FOUND)
381 {
382 PDMDEV_SET_ERROR(pDevIns, rc,
383 N_("Configuration error: Failed to get the \"Gateway\" value"));
384 return rc;
385 }
386 return VINF_SUCCESS;
387}
388
389/**
390 * Wait until data can be received.
391 *
392 * @returns VBox status code. VINF_SUCCESS means there is at least one receive descriptor available.
393 * @param pInterface PDM network port interface pointer.
394 * @param cMillies Number of milliseconds to wait. 0 means return immediately.
395 */
396static DECLCALLBACK(int) devINIPNetworkDown_WaitInputAvail(PPDMINETWORKDOWN pInterface, RTMSINTERVAL cMillies)
397{
398 LogFlow(("%s: pInterface=%p\n", __FUNCTION__, pInterface));
399 LogFlow(("%s: return VINF_SUCCESS\n", __FUNCTION__));
400 return VINF_SUCCESS;
401}
402
403/**
404 * Receive data and pass it to lwIP for processing.
405 *
406 * @returns VBox status code
407 * @param pInterface PDM network port interface pointer.
408 * @param pvBuf Pointer to frame data.
409 * @param cb Frame size.
410 */
411static DECLCALLBACK(int) devINIPNetworkDown_Input(PPDMINETWORKDOWN pInterface, const void *pvBuf, size_t cb)
412{
413 const uint8_t *pbBuf = (const uint8_t *)pvBuf;
414 size_t len = cb;
415 const struct eth_hdr *ethhdr;
416 struct pbuf *p, *q;
417 int rc = VINF_SUCCESS;
418
419 LogFlow(("%s: pInterface=%p pvBuf=%p cb=%lu\n", __FUNCTION__, pInterface, pvBuf, cb));
420 Assert(g_pDevINIPData);
421 Assert(g_pDevINIPData->pDrv);
422
423 /* Silently ignore packets being received while lwIP isn't set up. */
424 if (!g_pDevINIPData)
425 {
426 LogFlow(("%s: return %Rrc (no global)\n", __FUNCTION__, rc));
427 return VINF_SUCCESS;
428 }
429
430#if ETH_PAD_SIZE
431 len += ETH_PAD_SIZE; /* allow room for Ethernet padding */
432#endif
433
434 /* We allocate a pbuf chain of pbufs from the pool. */
435 p = lwip_pbuf_alloc(PBUF_RAW, len, PBUF_POOL);
436 if (p != NULL)
437 {
438#if ETH_PAD_SIZE
439 lwip_pbuf_header(p, -ETH_PAD_SIZE); /* drop the padding word */
440#endif
441
442 for (q = p; q != NULL; q = q->next)
443 {
444 /* Fill the buffers, and clean out unused buffer space. */
445 memcpy(q->payload, pbBuf, RT_MIN(cb, q->len));
446 pbBuf += RT_MIN(cb, q->len);
447 if (q->len > cb)
448 memset(((uint8_t *)q->payload) + cb, '\0', q->len - cb);
449 cb -= RT_MIN(cb, q->len);
450 }
451
452 ethhdr = (const struct eth_hdr *)p->payload;
453 struct netif *iface = &g_pDevINIPData->IntNetIF;
454 err_t lrc;
455#ifndef VBOX_WITH_NEW_LWIP
456 switch (htons(ethhdr->type))
457 {
458 case ETHTYPE_IP: /* IP packet */
459 lwip_pbuf_header(p, -(ssize_t)sizeof(struct eth_hdr));
460 lrc = iface->input(p, iface);
461 if (lrc)
462 rc = VERR_NET_IO_ERROR;
463 break;
464 case ETHTYPE_ARP: /* ARP packet */
465 lwip_etharp_arp_input(iface, (struct eth_addr *)iface->hwaddr, p);
466 break;
467 default:
468 lwip_pbuf_free(p);
469 }
470#else
471 /* We've setup flags NETIF_FLAG_ETHARP and NETIF_FLAG_ETHERNET
472 so this should be thread-safe. */
473 tcpip_input(p,iface);
474#endif
475 }
476
477out:
478 LogFlow(("%s: return %Rrc\n", __FUNCTION__, rc));
479 return rc;
480}
481
482/**
483 * @interface_method_impl{PDMINETWORKDOWN,pfnXmitPending}
484 */
485static DECLCALLBACK(void) devINIPNetworkDown_XmitPending(PPDMINETWORKDOWN pInterface)
486{
487 NOREF(pInterface);
488}
489
490
491/**
492 * Signals the end of lwIP TCPIP initialization.
493 *
494 * @note: TCPIP thread, corresponding EMT waiting on semaphore.
495 * @param arg opaque argument, here the pointer to the PDEVINTNETIP.
496 */
497static DECLCALLBACK(void) devINIPTcpipInitDone(void *arg)
498{
499 PDEVINTNETIP pThis = (PDEVINTNETIP)arg;
500 AssertPtrReturnVoid(arg);
501
502 pThis->rcInitialization = VINF_SUCCESS;
503#ifndef VBOX_WITH_NEW_LWIP
504 /* see PDEVINTNETIP::fTermination */
505 if (!pThis->fTermination)
506 {
507#endif
508 struct netif *ret;
509 struct ip_addr ipaddr, netmask, gw;
510 struct in_addr ip;
511
512 if (!inet_aton(pThis->pszIP, &ip))
513 {
514 pThis->rcInitialization = VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
515 PDMDEV_SET_ERROR(pThis->pDevIns,
516 pThis->rcInitialization,
517 N_("Configuration error: Invalid \"IP\" value"));
518 goto done;
519 }
520 memcpy(&ipaddr, &ip, sizeof(ipaddr));
521
522 if (!inet_aton(pThis->pszNetmask, &ip))
523 {
524 pThis->rcInitialization = VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
525 PDMDEV_SET_ERROR(pThis->pDevIns,
526 pThis->rcInitialization,
527 N_("Configuration error: Invalid \"Netmask\" value"));
528 goto done;
529 }
530 memcpy(&netmask, &ip, sizeof(netmask));
531
532 if (pThis->pszGateway)
533 {
534 if (!inet_aton(pThis->pszGateway, &ip))
535 {
536 pThis->rcInitialization = VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES;
537 PDMDEV_SET_ERROR(pThis->pDevIns,
538 pThis->rcInitialization,
539 N_("Configuration error: Invalid \"Gateway\" value"));
540 goto done;
541 }
542
543 }
544 else
545 {
546 inet_aton(pThis->pszIP, &ip);
547 }
548 memcpy(&gw, &ip, sizeof(gw));
549
550 pThis->IntNetIF.name[0] = 'I';
551 pThis->IntNetIF.name[1] = 'N';
552
553 ret = netif_add(&pThis->IntNetIF, &ipaddr, &netmask, &gw, NULL,
554 devINIPInterface, lwip_tcpip_input);
555
556 if (!ret)
557 {
558
559 pThis->rcInitialization = VERR_NET_NO_NETWORK;
560 PDMDEV_SET_ERROR(pThis->pDevIns,
561 pThis->rcInitialization,
562 N_("netif_add failed"));
563 goto done;
564 }
565
566 lwip_netif_set_default(&pThis->IntNetIF);
567 lwip_netif_set_up(&pThis->IntNetIF);
568
569#ifndef VBOX_WITH_NEW_LWIP
570 }
571 done:
572 lwip_sys_sem_signal(pThis->LWIPTcpInitSem);
573#else
574 done:
575 return;
576#endif
577}
578
579#ifdef VBOX_WITH_NEW_LWIP
580/**
581 * This callback is for finitializing our activity on TCPIP thread.
582 * XXX: We do it only for new LWIP, old LWIP will stay broken for now.
583 */
584static DECLCALLBACK(void) devINIPTcpipFiniDone(void *arg)
585{
586 PDEVINTNETIP pThis = (PDEVINTNETIP)arg;
587 AssertPtrReturnVoid(arg);
588
589 netif_set_link_down(&pThis->IntNetIF);
590 netif_set_down(&pThis->IntNetIF);
591 netif_remove(&pThis->IntNetIF);
592}
593#endif
594
595
596/**
597 * Gets the current Media Access Control (MAC) address.
598 *
599 * @returns VBox status code.
600 * @param pInterface Pointer to the interface structure containing the called function pointer.
601 * @param pMac Where to store the MAC address.
602 * @thread EMT
603 */
604static DECLCALLBACK(int) devINIPGetMac(PPDMINETWORKCONFIG pInterface, PRTMAC pMac)
605{
606 PDEVINTNETIP pThis = RT_FROM_MEMBER(pInterface, DEVINTNETIP, INetworkConfig);
607 memcpy(pMac, pThis->MAC.au8, sizeof(RTMAC));
608 return VINF_SUCCESS;
609}
610
611/**
612 * Gets the new link state.
613 *
614 * @returns The current link state.
615 * @param pInterface Pointer to the interface structure containing the called function pointer.
616 * @thread EMT
617 */
618static DECLCALLBACK(PDMNETWORKLINKSTATE) devINIPGetLinkState(PPDMINETWORKCONFIG pInterface)
619{
620 PDEVINTNETIP pThis = RT_FROM_MEMBER(pInterface, DEVINTNETIP, INetworkConfig);
621 if (pThis->fLnkUp)
622 return PDMNETWORKLINKSTATE_UP;
623 return PDMNETWORKLINKSTATE_DOWN;
624}
625
626
627/**
628 * Sets the new link state.
629 *
630 * @returns VBox status code.
631 * @param pInterface Pointer to the interface structure containing the called function pointer.
632 * @param enmState The new link state
633 */
634static DECLCALLBACK(int) devINIPSetLinkState(PPDMINETWORKCONFIG pInterface, PDMNETWORKLINKSTATE enmState)
635{
636 PDEVINTNETIP pThis = RT_FROM_MEMBER(pInterface, DEVINTNETIP, INetworkConfig);
637 bool fNewUp = enmState == PDMNETWORKLINKSTATE_UP;
638
639 if (fNewUp != pThis->fLnkUp)
640 {
641 if (fNewUp)
642 {
643 LogFlowFunc(("Link is up\n"));
644 pThis->fLnkUp = true;
645 }
646 else
647 {
648 LogFlowFunc(("Link is down\n"));
649 pThis->fLnkUp = false;
650 }
651 if (pThis->pDrv)
652 pThis->pDrv->pfnNotifyLinkChanged(pThis->pDrv, enmState);
653 }
654 return VINF_SUCCESS;
655}
656
657/* -=-=-=-=- PDMIBASE -=-=-=-=- */
658
659/**
660 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
661 */
662static DECLCALLBACK(void *) devINIPQueryInterface(PPDMIBASE pInterface,
663 const char *pszIID)
664{
665 PDEVINTNETIP pThis = RT_FROM_MEMBER(pInterface, DEVINTNETIP, IBase);
666 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->IBase);
667 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKDOWN, &pThis->INetworkDown);
668 PDMIBASE_RETURN_INTERFACE(pszIID, PDMINETWORKCONFIG, &pThis->INetworkConfig);
669 return NULL;
670}
671
672/* -=-=-=-=- PDMDEVREG -=-=-=-=- */
673
674/**
675 * Destruct a device instance.
676 *
677 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
678 * resources can be freed correctly.
679 *
680 * @returns VBox status.
681 * @param pDevIns The device instance data.
682 */
683static DECLCALLBACK(int) devINIPDestruct(PPDMDEVINS pDevIns)
684{
685 PDEVINTNETIP pThis = PDMINS_2_DATA(pDevIns, PDEVINTNETIP);
686
687 LogFlow(("%s: pDevIns=%p\n", __FUNCTION__, pDevIns));
688 PDMDEV_CHECK_VERSIONS_RETURN_QUIET(pDevIns);
689
690 if (g_pDevINIPData != NULL)
691 {
692#ifndef VBOX_WITH_NEW_LWIP
693 netif_set_down(&pThis->IntNetIF);
694 netif_remove(&pThis->IntNetIF);
695 pThis->fTermination = true;
696 tcpip_terminate();
697 lwip_sys_sem_wait(pThis->LWIPTcpInitSem);
698 lwip_sys_sem_free(pThis->LWIPTcpInitSem);
699#else
700 vboxLwipCoreFinalize(devINIPTcpipFiniDone, pThis);
701#endif
702 }
703
704 MMR3HeapFree(pThis->pszIP);
705 pThis->pszIP = NULL;
706 MMR3HeapFree(pThis->pszNetmask);
707 pThis->pszNetmask = NULL;
708 MMR3HeapFree(pThis->pszGateway);
709 pThis->pszGateway = NULL;
710
711 LogFlow(("%s: success\n", __FUNCTION__));
712 return VINF_SUCCESS;
713}
714
715
716/**
717 * @interface_method_impl{PDMDEVREG,pfnConstruct}
718 */
719static DECLCALLBACK(int) devINIPConstruct(PPDMDEVINS pDevIns, int iInstance, PCFGMNODE pCfg)
720{
721 PDEVINTNETIP pThis = PDMINS_2_DATA(pDevIns, PDEVINTNETIP);
722 int rc = VINF_SUCCESS;
723#ifdef VBOX_WITH_NEW_LWIP
724 err_t errRc = ERR_OK;
725#endif
726 LogFlow(("%s: pDevIns=%p iInstance=%d pCfg=%p\n", __FUNCTION__,
727 pDevIns, iInstance, pCfg));
728
729 Assert(iInstance == 0);
730 PDMDEV_CHECK_VERSIONS_RETURN(pDevIns);
731
732 /*
733 * Validate the config.
734 */
735 if (!CFGMR3AreValuesValid(pCfg, "MAC\0IP\0"
736#ifdef VBOX_WITH_NEW_LWIP
737 "IPv6\0"
738#endif
739 "Netmask\0Gateway\0"))
740 return PDMDEV_SET_ERROR(pDevIns, VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
741 N_("Unknown Internal Networking IP configuration option"));
742
743 /*
744 * Init the static parts.
745 */
746 pThis->pszIP = NULL;
747 pThis->pszNetmask = NULL;
748 pThis->pszGateway = NULL;
749 /* Pointer to device instance */
750 pThis->pDevIns = pDevIns;
751 /* IBase */
752 pThis->IBase.pfnQueryInterface = devINIPQueryInterface;
753 /* INetworkDown */
754 pThis->INetworkDown.pfnWaitReceiveAvail = devINIPNetworkDown_WaitInputAvail;
755 pThis->INetworkDown.pfnReceive = devINIPNetworkDown_Input;
756 pThis->INetworkDown.pfnXmitPending = devINIPNetworkDown_XmitPending;
757 /* INetworkConfig */
758 pThis->INetworkConfig.pfnGetMac = devINIPGetMac;
759 pThis->INetworkConfig.pfnGetLinkState = devINIPGetLinkState;
760 pThis->INetworkConfig.pfnSetLinkState = devINIPSetLinkState;
761
762 /*
763 * Get the configuration settings.
764 */
765 rc = CFGMR3QueryBytes(pCfg, "MAC", &pThis->MAC, sizeof(pThis->MAC));
766 if (rc == VERR_CFGM_NOT_BYTES)
767 {
768 char szMAC[64];
769 rc = CFGMR3QueryString(pCfg, "MAC", &szMAC[0], sizeof(szMAC));
770 if (RT_SUCCESS(rc))
771 {
772 char *macStr = &szMAC[0];
773 char *pMac = (char *)&pThis->MAC;
774 for (uint32_t i = 0; i < 6; i++)
775 {
776 if ( !*macStr || !*(macStr + 1)
777 || *macStr == ':' || *(macStr + 1) == ':')
778 return PDMDEV_SET_ERROR(pDevIns,
779 VERR_PDM_DEVINS_UNKNOWN_CFG_VALUES,
780 N_("Configuration error: Invalid \"MAC\" value"));
781 char c1 = *macStr++ - '0';
782 if (c1 > 9)
783 c1 -= 7;
784 char c2 = *macStr++ - '0';
785 if (c2 > 9)
786 c2 -= 7;
787 *pMac++ = ((c1 & 0x0f) << 4) | (c2 & 0x0f);
788 if (i != 5 && *macStr == ':')
789 macStr++;
790 }
791 }
792 }
793 if (RT_FAILURE(rc))
794 return PDMDEV_SET_ERROR(pDevIns, rc,
795 N_("Configuration error: Failed to get the \"MAC\" value"));
796 rc = devINIPNetworkConfiguration(pDevIns, pThis, pCfg);
797 AssertLogRelRCReturn(rc, rc);
798
799 /*
800 * Attach driver and query the network connector interface.
801 */
802 rc = PDMDevHlpDriverAttach(pDevIns, 0, &pThis->IBase, &pThis->pDrvBase, "Network Port");
803 if (RT_FAILURE(rc))
804 {
805 pThis->pDrvBase = NULL;
806 pThis->pDrv = NULL;
807 return PDMDEV_SET_ERROR(pDevIns, rc, N_("Error attaching device below us"));
808 }
809 pThis->pDrv = PDMIBASE_QUERY_INTERFACE(pThis->pDrvBase, PDMINETWORKUP);
810 AssertMsgReturn(pThis->pDrv, ("Failed to obtain the PDMINETWORKUP interface!\n"), VERR_PDM_MISSING_INTERFACE_BELOW);
811
812
813 /*
814 * Set up global pointer to interface data.
815 */
816 g_pDevINIPData = pThis;
817
818
819 /* link hack */
820 pThis->pLinkHack = g_pDevINILinkHack;
821
822 /*
823 * Initialize lwIP.
824 */
825#ifndef VBOX_WITH_NEW_LWIP
826 lwip_stats_init();
827 lwip_sys_init();
828# if MEM_LIBC_MALLOC == 0
829 lwip_mem_init();
830# endif
831 lwip_memp_init();
832 lwip_pbuf_init();
833 lwip_netif_init();
834 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, devINIPARPTimer, pThis,
835 TMTIMER_FLAGS_NO_CRIT_SECT, "lwIP ARP", &pThis->ARPTimer);
836 AssertRCReturn(rc, rc);
837 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, devINIPTCPFastTimer, pThis,
838 TMTIMER_FLAGS_NO_CRIT_SECT, "lwIP fast TCP", &pThis->TCPFastTimer);
839 AssertRCReturn(rc, rc);
840 TMTimerSetMillies(pThis->TCPFastTimer, TCP_FAST_INTERVAL);
841 rc = PDMDevHlpTMTimerCreate(pDevIns, TMCLOCK_VIRTUAL, devINIPTCPSlowTimer, pThis,
842 TMTIMER_FLAGS_NO_CRIT_SECT, "lwIP slow TCP", &pThis->TCPSlowTimer);
843 AssertRCReturn(rc, rc);
844 TMTimerSetMillies(pThis->TCPFastTimer, TCP_SLOW_INTERVAL);
845
846 pThis->LWIPTcpInitSem = lwip_sys_sem_new(0);
847
848 lwip_tcpip_init(devINIPTcpipInitDone, pThis);
849 lwip_sys_sem_wait(pThis->LWIPTcpInitSem);
850
851#else /* VBOX_WITH_NEW_LWIP */
852 vboxLwipCoreInitialize(devINIPTcpipInitDone, pThis);
853#endif
854
855 /* this rc could be updated in devINIPTcpInitDone thread */
856 AssertRCReturn(pThis->rcInitialization, pThis->rcInitialization);
857
858
859 LogFlow(("%s: return %Rrc\n", __FUNCTION__, rc));
860 return rc;
861}
862
863
864/**
865 * Query whether lwIP is initialized or not. Since there is only a single
866 * instance of this device ever for a VM, it can be a global function.
867 *
868 * @returns True if lwIP is initialized.
869 */
870bool DevINIPConfigured(void)
871{
872 return g_pDevINIPData != NULL;
873}
874
875
876/**
877 * Internal network IP stack device registration record.
878 */
879const PDMDEVREG g_DeviceINIP =
880{
881 /* u32Version */
882 PDM_DEVREG_VERSION,
883 /* szName */
884 "IntNetIP",
885 /* szRCMod/szR0Mod */
886 "",
887 "",
888 /* pszDescription */
889 "Internal Network IP stack device",
890 /* fFlags */
891 PDM_DEVREG_FLAGS_DEFAULT_BITS,
892 /* fClass. As this is used by the storage devices, it must come earlier. */
893 PDM_DEVREG_CLASS_VMM_DEV,
894 /* cMaxInstances */
895 1,
896 /* cbInstance */
897 sizeof(DEVINTNETIP),
898 /* pfnConstruct */
899 devINIPConstruct,
900 /* pfnDestruct */
901 devINIPDestruct,
902 /* pfnRelocate */
903 NULL,
904 /* pfnMemSetup */
905 NULL,
906 /* pfnPowerOn */
907 NULL,
908 /* pfnReset */
909 NULL,
910 /* pfnSuspend */
911 NULL,
912 /* pfnResume */
913 NULL,
914 /* pfnAttach */
915 NULL,
916 /* pfnDetach */
917 NULL,
918 /* pfnQueryInterface */
919 NULL,
920 /* pfnInitComplete */
921 NULL,
922 /* pfnPowerOff */
923 NULL,
924 /* pfnSoftReset */
925 NULL,
926 /* u32VersionEnd */
927 PDM_DEVREG_VERSION
928};
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