mirror of
https://github.com/zerotier/ZeroTierOne.git
synced 2025-01-18 18:56:24 +00:00
1210 lines
36 KiB
C
1210 lines
36 KiB
C
/*
|
|
* TAP-Windows -- A kernel driver to provide virtual tap
|
|
* device functionality on Windows.
|
|
*
|
|
* This code was inspired by the CIPE-Win32 driver by Damion K. Wilson.
|
|
*
|
|
* This source code is Copyright (C) 2002-2014 OpenVPN Technologies, Inc.,
|
|
* and is released under the GPL version 2 (see below).
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program (see the file COPYING included with this
|
|
* distribution); if not, write to the Free Software Foundation, Inc.,
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
//
|
|
// Include files.
|
|
//
|
|
|
|
#include "tap.h"
|
|
#include <wdmsec.h> // for SDDLs
|
|
|
|
//======================================================================
|
|
// TAP Win32 Device I/O Callbacks
|
|
//======================================================================
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text( PAGE, TapDeviceCreate)
|
|
#pragma alloc_text( PAGE, TapDeviceControl)
|
|
#pragma alloc_text( PAGE, TapDeviceCleanup)
|
|
#pragma alloc_text( PAGE, TapDeviceClose)
|
|
#endif // ALLOC_PRAGMA
|
|
|
|
//===================================================================
|
|
// Go back to default TAP mode from Point-To-Point mode.
|
|
// Also reset (i.e. disable) DHCP Masq mode.
|
|
//===================================================================
|
|
VOID tapResetAdapterState(
|
|
__in PTAP_ADAPTER_CONTEXT Adapter
|
|
)
|
|
{
|
|
/*
|
|
// Point-To-Point
|
|
Adapter->m_tun = FALSE;
|
|
Adapter->m_localIP = 0;
|
|
Adapter->m_remoteNetwork = 0;
|
|
Adapter->m_remoteNetmask = 0;
|
|
NdisZeroMemory (&Adapter->m_TapToUser, sizeof (Adapter->m_TapToUser));
|
|
NdisZeroMemory (&Adapter->m_UserToTap, sizeof (Adapter->m_UserToTap));
|
|
NdisZeroMemory (&Adapter->m_UserToTap_IPv6, sizeof (Adapter->m_UserToTap_IPv6));
|
|
*/
|
|
|
|
// DHCP Masq
|
|
/*
|
|
Adapter->m_dhcp_enabled = FALSE;
|
|
Adapter->m_dhcp_server_arp = FALSE;
|
|
Adapter->m_dhcp_user_supplied_options_buffer_len = 0;
|
|
Adapter->m_dhcp_addr = 0;
|
|
Adapter->m_dhcp_netmask = 0;
|
|
Adapter->m_dhcp_server_ip = 0;
|
|
Adapter->m_dhcp_lease_time = 0;
|
|
Adapter->m_dhcp_received_discover = FALSE;
|
|
Adapter->m_dhcp_bad_requests = 0;
|
|
NdisZeroMemory (Adapter->m_dhcp_server_mac, MACADDR_SIZE);
|
|
*/
|
|
}
|
|
|
|
// IRP_MJ_CREATE
|
|
NTSTATUS
|
|
TapDeviceCreate(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called by the I/O system when the device is opened.
|
|
|
|
No action is performed other than completing the request successfully.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the object that represents the device
|
|
that I/O is to be done on.
|
|
|
|
Irp - a pointer to the I/O Request Packet for this request.
|
|
|
|
Return Value:
|
|
|
|
NT status code
|
|
|
|
--*/
|
|
{
|
|
NDIS_STATUS status;
|
|
PIO_STACK_LOCATION irpSp;// Pointer to current stack location
|
|
PTAP_ADAPTER_CONTEXT adapter = NULL;
|
|
PFILE_OBJECT originalFileObject;
|
|
|
|
PAGED_CODE();
|
|
|
|
DEBUGP (("[TAP] --> TapDeviceCreate\n"));
|
|
|
|
irpSp = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
//
|
|
// Invalidate file context
|
|
//
|
|
irpSp->FileObject->FsContext = NULL;
|
|
irpSp->FileObject->FsContext2 = NULL;
|
|
|
|
//
|
|
// Find adapter context for this device.
|
|
// -------------------------------------
|
|
// Returns with added reference on adapter context.
|
|
//
|
|
adapter = tapAdapterContextFromDeviceObject(DeviceObject);
|
|
|
|
// Insure that adapter exists.
|
|
ASSERT(adapter);
|
|
|
|
if(adapter == NULL )
|
|
{
|
|
DEBUGP (("[TAP] release [%d.%d] open request; adapter not found\n",
|
|
TAP_DRIVER_MAJOR_VERSION,
|
|
TAP_DRIVER_MINOR_VERSION
|
|
));
|
|
|
|
Irp->IoStatus.Status = STATUS_DEVICE_DOES_NOT_EXIST;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
return STATUS_DEVICE_DOES_NOT_EXIST;
|
|
}
|
|
|
|
DEBUGP(("[%s] [TAP] release [%d.%d] open request (TapFileIsOpen=%d)\n",
|
|
MINIPORT_INSTANCE_ID(adapter),
|
|
TAP_DRIVER_MAJOR_VERSION,
|
|
TAP_DRIVER_MINOR_VERSION,
|
|
adapter->TapFileIsOpen
|
|
));
|
|
|
|
// Enforce exclusive access
|
|
originalFileObject = InterlockedCompareExchangePointer(
|
|
&adapter->TapFileObject,
|
|
irpSp->FileObject,
|
|
NULL
|
|
);
|
|
|
|
if(originalFileObject == NULL)
|
|
{
|
|
irpSp->FileObject->FsContext = adapter; // Quick reference
|
|
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
// Release the lock.
|
|
//tapAdapterReleaseLock(adapter,FALSE);
|
|
|
|
if(status == STATUS_SUCCESS)
|
|
{
|
|
// Reset adapter state on successful open.
|
|
tapResetAdapterState(adapter);
|
|
|
|
adapter->TapFileIsOpen = 1; // Legacy...
|
|
|
|
// NOTE!!! Reference added by tapAdapterContextFromDeviceObject
|
|
// will be removed when file is closed.
|
|
}
|
|
else
|
|
{
|
|
DEBUGP (("[%s] TAP is presently unavailable (TapFileIsOpen=%d)\n",
|
|
MINIPORT_INSTANCE_ID(adapter), adapter->TapFileIsOpen
|
|
));
|
|
|
|
NOTE_ERROR();
|
|
|
|
// Remove reference added by tapAdapterContextFromDeviceObject.
|
|
tapAdapterContextDereference(adapter);
|
|
}
|
|
|
|
// Complete the IRP.
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
DEBUGP (("[TAP] <-- TapDeviceCreate; status = %8.8X\n",status));
|
|
|
|
return status;
|
|
}
|
|
|
|
//===================================================
|
|
// Tell Windows whether the TAP device should be
|
|
// considered "connected" or "disconnected".
|
|
//
|
|
// Allows application control of media connect state.
|
|
//===================================================
|
|
VOID
|
|
tapSetMediaConnectStatus(
|
|
__in PTAP_ADAPTER_CONTEXT Adapter,
|
|
__in BOOLEAN LogicalMediaState
|
|
)
|
|
{
|
|
NDIS_STATUS_INDICATION statusIndication;
|
|
NDIS_LINK_STATE linkState;
|
|
|
|
NdisZeroMemory(&statusIndication, sizeof(NDIS_STATUS_INDICATION));
|
|
NdisZeroMemory(&linkState, sizeof(NDIS_LINK_STATE));
|
|
|
|
//
|
|
// Fill in object headers
|
|
//
|
|
statusIndication.Header.Type = NDIS_OBJECT_TYPE_STATUS_INDICATION;
|
|
statusIndication.Header.Revision = NDIS_STATUS_INDICATION_REVISION_1;
|
|
statusIndication.Header.Size = sizeof(NDIS_STATUS_INDICATION);
|
|
|
|
linkState.Header.Revision = NDIS_LINK_STATE_REVISION_1;
|
|
linkState.Header.Type = NDIS_OBJECT_TYPE_DEFAULT;
|
|
linkState.Header.Size = sizeof(NDIS_LINK_STATE);
|
|
|
|
//
|
|
// Link state buffer
|
|
//
|
|
if(Adapter->LogicalMediaState == TRUE)
|
|
{
|
|
linkState.MediaConnectState = MediaConnectStateConnected;
|
|
}
|
|
|
|
linkState.MediaDuplexState = MediaDuplexStateFull;
|
|
linkState.RcvLinkSpeed = TAP_RECV_SPEED;
|
|
linkState.XmitLinkSpeed = TAP_XMIT_SPEED;
|
|
|
|
//
|
|
// Fill in the status buffer
|
|
//
|
|
statusIndication.StatusCode = NDIS_STATUS_LINK_STATE;
|
|
statusIndication.SourceHandle = Adapter->MiniportAdapterHandle;
|
|
statusIndication.DestinationHandle = NULL;
|
|
statusIndication.RequestId = 0;
|
|
|
|
statusIndication.StatusBuffer = &linkState;
|
|
statusIndication.StatusBufferSize = sizeof(NDIS_LINK_STATE);
|
|
|
|
// Fill in new media connect state.
|
|
if ( (Adapter->LogicalMediaState != LogicalMediaState) && !Adapter->MediaStateAlwaysConnected)
|
|
{
|
|
Adapter->LogicalMediaState = LogicalMediaState;
|
|
|
|
if (LogicalMediaState == TRUE)
|
|
{
|
|
linkState.MediaConnectState = MediaConnectStateConnected;
|
|
|
|
DEBUGP (("[TAP] Set MediaConnectState: Connected.\n"));
|
|
}
|
|
else
|
|
{
|
|
linkState.MediaConnectState = MediaConnectStateDisconnected;
|
|
|
|
DEBUGP (("[TAP] Set MediaConnectState: Disconnected.\n"));
|
|
}
|
|
}
|
|
|
|
// Make the status indication.
|
|
if(Adapter->Locked.AdapterState != MiniportHaltedState)
|
|
{
|
|
NdisMIndicateStatusEx(Adapter->MiniportAdapterHandle, &statusIndication);
|
|
}
|
|
}
|
|
|
|
/*
|
|
//======================================================
|
|
// If DHCP mode is used together with tun
|
|
// mode, consider the fact that the P2P remote subnet
|
|
// might enclose the DHCP masq server address.
|
|
//======================================================
|
|
VOID
|
|
CheckIfDhcpAndTunMode (
|
|
__in PTAP_ADAPTER_CONTEXT Adapter
|
|
)
|
|
{
|
|
if (Adapter->m_tun && Adapter->m_dhcp_enabled)
|
|
{
|
|
if ((Adapter->m_dhcp_server_ip & Adapter->m_remoteNetmask) == Adapter->m_remoteNetwork)
|
|
{
|
|
ETH_COPY_NETWORK_ADDRESS (Adapter->m_dhcp_server_mac, Adapter->m_TapToUser.dest);
|
|
Adapter->m_dhcp_server_arp = FALSE;
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
|
|
// IRP_MJ_DEVICE_CONTROL callback.
|
|
NTSTATUS
|
|
TapDeviceControl(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine is called by the I/O system to perform a device I/O
|
|
control function.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the object that represents the device
|
|
that I/O is to be done on.
|
|
|
|
Irp - a pointer to the I/O Request Packet for this request.
|
|
|
|
Return Value:
|
|
|
|
NT status code
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS ntStatus = STATUS_SUCCESS; // Assume success
|
|
PIO_STACK_LOCATION irpSp; // Pointer to current stack location
|
|
PTAP_ADAPTER_CONTEXT adapter = NULL;
|
|
ULONG inBufLength; // Input buffer length
|
|
ULONG outBufLength; // Output buffer length
|
|
PCHAR inBuf, outBuf; // pointer to Input and output buffer
|
|
PMDL mdl = NULL;
|
|
PCHAR buffer = NULL;
|
|
|
|
PAGED_CODE();
|
|
|
|
irpSp = IoGetCurrentIrpStackLocation( Irp );
|
|
|
|
//
|
|
// Fetch adapter context for this device.
|
|
// --------------------------------------
|
|
// Adapter pointer was stashed in FsContext when handle was opened.
|
|
//
|
|
adapter = (PTAP_ADAPTER_CONTEXT )(irpSp->FileObject)->FsContext;
|
|
|
|
ASSERT(adapter);
|
|
|
|
inBufLength = irpSp->Parameters.DeviceIoControl.InputBufferLength;
|
|
outBufLength = irpSp->Parameters.DeviceIoControl.OutputBufferLength;
|
|
|
|
if (!inBufLength || !outBufLength)
|
|
{
|
|
ntStatus = STATUS_INVALID_PARAMETER;
|
|
goto End;
|
|
}
|
|
|
|
//
|
|
// Determine which I/O control code was specified.
|
|
//
|
|
switch ( irpSp->Parameters.DeviceIoControl.IoControlCode )
|
|
{
|
|
case TAP_WIN_IOCTL_GET_MAC:
|
|
{
|
|
if (outBufLength >= MACADDR_SIZE )
|
|
{
|
|
ETH_COPY_NETWORK_ADDRESS(
|
|
Irp->AssociatedIrp.SystemBuffer,
|
|
adapter->CurrentAddress
|
|
);
|
|
|
|
Irp->IoStatus.Information = MACADDR_SIZE;
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_BUFFER_TOO_SMALL;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TAP_WIN_IOCTL_GET_VERSION:
|
|
{
|
|
const ULONG size = sizeof (ULONG) * 3;
|
|
|
|
if (outBufLength >= size)
|
|
{
|
|
((PULONG) (Irp->AssociatedIrp.SystemBuffer))[0]
|
|
= TAP_DRIVER_MAJOR_VERSION;
|
|
|
|
((PULONG) (Irp->AssociatedIrp.SystemBuffer))[1]
|
|
= TAP_DRIVER_MINOR_VERSION;
|
|
|
|
((PULONG) (Irp->AssociatedIrp.SystemBuffer))[2]
|
|
#if DBG
|
|
= 1;
|
|
#else
|
|
= 0;
|
|
#endif
|
|
Irp->IoStatus.Information = size;
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_BUFFER_TOO_SMALL;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TAP_WIN_IOCTL_GET_MTU:
|
|
{
|
|
const ULONG size = sizeof (ULONG) * 1;
|
|
|
|
if (outBufLength >= size)
|
|
{
|
|
((PULONG) (Irp->AssociatedIrp.SystemBuffer))[0]
|
|
= adapter->MtuSize;
|
|
|
|
Irp->IoStatus.Information = size;
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_BUFFER_TOO_SMALL;
|
|
}
|
|
}
|
|
break;
|
|
|
|
// Allow ZeroTier One to get multicast memberships at the L2 level in a
|
|
// protocol-neutral manner.
|
|
case TAP_WIN_IOCTL_GET_MULTICAST_MEMBERSHIPS:
|
|
{
|
|
if (outBufLength < TAP_WIN_IOCTL_GET_MULTICAST_MEMBERSHIPS_OUTPUT_BUF_SIZE) {
|
|
/* output buffer too small */
|
|
NOTE_ERROR ();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_BUFFER_TOO_SMALL;
|
|
} else {
|
|
char *out = (char *)Irp->AssociatedIrp.SystemBuffer;
|
|
char *end = out + TAP_WIN_IOCTL_GET_MULTICAST_MEMBERSHIPS_OUTPUT_BUF_SIZE;
|
|
unsigned long i,j;
|
|
for(i=0;i<adapter->ulMCListSize;++i) {
|
|
if (i >= TAP_MAX_MCAST_LIST)
|
|
break;
|
|
for(j=0;j<6;++j)
|
|
*(out++) = adapter->MCList[i][j];
|
|
if (out >= end)
|
|
break;
|
|
}
|
|
while (out < end)
|
|
*(out++) = (char)0;
|
|
Irp->IoStatus.Information = TAP_WIN_IOCTL_GET_MULTICAST_MEMBERSHIPS_OUTPUT_BUF_SIZE;
|
|
Irp->IoStatus.Status = ntStatus = STATUS_SUCCESS;
|
|
}
|
|
break;
|
|
}
|
|
|
|
|
|
#if 0
|
|
case TAP_WIN_IOCTL_CONFIG_TUN:
|
|
{
|
|
if(inBufLength >= sizeof(IPADDR)*3)
|
|
{
|
|
MACADDR dest;
|
|
|
|
adapter->m_tun = FALSE;
|
|
|
|
GenerateRelatedMAC (dest, adapter->CurrentAddress, 1);
|
|
|
|
adapter->m_localIP = ((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[0];
|
|
adapter->m_remoteNetwork = ((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[1];
|
|
adapter->m_remoteNetmask = ((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[2];
|
|
|
|
// Sanity check on network/netmask
|
|
if ((adapter->m_remoteNetwork & adapter->m_remoteNetmask) != adapter->m_remoteNetwork)
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_INVALID_PARAMETER;
|
|
break;
|
|
}
|
|
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_TapToUser.src, adapter->CurrentAddress);
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_TapToUser.dest, dest);
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_UserToTap.src, dest);
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_UserToTap.dest, adapter->CurrentAddress);
|
|
|
|
adapter->m_TapToUser.proto = adapter->m_UserToTap.proto = htons (NDIS_ETH_TYPE_IPV4);
|
|
adapter->m_UserToTap_IPv6 = adapter->m_UserToTap;
|
|
adapter->m_UserToTap_IPv6.proto = htons(NDIS_ETH_TYPE_IPV6);
|
|
|
|
adapter->m_tun = TRUE;
|
|
|
|
CheckIfDhcpAndTunMode (adapter);
|
|
|
|
Irp->IoStatus.Information = 1; // Simple boolean value
|
|
|
|
DEBUGP (("[TAP] Set TUN mode.\n"));
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_INVALID_PARAMETER;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TAP_WIN_IOCTL_CONFIG_POINT_TO_POINT:
|
|
{
|
|
if(inBufLength >= sizeof(IPADDR)*2)
|
|
{
|
|
MACADDR dest;
|
|
|
|
adapter->m_tun = FALSE;
|
|
|
|
GenerateRelatedMAC (dest, adapter->CurrentAddress, 1);
|
|
|
|
adapter->m_localIP = ((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[0];
|
|
adapter->m_remoteNetwork = ((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[1];
|
|
adapter->m_remoteNetmask = ~0;
|
|
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_TapToUser.src, adapter->CurrentAddress);
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_TapToUser.dest, dest);
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_UserToTap.src, dest);
|
|
ETH_COPY_NETWORK_ADDRESS (adapter->m_UserToTap.dest, adapter->CurrentAddress);
|
|
|
|
adapter->m_TapToUser.proto = adapter->m_UserToTap.proto = htons (NDIS_ETH_TYPE_IPV4);
|
|
adapter->m_UserToTap_IPv6 = adapter->m_UserToTap;
|
|
adapter->m_UserToTap_IPv6.proto = htons(NDIS_ETH_TYPE_IPV6);
|
|
|
|
adapter->m_tun = TRUE;
|
|
|
|
CheckIfDhcpAndTunMode (adapter);
|
|
|
|
Irp->IoStatus.Information = 1; // Simple boolean value
|
|
|
|
DEBUGP (("[TAP] Set P2P mode.\n"));
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_INVALID_PARAMETER;
|
|
}
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
#if 0
|
|
case TAP_WIN_IOCTL_CONFIG_DHCP_MASQ:
|
|
{
|
|
if(inBufLength >= sizeof(IPADDR)*4)
|
|
{
|
|
adapter->m_dhcp_enabled = FALSE;
|
|
adapter->m_dhcp_server_arp = FALSE;
|
|
adapter->m_dhcp_user_supplied_options_buffer_len = 0;
|
|
|
|
// Adapter IP addr / netmask
|
|
adapter->m_dhcp_addr =
|
|
((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[0];
|
|
adapter->m_dhcp_netmask =
|
|
((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[1];
|
|
|
|
// IP addr of DHCP masq server
|
|
adapter->m_dhcp_server_ip =
|
|
((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[2];
|
|
|
|
// Lease time in seconds
|
|
adapter->m_dhcp_lease_time =
|
|
((IPADDR*) (Irp->AssociatedIrp.SystemBuffer))[3];
|
|
|
|
GenerateRelatedMAC(
|
|
adapter->m_dhcp_server_mac,
|
|
adapter->CurrentAddress,
|
|
2
|
|
);
|
|
|
|
adapter->m_dhcp_enabled = TRUE;
|
|
adapter->m_dhcp_server_arp = TRUE;
|
|
|
|
CheckIfDhcpAndTunMode (adapter);
|
|
|
|
Irp->IoStatus.Information = 1; // Simple boolean value
|
|
|
|
DEBUGP (("[TAP] Configured DHCP MASQ.\n"));
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_INVALID_PARAMETER;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case TAP_WIN_IOCTL_CONFIG_DHCP_SET_OPT:
|
|
{
|
|
if (inBufLength <= DHCP_USER_SUPPLIED_OPTIONS_BUFFER_SIZE
|
|
&& adapter->m_dhcp_enabled)
|
|
{
|
|
adapter->m_dhcp_user_supplied_options_buffer_len = 0;
|
|
|
|
NdisMoveMemory(
|
|
adapter->m_dhcp_user_supplied_options_buffer,
|
|
Irp->AssociatedIrp.SystemBuffer,
|
|
inBufLength
|
|
);
|
|
|
|
adapter->m_dhcp_user_supplied_options_buffer_len =
|
|
inBufLength;
|
|
|
|
Irp->IoStatus.Information = 1; // Simple boolean value
|
|
|
|
DEBUGP (("[TAP] Set DHCP OPT.\n"));
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_INVALID_PARAMETER;
|
|
}
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
#if 0
|
|
case TAP_WIN_IOCTL_GET_INFO:
|
|
{
|
|
char state[16];
|
|
|
|
// Fetch adapter (miniport) state.
|
|
if (tapAdapterSendAndReceiveReady(adapter) == NDIS_STATUS_SUCCESS)
|
|
state[0] = 'A';
|
|
else
|
|
state[0] = 'a';
|
|
|
|
if (tapAdapterReadAndWriteReady(adapter))
|
|
state[1] = 'T';
|
|
else
|
|
state[1] = 't';
|
|
|
|
state[2] = '0' + adapter->CurrentPowerState;
|
|
|
|
if (adapter->MediaStateAlwaysConnected)
|
|
state[3] = 'C';
|
|
else
|
|
state[3] = 'c';
|
|
|
|
state[4] = '\0';
|
|
|
|
// BUGBUG!!! What follows, and is not yet implemented, is a real mess.
|
|
// BUGBUG!!! Tied closely to the NDIS 5 implementation. Need to map
|
|
// as much as possible to the NDIS 6 implementation.
|
|
Irp->IoStatus.Status = ntStatus = RtlStringCchPrintfExA (
|
|
((LPTSTR) (Irp->AssociatedIrp.SystemBuffer)),
|
|
outBufLength,
|
|
NULL,
|
|
NULL,
|
|
STRSAFE_FILL_BEHIND_NULL | STRSAFE_IGNORE_NULLS,
|
|
#if PACKET_TRUNCATION_CHECK
|
|
"State=%s Err=[%s/%d] #O=%d Tx=[%d,%d,%d] Rx=[%d,%d,%d] IrpQ=[%d,%d,%d] PktQ=[%d,%d,%d] InjQ=[%d,%d,%d]",
|
|
#else
|
|
"State=%s Err=[%s/%d] #O=%d Tx=[%d,%d] Rx=[%d,%d] IrpQ=[%d,%d,%d] PktQ=[%d,%d,%d] InjQ=[%d,%d,%d]",
|
|
#endif
|
|
state,
|
|
g_LastErrorFilename,
|
|
g_LastErrorLineNumber,
|
|
(int)adapter->TapFileOpenCount,
|
|
(int)(adapter->FramesTxDirected + adapter->FramesTxMulticast + adapter->FramesTxBroadcast),
|
|
(int)adapter->TransmitFailuresOther,
|
|
#if PACKET_TRUNCATION_CHECK
|
|
(int)adapter->m_TxTrunc,
|
|
#endif
|
|
(int)adapter->m_Rx,
|
|
(int)adapter->m_RxErr,
|
|
#if PACKET_TRUNCATION_CHECK
|
|
(int)adapter->m_RxTrunc,
|
|
#endif
|
|
(int)adapter->PendingReadIrpQueue.Count,
|
|
(int)adapter->PendingReadIrpQueue.MaxCount,
|
|
(int)IRP_QUEUE_SIZE, // Ignored in NDIS 6 driver...
|
|
|
|
(int)adapter->SendPacketQueue.Count,
|
|
(int)adapter->SendPacketQueue.MaxCount,
|
|
(int)PACKET_QUEUE_SIZE,
|
|
|
|
(int)0, // adapter->InjectPacketQueue.Count - Unused
|
|
(int)0, // adapter->InjectPacketQueue.MaxCount - Unused
|
|
(int)INJECT_QUEUE_SIZE
|
|
);
|
|
|
|
Irp->IoStatus.Information = outBufLength;
|
|
|
|
// BUGBUG!!! Fail because this is not completely implemented.
|
|
ntStatus = STATUS_INVALID_DEVICE_REQUEST;
|
|
}
|
|
#endif
|
|
|
|
#if DBG
|
|
case TAP_WIN_IOCTL_GET_LOG_LINE:
|
|
{
|
|
if (GetDebugLine( (LPTSTR)Irp->AssociatedIrp.SystemBuffer,outBufLength))
|
|
{
|
|
Irp->IoStatus.Status = ntStatus = STATUS_SUCCESS;
|
|
}
|
|
else
|
|
{
|
|
Irp->IoStatus.Status = ntStatus = STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
Irp->IoStatus.Information = outBufLength;
|
|
|
|
break;
|
|
}
|
|
#endif
|
|
|
|
case TAP_WIN_IOCTL_SET_MEDIA_STATUS:
|
|
{
|
|
if(inBufLength >= sizeof(ULONG))
|
|
{
|
|
ULONG parm = ((PULONG) (Irp->AssociatedIrp.SystemBuffer))[0];
|
|
tapSetMediaConnectStatus (adapter, (BOOLEAN) parm);
|
|
Irp->IoStatus.Information = 1;
|
|
}
|
|
else
|
|
{
|
|
NOTE_ERROR();
|
|
Irp->IoStatus.Status = ntStatus = STATUS_INVALID_PARAMETER;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
|
|
//
|
|
// The specified I/O control code is unrecognized by this driver.
|
|
//
|
|
ntStatus = STATUS_INVALID_DEVICE_REQUEST;
|
|
break;
|
|
}
|
|
|
|
End:
|
|
|
|
//
|
|
// Finish the I/O operation by simply completing the packet and returning
|
|
// the same status as in the packet itself.
|
|
//
|
|
Irp->IoStatus.Status = ntStatus;
|
|
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
return ntStatus;
|
|
}
|
|
|
|
// Flush the pending read IRP queue.
|
|
VOID
|
|
tapFlushIrpQueues(
|
|
__in PTAP_ADAPTER_CONTEXT Adapter
|
|
)
|
|
{
|
|
|
|
DEBUGP (("[TAP] tapFlushIrpQueues: Flushing %d pending read IRPs\n",
|
|
Adapter->PendingReadIrpQueue.Count));
|
|
|
|
tapIrpCsqFlush(&Adapter->PendingReadIrpQueue);
|
|
}
|
|
|
|
// IRP_MJ_CLEANUP
|
|
NTSTATUS
|
|
TapDeviceCleanup(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Receipt of this request indicates that the last handle for a file
|
|
object that is associated with the target device object has been closed
|
|
(but, due to outstanding I/O requests, might not have been released).
|
|
|
|
A driver that holds pending IRPs internally must implement a routine for
|
|
IRP_MJ_CLEANUP. When the routine is called, the driver should cancel all
|
|
the pending IRPs that belong to the file object identified by the IRP_MJ_CLEANUP
|
|
call.
|
|
|
|
In other words, it should cancel all the IRPs that have the same file-object
|
|
pointer as the one supplied in the current I/O stack location of the IRP for the
|
|
IRP_MJ_CLEANUP call. Of course, IRPs belonging to other file objects should
|
|
not be canceled. Also, if an outstanding IRP is completed immediately, the
|
|
driver does not have to cancel it.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the object that represents the device
|
|
to be cleaned up.
|
|
|
|
Irp - a pointer to the I/O Request Packet for this request.
|
|
|
|
Return Value:
|
|
|
|
NT status code
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS status = NDIS_STATUS_SUCCESS; // Always succeed.
|
|
PIO_STACK_LOCATION irpSp; // Pointer to current stack location
|
|
PTAP_ADAPTER_CONTEXT adapter = NULL;
|
|
|
|
PAGED_CODE();
|
|
|
|
DEBUGP (("[TAP] --> TapDeviceCleanup\n"));
|
|
|
|
irpSp = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
//
|
|
// Fetch adapter context for this device.
|
|
// --------------------------------------
|
|
// Adapter pointer was stashed in FsContext when handle was opened.
|
|
//
|
|
adapter = (PTAP_ADAPTER_CONTEXT )(irpSp->FileObject)->FsContext;
|
|
|
|
// Insure that adapter exists.
|
|
ASSERT(adapter);
|
|
|
|
if(adapter == NULL )
|
|
{
|
|
DEBUGP (("[TAP] release [%d.%d] cleanup request; adapter not found\n",
|
|
TAP_DRIVER_MAJOR_VERSION,
|
|
TAP_DRIVER_MINOR_VERSION
|
|
));
|
|
}
|
|
|
|
if(adapter != NULL )
|
|
{
|
|
adapter->TapFileIsOpen = 0; // Legacy...
|
|
|
|
// Disconnect from media.
|
|
tapSetMediaConnectStatus(adapter,FALSE);
|
|
|
|
// Reset adapter state when cleaning up;
|
|
tapResetAdapterState(adapter);
|
|
|
|
// BUGBUG!!! Use RemoveLock???
|
|
|
|
//
|
|
// Flush pending send TAP packet queue.
|
|
//
|
|
tapFlushSendPacketQueue(adapter);
|
|
|
|
ASSERT(adapter->SendPacketQueue.Count == 0);
|
|
|
|
//
|
|
// Flush the pending IRP queues
|
|
//
|
|
tapFlushIrpQueues(adapter);
|
|
|
|
ASSERT(adapter->PendingReadIrpQueue.Count == 0);
|
|
}
|
|
|
|
// Complete the IRP.
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
DEBUGP (("[TAP] <-- TapDeviceCleanup; status = %8.8X\n",status));
|
|
|
|
return status;
|
|
}
|
|
|
|
// IRP_MJ_CLOSE
|
|
NTSTATUS
|
|
TapDeviceClose(
|
|
PDEVICE_OBJECT DeviceObject,
|
|
PIRP Irp
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Receipt of this request indicates that the last handle of the file
|
|
object that is associated with the target device object has been closed
|
|
and released.
|
|
|
|
All outstanding I/O requests have been completed or canceled.
|
|
|
|
Arguments:
|
|
|
|
DeviceObject - a pointer to the object that represents the device
|
|
to be closed.
|
|
|
|
Irp - a pointer to the I/O Request Packet for this request.
|
|
|
|
Return Value:
|
|
|
|
NT status code
|
|
|
|
--*/
|
|
|
|
{
|
|
NDIS_STATUS status = NDIS_STATUS_SUCCESS; // Always succeed.
|
|
PIO_STACK_LOCATION irpSp; // Pointer to current stack location
|
|
PTAP_ADAPTER_CONTEXT adapter = NULL;
|
|
|
|
PAGED_CODE();
|
|
|
|
DEBUGP (("[TAP] --> TapDeviceClose\n"));
|
|
|
|
irpSp = IoGetCurrentIrpStackLocation(Irp);
|
|
|
|
//
|
|
// Fetch adapter context for this device.
|
|
// --------------------------------------
|
|
// Adapter pointer was stashed in FsContext when handle was opened.
|
|
//
|
|
adapter = (PTAP_ADAPTER_CONTEXT )(irpSp->FileObject)->FsContext;
|
|
|
|
// Insure that adapter exists.
|
|
ASSERT(adapter);
|
|
|
|
if(adapter == NULL )
|
|
{
|
|
DEBUGP (("[TAP] release [%d.%d] close request; adapter not found\n",
|
|
TAP_DRIVER_MAJOR_VERSION,
|
|
TAP_DRIVER_MINOR_VERSION
|
|
));
|
|
}
|
|
|
|
if(adapter != NULL )
|
|
{
|
|
if(adapter->TapFileObject == NULL)
|
|
{
|
|
// Should never happen!!!
|
|
ASSERT(FALSE);
|
|
}
|
|
else
|
|
{
|
|
ASSERT(irpSp->FileObject->FsContext == adapter);
|
|
|
|
ASSERT(adapter->TapFileObject == irpSp->FileObject);
|
|
}
|
|
|
|
adapter->TapFileObject = NULL;
|
|
irpSp->FileObject = NULL;
|
|
|
|
// Remove reference added by when handle was opened.
|
|
tapAdapterContextDereference(adapter);
|
|
}
|
|
|
|
// Complete the IRP.
|
|
Irp->IoStatus.Status = status;
|
|
Irp->IoStatus.Information = 0;
|
|
|
|
IoCompleteRequest( Irp, IO_NO_INCREMENT );
|
|
|
|
DEBUGP (("[TAP] <-- TapDeviceClose; status = %8.8X\n",status));
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
tapConcatenateNdisStrings(
|
|
__inout PNDIS_STRING DestinationString,
|
|
__in_opt PNDIS_STRING SourceString1,
|
|
__in_opt PNDIS_STRING SourceString2,
|
|
__in_opt PNDIS_STRING SourceString3
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
|
|
ASSERT(SourceString1 && SourceString2 && SourceString3);
|
|
|
|
status = RtlAppendUnicodeStringToString(
|
|
DestinationString,
|
|
SourceString1
|
|
);
|
|
|
|
if(status == STATUS_SUCCESS)
|
|
{
|
|
status = RtlAppendUnicodeStringToString(
|
|
DestinationString,
|
|
SourceString2
|
|
);
|
|
|
|
if(status == STATUS_SUCCESS)
|
|
{
|
|
status = RtlAppendUnicodeStringToString(
|
|
DestinationString,
|
|
SourceString3
|
|
);
|
|
}
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
tapMakeDeviceNames(
|
|
__in PTAP_ADAPTER_CONTEXT Adapter
|
|
)
|
|
{
|
|
NDIS_STATUS status;
|
|
NDIS_STRING deviceNamePrefix = NDIS_STRING_CONST("\\Device\\");
|
|
NDIS_STRING tapNameSuffix = NDIS_STRING_CONST(".tap");
|
|
|
|
// Generate DeviceName from NetCfgInstanceId.
|
|
Adapter->DeviceName.Buffer = Adapter->DeviceNameBuffer;
|
|
Adapter->DeviceName.MaximumLength = sizeof(Adapter->DeviceNameBuffer);
|
|
|
|
status = tapConcatenateNdisStrings(
|
|
&Adapter->DeviceName,
|
|
&deviceNamePrefix,
|
|
&Adapter->NetCfgInstanceId,
|
|
&tapNameSuffix
|
|
);
|
|
|
|
if(status == STATUS_SUCCESS)
|
|
{
|
|
NDIS_STRING linkNamePrefix = NDIS_STRING_CONST("\\DosDevices\\Global\\");
|
|
|
|
Adapter->LinkName.Buffer = Adapter->LinkNameBuffer;
|
|
Adapter->LinkName.MaximumLength = sizeof(Adapter->LinkNameBuffer);
|
|
|
|
status = tapConcatenateNdisStrings(
|
|
&Adapter->LinkName,
|
|
&linkNamePrefix,
|
|
&Adapter->NetCfgInstanceId,
|
|
&tapNameSuffix
|
|
);
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NDIS_STATUS
|
|
CreateTapDevice(
|
|
__in PTAP_ADAPTER_CONTEXT Adapter
|
|
)
|
|
{
|
|
NDIS_STATUS status;
|
|
NDIS_DEVICE_OBJECT_ATTRIBUTES deviceAttribute;
|
|
PDRIVER_DISPATCH dispatchTable[IRP_MJ_MAXIMUM_FUNCTION+1];
|
|
|
|
DEBUGP (("[TAP] version [%d.%d] creating tap device: %wZ\n",
|
|
TAP_DRIVER_MAJOR_VERSION,
|
|
TAP_DRIVER_MINOR_VERSION,
|
|
&Adapter->NetCfgInstanceId));
|
|
|
|
// Generate DeviceName and LinkName from NetCfgInstanceId.
|
|
status = tapMakeDeviceNames(Adapter);
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
DEBUGP (("[TAP] DeviceName: %wZ\n",&Adapter->DeviceName));
|
|
DEBUGP (("[TAP] LinkName: %wZ\n",&Adapter->LinkName));
|
|
|
|
// Initialize dispatch table.
|
|
NdisZeroMemory(dispatchTable, (IRP_MJ_MAXIMUM_FUNCTION+1) * sizeof(PDRIVER_DISPATCH));
|
|
|
|
dispatchTable[IRP_MJ_CREATE] = TapDeviceCreate;
|
|
dispatchTable[IRP_MJ_CLEANUP] = TapDeviceCleanup;
|
|
dispatchTable[IRP_MJ_CLOSE] = TapDeviceClose;
|
|
dispatchTable[IRP_MJ_READ] = TapDeviceRead;
|
|
dispatchTable[IRP_MJ_WRITE] = TapDeviceWrite;
|
|
dispatchTable[IRP_MJ_DEVICE_CONTROL] = TapDeviceControl;
|
|
|
|
//
|
|
// Create a device object and register dispatch handlers
|
|
//
|
|
NdisZeroMemory(&deviceAttribute, sizeof(NDIS_DEVICE_OBJECT_ATTRIBUTES));
|
|
|
|
deviceAttribute.Header.Type = NDIS_OBJECT_TYPE_DEVICE_OBJECT_ATTRIBUTES;
|
|
deviceAttribute.Header.Revision = NDIS_DEVICE_OBJECT_ATTRIBUTES_REVISION_1;
|
|
deviceAttribute.Header.Size = sizeof(NDIS_DEVICE_OBJECT_ATTRIBUTES);
|
|
|
|
deviceAttribute.DeviceName = &Adapter->DeviceName;
|
|
deviceAttribute.SymbolicName = &Adapter->LinkName;
|
|
deviceAttribute.MajorFunctions = &dispatchTable[0];
|
|
//deviceAttribute.ExtensionSize = sizeof(FILTER_DEVICE_EXTENSION);
|
|
|
|
#if ENABLE_NONADMIN
|
|
if(Adapter->AllowNonAdmin)
|
|
{
|
|
//
|
|
// SDDL_DEVOBJ_SYS_ALL_WORLD_RWX_RES_RWX allows the kernel and system complete
|
|
// control over the device. By default the admin can access the entire device,
|
|
// but cannot change the ACL (the admin must take control of the device first)
|
|
//
|
|
// Everyone else, including "restricted" or "untrusted" code can read or write
|
|
// to the device. Traversal beneath the device is also granted (removing it
|
|
// would only effect storage devices, except if the "bypass-traversal"
|
|
// privilege was revoked).
|
|
//
|
|
deviceAttribute.DefaultSDDLString = &SDDL_DEVOBJ_SYS_ALL_ADM_RWX_WORLD_RWX_RES_RWX;
|
|
}
|
|
#endif
|
|
|
|
status = NdisRegisterDeviceEx(
|
|
Adapter->MiniportAdapterHandle,
|
|
&deviceAttribute,
|
|
&Adapter->DeviceObject,
|
|
&Adapter->DeviceHandle
|
|
);
|
|
}
|
|
|
|
ASSERT(NT_SUCCESS(status));
|
|
|
|
if (NT_SUCCESS(status))
|
|
{
|
|
// Set TAP device flags.
|
|
(Adapter->DeviceObject)->Flags &= ~DO_BUFFERED_IO;
|
|
(Adapter->DeviceObject)->Flags |= DO_DIRECT_IO;;
|
|
|
|
//========================
|
|
// Finalize initialization
|
|
//========================
|
|
|
|
Adapter->TapDeviceCreated = TRUE;
|
|
|
|
DEBUGP (("[%wZ] successfully created TAP device [%wZ]\n",
|
|
&Adapter->NetCfgInstanceId,
|
|
&Adapter->DeviceName
|
|
));
|
|
}
|
|
|
|
DEBUGP (("[TAP] <-- CreateTapDevice; status = %8.8X\n",status));
|
|
|
|
return status;
|
|
}
|
|
|
|
//
|
|
// DestroyTapDevice is called from AdapterHalt and NDIS miniport
|
|
// is in Halted state. Prior to entering the Halted state the
|
|
// miniport would have passed through the Pausing and Paused
|
|
// states. These miniport states have responsibility for waiting
|
|
// until NDIS network operations have completed.
|
|
//
|
|
VOID
|
|
DestroyTapDevice(
|
|
__in PTAP_ADAPTER_CONTEXT Adapter
|
|
)
|
|
{
|
|
DEBUGP (("[TAP] --> DestroyTapDevice; Adapter: %wZ\n",
|
|
&Adapter->NetCfgInstanceId));
|
|
|
|
//
|
|
// Let clients know we are shutting down
|
|
//
|
|
Adapter->TapDeviceCreated = FALSE;
|
|
|
|
//
|
|
// Flush pending send TAP packet queue.
|
|
//
|
|
tapFlushSendPacketQueue(Adapter);
|
|
|
|
ASSERT(Adapter->SendPacketQueue.Count == 0);
|
|
|
|
//
|
|
// Flush IRP queues. Wait for pending I/O. Etc.
|
|
// --------------------------------------------
|
|
// Exhaust IRP and packet queues. Any pending IRPs will
|
|
// be cancelled, causing user-space to get this error
|
|
// on overlapped reads:
|
|
//
|
|
// ERROR_OPERATION_ABORTED, code=995
|
|
//
|
|
// "The I/O operation has been aborted because of either a
|
|
// thread exit or an application request."
|
|
//
|
|
// It's important that user-space close the device handle
|
|
// when this code is returned, so that when we finally
|
|
// do a NdisMDeregisterDeviceEx, the device reference count
|
|
// is 0. Otherwise the driver will not unload even if the
|
|
// the last adapter has been halted.
|
|
//
|
|
// The act of flushing the queues at this point should result in the user-mode
|
|
// application closing the adapter's device handle. Closing the handle will
|
|
// result in the TapDeviceCleanup call being made, followed by the a call to
|
|
// the TapDeviceClose callback.
|
|
//
|
|
tapFlushIrpQueues(Adapter);
|
|
|
|
ASSERT(Adapter->PendingReadIrpQueue.Count == 0);
|
|
|
|
//
|
|
// Deregister the Win32 device.
|
|
// ----------------------------
|
|
// When a driver calls NdisDeregisterDeviceEx, the I/O manager deletes the
|
|
// target device object if there are no outstanding references to it. However,
|
|
// if any outstanding references remain, the I/O manager marks the device
|
|
// object as "delete pending" and deletes the device object when the references
|
|
// are finally released.
|
|
//
|
|
if(Adapter->DeviceHandle)
|
|
{
|
|
DEBUGP (("[TAP] Calling NdisDeregisterDeviceEx\n"));
|
|
NdisDeregisterDeviceEx(Adapter->DeviceHandle);
|
|
}
|
|
|
|
Adapter->DeviceHandle = NULL;
|
|
|
|
DEBUGP (("[TAP] <-- DestroyTapDevice\n"));
|
|
}
|
|
|