You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1765 lines
66 KiB
1765 lines
66 KiB
/*++
|
|
|
|
Copyright (c) 1989 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
afdstr.h
|
|
|
|
Abstract:
|
|
|
|
This module contains typedefs for structures used by AFD.
|
|
|
|
Author:
|
|
|
|
David Treadwell (davidtr) 21-Feb-1992
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#ifndef _AFDSTR_
|
|
#define _AFDSTR_
|
|
|
|
//
|
|
// This field defines the layout of a 32-bit pointer.
|
|
// This must be ULONG as wow64 support 4gb address on win64.
|
|
// You shouldn't POINTER_32 as this will sign-extend addresses when converting
|
|
// it to a 64-bit pointer and will end up having the wrong address.
|
|
//
|
|
typedef ULONG AFD_POINTER_32;
|
|
|
|
//
|
|
// Make sure that queued spinlocks are not used with
|
|
// regular spinlock functions by wrapping them into
|
|
// a different structure.
|
|
//
|
|
typedef struct _AFD_QSPIN_LOCK {
|
|
KSPIN_LOCK ActualSpinLock;
|
|
} AFD_QSPIN_LOCK, *PAFD_QSPIN_LOCK;
|
|
|
|
#if DBG
|
|
|
|
#ifndef REFERENCE_DEBUG
|
|
#define REFERENCE_DEBUG 1
|
|
#endif
|
|
|
|
#ifndef GLOBAL_REFERENCE_DEBUG
|
|
#define GLOBAL_REFERENCE_DEBUG 0
|
|
#endif
|
|
|
|
//
|
|
// Debug aid for queued spinlocks
|
|
// Allows us to verify that spinlock is released using
|
|
// the same handle as it was taken with.
|
|
//
|
|
typedef struct _AFD_LOCK_QUEUE_HANDLE {
|
|
KLOCK_QUEUE_HANDLE LockHandle;
|
|
PAFD_QSPIN_LOCK SpinLock;
|
|
} AFD_LOCK_QUEUE_HANDLE, *PAFD_LOCK_QUEUE_HANDLE;
|
|
|
|
#else
|
|
|
|
#ifndef REFERENCE_DEBUG
|
|
#define REFERENCE_DEBUG 0
|
|
#endif
|
|
|
|
#ifndef GLOBAL_REFERENCE_DEBUG
|
|
#define GLOBAL_REFERENCE_DEBUG 0
|
|
#endif
|
|
|
|
#define AFD_LOCK_QUEUE_HANDLE KLOCK_QUEUE_HANDLE
|
|
#define PAFD_LOCK_QUEUE_HANDLE PKLOCK_QUEUE_HANDLE
|
|
|
|
#endif // DBG
|
|
|
|
#if REFERENCE_DEBUG
|
|
|
|
#define AFD_REF_SHIFT 6
|
|
#define AFD_REF_MASK ((1<<AFD_REF_SHIFT)-1)
|
|
#define AFD_MAX_REF (1<<AFD_REF_SHIFT)
|
|
#define AFD_REF_TIME_BITS 16
|
|
#define AFD_MAX_REF_TIME ((1<<AFD_REF_TIME_BITS)-1)
|
|
#define AFD_TIME_EXP_SHIFT 4
|
|
#define AFD_TIME_EXP_BITS 2
|
|
#define AFD_REF_CNT_BITS 4
|
|
#define AFD_REF_LOC_BITS (32-AFD_REF_CNT_BITS-AFD_REF_TIME_BITS-AFD_TIME_EXP_BITS)
|
|
|
|
C_ASSERT (32-AFD_REF_SHIFT-AFD_REF_TIME_BITS <=
|
|
AFD_TIME_EXP_SHIFT*((1<<AFD_TIME_EXP_BITS)-1));
|
|
|
|
typedef union _AFD_REFERENCE_DEBUG {
|
|
struct {
|
|
ULONGLONG NewCount:AFD_REF_CNT_BITS;
|
|
ULONGLONG LocationId:AFD_REF_LOC_BITS;
|
|
ULONGLONG TimeExp:AFD_TIME_EXP_BITS;
|
|
ULONGLONG TimeDif:AFD_REF_TIME_BITS;
|
|
ULONGLONG Param:32;
|
|
};
|
|
ULONGLONG QuadPart;
|
|
} AFD_REFERENCE_DEBUG, *PAFD_REFERENCE_DEBUG;
|
|
C_ASSERT (sizeof (AFD_REFERENCE_DEBUG)==sizeof (ULONGLONG));
|
|
|
|
typedef struct _AFD_REFERENCE_LOCATION {
|
|
PCHAR Format;
|
|
PVOID Address;
|
|
} AFD_REFERENCE_LOCATION, *PAFD_REFERENCE_LOCATION;
|
|
|
|
LONG
|
|
AfdFindReferenceLocation (
|
|
IN PCHAR Format,
|
|
OUT PLONG LocationId
|
|
);
|
|
|
|
#define AFD_GET_ARL(_s) (_arl ? _arl : AfdFindReferenceLocation((_s),&_arl))
|
|
|
|
#define AFD_UPDATE_REFERENCE_DEBUG(_rd,_r,_l,_p) \
|
|
do { \
|
|
LONG _n, _n1; \
|
|
ULONGLONG _t = KeQueryInterruptTime (); \
|
|
_n = (_rd)->CurrentReferenceSlot; \
|
|
_n1 = (((LONG)(_t>>13))<<AFD_REF_SHIFT) + ((_n+1)&AFD_REF_MASK);\
|
|
if (InterlockedCompareExchange (&(_rd)->CurrentReferenceSlot, \
|
|
_n1,_n)==_n) { \
|
|
PAFD_REFERENCE_DEBUG _s; \
|
|
LONG _d,_e=0; \
|
|
(_rd)->CurrentTimeHigh = (LONG)(_t>>(13+32-AFD_REF_SHIFT)); \
|
|
_s = &(_rd)->ReferenceDebug[_n & AFD_REF_MASK]; \
|
|
_s->NewCount = _r; \
|
|
_s->LocationId = _l; \
|
|
_s->Param = _p; \
|
|
_d = (_n1-1-_n) >> AFD_REF_SHIFT; \
|
|
while (_d>=AFD_MAX_REF_TIME) { \
|
|
_e += 1; \
|
|
_d >>= _e*AFD_TIME_EXP_SHIFT; \
|
|
} \
|
|
_s->TimeDif = _d; \
|
|
_s->TimeExp = _e; \
|
|
break; \
|
|
} \
|
|
} while (1)
|
|
|
|
|
|
#if GLOBAL_REFERENCE_DEBUG
|
|
#define MAX_GLOBAL_REFERENCE 4096
|
|
|
|
typedef struct _AFD_GLOBAL_REFERENCE_DEBUG {
|
|
PVOID Info1;
|
|
PVOID Info2;
|
|
PVOID Connection;
|
|
ULONG_PTR Action;
|
|
LARGE_INTEGER TickCounter;
|
|
ULONG NewCount;
|
|
ULONG Dummy;
|
|
} AFD_GLOBAL_REFERENCE_DEBUG, *PAFD_GLOBAL_REFERENCE_DEBUG;
|
|
#endif
|
|
|
|
#endif
|
|
|
|
//
|
|
// A structure for maintaining work queue information in AFD.
|
|
//
|
|
|
|
typedef struct _AFD_WORK_ITEM {
|
|
LIST_ENTRY WorkItemListEntry;
|
|
PWORKER_THREAD_ROUTINE AfdWorkerRoutine;
|
|
PVOID Context;
|
|
} AFD_WORK_ITEM, *PAFD_WORK_ITEM;
|
|
|
|
//
|
|
// Structures for holding connect data pointers and lengths. This is
|
|
// kept separate from the normal structures to save space in those
|
|
// structures for transports that do not support and applications
|
|
// which do not use connect data.
|
|
//
|
|
|
|
typedef struct _AFD_CONNECT_DATA_INFO {
|
|
PVOID Buffer;
|
|
ULONG BufferLength;
|
|
} AFD_CONNECT_DATA_INFO, *PAFD_CONNECT_DATA_INFO;
|
|
|
|
typedef struct _AFD_CONNECT_DATA_BUFFERS {
|
|
AFD_CONNECT_DATA_INFO SendConnectData;
|
|
AFD_CONNECT_DATA_INFO SendConnectOptions;
|
|
AFD_CONNECT_DATA_INFO ReceiveConnectData;
|
|
AFD_CONNECT_DATA_INFO ReceiveConnectOptions;
|
|
AFD_CONNECT_DATA_INFO SendDisconnectData;
|
|
AFD_CONNECT_DATA_INFO SendDisconnectOptions;
|
|
AFD_CONNECT_DATA_INFO ReceiveDisconnectData;
|
|
AFD_CONNECT_DATA_INFO ReceiveDisconnectOptions;
|
|
TDI_CONNECTION_INFORMATION RequestConnectionInfo;
|
|
TDI_CONNECTION_INFORMATION ReturnConnectionInfo;
|
|
ULONG Flags;
|
|
} AFD_CONNECT_DATA_BUFFERS, *PAFD_CONNECT_DATA_BUFFERS;
|
|
|
|
//
|
|
// Structure used for holding disconnect context information.
|
|
//
|
|
|
|
typedef struct _AFD_ENDPOINT AFD_ENDPOINT, *PAFD_ENDPOINT;
|
|
typedef struct _AFD_CONNECTION AFD_CONNECTION, *PAFD_CONNECTION;
|
|
|
|
typedef struct _AFD_TPACKETS_INFO_INTERNAL
|
|
AFD_TPACKETS_INFO_INTERNAL,
|
|
*PAFD_TPACKETS_INFO_INTERNAL;
|
|
|
|
typedef DECLSPEC_ALIGN(MEMORY_ALLOCATION_ALIGNMENT)
|
|
struct _AFD_BUFFER
|
|
AFD_BUFFER,
|
|
*PAFD_BUFFER;
|
|
|
|
typedef struct _AFD_DISCONNECT_CONTEXT {
|
|
LARGE_INTEGER Timeout;
|
|
PIRP Irp;
|
|
} AFD_DISCONNECT_CONTEXT, *PAFD_DISCONNECT_CONTEXT;
|
|
|
|
|
|
typedef struct _AFD_LR_LIST_ITEM AFD_LR_LIST_ITEM, *PAFD_LR_LIST_ITEM;
|
|
typedef BOOLEAN (* PAFD_LR_LIST_ROUTINE) (PAFD_LR_LIST_ITEM Item);
|
|
|
|
struct _AFD_LR_LIST_ITEM {
|
|
SLIST_ENTRY SListLink; // Link in the list
|
|
PAFD_LR_LIST_ROUTINE Routine; // Processing routine;
|
|
};
|
|
|
|
|
|
//
|
|
// Endpoint and connection structures and related informaion.
|
|
//
|
|
//
|
|
// Block types that identify which fields are
|
|
// available in the strucutures.
|
|
//
|
|
|
|
#define AfdBlockTypeEndpoint 0xAFD0 // VC endpoint before connect/accept/listen
|
|
#define AfdBlockTypeDatagram 0xAFD1 // Datagram endpoint in any state
|
|
#define AfdBlockTypeVcConnecting 0xAFD2 // VC endpoint after connect/accept
|
|
#define AfdBlockTypeVcListening 0xAFD4 // VC endpoint after listen
|
|
#define AfdBlockTypeVcBoth 0xAFD6 // VC endpoint after connect+listen
|
|
// (used in multipoint root cases)
|
|
|
|
|
|
#define AfdBlockTypeHelper 0xAAFD // Regular helper endpoint
|
|
#define AfdBlockTypeSanHelper 0x0AFD // SAN helper endpoint
|
|
#define AfdBlockTypeSanEndpoint 0x1AFD // SAN endpoint
|
|
|
|
#define AfdBlockTypeInvalidEndpoint 0xCAFD // Right before we call ExFreePool
|
|
|
|
#define AfdBlockTypeConnection 0xAFD8 // Connection object
|
|
#define AfdBlockTypeInvalidConnection 0xEAFD // Right before we call ExFreePool
|
|
|
|
|
|
#if DBG
|
|
#define IS_AFD_ENDPOINT_TYPE( endpoint ) \
|
|
( (endpoint)->Type == AfdBlockTypeEndpoint || \
|
|
(endpoint)->Type == AfdBlockTypeDatagram || \
|
|
(endpoint)->Type == AfdBlockTypeVcConnecting || \
|
|
(endpoint)->Type == AfdBlockTypeVcListening || \
|
|
(endpoint)->Type == AfdBlockTypeVcBoth || \
|
|
(endpoint)->Type == AfdBlockTypeHelper || \
|
|
(endpoint)->Type == AfdBlockTypeSanHelper || \
|
|
(endpoint)->Type == AfdBlockTypeSanEndpoint )
|
|
#endif
|
|
|
|
enum {
|
|
AfdConnectionStateFree = 0, // On free/AcceptEx/DelayedAccept list
|
|
AfdConnectionStateUnaccepted = 1, // On unaccepted list
|
|
AfdConnectionStateReturned = 2, // On returned list
|
|
AfdConnectionStateConnected = 3, // Accepted or connected
|
|
AfdConnectionStateClosing = 4 // Not used anymore
|
|
};
|
|
|
|
//
|
|
// Flags that further qualify the state of the connection
|
|
//
|
|
typedef struct AFD_CONNECTION_STATE_FLAGS {
|
|
union {
|
|
struct {
|
|
LOGICAL TdiBufferring:1, // (Does not really belon here)
|
|
:3, // This spacing makes strcutures
|
|
// much more readable (hex) in the
|
|
// debugger and has no effect
|
|
// on the generated code as long
|
|
// as number of flags is less than
|
|
// 8 (we still take up full 32 bits
|
|
// because of aligment requiremens
|
|
// of most other fields)
|
|
Aborted:1,
|
|
AbortIndicated:1,
|
|
AbortFailed:1,
|
|
:1,
|
|
DisconnectIndicated:1,
|
|
:3,
|
|
ConnectedReferenceAdded:1,
|
|
:3,
|
|
SpecialCondition:1,
|
|
:3,
|
|
CleanupBegun:1,
|
|
:3,
|
|
ClosePendedTransmit:1,
|
|
:3,
|
|
OnLRList:1, // Low-resource list
|
|
SanConnection:1,
|
|
RcvInitiated:1, // AFD initiated receive to transport.
|
|
:1;
|
|
};
|
|
LOGICAL ConnectionStateFlags;
|
|
};
|
|
} AFD_CONNECTION_STATE_FLAGS;
|
|
C_ASSERT (sizeof (AFD_CONNECTION_STATE_FLAGS)==sizeof (LOGICAL));
|
|
|
|
struct _AFD_CONNECTION {
|
|
// *** Frequently used, mostly read-only fields (state/type/flag changes are rare).
|
|
USHORT Type;
|
|
USHORT State;
|
|
AFD_CONNECTION_STATE_FLAGS
|
|
#ifdef __cplusplus
|
|
StateFlags
|
|
#endif
|
|
;
|
|
|
|
PAFD_ENDPOINT Endpoint;
|
|
PFILE_OBJECT FileObject;
|
|
PDEVICE_OBJECT DeviceObject;
|
|
PEPROCESS OwningProcess;
|
|
|
|
|
|
union {
|
|
LONGLONG ConnectTime; // After connection is accepted.
|
|
PIRP AcceptIrp; // For AcceptEx
|
|
PIRP ListenIrp; // For delayed accept
|
|
PIRP ConnectIrp; // for SAN
|
|
};
|
|
|
|
// *** Frequently used volatile fields.
|
|
volatile LONG ReferenceCount;
|
|
|
|
union {
|
|
|
|
struct {
|
|
LARGE_INTEGER ReceiveBytesIndicated;
|
|
LARGE_INTEGER ReceiveBytesTaken;
|
|
LARGE_INTEGER ReceiveBytesOutstanding;
|
|
|
|
LARGE_INTEGER ReceiveExpeditedBytesIndicated;
|
|
LARGE_INTEGER ReceiveExpeditedBytesTaken;
|
|
LARGE_INTEGER ReceiveExpeditedBytesOutstanding;
|
|
BOOLEAN NonBlockingSendPossible;
|
|
BOOLEAN ZeroByteReceiveIndicated;
|
|
} Bufferring;
|
|
|
|
struct {
|
|
LIST_ENTRY ReceiveIrpListHead;
|
|
LIST_ENTRY ReceiveBufferListHead;
|
|
|
|
ULONG BufferredReceiveBytes;
|
|
ULONG BufferredExpeditedBytes;
|
|
|
|
USHORT BufferredReceiveCount;
|
|
USHORT BufferredExpeditedCount;
|
|
ULONG ReceiveBytesInTransport;
|
|
|
|
LIST_ENTRY SendIrpListHead;
|
|
|
|
ULONG BufferredSendBytes;
|
|
ULONG BufferredSendCount;
|
|
|
|
PIRP DisconnectIrp;
|
|
|
|
LONG ReceiveIrpsInTransport; // debug only.
|
|
} NonBufferring;
|
|
|
|
} Common;
|
|
|
|
|
|
ULONG MaxBufferredReceiveBytes;
|
|
ULONG MaxBufferredSendBytes;
|
|
|
|
PTRANSPORT_ADDRESS RemoteAddress;
|
|
ULONG RemoteAddressLength;
|
|
LONG Sequence;
|
|
|
|
HANDLE Handle; // Ref count protection
|
|
|
|
|
|
|
|
union {
|
|
AFD_WORK_ITEM WorkItem; // Work item to free the connection
|
|
// Connection has to be at ref 0 to be
|
|
// on the work queue, so it cannot be
|
|
// on the lists below or being disconnected
|
|
// because when on any of these lists the
|
|
// ref count is above 0.
|
|
struct {
|
|
union {
|
|
AFD_DISCONNECT_CONTEXT DisconnectContext;
|
|
// Disconnect operation context, we cannot be
|
|
// on the listening endpoint list
|
|
SLIST_ENTRY SListEntry;
|
|
// Links for listening endpoint lists
|
|
LIST_ENTRY ListEntry;
|
|
};
|
|
AFD_LR_LIST_ITEM LRListItem; // Link for low resource list. When on this
|
|
// list connection is referenced, but it can
|
|
// also be on the listening endpoint list or
|
|
// in the process of disconnecting.
|
|
};
|
|
};
|
|
|
|
PAFD_CONNECT_DATA_BUFFERS ConnectDataBuffers;
|
|
|
|
#if REFERENCE_DEBUG
|
|
LONG CurrentTimeHigh;
|
|
volatile LONG CurrentReferenceSlot;
|
|
AFD_REFERENCE_DEBUG ReferenceDebug[AFD_MAX_REF];
|
|
#endif
|
|
|
|
#ifdef _AFD_VERIFY_DATA_
|
|
ULONGLONG VerifySequenceNumber;
|
|
#endif // _AFD_VERIFY_DATA_
|
|
};
|
|
// Fields of the AFD_CONNECTION are protected by the endpoint
|
|
// spinlock, except:
|
|
// Type, FileObject, DeviceObject, OwningProcess, Handle which are
|
|
// only set once and cleared when reference count goes to 0
|
|
// Endpoint field itself (and thus the spinlock) can change though.
|
|
// - it is NULL when connection is on listening endpoint backlog
|
|
// and connection cannot be referenced outside of listening endpoint.
|
|
// - when it is assigned to listening endpoint it cannot change outside
|
|
// of listening endpoint spinlock
|
|
// - when it is assigned to connecting/accepting endpoint, it cannot
|
|
// change at all until reference count drops to 0 and
|
|
// connection is again placed on listening endpoint backlog or
|
|
// destroyed.
|
|
// ReferenceCount only updated via interlocked operations.
|
|
//
|
|
|
|
#ifdef _AFD_VERIFY_DATA_
|
|
VOID
|
|
AfdVerifyBuffer (
|
|
PAFD_CONNECTION Connection,
|
|
PVOID Buffer,
|
|
ULONG Length
|
|
);
|
|
VOID
|
|
AfdVerifyMdl (
|
|
PAFD_CONNECTION Connection,
|
|
PMDL Mdl,
|
|
ULONG Offset,
|
|
ULONG Length
|
|
);
|
|
VOID
|
|
AfdVerifyAddress (
|
|
PAFD_CONNECTION Connection,
|
|
PTRANSPORT_ADDRESS Address
|
|
);
|
|
|
|
#define AFD_VERIFY_BUFFER(_connection,_buffer,_length) \
|
|
AfdVerifyBuffer(_connection,_buffer,_length)
|
|
#define AFD_VERIFY_MDL(_connection,_mdl,_offset,_length) \
|
|
AfdVerifyMdl(_connection,_mdl,_offset,_length)
|
|
#define AFD_VERIFY_ADDRESS(_connection,_address) \
|
|
AfdVerifyAddress(_connection,_address)
|
|
#else
|
|
#define AFD_VERIFY_BUFFER(_connection,_buffer,_length)
|
|
#define AFD_VERIFY_MDL(_connection,_mdl,_offset,_length)
|
|
#define AFD_VERIFY_ADDRESS(_connection,_address)
|
|
#endif // _AFD_VERIFY_DATA_
|
|
|
|
//
|
|
// Some macros that make code more readable.
|
|
//
|
|
|
|
#define VcNonBlockingSendPossible Common.Bufferring.NonBlockingSendPossible
|
|
#define VcZeroByteReceiveIndicated Common.Bufferring.ZeroByteReceiveIndicated
|
|
|
|
#define VcReceiveIrpListHead Common.NonBufferring.ReceiveIrpListHead
|
|
#define VcReceiveBufferListHead Common.NonBufferring.ReceiveBufferListHead
|
|
#define VcSendIrpListHead Common.NonBufferring.SendIrpListHead
|
|
|
|
#define VcBufferredReceiveBytes Common.NonBufferring.BufferredReceiveBytes
|
|
#define VcBufferredExpeditedBytes Common.NonBufferring.BufferredExpeditedBytes
|
|
#define VcBufferredReceiveCount Common.NonBufferring.BufferredReceiveCount
|
|
#define VcBufferredExpeditedCount Common.NonBufferring.BufferredExpeditedCount
|
|
|
|
#define VcReceiveBytesInTransport Common.NonBufferring.ReceiveBytesInTransport
|
|
#if DBG
|
|
#define VcReceiveIrpsInTransport Common.NonBufferring.ReceiveIrpsInTransport
|
|
#endif
|
|
|
|
#define VcBufferredSendBytes Common.NonBufferring.BufferredSendBytes
|
|
#define VcBufferredSendCount Common.NonBufferring.BufferredSendCount
|
|
|
|
#define VcDisconnectIrp Common.NonBufferring.DisconnectIrp
|
|
|
|
//
|
|
// Information stored about each transport device name for which there
|
|
// is an open endpoint.
|
|
//
|
|
typedef
|
|
NTSTATUS
|
|
(FASTCALL *PIO_CALL_DRIVER) (
|
|
IN PDEVICE_OBJECT DeviceObject,
|
|
IN PIRP Irp
|
|
);
|
|
|
|
typedef
|
|
PAFD_BUFFER
|
|
(FASTCALL *PAFD_GET_BUFFER) (
|
|
IN ULONG BufferDataSize,
|
|
IN ULONG AddressSize,
|
|
IN PEPROCESS Process
|
|
);
|
|
|
|
typedef
|
|
PAFD_TPACKETS_INFO_INTERNAL
|
|
(FASTCALL *PAFD_GET_TPINFO) (
|
|
IN ULONG ElementCount
|
|
);
|
|
|
|
typedef struct _AFD_TRANSPORT_INFO {
|
|
LIST_ENTRY TransportInfoListEntry;
|
|
volatile LONG ReferenceCount;
|
|
BOOLEAN InfoValid;
|
|
#ifdef _AFD_VARIABLE_STACK_
|
|
CCHAR StackSize;
|
|
PAFD_GET_BUFFER GetBuffer;
|
|
PAFD_GET_TPINFO GetTpInfo;
|
|
PIO_CALL_DRIVER CallDriver;
|
|
#endif // _AFD_VARIABLE_STACK_
|
|
UNICODE_STRING TransportDeviceName;
|
|
TDI_PROVIDER_INFO ProviderInfo;
|
|
//WCHAR TransportDeviceNameStructure;
|
|
} AFD_TRANSPORT_INFO, *PAFD_TRANSPORT_INFO;
|
|
|
|
//
|
|
// Endpoint state definitions (can't be <=0 or state change
|
|
// macros won't work correctly).
|
|
//
|
|
enum {
|
|
AfdEndpointStateOpen =1, // Socket created but no yet bound
|
|
AfdEndpointStateBound =2,
|
|
AfdEndpointStateConnected =3,
|
|
AfdEndpointStateCleanup =4, // Not used anymore
|
|
AfdEndpointStateClosing =5,
|
|
AfdEndpointStateTransmitClosing =6, // TransmitFile with reuse
|
|
AfdEndpointStateInvalid =7 // Helper endpoints
|
|
};
|
|
|
|
//
|
|
// Flags that further qualify the state of the endpoint
|
|
//
|
|
typedef struct AFD_ENDPOINT_STATE_FLAGS {
|
|
union {
|
|
struct {
|
|
LOGICAL Listening:1,
|
|
DelayedAcceptance:1,
|
|
:2, // This spacing makes strcutures
|
|
// much more readable (hex) in the
|
|
// debugger and has no effect
|
|
// on the generated code as long
|
|
// as number of flags is less than
|
|
// 8 (we still take up full 32 bits
|
|
// because of aligment requiremens
|
|
// of most other fields)
|
|
NonBlocking:1,
|
|
:3,
|
|
InLine:1,
|
|
:3,
|
|
EndpointCleanedUp:1,
|
|
:3,
|
|
PollCalled:1,
|
|
:3,
|
|
RoutingQueryReferenced:1,
|
|
RoutingQueryIPv6:1,
|
|
:2,
|
|
DisableFastIoSend:1,
|
|
EnableSendEvent:1,
|
|
:2,
|
|
DisableFastIoRecv:1,
|
|
:3;
|
|
};
|
|
LOGICAL EndpointStateFlags;
|
|
};
|
|
} AFD_ENDPOINT_STATE_FLAGS;
|
|
C_ASSERT (sizeof (AFD_ENDPOINT_STATE_FLAGS)==sizeof (LOGICAL));
|
|
|
|
struct _AFD_ENDPOINT {
|
|
// *** Frequently used, mostly read-only fields (state/type/flag changes are rare).
|
|
USHORT Type;
|
|
UCHAR State;
|
|
BOOLEAN AdminAccessGranted;
|
|
ULONG TdiServiceFlags; // Tdi transport flags cached for quick access.
|
|
|
|
AFD_ENDPOINT_FLAGS __f; // As requested by the application through the
|
|
// Winsock2 provider flags and/or socket type
|
|
AFD_ENDPOINT_STATE_FLAGS
|
|
#ifdef __cplusplus
|
|
StateFlags
|
|
#endif
|
|
;
|
|
PFILE_OBJECT AddressFileObject;
|
|
union {
|
|
PDEVICE_OBJECT AddressDeviceObject;
|
|
PSECURITY_DESCRIPTOR SecurityDescriptor; // Only
|
|
};
|
|
PEPROCESS OwningProcess;
|
|
|
|
// *** Frequently used volatile fields.
|
|
AFD_QSPIN_LOCK SpinLock; // Pointer sized.
|
|
volatile LONG ReferenceCount;
|
|
ULONG EventsActive;
|
|
|
|
//
|
|
// Use a union to overlap the fields that are exclusive to datagram
|
|
// connecting, or listening endpoints. Since many fields are
|
|
// relevant to only one type of socket, it makes no sense to
|
|
// maintain the fields for all sockets--instead, save some nonpaged
|
|
// pool by combining them.
|
|
//
|
|
|
|
union {
|
|
//
|
|
// Information for circuit-based endpoints
|
|
//
|
|
struct {
|
|
union {
|
|
struct {
|
|
// These members are valid for listening endpoints
|
|
// (AfdBlockTypeVcListening).
|
|
LIST_ENTRY UnacceptedConnectionListHead;
|
|
LIST_ENTRY ReturnedConnectionListHead;
|
|
LIST_ENTRY ListeningIrpListHead;
|
|
|
|
// The below lists have their own lock which gets
|
|
// used on the machines that do not support 64-bit
|
|
// interlocked compare and exchange. On these machines
|
|
// using endpoint spinlock to synchronize causes all kinds
|
|
// of nasty deadlock conditions.
|
|
union {
|
|
LIST_ENTRY ListenConnectionListHead; // Delayed accept
|
|
SLIST_HEADER FreeConnectionListHead;
|
|
};
|
|
SLIST_HEADER PreacceptedConnectionsListHead;
|
|
|
|
LONG FailedConnectionAdds;
|
|
LONG TdiAcceptPendingCount;
|
|
|
|
LONG Sequence;
|
|
BOOLEAN EnableDynamicBacklog;
|
|
BOOLEAN BacklogReplenishActive; // Worker is scheduled
|
|
USHORT MaxExtraConnections; // Extra connections we keep in the free queue
|
|
// based on maximum number of AcceptEx requests
|
|
} Listening;
|
|
struct {
|
|
KAPC Apc;
|
|
USHORT RemoteSocketAddressOffset; // Offset inside of socket context
|
|
// pointing to remote address.
|
|
USHORT RemoteSocketAddressLength; // Length of the address.
|
|
#ifndef i386
|
|
BOOLEAN FixAddressAlignment; // Fix address alignment in
|
|
// SuperAccept
|
|
#endif
|
|
};
|
|
};
|
|
// These members are valid for all vc endpoints (but
|
|
// can be NULL)
|
|
PAFD_CONNECTION Connection;
|
|
struct _AFD_ENDPOINT *ListenEndpoint;
|
|
PAFD_CONNECT_DATA_BUFFERS ConnectDataBuffers;
|
|
} VirtualCircuit;
|
|
|
|
#define VcConnecting VirtualCircuit
|
|
#define VcListening VirtualCircuit.Listening
|
|
#define VcConnection VirtualCircuit.Connection
|
|
//
|
|
// Information for datagram endpoints. Note that different
|
|
// information is kept depending on whether the underlying
|
|
// transport buffers internally.
|
|
//
|
|
|
|
struct {
|
|
LIST_ENTRY ReceiveIrpListHead;
|
|
LIST_ENTRY PeekIrpListHead;
|
|
LIST_ENTRY ReceiveBufferListHead;
|
|
|
|
ULONG BufferredReceiveBytes;
|
|
ULONG BufferredReceiveCount;
|
|
|
|
ULONG MaxBufferredReceiveBytes;
|
|
ULONG BufferredSendBytes;
|
|
ULONG MaxBufferredSendBytes;
|
|
|
|
|
|
ULONG RemoteAddressLength;
|
|
PTRANSPORT_ADDRESS RemoteAddress;
|
|
|
|
union {
|
|
struct {
|
|
LOGICAL CircularQueueing:1,
|
|
:3,
|
|
HalfConnect:1,
|
|
:3,
|
|
DisablePUError:1,
|
|
:3,
|
|
AddressDrop:1,
|
|
ResourceDrop:1,
|
|
BufferDrop:1,
|
|
ErrorDrop:1;
|
|
};
|
|
LOGICAL Flags;
|
|
};
|
|
} Datagram;
|
|
|
|
struct {
|
|
LIST_ENTRY SanListLink;
|
|
PVOID IoCompletionPort;
|
|
PKEVENT IoCompletionEvent;
|
|
LONG Plsn; // Provider list sequence number
|
|
LONG PendingRequests; // Number of pending requests to this process
|
|
} SanHlpr;
|
|
|
|
struct {
|
|
struct _AFD_ENDPOINT *SanHlpr;
|
|
PFILE_OBJECT FileObject;
|
|
union {
|
|
//
|
|
// We can only have either one of two at any given time.
|
|
//
|
|
PAFD_SWITCH_CONTEXT SwitchContext;
|
|
PVOID SavedContext;
|
|
};
|
|
PAFD_SWITCH_CONTEXT LocalContext;
|
|
LIST_ENTRY IrpList;
|
|
ULONG SavedContextLength;
|
|
ULONG RequestId;
|
|
ULONG SelectEventsActive;
|
|
NTSTATUS CtxTransferStatus;
|
|
BOOLEAN ImplicitDup; // Dup-ed into another process without
|
|
// explicit request from the applicaiton
|
|
|
|
} SanEndp;
|
|
|
|
} Common;
|
|
|
|
|
|
volatile PVOID Context;
|
|
ULONG ContextLength;
|
|
|
|
ULONG LocalAddressLength;
|
|
PTRANSPORT_ADDRESS LocalAddress;
|
|
|
|
ULONG DisconnectMode;
|
|
LONG OutstandingIrpCount;
|
|
|
|
HANDLE AddressHandle; // TDI transport address object
|
|
PAFD_TRANSPORT_INFO TransportInfo;
|
|
|
|
LIST_ENTRY RoutingNotifications; // For non-blocking sockets
|
|
LIST_ENTRY RequestList; // For misc requests
|
|
|
|
LIST_ENTRY GlobalEndpointListEntry;
|
|
AFD_WORK_ITEM WorkItem;
|
|
PIRP Irp; // AcceptEx or TransmitPackets IRP
|
|
//
|
|
// Non-zero when state change such as binding, accepting,
|
|
// conntecting, and transmit file are in progress.
|
|
LONG StateChangeInProgress;
|
|
|
|
//
|
|
// EventSelect info.
|
|
//
|
|
|
|
ULONG EventsEnabled;
|
|
NTSTATUS EventStatus[AFD_NUM_POLL_EVENTS]; // Currently 13 events
|
|
PKEVENT EventObject;
|
|
|
|
//
|
|
// Socket grouping.
|
|
//
|
|
|
|
LONG GroupID;
|
|
AFD_GROUP_TYPE GroupType;
|
|
LIST_ENTRY ConstrainedEndpointListEntry;
|
|
|
|
//
|
|
// Debug stuff.
|
|
//
|
|
|
|
#if REFERENCE_DEBUG
|
|
LONG CurrentTimeHigh;
|
|
LONG CurrentReferenceSlot;
|
|
AFD_REFERENCE_DEBUG ReferenceDebug[AFD_MAX_REF];
|
|
#endif
|
|
|
|
#if DBG
|
|
LIST_ENTRY OutstandingIrpListHead;
|
|
LONG ObReferenceBias;
|
|
#endif
|
|
};
|
|
// Fields AFD_ENDPOINT structure are protected by the embedded
|
|
// spinlock, except:
|
|
// AdminAccessGranted, TdiServiceFlags, __f, OwningProcess, TransportInfo,
|
|
// IoCompletionPort, IoCompletionEvent, AddressHandle which are
|
|
// set once and are not touched until reference count goes to 0
|
|
// Type, State, LocalAddress, LocalAddressLength, AddressFileObject,
|
|
// AddressDeviceObject, AddressHandle, TdiServiceFlags are protected by the
|
|
// StateChange interlocked mutex (AddressHandle and TdiServiceFlags can only be set
|
|
// once under the mutex).
|
|
// AddressFileObject,AddressDeviceObject,LocalAddress, LocalAddressLength can only be
|
|
// reset to NULL under the endpoint spinlock in case of TransmitFile (REUSE) on
|
|
// accepting endpoint (StateChange protection still applies).
|
|
// Counters: ReferenceCount, FailedConnectionAdds, TdiAcceptPendingCount,
|
|
// Sequence, PendingRequests, SelectEventsActive are accessed with interlocked operations.
|
|
// Context, ContextLength, SwitchContext, SavedContext, LocalContext,
|
|
// SavedContextLength, RemoveSocketAddressOffset, RemoteSocketAddressLength
|
|
// are protected by the context lock (AfdLockContext).
|
|
|
|
|
|
typedef struct _AFD_POLL_ENDPOINT_INFO {
|
|
PAFD_ENDPOINT Endpoint;
|
|
PFILE_OBJECT FileObject;
|
|
HANDLE Handle;
|
|
ULONG PollEvents;
|
|
} AFD_POLL_ENDPOINT_INFO, *PAFD_POLL_ENDPOINT_INFO;
|
|
|
|
typedef struct _AFD_POLL_INFO_INTERNAL {
|
|
LIST_ENTRY PollListEntry;
|
|
ULONG NumberOfEndpoints;
|
|
PIRP Irp;
|
|
union {
|
|
struct {
|
|
KDPC Dpc;
|
|
KTIMER Timer;
|
|
};
|
|
KAPC Apc; // for SAN
|
|
};
|
|
BOOLEAN Unique;
|
|
BOOLEAN TimerStarted;
|
|
BOOLEAN SanPoll;
|
|
AFD_POLL_ENDPOINT_INFO EndpointInfo[1];
|
|
} AFD_POLL_INFO_INTERNAL, *PAFD_POLL_INFO_INTERNAL;
|
|
|
|
//
|
|
// A couple of useful manifests that make code more readable.
|
|
//
|
|
|
|
#define ReceiveDatagramIrpListHead Common.Datagram.ReceiveIrpListHead
|
|
#define PeekDatagramIrpListHead Common.Datagram.PeekIrpListHead
|
|
#define ReceiveDatagramBufferListHead Common.Datagram.ReceiveBufferListHead
|
|
#define DgBufferredReceiveCount Common.Datagram.BufferredReceiveCount
|
|
#define DgBufferredReceiveBytes Common.Datagram.BufferredReceiveBytes
|
|
#define DgBufferredSendBytes Common.Datagram.BufferredSendBytes
|
|
|
|
#define AFD_CONNECTION_FROM_ENDPOINT( endpoint ) ( \
|
|
(((endpoint)->Type & AfdBlockTypeVcConnecting)==AfdBlockTypeVcConnecting) \
|
|
? (endpoint)->Common.VirtualCircuit.Connection \
|
|
: (PAFD_CONNECTION)NULL \
|
|
)
|
|
|
|
//
|
|
// A structure which describes buffers used by AFD to perform bufferring
|
|
// for TDI providers which do not perform internal bufferring.
|
|
// It is also used in other code path as buffer descriptors.
|
|
//
|
|
typedef struct _AFD_BUFFER_HEADER AFD_BUFFER_HEADER, *PAFD_BUFFER_HEADER;
|
|
|
|
#define _AFD_BUFFER_HEADER_ \
|
|
union { \
|
|
TDI_CONNECTION_INFORMATION TdiInfo; /*holds info for TDI requests */ \
|
|
/*with remote address */ \
|
|
struct { \
|
|
union { /* Links */ \
|
|
struct { \
|
|
union { \
|
|
SINGLE_LIST_ENTRY SList; /* for buffer lookaside lists */ \
|
|
PAFD_BUFFER_HEADER Next; /* for transmit packet lists */ \
|
|
}; \
|
|
PFILE_OBJECT FileObject; /* for cached file in transmit file */ \
|
|
}; \
|
|
LIST_ENTRY BufferListEntry; /* for endpoint/connection lists */ \
|
|
}; \
|
|
union { \
|
|
struct { \
|
|
ULONG DataOffset; /* offset in buffer to start of unread data*/\
|
|
union { \
|
|
ULONG DatagramFlags;/* flags for datagrams with control info */\
|
|
LONG RefCount; /* Permit partial copy outside the lock*/ \
|
|
}; \
|
|
}; \
|
|
LARGE_INTEGER FileOffset;/* data offset from the start of the file */\
|
|
}; \
|
|
UCHAR _Test; /* used to test relative field pos in the union */ \
|
|
}; \
|
|
}; \
|
|
union { \
|
|
PVOID Context; /* stores context info (endp/conn/etc)*/ \
|
|
NTSTATUS Status;/* stores status of completed operation */ \
|
|
}; \
|
|
PMDL Mdl; /* pointer to an MDL describing the buffer*/ \
|
|
ULONG DataLength; /* actual data in the buffer */ \
|
|
ULONG BufferLength; /* amount of space allocated for the buffer */\
|
|
union { \
|
|
struct { \
|
|
/* Flags that describe data in the buffer */ \
|
|
UCHAR ExpeditedData:1, /* The buffer contains expedited data*/ \
|
|
:3, \
|
|
PartialMessage:1,/* This is a partial message*/ \
|
|
:3; \
|
|
/* Flags that keep allocation information */ \
|
|
UCHAR NdisPacket:1, /* Context is a packet to return to NDIS/TDI*/\
|
|
:3, \
|
|
Placement:2, /* Relative placement of the pieces */ \
|
|
AlignmentAdjusted:1, /* MM block alignment was adjusted to */ \
|
|
/* meet AFD buffer alignment requirement */ \
|
|
Lookaside:1; /* Poped from Slist (no quota charge)*/ \
|
|
}; \
|
|
USHORT Flags; \
|
|
}; \
|
|
USHORT AllocatedAddressLength/* length allocated for address */ \
|
|
|
|
|
|
struct _AFD_BUFFER_HEADER {
|
|
_AFD_BUFFER_HEADER_ ;
|
|
};
|
|
|
|
//
|
|
// The buffer management code makes the following assumptions
|
|
// about the union at the top the buffer header so that list
|
|
// links and DataOffset field are not in conflict with
|
|
// RemoteAddress fields of the TDI_CONNECTION_INFORMATION.
|
|
//
|
|
C_ASSERT (FIELD_OFFSET (AFD_BUFFER_HEADER, TdiInfo.RemoteAddress) >=
|
|
FIELD_OFFSET (AFD_BUFFER_HEADER, _Test));
|
|
C_ASSERT (FIELD_OFFSET (AFD_BUFFER_HEADER, TdiInfo.RemoteAddressLength)>=
|
|
FIELD_OFFSET (AFD_BUFFER_HEADER, _Test));
|
|
|
|
C_ASSERT(FIELD_OFFSET (AFD_BUFFER_HEADER, AllocatedAddressLength)==
|
|
FIELD_OFFSET(AFD_BUFFER_HEADER, Flags)+sizeof (USHORT));
|
|
|
|
typedef struct AFD_BUFFER_TAG {
|
|
union {
|
|
struct {
|
|
_AFD_BUFFER_HEADER_; // easy access to individual members
|
|
};
|
|
AFD_BUFFER_HEADER Header; // access to the header as a whole
|
|
};
|
|
#if DBG
|
|
PVOID Caller;
|
|
PVOID CallersCaller;
|
|
#endif
|
|
// UCHAR Address[]; // address of datagram sender
|
|
} AFD_BUFFER_TAG, *PAFD_BUFFER_TAG;
|
|
|
|
struct _AFD_BUFFER {
|
|
union {
|
|
struct {
|
|
_AFD_BUFFER_HEADER_; // easy access to individual members
|
|
};
|
|
AFD_BUFFER_HEADER Header; // access to the header as a whole
|
|
};
|
|
PIRP Irp; // pointer to the IRP associated w/the buffer
|
|
PVOID Buffer; // a pointer to the actual data buffer
|
|
|
|
#if DBG
|
|
LIST_ENTRY DebugListEntry;
|
|
PVOID Caller;
|
|
PVOID CallersCaller;
|
|
#endif
|
|
// IRP Irp; // the IRP follows this structure
|
|
// MDL Mdl; // the MDL follows the IRP
|
|
// UCHAR Address[]; // address of datagram sender
|
|
// UCHAR Buffer[BufferLength]; // the actual data buffer is last
|
|
} ;
|
|
|
|
//
|
|
// Placement of pieces that comprise the AFD_BUFFER.
|
|
// We have four pieces: header, IRP, MDL, data buffer
|
|
// and use 2 bits to encode each.
|
|
// We need to save the first piece so we know where the memory block
|
|
// starts when we need to return it to the memory manager.
|
|
//
|
|
enum {
|
|
AFD_PLACEMENT_HDR =0,
|
|
AFD_PLACEMENT_IRP =1,
|
|
AFD_PLACEMENT_MDL =2,
|
|
AFD_PLACEMENT_BUFFER =3,
|
|
AFD_PLACEMENT_HDR_IRP =(AFD_PLACEMENT_HDR|(AFD_PLACEMENT_IRP<<2)),
|
|
AFD_PLACEMENT_HDR_MDL =(AFD_PLACEMENT_HDR|(AFD_PLACEMENT_MDL<<2)),
|
|
AFD_PLACEMENT_IRP_MDL =(AFD_PLACEMENT_IRP|(AFD_PLACEMENT_MDL<<2)),
|
|
AFD_PLACEMENT_HDR_IRP_MDL=(AFD_PLACEMENT_HDR|(AFD_PLACEMENT_IRP<<2)|(AFD_PLACEMENT_MDL<<4))
|
|
};
|
|
|
|
|
|
//
|
|
// ALIGN_DOWN_A aligns to given alignment requirement
|
|
// (as opposed to the type in the original ALIGN_DOWN macro)
|
|
//
|
|
#define ALIGN_DOWN_A(length,alignment) \
|
|
(((ULONG)(length)) & ~ ((alignment)-1))
|
|
|
|
//
|
|
// ALIGN_DOWN_A for pointers.
|
|
//
|
|
#define ALIGN_DOWN_A_POINTER(address,alignment) \
|
|
((PVOID)(((ULONG_PTR)(address)) & ~ ((ULONG_PTR)(alignment)-1)))
|
|
|
|
|
|
//
|
|
// ALIGN_UP_A aligns to given alignment requirement
|
|
// (as opposed to the type in the original ALIGN_UP macro)
|
|
//
|
|
#define ALIGN_UP_A(length,alignment) \
|
|
((((ULONG)(length)) + (alignment)-1) & ~ ((alignment)-1))
|
|
|
|
//
|
|
// ALIGN_UP_A for pointers.
|
|
//
|
|
#define ALIGN_UP_A_POINTER(address,alignment) \
|
|
ALIGN_DOWN_A_POINTER(((ULONG_PTR)(address) + alignment-1), alignment)
|
|
|
|
//
|
|
// ALIGN_UP_TO_TYPE aligns size to make sure it meets
|
|
// the type alignment requirement
|
|
//
|
|
#define ALIGN_UP_TO_TYPE(length,type) \
|
|
ALIGN_UP_A(length,TYPE_ALIGNMENT(type))
|
|
|
|
//
|
|
// ALIGN_UP_TO_TYPE for pointers.
|
|
//
|
|
#define ALIGN_UP_TO_TYPE_POINTER(address,type) \
|
|
ALIGN_UP_A_POINTER(address,TYPE_ALIGNMENT(type))
|
|
|
|
#if DBG
|
|
|
|
#define IS_VALID_AFD_BUFFER(b) ( \
|
|
((b)->Placement==AFD_PLACEMENT_HDR) \
|
|
? ((PUCHAR)b<(PUCHAR)(b)->Buffer && (PUCHAR)b<(PUCHAR)(b)->Mdl && (PUCHAR)b<(PUCHAR)(b)->Irp) \
|
|
: (((b)->Placement==AFD_PLACEMENT_MDL) \
|
|
? ((PUCHAR)(b)->Mdl<(PUCHAR)(b)->Buffer && (PUCHAR)(b)->Mdl<(PUCHAR)b && (PUCHAR)(b)->Mdl<(PUCHAR)(b)->Irp) \
|
|
: ((b->Placement==AFD_PLACEMENT_IRP) \
|
|
? ((PUCHAR)(b)->Irp<(PUCHAR)(b)->Buffer && (PUCHAR)(b)->Irp<(PUCHAR)b && (PUCHAR)(b)->Irp<(PUCHAR)(b)->Mdl) \
|
|
: ((PUCHAR)(b)->Buffer<(PUCHAR)(b)->Irp && (PUCHAR)(b)->Buffer<(PUCHAR)b && (PUCHAR)(b)->Buffer<(PUCHAR)(b)->Mdl)) \
|
|
) \
|
|
) \
|
|
|
|
#endif
|
|
|
|
|
|
//
|
|
// Pointer to an IRP cleanup routine. This is used as a parameter to
|
|
// AfdCompleteIrpList().
|
|
//
|
|
|
|
typedef
|
|
BOOLEAN
|
|
(NTAPI * PAFD_IRP_CLEANUP_ROUTINE)(
|
|
IN PIRP Irp
|
|
);
|
|
|
|
//
|
|
// Debug statistics.
|
|
//
|
|
|
|
typedef struct _AFD_QUOTA_STATS {
|
|
LARGE_INTEGER Charged;
|
|
LARGE_INTEGER Returned;
|
|
} AFD_QUOTA_STATS;
|
|
|
|
typedef struct _AFD_HANDLE_STATS {
|
|
LONG AddrOpened;
|
|
LONG AddrClosed;
|
|
LONG AddrRef;
|
|
LONG AddrDeref;
|
|
LONG ConnOpened;
|
|
LONG ConnClosed;
|
|
LONG ConnRef;
|
|
LONG ConnDeref;
|
|
LONG FileRef;
|
|
LONG FileDeref;
|
|
} AFD_HANDLE_STATS;
|
|
|
|
typedef struct _AFD_QUEUE_STATS {
|
|
LONG AfdWorkItemsQueued;
|
|
LONG ExWorkItemsQueued;
|
|
LONG WorkerEnter;
|
|
LONG WorkerLeave;
|
|
LONG AfdWorkItemsProcessed;
|
|
PETHREAD AfdWorkerThread;
|
|
} AFD_QUEUE_STATS;
|
|
|
|
typedef struct _AFD_CONNECTION_STATS {
|
|
LONG ConnectedReferencesAdded;
|
|
LONG ConnectedReferencesDeleted;
|
|
LONG GracefulDisconnectsInitiated;
|
|
LONG GracefulDisconnectsCompleted;
|
|
LONG GracefulDisconnectIndications;
|
|
LONG AbortiveDisconnectsInitiated;
|
|
LONG AbortiveDisconnectsCompleted;
|
|
LONG AbortiveDisconnectIndications;
|
|
LONG ConnectionIndications;
|
|
LONG ConnectionsDropped;
|
|
LONG ConnectionsAccepted;
|
|
LONG ConnectionsPreaccepted;
|
|
LONG ConnectionsReused;
|
|
LONG EndpointsReused;
|
|
} AFD_CONNECTION_STATS;
|
|
|
|
//
|
|
// Global data. Resouces and lookaside list descriptors
|
|
// cannot be statically allocated, as they need to ALWAYS be nonpageable,
|
|
// even when the entire driver is paged out.
|
|
// Alignment table is variable-size and also cannot be statically allocated.
|
|
//
|
|
|
|
enum {
|
|
AFD_LARGE_BUFFER_LIST=0,
|
|
AFD_MEDIUM_BUFFER_LIST,
|
|
AFD_SMALL_BUFFER_LIST,
|
|
AFD_BUFFER_TAG_LIST,
|
|
AFD_TP_INFO_LIST,
|
|
AFD_REMOTE_ADDR_LIST,
|
|
AFD_NUM_LOOKASIDE_LISTS
|
|
} AFD_LOOKASIDE_LISTS_INDEX;
|
|
|
|
typedef struct _AFD_GLOBAL_DATA {
|
|
ERESOURCE Resource;
|
|
NPAGED_LOOKASIDE_LIST List[AFD_NUM_LOOKASIDE_LISTS];
|
|
#define LargeBufferList List[AFD_LARGE_BUFFER_LIST]
|
|
#define MediumBufferList List[AFD_MEDIUM_BUFFER_LIST]
|
|
#define SmallBufferList List[AFD_SMALL_BUFFER_LIST]
|
|
#define BufferTagList List[AFD_BUFFER_TAG_LIST]
|
|
#define TpInfoList List[AFD_TP_INFO_LIST]
|
|
#define RemoteAddrList List[AFD_REMOTE_ADDR_LIST]
|
|
LONG TrimFlags;
|
|
KTIMER Timer;
|
|
KDPC Dpc;
|
|
UCHAR BufferAlignmentTable[ANYSIZE_ARRAY];
|
|
} AFD_GLOBAL_DATA, *PAFD_GLOBAL_DATA;
|
|
|
|
//
|
|
// Context structure for misc requests pended in AFD.
|
|
//
|
|
typedef struct _AFD_REQUEST_CONTEXT AFD_REQUEST_CONTEXT, *PAFD_REQUEST_CONTEXT;
|
|
|
|
//
|
|
// The routine is called after request is removed from endpoint list
|
|
// for cleanup purposes
|
|
//
|
|
typedef BOOLEAN (* PAFD_REQUEST_CLEANUP) (
|
|
PAFD_ENDPOINT Endpoint,
|
|
PAFD_REQUEST_CONTEXT NotifyCtx
|
|
);
|
|
|
|
//
|
|
// This structure has to no more 16 bytes long so we can
|
|
// reuse IrpSp->Parameters for it.
|
|
//
|
|
struct _AFD_REQUEST_CONTEXT {
|
|
LIST_ENTRY EndpointListLink; // Link in endpoint list
|
|
PAFD_REQUEST_CLEANUP CleanupRoutine; // Routine to call to cancel
|
|
PVOID Context; // Request dependent context
|
|
// (PIRP)
|
|
};
|
|
|
|
//
|
|
// We use list entry fields to synchronize completion with cleanup/cancel
|
|
// routine assuming that as long as the entry is in the list
|
|
// both Flink and Blink fields cannot be NULL. (using these
|
|
// fields for synchronization allows us to cut down on
|
|
// cancel spinlock usage)
|
|
//
|
|
|
|
#define AfdEnqueueRequest(Endpoint,Request) \
|
|
ExInterlockedInsertTailList(&(Endpoint)->RequestList, \
|
|
&(Request)->EndpointListLink, \
|
|
&(Endpoint)->SpinLock)
|
|
|
|
|
|
#define AfdIsRequestInQueue(Request) \
|
|
((Request)->EndpointListLink.Flink!=NULL)
|
|
|
|
#define AfdMarkRequestCompleted(Request) \
|
|
(Request)->EndpointListLink.Blink = NULL
|
|
|
|
#define AfdIsRequestCompleted(Request) \
|
|
((Request)->EndpointListLink.Blink==NULL)
|
|
|
|
|
|
typedef struct _ROUTING_NOTIFY {
|
|
LIST_ENTRY NotifyListLink;
|
|
PIRP NotifyIrp;
|
|
PVOID NotifyContext;
|
|
} ROUTING_NOTIFY, *PROUTING_NOTIFY;
|
|
|
|
typedef struct _AFD_ADDRESS_ENTRY {
|
|
LIST_ENTRY AddressListLink;
|
|
UNICODE_STRING DeviceName;
|
|
TA_ADDRESS Address;
|
|
} AFD_ADDRESS_ENTRY, *PAFD_ADDRESS_ENTRY;
|
|
|
|
typedef struct _AFD_ADDRESS_CHANGE {
|
|
LIST_ENTRY ChangeListLink;
|
|
union {
|
|
PAFD_ENDPOINT Endpoint;
|
|
PIRP Irp;
|
|
};
|
|
USHORT AddressType;
|
|
BOOLEAN NonBlocking;
|
|
} AFD_ADDRESS_CHANGE, *PAFD_ADDRESS_CHANGE;
|
|
|
|
|
|
typedef
|
|
NTSTATUS
|
|
(* PAFD_IMMEDIATE_CALL) (
|
|
IN PFILE_OBJECT FileObject,
|
|
IN ULONG IoctlCode,
|
|
IN KPROCESSOR_MODE RequestorMode,
|
|
IN PVOID InputBuffer,
|
|
IN ULONG InputBufferLength,
|
|
IN PVOID OutputBuffer,
|
|
IN ULONG OutputBufferLength,
|
|
OUT PULONG_PTR Information
|
|
);
|
|
|
|
typedef
|
|
NTSTATUS
|
|
(FASTCALL * PAFD_IRP_CALL) (
|
|
IN PIRP Irp,
|
|
IN PIO_STACK_LOCATION IrpSp
|
|
);
|
|
|
|
typedef struct _AFD_TRANSMIT_PACKETS_ELEMENT {
|
|
#define TP_MDL 0x80000000
|
|
#define TP_COMBINE 0x40000000
|
|
ULONG Flags;
|
|
ULONG Length;
|
|
union {
|
|
struct {
|
|
LARGE_INTEGER FileOffset;
|
|
PFILE_OBJECT FileObject;
|
|
};
|
|
struct {
|
|
PVOID Buffer;
|
|
PMDL Mdl;
|
|
};
|
|
};
|
|
} AFD_TRANSMIT_PACKETS_ELEMENT, *PAFD_TRANSMIT_PACKETS_ELEMENT;
|
|
|
|
//
|
|
// Structure to keep track of transmit packets request
|
|
//
|
|
struct _AFD_TPACKETS_INFO_INTERNAL {
|
|
union {
|
|
SINGLE_LIST_ENTRY SListEntry; // Link on S-List
|
|
PFILE_OBJECT TdiFileObject; // Tdi objects (sending to)
|
|
};
|
|
|
|
PDEVICE_OBJECT TdiDeviceObject;
|
|
|
|
PMDL HeadMdl; // Ready to send chain
|
|
PMDL *TailMdl;
|
|
|
|
PAFD_BUFFER_HEADER HeadPd; // Corresponding packet chain
|
|
PAFD_BUFFER_HEADER *TailPd;
|
|
|
|
PIRP ReadIrp; // Irp used for file reads.
|
|
PAFD_TRANSMIT_PACKETS_ELEMENT
|
|
ElementArray; // Packet array
|
|
|
|
ULONG NextElement; // Next element to send.
|
|
ULONG ElementCount; // Total number of elements in the array
|
|
|
|
ULONG RemainingPkts; // Number of packets remaining to be sent.
|
|
USHORT NumSendIrps; // Actual number of send IRPs
|
|
BOOLEAN ArrayAllocated; // Element array is allocated (not built-in).
|
|
BOOLEAN PdNeedsPps; // Packet descriptor being built needs post-
|
|
// processing after completion.
|
|
ULONG PdLength; // Currently accumulated send length
|
|
ULONG SendPacketLength; // Maximum length of the packet
|
|
// APC/Work item for worker scheduling
|
|
union {
|
|
KAPC Apc;
|
|
WORK_QUEUE_ITEM WorkItem;
|
|
};
|
|
|
|
#if REFERENCE_DEBUG
|
|
LONG CurrentTimeHigh;
|
|
LONG CurrentReferenceSlot;
|
|
AFD_REFERENCE_DEBUG ReferenceDebug[AFD_MAX_REF];
|
|
#endif
|
|
#if AFD_PERF_DBG
|
|
LONG WorkersExecuted;
|
|
#endif
|
|
#define AFD_TP_MIN_SEND_IRPS 2 // Need at least two to keep transport busy
|
|
#define AFD_TP_MAX_SEND_IRPS 8 // Max is based on current flags layout below
|
|
PIRP SendIrp[AFD_TP_MAX_SEND_IRPS];
|
|
// ElementArray
|
|
// SendIrp1
|
|
// SendIrp2
|
|
};
|
|
|
|
//
|
|
// Structure maintained in driver context of the TPackets IRP
|
|
//
|
|
typedef struct _AFD_TPACKETS_IRP_CTX AFD_TPACKETS_IRP_CTX, *PAFD_TPACKETS_IRP_CTX;
|
|
#define AFD_GET_TPIC(_i) ((PAFD_TPACKETS_IRP_CTX)&(_i)->Tail.Overlay.DriverContext)
|
|
struct _AFD_TPACKETS_IRP_CTX {
|
|
PAFD_TPACKETS_IRP_CTX Next; // Next TPackets IRP in the list
|
|
LONG Flags; // Application flags.
|
|
volatile LONG ReferenceCount; // IRP reference count
|
|
volatile LONG StateFlags; // TPackets state flags
|
|
};
|
|
|
|
#define AFD_GET_TPIRP(_i) CONTAINING_RECORD(_i,IRP,Tail.Overlay.DriverContext)
|
|
|
|
#define AFD_TP_ABORT_PENDING 0x00000001 // Request is being aborted
|
|
#define AFD_TP_WORKER_SCHEDULED 0x00000002 // Worker is scheduled or active
|
|
#define AFD_TP_SENDS_POSTED 0x00000010 // All sends have been posted.
|
|
#define AFD_TP_QUEUED 0x00000020 // Irp is in the queue.
|
|
#define AFD_TP_SEND 0x00000100 // This is a plain send in TP queue.
|
|
#define AFD_TP_AFD_SEND 0x00000200 // This is AFD send IRP.
|
|
#ifdef TDI_SERVICE_SEND_AND_DISCONNECT
|
|
#define AFD_TP_SEND_AND_DISCONNECT 0x00000400 // S&D is enabled
|
|
#endif // TDI_SERVICE_SEND_AND_DISCONNECT
|
|
|
|
#define AFD_TP_READ_CALL_PENDING 0x00001000 // MDL_READ call is imminent or in progress on ReadIrp
|
|
#define AFD_TP_READ_COMP_PENDING 0x00002000 // Read completion is expected on ReadIrp
|
|
#define AFD_TP_READ_BUSY (AFD_TP_READ_CALL_PENDING|AFD_TP_READ_COMP_PENDING)
|
|
|
|
// TDI_SEND call is imminent or in progress on send Irp i
|
|
#define AFD_TP_SEND_CALL_PENDING(i) (0x00010000<<((i)*2))
|
|
// Send completion is expected on send Irp i
|
|
#define AFD_TP_SEND_COMP_PENDING(i) (0x00020000<<((i)*2))
|
|
#define AFD_TP_SEND_BUSY(i) (0x00030000<<((i)*2))
|
|
#define AFD_TP_SEND_MASK (0x55550000)
|
|
|
|
|
|
#if DBG
|
|
//
|
|
// N.B. This structure MUST be alligned on memory allocation boundary
|
|
//
|
|
|
|
typedef struct DECLSPEC_ALIGN(MEMORY_ALLOCATION_ALIGNMENT) _AFD_POOL_HEADER {
|
|
SIZE_T Size;
|
|
PCHAR FileName;
|
|
ULONG LineNumber;
|
|
LONG InUse;
|
|
} AFD_POOL_HEADER, *PAFD_POOL_HEADER;
|
|
#define AFD_POOL_OVERHEAD (sizeof(AFD_POOL_HEADER))
|
|
#else
|
|
#define AFD_POOL_OVERHEAD 0
|
|
#endif
|
|
|
|
//
|
|
// This macro verifies that the 32-bit mapping of the structures in 64-bit
|
|
// compiler match original 32-bit structures. Note, that the verification is
|
|
// performed when this file is compiled by 32 bit compiler, but
|
|
// the actual structures are used by 64 bit code.
|
|
//
|
|
|
|
#ifdef _WIN64
|
|
#define AFD_CHECK32on64(_str,_fld)
|
|
#define AFD_MAX_NATURAL_ALIGNMENT32 sizeof(ULONG)
|
|
#else
|
|
#define AFD_CHECK32on64(_str,_fld) \
|
|
C_ASSERT (FIELD_OFFSET (_str,_fld)==FIELD_OFFSET(_str##32,_fld))
|
|
#endif
|
|
|
|
//
|
|
// Structures for mapping IOCTL parameters for 32-bit clients on 64-bit
|
|
// platform.
|
|
//
|
|
typedef UNALIGNED struct _WSABUF32 {
|
|
ULONG len;
|
|
AFD_POINTER_32 buf;
|
|
} WSABUF32, *LPWSABUF32;
|
|
AFD_CHECK32on64(WSABUF,len);
|
|
AFD_CHECK32on64(WSABUF,buf);
|
|
|
|
typedef UNALIGNED struct _QualityOfService32 {
|
|
FLOWSPEC SendingFlowspec; /* the flow spec for data sending */
|
|
FLOWSPEC ReceivingFlowspec; /* the flow spec for data receiving */
|
|
WSABUF32 ProviderSpecific; /* additional provider specific stuff */
|
|
} QOS32, * LPQOS32;
|
|
AFD_CHECK32on64(QOS,SendingFlowspec);
|
|
AFD_CHECK32on64(QOS,ReceivingFlowspec);
|
|
AFD_CHECK32on64(QOS,ProviderSpecific);
|
|
|
|
typedef UNALIGNED struct _AFD_ACCEPT_INFO32 {
|
|
BOOLEAN SanActive;
|
|
LONG Sequence;
|
|
VOID * POINTER_32 AcceptHandle;
|
|
} AFD_ACCEPT_INFO32, *PAFD_ACCEPT_INFO32;
|
|
AFD_CHECK32on64(AFD_ACCEPT_INFO,SanActive);
|
|
AFD_CHECK32on64(AFD_ACCEPT_INFO,Sequence);
|
|
AFD_CHECK32on64(AFD_ACCEPT_INFO,AcceptHandle);
|
|
|
|
typedef UNALIGNED struct _AFD_SUPER_ACCEPT_INFO32 {
|
|
BOOLEAN SanActive;
|
|
BOOLEAN FixAddressAlignment;
|
|
VOID * POINTER_32 AcceptHandle;
|
|
ULONG ReceiveDataLength;
|
|
ULONG LocalAddressLength;
|
|
ULONG RemoteAddressLength;
|
|
} AFD_SUPER_ACCEPT_INFO32, *PAFD_SUPER_ACCEPT_INFO32;
|
|
AFD_CHECK32on64(AFD_SUPER_ACCEPT_INFO,SanActive);
|
|
AFD_CHECK32on64(AFD_SUPER_ACCEPT_INFO,FixAddressAlignment);
|
|
AFD_CHECK32on64(AFD_SUPER_ACCEPT_INFO,AcceptHandle);
|
|
AFD_CHECK32on64(AFD_SUPER_ACCEPT_INFO,ReceiveDataLength);
|
|
AFD_CHECK32on64(AFD_SUPER_ACCEPT_INFO,LocalAddressLength);
|
|
AFD_CHECK32on64(AFD_SUPER_ACCEPT_INFO,RemoteAddressLength);
|
|
|
|
typedef UNALIGNED struct _AFD_POLL_HANDLE_INFO32 {
|
|
VOID * POINTER_32 Handle;
|
|
ULONG PollEvents;
|
|
NTSTATUS Status;
|
|
} AFD_POLL_HANDLE_INFO32, *PAFD_POLL_HANDLE_INFO32;
|
|
AFD_CHECK32on64(AFD_POLL_HANDLE_INFO,Handle);
|
|
AFD_CHECK32on64(AFD_POLL_HANDLE_INFO,PollEvents);
|
|
AFD_CHECK32on64(AFD_POLL_HANDLE_INFO,Status);
|
|
|
|
typedef UNALIGNED struct _AFD_POLL_INFO32 {
|
|
LARGE_INTEGER Timeout;
|
|
ULONG NumberOfHandles;
|
|
BOOLEAN Unique;
|
|
AFD_POLL_HANDLE_INFO32 Handles[1];
|
|
} AFD_POLL_INFO32, *PAFD_POLL_INFO32;
|
|
AFD_CHECK32on64(AFD_POLL_INFO,Timeout);
|
|
AFD_CHECK32on64(AFD_POLL_INFO,NumberOfHandles);
|
|
AFD_CHECK32on64(AFD_POLL_INFO,Unique);
|
|
AFD_CHECK32on64(AFD_POLL_INFO,Handles);
|
|
|
|
typedef UNALIGNED struct _AFD_HANDLE_INFO632 {
|
|
VOID * POINTER_32 TdiAddressHandle;
|
|
VOID * POINTER_32 TdiConnectionHandle;
|
|
} AFD_HANDLE_INFO32, *PAFD_HANDLE_INFO32;
|
|
AFD_CHECK32on64(AFD_HANDLE_INFO,TdiAddressHandle);
|
|
AFD_CHECK32on64(AFD_HANDLE_INFO,TdiConnectionHandle);
|
|
|
|
typedef UNALIGNED struct _AFD_TRANSMIT_FILE_INFO32 {
|
|
LARGE_INTEGER Offset;
|
|
LARGE_INTEGER WriteLength;
|
|
ULONG SendPacketLength;
|
|
VOID * POINTER_32 FileHandle;
|
|
AFD_POINTER_32 Head;
|
|
ULONG HeadLength;
|
|
AFD_POINTER_32 Tail;
|
|
ULONG TailLength;
|
|
ULONG Flags;
|
|
} AFD_TRANSMIT_FILE_INFO32, *PAFD_TRANSMIT_FILE_INFO32;
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,Offset);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,WriteLength);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,SendPacketLength);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,FileHandle);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,Head);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,HeadLength);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,Tail);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,TailLength);
|
|
AFD_CHECK32on64(AFD_TRANSMIT_FILE_INFO,Flags);
|
|
|
|
typedef UNALIGNED struct _AFD_SEND_INFO32 {
|
|
AFD_POINTER_32 BufferArray;
|
|
ULONG BufferCount;
|
|
ULONG AfdFlags;
|
|
ULONG TdiFlags;
|
|
} AFD_SEND_INFO32, *PAFD_SEND_INFO32;
|
|
AFD_CHECK32on64(AFD_SEND_INFO,BufferArray);
|
|
AFD_CHECK32on64(AFD_SEND_INFO,BufferCount);
|
|
AFD_CHECK32on64(AFD_SEND_INFO,AfdFlags);
|
|
AFD_CHECK32on64(AFD_SEND_INFO,TdiFlags);
|
|
|
|
typedef UNALIGNED struct _TDI_REQUEST32 {
|
|
union {
|
|
VOID * POINTER_32 AddressHandle;
|
|
VOID * POINTER_32 ConnectionContext;
|
|
VOID * POINTER_32 ControlChannel;
|
|
} Handle;
|
|
|
|
VOID * POINTER_32 RequestNotifyObject;
|
|
VOID * POINTER_32 RequestContext;
|
|
TDI_STATUS TdiStatus;
|
|
} TDI_REQUEST32, *PTDI_REQUEST32;
|
|
AFD_CHECK32on64(TDI_REQUEST,Handle);
|
|
AFD_CHECK32on64(TDI_REQUEST,RequestNotifyObject);
|
|
AFD_CHECK32on64(TDI_REQUEST,RequestContext);
|
|
AFD_CHECK32on64(TDI_REQUEST,TdiStatus);
|
|
|
|
typedef UNALIGNED struct _TDI_CONNECTION_INFORMATION32 {
|
|
LONG UserDataLength; // length of user data buffer
|
|
AFD_POINTER_32 UserData; // pointer to user data buffer
|
|
LONG OptionsLength; // length of follwoing buffer
|
|
AFD_POINTER_32 Options; // pointer to buffer containing options
|
|
LONG RemoteAddressLength; // length of following buffer
|
|
AFD_POINTER_32 RemoteAddress; // buffer containing the remote address
|
|
} TDI_CONNECTION_INFORMATION32, *PTDI_CONNECTION_INFORMATION32;
|
|
AFD_CHECK32on64(TDI_CONNECTION_INFORMATION,UserDataLength);
|
|
AFD_CHECK32on64(TDI_CONNECTION_INFORMATION,UserData);
|
|
AFD_CHECK32on64(TDI_CONNECTION_INFORMATION,OptionsLength);
|
|
AFD_CHECK32on64(TDI_CONNECTION_INFORMATION,Options);
|
|
AFD_CHECK32on64(TDI_CONNECTION_INFORMATION,RemoteAddressLength);
|
|
AFD_CHECK32on64(TDI_CONNECTION_INFORMATION,RemoteAddress);
|
|
|
|
typedef UNALIGNED struct _TDI_REQUEST_SEND_DATAGRAM32 {
|
|
TDI_REQUEST32 Request;
|
|
AFD_POINTER_32 SendDatagramInformation;
|
|
} TDI_REQUEST_SEND_DATAGRAM32, *PTDI_REQUEST_SEND_DATAGRAM32;
|
|
AFD_CHECK32on64(TDI_REQUEST_SEND_DATAGRAM,Request);
|
|
AFD_CHECK32on64(TDI_REQUEST_SEND_DATAGRAM,SendDatagramInformation);
|
|
|
|
|
|
typedef UNALIGNED struct _AFD_SEND_DATAGRAM_INFO32 {
|
|
AFD_POINTER_32 BufferArray;
|
|
ULONG BufferCount;
|
|
ULONG AfdFlags;
|
|
TDI_REQUEST_SEND_DATAGRAM32 TdiRequest;
|
|
TDI_CONNECTION_INFORMATION32 TdiConnInfo;
|
|
} AFD_SEND_DATAGRAM_INFO32, *PAFD_SEND_DATAGRAM_INFO32;
|
|
AFD_CHECK32on64(AFD_SEND_DATAGRAM_INFO,BufferArray);
|
|
AFD_CHECK32on64(AFD_SEND_DATAGRAM_INFO,BufferCount);
|
|
AFD_CHECK32on64(AFD_SEND_DATAGRAM_INFO,AfdFlags);
|
|
AFD_CHECK32on64(AFD_SEND_DATAGRAM_INFO,TdiRequest);
|
|
AFD_CHECK32on64(AFD_SEND_DATAGRAM_INFO,TdiConnInfo);
|
|
|
|
typedef UNALIGNED struct _AFD_RECV_INFO32 {
|
|
AFD_POINTER_32 BufferArray;
|
|
ULONG BufferCount;
|
|
ULONG AfdFlags;
|
|
ULONG TdiFlags;
|
|
} AFD_RECV_INFO32, *PAFD_RECV_INFO32;
|
|
AFD_CHECK32on64(AFD_RECV_INFO,BufferArray);
|
|
AFD_CHECK32on64(AFD_RECV_INFO,BufferCount);
|
|
AFD_CHECK32on64(AFD_RECV_INFO,AfdFlags);
|
|
AFD_CHECK32on64(AFD_RECV_INFO,TdiFlags);
|
|
|
|
typedef UNALIGNED struct _AFD_RECV_DATAGRAM_INFO32 {
|
|
AFD_POINTER_32 BufferArray;
|
|
ULONG BufferCount;
|
|
ULONG AfdFlags;
|
|
ULONG TdiFlags;
|
|
AFD_POINTER_32 Address;
|
|
AFD_POINTER_32 AddressLength;
|
|
} AFD_RECV_DATAGRAM_INFO32, *PAFD_RECV_DATAGRAM_INFO32;
|
|
AFD_CHECK32on64(AFD_RECV_DATAGRAM_INFO,BufferArray);
|
|
AFD_CHECK32on64(AFD_RECV_DATAGRAM_INFO,BufferCount);
|
|
AFD_CHECK32on64(AFD_RECV_DATAGRAM_INFO,AfdFlags);
|
|
AFD_CHECK32on64(AFD_RECV_DATAGRAM_INFO,TdiFlags);
|
|
AFD_CHECK32on64(AFD_RECV_DATAGRAM_INFO,Address);
|
|
AFD_CHECK32on64(AFD_RECV_DATAGRAM_INFO,AddressLength);
|
|
|
|
typedef UNALIGNED struct _AFD_CONNECT_JOIN_INFO32 {
|
|
BOOLEAN SanActive;
|
|
VOID * POINTER_32 RootEndpoint; // Root endpoint for joins
|
|
VOID * POINTER_32 ConnectEndpoint;// Connect/leaf endpoint for async connects
|
|
TRANSPORT_ADDRESS RemoteAddress; // Remote address
|
|
} AFD_CONNECT_JOIN_INFO32, *PAFD_CONNECT_JOIN_INFO32;
|
|
AFD_CHECK32on64(AFD_CONNECT_JOIN_INFO,SanActive);
|
|
AFD_CHECK32on64(AFD_CONNECT_JOIN_INFO,RootEndpoint);
|
|
AFD_CHECK32on64(AFD_CONNECT_JOIN_INFO,ConnectEndpoint);
|
|
AFD_CHECK32on64(AFD_CONNECT_JOIN_INFO,RemoteAddress);
|
|
|
|
typedef UNALIGNED struct _AFD_EVENT_SELECT_INFO32 {
|
|
VOID * POINTER_32 Event;
|
|
ULONG PollEvents;
|
|
} AFD_EVENT_SELECT_INFO32, *PAFD_EVENT_SELECT_INFO32;
|
|
AFD_CHECK32on64(AFD_EVENT_SELECT_INFO,Event);
|
|
AFD_CHECK32on64(AFD_EVENT_SELECT_INFO,PollEvents);
|
|
|
|
typedef UNALIGNED struct _AFD_QOS_INFO32 {
|
|
QOS32 Qos;
|
|
BOOLEAN GroupQos;
|
|
} AFD_QOS_INFO32, *PAFD_QOS_INFO32;
|
|
AFD_CHECK32on64(AFD_QOS_INFO,Qos);
|
|
AFD_CHECK32on64(AFD_QOS_INFO,GroupQos);
|
|
|
|
typedef UNALIGNED struct _AFD_TRANSPORT_IOCTL_INFO32 {
|
|
VOID * POINTER_32 Handle;
|
|
AFD_POINTER_32 InputBuffer;
|
|
ULONG InputBufferLength;
|
|
ULONG IoControlCode;
|
|
ULONG AfdFlags;
|
|
ULONG PollEvent;
|
|
} AFD_TRANSPORT_IOCTL_INFO32, *PAFD_TRANSPORT_IOCTL_INFO32;
|
|
AFD_CHECK32on64(AFD_TRANSPORT_IOCTL_INFO,Handle);
|
|
AFD_CHECK32on64(AFD_TRANSPORT_IOCTL_INFO,InputBuffer);
|
|
AFD_CHECK32on64(AFD_TRANSPORT_IOCTL_INFO,InputBufferLength);
|
|
AFD_CHECK32on64(AFD_TRANSPORT_IOCTL_INFO,IoControlCode);
|
|
AFD_CHECK32on64(AFD_TRANSPORT_IOCTL_INFO,AfdFlags);
|
|
AFD_CHECK32on64(AFD_TRANSPORT_IOCTL_INFO,PollEvent);
|
|
|
|
typedef UNALIGNED struct _TRANSMIT_PACKETS_ELEMENT32 {
|
|
ULONG dwElFlags;
|
|
#define TP_MEMORY 1
|
|
#define TP_FILE 2
|
|
#define TP_EOP 4
|
|
ULONG cLength;
|
|
union {
|
|
struct {
|
|
LARGE_INTEGER nFileOffset;
|
|
VOID * POINTER_32 hFile;
|
|
};
|
|
AFD_POINTER_32 pBuffer;
|
|
};
|
|
} TRANSMIT_PACKETS_ELEMENT32, *LPTRANSMIT_PACKETS_ELEMENT32;
|
|
AFD_CHECK32on64(TRANSMIT_PACKETS_ELEMENT,dwElFlags);
|
|
AFD_CHECK32on64(TRANSMIT_PACKETS_ELEMENT,nFileOffset);
|
|
AFD_CHECK32on64(TRANSMIT_PACKETS_ELEMENT,hFile);
|
|
AFD_CHECK32on64(TRANSMIT_PACKETS_ELEMENT,pBuffer);
|
|
|
|
typedef UNALIGNED struct _AFD_TPACKETS_INFO32 {
|
|
AFD_POINTER_32 ElementArray;
|
|
ULONG ElementCount;
|
|
ULONG SendSize;
|
|
ULONG Flags;
|
|
} AFD_TPACKETS_INFO32, *PAFD_TPACKETS_INFO32;
|
|
AFD_CHECK32on64(AFD_TPACKETS_INFO,ElementArray);
|
|
AFD_CHECK32on64(AFD_TPACKETS_INFO,ElementCount);
|
|
AFD_CHECK32on64(AFD_TPACKETS_INFO,SendSize);
|
|
AFD_CHECK32on64(AFD_TPACKETS_INFO,Flags);
|
|
|
|
typedef UNALIGNED struct _AFD_RECV_MESSAGE_INFO32 {
|
|
AFD_RECV_DATAGRAM_INFO32 dgi;
|
|
AFD_POINTER_32 ControlBuffer;
|
|
AFD_POINTER_32 ControlLength;
|
|
AFD_POINTER_32 MsgFlags;
|
|
} AFD_RECV_MESSAGE_INFO32, *PAFD_RECV_MESSAGE_INFO32;
|
|
AFD_CHECK32on64(AFD_RECV_MESSAGE_INFO,dgi);
|
|
AFD_CHECK32on64(AFD_RECV_MESSAGE_INFO,MsgFlags);
|
|
AFD_CHECK32on64(AFD_RECV_MESSAGE_INFO,ControlBuffer);
|
|
AFD_CHECK32on64(AFD_RECV_MESSAGE_INFO,ControlLength);
|
|
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_OPEN_PACKET32 {
|
|
VOID * POINTER_32 CompletionPort;
|
|
VOID * POINTER_32 CompletionEvent;
|
|
} AFD_SWITCH_OPEN_PACKET32, *PAFD_SWITCH_OPEN_PACKET32;
|
|
AFD_CHECK32on64(AFD_SWITCH_OPEN_PACKET,CompletionPort);
|
|
AFD_CHECK32on64(AFD_SWITCH_OPEN_PACKET,CompletionEvent);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_CONTEXT_INFO32 {
|
|
VOID * POINTER_32 SocketHandle;
|
|
AFD_POINTER_32 SwitchContext;
|
|
} AFD_SWITCH_CONTEXT_INFO32, *PAFD_SWITCH_CONTEXT_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_CONTEXT_INFO,SocketHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_CONTEXT_INFO,SwitchContext);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_CONNECT_INFO32 {
|
|
VOID * POINTER_32 ListenHandle;
|
|
AFD_POINTER_32 SwitchContext;
|
|
TRANSPORT_ADDRESS RemoteAddress;
|
|
} AFD_SWITCH_CONNECT_INFO32, *PAFD_SWITCH_CONNECT_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_CONNECT_INFO,ListenHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_CONNECT_INFO,SwitchContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_CONNECT_INFO,RemoteAddress);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_ACCEPT_INFO32 {
|
|
VOID * POINTER_32 AcceptHandle;
|
|
ULONG ReceiveLength;
|
|
} AFD_SWITCH_ACCEPT_INFO32, *PAFD_SWITCH_ACCEPT_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_ACCEPT_INFO,AcceptHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_ACCEPT_INFO,ReceiveLength);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_EVENT_INFO32 {
|
|
VOID * POINTER_32 SocketHandle;
|
|
AFD_POINTER_32 SwitchContext;
|
|
ULONG EventBit;
|
|
NTSTATUS Status;
|
|
} AFD_SWITCH_EVENT_INFO32, *PAFD_SWITCH_EVENT_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_EVENT_INFO,SocketHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_EVENT_INFO,SwitchContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_EVENT_INFO,EventBit);
|
|
AFD_CHECK32on64(AFD_SWITCH_EVENT_INFO,Status);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_REQUEST_INFO32 {
|
|
VOID * POINTER_32 SocketHandle;
|
|
AFD_POINTER_32 SwitchContext;
|
|
AFD_POINTER_32 RequestContext;
|
|
NTSTATUS RequestStatus;
|
|
ULONG DataOffset;
|
|
} AFD_SWITCH_REQUEST_INFO32, *PAFD_SWITCH_REQUEST_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_REQUEST_INFO,SocketHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_REQUEST_INFO,SwitchContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_REQUEST_INFO,RequestContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_REQUEST_INFO,RequestStatus);
|
|
AFD_CHECK32on64(AFD_SWITCH_REQUEST_INFO,DataOffset);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_ACQUIRE_CTX_INFO32 {
|
|
VOID * POINTER_32 SocketHandle;
|
|
AFD_POINTER_32 SwitchContext;
|
|
AFD_POINTER_32 SocketCtxBuf;
|
|
ULONG SocketCtxBufSize;
|
|
} AFD_SWITCH_ACQUIRE_CTX_INFO32, *PAFD_SWITCH_ACQUIRE_CTX_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_ACQUIRE_CTX_INFO,SocketHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_ACQUIRE_CTX_INFO,SwitchContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_ACQUIRE_CTX_INFO,SocketCtxBuf);
|
|
AFD_CHECK32on64(AFD_SWITCH_ACQUIRE_CTX_INFO,SocketCtxBufSize);
|
|
|
|
typedef UNALIGNED struct _AFD_SWITCH_TRANSFER_CTX_INFO32 {
|
|
VOID * POINTER_32 SocketHandle;
|
|
AFD_POINTER_32 SwitchContext;
|
|
AFD_POINTER_32 RequestContext;
|
|
AFD_POINTER_32 SocketCtxBuf;
|
|
ULONG SocketCtxBufSize;
|
|
AFD_POINTER_32 RcvBufferArray;
|
|
ULONG RcvBufferCount;
|
|
NTSTATUS Status;
|
|
} AFD_SWITCH_TRANSFER_CTX_INFO32, *PAFD_SWITCH_TRANSFER_CTX_INFO32;
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,SocketHandle);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,SwitchContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,RequestContext);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,SocketCtxBuf);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,SocketCtxBufSize);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,RcvBufferArray);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,RcvBufferCount);
|
|
AFD_CHECK32on64(AFD_SWITCH_TRANSFER_CTX_INFO,Status);
|
|
|
|
typedef UNALIGNED struct _AFD_PARTIAL_DISCONNECT_INFO32 {
|
|
ULONG DisconnectMode;
|
|
LARGE_INTEGER Timeout;
|
|
} AFD_PARTIAL_DISCONNECT_INFO32, *PAFD_PARTIAL_DISCONNECT_INFO32;
|
|
AFD_CHECK32on64(AFD_PARTIAL_DISCONNECT_INFO,DisconnectMode);
|
|
AFD_CHECK32on64(AFD_PARTIAL_DISCONNECT_INFO,Timeout);
|
|
|
|
typedef UNALIGNED struct _AFD_SUPER_DISCONNECT_INFO32 {
|
|
ULONG Flags;
|
|
} AFD_SUPER_DISCONNECT_INFO32, *PAFD_SUPER_DISCONNECT_INFO32;
|
|
AFD_CHECK32on64(AFD_SUPER_DISCONNECT_INFO,Flags);
|
|
|
|
typedef UNALIGNED struct _AFD_INFORMATION32 {
|
|
ULONG InformationType;
|
|
union {
|
|
BOOLEAN Boolean;
|
|
ULONG Ulong;
|
|
LARGE_INTEGER LargeInteger;
|
|
} Information;
|
|
} AFD_INFORMATION32, *PAFD_INFORMATION32;
|
|
AFD_CHECK32on64(AFD_INFORMATION,InformationType);
|
|
AFD_CHECK32on64(AFD_INFORMATION,Information);
|
|
AFD_CHECK32on64(AFD_INFORMATION,Information.Boolean);
|
|
AFD_CHECK32on64(AFD_INFORMATION,Information.Ulong);
|
|
AFD_CHECK32on64(AFD_INFORMATION,Information.LargeInteger);
|
|
|
|
|
|
typedef UNALIGNED struct _TDI_CMSGHDR32 {
|
|
ULONG cmsg_len;
|
|
LONG cmsg_level;
|
|
LONG cmsg_type;
|
|
/* followed by UCHAR cmsg_data[] */
|
|
} TDI_CMSGHDR32, *PTDI_CMSGHDR32;
|
|
AFD_CHECK32on64(TDI_CMSGHDR,cmsg_len);
|
|
AFD_CHECK32on64(TDI_CMSGHDR,cmsg_level);
|
|
AFD_CHECK32on64(TDI_CMSGHDR,cmsg_type);
|
|
|
|
#ifdef _WIN64
|
|
#define TDI_CMSGHDR_ALIGN32(length) \
|
|
( ((length) + TYPE_ALIGNMENT(TDI_CMSGHDR32)-1) &\
|
|
(~(TYPE_ALIGNMENT(TDI_CMSGHDR32)-1)) ) \
|
|
|
|
#define TDI_CMSGDATA_ALIGN32(length) \
|
|
( ((length) + AFD_MAX_NATURAL_ALIGNMENT32-1) & \
|
|
(~(AFD_MAX_NATURAL_ALIGNMENT32-1)) )
|
|
#endif //_WIN64
|
|
|
|
#endif // ndef _AFDSTR_
|
|
|