VirtualBox

source: vbox/trunk/src/VBox/Devices/Network/SrvIntNetR0.cpp@ 27340

Last change on this file since 27340 was 26576, checked in by vboxsync, 15 years ago

SrvIntNetR0.cpp: gcc maybe used uninitialized.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 144.0 KB
Line 
1/* $Id: SrvIntNetR0.cpp 26576 2010-02-16 12:53:32Z vboxsync $ */
2/** @file
3 * Internal networking - The ring 0 service.
4 */
5
6/*
7 * Copyright (C) 2006-2008 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_SRV_INTNET
27#include <VBox/intnet.h>
28#include <VBox/intnetinline.h>
29#include <VBox/sup.h>
30#include <VBox/pdm.h>
31#include <VBox/log.h>
32#include <iprt/asm.h>
33#include <iprt/alloc.h>
34#include <iprt/semaphore.h>
35#include <iprt/spinlock.h>
36#include <iprt/thread.h>
37#include <iprt/assert.h>
38#include <iprt/string.h>
39#include <iprt/time.h>
40#include <iprt/handletable.h>
41#include <iprt/net.h>
42
43
44/*******************************************************************************
45* Defined Constants And Macros *
46*******************************************************************************/
47/** @def INTNET_WITH_DHCP_SNOOPING
48 * Enabled DHCP snooping when in shared-mac-on-the-wire mode. */
49#define INTNET_WITH_DHCP_SNOOPING
50
51
52/*******************************************************************************
53* Structures and Typedefs *
54*******************************************************************************/
55typedef enum INTNETADDRTYPE
56{
57 /** The invalid 0 entry. */
58 kIntNetAddrType_Invalid = 0,
59 /** IP version 4. */
60 kIntNetAddrType_IPv4,
61 /** IP version 6. */
62 kIntNetAddrType_IPv6,
63 /** IPX. */
64 kIntNetAddrType_IPX,
65 /** The end of the valid values. */
66 kIntNetAddrType_End,
67 /** The usual 32-bit hack. */
68 kIntNetAddrType_32BitHack = 0x7fffffff
69} INTNETADDRTYPE;
70/** Pointer to a network layer address type. */
71typedef INTNETADDRTYPE *PINTNETADDRTYPE;
72
73
74/**
75 * Address and type.
76 */
77typedef struct INTNETADDR
78{
79 /** The address type. */
80 INTNETADDRTYPE enmType;
81 /** The address. */
82 RTNETADDRU Addr;
83} INTNETADDR;
84/** Pointer to an address. */
85typedef INTNETADDR *PINTNETADDR;
86/** Pointer to a const address. */
87typedef INTNETADDR const *PCINTNETADDR;
88
89
90/**
91 * Address cache for a specific network layer.
92 */
93typedef struct INTNETADDRCACHE
94{
95 /** Pointer to the table of addresses. */
96 uint8_t *pbEntries;
97 /** The number of valid address entries. */
98 uint8_t cEntries;
99 /** The number of allocated address entries. */
100 uint8_t cEntriesAlloc;
101 /** The address size. */
102 uint8_t cbAddress;
103 /** The size of an entry. */
104 uint8_t cbEntry;
105} INTNETADDRCACHE;
106/** Pointer to an address cache. */
107typedef INTNETADDRCACHE *PINTNETADDRCACHE;
108/** Pointer to a const address cache. */
109typedef INTNETADDRCACHE const *PCINTNETADDRCACHE;
110
111
112/**
113 * A network interface.
114 *
115 * Unless explicitly stated, all members are protect by the network semaphore.
116 */
117typedef struct INTNETIF
118{
119 /** Pointer to the next interface.
120 * This is protected by the INTNET::FastMutex. */
121 struct INTNETIF *pNext;
122 /** The current MAC address for the interface. */
123 RTMAC Mac;
124 /** Set if the INTNET::Mac member is valid. */
125 bool fMacSet;
126 /** Set if the interface is in promiscuous mode.
127 * In promiscuous mode the interface will receive all packages except the one it's sending. */
128 bool fPromiscuous;
129 /** Whether the interface is active or not. */
130 bool fActive;
131 /** Whether someone is currently in the destructor. */
132 bool volatile fDestroying;
133 /** Number of yields done to try make the interface read pending data.
134 * We will stop yeilding when this reaches a threshold assuming that the VM is paused or
135 * that it simply isn't worth all the delay. It is cleared when a successful send has been done.
136 */
137 uint32_t cYields;
138 /** Pointer to the current exchange buffer (ring-0). */
139 PINTNETBUF pIntBuf;
140 /** Pointer to ring-3 mapping of the current exchange buffer. */
141 R3PTRTYPE(PINTNETBUF) pIntBufR3;
142 /** Pointer to the default exchange buffer for the interface. */
143 PINTNETBUF pIntBufDefault;
144 /** Pointer to ring-3 mapping of the default exchange buffer. */
145 R3PTRTYPE(PINTNETBUF) pIntBufDefaultR3;
146 /** Event semaphore which a receiver thread will sleep on while waiting for data to arrive. */
147 RTSEMEVENT volatile Event;
148 /** Number of threads sleeping on the Event semaphore. */
149 uint32_t cSleepers;
150 /** The interface handle.
151 * When this is INTNET_HANDLE_INVALID a sleeper which is waking up
152 * should return with the appropriate error condition. */
153 INTNETIFHANDLE volatile hIf;
154 /** Pointer to the network this interface is connected to.
155 * This is protected by the INTNET::FastMutex. */
156 struct INTNETNETWORK *pNetwork;
157 /** The session this interface is associated with. */
158 PSUPDRVSESSION pSession;
159 /** The SUPR0 object id. */
160 void *pvObj;
161 /** The network layer address cache. (Indexed by type, 0 entry isn't used.) */
162 INTNETADDRCACHE aAddrCache[kIntNetAddrType_End];
163} INTNETIF;
164/** Pointer to an internal network interface. */
165typedef INTNETIF *PINTNETIF;
166
167
168/**
169 * A trunk interface.
170 */
171typedef struct INTNETTRUNKIF
172{
173 /** The port interface we present to the component. */
174 INTNETTRUNKSWPORT SwitchPort;
175 /** The port interface we get from the component. */
176 PINTNETTRUNKIFPORT pIfPort;
177 /** The trunk mutex that serializes all calls <b>to</b> the component. */
178 RTSEMFASTMUTEX FastMutex;
179 /** Pointer to the network we're connect to.
180 * This may be NULL if we're orphaned? */
181 struct INTNETNETWORK *pNetwork;
182 /** The cached MAC address of the interface the trunk is attached to.
183 * This is for the situations where we cannot take the out-bound
184 * semaphore (the recv case) but need to make frame edits (ARP). */
185 RTMAC CachedMac;
186 /** Whether to supply physical addresses with the outbound SGs. */
187 bool volatile fPhysSG;
188 /** Set if the 'wire' is in promiscuous mode.
189 * The state of the 'host' is queried each time. */
190 bool fPromiscuousWire;
191} INTNETTRUNKIF;
192/** Pointer to a trunk interface. */
193typedef INTNETTRUNKIF *PINTNETTRUNKIF;
194
195/** Converts a pointer to INTNETTRUNKIF::SwitchPort to a PINTNETTRUNKIF. */
196#define INTNET_SWITCHPORT_2_TRUNKIF(pSwitchPort) ((PINTNETTRUNKIF)(pSwitchPort))
197
198
199/**
200 * Internal representation of a network.
201 */
202typedef struct INTNETNETWORK
203{
204 /** The Next network in the chain.
205 * This is protected by the INTNET::FastMutex. */
206 struct INTNETNETWORK *pNext;
207 /** List of interfaces connected to the network.
208 * This is protected by the INTNET::FastMutex. */
209 PINTNETIF pIFs;
210 /** Pointer to the trunk interface.
211 * Can be NULL if there is no trunk connection. */
212 PINTNETTRUNKIF pTrunkIF;
213 /** The network mutex.
214 * It protects everything dealing with this network. */
215 RTSEMFASTMUTEX FastMutex;
216 /** Pointer to the instance data. */
217 struct INTNET *pIntNet;
218 /** The SUPR0 object id. */
219 void *pvObj;
220 /** Pointer to the temporary buffer that is used when snooping fragmented packets.
221 * This is allocated after this structure if we're sharing the MAC address with
222 * the host. The buffer is INTNETNETWORK_TMP_SIZE big and aligned on a 64-byte boundrary. */
223 uint8_t *pbTmp;
224 /** Network creation flags (INTNET_OPEN_FLAGS_*). */
225 uint32_t fFlags;
226 /** The number of active interfaces (excluding the trunk). */
227 uint32_t cActiveIFs;
228 /** The length of the network name. */
229 uint8_t cchName;
230 /** The network name. */
231 char szName[INTNET_MAX_NETWORK_NAME];
232 /** The trunk type. */
233 INTNETTRUNKTYPE enmTrunkType;
234 /** The trunk name. */
235 char szTrunk[INTNET_MAX_TRUNK_NAME];
236} INTNETNETWORK;
237/** Pointer to an internal network. */
238typedef INTNETNETWORK *PINTNETNETWORK;
239
240/** The size of the buffer INTNETNETWORK::pbTmp points at. */
241#define INTNETNETWORK_TMP_SIZE 2048
242
243
244/**
245 * Internal networking instance.
246 */
247typedef struct INTNET
248{
249 /** Mutex protecting the network creation, opening and destruction.
250 * (This means all operations affecting the pNetworks list.) */
251 RTSEMFASTMUTEX FastMutex;
252 /** List of networks. Protected by INTNET::Spinlock. */
253 PINTNETNETWORK volatile pNetworks;
254 /** Handle table for the interfaces. */
255 RTHANDLETABLE hHtIfs;
256} INTNET;
257
258
259
260/*******************************************************************************
261* Internal Functions *
262*******************************************************************************/
263static PINTNETTRUNKIF intnetR0TrunkIfRetain(PINTNETTRUNKIF pThis);
264static void intnetR0TrunkIfRelease(PINTNETTRUNKIF pThis);
265static bool intnetR0TrunkIfOutLock(PINTNETTRUNKIF pThis);
266static void intnetR0TrunkIfOutUnlock(PINTNETTRUNKIF pThis);
267
268
269/**
270 * Initializes a scatter / gather buffer from a simple linear buffer.
271 *
272 * @returns Pointer to the start of the frame.
273 * @param pSG Pointer to the scatter / gather structure.
274 * (The pvOwnerData, fFlags, cUsers, and cSegsAlloc members are left untouched.)
275 * @param pvFrame Pointer to the frame
276 * @param cbFrame The size of the frame.
277 */
278DECLINLINE(void) intnetR0SgInitTemp(PINTNETSG pSG, void *pvFrame, uint32_t cbFrame)
279{
280 pSG->pvOwnerData = NULL;
281 pSG->pvUserData = NULL;
282 pSG->pvUserData2 = NULL;
283 pSG->cbTotal = cbFrame;
284 pSG->cUsers = 1;
285 pSG->fFlags = INTNETSG_FLAGS_TEMP;
286 pSG->cSegsAlloc = 1;
287 pSG->cSegsUsed = 1;
288 pSG->aSegs[0].Phys = NIL_RTHCPHYS;
289 pSG->aSegs[0].pv = pvFrame;
290 pSG->aSegs[0].cb = cbFrame;
291}
292
293
294/**
295 * Initializes a scatter / gather buffer from a internal networking packet.
296 *
297 * @returns Pointer to the start of the frame.
298 * @param pSG Pointer to the scatter / gather structure.
299 * (The pvOwnerData, fFlags, cUsers, and cSegsAlloc members are left untouched.)
300 * @param pHdr Pointer to the packet header.
301 * @param pBuf The buffer the header is within. Only used in strict builds.
302 */
303DECLINLINE(void) intnetR0SgInitFromPkt(PINTNETSG pSG, PCINTNETHDR pPktHdr, PCINTNETBUF pBuf)
304{
305 pSG->cSegsUsed = 1;
306 pSG->cbTotal = pSG->aSegs[0].cb = pPktHdr->cbFrame;
307 pSG->aSegs[0].pv = INTNETHdrGetFramePtr(pPktHdr, pBuf);
308 pSG->aSegs[0].Phys = NIL_RTHCPHYS;
309}
310
311
312/**
313 * Worker for intnetR0SgWritePart that deals with the case where the
314 * request doesn't fit into the first segment.
315 *
316 * @returns true, unless the request or SG invalid.
317 * @param pSG The SG list to write to.
318 * @param off Where to start writing (offset into the SG).
319 * @param cb How much to write.
320 * @param pvBuf The buffer to containing the bits to write.
321 */
322static bool intnetR0SgWritePartSlow(PCINTNETSG pSG, uint32_t off, uint32_t cb, void const *pvBuf)
323{
324 if (RT_UNLIKELY(off + cb > pSG->cbTotal))
325 return false;
326
327 /*
328 * Skip ahead to the segment where off starts.
329 */
330 unsigned const cSegs = pSG->cSegsUsed; Assert(cSegs == pSG->cSegsUsed);
331 unsigned iSeg = 0;
332 while (off > pSG->aSegs[iSeg].cb)
333 {
334 off -= pSG->aSegs[iSeg++].cb;
335 AssertReturn(iSeg < cSegs, false);
336 }
337
338 /*
339 * Copy the data, hoping that it's all from one segment...
340 */
341 uint32_t cbCanCopy = pSG->aSegs[iSeg].cb - off;
342 if (cbCanCopy >= cb)
343 memcpy((uint8_t *)pSG->aSegs[iSeg].pv + off, pvBuf, cb);
344 else
345 {
346 /* copy the portion in the current segment. */
347 memcpy((uint8_t *)pSG->aSegs[iSeg].pv + off, pvBuf, cbCanCopy);
348 cb -= cbCanCopy;
349
350 /* copy the portions in the other segments. */
351 do
352 {
353 pvBuf = (uint8_t const *)pvBuf + cbCanCopy;
354 iSeg++;
355 AssertReturn(iSeg < cSegs, false);
356
357 cbCanCopy = RT_MIN(cb, pSG->aSegs[iSeg].cb);
358 memcpy(pSG->aSegs[iSeg].pv, pvBuf, cbCanCopy);
359
360 cb -= cbCanCopy;
361 } while (cb > 0);
362 }
363
364 return true;
365}
366
367
368/**
369 * Writes to a part of an SG.
370 *
371 * @returns true on success, false on failure (out of bounds).
372 * @param pSG The SG list to write to.
373 * @param off Where to start writing (offset into the SG).
374 * @param cb How much to write.
375 * @param pvBuf The buffer to containing the bits to write.
376 */
377DECLINLINE(bool) intnetR0SgWritePart(PCINTNETSG pSG, uint32_t off, uint32_t cb, void const *pvBuf)
378{
379 Assert(off + cb > off);
380
381 /* The optimized case. */
382 if (RT_LIKELY( pSG->cSegsUsed == 1
383 || pSG->aSegs[0].cb >= off + cb))
384 {
385 Assert(pSG->cbTotal == pSG->aSegs[0].cb);
386 memcpy((uint8_t *)pSG->aSegs[0].pv + off, pvBuf, cb);
387 return true;
388 }
389 return intnetR0SgWritePartSlow(pSG, off, cb, pvBuf);
390}
391
392
393/**
394 * Reads a byte from a SG list.
395 *
396 * @returns The byte on success. 0xff on failure.
397 * @param pSG The SG list to read.
398 * @param off The offset (into the SG) off the byte.
399 */
400DECLINLINE(uint8_t) intnetR0SgReadByte(PCINTNETSG pSG, uint32_t off)
401{
402 if (RT_LIKELY(pSG->aSegs[0].cb > off))
403 return ((uint8_t const *)pSG->aSegs[0].pv)[off];
404
405 off -= pSG->aSegs[0].cb;
406 unsigned const cSegs = pSG->cSegsUsed; Assert(cSegs == pSG->cSegsUsed);
407 for (unsigned iSeg = 1; iSeg < cSegs; iSeg++)
408 {
409 if (pSG->aSegs[iSeg].cb > off)
410 return ((uint8_t const *)pSG->aSegs[iSeg].pv)[off];
411 off -= pSG->aSegs[iSeg].cb;
412 }
413 return false;
414}
415
416
417/**
418 * Worker for intnetR0SgReadPart that deals with the case where the
419 * requested data isn't in the first segment.
420 *
421 * @returns true, unless the SG is invalid.
422 * @param pSG The SG list to read.
423 * @param off Where to start reading (offset into the SG).
424 * @param cb How much to read.
425 * @param pvBuf The buffer to read into.
426 */
427static bool intnetR0SgReadPartSlow(PCINTNETSG pSG, uint32_t off, uint32_t cb, void *pvBuf)
428{
429 if (RT_UNLIKELY(off + cb > pSG->cbTotal))
430 return false;
431
432 /*
433 * Skip ahead to the segment where off starts.
434 */
435 unsigned const cSegs = pSG->cSegsUsed; Assert(cSegs == pSG->cSegsUsed);
436 unsigned iSeg = 0;
437 while (off > pSG->aSegs[iSeg].cb)
438 {
439 off -= pSG->aSegs[iSeg++].cb;
440 AssertReturn(iSeg < cSegs, false);
441 }
442
443 /*
444 * Copy the data, hoping that it's all from one segment...
445 */
446 uint32_t cbCanCopy = pSG->aSegs[iSeg].cb - off;
447 if (cbCanCopy >= cb)
448 memcpy(pvBuf, (uint8_t const *)pSG->aSegs[iSeg].pv + off, cb);
449 else
450 {
451 /* copy the portion in the current segment. */
452 memcpy(pvBuf, (uint8_t const *)pSG->aSegs[iSeg].pv + off, cbCanCopy);
453 cb -= cbCanCopy;
454
455 /* copy the portions in the other segments. */
456 do
457 {
458 pvBuf = (uint8_t *)pvBuf + cbCanCopy;
459 iSeg++;
460 AssertReturn(iSeg < cSegs, false);
461
462 cbCanCopy = RT_MIN(cb, pSG->aSegs[iSeg].cb);
463 memcpy(pvBuf, (uint8_t const *)pSG->aSegs[iSeg].pv, cbCanCopy);
464
465 cb -= cbCanCopy;
466 } while (cb > 0);
467 }
468
469 return true;
470}
471
472
473/**
474 * Reads a part of an SG into a buffer.
475 *
476 * @returns true on success, false on failure (out of bounds).
477 * @param pSG The SG list to read.
478 * @param off Where to start reading (offset into the SG).
479 * @param cb How much to read.
480 * @param pvBuf The buffer to read into.
481 */
482DECLINLINE(bool) intnetR0SgReadPart(PCINTNETSG pSG, uint32_t off, uint32_t cb, void *pvBuf)
483{
484 Assert(off + cb > off);
485
486 /* The optimized case. */
487 if (RT_LIKELY( pSG->cSegsUsed == 1
488 || pSG->aSegs[0].cb >= off + cb))
489 {
490 Assert(pSG->cbTotal == pSG->aSegs[0].cb);
491 memcpy(pvBuf, (uint8_t const *)pSG->aSegs[0].pv + off, cb);
492 return true;
493 }
494 return intnetR0SgReadPartSlow(pSG, off, cb, pvBuf);
495}
496
497
498/**
499 * Reads an entire SG into a fittingly size buffer.
500 *
501 * @param pSG The SG list to read.
502 * @param pvBuf The buffer to read into (at least pSG->cbTotal in size).
503 */
504DECLINLINE(void) intnetR0SgRead(PCINTNETSG pSG, void *pvBuf)
505{
506 if (pSG->cSegsUsed == 1)
507 {
508 Assert(pSG->cbTotal == pSG->aSegs[0].cb);
509 memcpy(pvBuf, pSG->aSegs[0].pv, pSG->cbTotal);
510 }
511 else
512 {
513 uint8_t *pbDst = (uint8_t *)pvBuf;
514 unsigned const cSegs = pSG->cSegsUsed; Assert(cSegs == pSG->cSegsUsed);
515 for (unsigned iSeg = 0; iSeg < cSegs; iSeg++)
516 {
517 uint32_t cbSeg = pSG->aSegs[iSeg].cb;
518 Assert(cbSeg <= pSG->cbTotal && (uintptr_t)(pbDst - (uint8_t *)pvBuf) + cbSeg <= pSG->cbTotal);
519 memcpy(pbDst, pSG->aSegs[iSeg].pv, cbSeg);
520 pbDst += cbSeg;
521 }
522 }
523}
524
525
526
527
528
529
530
531/**
532 * Retain an interface.
533 *
534 * @returns VBox status code, can assume success in most situations.
535 * @param pIf The interface instance.
536 * @param pSession The current session.
537 */
538DECLINLINE(int) intnetR0IfRetain(PINTNETIF pIf, PSUPDRVSESSION pSession)
539{
540 int rc = SUPR0ObjAddRefEx(pIf->pvObj, pSession, true /* fNoBlocking */);
541 AssertRCReturn(rc, rc);
542 return VINF_SUCCESS;
543}
544
545
546/**
547 * Release an interface previously retained by intnetR0IfRetain or
548 * by handle lookup/freeing.
549 *
550 * @returns true if destroyed, false if not.
551 * @param pIf The interface instance.
552 * @param pSession The current session.
553 */
554DECLINLINE(bool) intnetR0IfRelease(PINTNETIF pIf, PSUPDRVSESSION pSession)
555{
556 int rc = SUPR0ObjRelease(pIf->pvObj, pSession);
557 AssertRC(rc);
558 return rc == VINF_OBJECT_DESTROYED;
559}
560
561
562/**
563 * RTHandleCreateEx callback that retains an object in the
564 * handle table before returning it.
565 *
566 * (Avoids racing the freeing of the handle.)
567 *
568 * @returns VBox status code.
569 * @param hHandleTable The handle table (ignored).
570 * @param pvObj The object (INTNETIF).
571 * @param pvCtx The context (SUPDRVSESSION).
572 * @param pvUser The user context (ignored).
573 */
574static DECLCALLBACK(int) intnetR0IfRetainHandle(RTHANDLETABLE hHandleTable, void *pvObj, void *pvCtx, void *pvUser)
575{
576 NOREF(pvUser);
577 NOREF(hHandleTable);
578 PINTNETIF pIf = (PINTNETIF)pvObj;
579 if (pIf->hIf != INTNET_HANDLE_INVALID) /* Don't try retain it if called from intnetR0IfDestruct. */
580 return intnetR0IfRetain(pIf, (PSUPDRVSESSION)pvCtx);
581 return VINF_SUCCESS;
582}
583
584
585
586
587
588
589/**
590 * Checks if the IPv4 address is a broadcast address.
591 * @returns true/false.
592 * @param Addr The address, network endian.
593 */
594DECLINLINE(bool) intnetR0IPv4AddrIsBroadcast(RTNETADDRIPV4 Addr)
595{
596 /* Just check for 255.255.255.255 atm. */
597 return Addr.u == UINT32_MAX;
598}
599
600
601/**
602 * Checks if the IPv4 address is a good interface address.
603 * @returns true/false.
604 * @param Addr The address, network endian.
605 */
606DECLINLINE(bool) intnetR0IPv4AddrIsGood(RTNETADDRIPV4 Addr)
607{
608 /* Usual suspects. */
609 if ( Addr.u == UINT32_MAX /* 255.255.255.255 - broadcast. */
610 || Addr.au8[0] == 0) /* Current network, can be used as source address. */
611 return false;
612
613 /* Unusual suspects. */
614 if (RT_UNLIKELY( Addr.au8[0] == 127 /* Loopback */
615 || (Addr.au8[0] & 0xf0) == 224 /* Multicast */
616 ))
617 return false;
618 return true;
619}
620
621
622/**
623 * Gets the address size of a network layer type.
624 *
625 * @returns size in bytes.
626 * @param enmType The type.
627 */
628DECLINLINE(uint8_t) intnetR0AddrSize(INTNETADDRTYPE enmType)
629{
630 switch (enmType)
631 {
632 case kIntNetAddrType_IPv4: return 4;
633 case kIntNetAddrType_IPv6: return 16;
634 case kIntNetAddrType_IPX: return 4 + 6;
635 default: AssertFailedReturn(0);
636 }
637}
638
639
640/**
641 * Compares two address to see if they are equal, assuming naturally align structures.
642 *
643 * @returns true if equal, false if not.
644 * @param pAddr1 The first address.
645 * @param pAddr2 The second address.
646 * @param cbAddr The address size.
647 */
648DECLINLINE(bool) intnetR0AddrUIsEqualEx(PCRTNETADDRU pAddr1, PCRTNETADDRU pAddr2, uint8_t const cbAddr)
649{
650 switch (cbAddr)
651 {
652 case 4: /* IPv4 */
653 return pAddr1->au32[0] == pAddr2->au32[0];
654 case 16: /* IPv6 */
655 return pAddr1->au64[0] == pAddr2->au64[0]
656 && pAddr1->au64[1] == pAddr2->au64[1];
657 case 10: /* IPX */
658 return pAddr1->au64[0] == pAddr2->au64[0]
659 && pAddr1->au16[4] == pAddr2->au16[4];
660 default:
661 AssertFailedReturn(false);
662 }
663}
664
665
666/**
667 * Worker for intnetR0IfAddrCacheLookup that performs the lookup
668 * in the remaining cache entries after the caller has check the
669 * most likely ones.
670 *
671 * @returns -1 if not found, the index of the cache entry if found.
672 * @param pCache The cache.
673 * @param pAddr The address.
674 * @param cbAddr The address size (optimization).
675 */
676static int intnetR0IfAddrCacheLookupSlow(PCINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr)
677{
678 unsigned i = pCache->cEntries - 2;
679 uint8_t const *pbEntry = pCache->pbEntries + pCache->cbEntry * i;
680 while (i >= 1)
681 {
682 if (intnetR0AddrUIsEqualEx((PCRTNETADDRU)pbEntry, pAddr, cbAddr))
683 return i;
684 pbEntry -= pCache->cbEntry;
685 i--;
686 }
687
688 return -1;
689}
690
691/**
692 * Lookup an address in a cache without any expectations.
693 *
694 * @returns -1 if not found, the index of the cache entry if found.
695 * @param pCache The cache.
696 * @param pAddr The address.
697 * @param cbAddr The address size (optimization).
698 */
699DECLINLINE(int) intnetR0IfAddrCacheLookup(PCINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr)
700{
701 Assert(pCache->cbAddress == cbAddr);
702
703 /*
704 * The optimized case is when there is one cache entry and
705 * it doesn't match.
706 */
707 unsigned i = pCache->cEntries;
708 if ( i > 0
709 && intnetR0AddrUIsEqualEx((PCRTNETADDRU)pCache->pbEntries, pAddr, cbAddr))
710 return 0;
711 if (i <= 1)
712 return -1;
713
714 /*
715 * Check the last entry.
716 */
717 i--;
718 if (intnetR0AddrUIsEqualEx((PCRTNETADDRU)(pCache->pbEntries + pCache->cbEntry * i), pAddr, cbAddr))
719 return i;
720 if (i <= 1)
721 return -1;
722
723 return intnetR0IfAddrCacheLookupSlow(pCache, pAddr, cbAddr);
724}
725
726
727/** Same as intnetR0IfAddrCacheLookup except we expect the address to be present already. */
728DECLINLINE(int) intnetR0IfAddrCacheLookupLikely(PCINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr)
729{
730 /** @todo implement this. */
731 return intnetR0IfAddrCacheLookup(pCache, pAddr, cbAddr);
732}
733
734
735/**
736 * Worker for intnetR0IfAddrCacheLookupUnlikely that performs
737 * the lookup in the remaining cache entries after the caller
738 * has check the most likely ones.
739 *
740 * The routine is expecting not to find the address.
741 *
742 * @returns -1 if not found, the index of the cache entry if found.
743 * @param pCache The cache.
744 * @param pAddr The address.
745 * @param cbAddr The address size (optimization).
746 */
747static int intnetR0IfAddrCacheInCacheUnlikelySlow(PCINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr)
748{
749 /*
750 * Perform a full table lookup.
751 */
752 unsigned i = pCache->cEntries - 2;
753 uint8_t const *pbEntry = pCache->pbEntries + pCache->cbEntry * i;
754 while (i >= 1)
755 {
756 if (RT_UNLIKELY(intnetR0AddrUIsEqualEx((PCRTNETADDRU)pbEntry, pAddr, cbAddr)))
757 return i;
758 pbEntry -= pCache->cbEntry;
759 i--;
760 }
761
762 return -1;
763}
764
765
766/**
767 * Lookup an address in a cache expecting not to find it.
768 *
769 * @returns -1 if not found, the index of the cache entry if found.
770 * @param pCache The cache.
771 * @param pAddr The address.
772 * @param cbAddr The address size (optimization).
773 */
774DECLINLINE(int) intnetR0IfAddrCacheLookupUnlikely(PCINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr)
775{
776 Assert(pCache->cbAddress == cbAddr);
777
778 /*
779 * The optimized case is when there is one cache entry and
780 * it doesn't match.
781 */
782 unsigned i = pCache->cEntries;
783 if (RT_UNLIKELY( i > 0
784 && intnetR0AddrUIsEqualEx((PCRTNETADDRU)pCache->pbEntries, pAddr, cbAddr)))
785 return 0;
786 if (RT_LIKELY(i <= 1))
787 return -1;
788
789 /*
790 * Then check the last entry and return if there are just two cache entries.
791 */
792 i--;
793 if (RT_UNLIKELY(intnetR0AddrUIsEqualEx((PCRTNETADDRU)(pCache->pbEntries + pCache->cbEntry * i), pAddr, cbAddr)))
794 return i;
795 if (i <= 1)
796 return -1;
797
798 return intnetR0IfAddrCacheInCacheUnlikelySlow(pCache, pAddr, cbAddr);
799}
800
801
802/**
803 * Deletes a specific cache entry.
804 *
805 * Worker for intnetR0NetworkAddrCacheDelete and intnetR0NetworkAddrCacheDeleteMinusIf.
806 *
807 * @param pIf The interface (for logging).
808 * @param pCache The cache.
809 * @param iEntry The entry to delete.
810 * @param pszMsg Log message.
811 */
812static void intnetR0IfAddrCacheDeleteIt(PINTNETIF pIf, PINTNETADDRCACHE pCache, int iEntry, const char *pszMsg)
813{
814 AssertReturnVoid(iEntry < pCache->cEntries);
815 AssertReturnVoid(iEntry >= 0);
816#ifdef LOG_ENABLED
817 INTNETADDRTYPE enmAddrType = (INTNETADDRTYPE)(uintptr_t)(pCache - &pIf->aAddrCache[0]);
818 PCRTNETADDRU pAddr = (PCRTNETADDRU)(pCache->pbEntries + iEntry * pCache->cbEntry);
819 switch (enmAddrType)
820 {
821 case kIntNetAddrType_IPv4:
822 Log(("intnetR0IfAddrCacheDeleteIt: hIf=%#x MAC=%.6Rhxs IPv4 added #%d %d.%d.%d.%d %s\n",
823 pIf->hIf, &pIf->Mac, iEntry, pAddr->au8[0], pAddr->au8[1], pAddr->au8[2], pAddr->au8[3], pszMsg));
824 break;
825 default:
826 Log(("intnetR0IfAddrCacheDeleteIt: hIf=%RX32 MAC=%.6Rhxs type=%d #%d %.*Rhxs %s\n",
827 pIf->hIf, &pIf->Mac, enmAddrType, iEntry, pCache->cbAddress, pAddr, pszMsg));
828 break;
829 }
830#endif
831
832 pCache->cEntries--;
833 if (iEntry < pCache->cEntries)
834 memmove(pCache->pbEntries + iEntry * pCache->cbEntry,
835 pCache->pbEntries + (iEntry + 1) * pCache->cbEntry,
836 (pCache->cEntries - iEntry) * pCache->cbEntry);
837}
838
839
840/**
841 * Deletes an address from the cache, assuming it isn't actually in the cache.
842 *
843 * @param pIf The interface (for logging).
844 * @param pCache The cache.
845 * @param pAddr The address.
846 * @param cbAddr The address size (optimization).
847 */
848DECLINLINE(void) intnetR0IfAddrCacheDelete(PINTNETIF pIf, PINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr, const char *pszMsg)
849{
850 int i = intnetR0IfAddrCacheLookup(pCache, pAddr, cbAddr);
851 if (RT_UNLIKELY(i >= 0))
852 intnetR0IfAddrCacheDeleteIt(pIf, pCache, i, pszMsg);
853}
854
855
856/**
857 * Deletes the address from all the interface caches.
858 *
859 * This is used to remove stale entries that has been reassigned to
860 * other machines on the network.
861 *
862 * @param pNetwork The network.
863 * @param pAddr The address.
864 * @param enmType The address type.
865 * @param cbAddr The address size (optimization).
866 * @param pszMsg Log message.
867 */
868DECLINLINE(void) intnetR0NetworkAddrCacheDelete(PINTNETNETWORK pNetwork, PCRTNETADDRU pAddr, INTNETADDRTYPE const enmType,
869 uint8_t const cbAddr, const char *pszMsg)
870{
871 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
872 {
873 int i = intnetR0IfAddrCacheLookup(&pIf->aAddrCache[enmType], pAddr, cbAddr);
874 if (RT_UNLIKELY(i >= 0))
875 intnetR0IfAddrCacheDeleteIt(pIf, &pIf->aAddrCache[enmType], i, pszMsg);
876 }
877}
878
879
880/**
881 * Deletes the address from all the interface caches except the specified one.
882 *
883 * This is used to remove stale entries that has been reassigned to
884 * other machines on the network.
885 *
886 * @param pNetwork The network.
887 * @param pAddr The address.
888 * @param enmType The address type.
889 * @param cbAddr The address size (optimization).
890 * @param pszMsg Log message.
891 */
892DECLINLINE(void) intnetR0NetworkAddrCacheDeleteMinusIf(PINTNETNETWORK pNetwork, PINTNETIF pIfSender, PCRTNETADDRU pAddr,
893 INTNETADDRTYPE const enmType, uint8_t const cbAddr, const char *pszMsg)
894{
895 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
896 if (pIf != pIfSender)
897 {
898 int i = intnetR0IfAddrCacheLookup(&pIf->aAddrCache[enmType], pAddr, cbAddr);
899 if (RT_UNLIKELY(i >= 0))
900 intnetR0IfAddrCacheDeleteIt(pIf, &pIf->aAddrCache[enmType], i, pszMsg);
901 }
902}
903
904
905/**
906 * Lookup an address on the network, returning the (first) interface
907 * having it in its address cache.
908 *
909 * @returns Pointer to the interface on success, NULL if not found.
910 * @param pNetwork The network.
911 * @param pAddr The address to lookup.
912 * @param enmType The address type.
913 * @param cbAddr The size of the address.
914 */
915DECLINLINE(PINTNETIF) intnetR0NetworkAddrCacheLookupIf(PINTNETNETWORK pNetwork, PCRTNETADDRU pAddr, INTNETADDRTYPE const enmType, uint8_t const cbAddr)
916{
917 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
918 {
919 int i = intnetR0IfAddrCacheLookup(&pIf->aAddrCache[enmType], pAddr, cbAddr);
920 if (i >= 0)
921 return pIf;
922 }
923 return NULL;
924}
925
926
927/**
928 * Adds an address to the cache, the caller is responsible for making sure it'
929 * s not already in the cache.
930 *
931 * @param pIf The interface (for logging).
932 * @param pCache The address cache.
933 * @param pAddr The address.
934 * @param pszMsg log message.
935 */
936static void intnetR0IfAddrCacheAddIt(PINTNETIF pIf, PINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, const char *pszMsg)
937{
938 if (!pCache->cEntriesAlloc)
939 {
940 /* Allocate the first array */
941 pCache->pbEntries = (uint8_t *)RTMemAllocZ(pCache->cbEntry * 16);
942 if (!pCache->pbEntries)
943 return;
944 pCache->cEntriesAlloc = 16;
945 }
946 else if (pCache->cEntries >= pCache->cEntriesAlloc)
947 {
948 bool fReplace = true;
949 if (pCache->cEntriesAlloc < 64)
950 {
951 uint8_t cEntriesAlloc = pCache->cEntriesAlloc + 16;
952 void *pvNew = RTMemRealloc(pCache->pbEntries, pCache->cbEntry * cEntriesAlloc);
953 if (pvNew)
954 {
955 pCache->pbEntries = (uint8_t *)pvNew;
956 pCache->cEntriesAlloc = cEntriesAlloc;
957 fReplace = false;
958 }
959 }
960 if (fReplace)
961 {
962 /* simple FIFO, might consider usage/ageing here... */
963 Log(("intnetR0IfAddrCacheAddIt: type=%d replacing %.*Rhxs\n",
964 (int)(uintptr_t)(pCache - &pIf->aAddrCache[0]), pCache->cbAddress, pCache->pbEntries));
965 memmove(pCache->pbEntries, pCache->pbEntries + pCache->cbEntry, pCache->cbEntry * (pCache->cEntries - 1));
966 pCache->cEntries--;
967 }
968 }
969
970 /*
971 * Add the new entry to the end of the array.
972 */
973 uint8_t *pbEntry = pCache->pbEntries + pCache->cEntries * pCache->cbEntry;
974 memcpy(pbEntry, pAddr, pCache->cbAddress);
975 memset(pbEntry + pCache->cbAddress, '\0', pCache->cbEntry - pCache->cbAddress);
976#ifdef LOG_ENABLED
977 INTNETADDRTYPE enmAddrType = (INTNETADDRTYPE)(uintptr_t)(pCache - &pIf->aAddrCache[0]);
978 switch (enmAddrType)
979 {
980 case kIntNetAddrType_IPv4:
981 Log(("intnetR0IfAddrCacheAddIt: hIf=%#x MAC=%.6Rhxs IPv4 added #%d %d.%d.%d.%d %s\n",
982 pIf->hIf, &pIf->Mac, pCache->cEntries, pAddr->au8[0], pAddr->au8[1], pAddr->au8[2], pAddr->au8[3], pszMsg));
983 break;
984 default:
985 Log(("intnetR0IfAddrCacheAddIt: hIf=%#x MAC=%.6Rhxs type=%d added #%d %.*Rhxs %s\n",
986 pIf->hIf, &pIf->Mac, enmAddrType, pCache->cEntries, pCache->cbAddress, pAddr, pszMsg));
987 break;
988 }
989#endif
990 pCache->cEntries++;
991 Assert(pCache->cEntries <= pCache->cEntriesAlloc);
992}
993
994
995/**
996 * A intnetR0IfAddrCacheAdd worker that performs the rest of the lookup.
997 *
998 * @param pIf The interface (for logging).
999 * @param pCache The address cache.
1000 * @param pAddr The address.
1001 * @param cbAddr The size of the address (optimization).
1002 * @param pszMsg Log message.
1003 */
1004static void intnetR0IfAddrCacheAddSlow(PINTNETIF pIf, PINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr, const char *pszMsg)
1005{
1006 /*
1007 * Check all but the first and last entries, the caller
1008 * has already checked those.
1009 */
1010 int i = pCache->cEntries - 2;
1011 uint8_t const *pbEntry = pCache->pbEntries + pCache->cbEntry;
1012 while (i >= 1)
1013 {
1014 if (RT_LIKELY(intnetR0AddrUIsEqualEx((PCRTNETADDRU)pbEntry, pAddr, cbAddr)))
1015 return;
1016 pbEntry += pCache->cbEntry;
1017 i--;
1018 }
1019
1020 /*
1021 * Not found, add it.
1022 */
1023 intnetR0IfAddrCacheAddIt(pIf, pCache, pAddr, pszMsg);
1024}
1025
1026
1027/**
1028 * Adds an address to the cache if it's not already there.
1029 *
1030 * @param pIf The interface (for logging).
1031 * @param pCache The address cache.
1032 * @param pAddr The address.
1033 * @param cbAddr The size of the address (optimization).
1034 * @param pszMsg Log message.
1035 */
1036DECLINLINE(void) intnetR0IfAddrCacheAdd(PINTNETIF pIf, PINTNETADDRCACHE pCache, PCRTNETADDRU pAddr, uint8_t const cbAddr, const char *pszMsg)
1037{
1038 Assert(pCache->cbAddress == cbAddr);
1039
1040 /*
1041 * The optimized case is when the address the first or last cache entry.
1042 */
1043 unsigned i = pCache->cEntries;
1044 if (RT_LIKELY( i > 0
1045 && ( intnetR0AddrUIsEqualEx((PCRTNETADDRU)pCache->pbEntries, pAddr, cbAddr)
1046 || (i > 1
1047 && intnetR0AddrUIsEqualEx((PCRTNETADDRU)(pCache->pbEntries + pCache->cbEntry * i), pAddr, cbAddr))) ))
1048 return;
1049 intnetR0IfAddrCacheAddSlow(pIf, pCache, pAddr, cbAddr, pszMsg);
1050}
1051
1052
1053#ifdef INTNET_WITH_DHCP_SNOOPING
1054
1055/**
1056 * Snoops IP assignments and releases from the DHCPv4 traffic.
1057 *
1058 * The caller is responsible for making sure this traffic between the
1059 * BOOTPS and BOOTPC ports and validate the IP header. The UDP packet
1060 * need not be validated beyond the ports.
1061 *
1062 * @param pNetwork The network this frame was seen on.
1063 * @param pIpHdr Pointer to a valid IP header. This is for pseudo
1064 * header validation, so only the minimum header size
1065 * needs to be available and valid here.
1066 * @param pUdpHdr Pointer to the UDP header in the frame.
1067 * @param cbUdpPkt What's left of the frame when starting at the UDP header.
1068 */
1069static void intnetR0NetworkSnoopDhcp(PINTNETNETWORK pNetwork, PCRTNETIPV4 pIpHdr, PCRTNETUDP pUdpHdr, uint32_t cbUdpPkt)
1070{
1071 /*
1072 * Check if the DHCP message is valid and get the type.
1073 */
1074 if (!RTNetIPv4IsUDPValid(pIpHdr, pUdpHdr, pUdpHdr + 1, cbUdpPkt))
1075 {
1076 Log6(("Bad UDP packet\n"));
1077 return;
1078 }
1079 PCRTNETBOOTP pDhcp = (PCRTNETBOOTP)(pUdpHdr + 1);
1080 uint8_t MsgType;
1081 if (!RTNetIPv4IsDHCPValid(pUdpHdr, pDhcp, cbUdpPkt - sizeof(*pUdpHdr), &MsgType))
1082 {
1083 Log6(("Bad DHCP packet\n"));
1084 return;
1085 }
1086
1087#ifdef LOG_ENABLED
1088 /*
1089 * Log it.
1090 */
1091 const char *pszType = "unknown";
1092 switch (MsgType)
1093 {
1094 case RTNET_DHCP_MT_DISCOVER: pszType = "discover"; break;
1095 case RTNET_DHCP_MT_OFFER: pszType = "offer"; break;
1096 case RTNET_DHCP_MT_REQUEST: pszType = "request"; break;
1097 case RTNET_DHCP_MT_DECLINE: pszType = "decline"; break;
1098 case RTNET_DHCP_MT_ACK: pszType = "ack";break;
1099 case RTNET_DHCP_MT_NAC: pszType = "nac"; break;
1100 case RTNET_DHCP_MT_RELEASE: pszType = "release"; break;
1101 case RTNET_DHCP_MT_INFORM: pszType = "inform"; break;
1102 }
1103 Log6(("DHCP msg: %d (%s) client %.6Rhxs ciaddr=%d.%d.%d.%d yiaddr=%d.%d.%d.%d\n", MsgType, pszType, &pDhcp->bp_chaddr,
1104 pDhcp->bp_ciaddr.au8[0], pDhcp->bp_ciaddr.au8[1], pDhcp->bp_ciaddr.au8[2], pDhcp->bp_ciaddr.au8[3],
1105 pDhcp->bp_yiaddr.au8[0], pDhcp->bp_yiaddr.au8[1], pDhcp->bp_yiaddr.au8[2], pDhcp->bp_yiaddr.au8[3]));
1106#endif /* LOG_EANBLED */
1107
1108 /*
1109 * Act upon the message.
1110 */
1111 switch (MsgType)
1112 {
1113#if 0
1114 case RTNET_DHCP_MT_REQUEST:
1115 /** @todo Check for valid non-broadcast requests w/ IP for any of the MACs we
1116 * know, and add the IP to the cache. */
1117 break;
1118#endif
1119
1120
1121 /*
1122 * Lookup the interface by its MAC address and insert the IPv4 address into the cache.
1123 * Delete the old client address first, just in case it changed in a renewal.
1124 */
1125 case RTNET_DHCP_MT_ACK:
1126 if (intnetR0IPv4AddrIsGood(pDhcp->bp_yiaddr))
1127 for (PINTNETIF pCur = pNetwork->pIFs; pCur; pCur = pCur->pNext)
1128 if ( pCur->fMacSet
1129 && !memcmp(&pCur->Mac, &pDhcp->bp_chaddr, sizeof(RTMAC)))
1130 {
1131 intnetR0IfAddrCacheDelete(pCur, &pCur->aAddrCache[kIntNetAddrType_IPv4],
1132 (PCRTNETADDRU)&pDhcp->bp_ciaddr, sizeof(RTNETADDRIPV4), "DHCP_MT_ACK");
1133 intnetR0IfAddrCacheAdd(pCur, &pCur->aAddrCache[kIntNetAddrType_IPv4],
1134 (PCRTNETADDRU)&pDhcp->bp_yiaddr, sizeof(RTNETADDRIPV4), "DHCP_MT_ACK");
1135 break;
1136 }
1137 break;
1138
1139
1140 /*
1141 * Lookup the interface by its MAC address and remove the IPv4 address(es) from the cache.
1142 */
1143 case RTNET_DHCP_MT_RELEASE:
1144 {
1145 for (PINTNETIF pCur = pNetwork->pIFs; pCur; pCur = pCur->pNext)
1146 if ( pCur->fMacSet
1147 && !memcmp(&pCur->Mac, &pDhcp->bp_chaddr, sizeof(RTMAC)))
1148 {
1149 intnetR0IfAddrCacheDelete(pCur, &pCur->aAddrCache[kIntNetAddrType_IPv4],
1150 (PCRTNETADDRU)&pDhcp->bp_ciaddr, sizeof(RTNETADDRIPV4), "DHCP_MT_RELEASE");
1151 intnetR0IfAddrCacheDelete(pCur, &pCur->aAddrCache[kIntNetAddrType_IPv4],
1152 (PCRTNETADDRU)&pDhcp->bp_yiaddr, sizeof(RTNETADDRIPV4), "DHCP_MT_RELEASE");
1153 }
1154 break;
1155 }
1156 }
1157
1158}
1159
1160
1161/**
1162 * Worker for intnetR0TrunkIfSnoopAddr that takes care of what
1163 * is likely to be a DHCP message.
1164 *
1165 * The caller has already check that the UDP source and destination ports
1166 * are BOOTPS or BOOTPC.
1167 *
1168 * @param pNetwork The network this frame was seen on.
1169 * @param pSG The gather list for the frame.
1170 */
1171static void intnetR0TrunkIfSnoopDhcp(PINTNETNETWORK pNetwork, PCINTNETSG pSG)
1172{
1173 /*
1174 * Get a pointer to a linear copy of the full packet, using the
1175 * temporary buffer if necessary.
1176 */
1177 PCRTNETIPV4 pIpHdr = (PCRTNETIPV4)((PCRTNETETHERHDR)pSG->aSegs[0].pv + 1);
1178 uint32_t cbPacket = pSG->cbTotal - sizeof(RTNETETHERHDR);
1179 if (pSG->cSegsUsed > 1)
1180 {
1181 cbPacket = RT_MIN(cbPacket, INTNETNETWORK_TMP_SIZE);
1182 Log6(("intnetR0TrunkIfSnoopDhcp: Copying IPv4/UDP/DHCP pkt %u\n", cbPacket));
1183 if (!intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR), cbPacket, pNetwork->pbTmp))
1184 return;
1185 //pSG->fFlags |= INTNETSG_FLAGS_PKT_CP_IN_TMP;
1186 pIpHdr = (PCRTNETIPV4)pNetwork->pbTmp;
1187 }
1188
1189 /*
1190 * Validate the IP header and find the UDP packet.
1191 */
1192 if (!RTNetIPv4IsHdrValid(pIpHdr, cbPacket, pSG->cbTotal - sizeof(RTNETETHERHDR)))
1193 {
1194 Log(("intnetR0TrunkIfSnoopDhcp: bad ip header\n"));
1195 return;
1196 }
1197 uint32_t cbIpHdr = pIpHdr->ip_hl * 4;
1198
1199 /*
1200 * Hand it over to the common DHCP snooper.
1201 */
1202 intnetR0NetworkSnoopDhcp(pNetwork, pIpHdr, (PCRTNETUDP)((uintptr_t)pIpHdr + cbIpHdr), cbPacket - cbIpHdr);
1203}
1204
1205#endif /* INTNET_WITH_DHCP_SNOOPING */
1206
1207
1208/**
1209 * Snoops up source addresses from ARP requests and purge these
1210 * from the address caches.
1211 *
1212 * The purpose of this purging is to get rid of stale addresses.
1213 *
1214 * @param pNetwork The network this frame was seen on.
1215 * @param pSG The gather list for the frame.
1216 */
1217static void intnetR0TrunkIfSnoopArp(PINTNETNETWORK pNetwork, PCINTNETSG pSG)
1218{
1219 /*
1220 * Check the minimum size first.
1221 */
1222 if (RT_UNLIKELY(pSG->cbTotal < sizeof(RTNETETHERHDR) + sizeof(RTNETARPIPV4)))
1223 return;
1224
1225 /*
1226 * Copy to temporary buffer if necessary.
1227 */
1228 uint32_t cbPacket = RT_MIN(pSG->cbTotal, sizeof(RTNETARPIPV4));
1229 PCRTNETARPIPV4 pArpIPv4 = (PCRTNETARPIPV4)((uintptr_t)pSG->aSegs[0].pv + sizeof(RTNETETHERHDR));
1230 if ( pSG->cSegsUsed != 1
1231 && pSG->aSegs[0].cb < cbPacket)
1232 {
1233 if ( (pSG->fFlags & (INTNETSG_FLAGS_ARP_IPV4 | INTNETSG_FLAGS_PKT_CP_IN_TMP))
1234 != (INTNETSG_FLAGS_ARP_IPV4 | INTNETSG_FLAGS_PKT_CP_IN_TMP)
1235 && !intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR), cbPacket, pNetwork->pbTmp))
1236 return;
1237 pArpIPv4 = (PCRTNETARPIPV4)pNetwork->pbTmp;
1238 }
1239
1240 /*
1241 * Ignore packets which doesn't interest us or we perceive as malformed.
1242 */
1243 if (RT_UNLIKELY( pArpIPv4->Hdr.ar_hlen != sizeof(RTMAC)
1244 || pArpIPv4->Hdr.ar_plen != sizeof(RTNETADDRIPV4)
1245 || pArpIPv4->Hdr.ar_htype != RT_H2BE_U16(RTNET_ARP_ETHER)
1246 || pArpIPv4->Hdr.ar_ptype != RT_H2BE_U16(RTNET_ETHERTYPE_IPV4)))
1247 return;
1248 uint16_t ar_oper = RT_H2BE_U16(pArpIPv4->Hdr.ar_oper);
1249 if (RT_UNLIKELY( ar_oper != RTNET_ARPOP_REQUEST
1250 && ar_oper != RTNET_ARPOP_REPLY))
1251 {
1252 Log6(("ts-ar: op=%#x\n", ar_oper));
1253 return;
1254 }
1255
1256 /*
1257 * Delete the source address if it's OK.
1258 */
1259 if ( !(pArpIPv4->ar_sha.au8[0] & 1)
1260 && ( pArpIPv4->ar_sha.au16[0]
1261 || pArpIPv4->ar_sha.au16[1]
1262 || pArpIPv4->ar_sha.au16[2])
1263 && intnetR0IPv4AddrIsGood(pArpIPv4->ar_spa))
1264 {
1265 Log6(("ts-ar: %d.%d.%d.%d / %.6Rhxs\n", pArpIPv4->ar_spa.au8[0], pArpIPv4->ar_spa.au8[1],
1266 pArpIPv4->ar_spa.au8[2], pArpIPv4->ar_spa.au8[3], &pArpIPv4->ar_sha));
1267 intnetR0NetworkAddrCacheDelete(pNetwork, (PCRTNETADDRU)&pArpIPv4->ar_spa,
1268 kIntNetAddrType_IPv4, sizeof(pArpIPv4->ar_spa), "tif/arp");
1269 }
1270}
1271
1272
1273#ifdef INTNET_WITH_DHCP_SNOOPING
1274/**
1275 * Snoop up addresses from ARP and DHCP traffic from frames comming
1276 * over the trunk connection.
1277 *
1278 * The caller is responsible for do some basic filtering before calling
1279 * this function.
1280 * For IPv4 this means checking against the minimum DHCPv4 frame size.
1281 *
1282 * @param pNetwork The network.
1283 * @param pSG The SG list for the frame.
1284 * @param EtherType The Ethertype of the frame.
1285 */
1286static void intnetR0TrunkIfSnoopAddr(PINTNETNETWORK pNetwork, PCINTNETSG pSG, uint16_t EtherType)
1287{
1288 switch (EtherType)
1289 {
1290 case RTNET_ETHERTYPE_IPV4:
1291 {
1292 uint32_t cbIpHdr;
1293 uint8_t b;
1294
1295 Assert(pSG->cbTotal >= sizeof(RTNETETHERHDR) + RTNETIPV4_MIN_LEN + RTNETUDP_MIN_LEN + RTNETBOOTP_DHCP_MIN_LEN);
1296 if (pSG->aSegs[0].cb >= sizeof(RTNETETHERHDR) + RTNETIPV4_MIN_LEN)
1297 {
1298 /* check if the protocol is UDP */
1299 PCRTNETIPV4 pIpHdr = (PCRTNETIPV4)((uint8_t const *)pSG->aSegs[0].pv + sizeof(RTNETETHERHDR));
1300 if (pIpHdr->ip_p != RTNETIPV4_PROT_UDP)
1301 return;
1302
1303 /* get the TCP header length */
1304 cbIpHdr = pIpHdr->ip_hl * 4;
1305 }
1306 else
1307 {
1308 /* check if the protocol is UDP */
1309 if ( intnetR0SgReadByte(pSG, sizeof(RTNETETHERHDR) + RT_OFFSETOF(RTNETIPV4, ip_p))
1310 != RTNETIPV4_PROT_UDP)
1311 return;
1312
1313 /* get the TCP header length */
1314 b = intnetR0SgReadByte(pSG, sizeof(RTNETETHERHDR) + 0); /* (IPv4 first byte, a bitfield) */
1315 cbIpHdr = (b & 0x0f) * 4;
1316 }
1317 if (cbIpHdr < RTNETIPV4_MIN_LEN)
1318 return;
1319
1320 /* compare the ports. */
1321 if (pSG->aSegs[0].cb >= sizeof(RTNETETHERHDR) + cbIpHdr + RTNETUDP_MIN_LEN)
1322 {
1323 PCRTNETUDP pUdpHdr = (PCRTNETUDP)((uint8_t const *)pSG->aSegs[0].pv + sizeof(RTNETETHERHDR) + cbIpHdr);
1324 if ( ( RT_BE2H_U16(pUdpHdr->uh_sport) != RTNETIPV4_PORT_BOOTPS
1325 && RT_BE2H_U16(pUdpHdr->uh_dport) != RTNETIPV4_PORT_BOOTPS)
1326 || ( RT_BE2H_U16(pUdpHdr->uh_dport) != RTNETIPV4_PORT_BOOTPC
1327 && RT_BE2H_U16(pUdpHdr->uh_sport) != RTNETIPV4_PORT_BOOTPC))
1328 return;
1329 }
1330 else
1331 {
1332 /* get the lower byte of the UDP source port number. */
1333 b = intnetR0SgReadByte(pSG, sizeof(RTNETETHERHDR) + cbIpHdr + RT_OFFSETOF(RTNETUDP, uh_sport) + 1);
1334 if ( b != RTNETIPV4_PORT_BOOTPS
1335 && b != RTNETIPV4_PORT_BOOTPC)
1336 return;
1337 uint8_t SrcPort = b;
1338 b = intnetR0SgReadByte(pSG, sizeof(RTNETETHERHDR) + cbIpHdr + RT_OFFSETOF(RTNETUDP, uh_sport));
1339 if (b)
1340 return;
1341
1342 /* get the lower byte of the UDP destination port number. */
1343 b = intnetR0SgReadByte(pSG, sizeof(RTNETETHERHDR) + cbIpHdr + RT_OFFSETOF(RTNETUDP, uh_dport) + 1);
1344 if ( b != RTNETIPV4_PORT_BOOTPS
1345 && b != RTNETIPV4_PORT_BOOTPC)
1346 return;
1347 if (b == SrcPort)
1348 return;
1349 b = intnetR0SgReadByte(pSG, sizeof(RTNETETHERHDR) + cbIpHdr + RT_OFFSETOF(RTNETUDP, uh_dport));
1350 if (b)
1351 return;
1352 }
1353 intnetR0TrunkIfSnoopDhcp(pNetwork, pSG);
1354 break;
1355 }
1356
1357 case RTNET_ETHERTYPE_IPV6:
1358 {
1359 /** @todo IPv6: Check for ICMPv6. It looks like type 133 (Router solicitation) might
1360 * need to be edited. Check out how NDP works... */
1361 break;
1362 }
1363
1364 case RTNET_ETHERTYPE_ARP:
1365 intnetR0TrunkIfSnoopArp(pNetwork, pSG);
1366 break;
1367 }
1368}
1369#endif /* INTNET_WITH_DHCP_SNOOPING */
1370
1371
1372/**
1373 * Deals with an IPv4 packet.
1374 *
1375 * This will fish out the source IP address and add it to the cache.
1376 * Then it will look for DHCPRELEASE requests (?) and anything else
1377 * that we migh find useful later.
1378 *
1379 * @param pIf The interface that's sending the frame.
1380 * @param pIpHdr Pointer to the IPv4 header in the frame.
1381 * @param cbPacket The size of the packet, or more correctly the
1382 * size of the frame without the ethernet header.
1383 */
1384static void intnetR0IfSnoopIPv4SourceAddr(PINTNETIF pIf, PCRTNETIPV4 pIpHdr, uint32_t cbPacket)
1385{
1386 /*
1387 * Check the header size first to prevent access invalid data.
1388 */
1389 if (cbPacket < RTNETIPV4_MIN_LEN)
1390 return;
1391 uint32_t cbHdr = (uint32_t)pIpHdr->ip_hl * 4;
1392 if ( cbHdr < RTNETIPV4_MIN_LEN
1393 || cbPacket < cbHdr)
1394 return;
1395
1396 /*
1397 * If the source address is good (not broadcast or my network) and
1398 * not already in the address cache of the sender, add it. Validate
1399 * the IP header before adding it.
1400 */
1401 bool fValidatedIpHdr = false;
1402 RTNETADDRU Addr;
1403 Addr.IPv4 = pIpHdr->ip_src;
1404 if ( intnetR0IPv4AddrIsGood(Addr.IPv4)
1405 && intnetR0IfAddrCacheLookupLikely(&pIf->aAddrCache[kIntNetAddrType_IPv4], &Addr, sizeof(Addr.IPv4)) < 0)
1406 {
1407 if (!RTNetIPv4IsHdrValid(pIpHdr, cbPacket, cbPacket))
1408 {
1409 Log(("intnetR0IfSnoopIPv4SourceAddr: bad ip header\n"));
1410 return;
1411 }
1412 intnetR0IfAddrCacheAddIt(pIf, &pIf->aAddrCache[kIntNetAddrType_IPv4], &Addr, "if/ipv4");
1413 fValidatedIpHdr = true;
1414 }
1415
1416#ifdef INTNET_WITH_DHCP_SNOOPING
1417 /*
1418 * Check for potential DHCP packets.
1419 */
1420 if ( pIpHdr->ip_p == RTNETIPV4_PROT_UDP /* DHCP is UDP. */
1421 && cbPacket >= cbHdr + RTNETUDP_MIN_LEN + RTNETBOOTP_DHCP_MIN_LEN) /* Min DHCP packet len */
1422 {
1423 PCRTNETUDP pUdpHdr = (PCRTNETUDP)((uint8_t const *)pIpHdr + cbHdr);
1424 if ( ( RT_BE2H_U16(pUdpHdr->uh_dport) == RTNETIPV4_PORT_BOOTPS
1425 || RT_BE2H_U16(pUdpHdr->uh_sport) == RTNETIPV4_PORT_BOOTPS)
1426 && ( RT_BE2H_U16(pUdpHdr->uh_sport) == RTNETIPV4_PORT_BOOTPC
1427 || RT_BE2H_U16(pUdpHdr->uh_dport) == RTNETIPV4_PORT_BOOTPC))
1428 {
1429 if ( fValidatedIpHdr
1430 || RTNetIPv4IsHdrValid(pIpHdr, cbPacket, cbPacket))
1431 intnetR0NetworkSnoopDhcp(pIf->pNetwork, pIpHdr, pUdpHdr, cbPacket - cbHdr);
1432 else
1433 Log(("intnetR0IfSnoopIPv4SourceAddr: bad ip header (dhcp)\n"));
1434 }
1435 }
1436#endif /* INTNET_WITH_DHCP_SNOOPING */
1437}
1438
1439
1440/**
1441 * Snoop up source addresses from an ARP request or reply.
1442 *
1443 * @param pIf The interface that's sending the frame.
1444 * @param pHdr The ARP header.
1445 * @param cbPacket The size of the packet (migth be larger than the ARP
1446 * request 'cause of min ethernet frame size).
1447 * @param pfSgFlags Pointer to the SG flags. This is used to tag the packet so we
1448 * don't have to repeat the frame parsing in intnetR0TrunkIfSend.
1449 */
1450static void intnetR0IfSnoopArpAddr(PINTNETIF pIf, PCRTNETARPIPV4 pArpIPv4, uint32_t cbPacket, uint16_t *pfSgFlags)
1451{
1452 /*
1453 * Ignore packets which doesn't interest us or we perceive as malformed.
1454 */
1455 if (RT_UNLIKELY(cbPacket < sizeof(RTNETARPIPV4)))
1456 return;
1457 if (RT_UNLIKELY( pArpIPv4->Hdr.ar_hlen != sizeof(RTMAC)
1458 || pArpIPv4->Hdr.ar_plen != sizeof(RTNETADDRIPV4)
1459 || pArpIPv4->Hdr.ar_htype != RT_H2BE_U16(RTNET_ARP_ETHER)
1460 || pArpIPv4->Hdr.ar_ptype != RT_H2BE_U16(RTNET_ETHERTYPE_IPV4)))
1461 return;
1462 uint16_t ar_oper = RT_H2BE_U16(pArpIPv4->Hdr.ar_oper);
1463 if (RT_UNLIKELY( ar_oper != RTNET_ARPOP_REQUEST
1464 && ar_oper != RTNET_ARPOP_REPLY))
1465 {
1466 Log6(("ar_oper=%#x\n", ar_oper));
1467 return;
1468 }
1469
1470 /*
1471 * Tag the SG as ARP IPv4 for later editing, then check for addresses
1472 * which can be removed or added to the address cache of the sender.
1473 */
1474 *pfSgFlags |= INTNETSG_FLAGS_ARP_IPV4;
1475
1476 if ( ar_oper == RTNET_ARPOP_REPLY
1477 && !(pArpIPv4->ar_tha.au8[0] & 1)
1478 && ( pArpIPv4->ar_tha.au16[0]
1479 || pArpIPv4->ar_tha.au16[1]
1480 || pArpIPv4->ar_tha.au16[2])
1481 && intnetR0IPv4AddrIsGood(pArpIPv4->ar_tpa))
1482 intnetR0IfAddrCacheDelete(pIf, &pIf->aAddrCache[kIntNetAddrType_IPv4],
1483 (PCRTNETADDRU)&pArpIPv4->ar_tpa, sizeof(RTNETADDRIPV4), "if/arp");
1484
1485 if ( !memcmp(&pArpIPv4->ar_sha, &pIf->Mac, sizeof(RTMAC))
1486 && intnetR0IPv4AddrIsGood(pArpIPv4->ar_spa))
1487 intnetR0IfAddrCacheAdd(pIf, &pIf->aAddrCache[kIntNetAddrType_IPv4],
1488 (PCRTNETADDRU)&pArpIPv4->ar_spa, sizeof(RTNETADDRIPV4), "if/arp");
1489}
1490
1491
1492
1493/**
1494 * Checks packets send by a normal interface for new network
1495 * layer addresses.
1496 *
1497 * @param pIf The interface that's sending the frame.
1498 * @param pbFrame The frame.
1499 * @param cbFrame The size of the frame.
1500 * @param pfSgFlags Pointer to the SG flags. This is used to tag the packet so we
1501 * don't have to repeat the frame parsing in intnetR0TrunkIfSend.
1502 */
1503static void intnetR0IfSnoopAddr(PINTNETIF pIf, uint8_t const *pbFrame, uint32_t cbFrame, uint16_t *pfSgFlags)
1504{
1505 /*
1506 * Fish out the ethertype and look for stuff we can handle.
1507 */
1508 if (cbFrame <= sizeof(RTNETETHERHDR))
1509 return;
1510 cbFrame -= sizeof(RTNETETHERHDR);
1511
1512 uint16_t EtherType = RT_H2BE_U16(((PCRTNETETHERHDR)pbFrame)->EtherType);
1513 switch (EtherType)
1514 {
1515 case RTNET_ETHERTYPE_IPV4:
1516 intnetR0IfSnoopIPv4SourceAddr(pIf, (PCRTNETIPV4)((PCRTNETETHERHDR)pbFrame + 1), cbFrame);
1517 break;
1518#if 0 /** @todo IntNet: implement IPv6 for wireless MAC sharing. */
1519 case RTNET_ETHERTYPE_IPV6:
1520 /** @todo IPv6: Check for ICMPv6. It looks like type 133 (Router solicitation) might
1521 * need to be edited. Check out how NDP works... */
1522 intnetR0IfSnoopIPv6SourceAddr(pIf, (PCINTNETIPV6)((PCRTNETETHERHDR)pbFrame + 1), cbFrame, pfSgFlags);
1523 break;
1524#endif
1525#if 0 /** @todo IntNet: implement IPX for wireless MAC sharing? */
1526 case RTNET_ETHERTYPE_IPX_1:
1527 case RTNET_ETHERTYPE_IPX_2:
1528 case RTNET_ETHERTYPE_IPX_3:
1529 intnetR0IfSnoopIpxSourceAddr(pIf, (PCINTNETIPX)((PCRTNETETHERHDR)pbFrame + 1), cbFrame, pfSgFlags);
1530 break;
1531#endif
1532 case RTNET_ETHERTYPE_ARP:
1533 intnetR0IfSnoopArpAddr(pIf, (PCRTNETARPIPV4)((PCRTNETETHERHDR)pbFrame + 1), cbFrame, pfSgFlags);
1534 break;
1535 }
1536}
1537
1538
1539/**
1540 * Writes a frame packet to the ring buffer.
1541 *
1542 * @returns VBox status code.
1543 * @param pBuf The buffer.
1544 * @param pRingBuf The ring buffer to read from.
1545 * @param pSG The gather list.
1546 * @param pNewDstMac Set the destination MAC address to the address if specified.
1547 */
1548static int intnetR0RingWriteFrame(PINTNETRINGBUF pRingBuf, PCINTNETSG pSG, PCRTMAC pNewDstMac)
1549{
1550 PINTNETHDR pHdr = NULL; /* shut up gcc*/
1551 void *pvDst = NULL; /* ditto */
1552 int rc = INTNETRingAllocateFrame(pRingBuf, pSG->cbTotal, &pHdr, &pvDst);
1553 if (RT_SUCCESS(rc))
1554 {
1555 intnetR0SgRead(pSG, pvDst);
1556 if (pNewDstMac)
1557 ((PRTNETETHERHDR)pvDst)->DstMac = *pNewDstMac;
1558
1559 INTNETRingCommitFrame(pRingBuf, pHdr);
1560 return VINF_SUCCESS;
1561 }
1562 return rc;
1563}
1564
1565
1566/**
1567 * Sends a frame to a specific interface.
1568 *
1569 * @param pIf The interface.
1570 * @param pIfSender The interface sending the frame. This is NULL if it's the trunk.
1571 * @param pSG The gather buffer which data is being sent to the interface.
1572 * @param pNewDstMac Set the destination MAC address to the address if specified.
1573 */
1574static void intnetR0IfSend(PINTNETIF pIf, PINTNETIF pIfSender, PINTNETSG pSG, PCRTMAC pNewDstMac)
1575{
1576// LogFlow(("intnetR0IfSend: pIf=%p:{.hIf=%RX32}\n", pIf, pIf->hIf));
1577 int rc = intnetR0RingWriteFrame(&pIf->pIntBuf->Recv, pSG, pNewDstMac);
1578 if (RT_SUCCESS(rc))
1579 {
1580 pIf->cYields = 0;
1581 RTSemEventSignal(pIf->Event);
1582 return;
1583 }
1584
1585 Log(("intnetR0IfSend: overflow cb=%d hIf=%RX32\n", pSG->cbTotal, pIf->hIf));
1586
1587#if 0 /* This is bad stuff now as we're blocking while locking down the network.
1588 we really shouldn't delay the network traffic on the host just because
1589 some bugger isn't responding. Will have to deal with this in a different
1590 manner if required. */
1591 /*
1592 * Retry a few times, yielding the CPU in between.
1593 * But don't let a unresponsive VM harm performance, so give up after a couple of tries.
1594 */
1595 if ( pIf->fActive
1596 && pIf->cYields < 100)
1597 {
1598 unsigned cYields = 10;
1599#else
1600 /*
1601 * Scheduling hack, for unicore machines primarily.
1602 */
1603 if ( pIf->fActive
1604 && pIf->cYields < 4 /* just twice */
1605 && pIfSender /* but not if it's from the trunk */)
1606 {
1607 unsigned cYields = 2;
1608#endif
1609 while (--cYields > 0)
1610 {
1611 RTSemEventSignal(pIf->Event);
1612 RTThreadYield();
1613 rc = intnetR0RingWriteFrame(&pIf->pIntBuf->Recv, pSG, pNewDstMac);
1614 if (RT_SUCCESS(rc))
1615 {
1616 STAM_REL_COUNTER_INC(&pIf->pIntBuf->cStatYieldsOk);
1617 RTSemEventSignal(pIf->Event);
1618 return;
1619 }
1620 pIf->cYields++;
1621 }
1622 STAM_REL_COUNTER_INC(&pIf->pIntBuf->cStatYieldsNok);
1623 }
1624
1625 /* ok, the frame is lost. */
1626 STAM_REL_COUNTER_INC(&pIf->pIntBuf->cStatLost);
1627 RTSemEventSignal(pIf->Event);
1628}
1629
1630
1631/**
1632 * Sends a frame down the trunk.
1633 *
1634 * The caller must own the network mutex, might be abandond temporarily.
1635 * The fTrunkLock parameter indicates whether the trunk lock is held.
1636 *
1637 * @param pThis The trunk.
1638 * @param pNetwork The network the frame is being sent to.
1639 * @param pIfSender The IF sending the frame. Used for MAC address checks in shared MAC mode.
1640 * @param fDst The destination flags.
1641 * @param pSG Pointer to the gather list.
1642 * @param fTrunkLocked Whether the caller owns the out-bound trunk lock.
1643 */
1644static void intnetR0TrunkIfSend(PINTNETTRUNKIF pThis, PINTNETNETWORK pNetwork, PINTNETIF pIfSender,
1645 uint32_t fDst, PINTNETSG pSG, bool fTrunkLocked)
1646{
1647 /*
1648 * Quick sanity check.
1649 */
1650 AssertPtr(pThis);
1651 AssertPtr(pNetwork);
1652 AssertPtr(pSG);
1653 Assert(fDst);
1654 AssertReturnVoid(pThis->pIfPort);
1655
1656 /*
1657 * Edit the frame if we're sharing the MAC address with the host on the wire.
1658 *
1659 * If the frame is headed for both the host and the wire, we'll have to send
1660 * it to the host before making any modifications, and force the OS specific
1661 * backend to copy it. We do this by marking it as TEMP (which is always the
1662 * case right now).
1663 */
1664 if ( (pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
1665 && (fDst & INTNETTRUNKDIR_WIRE))
1666 {
1667 /* Dispatch it to the host before making changes. */
1668 if (fDst & INTNETTRUNKDIR_HOST)
1669 {
1670 Assert(pSG->fFlags & INTNETSG_FLAGS_TEMP); /* make sure copy is forced */
1671 intnetR0TrunkIfSend(pThis, pNetwork, pIfSender, INTNETTRUNKDIR_HOST, pSG, fTrunkLocked);
1672 fDst &= ~INTNETTRUNKDIR_HOST;
1673 }
1674
1675 /* ASSUME frame from INTNETR0IfSend! */
1676 AssertReturnVoid(pSG->cSegsUsed == 1);
1677 AssertReturnVoid(pSG->cbTotal >= sizeof(RTNETETHERHDR));
1678 AssertReturnVoid(fTrunkLocked);
1679 AssertReturnVoid(pIfSender);
1680 PRTNETETHERHDR pEthHdr = (PRTNETETHERHDR)pSG->aSegs[0].pv;
1681
1682 /*
1683 * Get the host mac address and update the ethernet header.
1684 *
1685 * The reason for caching it in the trunk structure is because
1686 * we cannot take the trunk out-bound semaphore when we make
1687 * edits in the intnetR0TrunkIfPortRecv path.
1688 */
1689 pThis->pIfPort->pfnGetMacAddress(pThis->pIfPort, &pThis->CachedMac);
1690 if (!memcmp(&pEthHdr->SrcMac, &pIfSender->Mac, sizeof(RTMAC)))
1691 pEthHdr->SrcMac = pThis->CachedMac;
1692
1693 /*
1694 * Deal with tags from the snooping phase.
1695 */
1696 if (pSG->fFlags & INTNETSG_FLAGS_ARP_IPV4)
1697 {
1698 /*
1699 * APR IPv4: replace hardware (MAC) addresses because these end up
1700 * in ARP caches. So, if we don't the other machiens will
1701 * send the packets to the MAC address of the guest
1702 * instead of the one of the host, which won't work on
1703 * wireless of course...
1704 */
1705 PRTNETARPIPV4 pArp = (PRTNETARPIPV4)(pEthHdr + 1);
1706 if (!memcmp(&pArp->ar_sha, &pIfSender->Mac, sizeof(RTMAC)))
1707 {
1708 Log6(("tw: ar_sha %.6Rhxs -> %.6Rhxs\n", &pArp->ar_sha, &pThis->CachedMac));
1709 pArp->ar_sha = pThis->CachedMac;
1710 }
1711 if (!memcmp(&pArp->ar_tha, &pIfSender->Mac, sizeof(RTMAC))) /* just in case... */
1712 {
1713 Log6(("tw: ar_tha %.6Rhxs -> %.6Rhxs\n", &pArp->ar_tha, &pThis->CachedMac));
1714 pArp->ar_tha = pThis->CachedMac;
1715 }
1716 }
1717 //else if (pSG->fFlags & INTNETSG_FLAGS_ICMPV6_NDP)
1718 //{ /// @todo move the editing into a different function
1719 //}
1720 }
1721
1722 /*
1723 * Temporarily leave the network lock while transmitting the frame.
1724 *
1725 * Note that we're relying on the out-bound lock to serialize threads down
1726 * in INTNETR0IfSend. It's theoretically possible for there to be race now
1727 * because I didn't implement async SG handling yet. Which is why we currently
1728 * require the trunk to be locked, well, one of the reasons.
1729 *
1730 * Another reason is that the intnetR0NetworkSendUnicast code may have to
1731 * call into the trunk interface component to do package switching.
1732 */
1733 AssertReturnVoid(fTrunkLocked); /* to be removed. */
1734
1735 int rc;
1736 if ( fTrunkLocked
1737 || intnetR0TrunkIfRetain(pThis))
1738 {
1739 rc = RTSemFastMutexRelease(pNetwork->FastMutex);
1740 AssertRC(rc);
1741 if (RT_SUCCESS(rc))
1742 {
1743 if ( fTrunkLocked
1744 || intnetR0TrunkIfOutLock(pThis))
1745 {
1746 rc = pThis->pIfPort->pfnXmit(pThis->pIfPort, pSG, fDst);
1747
1748 if (!fTrunkLocked)
1749 intnetR0TrunkIfOutUnlock(pThis);
1750 }
1751 else
1752 {
1753 AssertFailed();
1754 rc = VERR_SEM_DESTROYED;
1755 }
1756
1757 int rc2 = RTSemFastMutexRequest(pNetwork->FastMutex);
1758 AssertRC(rc2);
1759 }
1760
1761 if (!fTrunkLocked)
1762 intnetR0TrunkIfRelease(pThis);
1763 }
1764 else
1765 {
1766 AssertFailed();
1767 rc = VERR_SEM_DESTROYED;
1768 }
1769
1770 /** @todo failure statistics? */
1771 Log2(("intnetR0TrunkIfSend: %Rrc fDst=%d\n", rc, fDst));
1772}
1773
1774
1775/**
1776 * Edits an ARP packet arriving from the wire via the trunk connection.
1777 *
1778 * @param pNetwork The network the frame is being sent to.
1779 * @param pSG Pointer to the gather list for the frame.
1780 * The flags and data content may be updated.
1781 * @param pEthHdr Pointer to the ethernet header. This may also be
1782 * updated if it's a unicast...
1783 */
1784static void intnetR0NetworkEditArpFromWire(PINTNETNETWORK pNetwork, PINTNETSG pSG, PRTNETETHERHDR pEthHdr)
1785{
1786 /*
1787 * Check the minimum size and get a linear copy of the thing to work on,
1788 * using the temporary buffer if necessary.
1789 */
1790 if (RT_UNLIKELY(pSG->cbTotal < sizeof(RTNETETHERHDR) + sizeof(RTNETARPIPV4)))
1791 return;
1792 PRTNETARPIPV4 pArpIPv4 = (PRTNETARPIPV4)((uint8_t *)pSG->aSegs[0].pv + sizeof(RTNETETHERHDR));
1793 if ( pSG->cSegsUsed != 1
1794 && pSG->aSegs[0].cb < sizeof(RTNETETHERHDR) + sizeof(RTNETARPIPV4))
1795 {
1796 Log6(("fw: Copying ARP pkt %u\n", sizeof(RTNETARPIPV4)));
1797 if (!intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR), sizeof(RTNETARPIPV4), pNetwork->pbTmp))
1798 return;
1799 pSG->fFlags |= INTNETSG_FLAGS_PKT_CP_IN_TMP;
1800 pArpIPv4 = (PRTNETARPIPV4)pNetwork->pbTmp;
1801 }
1802
1803 /*
1804 * Ignore packets which doesn't interest us or we perceive as malformed.
1805 */
1806 if (RT_UNLIKELY( pArpIPv4->Hdr.ar_hlen != sizeof(RTMAC)
1807 || pArpIPv4->Hdr.ar_plen != sizeof(RTNETADDRIPV4)
1808 || pArpIPv4->Hdr.ar_htype != RT_H2BE_U16(RTNET_ARP_ETHER)
1809 || pArpIPv4->Hdr.ar_ptype != RT_H2BE_U16(RTNET_ETHERTYPE_IPV4)))
1810 return;
1811 uint16_t ar_oper = RT_H2BE_U16(pArpIPv4->Hdr.ar_oper);
1812 if (RT_UNLIKELY( ar_oper != RTNET_ARPOP_REQUEST
1813 && ar_oper != RTNET_ARPOP_REPLY))
1814 {
1815 Log6(("ar_oper=%#x\n", ar_oper));
1816 return;
1817 }
1818
1819 /* Tag it as ARP IPv4. */
1820 pSG->fFlags |= INTNETSG_FLAGS_ARP_IPV4;
1821
1822 /*
1823 * The thing we're interested in here is a reply to a query made by a guest
1824 * since we modified the MAC in the initial request the guest made.
1825 */
1826 if ( ar_oper == RTNET_ARPOP_REPLY
1827 && !memcmp(&pArpIPv4->ar_tha, &pNetwork->pTrunkIF->CachedMac, sizeof(RTMAC)))
1828 {
1829 PINTNETIF pIf = intnetR0NetworkAddrCacheLookupIf(pNetwork, (PCRTNETADDRU)&pArpIPv4->ar_tpa,
1830 kIntNetAddrType_IPv4, sizeof(pArpIPv4->ar_tpa));
1831 if (pIf)
1832 {
1833 Log6(("fw: ar_tha %.6Rhxs -> %.6Rhxs\n", &pArpIPv4->ar_tha, &pIf->Mac));
1834 pArpIPv4->ar_tha = pIf->Mac;
1835 if (!memcmp(&pEthHdr->DstMac, &pNetwork->pTrunkIF->CachedMac, sizeof(RTMAC)))
1836 {
1837 Log6(("fw: DstMac %.6Rhxs -> %.6Rhxs\n", &pEthHdr->DstMac, &pIf->Mac));
1838 pEthHdr->DstMac = pIf->Mac;
1839 if ((void *)pEthHdr != pSG->aSegs[0].pv)
1840 intnetR0SgWritePart(pSG, RT_OFFSETOF(RTNETETHERHDR, DstMac), sizeof(RTMAC), &pIf->Mac);
1841 }
1842
1843 /* Write back the packet if we've been making changes to a buffered copy. */
1844 if (pSG->fFlags & INTNETSG_FLAGS_PKT_CP_IN_TMP)
1845 intnetR0SgWritePart(pSG, sizeof(RTNETETHERHDR), sizeof(PRTNETARPIPV4), pArpIPv4);
1846 }
1847 }
1848}
1849
1850
1851/**
1852 * Detects and edits an DHCP packet arriving from the internal net.
1853 *
1854 * @param pNetwork The network the frame is being sent to.
1855 * @param pSG Pointer to the gather list for the frame.
1856 * The flags and data content may be updated.
1857 * @param pEthHdr Pointer to the ethernet header. This may also be
1858 * updated if it's a unicast...
1859 */
1860static void intnetR0NetworkEditDhcpFromIntNet(PINTNETNETWORK pNetwork, PINTNETSG pSG, PRTNETETHERHDR pEthHdr)
1861{
1862 /*
1863 * Check the minimum size and get a linear copy of the thing to work on,
1864 * using the temporary buffer if necessary.
1865 */
1866 if (RT_UNLIKELY(pSG->cbTotal < sizeof(RTNETETHERHDR) + RTNETIPV4_MIN_LEN + RTNETUDP_MIN_LEN + RTNETBOOTP_DHCP_MIN_LEN))
1867 return;
1868 /*
1869 * Get a pointer to a linear copy of the full packet, using the
1870 * temporary buffer if necessary.
1871 */
1872 PCRTNETIPV4 pIpHdr = (PCRTNETIPV4)((PCRTNETETHERHDR)pSG->aSegs[0].pv + 1);
1873 uint32_t cbPacket = pSG->cbTotal - sizeof(RTNETETHERHDR);
1874 if (pSG->cSegsUsed > 1)
1875 {
1876 cbPacket = RT_MIN(cbPacket, INTNETNETWORK_TMP_SIZE);
1877 Log6(("intnetR0NetworkEditDhcpFromIntNet: Copying IPv4/UDP/DHCP pkt %u\n", cbPacket));
1878 if (!intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR), cbPacket, pNetwork->pbTmp))
1879 return;
1880 //pSG->fFlags |= INTNETSG_FLAGS_PKT_CP_IN_TMP;
1881 pIpHdr = (PCRTNETIPV4)pNetwork->pbTmp;
1882 }
1883
1884 /*
1885 * Validate the IP header and find the UDP packet.
1886 */
1887 if (!RTNetIPv4IsHdrValid(pIpHdr, cbPacket, pSG->cbTotal - sizeof(RTNETETHERHDR)))
1888 {
1889 Log6(("intnetR0NetworkEditDhcpFromIntNet: bad ip header\n"));
1890 return;
1891 }
1892 size_t cbIpHdr = pIpHdr->ip_hl * 4;
1893 if ( pIpHdr->ip_p != RTNETIPV4_PROT_UDP /* DHCP is UDP. */
1894 || cbPacket < cbIpHdr + RTNETUDP_MIN_LEN + RTNETBOOTP_DHCP_MIN_LEN) /* Min DHCP packet len */
1895 return;
1896
1897 size_t cbUdpPkt = cbPacket - cbIpHdr;
1898 PCRTNETUDP pUdpHdr = (PCRTNETUDP)((uintptr_t)pIpHdr + cbIpHdr);
1899 /* We are only interested in DHCP packets coming from client to server. */
1900 if ( RT_BE2H_U16(pUdpHdr->uh_dport) != RTNETIPV4_PORT_BOOTPS
1901 || RT_BE2H_U16(pUdpHdr->uh_sport) != RTNETIPV4_PORT_BOOTPC)
1902 return;
1903
1904 /*
1905 * Check if the DHCP message is valid and get the type.
1906 */
1907 if (!RTNetIPv4IsUDPValid(pIpHdr, pUdpHdr, pUdpHdr + 1, cbUdpPkt))
1908 {
1909 Log6(("intnetR0NetworkEditDhcpFromIntNet: Bad UDP packet\n"));
1910 return;
1911 }
1912 PCRTNETBOOTP pDhcp = (PCRTNETBOOTP)(pUdpHdr + 1);
1913 uint8_t MsgType;
1914 if (!RTNetIPv4IsDHCPValid(pUdpHdr, pDhcp, cbUdpPkt - sizeof(*pUdpHdr), &MsgType))
1915 {
1916 Log6(("intnetR0NetworkEditDhcpFromIntNet: Bad DHCP packet\n"));
1917 return;
1918 }
1919
1920 switch (MsgType)
1921 {
1922 case RTNET_DHCP_MT_DISCOVER:
1923 case RTNET_DHCP_MT_REQUEST:
1924 Log6(("intnetR0NetworkEditDhcpFromIntNet: Setting broadcast flag in DHCP %#x, previously %x\n", MsgType, pDhcp->bp_flags));
1925 if (!(pDhcp->bp_flags & RT_H2BE_U16_C(RTNET_DHCP_FLAG_BROADCAST)))
1926 {
1927 /* Patch flags */
1928 uint16_t uFlags = pDhcp->bp_flags | RT_H2BE_U16_C(RTNET_DHCP_FLAG_BROADCAST);
1929 intnetR0SgWritePart(pSG, (uintptr_t)&pDhcp->bp_flags - (uintptr_t)pIpHdr + sizeof(RTNETETHERHDR), sizeof(uFlags), &uFlags);
1930 /* Patch UDP checksum */
1931 uint32_t uChecksum = (uint32_t)~pUdpHdr->uh_sum + RT_H2BE_U16_C(RTNET_DHCP_FLAG_BROADCAST);
1932 while (uChecksum >> 16)
1933 uChecksum = (uChecksum >> 16) + (uChecksum & 0xFFFF);
1934 uChecksum = ~uChecksum;
1935 intnetR0SgWritePart(pSG, (uintptr_t)&pUdpHdr->uh_sum - (uintptr_t)pIpHdr + sizeof(RTNETETHERHDR), sizeof(pUdpHdr->uh_sum), &uChecksum);
1936 }
1937 break;
1938 }
1939}
1940
1941
1942/**
1943 * Sends a broadcast frame.
1944 *
1945 * The caller must own the network mutex, might be abandond temporarily.
1946 * When pIfSender is not NULL, the caller must also own the trunk semaphore.
1947 *
1948 * @returns true if it's addressed to someone on the network, otherwise false.
1949 * @param pNetwork The network the frame is being sent to.
1950 * @param pIfSender The interface sending the frame. This is NULL if it's the trunk.
1951 * @param fSrc The source flags. This 0 if it's not from the trunk.
1952 * @param pSG Pointer to the gather list.
1953 * @param fTrunkLocked Whether the caller owns the out-bound trunk lock.
1954 * @param pEthHdr Pointer to the ethernet header.
1955 */
1956static bool intnetR0NetworkSendBroadcast(PINTNETNETWORK pNetwork, PINTNETIF pIfSender, uint32_t fSrc,
1957 PINTNETSG pSG, bool fTrunkLocked, PRTNETETHERHDR pEthHdr)
1958{
1959 /*
1960 * Check for ARP packets from the wire since we'll have to make
1961 * modification to them if we're sharing the MAC address with the host.
1962 */
1963 if ( (pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
1964 && (fSrc & INTNETTRUNKDIR_WIRE)
1965 && RT_BE2H_U16(pEthHdr->EtherType) == RTNET_ETHERTYPE_ARP)
1966 intnetR0NetworkEditArpFromWire(pNetwork, pSG, pEthHdr);
1967
1968 /*
1969 * Check for DHCP packets from the internal net since we'll have to set
1970 * broadcast flag in DHCP requests if we're sharing the MAC address with
1971 * the host.
1972 */
1973 if ( (pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
1974 && !fSrc
1975 && RT_BE2H_U16(pEthHdr->EtherType) == RTNET_ETHERTYPE_IPV4)
1976 intnetR0NetworkEditDhcpFromIntNet(pNetwork, pSG, pEthHdr);
1977
1978 /*
1979 * This is a broadcast or multicast address. For the present we treat those
1980 * two as the same - investigating multicast is left for later.
1981 *
1982 * Write the packet to all the interfaces and signal them.
1983 */
1984 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
1985 if (pIf != pIfSender)
1986 intnetR0IfSend(pIf, pIfSender, pSG, NULL);
1987
1988 /*
1989 * Unless the trunk is the origin, broadcast it to both the wire
1990 * and the host as well.
1991 */
1992 PINTNETTRUNKIF pTrunkIf = pNetwork->pTrunkIF;
1993 if ( pIfSender
1994 && pTrunkIf)
1995 intnetR0TrunkIfSend(pTrunkIf, pNetwork, pIfSender, INTNETTRUNKDIR_HOST | INTNETTRUNKDIR_WIRE, pSG, fTrunkLocked);
1996
1997 /*
1998 * Snoop address info from packet orginating from the trunk connection.
1999 */
2000 else if ( (pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
2001 && !pIfSender)
2002 {
2003#ifdef INTNET_WITH_DHCP_SNOOPING
2004 uint16_t EtherType = RT_BE2H_U16(pEthHdr->EtherType);
2005 if ( ( EtherType == RTNET_ETHERTYPE_IPV4 /* for DHCP */
2006 && pSG->cbTotal >= sizeof(RTNETETHERHDR) + RTNETIPV4_MIN_LEN + RTNETUDP_MIN_LEN + RTNETBOOTP_DHCP_MIN_LEN)
2007 || (pSG->fFlags & (INTNETSG_FLAGS_ARP_IPV4)) )
2008 intnetR0TrunkIfSnoopAddr(pNetwork, pSG, EtherType);
2009#else
2010 if (pSG->fFlags & (INTNETSG_FLAGS_ARP_IPV4))
2011 intnetR0TrunkIfSnoopArp(pNetwork, pSG);
2012#endif
2013 }
2014
2015 return false; /* broadcast frames are never dropped */
2016}
2017
2018
2019/**
2020 * Sends a multicast frame.
2021 *
2022 * The caller must own the network mutex, might be abandond temporarily.
2023 *
2024 * @returns true if it's addressed to someone on the network, otherwise false.
2025 * @param pNetwork The network the frame is being sent to.
2026 * @param pIfSender The interface sending the frame. This is NULL if it's the trunk.
2027 * @param fSrc The source flags. This 0 if it's not from the trunk.
2028 * @param pSG Pointer to the gather list.
2029 * @param fTrunkLocked Whether the caller owns the out-bound trunk lock.
2030 * @param pEthHdr Pointer to the ethernet header.
2031 */
2032static bool intnetR0NetworkSendMulticast(PINTNETNETWORK pNetwork, PINTNETIF pIfSender, uint32_t fSrc, PINTNETSG pSG, bool fTrunkLocked, PRTNETETHERHDR pEthHdr)
2033{
2034 /** @todo implement multicast */
2035 return intnetR0NetworkSendBroadcast(pNetwork, pIfSender, fSrc, pSG, fTrunkLocked, pEthHdr);
2036}
2037
2038
2039/**
2040 * Sends a unicast frame using the network layer address instead
2041 * of the link layer one.
2042 *
2043 * The caller must own the network mutex, might be abandond temporarily.
2044 *
2045 * @returns true if it's addressed to someone on the network, otherwise false.
2046 * @param pNetwork The network the frame is being sent to.
2047 * @param pSG Pointer to the gather list.
2048 * @param fTrunkLocked Whether the caller owns the out-bound trunk lock.
2049 * @param pEthHdr Pointer to the ethernet header.
2050 */
2051static bool intnetR0NetworkSendUnicastWithSharedMac(PINTNETNETWORK pNetwork, PINTNETSG pSG, bool fTrunkLocked, PRTNETETHERHDR pEthHdr)
2052{
2053 /*
2054 * Extract the network address from the packet.
2055 */
2056 RTNETADDRU Addr;
2057 INTNETADDRTYPE enmAddrType;
2058 uint8_t cbAddr;
2059 switch (RT_BE2H_U16(pEthHdr->EtherType))
2060 {
2061 case RTNET_ETHERTYPE_IPV4:
2062 if (RT_UNLIKELY(!intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR) + RT_OFFSETOF(RTNETIPV4, ip_dst), sizeof(Addr.IPv4), &Addr)))
2063 {
2064 Log(("intnetshareduni: failed to read ip_dst! cbTotal=%#x\n", pSG->cbTotal));
2065 return false;
2066 }
2067 enmAddrType = kIntNetAddrType_IPv4;
2068 cbAddr = sizeof(Addr.IPv4);
2069 Log6(("intnetshareduni: IPv4 %d.%d.%d.%d\n", Addr.au8[0], Addr.au8[1], Addr.au8[2], Addr.au8[3]));
2070 break;
2071
2072#if 0 /** @todo IntNet: implement IPv6 for wireless MAC sharing. */
2073 case RTNET_ETHERTYPE_IPV6
2074 if (RT_UNLIKELY(!intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR) + RT_OFFSETOF(RTNETIPV6, ip6_dst), sizeof(Addr.IPv6), &Addr)))
2075 {
2076 Log(("intnetshareduni: failed to read ip6_dst! cbTotal=%#x\n", pSG->cbTotal));
2077 return false;
2078 }
2079 enmAddrType = kIntNetAddrType_IPv6;
2080 cbAddr = sizeof(Addr.IPv6);
2081 break;
2082#endif
2083#if 0 /** @todo IntNet: implement IPX for wireless MAC sharing? */
2084 case RTNET_ETHERTYPE_IPX_1:
2085 case RTNET_ETHERTYPE_IPX_2:
2086 case RTNET_ETHERTYPE_IPX_3:
2087 if (RT_UNLIKELY(!intnetR0SgReadPart(pSG, sizeof(RTNETETHERHDR) + RT_OFFSETOF(RTNETIPX, ipx_dstnet), sizeof(Addr.IPX), &Addr)))
2088 {
2089 Log(("intnetshareduni: failed to read ipx_dstnet! cbTotal=%#x\n", pSG->cbTotal));
2090 return false;
2091 }
2092 enmAddrType = kIntNetAddrType_IPX;
2093 cbAddr = sizeof(Addr.IPX);
2094 break;
2095#endif
2096
2097 /*
2098 * Treat ARP is broadcast (it shouldn't end up here normally,
2099 * so it goes last in the switch).
2100 */
2101 case RTNET_ETHERTYPE_ARP:
2102 Log6(("intnetshareduni: ARP\n"));
2103 /** @todo revisit this broadcasting of unicast ARP frames! */
2104 return intnetR0NetworkSendBroadcast(pNetwork, NULL, INTNETTRUNKDIR_WIRE, pSG, fTrunkLocked, pEthHdr);
2105
2106 /*
2107 * Unknown packets are sent do all interfaces that are in promiscuous mode.
2108 */
2109 default:
2110 {
2111 Log6(("intnetshareduni: unknown ethertype=%#x\n", RT_BE2H_U16(pEthHdr->EtherType)));
2112 if (!(pNetwork->fFlags & (INTNET_OPEN_FLAGS_IGNORE_PROMISC | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC)))
2113 {
2114 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
2115 if (pIf->fPromiscuous)
2116 {
2117 Log2(("Dst=%.6Rhxs => %.6Rhxs\n", &pEthHdr->DstMac, &pIf->Mac));
2118 intnetR0IfSend(pIf, NULL, pSG, NULL);
2119 }
2120 }
2121 return false;
2122 }
2123 }
2124
2125 /*
2126 * Send it to interfaces with matching network addresses.
2127 */
2128 bool fExactIntNetRecipient = false;
2129 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
2130 {
2131 bool fIt = intnetR0IfAddrCacheLookup(&pIf->aAddrCache[enmAddrType], &Addr, cbAddr) >= 0;
2132 if ( fIt
2133 || ( pIf->fPromiscuous
2134 && !(pNetwork->fFlags & (INTNET_OPEN_FLAGS_IGNORE_PROMISC | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC))))
2135 {
2136 Log2(("Dst=%.6Rhxs => %.6Rhxs\n", &pEthHdr->DstMac, &pIf->Mac));
2137 fExactIntNetRecipient |= fIt;
2138 intnetR0IfSend(pIf, NULL, pSG, fIt ? &pIf->Mac : NULL);
2139 }
2140 }
2141
2142#ifdef INTNET_WITH_DHCP_SNOOPING
2143 /*
2144 * Perform DHCP snooping.
2145 */
2146 if ( enmAddrType == kIntNetAddrType_IPv4
2147 && pSG->cbTotal >= sizeof(RTNETETHERHDR) + RTNETIPV4_MIN_LEN + RTNETUDP_MIN_LEN + RTNETBOOTP_DHCP_MIN_LEN)
2148 intnetR0TrunkIfSnoopAddr(pNetwork, pSG, RT_BE2H_U16(pEthHdr->EtherType));
2149#endif /* INTNET_WITH_DHCP_SNOOPING */
2150
2151 return fExactIntNetRecipient;
2152}
2153
2154
2155/**
2156 * Sends a unicast frame.
2157 *
2158 * The caller must own the network mutex, might be abandond temporarily.
2159 *
2160 * @returns true if it's addressed to someone on the network, otherwise false.
2161 * @param pNetwork The network the frame is being sent to.
2162 * @param pIfSender The interface sending the frame. This is NULL if it's the trunk.
2163 * @param fSrc The source flags. This 0 if it's not from the trunk.
2164 * @param pSG Pointer to the gather list.
2165 * @param fTrunkLocked Whether the caller owns the out-bound trunk lock.
2166 * @param pEthHdr Pointer to the ethernet header.
2167 */
2168static bool intnetR0NetworkSendUnicast(PINTNETNETWORK pNetwork, PINTNETIF pIfSender, uint32_t fSrc, PINTNETSG pSG, bool fTrunkLocked, PCRTNETETHERHDR pEthHdr)
2169{
2170 /*
2171 * Only send to the interfaces with matching a MAC address.
2172 */
2173 bool fExactIntNetRecipient = false;
2174 for (PINTNETIF pIf = pNetwork->pIFs; pIf; pIf = pIf->pNext)
2175 {
2176 bool fIt = false;
2177 if ( ( !pIf->fMacSet
2178 || (fIt = !memcmp(&pIf->Mac, &pEthHdr->DstMac, sizeof(pIf->Mac))) )
2179 || ( pIf->fPromiscuous
2180 && !(pNetwork->fFlags & (INTNET_OPEN_FLAGS_IGNORE_PROMISC | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC))
2181 && pIf != pIfSender /* promiscuous mode: omit the sender */))
2182 {
2183 Log2(("Dst=%.6Rhxs => %.6Rhxs\n", &pEthHdr->DstMac, &pIf->Mac));
2184 fExactIntNetRecipient |= fIt;
2185 intnetR0IfSend(pIf, pIfSender, pSG, NULL);
2186 }
2187 }
2188
2189 /*
2190 * Send it to the trunk?
2191 * If we didn't find the recipient on the internal network the
2192 * frame will hit the wire.
2193 */
2194 uint32_t fDst = 0;
2195 PINTNETTRUNKIF pTrunkIf = pNetwork->pTrunkIF;
2196 if ( pIfSender
2197 && pTrunkIf
2198 && pTrunkIf->pIfPort)
2199 {
2200 Assert(!fSrc);
2201
2202 /* promiscuous checks first as they are cheaper than pfnIsHostMac. */
2203 if ( pTrunkIf->fPromiscuousWire
2204 && !(pNetwork->fFlags & (INTNET_OPEN_FLAGS_IGNORE_PROMISC | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC | INTNET_OPEN_FLAGS_IGNORE_PROMISC_TRUNK_WIRE | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC_TRUNK_WIRE)) )
2205 fDst |= INTNETTRUNKDIR_WIRE;
2206 if ( !(pNetwork->fFlags & (INTNET_OPEN_FLAGS_IGNORE_PROMISC | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC | INTNET_OPEN_FLAGS_IGNORE_PROMISC_TRUNK_HOST | INTNET_OPEN_FLAGS_QUIETLY_IGNORE_PROMISC_TRUNK_HOST))
2207 && pTrunkIf->pIfPort->pfnIsPromiscuous(pTrunkIf->pIfPort) )
2208 fDst |= INTNETTRUNKDIR_HOST;
2209
2210 if ( fDst != (INTNETTRUNKDIR_HOST | INTNETTRUNKDIR_WIRE)
2211 && !fExactIntNetRecipient /* if you have duplicate mac addresses, you're screwed. */ )
2212 {
2213 if (pTrunkIf->pIfPort->pfnIsHostMac(pTrunkIf->pIfPort, &pEthHdr->DstMac))
2214 fDst |= INTNETTRUNKDIR_HOST;
2215 else
2216 fDst |= INTNETTRUNKDIR_WIRE;
2217 }
2218
2219 if (fDst)
2220 intnetR0TrunkIfSend(pTrunkIf, pNetwork, pIfSender, fDst, pSG, fTrunkLocked);
2221 }
2222
2223 /* log it */
2224 if ( !fExactIntNetRecipient
2225 && !fDst
2226 && ( (pEthHdr->DstMac.au8[0] == 0x08 && pEthHdr->DstMac.au8[1] == 0x00 && pEthHdr->DstMac.au8[2] == 0x27)
2227 || (pEthHdr->SrcMac.au8[0] == 0x08 && pEthHdr->SrcMac.au8[1] == 0x00 && pEthHdr->SrcMac.au8[2] == 0x27)))
2228 Log2(("Dst=%.6Rhxs ??\n", &pEthHdr->DstMac));
2229
2230 return fExactIntNetRecipient;
2231}
2232
2233
2234/**
2235 * Sends a frame.
2236 *
2237 * This function will distribute the frame to the interfaces it is addressed to.
2238 * It will also update the MAC address of the sender.
2239 *
2240 * The caller must own the network mutex.
2241 *
2242 * @returns true if it's addressed to someone on the network, otherwise false.
2243 * @param pNetwork The network the frame is being sent to.
2244 * @param pIfSender The interface sending the frame. This is NULL if it's the trunk.
2245 * @param fSrc The source flags. This 0 if it's not from the trunk.
2246 * @param pSG Pointer to the gather list.
2247 * @param fTrunkLocked Whether the caller owns the out-bound trunk lock.
2248 */
2249static bool intnetR0NetworkSend(PINTNETNETWORK pNetwork, PINTNETIF pIfSender, uint32_t fSrc, PINTNETSG pSG, bool fTrunkLocked)
2250{
2251 bool fRc = false;
2252
2253 /*
2254 * Assert reality.
2255 */
2256 AssertPtr(pNetwork);
2257 AssertPtrNull(pIfSender);
2258 Assert(pIfSender ? fSrc == 0 : fSrc != 0);
2259 Assert(!pIfSender || pNetwork == pIfSender->pNetwork);
2260 AssertPtr(pSG);
2261 Assert(pSG->cSegsUsed >= 1);
2262 Assert(pSG->cSegsUsed <= pSG->cSegsAlloc);
2263 if (pSG->cbTotal < sizeof(RTNETETHERHDR))
2264 return fRc;
2265
2266 /*
2267 * Get the ethernet header (might theoretically involve multiple segments).
2268 */
2269 RTNETETHERHDR EthHdr;
2270 if (pSG->aSegs[0].cb >= sizeof(EthHdr))
2271 EthHdr = *(PCRTNETETHERHDR)pSG->aSegs[0].pv;
2272 else if (!intnetR0SgReadPart(pSG, 0, sizeof(EthHdr), &EthHdr))
2273 return false;
2274 if ( (EthHdr.DstMac.au8[0] == 0x08 && EthHdr.DstMac.au8[1] == 0x00 && EthHdr.DstMac.au8[2] == 0x27)
2275 || (EthHdr.SrcMac.au8[0] == 0x08 && EthHdr.SrcMac.au8[1] == 0x00 && EthHdr.SrcMac.au8[2] == 0x27)
2276 || (EthHdr.DstMac.au8[0] == 0x00 && EthHdr.DstMac.au8[1] == 0x16 && EthHdr.DstMac.au8[2] == 0xcb)
2277 || (EthHdr.SrcMac.au8[0] == 0x00 && EthHdr.SrcMac.au8[1] == 0x16 && EthHdr.SrcMac.au8[2] == 0xcb)
2278 || EthHdr.DstMac.au8[0] == 0xff
2279 || EthHdr.SrcMac.au8[0] == 0xff)
2280 Log2(("D=%.6Rhxs S=%.6Rhxs T=%04x f=%x z=%x\n",
2281 &EthHdr.DstMac, &EthHdr.SrcMac, RT_BE2H_U16(EthHdr.EtherType), fSrc, pSG->cbTotal));
2282
2283 /*
2284 * Inspect the header updating the mac address of the sender in the process.
2285 */
2286 if ( pIfSender
2287 && memcmp(&EthHdr.SrcMac, &pIfSender->Mac, sizeof(pIfSender->Mac)))
2288 {
2289 /** @todo stats */
2290 Log2(("IF MAC: %.6Rhxs -> %.6Rhxs\n", &pIfSender->Mac, &EthHdr.SrcMac));
2291 pIfSender->Mac = EthHdr.SrcMac;
2292 pIfSender->fMacSet = true;
2293 }
2294
2295 /*
2296 * Distribute the frame.
2297 */
2298 if ( EthHdr.DstMac.au16[0] == 0xffff /* broadcast address. */
2299 && EthHdr.DstMac.au16[1] == 0xffff
2300 && EthHdr.DstMac.au16[2] == 0xffff)
2301 fRc = intnetR0NetworkSendBroadcast(pNetwork, pIfSender, fSrc, pSG, fTrunkLocked, &EthHdr);
2302 else if (RT_UNLIKELY(EthHdr.DstMac.au8[0] & 1)) /* multicast address */
2303 fRc = intnetR0NetworkSendMulticast(pNetwork, pIfSender, fSrc, pSG, fTrunkLocked, &EthHdr);
2304 else if ( !(pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
2305 || !(fSrc & INTNETTRUNKDIR_WIRE))
2306 fRc = intnetR0NetworkSendUnicast(pNetwork, pIfSender, fSrc, pSG, fTrunkLocked, &EthHdr);
2307 else
2308 fRc = intnetR0NetworkSendUnicastWithSharedMac(pNetwork, pSG, fTrunkLocked, &EthHdr);
2309 return fRc;
2310}
2311
2312
2313/**
2314 * Sends one or more frames.
2315 *
2316 * The function will first the frame which is passed as the optional
2317 * arguments pvFrame and cbFrame. These are optional since it also
2318 * possible to chain together one or more frames in the send buffer
2319 * which the function will process after considering it's arguments.
2320 *
2321 * @returns VBox status code.
2322 * @param pIntNet The instance data.
2323 * @param hIf The interface handle.
2324 * @param pSession The caller's session.
2325 * @param pvFrame Pointer to the frame. Optional, please don't use.
2326 * @param cbFrame Size of the frame. Optional, please don't use.
2327 */
2328INTNETR0DECL(int) INTNETR0IfSend(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, const void *pvFrame, unsigned cbFrame)
2329{
2330 Log5(("INTNETR0IfSend: pIntNet=%p hIf=%RX32 pvFrame=%p cbFrame=%u\n", pIntNet, hIf, pvFrame, cbFrame));
2331
2332 /*
2333 * Validate input and translate the handle.
2334 */
2335 AssertReturn(pIntNet, VERR_INVALID_PARAMETER);
2336 if (pvFrame && cbFrame)
2337 {
2338 AssertReturn(cbFrame < 0x8000, VERR_INVALID_PARAMETER);
2339 AssertPtrReturn(pvFrame, VERR_INVALID_PARAMETER);
2340 AssertPtrReturn((uint8_t *)pvFrame + cbFrame - 1, VERR_INVALID_PARAMETER);
2341
2342 /* This is the better place to crash, probe the buffer. */
2343 ASMProbeReadBuffer(pvFrame, cbFrame);
2344 }
2345 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2346 if (!pIf)
2347 return VERR_INVALID_HANDLE;
2348
2349 /*
2350 * Lock the network. If there is a trunk retain it and grab its
2351 * out-bound lock (this requires leaving the network lock first).
2352 * Grabbing the out-bound lock here simplifies things quite a bit
2353 * later on, so while this is excessive and a bit expensive it's
2354 * not worth caring about right now.
2355 */
2356 PINTNETNETWORK pNetwork = pIf->pNetwork;
2357 int rc = RTSemFastMutexRequest(pNetwork->FastMutex);
2358 if (RT_FAILURE(rc))
2359 {
2360 intnetR0IfRelease(pIf, pSession);
2361 return rc;
2362 }
2363 PINTNETTRUNKIF pTrunkIf = intnetR0TrunkIfRetain(pNetwork->pTrunkIF);
2364 if (pTrunkIf)
2365 {
2366 RTSemFastMutexRelease(pIf->pNetwork->FastMutex);
2367
2368 if (!intnetR0TrunkIfOutLock(pTrunkIf))
2369 {
2370 intnetR0TrunkIfRelease(pTrunkIf);
2371 intnetR0IfRelease(pIf, pSession);
2372 return VERR_SEM_DESTROYED;
2373 }
2374
2375 rc = RTSemFastMutexRequest(pNetwork->FastMutex);
2376 if (RT_FAILURE(rc))
2377 {
2378 intnetR0TrunkIfOutUnlock(pTrunkIf);
2379 intnetR0TrunkIfRelease(pTrunkIf);
2380 intnetR0IfRelease(pIf, pSession);
2381 return rc;
2382 }
2383 }
2384
2385 INTNETSG Sg; /** @todo this will have to be changed if we're going to use async sending
2386 * with buffer sharing for some OS or service. Darwin copies everything so
2387 * I won't bother allocating and managing SGs rigth now. Sorry. */
2388
2389 /*
2390 * Process the argument.
2391 */
2392 if (pvFrame && cbFrame)
2393 {
2394 intnetR0SgInitTemp(&Sg, (void *)pvFrame, cbFrame);
2395 if (pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
2396 intnetR0IfSnoopAddr(pIf, (uint8_t *)pvFrame, cbFrame, (uint16_t *)&Sg.fFlags);
2397 intnetR0NetworkSend(pNetwork, pIf, 0, &Sg, !!pTrunkIf);
2398 }
2399
2400 /*
2401 * Process the send buffer.
2402 */
2403 PINTNETHDR pHdr;
2404 while ((pHdr = INTNETRingGetNextFrameToRead(&pIf->pIntBuf->Send)) != NULL)
2405 {
2406 /* Send the frame if the type is sane. */
2407 if (pHdr->u16Type == INTNETHDR_TYPE_FRAME)
2408 {
2409 void *pvCurFrame = INTNETHdrGetFramePtr(pHdr, pIf->pIntBuf);
2410 if (pvCurFrame)
2411 {
2412 intnetR0SgInitTemp(&Sg, pvCurFrame, pHdr->cbFrame);
2413 if (pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
2414 intnetR0IfSnoopAddr(pIf, (uint8_t *)pvCurFrame, pHdr->cbFrame, (uint16_t *)&Sg.fFlags);
2415 intnetR0NetworkSend(pNetwork, pIf, 0, &Sg, !!pTrunkIf);
2416 }
2417 }
2418 else
2419 STAM_REL_COUNTER_INC(&pIf->pIntBuf->cStatBadFrames); /* ignore */
2420
2421 /* Skip to the next frame. */
2422 INTNETRingSkipFrame(&pIf->pIntBuf->Send);
2423 }
2424
2425 /*
2426 * Release the semaphore(s) and release references.
2427 */
2428 rc = RTSemFastMutexRelease(pNetwork->FastMutex);
2429 if (pTrunkIf)
2430 {
2431 intnetR0TrunkIfOutUnlock(pTrunkIf);
2432 intnetR0TrunkIfRelease(pTrunkIf);
2433 }
2434
2435 intnetR0IfRelease(pIf, pSession);
2436 return rc;
2437}
2438
2439
2440/**
2441 * VMMR0 request wrapper for INTNETR0IfSend.
2442 *
2443 * @returns see INTNETR0IfSend.
2444 * @param pIntNet The internal networking instance.
2445 * @param pSession The caller's session.
2446 * @param pReq The request packet.
2447 */
2448INTNETR0DECL(int) INTNETR0IfSendReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFSENDREQ pReq)
2449{
2450 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
2451 return VERR_INVALID_PARAMETER;
2452 return INTNETR0IfSend(pIntNet, pReq->hIf, pSession, NULL, 0);
2453}
2454
2455
2456/**
2457 * Maps the default buffer into ring 3.
2458 *
2459 * @returns VBox status code.
2460 * @param pIntNet The instance data.
2461 * @param hIf The interface handle.
2462 * @param pSession The caller's session.
2463 * @param ppRing3Buf Where to store the address of the ring-3 mapping.
2464 */
2465INTNETR0DECL(int) INTNETR0IfGetRing3Buffer(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, R3PTRTYPE(PINTNETBUF) *ppRing3Buf)
2466{
2467 LogFlow(("INTNETR0IfGetRing3Buffer: pIntNet=%p hIf=%RX32 ppRing3Buf=%p\n", pIntNet, hIf, ppRing3Buf));
2468
2469 /*
2470 * Validate input.
2471 */
2472 AssertReturn(pIntNet, VERR_INVALID_PARAMETER);
2473 AssertPtrReturn(ppRing3Buf, VERR_INVALID_PARAMETER);
2474 *ppRing3Buf = 0;
2475 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2476 if (!pIf)
2477 return VERR_INVALID_HANDLE;
2478
2479 /*
2480 * ASSUMES that only the process that created an interface can use it.
2481 * ASSUMES that we created the ring-3 mapping when selecting or
2482 * allocating the buffer.
2483 */
2484 int rc = RTSemFastMutexRequest(pIf->pNetwork->FastMutex);
2485 if (RT_SUCCESS(rc))
2486 {
2487 *ppRing3Buf = pIf->pIntBufR3;
2488 rc = RTSemFastMutexRelease(pIf->pNetwork->FastMutex);
2489 }
2490
2491 intnetR0IfRelease(pIf, pSession);
2492 LogFlow(("INTNETR0IfGetRing3Buffer: returns %Rrc *ppRing3Buf=%p\n", rc, *ppRing3Buf));
2493 return rc;
2494}
2495
2496
2497/**
2498 * VMMR0 request wrapper for INTNETR0IfGetRing3Buffer.
2499 *
2500 * @returns see INTNETR0IfGetRing3Buffer.
2501 * @param pIntNet The internal networking instance.
2502 * @param pSession The caller's session.
2503 * @param pReq The request packet.
2504 */
2505INTNETR0DECL(int) INTNETR0IfGetRing3BufferReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFGETRING3BUFFERREQ pReq)
2506{
2507 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
2508 return VERR_INVALID_PARAMETER;
2509 return INTNETR0IfGetRing3Buffer(pIntNet, pReq->hIf, pSession, &pReq->pRing3Buf);
2510}
2511
2512
2513/**
2514 * Gets the ring-0 address of the current buffer.
2515 *
2516 * @returns VBox status code.
2517 * @param pIntNet The instance data.
2518 * @param hIf The interface handle.
2519 * @param pSession The caller's session.
2520 * @param ppRing0Buf Where to store the address of the ring-3 mapping.
2521 */
2522INTNETR0DECL(int) INTNETR0IfGetRing0Buffer(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, PINTNETBUF *ppRing0Buf)
2523{
2524 LogFlow(("INTNETR0IfGetRing0Buffer: pIntNet=%p hIf=%RX32 ppRing0Buf=%p\n", pIntNet, hIf, ppRing0Buf));
2525
2526 /*
2527 * Validate input.
2528 */
2529 AssertPtrReturn(ppRing0Buf, VERR_INVALID_PARAMETER);
2530 *ppRing0Buf = NULL;
2531 AssertPtrReturn(pIntNet, VERR_INVALID_PARAMETER);
2532 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2533 if (!pIf)
2534 return VERR_INVALID_HANDLE;
2535
2536 /*
2537 * Grab the lock and get the data.
2538 * ASSUMES that the handle isn't closed while we're here.
2539 */
2540 int rc = RTSemFastMutexRequest(pIf->pNetwork->FastMutex);
2541 if (RT_SUCCESS(rc))
2542 {
2543 *ppRing0Buf = pIf->pIntBuf;
2544
2545 rc = RTSemFastMutexRelease(pIf->pNetwork->FastMutex);
2546 }
2547 intnetR0IfRelease(pIf, pSession);
2548 LogFlow(("INTNETR0IfGetRing0Buffer: returns %Rrc *ppRing0Buf=%p\n", rc, *ppRing0Buf));
2549 return rc;
2550}
2551
2552
2553#if 0
2554/**
2555 * Gets the physical addresses of the default interface buffer.
2556 *
2557 * @returns VBox status code.
2558 * @param pIntNet The instance data.
2559 * @param hIF The interface handle.
2560 * @param paPages Where to store the addresses. (The reserved fields will be set to zero.)
2561 * @param cPages
2562 */
2563INTNETR0DECL(int) INTNETR0IfGetPhysBuffer(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPPAGE paPages, unsigned cPages)
2564{
2565 /*
2566 * Validate input.
2567 */
2568 AssertReturn(pIntNet, VERR_INVALID_PARAMETER);
2569 AssertPtrReturn(paPages, VERR_INVALID_PARAMETER);
2570 AssertPtrReturn((uint8_t *)&paPages[cPages] - 1, VERR_INVALID_PARAMETER);
2571 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2572 if (!pIf)
2573 return VERR_INVALID_HANDLE;
2574
2575 /*
2576 * Grab the lock and get the data.
2577 * ASSUMES that the handle isn't closed while we're here.
2578 */
2579 int rc = RTSemFastMutexRequest(pIf->pNetwork->FastMutex);
2580 if (RT_SUCCESS(rc))
2581 {
2582 /** @todo make a SUPR0 api for obtaining the array. SUPR0/IPRT is keeping track of everything, there
2583 * is no need for any extra bookkeeping here.. */
2584
2585 rc = RTSemFastMutexRelease(pIf->pNetwork->FastMutex);
2586 }
2587 intnetR0IfRelease(pIf, pSession);
2588 return VERR_NOT_IMPLEMENTED;
2589}
2590#endif
2591
2592
2593/**
2594 * Sets the promiscuous mode property of an interface.
2595 *
2596 * @returns VBox status code.
2597 * @param pIntNet The instance handle.
2598 * @param hIf The interface handle.
2599 * @param pSession The caller's session.
2600 * @param fPromiscuous Set if the interface should be in promiscuous mode, clear if not.
2601 */
2602INTNETR0DECL(int) INTNETR0IfSetPromiscuousMode(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, bool fPromiscuous)
2603{
2604 LogFlow(("INTNETR0IfSetPromiscuousMode: pIntNet=%p hIf=%RX32 fPromiscuous=%d\n", pIntNet, hIf, fPromiscuous));
2605
2606 /*
2607 * Validate & translate input.
2608 */
2609 AssertReturn(pIntNet, VERR_INVALID_PARAMETER);
2610 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2611 if (!pIf)
2612 {
2613 Log(("INTNETR0IfSetPromiscuousMode: returns VERR_INVALID_HANDLE\n"));
2614 return VERR_INVALID_HANDLE;
2615 }
2616
2617 /*
2618 * Grab the network semaphore and make the change.
2619 */
2620 int rc;
2621 PINTNETNETWORK pNetwork = pIf->pNetwork;
2622 if (pNetwork)
2623 {
2624 rc = RTSemFastMutexRequest(pNetwork->FastMutex);
2625 if (RT_SUCCESS(rc))
2626 {
2627 if (pIf->fPromiscuous != fPromiscuous)
2628 {
2629 Log(("INTNETR0IfSetPromiscuousMode: hIf=%RX32: Changed from %d -> %d\n",
2630 hIf, !fPromiscuous, !!fPromiscuous));
2631 ASMAtomicUoWriteBool(&pIf->fPromiscuous, fPromiscuous);
2632 }
2633
2634 rc = RTSemFastMutexRelease(pNetwork->FastMutex);
2635 }
2636 }
2637 else
2638 rc = VERR_WRONG_ORDER;
2639
2640 intnetR0IfRelease(pIf, pSession);
2641 return rc;
2642}
2643
2644
2645/**
2646 * VMMR0 request wrapper for INTNETR0IfSetPromiscuousMode.
2647 *
2648 * @returns see INTNETR0IfSetPromiscuousMode.
2649 * @param pIntNet The internal networking instance.
2650 * @param pSession The caller's session.
2651 * @param pReq The request packet.
2652 */
2653INTNETR0DECL(int) INTNETR0IfSetPromiscuousModeReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFSETPROMISCUOUSMODEREQ pReq)
2654{
2655 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
2656 return VERR_INVALID_PARAMETER;
2657 return INTNETR0IfSetPromiscuousMode(pIntNet, pReq->hIf, pSession, pReq->fPromiscuous);
2658}
2659
2660
2661/**
2662 * Sets the MAC address of an interface.
2663 *
2664 * @returns VBox status code.
2665 * @param pIntNet The instance handle.
2666 * @param hIf The interface handle.
2667 * @param pSession The caller's session.
2668 * @param pMAC The new MAC address.
2669 */
2670INTNETR0DECL(int) INTNETR0IfSetMacAddress(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, PCRTMAC pMac)
2671{
2672 LogFlow(("INTNETR0IfSetMacAddress: pIntNet=%p hIf=%RX32 pMac=%p:{%.6Rhxs}\n", pIntNet, hIf, pMac, pMac));
2673
2674 /*
2675 * Validate & translate input.
2676 */
2677 AssertPtrReturn(pIntNet, VERR_INVALID_PARAMETER);
2678 AssertPtrReturn(pMac, VERR_INVALID_PARAMETER);
2679 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2680 if (!pIf)
2681 {
2682 Log(("INTNETR0IfSetMacAddress: returns VERR_INVALID_HANDLE\n"));
2683 return VERR_INVALID_HANDLE;
2684 }
2685
2686 /*
2687 * Grab the network semaphore and make the change.
2688 */
2689 int rc;
2690 PINTNETNETWORK pNetwork = pIf->pNetwork;
2691 if (pNetwork)
2692 {
2693 rc = RTSemFastMutexRequest(pNetwork->FastMutex);
2694 if (RT_SUCCESS(rc))
2695 {
2696 if (memcmp(&pIf->Mac, pMac, sizeof(pIf->Mac)))
2697 {
2698 Log(("INTNETR0IfSetMacAddress: hIf=%RX32: Changed from %.6Rhxs -> %.6Rhxs\n",
2699 hIf, &pIf->Mac, pMac));
2700 pIf->Mac = *pMac;
2701 pIf->fMacSet = true;
2702 }
2703
2704 rc = RTSemFastMutexRelease(pNetwork->FastMutex);
2705 }
2706 }
2707 else
2708 rc = VERR_WRONG_ORDER;
2709
2710 intnetR0IfRelease(pIf, pSession);
2711 return rc;
2712}
2713
2714
2715/**
2716 * VMMR0 request wrapper for INTNETR0IfSetMacAddress.
2717 *
2718 * @returns see INTNETR0IfSetMacAddress.
2719 * @param pIntNet The internal networking instance.
2720 * @param pSession The caller's session.
2721 * @param pReq The request packet.
2722 */
2723INTNETR0DECL(int) INTNETR0IfSetMacAddressReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFSETMACADDRESSREQ pReq)
2724{
2725 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
2726 return VERR_INVALID_PARAMETER;
2727 return INTNETR0IfSetMacAddress(pIntNet, pReq->hIf, pSession, &pReq->Mac);
2728}
2729
2730
2731/**
2732 * Worker for intnetR0IfSetActive.
2733 *
2734 * This function will update the active interface count on the network and
2735 * activate or deactivate the trunk connection if necessary. Note that in
2736 * order to do this it is necessary to abandond the network semaphore.
2737 *
2738 * @returns VBox status code.
2739 * @param pNetwork The network.
2740 * @param fIf The interface.
2741 * @param fActive What to do.
2742 */
2743static int intnetR0NetworkSetIfActive(PINTNETNETWORK pNetwork, PINTNETIF pIf, bool fActive)
2744{
2745 /* quick santiy check */
2746 AssertPtr(pNetwork);
2747 AssertPtr(pIf);
2748
2749 /*
2750 * If we've got a trunk, lock it now in case we need to call out, and
2751 * then lock the network.
2752 */
2753 PINTNETTRUNKIF pTrunkIf = pNetwork->pTrunkIF;
2754 if (pTrunkIf && !intnetR0TrunkIfOutLock(pTrunkIf))
2755 return VERR_SEM_DESTROYED;
2756
2757 int rc = RTSemFastMutexRequest(pNetwork->FastMutex); AssertRC(rc);
2758 if (RT_SUCCESS(rc))
2759 {
2760 bool fNetworkLocked = true;
2761
2762 /*
2763 * Make the change if necessary.
2764 */
2765 if (pIf->fActive != fActive)
2766 {
2767 pIf->fActive = fActive;
2768
2769 uint32_t const cActiveIFs = pNetwork->cActiveIFs;
2770 Assert((int32_t)cActiveIFs + (fActive ? 1 : -1) >= 0);
2771 pNetwork->cActiveIFs += fActive ? 1 : -1;
2772
2773 if ( pTrunkIf
2774 && ( !pNetwork->cActiveIFs
2775 || !cActiveIFs))
2776 {
2777 /*
2778 * We'll have to change the trunk status, so, leave
2779 * the network semaphore so we don't create any deadlocks.
2780 */
2781 int rc2 = RTSemFastMutexRelease(pNetwork->FastMutex); AssertRC(rc2);
2782 fNetworkLocked = false;
2783
2784 if (pTrunkIf->pIfPort)
2785 pTrunkIf->pIfPort->pfnSetActive(pTrunkIf->pIfPort, fActive);
2786 }
2787 }
2788
2789 if (fNetworkLocked)
2790 RTSemFastMutexRelease(pNetwork->FastMutex);
2791 }
2792 if (pTrunkIf)
2793 intnetR0TrunkIfOutUnlock(pTrunkIf);
2794 return rc;
2795}
2796
2797
2798/**
2799 * Activates or deactivates a interface.
2800 *
2801 * This is used to enable and disable the trunk connection on demans as well as
2802 * know when not to expect an interface to want to receive packets.
2803 *
2804 * @returns VBox status code.
2805 * @param pIf The interface.
2806 * @param fActive What to do.
2807 */
2808static int intnetR0IfSetActive(PINTNETIF pIf, bool fActive)
2809{
2810 /* quick sanity check */
2811 AssertPtrReturn(pIf, VERR_INVALID_POINTER);
2812
2813 /*
2814 * Hand it to the network since it might involve the trunk
2815 * and things are tricky there wrt to locking order.
2816 */
2817 PINTNETNETWORK pNetwork = pIf->pNetwork;
2818 if (!pNetwork)
2819 return VERR_WRONG_ORDER;
2820 return intnetR0NetworkSetIfActive(pNetwork, pIf, fActive);
2821}
2822
2823
2824/**
2825 * Sets the active property of an interface.
2826 *
2827 * @returns VBox status code.
2828 * @param pIntNet The instance handle.
2829 * @param hIf The interface handle.
2830 * @param pSession The caller's session.
2831 * @param fActive The new state.
2832 */
2833INTNETR0DECL(int) INTNETR0IfSetActive(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, bool fActive)
2834{
2835 LogFlow(("INTNETR0IfSetActive: pIntNet=%p hIf=%RX32 fActive=%RTbool\n", pIntNet, hIf, fActive));
2836
2837 /*
2838 * Validate & translate input.
2839 */
2840 AssertPtrReturn(pIntNet, VERR_INVALID_PARAMETER);
2841 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2842 if (!pIf)
2843 {
2844 Log(("INTNETR0IfSetActive: returns VERR_INVALID_HANDLE\n"));
2845 return VERR_INVALID_HANDLE;
2846 }
2847
2848 /*
2849 * Hand it to the network since it might involve the trunk
2850 * and things are tricky there wrt to locking order.
2851 */
2852 int rc;
2853 PINTNETNETWORK pNetwork = pIf->pNetwork;
2854 if (pNetwork)
2855 rc = intnetR0NetworkSetIfActive(pNetwork, pIf, fActive);
2856 else
2857 rc = VERR_WRONG_ORDER;
2858
2859 intnetR0IfRelease(pIf, pSession);
2860 return rc;
2861}
2862
2863
2864/**
2865 * VMMR0 request wrapper for INTNETR0IfSetActive.
2866 *
2867 * @returns see INTNETR0IfSetActive.
2868 * @param pIntNet The internal networking instance.
2869 * @param pSession The caller's session.
2870 * @param pReq The request packet.
2871 */
2872INTNETR0DECL(int) INTNETR0IfSetActiveReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFSETACTIVEREQ pReq)
2873{
2874 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
2875 return VERR_INVALID_PARAMETER;
2876 return INTNETR0IfSetActive(pIntNet, pReq->hIf, pSession, pReq->fActive);
2877}
2878
2879
2880/**
2881 * Wait for the interface to get signaled.
2882 * The interface will be signaled when is put into the receive buffer.
2883 *
2884 * @returns VBox status code.
2885 * @param pIntNet The instance handle.
2886 * @param hIf The interface handle.
2887 * @param pSession The caller's session.
2888 * @param cMillies Number of milliseconds to wait. RT_INDEFINITE_WAIT should be
2889 * used if indefinite wait is desired.
2890 */
2891INTNETR0DECL(int) INTNETR0IfWait(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession, uint32_t cMillies)
2892{
2893 Log4(("INTNETR0IfWait: pIntNet=%p hIf=%RX32 cMillies=%u\n", pIntNet, hIf, cMillies));
2894
2895 /*
2896 * Get and validate essential handles.
2897 */
2898 AssertPtrReturn(pIntNet, VERR_INVALID_PARAMETER);
2899 PINTNETIF pIf = (PINTNETIF)RTHandleTableLookupWithCtx(pIntNet->hHtIfs, hIf, pSession);
2900 if (!pIf)
2901 {
2902 Log(("INTNETR0IfWait: returns VERR_INVALID_HANDLE\n"));
2903 return VERR_INVALID_HANDLE;
2904 }
2905 const INTNETIFHANDLE hIfSelf = pIf->hIf;
2906 const RTSEMEVENT Event = pIf->Event;
2907 if ( hIfSelf != hIf /* paranoia */
2908 && Event != NIL_RTSEMEVENT)
2909 {
2910 Log(("INTNETR0IfWait: returns VERR_SEM_DESTROYED\n"));
2911 return VERR_SEM_DESTROYED;
2912 }
2913
2914 /*
2915 * It is tempting to check if there is data to be read here,
2916 * but the problem with such an approach is that it will cause
2917 * one unnecessary supervisor->user->supervisor trip. There is
2918 * already a slight risk for such, so no need to increase it.
2919 */
2920
2921 /*
2922 * Increment the number of waiters before starting the wait.
2923 * Upon wakeup we must assert reality, checking that we're not
2924 * already destroyed or in the process of being destroyed. This
2925 * code must be aligned with the waiting code in intnetR0IfDestruct.
2926 */
2927 ASMAtomicIncU32(&pIf->cSleepers);
2928 int rc = RTSemEventWaitNoResume(Event, cMillies);
2929 if (pIf->Event == Event)
2930 {
2931 ASMAtomicDecU32(&pIf->cSleepers);
2932 if (!pIf->fDestroying)
2933 {
2934 if (intnetR0IfRelease(pIf, pSession))
2935 rc = VERR_SEM_DESTROYED;
2936 }
2937 else
2938 rc = VERR_SEM_DESTROYED;
2939 }
2940 else
2941 rc = VERR_SEM_DESTROYED;
2942 Log4(("INTNETR0IfWait: returns %Rrc\n", rc));
2943 return rc;
2944}
2945
2946
2947/**
2948 * VMMR0 request wrapper for INTNETR0IfWait.
2949 *
2950 * @returns see INTNETR0IfWait.
2951 * @param pIntNet The internal networking instance.
2952 * @param pSession The caller's session.
2953 * @param pReq The request packet.
2954 */
2955INTNETR0DECL(int) INTNETR0IfWaitReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFWAITREQ pReq)
2956{
2957 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
2958 return VERR_INVALID_PARAMETER;
2959 return INTNETR0IfWait(pIntNet, pReq->hIf, pSession, pReq->cMillies);
2960}
2961
2962
2963/**
2964 * Close an interface.
2965 *
2966 * @returns VBox status code.
2967 * @param pIntNet The instance handle.
2968 * @param hIf The interface handle.
2969 * @param pSession The caller's session.
2970 */
2971INTNETR0DECL(int) INTNETR0IfClose(PINTNET pIntNet, INTNETIFHANDLE hIf, PSUPDRVSESSION pSession)
2972{
2973 LogFlow(("INTNETR0IfClose: pIntNet=%p hIf=%RX32\n", pIntNet, hIf));
2974
2975 /*
2976 * Validate and free the handle.
2977 */
2978 AssertPtrReturn(pIntNet, VERR_INVALID_PARAMETER);
2979 PINTNETIF pIf = (PINTNETIF)RTHandleTableFreeWithCtx(pIntNet->hHtIfs, hIf, pSession);
2980 if (!pIf)
2981 return VERR_INVALID_HANDLE;
2982
2983 /* mark the handle as freed so intnetR0IfDestruct won't free it again. */
2984 ASMAtomicWriteU32(&pIf->hIf, INTNET_HANDLE_INVALID);
2985
2986
2987 /*
2988 * Release the references to the interface object (handle + free lookup).
2989 * But signal the event semaphore first so any waiter holding a reference
2990 * will wake up too (he'll see hIf == invalid and return correctly).
2991 */
2992 RTSemEventSignal(pIf->Event);
2993
2994 void *pvObj = pIf->pvObj;
2995 intnetR0IfRelease(pIf, pSession); /* (RTHandleTableFreeWithCtx) */
2996
2997 int rc = SUPR0ObjRelease(pvObj, pSession);
2998 LogFlow(("INTNETR0IfClose: returns %Rrc\n", rc));
2999 return rc;
3000}
3001
3002
3003/**
3004 * VMMR0 request wrapper for INTNETR0IfCloseReq.
3005 *
3006 * @returns see INTNETR0IfClose.
3007 * @param pIntNet The internal networking instance.
3008 * @param pSession The caller's session.
3009 * @param pReq The request packet.
3010 */
3011INTNETR0DECL(int) INTNETR0IfCloseReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETIFCLOSEREQ pReq)
3012{
3013 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
3014 return VERR_INVALID_PARAMETER;
3015 return INTNETR0IfClose(pIntNet, pReq->hIf, pSession);
3016}
3017
3018
3019/**
3020 * Interface destructor callback.
3021 * This is called for reference counted objectes when the count reaches 0.
3022 *
3023 * @param pvObj The object pointer.
3024 * @param pvUser1 Pointer to the interface.
3025 * @param pvUser2 Pointer to the INTNET instance data.
3026 */
3027static DECLCALLBACK(void) intnetR0IfDestruct(void *pvObj, void *pvUser1, void *pvUser2)
3028{
3029 PINTNETIF pIf = (PINTNETIF)pvUser1;
3030 PINTNET pIntNet = (PINTNET)pvUser2;
3031 Log(("intnetR0IfDestruct: pvObj=%p pIf=%p pIntNet=%p hIf=%RX32\n", pvObj, pIf, pIntNet, pIf->hIf));
3032
3033 RTSemFastMutexRequest(pIntNet->FastMutex);
3034
3035 /*
3036 * Mark the interface as being destroyed so the waiter
3037 * can behave appropriately (theoretical case).
3038 */
3039 ASMAtomicWriteBool(&pIf->fDestroying, true);
3040
3041 /*
3042 * Delete the interface handle so the object no longer can be used.
3043 * (Can happen if the client didn't close its session.)
3044 */
3045 INTNETIFHANDLE hIf = ASMAtomicXchgU32(&pIf->hIf, INTNET_HANDLE_INVALID);
3046 if (hIf != INTNET_HANDLE_INVALID)
3047 {
3048 void *pvObj2 = RTHandleTableFreeWithCtx(pIntNet->hHtIfs, hIf, pIf->pSession); NOREF(pvObj2);
3049 AssertMsg(pvObj2 == pIf, ("%p, %p, hIf=%RX32 pSession=%p\n", pvObj2, pIf, hIf, pIf->pSession));
3050 }
3051
3052 /*
3053 * If we've got a network deactivate and unlink ourselves from it.
3054 * Because of cleanup order we might be an orphan now.
3055 */
3056 PINTNETNETWORK pNetwork = pIf->pNetwork;
3057 if (pNetwork)
3058 {
3059 intnetR0IfSetActive(pIf, false);
3060
3061 if (pNetwork->pIFs == pIf)
3062 pNetwork->pIFs = pIf->pNext;
3063 else
3064 {
3065 PINTNETIF pPrev = pNetwork->pIFs;
3066 while (pPrev)
3067 {
3068 if (pPrev->pNext == pIf)
3069 {
3070 pPrev->pNext = pIf->pNext;
3071 break;
3072 }
3073 pPrev = pPrev->pNext;
3074 }
3075 Assert(pPrev);
3076 }
3077 pIf->pNext = NULL;
3078
3079 /*
3080 * Release our reference to the network.
3081 */
3082 RTSemFastMutexRelease(pIntNet->FastMutex);
3083
3084 SUPR0ObjRelease(pNetwork->pvObj, pIf->pSession);
3085 pIf->pNetwork = NULL;
3086 }
3087 else
3088 RTSemFastMutexRelease(pIntNet->FastMutex);
3089
3090 /*
3091 * Wakeup anyone waiting on this interface.
3092 *
3093 * We *must* make sure they have woken up properly and realized
3094 * that the interface is no longer valid.
3095 */
3096 if (pIf->Event != NIL_RTSEMEVENT)
3097 {
3098 RTSEMEVENT Event = pIf->Event;
3099 unsigned cMaxWait = 0x1000;
3100 while (pIf->cSleepers && cMaxWait-- > 0)
3101 {
3102 RTSemEventSignal(Event);
3103 RTThreadYield();
3104 }
3105 if (pIf->cSleepers)
3106 {
3107 RTThreadSleep(1);
3108
3109 cMaxWait = pIf->cSleepers;
3110 while (pIf->cSleepers && cMaxWait-- > 0)
3111 {
3112 RTSemEventSignal(Event);
3113 RTThreadSleep(10);
3114 }
3115 }
3116
3117 RTSemEventDestroy(Event);
3118 pIf->Event = NIL_RTSEMEVENT;
3119 }
3120
3121 /*
3122 * Unmap user buffer.
3123 */
3124 if (pIf->pIntBuf != pIf->pIntBufDefault)
3125 {
3126 /** @todo user buffer */
3127 }
3128
3129 /*
3130 * Unmap and Free the default buffer.
3131 */
3132 if (pIf->pIntBufDefault)
3133 {
3134 SUPR0MemFree(pIf->pSession, (RTHCUINTPTR)pIf->pIntBufDefault);
3135 pIf->pIntBufDefault = NULL;
3136 pIf->pIntBufDefaultR3 = 0;
3137 pIf->pIntBuf = NULL;
3138 pIf->pIntBufR3 = 0;
3139 }
3140
3141 /*
3142 * The interface.
3143 */
3144 pIf->pvObj = NULL;
3145 RTMemFree(pIf);
3146}
3147
3148
3149/**
3150 * Creates a new network interface.
3151 *
3152 * The call must have opened the network for the new interface
3153 * and is responsible for closing it on failure. On success
3154 * it must leave the network opened so the interface destructor
3155 * can close it.
3156 *
3157 * @returns VBox status code.
3158 * @param pNetwork The network.
3159 * @param pSession The session handle.
3160 * @param cbSend The size of the send buffer.
3161 * @param cbRecv The size of the receive buffer.
3162 * @param phIf Where to store the interface handle.
3163 */
3164static int intnetR0NetworkCreateIf(PINTNETNETWORK pNetwork, PSUPDRVSESSION pSession, unsigned cbSend, unsigned cbRecv, bool *pfCloseNetwork, PINTNETIFHANDLE phIf)
3165{
3166 LogFlow(("intnetR0NetworkCreateIf: pNetwork=%p pSession=%p cbSend=%u cbRecv=%u phIf=%p\n",
3167 pNetwork, pSession, cbSend, cbRecv, phIf));
3168
3169 /*
3170 * Assert input.
3171 */
3172 AssertPtr(pNetwork);
3173 AssertPtr(phIf);
3174 AssertPtr(pfCloseNetwork);
3175 *pfCloseNetwork = false;
3176
3177 /*
3178 * Allocate and initialize the interface structure.
3179 */
3180 PINTNETIF pIf = (PINTNETIF)RTMemAllocZ(sizeof(*pIf));
3181 if (!pIf)
3182 return VERR_NO_MEMORY;
3183 //pIf->pNext = NULL;
3184 memset(&pIf->Mac, 0xff, sizeof(pIf->Mac)); /* broadcast */
3185 //pIf->fMacSet = false;
3186 //pIf->fPromiscuous = false;
3187 //pIf->fActive = false;
3188 //pIf->fDestroying = false;
3189 //pIf->pIntBuf = 0;
3190 //pIf->pIntBufR3 = NIL_RTR3PTR;
3191 //pIf->pIntBufDefault = 0;
3192 //pIf->pIntBufDefaultR3 = NIL_RTR3PTR;
3193 //pIf->cYields = 0;
3194 pIf->Event = NIL_RTSEMEVENT;
3195 //pIf->cSleepers = 0;
3196 pIf->hIf = INTNET_HANDLE_INVALID;
3197 pIf->pNetwork = pNetwork;
3198 pIf->pSession = pSession;
3199 //pIf->pvObj = NULL;
3200 //pIf->aAddrCache[kIntNetAddrType_Invalid] = {0};
3201 //pIf->aAddrCache[kIntNetAddrType_IPv4].pbEntries = NULL;
3202 //pIf->aAddrCache[kIntNetAddrType_IPv4].cEntries = 0;
3203 //pIf->aAddrCache[kIntNetAddrType_IPv4].cEntriesAlloc = 0;
3204 pIf->aAddrCache[kIntNetAddrType_IPv4].cbAddress = intnetR0AddrSize(kIntNetAddrType_IPv4);
3205 pIf->aAddrCache[kIntNetAddrType_IPv4].cbEntry = intnetR0AddrSize(kIntNetAddrType_IPv4);
3206 //pIf->aAddrCache[kIntNetAddrType_IPv6].pbEntries = NULL;
3207 //pIf->aAddrCache[kIntNetAddrType_IPv6].cEntries = 0;
3208 //pIf->aAddrCache[kIntNetAddrType_IPv6].cEntriesAlloc = 0;
3209 pIf->aAddrCache[kIntNetAddrType_IPv6].cbAddress = intnetR0AddrSize(kIntNetAddrType_IPv6);
3210 pIf->aAddrCache[kIntNetAddrType_IPv6].cbEntry = intnetR0AddrSize(kIntNetAddrType_IPv6);
3211 //pIf->aAddrCache[kIntNetAddrType_IPX].pbEntries = NULL;
3212 //pIf->aAddrCache[kIntNetAddrType_IPX].cEntries = 0;
3213 //pIf->aAddrCache[kIntNetAddrType_IPX].cEntriesAlloc = 0;
3214 pIf->aAddrCache[kIntNetAddrType_IPX].cbAddress = intnetR0AddrSize(kIntNetAddrType_IPX);
3215 pIf->aAddrCache[kIntNetAddrType_IPX].cbEntry = RT_ALIGN_32(intnetR0AddrSize(kIntNetAddrType_IPv4), 16);
3216 int rc = RTSemEventCreate((PRTSEMEVENT)&pIf->Event);
3217 if (RT_SUCCESS(rc))
3218 {
3219 /*
3220 * Create the default buffer.
3221 */
3222 /** @todo adjust with minimums and apply defaults here. */
3223 cbRecv = RT_ALIGN(RT_MAX(cbRecv, sizeof(INTNETHDR) * 4), INTNETRINGBUF_ALIGNMENT);
3224 cbSend = RT_ALIGN(RT_MAX(cbSend, sizeof(INTNETHDR) * 4), INTNETRINGBUF_ALIGNMENT);
3225 const unsigned cbBuf = RT_ALIGN(sizeof(*pIf->pIntBuf), INTNETRINGBUF_ALIGNMENT) + cbRecv + cbSend;
3226 rc = SUPR0MemAlloc(pIf->pSession, cbBuf, (PRTR0PTR)&pIf->pIntBufDefault, (PRTR3PTR)&pIf->pIntBufDefaultR3);
3227 if (RT_SUCCESS(rc))
3228 {
3229 ASMMemZero32(pIf->pIntBufDefault, cbBuf); /** @todo I thought I specified these buggers as clearing the memory... */
3230
3231 pIf->pIntBuf = pIf->pIntBufDefault;
3232 pIf->pIntBufR3 = pIf->pIntBufDefaultR3;
3233 INTNETBufInit(pIf->pIntBuf, cbBuf, cbRecv, cbSend);
3234
3235 /*
3236 * Link the interface to the network.
3237 */
3238 rc = RTSemFastMutexRequest(pNetwork->FastMutex);
3239 if (RT_SUCCESS(rc))
3240 {
3241 pIf->pNext = pNetwork->pIFs;
3242 pNetwork->pIFs = pIf;
3243 RTSemFastMutexRelease(pNetwork->FastMutex);
3244
3245 /*
3246 * Register the interface with the session.
3247 */
3248 pIf->pvObj = SUPR0ObjRegister(pSession, SUPDRVOBJTYPE_INTERNAL_NETWORK_INTERFACE, intnetR0IfDestruct, pIf, pNetwork->pIntNet);
3249 if (pIf->pvObj)
3250 {
3251 rc = RTHandleTableAllocWithCtx(pNetwork->pIntNet->hHtIfs, pIf, pSession, (uint32_t *)&pIf->hIf);
3252 if (RT_SUCCESS(rc))
3253 {
3254 *phIf = pIf->hIf;
3255 Log(("intnetR0NetworkCreateIf: returns VINF_SUCCESS *phIf=%RX32 cbSend=%u cbRecv=%u cbBuf=%u\n",
3256 *phIf, pIf->pIntBufDefault->cbSend, pIf->pIntBufDefault->cbRecv, pIf->pIntBufDefault->cbBuf));
3257 return VINF_SUCCESS;
3258 }
3259
3260 SUPR0ObjRelease(pIf->pvObj, pSession);
3261 LogFlow(("intnetR0NetworkCreateIf: returns %Rrc\n", rc));
3262 return rc;
3263 }
3264
3265 RTSemFastMutexDestroy(pNetwork->FastMutex);
3266 pNetwork->FastMutex = NIL_RTSEMFASTMUTEX;
3267 }
3268
3269 SUPR0MemFree(pIf->pSession, (RTHCUINTPTR)pIf->pIntBufDefault);
3270 pIf->pIntBufDefault = NULL;
3271 pIf->pIntBuf = NULL;
3272 }
3273
3274 RTSemEventDestroy(pIf->Event);
3275 pIf->Event = NIL_RTSEMEVENT;
3276 }
3277 RTMemFree(pIf);
3278 LogFlow(("intnetR0NetworkCreateIf: returns %Rrc\n", rc));
3279 *pfCloseNetwork = true;
3280 return rc;
3281}
3282
3283
3284/** @copydoc INTNETTRUNKSWPORT::pfnSetSGPhys */
3285static DECLCALLBACK(bool) intnetR0TrunkIfPortSetSGPhys(PINTNETTRUNKSWPORT pSwitchPort, bool fEnable)
3286{
3287 PINTNETTRUNKIF pThis = INTNET_SWITCHPORT_2_TRUNKIF(pSwitchPort);
3288 AssertMsgFailed(("Not implemented because it wasn't required on Darwin\n"));
3289 return ASMAtomicXchgBool(&pThis->fPhysSG, fEnable);
3290}
3291
3292
3293/** @copydoc INTNETTRUNKSWPORT::pfnRecv */
3294static DECLCALLBACK(bool) intnetR0TrunkIfPortRecv(PINTNETTRUNKSWPORT pSwitchPort, PINTNETSG pSG, uint32_t fSrc)
3295{
3296 PINTNETTRUNKIF pThis = INTNET_SWITCHPORT_2_TRUNKIF(pSwitchPort);
3297 PINTNETNETWORK pNetwork = pThis->pNetwork;
3298
3299 /* assert some sanity */
3300 AssertPtrReturn(pNetwork, false);
3301 AssertReturn(pNetwork->FastMutex != NIL_RTSEMFASTMUTEX, false);
3302 AssertPtr(pSG);
3303 Assert(fSrc);
3304
3305 /*
3306 * Lock the network and send the frame to it.
3307 */
3308 int rc = RTSemFastMutexRequest(pNetwork->FastMutex);
3309 AssertRCReturn(rc, false);
3310
3311 bool fRc;
3312 if (RT_LIKELY(pNetwork->cActiveIFs > 0))
3313 fRc = intnetR0NetworkSend(pNetwork, NULL, fSrc, pSG, false /* fTrunkLocked */);
3314 else
3315 fRc = false; /* don't drop it */
3316
3317 rc = RTSemFastMutexRelease(pNetwork->FastMutex);
3318 AssertRC(rc);
3319
3320 return fRc;
3321}
3322
3323
3324/** @copydoc INTNETTRUNKSWPORT::pfnSGRetain */
3325static DECLCALLBACK(void) intnetR0TrunkIfPortSGRetain(PINTNETTRUNKSWPORT pSwitchPort, PINTNETSG pSG)
3326{
3327 PINTNETTRUNKIF pThis = INTNET_SWITCHPORT_2_TRUNKIF(pSwitchPort);
3328 PINTNETNETWORK pNetwork = pThis->pNetwork;
3329
3330 /* assert some sanity */
3331 AssertPtrReturnVoid(pNetwork);
3332 AssertReturnVoid(pNetwork->FastMutex != NIL_RTSEMFASTMUTEX);
3333 AssertPtr(pSG);
3334 Assert(pSG->cUsers > 0);
3335
3336 /* do it. */
3337 ++pSG->cUsers;
3338}
3339
3340
3341/** @copydoc INTNETTRUNKSWPORT::pfnSGRelease */
3342static DECLCALLBACK(void) intnetR0TrunkIfPortSGRelease(PINTNETTRUNKSWPORT pSwitchPort, PINTNETSG pSG)
3343{
3344 PINTNETTRUNKIF pThis = INTNET_SWITCHPORT_2_TRUNKIF(pSwitchPort);
3345 PINTNETNETWORK pNetwork = pThis->pNetwork;
3346
3347 /* assert some sanity */
3348 AssertPtrReturnVoid(pNetwork);
3349 AssertReturnVoid(pNetwork->FastMutex != NIL_RTSEMFASTMUTEX);
3350 AssertPtr(pSG);
3351 Assert(pSG->cUsers > 0);
3352
3353 /*
3354 * Free it?
3355 */
3356 if (!--pSG->cUsers)
3357 {
3358 /** @todo later */
3359 }
3360}
3361
3362
3363/**
3364 * Retain the trunk interface.
3365 *
3366 * @returns pThis if retained.
3367 *
3368 * @param pThis The trunk.
3369 *
3370 * @remarks Any locks.
3371 */
3372static PINTNETTRUNKIF intnetR0TrunkIfRetain(PINTNETTRUNKIF pThis)
3373{
3374 if (pThis && pThis->pIfPort)
3375 {
3376 pThis->pIfPort->pfnRetain(pThis->pIfPort);
3377 return pThis;
3378 }
3379 return NULL;
3380}
3381
3382
3383/**
3384 * Release the trunk interface.
3385 *
3386 * @param pThis The trunk.
3387 */
3388static void intnetR0TrunkIfRelease(PINTNETTRUNKIF pThis)
3389{
3390 if (pThis && pThis->pIfPort)
3391 pThis->pIfPort->pfnRelease(pThis->pIfPort);
3392}
3393
3394
3395/**
3396 * Takes the out-bound trunk lock.
3397 *
3398 * This will ensure that pIfPort is valid.
3399 *
3400 * @returns success indicator.
3401 * @param pThis The trunk.
3402 *
3403 * @remarks No locks other than the create/destroy one.
3404 */
3405static bool intnetR0TrunkIfOutLock(PINTNETTRUNKIF pThis)
3406{
3407 AssertPtrReturn(pThis, false);
3408 int rc = RTSemFastMutexRequest(pThis->FastMutex);
3409 if (RT_SUCCESS(rc))
3410 {
3411 if (RT_LIKELY(pThis->pIfPort))
3412 return true;
3413 RTSemFastMutexRelease(pThis->FastMutex);
3414 }
3415 else
3416 AssertMsg(rc == VERR_SEM_DESTROYED, ("%Rrc\n", rc));
3417 return false;
3418}
3419
3420
3421/**
3422 * Releases the out-bound trunk lock.
3423 *
3424 * @param pThis The trunk.
3425 */
3426static void intnetR0TrunkIfOutUnlock(PINTNETTRUNKIF pThis)
3427{
3428 if (pThis)
3429 {
3430 int rc = RTSemFastMutexRelease(pThis->FastMutex);
3431 AssertRC(rc);
3432 }
3433}
3434
3435
3436/**
3437 * Activates the trunk interface.
3438 *
3439 * @param pThis The trunk.
3440 * @param fActive What to do with it.
3441 *
3442 * @remarks Caller may only own the create/destroy lock.
3443 */
3444static void intnetR0TrunkIfActivate(PINTNETTRUNKIF pThis, bool fActive)
3445{
3446 if (intnetR0TrunkIfOutLock(pThis))
3447 {
3448 pThis->pIfPort->pfnSetActive(pThis->pIfPort, fActive);
3449 intnetR0TrunkIfOutUnlock(pThis);
3450 }
3451}
3452
3453
3454/**
3455 * Shutdown the trunk interface.
3456 *
3457 * @param pThis The trunk.
3458 * @param pNetworks The network.
3459 *
3460 * @remarks The caller must *NOT* hold the network lock. The global
3461 * create/destroy lock is fine though.
3462 */
3463static void intnetR0TrunkIfDestroy(PINTNETTRUNKIF pThis, PINTNETNETWORK pNetwork)
3464{
3465 /* assert sanity */
3466 if (!pThis)
3467 return;
3468 AssertPtr(pThis);
3469 Assert(pThis->pNetwork == pNetwork);
3470 AssertPtrNull(pThis->pIfPort);
3471
3472 /*
3473 * The interface has already been deactivated, we just to wait for
3474 * it to become idle before we can disconnect and release it.
3475 */
3476 PINTNETTRUNKIFPORT pIfPort = pThis->pIfPort;
3477 if (pIfPort)
3478 {
3479 intnetR0TrunkIfOutLock(pThis);
3480
3481 /* unset it */
3482 pThis->pIfPort = NULL;
3483
3484 /* wait in portions so we can complain ever now an then. */
3485 uint64_t StartTS = RTTimeSystemNanoTS();
3486 int rc = pIfPort->pfnWaitForIdle(pIfPort, 10*1000);
3487 if (RT_FAILURE(rc))
3488 {
3489 LogRel(("intnet: '%s' did't become idle in %RU64 ns (%Rrc).\n",
3490 pNetwork->szName, RTTimeSystemNanoTS() - StartTS, rc));
3491 Assert(rc == VERR_TIMEOUT);
3492 while ( RT_FAILURE(rc)
3493 && RTTimeSystemNanoTS() - StartTS < UINT64_C(30000000000)) /* 30 sec */
3494 rc = pIfPort->pfnWaitForIdle(pIfPort, 10*1000);
3495 if (rc == VERR_TIMEOUT)
3496 {
3497 LogRel(("intnet: '%s' did't become idle in %RU64 ns (%Rrc).\n",
3498 pNetwork->szName, RTTimeSystemNanoTS() - StartTS, rc));
3499 while ( rc == VERR_TIMEOUT
3500 && RTTimeSystemNanoTS() - StartTS < UINT64_C(360000000000)) /* 360 sec */
3501 rc = pIfPort->pfnWaitForIdle(pIfPort, 30*1000);
3502 if (RT_FAILURE(rc))
3503 {
3504 LogRel(("intnet: '%s' did't become idle in %RU64 ns (%Rrc), giving up.\n",
3505 pNetwork->szName, RTTimeSystemNanoTS() - StartTS, rc));
3506 AssertRC(rc);
3507 }
3508 }
3509 }
3510
3511 /* disconnect & release it. */
3512 pIfPort->pfnDisconnectAndRelease(pIfPort);
3513 }
3514
3515 /*
3516 * Free up the resources.
3517 */
3518 RTSEMFASTMUTEX FastMutex = pThis->FastMutex;
3519 pThis->FastMutex = NIL_RTSEMFASTMUTEX;
3520 pThis->pNetwork = NULL;
3521 RTSemFastMutexRelease(FastMutex);
3522 RTSemFastMutexDestroy(FastMutex);
3523 RTMemFree(pThis);
3524}
3525
3526
3527/**
3528 * Creates the trunk connection (if any).
3529 *
3530 * @returns VBox status code.
3531 *
3532 * @param pNetwork The newly created network.
3533 * @param pSession The session handle.
3534 */
3535static int intnetR0NetworkCreateTrunkIf(PINTNETNETWORK pNetwork, PSUPDRVSESSION pSession)
3536{
3537 const char *pszName;
3538 switch (pNetwork->enmTrunkType)
3539 {
3540 /*
3541 * The 'None' case, simple.
3542 */
3543 case kIntNetTrunkType_None:
3544 case kIntNetTrunkType_WhateverNone:
3545 return VINF_SUCCESS;
3546
3547 /* Can't happen, but makes GCC happy. */
3548 default:
3549 return VERR_NOT_IMPLEMENTED;
3550
3551 /*
3552 * Translate enum to component factory name.
3553 */
3554 case kIntNetTrunkType_NetFlt:
3555 pszName = "VBoxNetFlt";
3556 break;
3557 case kIntNetTrunkType_NetAdp:
3558#if defined(RT_OS_DARWIN) && !defined(VBOXNETADP_DO_NOT_USE_NETFLT)
3559 pszName = "VBoxNetFlt";
3560#else /* VBOXNETADP_DO_NOT_USE_NETFLT */
3561 pszName = "VBoxNetAdp";
3562#endif /* VBOXNETADP_DO_NOT_USE_NETFLT */
3563 break;
3564 case kIntNetTrunkType_SrvNat:
3565 pszName = "VBoxSrvNat";
3566 break;
3567 }
3568
3569 /*
3570 * Allocate the trunk interface.
3571 */
3572 PINTNETTRUNKIF pTrunkIF = (PINTNETTRUNKIF)RTMemAllocZ(sizeof(*pTrunkIF));
3573 if (!pTrunkIF)
3574 return VERR_NO_MEMORY;
3575 pTrunkIF->SwitchPort.u32Version = INTNETTRUNKSWPORT_VERSION;
3576 pTrunkIF->SwitchPort.pfnSetSGPhys = intnetR0TrunkIfPortSetSGPhys;
3577 pTrunkIF->SwitchPort.pfnRecv = intnetR0TrunkIfPortRecv;
3578 pTrunkIF->SwitchPort.pfnSGRetain = intnetR0TrunkIfPortSGRetain;
3579 pTrunkIF->SwitchPort.pfnSGRelease = intnetR0TrunkIfPortSGRelease;
3580 pTrunkIF->SwitchPort.u32VersionEnd = INTNETTRUNKSWPORT_VERSION;
3581 //pTrunkIF->pIfPort = NULL;
3582 pTrunkIF->pNetwork = pNetwork;
3583 //pTrunkIF->fPhysSG = false;
3584 //pTrunkIF->fPromiscuousWire = false;
3585 pTrunkIF->CachedMac.au8[0] = 0xfe;
3586 pTrunkIF->CachedMac.au8[1] = 0xff;
3587 pTrunkIF->CachedMac.au8[2] = 0xff;
3588 pTrunkIF->CachedMac.au8[3] = 0xff;
3589 pTrunkIF->CachedMac.au8[4] = 0xff;
3590 pTrunkIF->CachedMac.au8[5] = 0xff;
3591 int rc = RTSemFastMutexCreate(&pTrunkIF->FastMutex);
3592 if (RT_SUCCESS(rc))
3593 {
3594#ifdef IN_RING0 /* (testcase is ring-3) */
3595 /*
3596 * Query the factory we want, then use it create and connect the trunk.
3597 */
3598 PINTNETTRUNKFACTORY pTrunkFactory = NULL;
3599 rc = SUPR0ComponentQueryFactory(pSession, pszName, INTNETTRUNKFACTORY_UUID_STR, (void **)&pTrunkFactory);
3600 if (RT_SUCCESS(rc))
3601 {
3602 rc = pTrunkFactory->pfnCreateAndConnect(pTrunkFactory, pNetwork->szTrunk, &pTrunkIF->SwitchPort,
3603 pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE
3604 ? INTNETTRUNKFACTORY_FLAG_NO_PROMISC : 0,
3605 &pTrunkIF->pIfPort);
3606 pTrunkFactory->pfnRelease(pTrunkFactory);
3607 if (RT_SUCCESS(rc))
3608 {
3609 Assert(pTrunkIF->pIfPort);
3610 pNetwork->pTrunkIF = pTrunkIF;
3611 Log(("intnetR0NetworkCreateTrunkIf: VINF_SUCCESS - pszName=%s szTrunk=%s%s Network=%s\n",
3612 pszName, pNetwork->szTrunk, pNetwork->fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE ? " shared-mac" : "", pNetwork->szName));
3613 return VINF_SUCCESS;
3614 }
3615 }
3616#endif /* IN_RING0 */
3617 RTSemFastMutexDestroy(pTrunkIF->FastMutex);
3618 }
3619 RTMemFree(pTrunkIF);
3620 LogFlow(("intnetR0NetworkCreateTrunkIf: %Rrc - pszName=%s szTrunk=%s Network=%s\n",
3621 rc, pszName, pNetwork->szTrunk, pNetwork->szName));
3622 return rc;
3623}
3624
3625
3626
3627/**
3628 * Close a network which was opened/created using intnetR0OpenNetwork()/intnetR0CreateNetwork().
3629 *
3630 * @param pNetwork The network to close.
3631 * @param pSession The session handle.
3632 */
3633static int intnetR0NetworkClose(PINTNETNETWORK pNetwork, PSUPDRVSESSION pSession)
3634{
3635 LogFlow(("intnetR0NetworkClose: pNetwork=%p pSession=%p\n", pNetwork, pSession));
3636 AssertPtrReturn(pSession, VERR_INVALID_PARAMETER);
3637 AssertPtrReturn(pNetwork, VERR_INVALID_PARAMETER);
3638
3639 int rc = SUPR0ObjRelease(pNetwork->pvObj, pSession);
3640 LogFlow(("intnetR0NetworkClose: return %Rrc\n", rc));
3641 return rc;
3642}
3643
3644
3645/**
3646 * Object destructor callback.
3647 * This is called for reference counted objectes when the count reaches 0.
3648 *
3649 * @param pvObj The object pointer.
3650 * @param pvUser1 Pointer to the network.
3651 * @param pvUser2 Pointer to the INTNET instance data.
3652 */
3653static DECLCALLBACK(void) intnetR0NetworkDestruct(void *pvObj, void *pvUser1, void *pvUser2)
3654{
3655 PINTNETNETWORK pNetwork = (PINTNETNETWORK)pvUser1;
3656 PINTNET pIntNet = (PINTNET)pvUser2;
3657 Log(("intnetR0NetworkDestruct: pvObj=%p pNetwork=%p pIntNet=%p %s\n", pvObj, pNetwork, pIntNet, pNetwork->szName));
3658 Assert(pNetwork->pIntNet == pIntNet);
3659
3660 /* take the create/destroy sem. */
3661 RTSemFastMutexRequest(pIntNet->FastMutex);
3662
3663 /*
3664 * Deactivate the trunk connection first (if any).
3665 */
3666 if (pNetwork->pTrunkIF)
3667 intnetR0TrunkIfActivate(pNetwork->pTrunkIF, false /* fActive */);
3668
3669 /*
3670 * Unlink the network.
3671 * Note that it needn't be in the list if we failed during creation.
3672 */
3673 PINTNETNETWORK pPrev = pIntNet->pNetworks;
3674 if (pPrev == pNetwork)
3675 pIntNet->pNetworks = pNetwork->pNext;
3676 else
3677 {
3678 for (; pPrev; pPrev = pPrev->pNext)
3679 if (pPrev->pNext == pNetwork)
3680 {
3681 pPrev->pNext = pNetwork->pNext;
3682 break;
3683 }
3684 }
3685 pNetwork->pNext = NULL;
3686 pNetwork->pvObj = NULL;
3687
3688 /*
3689 * Because of the undefined order of the per session object dereferencing when closing a session,
3690 * we have to handle the case where the network is destroyed before the interfaces. We'll
3691 * deal with this by simply orphaning the interfaces.
3692 */
3693 RTSemFastMutexRequest(pNetwork->FastMutex);
3694
3695 PINTNETIF pCur = pNetwork->pIFs;
3696 while (pCur)
3697 {
3698 PINTNETIF pNext = pCur->pNext;
3699 pCur->pNext = NULL;
3700 pCur->pNetwork = NULL;
3701 pCur = pNext;
3702 }
3703
3704 /* Grab and zap the trunk pointer before leaving the mutex. */
3705 PINTNETTRUNKIF pTrunkIF = pNetwork->pTrunkIF;
3706 pNetwork->pTrunkIF = NULL;
3707
3708 RTSemFastMutexRelease(pNetwork->FastMutex);
3709
3710 /*
3711 * If there is a trunk, delete it.
3712 * Note that this may tak a while if we're unlucky...
3713 */
3714 if (pTrunkIF)
3715 intnetR0TrunkIfDestroy(pTrunkIF, pNetwork);
3716
3717 /*
3718 * Free resources.
3719 */
3720 RTSemFastMutexDestroy(pNetwork->FastMutex);
3721 pNetwork->FastMutex = NIL_RTSEMFASTMUTEX;
3722 RTMemFree(pNetwork);
3723
3724 /* release the create/destroy sem. (can be done before trunk destruction.) */
3725 RTSemFastMutexRelease(pIntNet->FastMutex);
3726}
3727
3728
3729/**
3730 * Opens an existing network.
3731 *
3732 * @returns VBox status code.
3733 * @param pIntNet The instance data.
3734 * @param pSession The current session.
3735 * @param pszNetwork The network name. This has a valid length.
3736 * @param enmTrunkType The trunk type.
3737 * @param pszTrunk The trunk name. Its meaning is specfic to the type.
3738 * @param fFlags Flags, see INTNET_OPEN_FLAGS_*.
3739 * @param ppNetwork Where to store the pointer to the network on success.
3740 */
3741static int intnetR0OpenNetwork(PINTNET pIntNet, PSUPDRVSESSION pSession, const char *pszNetwork, INTNETTRUNKTYPE enmTrunkType,
3742 const char *pszTrunk, uint32_t fFlags, PINTNETNETWORK *ppNetwork)
3743{
3744 LogFlow(("intnetR0OpenNetwork: pIntNet=%p pSession=%p pszNetwork=%p:{%s} enmTrunkType=%d pszTrunk=%p:{%s} fFlags=%#x ppNetwork=%p\n",
3745 pIntNet, pSession, pszNetwork, pszNetwork, enmTrunkType, pszTrunk, pszTrunk, fFlags, ppNetwork));
3746
3747 /* just pro forma validation, the caller is internal. */
3748 AssertPtr(pIntNet);
3749 AssertPtr(pSession);
3750 AssertPtr(pszNetwork);
3751 Assert(enmTrunkType > kIntNetTrunkType_Invalid && enmTrunkType < kIntNetTrunkType_End);
3752 AssertPtr(pszTrunk);
3753 Assert(!(fFlags & ~(INTNET_OPEN_FLAGS_MASK)));
3754 AssertPtr(ppNetwork);
3755 *ppNetwork = NULL;
3756
3757 /*
3758 * Search networks by name.
3759 */
3760 PINTNETNETWORK pCur;
3761 uint8_t cchName = (uint8_t)strlen(pszNetwork);
3762 Assert(cchName && cchName < sizeof(pCur->szName)); /* caller ensures this */
3763
3764 pCur = pIntNet->pNetworks;
3765 while (pCur)
3766 {
3767 if ( pCur->cchName == cchName
3768 && !memcmp(pCur->szName, pszNetwork, cchName))
3769 {
3770 /*
3771 * Found the network, now check that we have the same ideas
3772 * about the trunk setup and security.
3773 */
3774 int rc;
3775 if ( enmTrunkType == kIntNetTrunkType_WhateverNone
3776 || ( pCur->enmTrunkType == enmTrunkType
3777 && !strcmp(pCur->szTrunk, pszTrunk)))
3778 {
3779 if (!((pCur->fFlags ^ fFlags) & INTNET_OPEN_FLAGS_COMPATIBILITY_XOR_MASK))
3780 {
3781
3782 /*
3783 * Increment the reference and check that the session
3784 * can access this network.
3785 */
3786 rc = SUPR0ObjAddRef(pCur->pvObj, pSession);
3787 if (RT_SUCCESS(rc))
3788 {
3789 if (!(pCur->fFlags & INTNET_OPEN_FLAGS_PUBLIC))
3790 rc = SUPR0ObjVerifyAccess(pCur->pvObj, pSession, pCur->szName);
3791 if (RT_SUCCESS(rc))
3792 {
3793 pCur->fFlags |= fFlags & INTNET_OPEN_FLAGS_SECURITY_OR_MASK;
3794
3795 *ppNetwork = pCur;
3796 }
3797 else
3798 SUPR0ObjRelease(pCur->pvObj, pSession);
3799 }
3800 else if (rc == VERR_WRONG_ORDER)
3801 rc = VERR_NOT_FOUND; /* destruction race, pretend the other isn't there. */
3802 }
3803 else
3804 rc = VERR_INTNET_INCOMPATIBLE_FLAGS;
3805 }
3806 else
3807 rc = VERR_INTNET_INCOMPATIBLE_TRUNK;
3808
3809 LogFlow(("intnetR0OpenNetwork: returns %Rrc *ppNetwork=%p\n", rc, *ppNetwork));
3810 return rc;
3811 }
3812 pCur = pCur->pNext;
3813 }
3814
3815 LogFlow(("intnetR0OpenNetwork: returns VERR_NOT_FOUND\n"));
3816 return VERR_NOT_FOUND;
3817}
3818
3819
3820/**
3821 * Creates a new network.
3822 *
3823 * The call must own the INTNET::FastMutex and has already attempted
3824 * opening the network and found it to be non-existing.
3825 *
3826 * @returns VBox status code.
3827 * @param pIntNet The instance data.
3828 * @param pSession The session handle.
3829 * @param pszNetwork The name of the network. This must be at least one character long and no longer
3830 * than the INTNETNETWORK::szName.
3831 * @param enmTrunkType The trunk type.
3832 * @param pszTrunk The trunk name. Its meaning is specfic to the type.
3833 * @param fFlags Flags, see INTNET_OPEN_FLAGS_*.
3834 * @param ppNetwork Where to store the network. In the case of failure whatever is returned
3835 * here should be dereferenced outside the INTNET::FastMutex.
3836 */
3837static int intnetR0CreateNetwork(PINTNET pIntNet, PSUPDRVSESSION pSession, const char *pszNetwork, INTNETTRUNKTYPE enmTrunkType,
3838 const char *pszTrunk, uint32_t fFlags, PINTNETNETWORK *ppNetwork)
3839{
3840 LogFlow(("intnetR0CreateNetwork: pIntNet=%p pSession=%p pszNetwork=%p:{%s} enmTrunkType=%d pszTrunk=%p:{%s} fFlags=%#x ppNetwork=%p\n",
3841 pIntNet, pSession, pszNetwork, pszNetwork, enmTrunkType, pszTrunk, pszTrunk, fFlags, ppNetwork));
3842
3843 /* just pro forma validation, the caller is internal. */
3844 AssertPtr(pIntNet);
3845 AssertPtr(pSession);
3846 AssertPtr(pszNetwork);
3847 Assert(enmTrunkType > kIntNetTrunkType_Invalid && enmTrunkType < kIntNetTrunkType_End);
3848 AssertPtr(pszTrunk);
3849 Assert(!(fFlags & ~INTNET_OPEN_FLAGS_MASK));
3850 AssertPtr(ppNetwork);
3851 *ppNetwork = NULL;
3852
3853 /*
3854 * Allocate and initialize.
3855 */
3856 size_t cb = sizeof(INTNETNETWORK);
3857 if (fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
3858 cb += INTNETNETWORK_TMP_SIZE + 64;
3859 PINTNETNETWORK pNew = (PINTNETNETWORK)RTMemAllocZ(cb);
3860 if (!pNew)
3861 return VERR_NO_MEMORY;
3862 int rc = RTSemFastMutexCreate(&pNew->FastMutex);
3863 if (RT_SUCCESS(rc))
3864 {
3865 //pNew->pIFs = NULL;
3866 pNew->pIntNet = pIntNet;
3867 //pNew->cActiveIFs = 0;
3868 pNew->fFlags = fFlags;
3869 size_t cchName = strlen(pszNetwork);
3870 pNew->cchName = (uint8_t)cchName;
3871 Assert(cchName && cchName < sizeof(pNew->szName)); /* caller's responsibility. */
3872 memcpy(pNew->szName, pszNetwork, cchName); /* '\0' by alloc. */
3873 pNew->enmTrunkType = enmTrunkType;
3874 Assert(strlen(pszTrunk) < sizeof(pNew->szTrunk)); /* caller's responsibility. */
3875 strcpy(pNew->szTrunk, pszTrunk);
3876 if (fFlags & INTNET_OPEN_FLAGS_SHARED_MAC_ON_WIRE)
3877 pNew->pbTmp = RT_ALIGN_PT(pNew + 1, 64, uint8_t *);
3878 //else
3879 // pNew->pbTmp = NULL;
3880
3881 /*
3882 * Register the object in the current session and link it into the network list.
3883 */
3884 pNew->pvObj = SUPR0ObjRegister(pSession, SUPDRVOBJTYPE_INTERNAL_NETWORK, intnetR0NetworkDestruct, pNew, pIntNet);
3885 if (pNew->pvObj)
3886 {
3887 pNew->pNext = pIntNet->pNetworks;
3888 pIntNet->pNetworks = pNew;
3889
3890 /*
3891 * Check if the current session is actually allowed to create and open
3892 * the network. It is possible to implement network name based policies
3893 * and these must be checked now. SUPR0ObjRegister does no such checks.
3894 */
3895 rc = SUPR0ObjVerifyAccess(pNew->pvObj, pSession, pNew->szName);
3896 if (RT_SUCCESS(rc))
3897 {
3898 /*
3899 * Connect the trunk.
3900 */
3901 rc = intnetR0NetworkCreateTrunkIf(pNew, pSession);
3902 if (RT_SUCCESS(rc))
3903 {
3904 *ppNetwork = pNew;
3905 LogFlow(("intnetR0CreateNetwork: returns VINF_SUCCESS *ppNetwork=%p\n", pNew));
3906 return VINF_SUCCESS;
3907 }
3908 }
3909
3910 /*
3911 * We unlink it here so it cannot be opened when the caller leaves
3912 * INTNET::FastMutex before dereferencing it.
3913 */
3914 Assert(pIntNet->pNetworks == pNew);
3915 pIntNet->pNetworks = pNew->pNext;
3916 pNew->pNext = NULL;
3917
3918 *ppNetwork = pNew;
3919 LogFlow(("intnetR0CreateNetwork: returns %Rrc\n", rc));
3920 return rc;
3921 }
3922 rc = VERR_NO_MEMORY;
3923
3924 RTSemFastMutexDestroy(pNew->FastMutex);
3925 pNew->FastMutex = NIL_RTSEMFASTMUTEX;
3926 }
3927 RTMemFree(pNew);
3928 LogFlow(("intnetR0CreateNetwork: returns %Rrc\n", rc));
3929 return rc;
3930}
3931
3932
3933/**
3934 * Opens a network interface and connects it to the specified network.
3935 *
3936 * @returns VBox status code.
3937 * @param pIntNet The internal network instance.
3938 * @param pSession The session handle.
3939 * @param pszNetwork The network name.
3940 * @param enmTrunkType The trunk type.
3941 * @param pszTrunk The trunk name. Its meaning is specfic to the type.
3942 * @param fFlags Flags, see INTNET_OPEN_FLAGS_*.
3943 * @param fRestrictAccess Whether new participants should be subjected to access check or not.
3944 * @param cbSend The send buffer size.
3945 * @param cbRecv The receive buffer size.
3946 * @param phIf Where to store the handle to the network interface.
3947 */
3948INTNETR0DECL(int) INTNETR0Open(PINTNET pIntNet, PSUPDRVSESSION pSession, const char *pszNetwork,
3949 INTNETTRUNKTYPE enmTrunkType, const char *pszTrunk, uint32_t fFlags,
3950 unsigned cbSend, unsigned cbRecv, PINTNETIFHANDLE phIf)
3951{
3952 LogFlow(("INTNETR0Open: pIntNet=%p pSession=%p pszNetwork=%p:{%s} enmTrunkType=%d pszTrunk=%p:{%s} fFlags=%#x cbSend=%u cbRecv=%u phIf=%p\n",
3953 pIntNet, pSession, pszNetwork, pszNetwork, enmTrunkType, pszTrunk, pszTrunk, fFlags, cbSend, cbRecv, phIf));
3954
3955 /*
3956 * Validate input.
3957 */
3958 AssertPtrReturn(pIntNet, VERR_INVALID_PARAMETER);
3959
3960 AssertPtrReturn(pszNetwork, VERR_INVALID_PARAMETER);
3961 const char *pszNetworkEnd = (const char *)memchr(pszNetwork, '\0', INTNET_MAX_NETWORK_NAME);
3962 AssertReturn(pszNetworkEnd, VERR_INVALID_PARAMETER);
3963 size_t cchNetwork = pszNetworkEnd - pszNetwork;
3964 AssertReturn(cchNetwork, VERR_INVALID_PARAMETER);
3965
3966 if (pszTrunk)
3967 {
3968 AssertPtrReturn(pszTrunk, VERR_INVALID_PARAMETER);
3969 const char *pszTrunkEnd = (const char *)memchr(pszTrunk, '\0', INTNET_MAX_TRUNK_NAME);
3970 AssertReturn(pszTrunkEnd, VERR_INVALID_PARAMETER);
3971 }
3972 else
3973 pszTrunk = "";
3974
3975 AssertMsgReturn(enmTrunkType > kIntNetTrunkType_Invalid && enmTrunkType < kIntNetTrunkType_End,
3976 ("%d\n", enmTrunkType), VERR_INVALID_PARAMETER);
3977 switch (enmTrunkType)
3978 {
3979 case kIntNetTrunkType_None:
3980 case kIntNetTrunkType_WhateverNone:
3981 if (*pszTrunk)
3982 return VERR_INVALID_PARAMETER;
3983 break;
3984
3985 case kIntNetTrunkType_NetFlt:
3986 case kIntNetTrunkType_NetAdp:
3987 if (!*pszTrunk)
3988 return VERR_INVALID_PARAMETER;
3989 break;
3990
3991 default:
3992 return VERR_NOT_IMPLEMENTED;
3993 }
3994
3995 AssertMsgReturn(!(fFlags & ~INTNET_OPEN_FLAGS_MASK), ("%#x\n", fFlags), VERR_INVALID_PARAMETER);
3996 AssertPtrReturn(phIf, VERR_INVALID_PARAMETER);
3997
3998 /*
3999 * Acquire the mutex to serialize open/create.
4000 */
4001 int rc = RTSemFastMutexRequest(pIntNet->FastMutex);
4002 if (RT_FAILURE(rc))
4003 return rc;
4004
4005 /*
4006 * Try open / create the network and create an interface on it for the
4007 * caller to use.
4008 *
4009 * Note! Because of the destructors grabbing INTNET::FastMutex and us being
4010 * required to own this semaphore for the entire network opening / creation
4011 * and interface creation sequence, intnetR0CreateNetwork will have to
4012 * defer the network cleanup to us on failure.
4013 */
4014 PINTNETNETWORK pNetwork = NULL;
4015 rc = intnetR0OpenNetwork(pIntNet, pSession, pszNetwork, enmTrunkType, pszTrunk, fFlags, &pNetwork);
4016 if (RT_SUCCESS(rc) || rc == VERR_NOT_FOUND)
4017 {
4018 bool fCloseNetwork = true;
4019 bool fNewNet = rc == VERR_NOT_FOUND;
4020 if (fNewNet)
4021 rc = intnetR0CreateNetwork(pIntNet, pSession, pszNetwork, enmTrunkType, pszTrunk, fFlags, &pNetwork);
4022 if (RT_SUCCESS(rc))
4023 rc = intnetR0NetworkCreateIf(pNetwork, pSession, cbSend, cbRecv, &fCloseNetwork, phIf);
4024
4025 RTSemFastMutexRelease(pIntNet->FastMutex);
4026
4027 if (RT_FAILURE(rc))
4028 {
4029 if(pNetwork && fCloseNetwork)
4030 intnetR0NetworkClose(pNetwork, pSession);
4031 }
4032 else if(!fNewNet)
4033 rc = VINF_ALREADY_INITIALIZED;
4034 }
4035 else
4036 RTSemFastMutexRelease(pIntNet->FastMutex);
4037
4038 LogFlow(("INTNETR0Open: return %Rrc *phIf=%RX32\n", rc, *phIf));
4039 return rc;
4040}
4041
4042
4043/**
4044 * VMMR0 request wrapper for GMMR0MapUnmapChunk.
4045 *
4046 * @returns see GMMR0MapUnmapChunk.
4047 * @param pIntNet The internal networking instance.
4048 * @param pSession The caller's session.
4049 * @param pReq The request packet.
4050 */
4051INTNETR0DECL(int) INTNETR0OpenReq(PINTNET pIntNet, PSUPDRVSESSION pSession, PINTNETOPENREQ pReq)
4052{
4053 if (RT_UNLIKELY(pReq->Hdr.cbReq != sizeof(*pReq)))
4054 return VERR_INVALID_PARAMETER;
4055 return INTNETR0Open(pIntNet, pSession, &pReq->szNetwork[0], pReq->enmTrunkType, pReq->szTrunk,
4056 pReq->fFlags, pReq->cbSend, pReq->cbRecv, &pReq->hIf);
4057}
4058
4059
4060/**
4061 * Destroys an instance of the Ring-0 internal networking service.
4062 *
4063 * @param pIntNet Pointer to the instance data.
4064 */
4065INTNETR0DECL(void) INTNETR0Destroy(PINTNET pIntNet)
4066{
4067 LogFlow(("INTNETR0Destroy: pIntNet=%p\n", pIntNet));
4068
4069 /*
4070 * Allow NULL pointers.
4071 */
4072 if (!pIntNet)
4073 return;
4074 AssertPtrReturnVoid(pIntNet);
4075
4076 /*
4077 * There is not supposed to be any networks hanging around at this time.
4078 */
4079 Assert(pIntNet->pNetworks == NULL);
4080 if (pIntNet->FastMutex != NIL_RTSEMFASTMUTEX)
4081 {
4082 RTSemFastMutexDestroy(pIntNet->FastMutex);
4083 pIntNet->FastMutex = NIL_RTSEMFASTMUTEX;
4084 }
4085 if (pIntNet->hHtIfs != NIL_RTHANDLETABLE)
4086 {
4087 /** @todo does it make sense to have a deleter here? */
4088 RTHandleTableDestroy(pIntNet->hHtIfs, NULL, NULL);
4089 pIntNet->hHtIfs = NIL_RTHANDLETABLE;
4090 }
4091
4092 RTMemFree(pIntNet);
4093}
4094
4095
4096/**
4097 * Create an instance of the Ring-0 internal networking service.
4098 *
4099 * @returns VBox status code.
4100 * @param ppIntNet Where to store the instance pointer.
4101 */
4102INTNETR0DECL(int) INTNETR0Create(PINTNET *ppIntNet)
4103{
4104 LogFlow(("INTNETR0Create: ppIntNet=%p\n", ppIntNet));
4105 int rc = VERR_NO_MEMORY;
4106 PINTNET pIntNet = (PINTNET)RTMemAllocZ(sizeof(*pIntNet));
4107 if (pIntNet)
4108 {
4109 //pIntNet->pNetworks = NULL;
4110
4111 rc = RTSemFastMutexCreate(&pIntNet->FastMutex);
4112 if (RT_SUCCESS(rc))
4113 {
4114 rc = RTHandleTableCreateEx(&pIntNet->hHtIfs, RTHANDLETABLE_FLAGS_LOCKED | RTHANDLETABLE_FLAGS_CONTEXT,
4115 UINT32_C(0x8ffe0000), 4096, intnetR0IfRetainHandle, NULL);
4116 if (RT_SUCCESS(rc))
4117 {
4118 *ppIntNet = pIntNet;
4119 LogFlow(("INTNETR0Create: returns VINF_SUCCESS *ppIntNet=%p\n", pIntNet));
4120 return VINF_SUCCESS;
4121 }
4122
4123 RTSemFastMutexDestroy(pIntNet->FastMutex);
4124 }
4125 RTMemFree(pIntNet);
4126 }
4127 *ppIntNet = NULL;
4128 LogFlow(("INTNETR0Create: returns %Rrc\n", rc));
4129 return rc;
4130}
4131
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