VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/VBoxNetFlt/win/drv/VBoxNetFltRt-win.cpp@ 58956

Last change on this file since 58956 was 57937, checked in by vboxsync, 9 years ago

HostDrivers/NetFlt: added timeout in filter unload for unregistration of factory

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 110.0 KB
Line 
1/* $Id: VBoxNetFltRt-win.cpp 57937 2015-09-29 11:33:00Z vboxsync $ */
2/** @file
3 * VBoxNetFltRt-win.cpp - Bridged Networking Driver, Windows Specific Code.
4 * NetFlt Runtime
5 */
6/*
7 * Copyright (C) 2011-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#include "VBoxNetFltCmn-win.h"
18#include <VBox/intnetinline.h>
19#include <iprt/thread.h>
20
21RT_C_DECLS_BEGIN
22#include <tdikrnl.h>
23RT_C_DECLS_END
24#include <mstcpip.h>
25
26/** represents the job element of the job queue
27 * see comments for VBOXNETFLT_JOB_QUEUE */
28typedef struct VBOXNETFLT_JOB
29{
30 /** link in the job queue */
31 LIST_ENTRY ListEntry;
32 /** job function to be executed */
33 PFNVBOXNETFLT_JOB_ROUTINE pfnRoutine;
34 /** parameter to be passed to the job function */
35 PVOID pContext;
36 /** event that will be fired on job completion */
37 KEVENT CompletionEvent;
38 /** true if the job manager should use the completion even for completion indication, false-otherwise*/
39 bool bUseCompletionEvent;
40} VBOXNETFLT_JOB, *PVBOXNETFLT_JOB;
41
42/**
43 * represents the queue of jobs processed by the worker thread
44 *
45 * we use the thread to process tasks which are required to be done at passive level
46 * our callbacks may be called at APC level by IntNet, there are some tasks that we can not create at APC,
47 * e.g. thread creation. This is why we schedule such jobs to the worker thread working at passive level
48 */
49typedef struct VBOXNETFLT_JOB_QUEUE
50{
51 /* jobs */
52 LIST_ENTRY Jobs;
53 /* we are using ExInterlocked..List functions to access the jobs list */
54 KSPIN_LOCK Lock;
55 /** this event is used to initiate a job worker thread kill */
56 KEVENT KillEvent;
57 /** this event is used to notify a worker thread that jobs are added to the queue */
58 KEVENT NotifyEvent;
59 /** worker thread */
60 PKTHREAD pThread;
61} VBOXNETFLT_JOB_QUEUE, *PVBOXNETFLT_JOB_QUEUE;
62
63typedef struct _CREATE_INSTANCE_CONTEXT
64{
65#ifndef VBOXNETADP
66 PNDIS_STRING pOurName;
67 PNDIS_STRING pBindToName;
68#else
69 NDIS_HANDLE hMiniportAdapter;
70 NDIS_HANDLE hWrapperConfigurationContext;
71#endif
72 NDIS_STATUS Status;
73}CREATE_INSTANCE_CONTEXT, *PCREATE_INSTANCE_CONTEXT;
74
75/*contexts used for our jobs */
76/* Attach context */
77typedef struct _ATTACH_INFO
78{
79 PVBOXNETFLTINS pNetFltIf;
80 PCREATE_INSTANCE_CONTEXT pCreateContext;
81 bool fRediscovery;
82 int Status;
83}ATTACH_INFO, *PATTACH_INFO;
84
85/* general worker context */
86typedef struct _WORKER_INFO
87{
88 PVBOXNETFLTINS pNetFltIf;
89 int Status;
90}WORKER_INFO, *PWORKER_INFO;
91
92/* idc initialization */
93typedef struct _INIT_IDC_INFO
94{
95 VBOXNETFLT_JOB Job;
96 bool bInitialized;
97 volatile bool bStop;
98 volatile int rc;
99 KEVENT hCompletionEvent;
100}INIT_IDC_INFO, *PINIT_IDC_INFO;
101
102
103/** globals */
104/** global job queue. some operations are required to be done at passive level, e.g. thread creation, adapter bind/unbind initiation,
105 * while IntNet typically calls us APC_LEVEL, so we just create a system thread in our DriverEntry and enqueue the jobs to that thread */
106static VBOXNETFLT_JOB_QUEUE g_VBoxJobQueue;
107volatile static bool g_bVBoxIdcInitialized;
108INIT_IDC_INFO g_VBoxInitIdcInfo;
109/**
110 * The (common) global data.
111 */
112static VBOXNETFLTGLOBALS g_VBoxNetFltGlobals;
113/* win-specific global data */
114VBOXNETFLTGLOBALS_WIN g_VBoxNetFltGlobalsWin = {0};
115
116#define LIST_ENTRY_2_JOB(pListEntry) \
117 ( (PVBOXNETFLT_JOB)((uint8_t *)(pListEntry) - RT_OFFSETOF(VBOXNETFLT_JOB, ListEntry)) )
118
119static int vboxNetFltWinAttachToInterface(PVBOXNETFLTINS pThis, void * pContext, bool fRediscovery);
120static int vboxNetFltWinConnectIt(PVBOXNETFLTINS pThis);
121static int vboxNetFltWinFiniIdc();
122static void vboxNetFltWinFiniNetFltBase();
123static int vboxNetFltWinInitNetFltBase();
124static int vboxNetFltWinFiniNetFlt();
125static int vboxNetFltWinStartInitIdcProbing();
126static int vboxNetFltWinStopInitIdcProbing();
127
128/** makes the current thread to sleep for the given number of miliseconds */
129DECLHIDDEN(void) vboxNetFltWinSleep(ULONG milis)
130{
131 RTThreadSleep(milis);
132}
133
134/** wait for the given device to be dereferenced */
135DECLHIDDEN(void) vboxNetFltWinWaitDereference(PVBOXNETFLT_WINIF_DEVICE pState)
136{
137#ifdef DEBUG
138 uint64_t StartNanoTS = RTTimeSystemNanoTS();
139 uint64_t CurNanoTS;
140#endif
141 Assert(KeGetCurrentIrql() < DISPATCH_LEVEL);
142
143 while (ASMAtomicUoReadU32((volatile uint32_t *)&pState->cReferences))
144 {
145 vboxNetFltWinSleep(2);
146#ifdef DEBUG
147 CurNanoTS = RTTimeSystemNanoTS();
148 if (CurNanoTS - StartNanoTS > 20000000)
149 {
150 LogRel(("device not idle"));
151 AssertFailed();
152// break;
153 }
154#endif
155 }
156}
157
158/**
159 * mem functions
160 */
161/* allocates and zeroes the nonpaged memory of a given size */
162DECLHIDDEN(NDIS_STATUS) vboxNetFltWinMemAlloc(PVOID* ppMemBuf, UINT cbLength)
163{
164#ifdef DEBUG_NETFLT_USE_EXALLOC
165 *ppMemBuf = ExAllocatePoolWithTag(NonPagedPool, cbLength, VBOXNETFLT_MEM_TAG);
166 if (*ppMemBuf)
167 {
168 NdisZeroMemory(*ppMemBuf, cbLength);
169 return NDIS_STATUS_SUCCESS;
170 }
171 return NDIS_STATUS_FAILURE;
172#else
173 NDIS_STATUS fStatus = NdisAllocateMemoryWithTag(ppMemBuf, cbLength, VBOXNETFLT_MEM_TAG);
174 if (fStatus == NDIS_STATUS_SUCCESS)
175 {
176 NdisZeroMemory(*ppMemBuf, cbLength);
177 }
178 return fStatus;
179#endif
180}
181
182/* frees memory allocated with vboxNetFltWinMemAlloc */
183DECLHIDDEN(void) vboxNetFltWinMemFree(PVOID pvMemBuf)
184{
185#ifdef DEBUG_NETFLT_USE_EXALLOC
186 ExFreePool(pvMemBuf);
187#else
188 NdisFreeMemory(pvMemBuf, 0, 0);
189#endif
190}
191
192#ifndef VBOXNETFLT_NO_PACKET_QUEUE
193
194/* initializes packet info pool and allocates the cSize packet infos for the pool */
195static NDIS_STATUS vboxNetFltWinPpAllocatePacketInfoPool(PVBOXNETFLT_PACKET_INFO_POOL pPool, UINT cSize)
196{
197 UINT cbBufSize = sizeof(PACKET_INFO)*cSize;
198 PACKET_INFO * pPacketInfos;
199 NDIS_STATUS fStatus;
200 UINT i;
201
202 Assert(cSize > 0);
203
204 INIT_INTERLOCKED_PACKET_QUEUE(&pPool->Queue);
205
206 fStatus = vboxNetFltWinMemAlloc((PVOID*)&pPacketInfos, cbBufSize);
207
208 if (fStatus == NDIS_STATUS_SUCCESS)
209 {
210 PVBOXNETFLTPACKET_INFO pInfo;
211 pPool->pBuffer = pPacketInfos;
212
213 for (i = 0; i < cSize; i++)
214 {
215 pInfo = &pPacketInfos[i];
216 vboxNetFltWinQuEnqueueTail(&pPool->Queue.Queue, pInfo);
217 pInfo->pPool = pPool;
218 }
219 }
220 else
221 {
222 AssertFailed();
223 }
224
225 return fStatus;
226}
227
228/* frees the packet info pool */
229VOID vboxNetFltWinPpFreePacketInfoPool(PVBOXNETFLT_PACKET_INFO_POOL pPool)
230{
231 vboxNetFltWinMemFree(pPool->pBuffer);
232
233 FINI_INTERLOCKED_PACKET_QUEUE(&pPool->Queue)
234}
235
236#endif
237
238/**
239 * copies one string to another. in case the destination string size is not enough to hold the complete source string
240 * does nothing and returns NDIS_STATUS_RESOURCES .
241 */
242DECLHIDDEN(NDIS_STATUS) vboxNetFltWinCopyString(PNDIS_STRING pDst, PNDIS_STRING pSrc)
243{
244 NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
245
246 if (pDst != pSrc)
247 {
248 if (pDst->MaximumLength < pSrc->Length)
249 {
250 AssertFailed();
251 Status = NDIS_STATUS_RESOURCES;
252 }
253 else
254 {
255 pDst->Length = pSrc->Length;
256
257 if (pDst->Buffer != pSrc->Buffer)
258 {
259 NdisMoveMemory(pDst->Buffer, pSrc->Buffer, pSrc->Length);
260 }
261 }
262 }
263 return Status;
264}
265
266/************************************************************************************
267 * PINTNETSG pSG manipulation functions
268 ************************************************************************************/
269
270/* moves the contents of the given NDIS_BUFFER and all other buffers chained to it to the PINTNETSG
271 * the PINTNETSG is expected to contain one segment whose bugger is large enough to maintain
272 * the contents of the given NDIS_BUFFER and all other buffers chained to it */
273static NDIS_STATUS vboxNetFltWinNdisBufferMoveToSG0(PNDIS_BUFFER pBuffer, PINTNETSG pSG)
274{
275 UINT cSegs = 0;
276 PINTNETSEG paSeg;
277 uint8_t * ptr;
278 PVOID pVirtualAddress;
279 UINT cbCurrentLength;
280 NDIS_STATUS fStatus = NDIS_STATUS_SUCCESS;
281
282 Assert(pSG->cSegsAlloc == 1);
283
284 paSeg = pSG->aSegs;
285 ptr = (uint8_t*)paSeg->pv;
286 paSeg->cb = 0;
287 paSeg->Phys = NIL_RTHCPHYS;
288 pSG->cbTotal = 0;
289
290 Assert(paSeg->pv);
291
292 while (pBuffer)
293 {
294 NdisQueryBufferSafe(pBuffer, &pVirtualAddress, &cbCurrentLength, NormalPagePriority);
295
296 if (!pVirtualAddress)
297 {
298 fStatus = NDIS_STATUS_FAILURE;
299 break;
300 }
301
302 pSG->cbTotal += cbCurrentLength;
303 paSeg->cb += cbCurrentLength;
304 NdisMoveMemory(ptr, pVirtualAddress, cbCurrentLength);
305 ptr += cbCurrentLength;
306
307 NdisGetNextBuffer(pBuffer, &pBuffer);
308 }
309
310 if (fStatus == NDIS_STATUS_SUCCESS)
311 {
312 pSG->cSegsUsed = 1;
313 Assert(pSG->cbTotal == paSeg->cb);
314 }
315 return fStatus;
316}
317
318/* converts the PNDIS_BUFFER to PINTNETSG by making the PINTNETSG segments to point to the memory buffers the
319 * ndis buffer(s) point to (as opposed to vboxNetFltWinNdisBufferMoveToSG0 which copies the memory from ndis buffers(s) to PINTNETSG) */
320static NDIS_STATUS vboxNetFltWinNdisBuffersToSG(PNDIS_BUFFER pBuffer, PINTNETSG pSG)
321{
322 UINT cSegs = 0;
323 NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
324 PVOID pVirtualAddress;
325 UINT cbCurrentLength;
326
327 while (pBuffer)
328 {
329 NdisQueryBufferSafe(pBuffer, &pVirtualAddress, &cbCurrentLength, NormalPagePriority);
330
331 if (!pVirtualAddress)
332 {
333 Status = NDIS_STATUS_FAILURE;
334 break;
335 }
336
337 pSG->cbTotal += cbCurrentLength;
338 pSG->aSegs[cSegs].cb = cbCurrentLength;
339 pSG->aSegs[cSegs].pv = pVirtualAddress;
340 pSG->aSegs[cSegs].Phys = NIL_RTHCPHYS;
341 cSegs++;
342
343 NdisGetNextBuffer(pBuffer, &pBuffer);
344 }
345
346 AssertFatal(cSegs <= pSG->cSegsAlloc);
347
348 if (Status == NDIS_STATUS_SUCCESS)
349 {
350 pSG->cSegsUsed = cSegs;
351 }
352
353 return Status;
354}
355
356static void vboxNetFltWinDeleteSG(PINTNETSG pSG)
357{
358 vboxNetFltWinMemFree(pSG);
359}
360
361static PINTNETSG vboxNetFltWinCreateSG(uint32_t cSegs)
362{
363 PINTNETSG pSG;
364 NTSTATUS Status = vboxNetFltWinMemAlloc((PVOID*)&pSG, RT_OFFSETOF(INTNETSG, aSegs[cSegs]));
365 if (Status == STATUS_SUCCESS)
366 {
367 IntNetSgInitTempSegs(pSG, 0 /*cbTotal*/, cSegs, 0 /*cSegsUsed*/);
368 return pSG;
369 }
370
371 return NULL;
372}
373
374/************************************************************************************
375 * packet queue functions
376 ************************************************************************************/
377#ifndef VBOXNETFLT_NO_PACKET_QUEUE
378#if !defined(VBOXNETADP)
379static NDIS_STATUS vboxNetFltWinQuPostPacket(PVBOXNETFLTINS pNetFlt, PNDIS_PACKET pPacket, PINTNETSG pSG, uint32_t fFlags
380# ifdef DEBUG_NETFLT_PACKETS
381 , PNDIS_PACKET pTmpPacket
382# endif
383 )
384{
385 NDIS_STATUS Status;
386 PNDIS_PACKET pMyPacket;
387 bool bSrcHost = fFlags & PACKET_SRC_HOST;
388
389 LogFlow(("posting packet back to driver stack..\n"));
390
391 if (!pPacket)
392 {
393 /* INTNETSG was in the packet queue, create a new NdisPacket from INTNETSG*/
394 pMyPacket = vboxNetFltWinNdisPacketFromSG(pNetFlt,
395 pSG, /* PINTNETSG */
396 pSG, /* PVOID pBufToFree */
397 bSrcHost, /* bool bToWire */
398 false); /* bool bCopyMemory */
399
400 Assert(pMyPacket);
401
402 NDIS_SET_PACKET_STATUS(pMyPacket, NDIS_STATUS_SUCCESS);
403
404 DBG_CHECK_PACKET_AND_SG(pMyPacket, pSG);
405
406#ifdef DEBUG_NETFLT_PACKETS
407 Assert(pTmpPacket);
408
409 DBG_CHECK_PACKET_AND_SG(pTmpPacket, pSG);
410
411 DBG_CHECK_PACKETS(pTmpPacket, pMyPacket);
412#endif
413
414 LogFlow(("non-ndis packet info, packet created (%p)\n", pMyPacket));
415 }
416 else
417 {
418 /* NDIS_PACKET was in the packet queue */
419 DBG_CHECK_PACKET_AND_SG(pPacket, pSG);
420
421 if (!(fFlags & PACKET_MINE))
422 {
423 /* the packet is the one that was passed to us in send/receive callback
424 * According to the DDK, we can not post it further,
425 * instead we should allocate our own packet.
426 * So, allocate our own packet (pMyPacket) and copy the packet info there */
427 if (bSrcHost)
428 {
429 Status = vboxNetFltWinPrepareSendPacket(pNetFlt, pPacket, &pMyPacket/*, true*/);
430 LogFlow(("packet from wire, packet created (%p)\n", pMyPacket));
431 }
432 else
433 {
434 Status = vboxNetFltWinPrepareRecvPacket(pNetFlt, pPacket, &pMyPacket, false);
435 LogFlow(("packet from wire, packet created (%p)\n", pMyPacket));
436 }
437 }
438 else
439 {
440 /* the packet enqueued is ours, simply assign pMyPacket and zero pPacket */
441 pMyPacket = pPacket;
442 pPacket = NULL;
443 }
444 Assert(pMyPacket);
445 }
446
447 if (pMyPacket)
448 {
449 /* we have successfully initialized our packet, post it to the host or to the wire */
450 if (bSrcHost)
451 {
452#if defined(DEBUG_NETFLT_PACKETS) || !defined(VBOX_LOOPBACK_USEFLAGS)
453 vboxNetFltWinLbPutSendPacket(pNetFlt, pMyPacket, false /* bFromIntNet */);
454#endif
455 NdisSend(&Status, pNetFlt->u.s.hBinding, pMyPacket);
456
457 if (Status != NDIS_STATUS_PENDING)
458 {
459#if defined(DEBUG_NETFLT_PACKETS) || !defined(VBOX_LOOPBACK_USEFLAGS)
460 /* the status is NOT pending, complete the packet */
461 bool bTmp = vboxNetFltWinLbRemoveSendPacket(pNetFlt, pMyPacket);
462 Assert(bTmp);
463#endif
464 if (pPacket)
465 {
466 LogFlow(("status is not pending, completing packet (%p)\n", pPacket));
467
468 NdisIMCopySendCompletePerPacketInfo (pPacket, pMyPacket);
469
470 NdisFreePacket(pMyPacket);
471 }
472 else
473 {
474 /* should never be here since the PINTNETSG is stored only when the underlying miniport
475 * indicates NDIS_STATUS_RESOURCES, we should never have this when processing
476 * the "from-host" packets */
477 AssertFailed();
478 LogFlow(("status is not pending, freeing myPacket (%p)\n", pMyPacket));
479 vboxNetFltWinFreeSGNdisPacket(pMyPacket, false);
480 }
481 }
482 }
483 else
484 {
485 NdisMIndicateReceivePacket(pNetFlt->u.s.hMiniport, &pMyPacket, 1);
486
487 Status = NDIS_STATUS_PENDING;
488 /* the packet receive completion is always indicated via MiniportReturnPacket */
489 }
490 }
491 else
492 {
493 /*we failed to create our packet */
494 AssertFailed();
495 Status = NDIS_STATUS_FAILURE;
496 }
497
498 return Status;
499}
500#endif
501
502static bool vboxNetFltWinQuProcessInfo(PVBOXNETFLTINS pNetFltIf, PPACKET_QUEUE_WORKER pWorker, PVOID pvPacket, const UINT fFlags)
503#else
504DECLHIDDEN(bool) vboxNetFltWinPostIntnet(PVBOXNETFLTINS pNetFltIf, PVOID pvPacket, const UINT fFlags)
505#endif
506{
507 PNDIS_PACKET pPacket = NULL;
508 PINTNETSG pSG = NULL;
509 NDIS_STATUS Status;
510#ifndef VBOXNETADP
511 bool bSrcHost;
512 bool bDropIt;
513# ifndef VBOXNETFLT_NO_PACKET_QUEUE
514 bool bPending;
515# endif
516#endif
517#ifdef VBOXNETFLT_NO_PACKET_QUEUE
518 bool bDeleteSG = false;
519#endif
520#ifdef DEBUG_NETFLT_PACKETS
521 /* packet used for matching */
522 PNDIS_PACKET pTmpPacket = NULL;
523#endif
524
525#ifndef VBOXNETADP
526 bSrcHost = (fFlags & VBOXNETFLT_PACKET_SRC_HOST) != 0;
527#endif
528
529 /* we first need to obtain the INTNETSG to be passed to intnet */
530
531 /* the queue may contain two "types" of packets:
532 * the NDIS_PACKET and the INTNETSG.
533 * I.e. on send/receive we typically enqueue the NDIS_PACKET passed to us by ndis,
534 * however in case our ProtocolReceive is called or the packet's status is set to NDIS_STSTUS_RESOURCES
535 * in ProtocolReceivePacket, we must return the packet immediately on ProtocolReceive*** exit
536 * In this case we allocate the INTNETSG, copy the ndis packet data there and enqueue it.
537 * In this case the packet info flags has the VBOXNETFLT_PACKET_SG fag set
538 *
539 * Besides that the NDIS_PACKET contained in the queue could be either the one passed to us in our send/receive callback
540 * or the one created by us. The latter is possible in case our ProtocolReceive callback is called and we call NdisTransferData
541 * in this case we need to allocate the packet the data to be transferred to.
542 * If the enqueued packet is the one allocated by us the VBOXNETFLT_PACKET_MINE flag is set
543 * */
544 if ((fFlags & VBOXNETFLT_PACKET_SG) == 0)
545 {
546 /* we have NDIS_PACKET enqueued, we need to convert it to INTNETSG to be passed to intnet */
547 PNDIS_BUFFER pCurrentBuffer = NULL;
548 UINT cBufferCount;
549 UINT uBytesCopied = 0;
550 UINT cbPacketLength;
551
552 pPacket = (PNDIS_PACKET)pvPacket;
553
554 LogFlow(("ndis packet info, packet (%p)\n", pPacket));
555
556 LogFlow(("preparing pSG"));
557 NdisQueryPacket(pPacket, NULL, &cBufferCount, &pCurrentBuffer, &cbPacketLength);
558 Assert(cBufferCount);
559
560#ifdef VBOXNETFLT_NO_PACKET_QUEUE
561 pSG = vboxNetFltWinCreateSG(cBufferCount);
562#else
563 /* we can not allocate the INTNETSG on stack since in this case we may get stack overflow
564 * somewhere outside of our driver (3 pages of system thread stack does not seem to be enough)
565 *
566 * since we have a "serialized" packet processing, i.e. all packets are being processed and passed
567 * to intnet by this thread, we just use one previously allocated INTNETSG which is stored in PVBOXNETFLTINS */
568 pSG = pWorker->pSG;
569
570 if (cBufferCount > pSG->cSegsAlloc)
571 {
572 pSG = vboxNetFltWinCreateSG(cBufferCount + 2);
573 if (pSG)
574 {
575 vboxNetFltWinDeleteSG(pWorker->pSG);
576 pWorker->pSG = pSG;
577 }
578 else
579 {
580 LogRel(("Failed to reallocate the pSG\n"));
581 }
582 }
583#endif
584
585 if (pSG)
586 {
587#ifdef VBOXNETFLT_NO_PACKET_QUEUE
588 bDeleteSG = true;
589#endif
590 /* reinitialize */
591 IntNetSgInitTempSegs(pSG, 0 /*cbTotal*/, pSG->cSegsAlloc, 0 /*cSegsUsed*/);
592
593 /* convert the ndis buffers to INTNETSG */
594 Status = vboxNetFltWinNdisBuffersToSG(pCurrentBuffer, pSG);
595 if (Status != NDIS_STATUS_SUCCESS)
596 {
597 pSG = NULL;
598 }
599 else
600 {
601 DBG_CHECK_PACKET_AND_SG(pPacket, pSG);
602 }
603 }
604 }
605 else
606 {
607 /* we have the INTNETSG enqueued. (see the above comment explaining why/when this may happen)
608 * just use the INTNETSG to pass it to intnet */
609#ifndef VBOXNETADP
610 /* the PINTNETSG is stored only when the underlying miniport
611 * indicates NDIS_STATUS_RESOURCES, we should never have this when processing
612 * the "from-host" packedts */
613 Assert(!bSrcHost);
614#endif
615 pSG = (PINTNETSG)pvPacket;
616
617 LogFlow(("not ndis packet info, pSG (%p)\n", pSG));
618 }
619
620#ifdef DEBUG_NETFLT_PACKETS
621 if (!pPacket && !pTmpPacket)
622 {
623 /* create tmp packet that woud be used for matching */
624 pTmpPacket = vboxNetFltWinNdisPacketFromSG(pNetFltIf,
625 pSG, /* PINTNETSG */
626 pSG, /* PVOID pBufToFree */
627 bSrcHost, /* bool bToWire */
628 true); /* bool bCopyMemory */
629
630 NDIS_SET_PACKET_STATUS(pTmpPacket, NDIS_STATUS_SUCCESS);
631
632 DBG_CHECK_PACKET_AND_SG(pTmpPacket, pSG);
633
634 Assert(pTmpPacket);
635 }
636#endif
637 do
638 {
639#ifndef VBOXNETADP
640 /* the pSG was successfully initialized, post it to the netFlt*/
641 bDropIt = pSG ? pNetFltIf->pSwitchPort->pfnRecv(pNetFltIf->pSwitchPort, NULL /* pvIf */, pSG,
642 bSrcHost ? INTNETTRUNKDIR_HOST : INTNETTRUNKDIR_WIRE
643 )
644 : false;
645#else
646 if (pSG)
647 {
648 pNetFltIf->pSwitchPort->pfnRecv(pNetFltIf->pSwitchPort, NULL /* pvIf */, pSG, INTNETTRUNKDIR_HOST);
649 STATISTIC_INCREASE(pNetFltIf->u.s.WinIf.cTxSuccess);
650 }
651 else
652 {
653 STATISTIC_INCREASE(pNetFltIf->u.s.WinIf.cTxError);
654 }
655#endif
656
657#ifndef VBOXNETFLT_NO_PACKET_QUEUE
658
659# if !defined(VBOXNETADP)
660 if (!bDropIt)
661 {
662 Status = vboxNetFltWinQuPostPacket(pNetFltIf, pPacket, pSG, fFlags
663# ifdef DEBUG_NETFLT_PACKETS
664 , pTmpPacket
665# endif
666 );
667
668 if (Status == NDIS_STATUS_PENDING)
669 {
670 /* we will process packet completion in the completion routine */
671 bPending = true;
672 break;
673 }
674 }
675 else
676# endif
677 {
678 Status = NDIS_STATUS_SUCCESS;
679 }
680
681 /* drop it */
682 if (pPacket)
683 {
684 if (!(fFlags & PACKET_MINE))
685 {
686# if !defined(VBOXNETADP)
687 /* complete the packets */
688 if (fFlags & PACKET_SRC_HOST)
689 {
690# endif
691/* NDIS_SET_PACKET_STATUS(pPacket, Status); */
692 NdisMSendComplete(pNetFltIf->u.s.hMiniport, pPacket, Status);
693# if !defined(VBOXNETADP)
694 }
695 else
696 {
697# endif
698# ifndef VBOXNETADP
699 NdisReturnPackets(&pPacket, 1);
700# endif
701# if !defined(VBOXNETADP)
702 }
703# endif
704 }
705 else
706 {
707 Assert(!(fFlags & PACKET_SRC_HOST));
708 vboxNetFltWinFreeSGNdisPacket(pPacket, true);
709 }
710 }
711 else
712 {
713 Assert(pSG);
714 vboxNetFltWinMemFree(pSG);
715 }
716# ifndef VBOXNETADP
717 bPending = false;
718# endif
719 } while (0);
720
721#ifdef DEBUG_NETFLT_PACKETS
722 if (pTmpPacket)
723 {
724 vboxNetFltWinFreeSGNdisPacket(pTmpPacket, true);
725 }
726#endif
727
728#ifndef VBOXNETADP
729 return bPending;
730#else
731 return false;
732#endif
733#else /* #ifdef VBOXNETFLT_NO_PACKET_QUEUE */
734 } while (0);
735
736 if (bDeleteSG)
737 vboxNetFltWinMemFree(pSG);
738
739# ifndef VBOXNETADP
740 return bDropIt;
741# else
742 return true;
743# endif
744#endif
745}
746#ifndef VBOXNETFLT_NO_PACKET_QUEUE
747/*
748 * thread start function for the thread which processes the packets enqueued in our send and receive callbacks called by ndis
749 *
750 * ndis calls us at DISPATCH_LEVEL, while IntNet is using kernel functions which require Irql<DISPATCH_LEVEL
751 * this is why we can not immediately post packets to IntNet from our sen/receive callbacks
752 * instead we put the incoming packets to the queue and maintain the system thread running at passive level
753 * which processes the queue and posts the packets to IntNet, and further to the host or to the wire.
754 */
755static VOID vboxNetFltWinQuPacketQueueWorkerThreadProc(PVBOXNETFLTINS pNetFltIf)
756{
757 bool fResume = true;
758 NTSTATUS fStatus;
759 PPACKET_QUEUE_WORKER pWorker = &pNetFltIf->u.s.PacketQueueWorker;
760
761 PVOID apEvents[] = {
762 (PVOID)&pWorker->KillEvent,
763 (PVOID)&pWorker->NotifyEvent
764 };
765
766 while (fResume)
767 {
768 uint32_t cNumProcessed;
769 uint32_t cNumPostedToHostWire;
770
771 fStatus = KeWaitForMultipleObjects(RT_ELEMENTS(apEvents), apEvents, WaitAny, Executive, KernelMode, FALSE, NULL, NULL);
772 if (!NT_SUCCESS(fStatus) || fStatus == STATUS_WAIT_0)
773 {
774 /* "kill" event was set
775 * will process queued packets and exit */
776 fResume = false;
777 }
778
779 LogFlow(("processing vboxNetFltWinQuPacketQueueWorkerThreadProc\n"));
780
781 cNumProcessed = 0;
782 cNumPostedToHostWire = 0;
783
784 do
785 {
786 PVBOXNETFLTPACKET_INFO pInfo;
787
788#ifdef DEBUG_NETFLT_PACKETS
789 /* packet used for matching */
790 PNDIS_PACKET pTmpPacket = NULL;
791#endif
792
793 /*TODO: FIXME: !!! the better approach for performance would be to dequeue all packets at once
794 * and then go through all dequeued packets
795 * the same should be done for enqueue !!! */
796 pInfo = vboxNetFltWinQuInterlockedDequeueHead(&pWorker->PacketQueue);
797
798 if (!pInfo)
799 {
800 break;
801 }
802
803 LogFlow(("found info (0x%p)\n", pInfo));
804
805 if (vboxNetFltWinQuProcessInfo(pNetFltIf, pWorker, pInfo->pPacket, pInfo->fFlags))
806 {
807 cNumPostedToHostWire++;
808 }
809
810 vboxNetFltWinPpFreePacketInfo(pInfo);
811
812 cNumProcessed++;
813 } while (TRUE);
814
815 if (cNumProcessed)
816 {
817 vboxNetFltWinDecReferenceNetFlt(pNetFltIf, cNumProcessed);
818
819 Assert(cNumProcessed >= cNumPostedToHostWire);
820
821 if (cNumProcessed != cNumPostedToHostWire)
822 {
823 vboxNetFltWinDecReferenceWinIf(pNetFltIf, cNumProcessed - cNumPostedToHostWire);
824 }
825 }
826 }
827
828 PsTerminateSystemThread(STATUS_SUCCESS);
829}
830#endif
831/**
832 * thread start function for the job processing thread
833 *
834 * see comments for PVBOXNETFLT_JOB_QUEUE
835 */
836static VOID vboxNetFltWinJobWorkerThreadProc(PVBOXNETFLT_JOB_QUEUE pQueue)
837{
838 bool fResume = true;
839 NTSTATUS Status;
840
841 PVOID apEvents[] = {
842 (PVOID)&pQueue->KillEvent,
843 (PVOID)&pQueue->NotifyEvent,
844 };
845
846 do
847 {
848 Status = KeWaitForMultipleObjects(RT_ELEMENTS(apEvents), apEvents, WaitAny, Executive, KernelMode, FALSE, NULL, NULL);
849 Assert(NT_SUCCESS(Status));
850 if (!NT_SUCCESS(Status) || Status == STATUS_WAIT_0)
851 {
852 /* will process queued jobs and exit */
853 Assert(Status == STATUS_WAIT_0);
854 fResume = false;
855 }
856
857 do
858 {
859 PLIST_ENTRY pJobEntry = ExInterlockedRemoveHeadList(&pQueue->Jobs, &pQueue->Lock);
860 PVBOXNETFLT_JOB pJob;
861
862 if (!pJobEntry)
863 break;
864
865 pJob = LIST_ENTRY_2_JOB(pJobEntry);
866
867 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
868 pJob->pfnRoutine(pJob->pContext);
869 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
870
871 if (pJob->bUseCompletionEvent)
872 {
873 KeSetEvent(&pJob->CompletionEvent, 1, FALSE);
874 }
875 } while (TRUE);
876 } while (fResume);
877
878 Assert(Status == STATUS_WAIT_0);
879
880 PsTerminateSystemThread(STATUS_SUCCESS);
881}
882
883/**
884 * enqueues the job to the job queue to be processed by the job worker thread
885 * see comments for PVBOXNETFLT_JOB_QUEUE
886 */
887static VOID vboxNetFltWinJobEnqueueJob(PVBOXNETFLT_JOB_QUEUE pQueue, PVBOXNETFLT_JOB pJob, bool bEnqueueHead)
888{
889 if (bEnqueueHead)
890 {
891 ExInterlockedInsertHeadList(&pQueue->Jobs, &pJob->ListEntry, &pQueue->Lock);
892 }
893 else
894 {
895 ExInterlockedInsertTailList(&pQueue->Jobs, &pJob->ListEntry, &pQueue->Lock);
896 }
897
898 KeSetEvent(&pQueue->NotifyEvent, 1, FALSE);
899}
900
901DECLINLINE(VOID) vboxNetFltWinJobInit(PVBOXNETFLT_JOB pJob, PFNVBOXNETFLT_JOB_ROUTINE pfnRoutine, PVOID pContext, bool bUseEvent)
902{
903 pJob->pfnRoutine = pfnRoutine;
904 pJob->pContext = pContext;
905 pJob->bUseCompletionEvent = bUseEvent;
906 if (bUseEvent)
907 KeInitializeEvent(&pJob->CompletionEvent, NotificationEvent, FALSE);
908}
909
910/**
911 * enqueues the job to the job queue to be processed by the job worker thread and
912 * blocks until the job is done
913 * see comments for PVBOXNETFLT_JOB_QUEUE
914 */
915static VOID vboxNetFltWinJobSynchExec(PVBOXNETFLT_JOB_QUEUE pQueue, PFNVBOXNETFLT_JOB_ROUTINE pfnRoutine, PVOID pContext)
916{
917 VBOXNETFLT_JOB Job;
918
919 Assert(KeGetCurrentIrql() < DISPATCH_LEVEL);
920
921 vboxNetFltWinJobInit(&Job, pfnRoutine, pContext, true);
922
923 vboxNetFltWinJobEnqueueJob(pQueue, &Job, false);
924
925 KeWaitForSingleObject(&Job.CompletionEvent, Executive, KernelMode, FALSE, NULL);
926}
927
928/**
929 * enqueues the job to be processed by the job worker thread at passive level and
930 * blocks until the job is done
931 */
932DECLHIDDEN(VOID) vboxNetFltWinJobSynchExecAtPassive(PFNVBOXNETFLT_JOB_ROUTINE pfnRoutine, PVOID pContext)
933{
934 vboxNetFltWinJobSynchExec(&g_VBoxJobQueue, pfnRoutine, pContext);
935}
936
937/**
938 * helper function used for system thread creation
939 */
940static NTSTATUS vboxNetFltWinQuCreateSystemThread(PKTHREAD *ppThread, PKSTART_ROUTINE pfnStartRoutine, PVOID pvStartContext)
941{
942 OBJECT_ATTRIBUTES ObjectAttributes;
943 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
944
945 InitializeObjectAttributes(&ObjectAttributes, NULL, OBJ_KERNEL_HANDLE, NULL, NULL);
946
947 HANDLE hThread;
948 NTSTATUS Status = PsCreateSystemThread(&hThread, THREAD_ALL_ACCESS, &ObjectAttributes, NULL, NULL, (PKSTART_ROUTINE)pfnStartRoutine, pvStartContext);
949 Assert(Status == STATUS_SUCCESS);
950 if (Status == STATUS_SUCCESS)
951 {
952 Status = ObReferenceObjectByHandle(hThread, THREAD_ALL_ACCESS, NULL, KernelMode, (PVOID*)ppThread, NULL);
953 Assert(Status == STATUS_SUCCESS);
954 ZwClose(hThread);
955 if (Status == STATUS_SUCCESS)
956 {
957 return STATUS_SUCCESS;
958 }
959
960 /* @todo: how would we fail in this case ?*/
961 }
962 return Status;
963}
964
965/**
966 * initialize the job queue
967 * see comments for PVBOXNETFLT_JOB_QUEUE
968 */
969static NTSTATUS vboxNetFltWinJobInitQueue(PVBOXNETFLT_JOB_QUEUE pQueue)
970{
971 NTSTATUS fStatus;
972
973 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
974
975 NdisZeroMemory(pQueue, sizeof(VBOXNETFLT_JOB_QUEUE));
976
977 KeInitializeEvent(&pQueue->KillEvent, NotificationEvent, FALSE);
978
979 KeInitializeEvent(&pQueue->NotifyEvent, SynchronizationEvent, FALSE);
980
981 InitializeListHead(&pQueue->Jobs);
982
983 fStatus = vboxNetFltWinQuCreateSystemThread(&pQueue->pThread, (PKSTART_ROUTINE)vboxNetFltWinJobWorkerThreadProc, pQueue);
984 if (fStatus != STATUS_SUCCESS)
985 {
986 pQueue->pThread = NULL;
987 }
988 else
989 {
990 Assert(pQueue->pThread);
991 }
992
993 return fStatus;
994}
995
996/**
997 * deinitialize the job queue
998 * see comments for PVBOXNETFLT_JOB_QUEUE
999 */
1000static void vboxNetFltWinJobFiniQueue(PVBOXNETFLT_JOB_QUEUE pQueue)
1001{
1002 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
1003
1004 if (pQueue->pThread)
1005 {
1006 KeSetEvent(&pQueue->KillEvent, 0, FALSE);
1007
1008 KeWaitForSingleObject(pQueue->pThread, Executive,
1009 KernelMode, FALSE, NULL);
1010 }
1011}
1012
1013#ifndef VBOXNETFLT_NO_PACKET_QUEUE
1014
1015/**
1016 * initializes the packet queue
1017 * */
1018DECLHIDDEN(NTSTATUS) vboxNetFltWinQuInitPacketQueue(PVBOXNETFLTINS pInstance)
1019{
1020 NTSTATUS Status;
1021 PPACKET_QUEUE_WORKER pWorker = &pInstance->u.s.PacketQueueWorker;
1022
1023 AssertFatal(!pWorker->pSG);
1024
1025 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
1026
1027 KeInitializeEvent(&pWorker->KillEvent, NotificationEvent, FALSE);
1028
1029 KeInitializeEvent(&pWorker->NotifyEvent, SynchronizationEvent, FALSE);
1030
1031 INIT_INTERLOCKED_PACKET_QUEUE(&pWorker->PacketQueue);
1032
1033 do
1034 {
1035 Status = vboxNetFltWinPpAllocatePacketInfoPool(&pWorker->PacketInfoPool, VBOXNETFLT_PACKET_INFO_POOL_SIZE);
1036
1037 if (Status == NDIS_STATUS_SUCCESS)
1038 {
1039 pWorker->pSG = vboxNetFltWinCreateSG(PACKET_QUEUE_SG_SEGS_ALLOC);
1040 if (!pWorker->pSG)
1041 {
1042 Status = STATUS_INSUFFICIENT_RESOURCES;
1043 break;
1044 }
1045
1046 Status = vboxNetFltWinQuCreateSystemThread(&pWorker->pThread, (PKSTART_ROUTINE)vboxNetFltWinQuPacketQueueWorkerThreadProc, pInstance);
1047 if (Status != STATUS_SUCCESS)
1048 {
1049 vboxNetFltWinPpFreePacketInfoPool(&pWorker->PacketInfoPool);
1050 vboxNetFltWinMemFree(pWorker->pSG);
1051 pWorker->pSG = NULL;
1052 break;
1053 }
1054 }
1055
1056 } while (0);
1057
1058 return Status;
1059}
1060
1061/*
1062 * deletes the packet queue
1063 */
1064DECLHIDDEN(void) vboxNetFltWinQuFiniPacketQueue(PVBOXNETFLTINS pInstance)
1065{
1066 PINTNETSG pSG;
1067 PPACKET_QUEUE_WORKER pWorker = &pInstance->u.s.PacketQueueWorker;
1068 Assert(KeGetCurrentIrql() < DISPATCH_LEVEL);
1069
1070 /* using the pPacketQueueSG as an indicator that the packet queue is initialized */
1071 RTSpinlockAcquire((pInstance)->hSpinlock);
1072 if (pWorker->pSG)
1073 {
1074 pSG = pWorker->pSG;
1075 pWorker->pSG = NULL;
1076 RTSpinlockRelease((pInstance)->hSpinlock);
1077 KeSetEvent(&pWorker->KillEvent, 0, FALSE);
1078
1079 KeWaitForSingleObject(pWorker->pThread, Executive,
1080 KernelMode, FALSE, NULL);
1081
1082 vboxNetFltWinPpFreePacketInfoPool(&pWorker->PacketInfoPool);
1083
1084 vboxNetFltWinDeleteSG(pSG);
1085
1086 FINI_INTERLOCKED_PACKET_QUEUE(&pWorker->PacketQueue);
1087 }
1088 else
1089 {
1090 RTSpinlockRelease((pInstance)->hSpinlock);
1091 }
1092}
1093
1094#endif
1095
1096/*
1097 * creates the INTNETSG containing one segment pointing to the buffer of size cbBufSize
1098 * the INTNETSG created should be cleaned with vboxNetFltWinMemFree
1099 */
1100DECLHIDDEN(NDIS_STATUS) vboxNetFltWinAllocSG(UINT cbPacket, PINTNETSG *ppSG)
1101{
1102 NDIS_STATUS Status;
1103 PINTNETSG pSG;
1104
1105 /* allocation:
1106 * 1. SG_PACKET - with one aSegs pointing to
1107 * 2. buffer of cbPacket containing the entire packet */
1108 AssertCompileSizeAlignment(INTNETSG, sizeof(PVOID));
1109 Status = vboxNetFltWinMemAlloc((PVOID*)&pSG, cbPacket + sizeof(INTNETSG));
1110 if (Status == NDIS_STATUS_SUCCESS)
1111 {
1112 IntNetSgInitTemp(pSG, pSG + 1, cbPacket);
1113 LogFlow(("pSG created (%p)\n", pSG));
1114 *ppSG = pSG;
1115 }
1116 return Status;
1117}
1118
1119#ifndef VBOXNETFLT_NO_PACKET_QUEUE
1120/**
1121 * put the packet info to the queue
1122 */
1123DECLINLINE(void) vboxNetFltWinQuEnqueueInfo(PVBOXNETFLTPACKET_QUEUE_WORKER pWorker, PVBOXNETFLTPACKET_INFO pInfo)
1124{
1125 vboxNetFltWinQuInterlockedEnqueueTail(&pWorker->PacketQueue, pInfo);
1126
1127 KeSetEvent(&pWorker->NotifyEvent, IO_NETWORK_INCREMENT, FALSE);
1128}
1129
1130/**
1131 * puts the packet to the queue
1132 *
1133 * @return NDIST_STATUS_SUCCESS iff the packet was enqueued successfully
1134 * and error status otherwise.
1135 * NOTE: that the success status does NOT mean that the packet processing is completed, but only that it was enqueued successfully
1136 * the packet can be returned to the caller protocol/moniport only in case the bReleasePacket was set to true (in this case the copy of the packet was enqueued)
1137 * or if vboxNetFltWinQuEnqueuePacket failed, i.e. the packet was NOT enqueued
1138 */
1139DECLHIDDEN(NDIS_STATUS) vboxNetFltWinQuEnqueuePacket(PVBOXNETFLTINS pInstance, PVOID pPacket, const UINT fPacketFlags)
1140{
1141 PVBOXNETFLT_PACKET_INFO pInfo;
1142 PVBOXNETFLT_PACKET_QUEUE_WORKER pWorker = &pInstance->u.s.PacketQueueWorker;
1143 NDIS_STATUS fStatus = NDIS_STATUS_SUCCESS;
1144
1145 do
1146 {
1147 if (fPacketFlags & PACKET_COPY)
1148 {
1149 PNDIS_BUFFER pBuffer = NULL;
1150 UINT cBufferCount;
1151 UINT uBytesCopied = 0;
1152 UINT cbPacketLength;
1153 PINTNETSG pSG;
1154
1155 /* the packet is Ndis packet */
1156 Assert(!(fPacketFlags & PACKET_SG));
1157 Assert(!(fPacketFlags & PACKET_MINE));
1158
1159 NdisQueryPacket((PNDIS_PACKET)pPacket,
1160 NULL,
1161 &cBufferCount,
1162 &pBuffer,
1163 &cbPacketLength);
1164
1165
1166 Assert(cBufferCount);
1167
1168 fStatus = vboxNetFltWinAllocSG(cbPacketLength, &pSG);
1169 if (fStatus != NDIS_STATUS_SUCCESS)
1170 {
1171 AssertFailed();
1172 break;
1173 }
1174
1175 pInfo = vboxNetFltWinPpAllocPacketInfo(&pWorker->PacketInfoPool);
1176
1177 if (!pInfo)
1178 {
1179 AssertFailed();
1180 /* TODO: what status to set? */
1181 fStatus = NDIS_STATUS_FAILURE;
1182 vboxNetFltWinMemFree(pSG);
1183 break;
1184 }
1185
1186 Assert(pInfo->pPool);
1187
1188 /* the packet we are queueing is SG, add PACKET_SG to flags */
1189 SET_FLAGS_TO_INFO(pInfo, fPacketFlags | PACKET_SG);
1190 SET_PACKET_TO_INFO(pInfo, pSG);
1191
1192 fStatus = vboxNetFltWinNdisBufferMoveToSG0(pBuffer, pSG);
1193 if (fStatus != NDIS_STATUS_SUCCESS)
1194 {
1195 AssertFailed();
1196 vboxNetFltWinPpFreePacketInfo(pInfo);
1197 vboxNetFltWinMemFree(pSG);
1198 break;
1199 }
1200
1201 DBG_CHECK_PACKET_AND_SG((PNDIS_PACKET)pPacket, pSG);
1202 }
1203 else
1204 {
1205 pInfo = vboxNetFltWinPpAllocPacketInfo(&pWorker->PacketInfoPool);
1206
1207 if (!pInfo)
1208 {
1209 AssertFailed();
1210 /* TODO: what status to set? */
1211 fStatus = NDIS_STATUS_FAILURE;
1212 break;
1213 }
1214
1215 Assert(pInfo->pPool);
1216
1217 SET_FLAGS_TO_INFO(pInfo, fPacketFlags);
1218 SET_PACKET_TO_INFO(pInfo, pPacket);
1219 }
1220
1221 vboxNetFltWinQuEnqueueInfo(pWorker, pInfo);
1222
1223 } while (0);
1224
1225 return fStatus;
1226}
1227#endif
1228
1229
1230/*
1231 * netflt
1232 */
1233#ifndef VBOXNETADP
1234static NDIS_STATUS vboxNetFltWinSynchNdisRequest(PVBOXNETFLTINS pNetFlt, PNDIS_REQUEST pRequest)
1235{
1236 int rc;
1237
1238 Assert(KeGetCurrentIrql() < DISPATCH_LEVEL);
1239
1240 /* 1. serialize */
1241 rc = RTSemFastMutexRequest(pNetFlt->u.s.WinIf.hSynchRequestMutex); AssertRC(rc);
1242 if (RT_SUCCESS(rc))
1243 {
1244 NDIS_STATUS fRequestStatus = NDIS_STATUS_SUCCESS;
1245
1246 /* 2. set pNetFlt->u.s.pSynchRequest */
1247 Assert(!pNetFlt->u.s.WinIf.pSynchRequest);
1248 pNetFlt->u.s.WinIf.pSynchRequest = pRequest;
1249
1250 /* 3. call NdisRequest */
1251 NdisRequest(&fRequestStatus, pNetFlt->u.s.WinIf.hBinding, pRequest);
1252
1253 if (fRequestStatus == NDIS_STATUS_PENDING)
1254 {
1255 /* 3.1 if pending wait and assign the resulting status */
1256 KeWaitForSingleObject(&pNetFlt->u.s.WinIf.hSynchCompletionEvent, Executive,
1257 KernelMode, FALSE, NULL);
1258
1259 fRequestStatus = pNetFlt->u.s.WinIf.SynchCompletionStatus;
1260 }
1261
1262 /* 4. clear the pNetFlt->u.s.pSynchRequest */
1263 pNetFlt->u.s.WinIf.pSynchRequest = NULL;
1264
1265 RTSemFastMutexRelease(pNetFlt->u.s.WinIf.hSynchRequestMutex); AssertRC(rc);
1266 return fRequestStatus;
1267 }
1268 return NDIS_STATUS_FAILURE;
1269}
1270
1271
1272DECLHIDDEN(NDIS_STATUS) vboxNetFltWinGetMacAddress(PVBOXNETFLTINS pNetFlt, PRTMAC pMac)
1273{
1274 NDIS_REQUEST request;
1275 NDIS_STATUS status;
1276 request.RequestType = NdisRequestQueryInformation;
1277 request.DATA.QUERY_INFORMATION.InformationBuffer = pMac;
1278 request.DATA.QUERY_INFORMATION.InformationBufferLength = sizeof(RTMAC);
1279 request.DATA.QUERY_INFORMATION.Oid = OID_802_3_CURRENT_ADDRESS;
1280 status = vboxNetFltWinSynchNdisRequest(pNetFlt, &request);
1281 if (status != NDIS_STATUS_SUCCESS)
1282 {
1283 /* TODO */
1284 AssertFailed();
1285 }
1286
1287 return status;
1288
1289}
1290
1291DECLHIDDEN(NDIS_STATUS) vboxNetFltWinQueryPhysicalMedium(PVBOXNETFLTINS pNetFlt, NDIS_PHYSICAL_MEDIUM * pMedium)
1292{
1293 NDIS_REQUEST Request;
1294 NDIS_STATUS Status;
1295 Request.RequestType = NdisRequestQueryInformation;
1296 Request.DATA.QUERY_INFORMATION.InformationBuffer = pMedium;
1297 Request.DATA.QUERY_INFORMATION.InformationBufferLength = sizeof(NDIS_PHYSICAL_MEDIUM);
1298 Request.DATA.QUERY_INFORMATION.Oid = OID_GEN_PHYSICAL_MEDIUM;
1299 Status = vboxNetFltWinSynchNdisRequest(pNetFlt, &Request);
1300 if (Status != NDIS_STATUS_SUCCESS)
1301 {
1302 if (Status == NDIS_STATUS_NOT_SUPPORTED || Status == NDIS_STATUS_NOT_RECOGNIZED || Status == NDIS_STATUS_INVALID_OID)
1303 {
1304 Status = NDIS_STATUS_NOT_SUPPORTED;
1305 }
1306 else
1307 {
1308 LogRel(("OID_GEN_PHYSICAL_MEDIUM failed: Status (0x%x)", Status));
1309 AssertFailed();
1310 }
1311 }
1312 return Status;
1313}
1314
1315DECLHIDDEN(bool) vboxNetFltWinIsPromiscuous(PVBOXNETFLTINS pNetFlt)
1316{
1317 /** @todo r=bird: This is too slow and is probably returning the wrong
1318 * information. What we're interested in is whether someone besides us
1319 * has put the interface into promiscuous mode. */
1320 NDIS_REQUEST request;
1321 NDIS_STATUS status;
1322 ULONG filter;
1323 Assert(VBOXNETFLT_PROMISCUOUS_SUPPORTED(pNetFlt));
1324 request.RequestType = NdisRequestQueryInformation;
1325 request.DATA.QUERY_INFORMATION.InformationBuffer = &filter;
1326 request.DATA.QUERY_INFORMATION.InformationBufferLength = sizeof(filter);
1327 request.DATA.QUERY_INFORMATION.Oid = OID_GEN_CURRENT_PACKET_FILTER;
1328 status = vboxNetFltWinSynchNdisRequest(pNetFlt, &request);
1329 if (status != NDIS_STATUS_SUCCESS)
1330 {
1331 /* TODO */
1332 AssertFailed();
1333 return false;
1334 }
1335 return (filter & NDIS_PACKET_TYPE_PROMISCUOUS) != 0;
1336}
1337
1338DECLHIDDEN(NDIS_STATUS) vboxNetFltWinSetPromiscuous(PVBOXNETFLTINS pNetFlt, bool bYes)
1339{
1340/** @todo Need to report changes to the switch via:
1341 * pThis->pSwitchPort->pfnReportPromiscuousMode(pThis->pSwitchPort, fPromisc);
1342 */
1343 Assert(VBOXNETFLT_PROMISCUOUS_SUPPORTED(pNetFlt));
1344 if (VBOXNETFLT_PROMISCUOUS_SUPPORTED(pNetFlt))
1345 {
1346 NDIS_REQUEST Request;
1347 NDIS_STATUS fStatus;
1348 ULONG fFilter;
1349 ULONG fExpectedFilter;
1350 ULONG fOurFilter;
1351 Request.RequestType = NdisRequestQueryInformation;
1352 Request.DATA.QUERY_INFORMATION.InformationBuffer = &fFilter;
1353 Request.DATA.QUERY_INFORMATION.InformationBufferLength = sizeof(fFilter);
1354 Request.DATA.QUERY_INFORMATION.Oid = OID_GEN_CURRENT_PACKET_FILTER;
1355 fStatus = vboxNetFltWinSynchNdisRequest(pNetFlt, &Request);
1356 if (fStatus != NDIS_STATUS_SUCCESS)
1357 {
1358 /* TODO: */
1359 AssertFailed();
1360 return fStatus;
1361 }
1362
1363 if (!pNetFlt->u.s.WinIf.StateFlags.fUpperProtSetFilterInitialized)
1364 {
1365 /* the cache was not initialized yet, initiate it with the current filter value */
1366 pNetFlt->u.s.WinIf.fUpperProtocolSetFilter = fFilter;
1367 pNetFlt->u.s.WinIf.StateFlags.fUpperProtSetFilterInitialized = TRUE;
1368 }
1369
1370
1371 if (bYes)
1372 {
1373 fExpectedFilter = NDIS_PACKET_TYPE_PROMISCUOUS;
1374 fOurFilter = NDIS_PACKET_TYPE_PROMISCUOUS;
1375 }
1376 else
1377 {
1378 fExpectedFilter = pNetFlt->u.s.WinIf.fUpperProtocolSetFilter;
1379 fOurFilter = 0;
1380 }
1381
1382 if (fExpectedFilter != fFilter)
1383 {
1384 Request.RequestType = NdisRequestSetInformation;
1385 Request.DATA.SET_INFORMATION.InformationBuffer = &fExpectedFilter;
1386 Request.DATA.SET_INFORMATION.InformationBufferLength = sizeof(fExpectedFilter);
1387 Request.DATA.SET_INFORMATION.Oid = OID_GEN_CURRENT_PACKET_FILTER;
1388 fStatus = vboxNetFltWinSynchNdisRequest(pNetFlt, &Request);
1389 if (fStatus != NDIS_STATUS_SUCCESS)
1390 {
1391 /* TODO */
1392 AssertFailed();
1393 return fStatus;
1394 }
1395 }
1396 pNetFlt->u.s.WinIf.fOurSetFilter = fOurFilter;
1397 return fStatus;
1398 }
1399 return NDIS_STATUS_NOT_SUPPORTED;
1400}
1401#else /* if defined VBOXNETADP */
1402
1403/**
1404 * Generates a new unique MAC address based on our vendor ID
1405 */
1406DECLHIDDEN(void) vboxNetFltWinGenerateMACAddress(RTMAC *pMac)
1407{
1408 /* temporary use a time info */
1409 uint64_t NanoTS = RTTimeSystemNanoTS();
1410 pMac->au8[0] = (uint8_t)((VBOXNETADP_VENDOR_ID >> 16) & 0xff);
1411 pMac->au8[1] = (uint8_t)((VBOXNETADP_VENDOR_ID >> 8) & 0xff);
1412 pMac->au8[2] = (uint8_t)(VBOXNETADP_VENDOR_ID & 0xff);
1413 pMac->au8[3] = (uint8_t)(NanoTS & 0xff0000);
1414 pMac->au16[2] = (uint16_t)(NanoTS & 0xffff);
1415}
1416
1417DECLHIDDEN(int) vboxNetFltWinMAC2NdisString(RTMAC *pMac, PNDIS_STRING pNdisString)
1418{
1419 static const char s_achDigits[17] = "0123456789abcdef";
1420 PWSTR pString;
1421
1422 /* validate parameters */
1423 AssertPtrReturn(pMac, VERR_INVALID_PARAMETER);
1424 AssertPtrReturn(pNdisString, VERR_INVALID_PARAMETER);
1425 AssertReturn(pNdisString->MaximumLength >= 13*sizeof(pNdisString->Buffer[0]), VERR_INVALID_PARAMETER);
1426
1427 pString = pNdisString->Buffer;
1428
1429 for (int i = 0; i < 6; i++)
1430 {
1431 uint8_t u8 = pMac->au8[i];
1432 pString[0] = s_achDigits[(u8 >> 4) & 0xf];
1433 pString[1] = s_achDigits[(u8/*>>0*/)& 0xf];
1434 pString += 2;
1435 }
1436
1437 pNdisString->Length = 12*sizeof(pNdisString->Buffer[0]);
1438
1439 *pString = L'\0';
1440
1441 return VINF_SUCCESS;
1442}
1443
1444static int vboxNetFltWinWchar2Int(WCHAR c, uint8_t * pv)
1445{
1446 if (c >= L'A' && c <= L'F')
1447 {
1448 *pv = (c - L'A') + 10;
1449 }
1450 else if (c >= L'a' && c <= L'f')
1451 {
1452 *pv = (c - L'a') + 10;
1453 }
1454 else if (c >= L'0' && c <= L'9')
1455 {
1456 *pv = (c - L'0');
1457 }
1458 else
1459 {
1460 return VERR_INVALID_PARAMETER;
1461 }
1462 return VINF_SUCCESS;
1463}
1464
1465DECLHIDDEN(int) vboxNetFltWinMACFromNdisString(RTMAC *pMac, PNDIS_STRING pNdisString)
1466{
1467 int i, rc;
1468 PWSTR pString;
1469
1470 /* validate parameters */
1471 AssertPtrReturn(pMac, VERR_INVALID_PARAMETER);
1472 AssertPtrReturn(pNdisString, VERR_INVALID_PARAMETER);
1473 AssertReturn(pNdisString->Length >= 12*sizeof(pNdisString->Buffer[0]), VERR_INVALID_PARAMETER);
1474
1475 pString = pNdisString->Buffer;
1476
1477 for (i = 0; i < 6; i++)
1478 {
1479 uint8_t v1, v2;
1480 rc = vboxNetFltWinWchar2Int(pString[0], &v1);
1481 if (RT_FAILURE(rc))
1482 {
1483 break;
1484 }
1485
1486 rc = vboxNetFltWinWchar2Int(pString[1], &v2);
1487 if (RT_FAILURE(rc))
1488 {
1489 break;
1490 }
1491
1492 pMac->au8[i] = (v1 << 4) | v2;
1493
1494 pString += 2;
1495 }
1496
1497 return rc;
1498}
1499
1500#endif
1501/**
1502 * creates a NDIS_PACKET from the PINTNETSG
1503 */
1504DECLHIDDEN(PNDIS_PACKET) vboxNetFltWinNdisPacketFromSG(PVBOXNETFLTINS pNetFlt, PINTNETSG pSG, PVOID pBufToFree, bool bToWire, bool bCopyMemory)
1505{
1506 NDIS_STATUS fStatus;
1507 PNDIS_PACKET pPacket;
1508
1509 Assert(pSG->aSegs[0].pv);
1510 Assert(pSG->cbTotal >= sizeof(VBOXNETFLT_PACKET_ETHEADER_SIZE));
1511
1512/** @todo Hrmpf, how can we fix this assumption? I fear this'll cause data
1513 * corruption and maybe even BSODs ... */
1514 AssertReturn(pSG->cSegsUsed == 1 || bCopyMemory, NULL);
1515
1516#ifdef VBOXNETADP
1517 NdisAllocatePacket(&fStatus, &pPacket, pNetFlt->u.s.WinIf.hRecvPacketPool);
1518#else
1519 NdisAllocatePacket(&fStatus, &pPacket, bToWire ? pNetFlt->u.s.WinIf.hSendPacketPool : pNetFlt->u.s.WinIf.hRecvPacketPool);
1520#endif
1521 if (fStatus == NDIS_STATUS_SUCCESS)
1522 {
1523 PNDIS_BUFFER pBuffer;
1524 PVOID pvMemBuf;
1525
1526 /* @todo: generally we do not always need to zero-initialize the complete OOB data here, reinitialize only when/what we need,
1527 * however we DO need to reset the status for the packets we indicate via NdisMIndicateReceivePacket to avoid packet loss
1528 * in case the status contains NDIS_STATUS_RESOURCES */
1529 VBOXNETFLT_OOB_INIT(pPacket);
1530
1531 if (bCopyMemory)
1532 {
1533 fStatus = vboxNetFltWinMemAlloc(&pvMemBuf, pSG->cbTotal);
1534 Assert(fStatus == NDIS_STATUS_SUCCESS);
1535 if (fStatus == NDIS_STATUS_SUCCESS)
1536 IntNetSgRead(pSG, pvMemBuf);
1537 }
1538 else
1539 {
1540 pvMemBuf = pSG->aSegs[0].pv;
1541 }
1542 if (fStatus == NDIS_STATUS_SUCCESS)
1543 {
1544#ifdef VBOXNETADP
1545 NdisAllocateBuffer(&fStatus, &pBuffer,
1546 pNetFlt->u.s.WinIf.hRecvBufferPool,
1547 pvMemBuf,
1548 pSG->cbTotal);
1549#else
1550 NdisAllocateBuffer(&fStatus, &pBuffer,
1551 bToWire ? pNetFlt->u.s.WinIf.hSendBufferPool : pNetFlt->u.s.WinIf.hRecvBufferPool,
1552 pvMemBuf,
1553 pSG->cbTotal);
1554#endif
1555
1556 if (fStatus == NDIS_STATUS_SUCCESS)
1557 {
1558 NdisChainBufferAtBack(pPacket, pBuffer);
1559
1560 if (bToWire)
1561 {
1562 PVBOXNETFLT_PKTRSVD_PT pSendInfo = (PVBOXNETFLT_PKTRSVD_PT)pPacket->ProtocolReserved;
1563 pSendInfo->pOrigPacket = NULL;
1564 pSendInfo->pBufToFree = pBufToFree;
1565#ifdef VBOX_LOOPBACK_USEFLAGS
1566 /* set "don't loopback" flags */
1567 NdisGetPacketFlags(pPacket) = g_VBoxNetFltGlobalsWin.fPacketDontLoopBack;
1568#else
1569 NdisGetPacketFlags(pPacket) = 0;
1570#endif
1571 }
1572 else
1573 {
1574 PVBOXNETFLT_PKTRSVD_MP pRecvInfo = (PVBOXNETFLT_PKTRSVD_MP)pPacket->MiniportReserved;
1575 pRecvInfo->pOrigPacket = NULL;
1576 pRecvInfo->pBufToFree = pBufToFree;
1577
1578 /* we must set the header size on receive */
1579 NDIS_SET_PACKET_HEADER_SIZE(pPacket, VBOXNETFLT_PACKET_ETHEADER_SIZE);
1580 /* NdisAllocatePacket zero-initializes the OOB data,
1581 * but keeps the packet flags, clean them here */
1582 NdisGetPacketFlags(pPacket) = 0;
1583 }
1584 /* TODO: set out of bound data */
1585 }
1586 else
1587 {
1588 AssertFailed();
1589 if (bCopyMemory)
1590 {
1591 vboxNetFltWinMemFree(pvMemBuf);
1592 }
1593 NdisFreePacket(pPacket);
1594 pPacket = NULL;
1595 }
1596 }
1597 else
1598 {
1599 AssertFailed();
1600 NdisFreePacket(pPacket);
1601 pPacket = NULL;
1602 }
1603 }
1604 else
1605 {
1606 pPacket = NULL;
1607 }
1608
1609 DBG_CHECK_PACKET_AND_SG(pPacket, pSG);
1610
1611 return pPacket;
1612}
1613
1614/*
1615 * frees NDIS_PACKET created with vboxNetFltWinNdisPacketFromSG
1616 */
1617DECLHIDDEN(void) vboxNetFltWinFreeSGNdisPacket(PNDIS_PACKET pPacket, bool bFreeMem)
1618{
1619 UINT cBufCount;
1620 PNDIS_BUFFER pFirstBuffer;
1621 UINT uTotalPacketLength;
1622 PNDIS_BUFFER pBuffer;
1623
1624 NdisQueryPacket(pPacket, NULL, &cBufCount, &pFirstBuffer, &uTotalPacketLength);
1625
1626 Assert(cBufCount == 1);
1627
1628 do
1629 {
1630 NdisUnchainBufferAtBack(pPacket, &pBuffer);
1631 if (pBuffer != NULL)
1632 {
1633 PVOID pvMemBuf;
1634 UINT cbLength;
1635
1636 NdisQueryBufferSafe(pBuffer, &pvMemBuf, &cbLength, NormalPagePriority);
1637 NdisFreeBuffer(pBuffer);
1638 if (bFreeMem)
1639 {
1640 vboxNetFltWinMemFree(pvMemBuf);
1641 }
1642 }
1643 else
1644 {
1645 break;
1646 }
1647 } while (true);
1648
1649 NdisFreePacket(pPacket);
1650}
1651
1652#if !defined(VBOXNETADP)
1653static void vboxNetFltWinAssociateMiniportProtocol(PVBOXNETFLTGLOBALS_WIN pGlobalsWin)
1654{
1655 NdisIMAssociateMiniport(pGlobalsWin->Mp.hMiniport, pGlobalsWin->Pt.hProtocol);
1656}
1657#endif
1658
1659/*
1660 * NetFlt driver unload function
1661 */
1662DECLHIDDEN(VOID) vboxNetFltWinUnload(IN PDRIVER_OBJECT DriverObject)
1663{
1664 int rc;
1665 UNREFERENCED_PARAMETER(DriverObject);
1666
1667 LogFlow((__FUNCTION__" ==> DO (0x%x)\n", DriverObject));
1668
1669 rc = vboxNetFltWinFiniIdc();
1670 if (RT_FAILURE(rc))
1671 {
1672 /* TODO: we can not prevent driver unload here */
1673 AssertFailed();
1674
1675 Log((__FUNCTION__": vboxNetFltWinFiniIdc - failed, busy.\n"));
1676 }
1677
1678 vboxNetFltWinJobFiniQueue(&g_VBoxJobQueue);
1679#ifndef VBOXNETADP
1680 vboxNetFltWinPtDeregister(&g_VBoxNetFltGlobalsWin.Pt);
1681#endif
1682
1683 vboxNetFltWinMpDeregister(&g_VBoxNetFltGlobalsWin.Mp);
1684
1685#ifndef VBOXNETADP
1686 NdisFreeSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
1687#endif /* VBOXNETADP */
1688
1689 LogFlow((__FUNCTION__" <== DO (0x%x)\n", DriverObject));
1690
1691 vboxNetFltWinFiniNetFltBase();
1692 /* don't use logging or any RT after de-init */
1693}
1694
1695RT_C_DECLS_BEGIN
1696
1697NTSTATUS DriverEntry(IN PDRIVER_OBJECT DriverObject, IN PUNICODE_STRING RegistryPath);
1698
1699RT_C_DECLS_END
1700
1701NTSTATUS DriverEntry(IN PDRIVER_OBJECT DriverObject, IN PUNICODE_STRING RegistryPath)
1702{
1703 NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
1704 int rc;
1705
1706 /* the idc registration is initiated via IOCTL since our driver
1707 * can be loaded when the VBoxDrv is not in case we are a Ndis IM driver */
1708 rc = vboxNetFltWinInitNetFltBase();
1709 AssertRC(rc);
1710 if (RT_SUCCESS(rc))
1711 {
1712 Status = vboxNetFltWinJobInitQueue(&g_VBoxJobQueue);
1713 Assert(Status == STATUS_SUCCESS);
1714 if (Status == STATUS_SUCCESS)
1715 {
1716 ULONG MjVersion;
1717 ULONG MnVersion;
1718
1719 /* note: we do it after we initialize the Job Queue */
1720 vboxNetFltWinStartInitIdcProbing();
1721
1722 NdisZeroMemory(&g_VBoxNetFltGlobalsWin, sizeof (g_VBoxNetFltGlobalsWin));
1723 KeInitializeEvent(&g_VBoxNetFltGlobalsWin.SynchEvent, SynchronizationEvent, TRUE /* signalled*/);
1724
1725 PsGetVersion(&MjVersion, &MnVersion,
1726 NULL, /* PULONG BuildNumber OPTIONAL */
1727 NULL /* PUNICODE_STRING CSDVersion OPTIONAL */
1728 );
1729
1730 g_VBoxNetFltGlobalsWin.fPacketDontLoopBack = NDIS_FLAGS_DONT_LOOPBACK;
1731
1732 if (MjVersion == 5 && MnVersion == 0)
1733 {
1734 /* this is Win2k, we don't support it actually, but just in case */
1735 g_VBoxNetFltGlobalsWin.fPacketDontLoopBack |= NDIS_FLAGS_SKIP_LOOPBACK_W2K;
1736 }
1737
1738 g_VBoxNetFltGlobalsWin.fPacketIsLoopedBack = NDIS_FLAGS_IS_LOOPBACK_PACKET;
1739
1740#ifndef VBOXNETADP
1741 RTListInit(&g_VBoxNetFltGlobalsWin.listFilters);
1742 NdisAllocateSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
1743#endif
1744
1745 Status = vboxNetFltWinMpRegister(&g_VBoxNetFltGlobalsWin.Mp, DriverObject, RegistryPath);
1746 Assert(Status == STATUS_SUCCESS);
1747 if (Status == NDIS_STATUS_SUCCESS)
1748 {
1749#ifndef VBOXNETADP
1750 Status = vboxNetFltWinPtRegister(&g_VBoxNetFltGlobalsWin.Pt, DriverObject, RegistryPath);
1751 Assert(Status == STATUS_SUCCESS);
1752 if (Status == NDIS_STATUS_SUCCESS)
1753#endif
1754 {
1755#ifndef VBOXNETADP
1756 vboxNetFltWinAssociateMiniportProtocol(&g_VBoxNetFltGlobalsWin);
1757#endif
1758 return STATUS_SUCCESS;
1759
1760//#ifndef VBOXNETADP
1761// vboxNetFltWinPtDeregister(&g_VBoxNetFltGlobalsWin.Pt);
1762//#endif
1763 }
1764 vboxNetFltWinMpDeregister(&g_VBoxNetFltGlobalsWin.Mp);
1765#ifndef VBOXNETADP
1766 NdisFreeSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
1767#endif /* VBOXNETADP */
1768 }
1769 vboxNetFltWinJobFiniQueue(&g_VBoxJobQueue);
1770 }
1771 vboxNetFltWinFiniNetFlt();
1772 }
1773 else
1774 {
1775 Status = NDIS_STATUS_FAILURE;
1776 }
1777
1778 return Status;
1779}
1780
1781#ifndef VBOXNETADP
1782/**
1783 * creates and initializes the packet to be sent to the underlying miniport given a packet posted to our miniport edge
1784 * according to DDK docs we must create our own packet rather than posting the one passed to us
1785 */
1786DECLHIDDEN(NDIS_STATUS) vboxNetFltWinPrepareSendPacket(PVBOXNETFLTINS pNetFlt, PNDIS_PACKET pPacket, PNDIS_PACKET *ppMyPacket)
1787{
1788 NDIS_STATUS Status;
1789
1790 NdisAllocatePacket(&Status, ppMyPacket, pNetFlt->u.s.WinIf.hSendPacketPool);
1791
1792 if (Status == NDIS_STATUS_SUCCESS)
1793 {
1794 PVBOXNETFLT_PKTRSVD_PT pSendInfo = (PVBOXNETFLT_PKTRSVD_PT)((*ppMyPacket)->ProtocolReserved);
1795 pSendInfo->pOrigPacket = pPacket;
1796 pSendInfo->pBufToFree = NULL;
1797 /* the rest will be filled on send */
1798
1799 vboxNetFltWinCopyPacketInfoOnSend(*ppMyPacket, pPacket);
1800
1801#ifdef VBOX_LOOPBACK_USEFLAGS
1802 NdisGetPacketFlags(*ppMyPacket) |= g_VBoxNetFltGlobalsWin.fPacketDontLoopBack;
1803#endif
1804 }
1805 else
1806 {
1807 *ppMyPacket = NULL;
1808 }
1809
1810 return Status;
1811}
1812
1813/**
1814 * creates and initializes the packet to be sent to the upperlying protocol given a packet indicated to our protocol edge
1815 * according to DDK docs we must create our own packet rather than posting the one passed to us
1816 */
1817DECLHIDDEN(NDIS_STATUS) vboxNetFltWinPrepareRecvPacket(PVBOXNETFLTINS pNetFlt, PNDIS_PACKET pPacket, PNDIS_PACKET *ppMyPacket, bool bDpr)
1818{
1819 NDIS_STATUS Status;
1820
1821 if (bDpr)
1822 {
1823 Assert(KeGetCurrentIrql() == DISPATCH_LEVEL);
1824 NdisDprAllocatePacket(&Status, ppMyPacket, pNetFlt->u.s.WinIf.hRecvPacketPool);
1825 }
1826 else
1827 {
1828 NdisAllocatePacket(&Status, ppMyPacket, pNetFlt->u.s.WinIf.hRecvPacketPool);
1829 }
1830
1831 if (Status == NDIS_STATUS_SUCCESS)
1832 {
1833 PVBOXNETFLT_PKTRSVD_MP pRecvInfo = (PVBOXNETFLT_PKTRSVD_MP)((*ppMyPacket)->MiniportReserved);
1834 pRecvInfo->pOrigPacket = pPacket;
1835 pRecvInfo->pBufToFree = NULL;
1836
1837 Status = vboxNetFltWinCopyPacketInfoOnRecv(*ppMyPacket, pPacket, false);
1838 }
1839 else
1840 {
1841 *ppMyPacket = NULL;
1842 }
1843 return Status;
1844}
1845#endif
1846/**
1847 * initializes the VBOXNETFLTINS (our context structure) and binds to the given adapter
1848 */
1849#if defined(VBOXNETADP)
1850DECLHIDDEN(NDIS_STATUS) vboxNetFltWinPtInitBind(PVBOXNETFLTINS *ppNetFlt, NDIS_HANDLE hMiniportAdapter, PNDIS_STRING pBindToMiniportName /* actually this is our miniport name*/, NDIS_HANDLE hWrapperConfigurationContext)
1851#else
1852DECLHIDDEN(NDIS_STATUS) vboxNetFltWinPtInitBind(PVBOXNETFLTINS *ppNetFlt, PNDIS_STRING pOurMiniportName, PNDIS_STRING pBindToMiniportName)
1853#endif
1854{
1855 NDIS_STATUS Status;
1856 do
1857 {
1858 ANSI_STRING AnsiString;
1859 int rc;
1860 PVBOXNETFLTINS pInstance;
1861 USHORT cbAnsiName = pBindToMiniportName->Length;/* the length is is bytes ; *2 ;RtlUnicodeStringToAnsiSize(pBindToMiniportName)*/
1862 CREATE_INSTANCE_CONTEXT Context;
1863
1864# ifndef VBOXNETADP
1865 Context.pOurName = pOurMiniportName;
1866 Context.pBindToName = pBindToMiniportName;
1867# else
1868 Context.hMiniportAdapter = hMiniportAdapter;
1869 Context.hWrapperConfigurationContext = hWrapperConfigurationContext;
1870# endif
1871 Context.Status = NDIS_STATUS_SUCCESS;
1872
1873 AnsiString.Buffer = 0; /* will be allocated by RtlUnicodeStringToAnsiString */
1874 AnsiString.Length = 0;
1875 AnsiString.MaximumLength = cbAnsiName;
1876
1877 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
1878
1879 Status = RtlUnicodeStringToAnsiString(&AnsiString, pBindToMiniportName, true);
1880
1881 if (Status != STATUS_SUCCESS)
1882 {
1883 break;
1884 }
1885
1886 rc = vboxNetFltSearchCreateInstance(&g_VBoxNetFltGlobals, AnsiString.Buffer, &pInstance, &Context);
1887 RtlFreeAnsiString(&AnsiString);
1888 if (RT_FAILURE(rc))
1889 {
1890 AssertFailed();
1891 Status = Context.Status != NDIS_STATUS_SUCCESS ? Context.Status : NDIS_STATUS_FAILURE;
1892 break;
1893 }
1894
1895 Assert(pInstance);
1896
1897 if (rc == VINF_ALREADY_INITIALIZED)
1898 {
1899 /* the case when our adapter was unbound while IntNet was connected to it */
1900 /* the instance remains valid until IntNet disconnects from it, we simply search and re-use it*/
1901 rc = vboxNetFltWinAttachToInterface(pInstance, &Context, true);
1902 if (RT_FAILURE(rc))
1903 {
1904 AssertFailed();
1905 Status = Context.Status != NDIS_STATUS_SUCCESS ? Context.Status : NDIS_STATUS_FAILURE;
1906 /* release netflt */
1907 vboxNetFltRelease(pInstance, false);
1908
1909 break;
1910 }
1911 }
1912
1913 *ppNetFlt = pInstance;
1914
1915 } while (FALSE);
1916
1917 return Status;
1918}
1919/*
1920 * deinitializes the VBOXNETFLTWIN
1921 */
1922DECLHIDDEN(VOID) vboxNetFltWinPtFiniWinIf(PVBOXNETFLTWIN pWinIf)
1923{
1924#ifndef VBOXNETADP
1925 int rc;
1926#endif
1927
1928 LogFlow(("==>"__FUNCTION__" : pWinIf 0x%p\n", pWinIf));
1929
1930 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
1931#ifndef VBOXNETADP
1932 if (pWinIf->MpDeviceName.Buffer)
1933 {
1934 vboxNetFltWinMemFree(pWinIf->MpDeviceName.Buffer);
1935 }
1936
1937 FINI_INTERLOCKED_SINGLE_LIST(&pWinIf->TransferDataList);
1938# if defined(DEBUG_NETFLT_LOOPBACK) || !defined(VBOX_LOOPBACK_USEFLAGS)
1939 FINI_INTERLOCKED_SINGLE_LIST(&pWinIf->SendPacketQueue);
1940# endif
1941 NdisFreeBufferPool(pWinIf->hSendBufferPool);
1942 NdisFreePacketPool(pWinIf->hSendPacketPool);
1943 rc = RTSemFastMutexDestroy(pWinIf->hSynchRequestMutex); AssertRC(rc);
1944#endif
1945
1946 /* NOTE: NULL is a valid handle */
1947 NdisFreeBufferPool(pWinIf->hRecvBufferPool);
1948 NdisFreePacketPool(pWinIf->hRecvPacketPool);
1949
1950 LogFlow(("<=="__FUNCTION__" : pWinIf 0x%p\n", pWinIf));
1951}
1952
1953#ifndef VBOXNETADP
1954DECLHIDDEN(NDIS_STATUS) vboxNetFltWinPtInitWinIf(PVBOXNETFLTWIN pWinIf, IN PNDIS_STRING pOurDeviceName)
1955#else
1956DECLHIDDEN(NDIS_STATUS) vboxNetFltWinPtInitWinIf(PVBOXNETFLTWIN pWinIf)
1957#endif
1958{
1959 NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
1960#ifndef VBOXNETADP
1961 int rc;
1962#endif
1963 BOOLEAN bCallFiniOnFail = FALSE;
1964
1965 LogFlow(("==>"__FUNCTION__": pWinIf 0x%p\n", pWinIf));
1966
1967 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
1968
1969 NdisZeroMemory(pWinIf, sizeof (VBOXNETFLTWIN));
1970 NdisAllocatePacketPoolEx(&Status, &pWinIf->hRecvPacketPool,
1971 VBOXNETFLT_PACKET_POOL_SIZE_NORMAL,
1972 VBOXNETFLT_PACKET_POOL_SIZE_OVERFLOW,
1973 PROTOCOL_RESERVED_SIZE_IN_PACKET);
1974 Assert(Status == NDIS_STATUS_SUCCESS);
1975 if (Status == NDIS_STATUS_SUCCESS)
1976 {
1977 /* NOTE: NULL is a valid handle !!! */
1978 NdisAllocateBufferPool(&Status, &pWinIf->hRecvBufferPool, VBOXNETFLT_BUFFER_POOL_SIZE_RX);
1979 Assert(Status == NDIS_STATUS_SUCCESS);
1980 if (Status == NDIS_STATUS_SUCCESS)
1981 {
1982 pWinIf->MpState.PowerState = NdisDeviceStateD3;
1983 vboxNetFltWinSetOpState(&pWinIf->MpState, kVBoxNetDevOpState_Deinitialized);
1984#ifndef VBOXNETADP
1985 pWinIf->PtState.PowerState = NdisDeviceStateD3;
1986 vboxNetFltWinSetOpState(&pWinIf->PtState, kVBoxNetDevOpState_Deinitialized);
1987
1988 NdisAllocateBufferPool(&Status,
1989 &pWinIf->hSendBufferPool,
1990 VBOXNETFLT_BUFFER_POOL_SIZE_TX);
1991 Assert(Status == NDIS_STATUS_SUCCESS);
1992 if (Status == NDIS_STATUS_SUCCESS)
1993 {
1994 INIT_INTERLOCKED_SINGLE_LIST(&pWinIf->TransferDataList);
1995
1996# if defined(DEBUG_NETFLT_LOOPBACK) || !defined(VBOX_LOOPBACK_USEFLAGS)
1997 INIT_INTERLOCKED_SINGLE_LIST(&pWinIf->SendPacketQueue);
1998# endif
1999 NdisInitializeEvent(&pWinIf->OpenCloseEvent);
2000
2001 KeInitializeEvent(&pWinIf->hSynchCompletionEvent, SynchronizationEvent, FALSE);
2002
2003 NdisInitializeEvent(&pWinIf->MpInitCompleteEvent);
2004
2005 NdisAllocatePacketPoolEx(&Status, &pWinIf->hSendPacketPool,
2006 VBOXNETFLT_PACKET_POOL_SIZE_NORMAL,
2007 VBOXNETFLT_PACKET_POOL_SIZE_OVERFLOW,
2008 sizeof (PVBOXNETFLT_PKTRSVD_PT));
2009 Assert(Status == NDIS_STATUS_SUCCESS);
2010 if (Status == NDIS_STATUS_SUCCESS)
2011 {
2012 rc = RTSemFastMutexCreate(&pWinIf->hSynchRequestMutex);
2013 AssertRC(rc);
2014 if (RT_SUCCESS(rc))
2015 {
2016 Status = vboxNetFltWinMemAlloc((PVOID*)&pWinIf->MpDeviceName.Buffer, pOurDeviceName->Length);
2017 Assert(Status == NDIS_STATUS_SUCCESS);
2018 if (Status == NDIS_STATUS_SUCCESS)
2019 {
2020 pWinIf->MpDeviceName.MaximumLength = pOurDeviceName->Length;
2021 pWinIf->MpDeviceName.Length = 0;
2022 Status = vboxNetFltWinCopyString(&pWinIf->MpDeviceName, pOurDeviceName);
2023#endif
2024 return NDIS_STATUS_SUCCESS;
2025#ifndef VBOXNETADP
2026 vboxNetFltWinMemFree(pWinIf->MpDeviceName.Buffer);
2027 }
2028 RTSemFastMutexDestroy(pWinIf->hSynchRequestMutex);
2029 }
2030 else
2031 {
2032 Status = NDIS_STATUS_FAILURE;
2033 }
2034 NdisFreePacketPool(pWinIf->hSendPacketPool);
2035 }
2036 NdisFreeBufferPool(pWinIf->hSendBufferPool);
2037 }
2038#endif
2039 NdisFreeBufferPool(pWinIf->hRecvBufferPool);
2040 }
2041 NdisFreePacketPool(pWinIf->hRecvPacketPool);
2042 }
2043
2044 LogFlow(("<=="__FUNCTION__": pWinIf 0x%p, Status 0x%x\n", pWinIf, Status));
2045
2046 return Status;
2047}
2048
2049/**
2050 * match packets
2051 */
2052#define NEXT_LIST_ENTRY(_Entry) ((_Entry)->Flink)
2053#define PREV_LIST_ENTRY(_Entry) ((_Entry)->Blink)
2054#define FIRST_LIST_ENTRY NEXT_LIST_ENTRY
2055#define LAST_LIST_ENTRY PREV_LIST_ENTRY
2056
2057#define MIN(_a, _b) ((_a) < (_b) ? (_a) : (_b))
2058
2059#ifndef VBOXNETADP
2060
2061#ifdef DEBUG_misha
2062
2063RTMAC g_vboxNetFltWinVerifyMACBroadcast = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
2064RTMAC g_vboxNetFltWinVerifyMACGuest = {0x08, 0x00, 0x27, 0x01, 0x02, 0x03};
2065
2066DECLHIDDEN(PRTNETETHERHDR) vboxNetFltWinGetEthHdr(PNDIS_PACKET pPacket)
2067{
2068 UINT cBufCount1;
2069 PNDIS_BUFFER pBuffer1;
2070 UINT uTotalPacketLength1;
2071 RTNETETHERHDR* pEth;
2072 UINT cbLength1 = 0;
2073 UINT i = 0;
2074
2075 NdisQueryPacket(pPacket, NULL, &cBufCount1, &pBuffer1, &uTotalPacketLength1);
2076
2077 Assert(pBuffer1);
2078 Assert(uTotalPacketLength1 >= VBOXNETFLT_PACKET_ETHEADER_SIZE);
2079 if (uTotalPacketLength1 < VBOXNETFLT_PACKET_ETHEADER_SIZE)
2080 return NULL;
2081
2082 NdisQueryBufferSafe(pBuffer1, &pEth, &cbLength1, NormalPagePriority);
2083 Assert(cbLength1 >= VBOXNETFLT_PACKET_ETHEADER_SIZE);
2084 if (cbLength1 < VBOXNETFLT_PACKET_ETHEADER_SIZE)
2085 return NULL;
2086
2087 return pEth;
2088}
2089
2090DECLHIDDEN(PRTNETETHERHDR) vboxNetFltWinGetEthHdrSG(PINTNETSG pSG)
2091{
2092 Assert(pSG->cSegsUsed);
2093 Assert(pSG->cSegsAlloc >= pSG->cSegsUsed);
2094 Assert(pSG->aSegs[0].cb >= VBOXNETFLT_PACKET_ETHEADER_SIZE);
2095
2096 if (!pSG->cSegsUsed)
2097 return NULL;
2098
2099 if (pSG->aSegs[0].cb < VBOXNETFLT_PACKET_ETHEADER_SIZE)
2100 return NULL;
2101
2102 return (PRTNETETHERHDR)pSG->aSegs[0].pv;
2103}
2104
2105DECLHIDDEN(bool) vboxNetFltWinCheckMACs(PNDIS_PACKET pPacket, PRTMAC pDst, PRTMAC pSrc)
2106{
2107 PRTNETETHERHDR pHdr = vboxNetFltWinGetEthHdr(pPacket);
2108 Assert(pHdr);
2109
2110 if (!pHdr)
2111 return false;
2112
2113 if (pDst && memcmp(pDst, &pHdr->DstMac, sizeof(RTMAC)))
2114 return false;
2115
2116 if (pSrc && memcmp(pSrc, &pHdr->SrcMac, sizeof(RTMAC)))
2117 return false;
2118
2119 return true;
2120}
2121
2122DECLHIDDEN(bool) vboxNetFltWinCheckMACsSG(PINTNETSG pSG, PRTMAC pDst, PRTMAC pSrc)
2123{
2124 PRTNETETHERHDR pHdr = vboxNetFltWinGetEthHdrSG(pSG);
2125 Assert(pHdr);
2126
2127 if (!pHdr)
2128 return false;
2129
2130 if (pDst && memcmp(pDst, &pHdr->DstMac, sizeof(RTMAC)))
2131 return false;
2132
2133 if (pSrc && memcmp(pSrc, &pHdr->SrcMac, sizeof(RTMAC)))
2134 return false;
2135
2136 return true;
2137}
2138#endif
2139
2140# if !defined(VBOX_LOOPBACK_USEFLAGS) || defined(DEBUG_NETFLT_PACKETS)
2141/*
2142 * answers whether the two given packets match based on the packet length and the first cbMatch bytes of the packets
2143 * if cbMatch < 0 matches complete packets.
2144 */
2145DECLHIDDEN(bool) vboxNetFltWinMatchPackets(PNDIS_PACKET pPacket1, PNDIS_PACKET pPacket2, const INT cbMatch)
2146{
2147 UINT cBufCount1;
2148 PNDIS_BUFFER pBuffer1;
2149 UINT uTotalPacketLength1;
2150 uint8_t* pMemBuf1;
2151 UINT cbLength1 = 0;
2152
2153 UINT cBufCount2;
2154 PNDIS_BUFFER pBuffer2;
2155 UINT uTotalPacketLength2;
2156 uint8_t* pMemBuf2;
2157 UINT cbLength2 = 0;
2158 bool bMatch = true;
2159
2160#ifdef DEBUG_NETFLT_PACKETS
2161 bool bCompleteMatch = false;
2162#endif
2163
2164 NdisQueryPacket(pPacket1, NULL, &cBufCount1, &pBuffer1, &uTotalPacketLength1);
2165 NdisQueryPacket(pPacket2, NULL, &cBufCount2, &pBuffer2, &uTotalPacketLength2);
2166
2167 Assert(pBuffer1);
2168 Assert(pBuffer2);
2169
2170 if (uTotalPacketLength1 != uTotalPacketLength2)
2171 {
2172 bMatch = false;
2173 }
2174 else
2175 {
2176 UINT ucbLength2Match = 0;
2177 UINT ucbMatch;
2178 if (cbMatch < 0 || (UINT)cbMatch > uTotalPacketLength1)
2179 {
2180 /* NOTE: assuming uTotalPacketLength1 == uTotalPacketLength2*/
2181 ucbMatch = uTotalPacketLength1;
2182#ifdef DEBUG_NETFLT_PACKETS
2183 bCompleteMatch = true;
2184#endif
2185 }
2186 else
2187 {
2188 ucbMatch = (UINT)cbMatch;
2189 }
2190
2191 for (;;)
2192 {
2193 if (!cbLength1)
2194 {
2195 NdisQueryBufferSafe(pBuffer1, &pMemBuf1, &cbLength1, NormalPagePriority);
2196 NdisGetNextBuffer(pBuffer1, &pBuffer1);
2197 }
2198 else
2199 {
2200 Assert(pMemBuf1);
2201 Assert(ucbLength2Match);
2202 pMemBuf1 += ucbLength2Match;
2203 }
2204
2205 if (!cbLength2)
2206 {
2207 NdisQueryBufferSafe(pBuffer2, &pMemBuf2, &cbLength2, NormalPagePriority);
2208 NdisGetNextBuffer(pBuffer2, &pBuffer2);
2209 }
2210 else
2211 {
2212 Assert(pMemBuf2);
2213 Assert(ucbLength2Match);
2214 pMemBuf2 += ucbLength2Match;
2215 }
2216
2217 ucbLength2Match = MIN(ucbMatch, cbLength1);
2218 ucbLength2Match = MIN(ucbLength2Match, cbLength2);
2219
2220 if (memcmp((PVOID*)pMemBuf1, (PVOID*)pMemBuf2, ucbLength2Match))
2221 {
2222 bMatch = false;
2223 break;
2224 }
2225
2226 ucbMatch -= ucbLength2Match;
2227 if (!ucbMatch)
2228 break;
2229
2230 cbLength1 -= ucbLength2Match;
2231 cbLength2 -= ucbLength2Match;
2232 }
2233 }
2234
2235#ifdef DEBUG_NETFLT_PACKETS
2236 if (bMatch && !bCompleteMatch)
2237 {
2238 /* check that the packets fully match */
2239 DBG_CHECK_PACKETS(pPacket1, pPacket2);
2240 }
2241#endif
2242
2243 return bMatch;
2244}
2245
2246/*
2247 * answers whether the ndis packet and PINTNETSG match based on the packet length and the first cbMatch bytes of the packet and PINTNETSG
2248 * if cbMatch < 0 matches complete packets.
2249 */
2250DECLHIDDEN(bool) vboxNetFltWinMatchPacketAndSG(PNDIS_PACKET pPacket, PINTNETSG pSG, const INT cbMatch)
2251{
2252 UINT cBufCount1;
2253 PNDIS_BUFFER pBuffer1;
2254 UINT uTotalPacketLength1;
2255 uint8_t* pMemBuf1;
2256 UINT cbLength1 = 0;
2257 UINT uTotalPacketLength2 = pSG->cbTotal;
2258 uint8_t* pMemBuf2;
2259 UINT cbLength2 = 0;
2260 bool bMatch = true;
2261 bool bCompleteMatch = false;
2262 UINT i = 0;
2263
2264 NdisQueryPacket(pPacket, NULL, &cBufCount1, &pBuffer1, &uTotalPacketLength1);
2265
2266 Assert(pBuffer1);
2267 Assert(pSG->cSegsUsed);
2268 Assert(pSG->cSegsAlloc >= pSG->cSegsUsed);
2269
2270 if (uTotalPacketLength1 != uTotalPacketLength2)
2271 {
2272 AssertFailed();
2273 bMatch = false;
2274 }
2275 else
2276 {
2277 UINT ucbLength2Match = 0;
2278 UINT ucbMatch;
2279
2280 if (cbMatch < 0 || (UINT)cbMatch > uTotalPacketLength1)
2281 {
2282 /* NOTE: assuming uTotalPacketLength1 == uTotalPacketLength2*/
2283 ucbMatch = uTotalPacketLength1;
2284 bCompleteMatch = true;
2285 }
2286 else
2287 {
2288 ucbMatch = (UINT)cbMatch;
2289 }
2290
2291 for (;;)
2292 {
2293 if (!cbLength1)
2294 {
2295 NdisQueryBufferSafe(pBuffer1, &pMemBuf1, &cbLength1, NormalPagePriority);
2296 NdisGetNextBuffer(pBuffer1, &pBuffer1);
2297 }
2298 else
2299 {
2300 Assert(pMemBuf1);
2301 Assert(ucbLength2Match);
2302 pMemBuf1 += ucbLength2Match;
2303 }
2304
2305 if (!cbLength2)
2306 {
2307 Assert(i < pSG->cSegsUsed);
2308 pMemBuf2 = (uint8_t*)pSG->aSegs[i].pv;
2309 cbLength2 = pSG->aSegs[i].cb;
2310 i++;
2311 }
2312 else
2313 {
2314 Assert(pMemBuf2);
2315 Assert(ucbLength2Match);
2316 pMemBuf2 += ucbLength2Match;
2317 }
2318
2319 ucbLength2Match = MIN(ucbMatch, cbLength1);
2320 ucbLength2Match = MIN(ucbLength2Match, cbLength2);
2321
2322 if (memcmp((PVOID*)pMemBuf1, (PVOID*)pMemBuf2, ucbLength2Match))
2323 {
2324 bMatch = false;
2325 AssertFailed();
2326 break;
2327 }
2328
2329 ucbMatch -= ucbLength2Match;
2330 if (!ucbMatch)
2331 break;
2332
2333 cbLength1 -= ucbLength2Match;
2334 cbLength2 -= ucbLength2Match;
2335 }
2336 }
2337
2338 if (bMatch && !bCompleteMatch)
2339 {
2340 /* check that the packets fully match */
2341 DBG_CHECK_PACKET_AND_SG(pPacket, pSG);
2342 }
2343 return bMatch;
2344}
2345
2346# if 0
2347/*
2348 * answers whether the two PINTNETSGs match based on the packet length and the first cbMatch bytes of the PINTNETSG
2349 * if cbMatch < 0 matches complete packets.
2350 */
2351static bool vboxNetFltWinMatchSGs(PINTNETSG pSG1, PINTNETSG pSG2, const INT cbMatch)
2352{
2353 UINT uTotalPacketLength1 = pSG1->cbTotal;
2354 PVOID pMemBuf1;
2355 UINT cbLength1 = 0;
2356 UINT i1 = 0;
2357 UINT uTotalPacketLength2 = pSG2->cbTotal;
2358 PVOID pMemBuf2;
2359 UINT cbLength2 = 0;
2360
2361 bool bMatch = true;
2362 bool bCompleteMatch = false;
2363 UINT i2 = 0;
2364
2365 Assert(pSG1->cSegsUsed);
2366 Assert(pSG2->cSegsUsed);
2367 Assert(pSG1->cSegsAlloc >= pSG1->cSegsUsed);
2368 Assert(pSG2->cSegsAlloc >= pSG2->cSegsUsed);
2369
2370 if (uTotalPacketLength1 != uTotalPacketLength2)
2371 {
2372 AssertFailed();
2373 bMatch = false;
2374 }
2375 else
2376 {
2377 UINT ucbMatch;
2378 if (cbMatch < 0 || (UINT)cbMatch > uTotalPacketLength1)
2379 {
2380 /* NOTE: assuming uTotalPacketLength1 == uTotalPacketLength2*/
2381 ucbMatch = uTotalPacketLength1;
2382 bCompleteMatch = true;
2383 }
2384 else
2385 {
2386 ucbMatch = (UINT)cbMatch;
2387 }
2388
2389 do
2390 {
2391 UINT ucbLength2Match;
2392 if (!cbLength1)
2393 {
2394 Assert(i1 < pSG1->cSegsUsed);
2395 pMemBuf1 = pSG1->aSegs[i1].pv;
2396 cbLength1 = pSG1->aSegs[i1].cb;
2397 i1++;
2398 }
2399
2400 if (!cbLength2)
2401 {
2402 Assert(i2 < pSG2->cSegsUsed);
2403 pMemBuf2 = pSG2->aSegs[i2].pv;
2404 cbLength2 = pSG2->aSegs[i2].cb;
2405 i2++;
2406 }
2407
2408 ucbLength2Match = MIN(ucbMatch, cbLength1);
2409 ucbLength2Match = MIN(ucbLength2Match, cbLength2);
2410
2411 if (memcmp(pMemBuf1, pMemBuf2, ucbLength2Match))
2412 {
2413 bMatch = false;
2414 AssertFailed();
2415 break;
2416 }
2417 ucbMatch -= ucbLength2Match;
2418 cbLength1 -= ucbLength2Match;
2419 cbLength2 -= ucbLength2Match;
2420 } while (ucbMatch);
2421 }
2422
2423 if (bMatch && !bCompleteMatch)
2424 {
2425 /* check that the packets fully match */
2426 DBG_CHECK_SGS(pSG1, pSG2);
2427 }
2428 return bMatch;
2429}
2430# endif
2431# endif
2432#endif
2433
2434static void vboxNetFltWinFiniNetFltBase()
2435{
2436 do
2437 {
2438 vboxNetFltDeleteGlobals(&g_VBoxNetFltGlobals);
2439
2440 /*
2441 * Undo the work done during start (in reverse order).
2442 */
2443 memset(&g_VBoxNetFltGlobals, 0, sizeof(g_VBoxNetFltGlobals));
2444
2445 RTLogDestroy(RTLogRelSetDefaultInstance(NULL));
2446 RTLogDestroy(RTLogSetDefaultInstance(NULL));
2447
2448 RTR0Term();
2449 } while (0);
2450}
2451
2452/*
2453 * Defines max timeout for waiting for driver unloading
2454 * (3000 * 100 ms = 5 minutes)
2455 */
2456#define MAX_UNLOAD_PROBES 3000
2457
2458static int vboxNetFltWinFiniIdc()
2459{
2460 int rc;
2461 int i;
2462
2463 vboxNetFltWinStopInitIdcProbing();
2464
2465 if (g_bVBoxIdcInitialized)
2466 {
2467 for (i = 0; (rc = vboxNetFltTryDeleteIdc(&g_VBoxNetFltGlobals)) == VERR_WRONG_ORDER
2468 && i < MAX_UNLOAD_PROBES; i++)
2469 {
2470 RTThreadSleep(100);
2471 }
2472 if (i == MAX_UNLOAD_PROBES)
2473 {
2474 // seems something hungs in driver
2475 LogFlow(("vboxNetFltWinFiniIdc - Can't delete Idc. pInH=%p cFRefs=%d fIDcOpen=%s",
2476 g_VBoxNetFltGlobals.pInstanceHead, g_VBoxNetFltGlobals.cFactoryRefs,
2477 g_VBoxNetFltGlobals.fIDCOpen ? "true" : "false"));
2478 LogFlow(("vboxNetFltWinFiniIdc g_VBoxNetFltGlobalsWin cDvRefs=%d hDev=%x pDev=%p Mp=%x \n",
2479 g_VBoxNetFltGlobalsWin.cDeviceRefs, g_VBoxNetFltGlobalsWin.hDevice,
2480 g_VBoxNetFltGlobalsWin.pDevObj, g_VBoxNetFltGlobalsWin.Mp.hMiniport));
2481 Assert(i == MAX_UNLOAD_PROBES);
2482 return VERR_WRONG_ORDER;
2483 }
2484
2485 if (RT_SUCCESS(rc))
2486 {
2487 g_bVBoxIdcInitialized = false;
2488 }
2489 }
2490 else
2491 {
2492 rc = VINF_SUCCESS;
2493 }
2494 return rc;
2495
2496}
2497
2498static int vboxNetFltWinFiniNetFlt()
2499{
2500 int rc = vboxNetFltWinFiniIdc();
2501 if (RT_SUCCESS(rc))
2502 {
2503 vboxNetFltWinFiniNetFltBase();
2504 }
2505 return rc;
2506}
2507
2508/**
2509 * base netflt initialization
2510 */
2511static int vboxNetFltWinInitNetFltBase()
2512{
2513 int rc;
2514
2515 do
2516 {
2517 Assert(!g_bVBoxIdcInitialized);
2518
2519 rc = RTR0Init(0);
2520 if (!RT_SUCCESS(rc))
2521 {
2522 break;
2523 }
2524
2525 memset(&g_VBoxNetFltGlobals, 0, sizeof(g_VBoxNetFltGlobals));
2526 rc = vboxNetFltInitGlobals(&g_VBoxNetFltGlobals);
2527 if (!RT_SUCCESS(rc))
2528 {
2529 RTR0Term();
2530 break;
2531 }
2532 }while (0);
2533
2534 return rc;
2535}
2536
2537/**
2538 * initialize IDC
2539 */
2540static int vboxNetFltWinInitIdc()
2541{
2542 int rc;
2543
2544 do
2545 {
2546 if (g_bVBoxIdcInitialized)
2547 {
2548 rc = VINF_ALREADY_INITIALIZED;
2549 break;
2550 }
2551
2552 /*
2553 * connect to the support driver.
2554 *
2555 * This will call back vboxNetFltOsOpenSupDrv (and maybe vboxNetFltOsCloseSupDrv)
2556 * for establishing the connect to the support driver.
2557 */
2558 rc = vboxNetFltInitIdc(&g_VBoxNetFltGlobals);
2559 if (!RT_SUCCESS(rc))
2560 {
2561 break;
2562 }
2563
2564 g_bVBoxIdcInitialized = true;
2565 } while (0);
2566
2567 return rc;
2568}
2569
2570static VOID vboxNetFltWinInitIdcProbingWorker(PVOID pvContext)
2571{
2572 PINIT_IDC_INFO pInitIdcInfo = (PINIT_IDC_INFO)pvContext;
2573 int rc = vboxNetFltWinInitIdc();
2574 if (RT_FAILURE(rc))
2575 {
2576 bool bInterupted = ASMAtomicUoReadBool(&pInitIdcInfo->bStop);
2577 if (!bInterupted)
2578 {
2579 RTThreadSleep(1000); /* 1 s */
2580 bInterupted = ASMAtomicUoReadBool(&pInitIdcInfo->bStop);
2581 if (!bInterupted)
2582 {
2583 vboxNetFltWinJobEnqueueJob(&g_VBoxJobQueue, &pInitIdcInfo->Job, false);
2584 return;
2585 }
2586 }
2587
2588 /* it's interrupted */
2589 rc = VERR_INTERRUPTED;
2590 }
2591
2592 ASMAtomicUoWriteS32(&pInitIdcInfo->rc, rc);
2593 KeSetEvent(&pInitIdcInfo->hCompletionEvent, 0, FALSE);
2594}
2595
2596static int vboxNetFltWinStopInitIdcProbing()
2597{
2598 if (!g_VBoxInitIdcInfo.bInitialized)
2599 return VERR_INVALID_STATE;
2600
2601 ASMAtomicUoWriteBool(&g_VBoxInitIdcInfo.bStop, true);
2602 KeWaitForSingleObject(&g_VBoxInitIdcInfo.hCompletionEvent, Executive, KernelMode, FALSE, NULL);
2603
2604 return g_VBoxInitIdcInfo.rc;
2605}
2606
2607static int vboxNetFltWinStartInitIdcProbing()
2608{
2609 Assert(!g_bVBoxIdcInitialized);
2610 KeInitializeEvent(&g_VBoxInitIdcInfo.hCompletionEvent, NotificationEvent, FALSE);
2611 g_VBoxInitIdcInfo.bStop = false;
2612 g_VBoxInitIdcInfo.bInitialized = true;
2613 vboxNetFltWinJobInit(&g_VBoxInitIdcInfo.Job, vboxNetFltWinInitIdcProbingWorker, &g_VBoxInitIdcInfo, false);
2614 vboxNetFltWinJobEnqueueJob(&g_VBoxJobQueue, &g_VBoxInitIdcInfo.Job, false);
2615 return VINF_SUCCESS;
2616}
2617
2618static int vboxNetFltWinInitNetFlt()
2619{
2620 int rc;
2621
2622 do
2623 {
2624 rc = vboxNetFltWinInitNetFltBase();
2625 if (RT_FAILURE(rc))
2626 {
2627 AssertFailed();
2628 break;
2629 }
2630
2631 /*
2632 * connect to the support driver.
2633 *
2634 * This will call back vboxNetFltOsOpenSupDrv (and maybe vboxNetFltOsCloseSupDrv)
2635 * for establishing the connect to the support driver.
2636 */
2637 rc = vboxNetFltWinInitIdc();
2638 if (RT_FAILURE(rc))
2639 {
2640 AssertFailed();
2641 vboxNetFltWinFiniNetFltBase();
2642 break;
2643 }
2644 } while (0);
2645
2646 return rc;
2647}
2648
2649/* detach*/
2650static int vboxNetFltWinDeleteInstance(PVBOXNETFLTINS pThis)
2651{
2652 LogFlow(("vboxNetFltWinDeleteInstance: pThis=0x%p \n", pThis));
2653
2654 Assert(KeGetCurrentIrql() < DISPATCH_LEVEL);
2655 Assert(pThis);
2656 Assert(pThis->fDisconnectedFromHost);
2657 Assert(!pThis->fRediscoveryPending);
2658 Assert(pThis->enmTrunkState != INTNETTRUNKIFSTATE_ACTIVE);
2659#ifndef VBOXNETADP
2660 Assert(pThis->u.s.WinIf.PtState.OpState == kVBoxNetDevOpState_Deinitialized);
2661 Assert(!pThis->u.s.WinIf.hBinding);
2662#endif
2663 Assert(pThis->u.s.WinIf.MpState.OpState == kVBoxNetDevOpState_Deinitialized);
2664#ifndef VBOXNETFLT_NO_PACKET_QUEUE
2665 Assert(!pThis->u.s.PacketQueueWorker.pSG);
2666#endif
2667
2668 RTSemMutexDestroy(pThis->u.s.hWinIfMutex);
2669
2670 vboxNetFltWinDrvDereference();
2671
2672 return VINF_SUCCESS;
2673}
2674
2675static NDIS_STATUS vboxNetFltWinDisconnectIt(PVBOXNETFLTINS pInstance)
2676{
2677#ifndef VBOXNETFLT_NO_PACKET_QUEUE
2678 vboxNetFltWinQuFiniPacketQueue(pInstance);
2679#endif
2680 return NDIS_STATUS_SUCCESS;
2681}
2682
2683/* detach*/
2684DECLHIDDEN(NDIS_STATUS) vboxNetFltWinDetachFromInterface(PVBOXNETFLTINS pNetFlt, bool bOnUnbind)
2685{
2686 NDIS_STATUS Status;
2687 int rc;
2688 LogFlow((__FUNCTION__": pThis=%0xp\n", pNetFlt));
2689
2690 Assert(KeGetCurrentIrql() < DISPATCH_LEVEL);
2691 Assert(pNetFlt);
2692
2693 /* paranoia to ensure the instance is not removed while we're waiting on the mutex
2694 * in case ndis does something unpredictable, e.g. calls our miniport halt independently
2695 * from protocol unbind and concurrently with it*/
2696 vboxNetFltRetain(pNetFlt, false);
2697
2698 rc = RTSemMutexRequest(pNetFlt->u.s.hWinIfMutex, RT_INDEFINITE_WAIT);
2699 if (RT_SUCCESS(rc))
2700 {
2701 Assert(vboxNetFltWinGetWinIfState(pNetFlt) == kVBoxWinIfState_Connected);
2702 Assert(vboxNetFltWinGetOpState(&pNetFlt->u.s.WinIf.MpState) == kVBoxNetDevOpState_Initialized);
2703#ifndef VBOXNETADP
2704 Assert(vboxNetFltWinGetOpState(&pNetFlt->u.s.WinIf.PtState) == kVBoxNetDevOpState_Initialized);
2705#endif
2706 if (vboxNetFltWinGetWinIfState(pNetFlt) == kVBoxWinIfState_Connected)
2707 {
2708 vboxNetFltWinSetWinIfState(pNetFlt, kVBoxWinIfState_Disconnecting);
2709#ifndef VBOXNETADP
2710 Status = vboxNetFltWinPtDoUnbinding(pNetFlt, bOnUnbind);
2711#else
2712 Status = vboxNetFltWinMpDoDeinitialization(pNetFlt);
2713#endif
2714 Assert(Status == NDIS_STATUS_SUCCESS);
2715
2716 vboxNetFltWinSetWinIfState(pNetFlt, kVBoxWinIfState_Disconnected);
2717 Assert(vboxNetFltWinGetOpState(&pNetFlt->u.s.WinIf.MpState) == kVBoxNetDevOpState_Deinitialized);
2718#ifndef VBOXNETADP
2719 Assert(vboxNetFltWinGetOpState(&pNetFlt->u.s.WinIf.PtState) == kVBoxNetDevOpState_Deinitialized);
2720#endif
2721 vboxNetFltWinPtFiniWinIf(&pNetFlt->u.s.WinIf);
2722
2723 /* we're unbinding, make an unbind-related release */
2724 vboxNetFltRelease(pNetFlt, false);
2725 }
2726 else
2727 {
2728 AssertBreakpoint();
2729#ifndef VBOXNETADP
2730 pNetFlt->u.s.WinIf.OpenCloseStatus = NDIS_STATUS_FAILURE;
2731#endif
2732 if (!bOnUnbind)
2733 {
2734 vboxNetFltWinSetOpState(&pNetFlt->u.s.WinIf.MpState, kVBoxNetDevOpState_Deinitialized);
2735 }
2736 Status = NDIS_STATUS_FAILURE;
2737 }
2738 RTSemMutexRelease(pNetFlt->u.s.hWinIfMutex);
2739 }
2740 else
2741 {
2742 AssertBreakpoint();
2743 Status = NDIS_STATUS_FAILURE;
2744 }
2745
2746 /* release for the retain we made before waining on the mutex */
2747 vboxNetFltRelease(pNetFlt, false);
2748
2749 return Status;
2750}
2751
2752
2753/**
2754 * Checks if the host (not us) has put the adapter in promiscuous mode.
2755 *
2756 * @returns true if promiscuous, false if not.
2757 * @param pThis The instance.
2758 */
2759static bool vboxNetFltWinIsPromiscuous2(PVBOXNETFLTINS pThis)
2760{
2761#ifndef VBOXNETADP
2762 if (VBOXNETFLT_PROMISCUOUS_SUPPORTED(pThis))
2763 {
2764 bool bPromiscuous;
2765 if (!vboxNetFltWinReferenceWinIf(pThis))
2766 return false;
2767
2768 bPromiscuous = (pThis->u.s.WinIf.fUpperProtocolSetFilter & NDIS_PACKET_TYPE_PROMISCUOUS) == NDIS_PACKET_TYPE_PROMISCUOUS;
2769 /*vboxNetFltWinIsPromiscuous(pAdapt);*/
2770
2771 vboxNetFltWinDereferenceWinIf(pThis);
2772 return bPromiscuous;
2773 }
2774 return false;
2775#else
2776 return true;
2777#endif
2778}
2779
2780
2781/**
2782 * Report the MAC address, promiscuous mode setting, GSO capabilities and
2783 * no-preempt destinations to the internal network.
2784 *
2785 * Does nothing if we're not currently connected to an internal network.
2786 *
2787 * @param pThis The instance data.
2788 */
2789static void vboxNetFltWinReportStuff(PVBOXNETFLTINS pThis)
2790{
2791 /** @todo Keep these up to date, esp. the promiscuous mode bit. */
2792 if (pThis->pSwitchPort
2793 && vboxNetFltTryRetainBusyNotDisconnected(pThis))
2794 {
2795 pThis->pSwitchPort->pfnReportMacAddress(pThis->pSwitchPort, &pThis->u.s.MacAddr);
2796 pThis->pSwitchPort->pfnReportPromiscuousMode(pThis->pSwitchPort,
2797 vboxNetFltWinIsPromiscuous2(pThis));
2798 pThis->pSwitchPort->pfnReportGsoCapabilities(pThis->pSwitchPort, 0,
2799 INTNETTRUNKDIR_WIRE | INTNETTRUNKDIR_HOST);
2800 /** @todo We should be able to do pfnXmit at DISPATCH_LEVEL... */
2801 pThis->pSwitchPort->pfnReportNoPreemptDsts(pThis->pSwitchPort, 0 /* none */);
2802 vboxNetFltRelease(pThis, true /*fBusy*/);
2803 }
2804}
2805
2806/**
2807 * Worker for vboxNetFltWinAttachToInterface.
2808 *
2809 * @param pAttachInfo Structure for communicating with
2810 * vboxNetFltWinAttachToInterface.
2811 */
2812static void vboxNetFltWinAttachToInterfaceWorker(PATTACH_INFO pAttachInfo)
2813{
2814 PVBOXNETFLTINS pThis = pAttachInfo->pNetFltIf;
2815 NDIS_STATUS Status = NDIS_STATUS_SUCCESS;
2816 int rc;
2817
2818 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
2819
2820 /* to ensure we're not removed while we're here */
2821 vboxNetFltRetain(pThis, false);
2822
2823 rc = RTSemMutexRequest(pThis->u.s.hWinIfMutex, RT_INDEFINITE_WAIT);
2824 if (RT_SUCCESS(rc))
2825 {
2826 Assert(vboxNetFltWinGetWinIfState(pThis) == kVBoxWinIfState_Disconnected);
2827 Assert(vboxNetFltWinGetOpState(&pThis->u.s.WinIf.MpState) == kVBoxNetDevOpState_Deinitialized);
2828#ifndef VBOXNETADP
2829 Assert(vboxNetFltWinGetOpState(&pThis->u.s.WinIf.PtState) == kVBoxNetDevOpState_Deinitialized);
2830#endif
2831 if (vboxNetFltWinGetWinIfState(pThis) == kVBoxWinIfState_Disconnected)
2832 {
2833 if (pAttachInfo->fRediscovery)
2834 {
2835 /* rediscovery means adaptor bind is performed while intnet is already using it
2836 * i.e. adaptor was unbound while being used by intnet and now being bound back again */
2837 Assert(((VBOXNETFTLINSSTATE)ASMAtomicUoReadU32((uint32_t volatile *)&pThis->enmState)) == kVBoxNetFltInsState_Connected);
2838 }
2839#ifndef VBOXNETADP
2840 Status = vboxNetFltWinPtInitWinIf(&pThis->u.s.WinIf, pAttachInfo->pCreateContext->pOurName);
2841#else
2842 Status = vboxNetFltWinPtInitWinIf(&pThis->u.s.WinIf);
2843#endif
2844 if (Status == NDIS_STATUS_SUCCESS)
2845 {
2846 vboxNetFltWinSetWinIfState(pThis, kVBoxWinIfState_Connecting);
2847
2848#ifndef VBOXNETADP
2849 Status = vboxNetFltWinPtDoBinding(pThis, pAttachInfo->pCreateContext->pOurName, pAttachInfo->pCreateContext->pBindToName);
2850#else
2851 Status = vboxNetFltWinMpDoInitialization(pThis, pAttachInfo->pCreateContext->hMiniportAdapter, pAttachInfo->pCreateContext->hWrapperConfigurationContext);
2852#endif
2853 if (Status == NDIS_STATUS_SUCCESS)
2854 {
2855 if (!pAttachInfo->fRediscovery)
2856 {
2857 vboxNetFltWinDrvReference();
2858 }
2859#ifndef VBOXNETADP
2860 if (pThis->u.s.WinIf.OpenCloseStatus == NDIS_STATUS_SUCCESS)
2861#endif
2862 {
2863 vboxNetFltWinSetWinIfState(pThis, kVBoxWinIfState_Connected);
2864#ifndef VBOXNETADP
2865 Assert(vboxNetFltWinGetOpState(&pThis->u.s.WinIf.PtState) == kVBoxNetDevOpState_Initialized);
2866#endif
2867 /* 4. mark as connected */
2868 RTSpinlockAcquire(pThis->hSpinlock);
2869 ASMAtomicUoWriteBool(&pThis->fDisconnectedFromHost, false);
2870 RTSpinlockRelease(pThis->hSpinlock);
2871
2872 pAttachInfo->Status = VINF_SUCCESS;
2873 pAttachInfo->pCreateContext->Status = NDIS_STATUS_SUCCESS;
2874
2875 RTSemMutexRelease(pThis->u.s.hWinIfMutex);
2876
2877 vboxNetFltRelease(pThis, false);
2878
2879 /* 5. Report MAC address, promiscuousness and GSO capabilities. */
2880 vboxNetFltWinReportStuff(pThis);
2881
2882 return;
2883 }
2884 AssertBreakpoint();
2885
2886 if (!pAttachInfo->fRediscovery)
2887 {
2888 vboxNetFltWinDrvDereference();
2889 }
2890#ifndef VBOXNETADP
2891 vboxNetFltWinPtDoUnbinding(pThis, true);
2892#else
2893 vboxNetFltWinMpDoDeinitialization(pThis);
2894#endif
2895 }
2896 AssertBreakpoint();
2897 vboxNetFltWinPtFiniWinIf(&pThis->u.s.WinIf);
2898 }
2899 AssertBreakpoint();
2900 vboxNetFltWinSetWinIfState(pThis, kVBoxWinIfState_Disconnected);
2901 Assert(vboxNetFltWinGetOpState(&pThis->u.s.WinIf.MpState) == kVBoxNetDevOpState_Deinitialized);
2902#ifndef VBOXNETADP
2903 Assert(vboxNetFltWinGetOpState(&pThis->u.s.WinIf.PtState) == kVBoxNetDevOpState_Deinitialized);
2904#endif
2905 }
2906 AssertBreakpoint();
2907
2908 pAttachInfo->Status = VERR_GENERAL_FAILURE;
2909 pAttachInfo->pCreateContext->Status = Status;
2910 RTSemMutexRelease(pThis->u.s.hWinIfMutex);
2911 }
2912 else
2913 {
2914 AssertBreakpoint();
2915 pAttachInfo->Status = rc;
2916 }
2917
2918 vboxNetFltRelease(pThis, false);
2919
2920 return;
2921}
2922
2923/**
2924 * Common code for vboxNetFltOsInitInstance and
2925 * vboxNetFltOsMaybeRediscovered.
2926 *
2927 * @returns IPRT status code.
2928 * @param pThis The instance.
2929 * @param fRediscovery True if vboxNetFltOsMaybeRediscovered is calling,
2930 * false if it's vboxNetFltOsInitInstance.
2931 */
2932static int vboxNetFltWinAttachToInterface(PVBOXNETFLTINS pThis, void * pContext, bool fRediscovery)
2933{
2934 ATTACH_INFO Info;
2935 Info.pNetFltIf = pThis;
2936 Info.fRediscovery = fRediscovery;
2937 Info.pCreateContext = (PCREATE_INSTANCE_CONTEXT)pContext;
2938
2939 vboxNetFltWinAttachToInterfaceWorker(&Info);
2940
2941 return Info.Status;
2942}
2943static NTSTATUS vboxNetFltWinPtDevDispatch(IN PDEVICE_OBJECT pDevObj, IN PIRP pIrp)
2944{
2945 PIO_STACK_LOCATION pIrpSl = IoGetCurrentIrpStackLocation(pIrp);;
2946 NTSTATUS Status = STATUS_SUCCESS;
2947
2948 switch (pIrpSl->MajorFunction)
2949 {
2950 case IRP_MJ_DEVICE_CONTROL:
2951 Status = STATUS_NOT_SUPPORTED;
2952 break;
2953 case IRP_MJ_CREATE:
2954 case IRP_MJ_CLEANUP:
2955 case IRP_MJ_CLOSE:
2956 break;
2957 default:
2958 Assert(0);
2959 break;
2960 }
2961
2962 pIrp->IoStatus.Status = Status;
2963 IoCompleteRequest(pIrp, IO_NO_INCREMENT);
2964
2965 return Status;
2966}
2967
2968static NDIS_STATUS vboxNetFltWinDevCreate(PVBOXNETFLTGLOBALS_WIN pGlobals)
2969{
2970 NDIS_STRING DevName, LinkName;
2971 PDRIVER_DISPATCH aMajorFunctions[IRP_MJ_MAXIMUM_FUNCTION+1];
2972 NdisInitUnicodeString(&DevName, VBOXNETFLT_NAME_DEVICE);
2973 NdisInitUnicodeString(&LinkName, VBOXNETFLT_NAME_LINK);
2974
2975 Assert(!pGlobals->hDevice);
2976 Assert(!pGlobals->pDevObj);
2977 NdisZeroMemory(aMajorFunctions, sizeof (aMajorFunctions));
2978 aMajorFunctions[IRP_MJ_CREATE] = vboxNetFltWinPtDevDispatch;
2979 aMajorFunctions[IRP_MJ_CLEANUP] = vboxNetFltWinPtDevDispatch;
2980 aMajorFunctions[IRP_MJ_CLOSE] = vboxNetFltWinPtDevDispatch;
2981 aMajorFunctions[IRP_MJ_DEVICE_CONTROL] = vboxNetFltWinPtDevDispatch;
2982
2983 NDIS_STATUS Status = NdisMRegisterDevice(pGlobals->Mp.hNdisWrapper,
2984 &DevName, &LinkName,
2985 aMajorFunctions,
2986 &pGlobals->pDevObj,
2987 &pGlobals->hDevice);
2988 Assert(Status == NDIS_STATUS_SUCCESS);
2989 return Status;
2990}
2991
2992static NDIS_STATUS vboxNetFltWinDevDestroy(PVBOXNETFLTGLOBALS_WIN pGlobals)
2993{
2994 Assert(pGlobals->hDevice);
2995 Assert(pGlobals->pDevObj);
2996 NDIS_STATUS Status = NdisMDeregisterDevice(pGlobals->hDevice);
2997 Assert(Status == NDIS_STATUS_SUCCESS);
2998 if (Status == NDIS_STATUS_SUCCESS)
2999 {
3000 pGlobals->hDevice = NULL;
3001 pGlobals->pDevObj = NULL;
3002 }
3003 return Status;
3004}
3005
3006static NDIS_STATUS vboxNetFltWinDevCreateReference(PVBOXNETFLTGLOBALS_WIN pGlobals)
3007{
3008 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
3009 NDIS_STATUS Status = KeWaitForSingleObject(&pGlobals->SynchEvent, Executive, KernelMode, FALSE, NULL);
3010 Assert(Status == STATUS_SUCCESS);
3011 if (Status == STATUS_SUCCESS)
3012 {
3013 Assert(pGlobals->cDeviceRefs >= 0);
3014 if (++pGlobals->cDeviceRefs == 1)
3015 {
3016 Status = vboxNetFltWinDevCreate(pGlobals);
3017 if (Status == NDIS_STATUS_SUCCESS)
3018 {
3019 ObReferenceObject(pGlobals->pDevObj);
3020 }
3021 }
3022 else
3023 {
3024 Status = NDIS_STATUS_SUCCESS;
3025 }
3026 KeSetEvent(&pGlobals->SynchEvent, 0, FALSE);
3027 }
3028 else
3029 {
3030 /* should never happen actually */
3031 Assert(0);
3032 Status = NDIS_STATUS_FAILURE;
3033 }
3034 return Status;
3035}
3036
3037static NDIS_STATUS vboxNetFltWinDevDereference(PVBOXNETFLTGLOBALS_WIN pGlobals)
3038{
3039 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
3040 NDIS_STATUS Status = KeWaitForSingleObject(&pGlobals->SynchEvent, Executive, KernelMode, FALSE, NULL);
3041 Assert(Status == STATUS_SUCCESS);
3042 if (Status == STATUS_SUCCESS)
3043 {
3044 Assert(pGlobals->cDeviceRefs > 0);
3045 if (!(--pGlobals->cDeviceRefs))
3046 {
3047 ObDereferenceObject(pGlobals->pDevObj);
3048 Status = vboxNetFltWinDevDestroy(pGlobals);
3049 }
3050 else
3051 {
3052 Status = NDIS_STATUS_SUCCESS;
3053 }
3054 KeSetEvent(&pGlobals->SynchEvent, 0, FALSE);
3055 }
3056 else
3057 {
3058 /* should never happen actually */
3059 Assert(0);
3060 Status = NDIS_STATUS_FAILURE;
3061 }
3062 return Status;
3063}
3064
3065/* reference the driver module to prevent driver unload */
3066DECLHIDDEN(void) vboxNetFltWinDrvReference()
3067{
3068 vboxNetFltWinDevCreateReference(&g_VBoxNetFltGlobalsWin);
3069}
3070
3071/* dereference the driver module to prevent driver unload */
3072DECLHIDDEN(void) vboxNetFltWinDrvDereference()
3073{
3074 vboxNetFltWinDevDereference(&g_VBoxNetFltGlobalsWin);
3075}
3076
3077/*
3078 *
3079 * The OS specific interface definition
3080 *
3081 */
3082
3083
3084bool vboxNetFltOsMaybeRediscovered(PVBOXNETFLTINS pThis)
3085{
3086 /* AttachToInterface true if disconnected */
3087 return !ASMAtomicUoReadBool(&pThis->fDisconnectedFromHost);
3088}
3089
3090int vboxNetFltPortOsXmit(PVBOXNETFLTINS pThis, void *pvIfData, PINTNETSG pSG, uint32_t fDst)
3091{
3092 int rc = VINF_SUCCESS;
3093 uint32_t cRefs = 0;
3094#ifndef VBOXNETADP
3095 if (fDst & INTNETTRUNKDIR_WIRE)
3096 {
3097 cRefs++;
3098 }
3099 if (fDst & INTNETTRUNKDIR_HOST)
3100 {
3101 cRefs++;
3102 }
3103#else
3104 if (fDst & INTNETTRUNKDIR_WIRE || fDst & INTNETTRUNKDIR_HOST)
3105 {
3106 cRefs = 1;
3107 }
3108#endif
3109
3110 AssertReturn(cRefs, VINF_SUCCESS);
3111
3112 if (!vboxNetFltWinIncReferenceWinIf(pThis, cRefs))
3113 {
3114 return VERR_GENERAL_FAILURE;
3115 }
3116#ifndef VBOXNETADP
3117 if (fDst & INTNETTRUNKDIR_WIRE)
3118 {
3119 PNDIS_PACKET pPacket;
3120
3121 pPacket = vboxNetFltWinNdisPacketFromSG(pThis, pSG, NULL /*pBufToFree*/,
3122 true /*fToWire*/, true /*fCopyMemory*/);
3123
3124 if (pPacket)
3125 {
3126 NDIS_STATUS fStatus;
3127
3128#ifndef VBOX_LOOPBACK_USEFLAGS
3129 /* force "don't loopback" flags to prevent loopback branch invocation in any case
3130 * to avoid ndis misbehave */
3131 NdisGetPacketFlags(pPacket) |= g_VBoxNetFltGlobalsWin.fPacketDontLoopBack;
3132#else
3133 /* this is done by default in vboxNetFltWinNdisPacketFromSG */
3134#endif
3135
3136#if defined(DEBUG_NETFLT_PACKETS) || !defined(VBOX_LOOPBACK_USEFLAGS)
3137 vboxNetFltWinLbPutSendPacket(pThis, pPacket, true /* bFromIntNet */);
3138#endif
3139 NdisSend(&fStatus, pThis->u.s.WinIf.hBinding, pPacket);
3140 if (fStatus != NDIS_STATUS_PENDING)
3141 {
3142#if defined(DEBUG_NETFLT_PACKETS) || !defined(VBOX_LOOPBACK_USEFLAGS)
3143 /* the status is NOT pending, complete the packet */
3144 bool bTmp = vboxNetFltWinLbRemoveSendPacket(pThis, pPacket);
3145 Assert(bTmp);
3146#endif
3147 if (!NT_SUCCESS(fStatus))
3148 {
3149 /* TODO: convert status to VERR_xxx */
3150 rc = VERR_GENERAL_FAILURE;
3151 }
3152
3153 vboxNetFltWinFreeSGNdisPacket(pPacket, true);
3154 }
3155 else
3156 {
3157 /* pending, dereference on packet complete */
3158 cRefs--;
3159 }
3160 }
3161 else
3162 {
3163 AssertFailed();
3164 rc = VERR_NO_MEMORY;
3165 }
3166 }
3167#endif
3168
3169#ifndef VBOXNETADP
3170 if (fDst & INTNETTRUNKDIR_HOST)
3171#else
3172 if (cRefs)
3173#endif
3174 {
3175 PNDIS_PACKET pPacket = vboxNetFltWinNdisPacketFromSG(pThis, pSG, NULL /*pBufToFree*/,
3176 false /*fToWire*/, true /*fCopyMemory*/);
3177 if (pPacket)
3178 {
3179 NdisMIndicateReceivePacket(pThis->u.s.WinIf.hMiniport, &pPacket, 1);
3180 cRefs--;
3181#ifdef VBOXNETADP
3182 STATISTIC_INCREASE(pThis->u.s.WinIf.cRxSuccess);
3183#endif
3184 }
3185 else
3186 {
3187 AssertFailed();
3188#ifdef VBOXNETADP
3189 STATISTIC_INCREASE(pThis->u.s.WinIf.cRxError);
3190#endif
3191 rc = VERR_NO_MEMORY;
3192 }
3193 }
3194
3195 Assert(cRefs <= 2);
3196
3197 if (cRefs)
3198 {
3199 vboxNetFltWinDecReferenceWinIf(pThis, cRefs);
3200 }
3201
3202 return rc;
3203}
3204
3205void vboxNetFltPortOsSetActive(PVBOXNETFLTINS pThis, bool fActive)
3206{
3207#ifndef VBOXNETADP
3208 NDIS_STATUS Status;
3209#endif
3210 /* we first wait for all pending ops to complete
3211 * this might include all packets queued for processing */
3212 for (;;)
3213 {
3214 if (fActive)
3215 {
3216 if (!pThis->u.s.cModePassThruRefs)
3217 {
3218 break;
3219 }
3220 }
3221 else
3222 {
3223 if (!pThis->u.s.cModeNetFltRefs)
3224 {
3225 break;
3226 }
3227 }
3228 vboxNetFltWinSleep(2);
3229 }
3230
3231 if (!vboxNetFltWinReferenceWinIf(pThis))
3232 return;
3233#ifndef VBOXNETADP
3234
3235 if (fActive)
3236 {
3237#ifdef DEBUG_misha
3238 NDIS_PHYSICAL_MEDIUM PhMedium;
3239 bool bPromiscSupported;
3240
3241 Status = vboxNetFltWinQueryPhysicalMedium(pThis, &PhMedium);
3242 if (Status != NDIS_STATUS_SUCCESS)
3243 {
3244
3245 LogRel(("vboxNetFltWinQueryPhysicalMedium failed, Status (0x%x), setting medium to NdisPhysicalMediumUnspecified\n", Status));
3246 Assert(Status == NDIS_STATUS_NOT_SUPPORTED);
3247 if (Status != NDIS_STATUS_NOT_SUPPORTED)
3248 {
3249 LogRel(("vboxNetFltWinQueryPhysicalMedium failed, Status (0x%x), setting medium to NdisPhysicalMediumUnspecified\n", Status));
3250 }
3251 PhMedium = NdisPhysicalMediumUnspecified;
3252 }
3253 else
3254 {
3255 LogRel(("(SUCCESS) vboxNetFltWinQueryPhysicalMedium SUCCESS\n"));
3256 }
3257
3258 bPromiscSupported = (!(PhMedium == NdisPhysicalMediumWirelessWan
3259 || PhMedium == NdisPhysicalMediumWirelessLan
3260 || PhMedium == NdisPhysicalMediumNative802_11
3261 || PhMedium == NdisPhysicalMediumBluetooth
3262 /*|| PhMedium == NdisPhysicalMediumWiMax */
3263 ));
3264
3265 Assert(bPromiscSupported == VBOXNETFLT_PROMISCUOUS_SUPPORTED(pThis));
3266#endif
3267 }
3268
3269 if (VBOXNETFLT_PROMISCUOUS_SUPPORTED(pThis))
3270 {
3271 Status = vboxNetFltWinSetPromiscuous(pThis, fActive);
3272 if (Status != NDIS_STATUS_SUCCESS)
3273 {
3274 LogRel(("vboxNetFltWinSetPromiscuous failed, Status (0x%x), fActive (%d)\n", Status, fActive));
3275 AssertFailed();
3276 }
3277 }
3278#else
3279# ifdef VBOXNETADP_REPORT_DISCONNECTED
3280 if (fActive)
3281 {
3282 NdisMIndicateStatus(pThis->u.s.WinIf.hMiniport,
3283 NDIS_STATUS_MEDIA_CONNECT,
3284 (PVOID)NULL,
3285 0);
3286 }
3287 else
3288 {
3289 NdisMIndicateStatus(pThis->u.s.WinIf.hMiniport,
3290 NDIS_STATUS_MEDIA_DISCONNECT,
3291 (PVOID)NULL,
3292 0);
3293 }
3294#else
3295 if (fActive)
3296 {
3297 /* indicate status change to make the ip settings be re-picked for dhcp */
3298 NdisMIndicateStatus(pThis->u.s.WinIf.hMiniport,
3299 NDIS_STATUS_MEDIA_DISCONNECT,
3300 (PVOID)NULL,
3301 0);
3302
3303 NdisMIndicateStatus(pThis->u.s.WinIf.hMiniport,
3304 NDIS_STATUS_MEDIA_CONNECT,
3305 (PVOID)NULL,
3306 0);
3307 }
3308# endif
3309#endif
3310 vboxNetFltWinDereferenceWinIf(pThis);
3311
3312 return;
3313}
3314
3315#ifndef VBOXNETADP
3316
3317DECLINLINE(bool) vboxNetFltWinIsAddrLinkLocal4(PCRTNETADDRIPV4 pAddr)
3318{
3319 return (pAddr->s.Lo == 0xfea9); /* 169.254 */
3320}
3321
3322DECLINLINE(bool) vboxNetFltWinIsAddrLinkLocal6(PCRTNETADDRIPV6 pAddr)
3323{
3324 return ((pAddr->au8[0] == 0xfe) && ((pAddr->au8[1] & 0xc0) == 0x80));
3325}
3326
3327void vboxNetFltWinNotifyHostAddress(PTA_ADDRESS pAddress, bool fAdded)
3328{
3329 void *pvAddr = NULL;
3330 INTNETADDRTYPE enmAddrType = kIntNetAddrType_Invalid;
3331
3332 LogFlow(("==>vboxNetFltWinNotifyHostAddress: AddrType=%d %s\n",
3333 pAddress->AddressType, fAdded ? "added" : "deleted"));
3334 if (pAddress->AddressType == TDI_ADDRESS_TYPE_IP)
3335 {
3336 PTDI_ADDRESS_IP pTdiAddrIp = (PTDI_ADDRESS_IP)pAddress->Address;
3337 /*
3338 * Note that we do not get loopback addresses here. If we did we should
3339 * have checked and ignored them too.
3340 */
3341 if (!vboxNetFltWinIsAddrLinkLocal4((PCRTNETADDRIPV4)(&pTdiAddrIp->in_addr)))
3342 {
3343 pvAddr = &pTdiAddrIp->in_addr;
3344 enmAddrType = kIntNetAddrType_IPv4;
3345 }
3346 else
3347 Log2(("vboxNetFltWinNotifyHostAddress: ignoring link-local address %RTnaipv4\n",
3348 pTdiAddrIp->in_addr));
3349 }
3350 else if (pAddress->AddressType == TDI_ADDRESS_TYPE_IP6)
3351 {
3352 PTDI_ADDRESS_IP6 pTdiAddrIp6 = (PTDI_ADDRESS_IP6)pAddress->Address;
3353 if (!vboxNetFltWinIsAddrLinkLocal6((PCRTNETADDRIPV6)(pTdiAddrIp6->sin6_addr)))
3354 {
3355 pvAddr = pTdiAddrIp6->sin6_addr;
3356 enmAddrType = kIntNetAddrType_IPv6;
3357 }
3358 else
3359 Log2(("vboxNetFltWinNotifyHostAddress: ignoring link-local address %RTnaipv6\n",
3360 pTdiAddrIp6->sin6_addr));
3361 }
3362 else
3363 {
3364 Log2(("vboxNetFltWinNotifyHostAddress: ignoring irrelevant address type %d\n",
3365 pAddress->AddressType));
3366 LogFlow(("<==vboxNetFltWinNotifyHostAddress\n"));
3367 return;
3368 }
3369 if (pvAddr)
3370 {
3371 NdisAcquireSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
3372 /* At this point the list must contain at least one element. */
3373 PVBOXNETFLTWIN pFilter = NULL;
3374 PVBOXNETFLTINS pInstance = NULL;
3375 RTListForEach(&g_VBoxNetFltGlobalsWin.listFilters, pFilter, VBOXNETFLTWIN, node)
3376 {
3377 pInstance = RT_FROM_MEMBER(pFilter, VBOXNETFLTINS, u.s.WinIf);
3378 if (vboxNetFltWinReferenceWinIf(pInstance))
3379 {
3380 if (pInstance->pSwitchPort && pInstance->pSwitchPort->pfnNotifyHostAddress)
3381 break;
3382 vboxNetFltWinDereferenceWinIf(pInstance);
3383 }
3384 else
3385 Log2(("vboxNetFltWinNotifyHostAddress: failed to retain filter instance %p\n", pInstance));
3386 pInstance = NULL;
3387 }
3388 NdisReleaseSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
3389 if (pInstance)
3390 {
3391 if (enmAddrType == kIntNetAddrType_IPv4)
3392 Log2(("vboxNetFltWin%sAddressHandler: %RTnaipv4\n",
3393 fAdded ? "Add" : "Del", *(PCRTNETADDRIPV4)pvAddr));
3394 else
3395 Log2(("vboxNetFltWin%sAddressHandler: %RTnaipv6\n",
3396 fAdded ? "Add" : "Del", pvAddr));
3397 pInstance->pSwitchPort->pfnNotifyHostAddress(pInstance->pSwitchPort, fAdded,
3398 enmAddrType, pvAddr);
3399 vboxNetFltWinDereferenceWinIf(pInstance);
3400 }
3401 else
3402 Log2(("vboxNetFltWinNotifyHostAddress: no filters require notification\n"));
3403 }
3404 LogFlow(("<==vboxNetFltWinNotifyHostAddress\n"));
3405}
3406
3407void vboxNetFltWinAddAddressHandler(PTA_ADDRESS Address,
3408 PUNICODE_STRING DeviceName,
3409 PTDI_PNP_CONTEXT Context)
3410{
3411 vboxNetFltWinNotifyHostAddress(Address, true);
3412}
3413
3414void vboxNetFltWinDelAddressHandler(PTA_ADDRESS Address,
3415 PUNICODE_STRING DeviceName,
3416 PTDI_PNP_CONTEXT Context)
3417{
3418 vboxNetFltWinNotifyHostAddress(Address, false);
3419}
3420
3421void vboxNetFltWinRegisterIpAddrNotifier(PVBOXNETFLTINS pThis)
3422{
3423 LogFlow(("==>vboxNetFltWinRegisterIpAddrNotifier: instance=%p pThis->pSwitchPort=%p pThis->pSwitchPort->pfnNotifyHostAddress=%p\n",
3424 pThis, pThis->pSwitchPort, pThis->pSwitchPort ? pThis->pSwitchPort->pfnNotifyHostAddress : NULL));
3425 if (pThis->pSwitchPort && pThis->pSwitchPort->pfnNotifyHostAddress)
3426 {
3427 NdisAcquireSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
3428 bool fRegisterHandlers = RTListIsEmpty(&g_VBoxNetFltGlobalsWin.listFilters);
3429 RTListPrepend(&g_VBoxNetFltGlobalsWin.listFilters, &pThis->u.s.WinIf.node);
3430 NdisReleaseSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
3431
3432 if (fRegisterHandlers)
3433 {
3434 TDI_CLIENT_INTERFACE_INFO Info;
3435 UNICODE_STRING ClientName = RTL_CONSTANT_STRING(L"VBoxNetFlt");
3436 memset(&Info, 0, sizeof(Info));
3437 Info.MajorTdiVersion = 2;
3438 Info.MinorTdiVersion = 0;
3439 Info.ClientName = &ClientName;
3440 Info.AddAddressHandlerV2 = vboxNetFltWinAddAddressHandler;
3441 Info.DelAddressHandlerV2 = vboxNetFltWinDelAddressHandler;
3442 Assert(!g_VBoxNetFltGlobalsWin.hNotifier);
3443 NTSTATUS Status = TdiRegisterPnPHandlers(&Info, sizeof(Info), &g_VBoxNetFltGlobalsWin.hNotifier);
3444 Log2(("vboxNetFltWinRegisterIpAddrNotifier: TdiRegisterPnPHandlers returned %d\n", Status));
3445 }
3446 else
3447 Log2(("vboxNetFltWinRegisterIpAddrNotifier: already registed\n"));
3448 }
3449 else
3450 Log2(("vboxNetFltWinRegisterIpAddrNotifier: this instance does not require notifications, ignoring...\n"));
3451 LogFlow(("<==vboxNetFltWinRegisterIpAddrNotifier: notifier=%p\n", g_VBoxNetFltGlobalsWin.hNotifier));
3452}
3453
3454void vboxNetFltWinUnregisterIpAddrNotifier(PVBOXNETFLTINS pThis)
3455{
3456 LogFlow(("==>vboxNetFltWinUnregisterIpAddrNotifier: notifier=%p\n", g_VBoxNetFltGlobalsWin.hNotifier));
3457 if (pThis->pSwitchPort && pThis->pSwitchPort->pfnNotifyHostAddress)
3458 {
3459 NdisAcquireSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
3460 /* At this point the list must contain at least one element. */
3461 Assert(!RTListIsEmpty(&g_VBoxNetFltGlobalsWin.listFilters));
3462 RTListNodeRemove(&pThis->u.s.WinIf.node);
3463 HANDLE hNotifier = NULL;
3464 if (RTListIsEmpty(&g_VBoxNetFltGlobalsWin.listFilters))
3465 {
3466 /*
3467 * The list has become empty, so we need to deregister handlers. We
3468 * grab hNotifier and reset it while still holding the lock. This
3469 * guaranties that we won't interfere with setting it in
3470 * vboxNetFltWinRegisterIpAddrNotifier(). It is inconceivable that
3471 * vboxNetFltWinUnregisterIpAddrNotifier() will be called for the
3472 * same filter instance while it is still being processed by
3473 * vboxNetFltWinRegisterIpAddrNotifier(). This would require trunk
3474 * destruction in the middle of its creation. It is possible that
3475 * vboxNetFltWinUnregisterIpAddrNotifier() is called for another
3476 * filter instance, but in such case we won't even get here as the
3477 * list won't be empty.
3478 */
3479 hNotifier = g_VBoxNetFltGlobalsWin.hNotifier;
3480 g_VBoxNetFltGlobalsWin.hNotifier = NULL;
3481 }
3482 NdisReleaseSpinLock(&g_VBoxNetFltGlobalsWin.lockFilters);
3483 if (hNotifier)
3484 {
3485 NTSTATUS Status = TdiDeregisterPnPHandlers(hNotifier);
3486 Log2(("vboxNetFltWinUnregisterIpAddrNotifier: TdiDeregisterPnPHandlers(%p) returned %d\n",
3487 hNotifier, Status));
3488 }
3489 else
3490 Log2(("vboxNetFltWinUnregisterIpAddrNotifier: filters remain, do not deregister handlers yet\n"));
3491 }
3492 else
3493 Log2(("vboxNetFltWinUnregisterIpAddrNotifier: this instance did not require notifications, ignoring...\n"));
3494 LogFlow(("<==vboxNetFltWinUnregisterIpAddrNotifier\n"));
3495}
3496#else /* VBOXNETADP */
3497#define vboxNetFltWinRegisterIpAddrNotifier(x)
3498#define vboxNetFltWinUnregisterIpAddrNotifier(x)
3499#endif /* VBOXNETADP */
3500
3501int vboxNetFltOsDisconnectIt(PVBOXNETFLTINS pThis)
3502{
3503 NDIS_STATUS Status = vboxNetFltWinDisconnectIt(pThis);
3504 Log2(("vboxNetFltOsDisconnectIt: pThis=%p pThis->pSwitchPort=%p pThis->pSwitchPort->pfnNotifyHostAddress=%p\n",
3505 pThis, pThis->pSwitchPort, pThis->pSwitchPort ? pThis->pSwitchPort->pfnNotifyHostAddress : NULL));
3506 vboxNetFltWinUnregisterIpAddrNotifier(pThis);
3507 return Status == NDIS_STATUS_SUCCESS ? VINF_SUCCESS : VERR_GENERAL_FAILURE;
3508}
3509
3510static void vboxNetFltWinConnectItWorker(PVOID pvContext)
3511{
3512 PWORKER_INFO pInfo = (PWORKER_INFO)pvContext;
3513#if !defined(VBOXNETADP) || !defined(VBOXNETFLT_NO_PACKET_QUEUE)
3514 NDIS_STATUS Status;
3515#endif
3516 PVBOXNETFLTINS pInstance = pInfo->pNetFltIf;
3517
3518 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
3519
3520 /* this is not a rediscovery, initialize Mac cache */
3521 if (vboxNetFltWinReferenceWinIf(pInstance))
3522 {
3523#ifndef VBOXNETADP
3524 Status = vboxNetFltWinGetMacAddress(pInstance, &pInstance->u.s.MacAddr);
3525 if (Status == NDIS_STATUS_SUCCESS)
3526#endif
3527 {
3528#ifdef VBOXNETFLT_NO_PACKET_QUEUE
3529 pInfo->Status = VINF_SUCCESS;
3530#else
3531 Status = vboxNetFltWinQuInitPacketQueue(pInstance);
3532 if (Status == NDIS_STATUS_SUCCESS)
3533 {
3534 pInfo->Status = VINF_SUCCESS;
3535 }
3536 else
3537 {
3538 pInfo->Status = VERR_GENERAL_FAILURE;
3539 }
3540#endif
3541 }
3542#ifndef VBOXNETADP
3543 else
3544 {
3545 pInfo->Status = VERR_INTNET_FLT_IF_FAILED;
3546 }
3547#endif
3548
3549 vboxNetFltWinDereferenceWinIf(pInstance);
3550 }
3551 else
3552 {
3553 pInfo->Status = VERR_INTNET_FLT_IF_NOT_FOUND;
3554 }
3555}
3556
3557static int vboxNetFltWinConnectIt(PVBOXNETFLTINS pThis)
3558{
3559 WORKER_INFO Info;
3560 Info.pNetFltIf = pThis;
3561
3562 vboxNetFltWinJobSynchExecAtPassive(vboxNetFltWinConnectItWorker, &Info);
3563
3564 if (RT_SUCCESS(Info.Status))
3565 vboxNetFltWinReportStuff(pThis);
3566
3567 return Info.Status;
3568}
3569
3570int vboxNetFltOsConnectIt(PVBOXNETFLTINS pThis)
3571{
3572 Log2(("vboxNetFltOsConnectIt: pThis=%p pThis->pSwitchPort=%p pThis->pSwitchPort->pfnNotifyHostAddress=%p\n",
3573 pThis, pThis->pSwitchPort, pThis->pSwitchPort ? pThis->pSwitchPort->pfnNotifyHostAddress : NULL));
3574 vboxNetFltWinRegisterIpAddrNotifier(pThis);
3575 return vboxNetFltWinConnectIt(pThis);
3576}
3577
3578void vboxNetFltOsDeleteInstance(PVBOXNETFLTINS pThis)
3579{
3580 vboxNetFltWinDeleteInstance(pThis);
3581}
3582
3583int vboxNetFltOsInitInstance(PVBOXNETFLTINS pThis, void *pvContext)
3584{
3585 int rc = RTSemMutexCreate(&pThis->u.s.hWinIfMutex);
3586 if (RT_SUCCESS(rc))
3587 {
3588 rc = vboxNetFltWinAttachToInterface(pThis, pvContext, false /*fRediscovery*/ );
3589 if (RT_SUCCESS(rc))
3590 {
3591 return rc;
3592 }
3593 RTSemMutexDestroy(pThis->u.s.hWinIfMutex);
3594 }
3595 return rc;
3596}
3597
3598int vboxNetFltOsPreInitInstance(PVBOXNETFLTINS pThis)
3599{
3600 pThis->u.s.cModeNetFltRefs = 0;
3601 pThis->u.s.cModePassThruRefs = 0;
3602 vboxNetFltWinSetWinIfState(pThis, kVBoxWinIfState_Disconnected);
3603 vboxNetFltWinSetOpState(&pThis->u.s.WinIf.MpState, kVBoxNetDevOpState_Deinitialized);
3604#ifndef VBOXNETADP
3605 vboxNetFltWinSetOpState(&pThis->u.s.WinIf.PtState, kVBoxNetDevOpState_Deinitialized);
3606#endif
3607 return VINF_SUCCESS;
3608}
3609
3610void vboxNetFltPortOsNotifyMacAddress(PVBOXNETFLTINS pThis, void *pvIfData, PCRTMAC pMac)
3611{
3612}
3613
3614int vboxNetFltPortOsConnectInterface(PVBOXNETFLTINS pThis, void *pvIf, void **ppvIfData)
3615{
3616 /* Nothing to do */
3617 return VINF_SUCCESS;
3618}
3619
3620int vboxNetFltPortOsDisconnectInterface(PVBOXNETFLTINS pThis, void *pvIfData)
3621{
3622 /* Nothing to do */
3623 return VINF_SUCCESS;
3624}
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