mirror of
https://github.com/reactos/reactos.git
synced 2024-10-30 03:27:31 +00:00
11b706429c
- Mismatch NTSTATUS/DWORD for win32 errors; - Close handle returned from openTcpFile() with closeTcpFile().
818 lines
21 KiB
C
818 lines
21 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS WinSock Helper DLL for TCP/IP
|
|
* FILE: wshtcpip.c
|
|
* PURPOSE: DLL entry
|
|
* PROGRAMMERS: Casper S. Hornstrup (chorns@users.sourceforge.net)
|
|
* REVISIONS:
|
|
* CSH 01/09-2000 Created
|
|
*/
|
|
#include "wshtcpip.h"
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
BOOL
|
|
EXPORT
|
|
DllMain(HANDLE hInstDll,
|
|
ULONG dwReason,
|
|
PVOID Reserved)
|
|
{
|
|
DPRINT("DllMain of wshtcpip.dll\n");
|
|
|
|
switch (dwReason) {
|
|
case DLL_PROCESS_ATTACH:
|
|
/* Don't need thread attach notifications
|
|
so disable them to improve performance */
|
|
DisableThreadLibraryCalls(hInstDll);
|
|
break;
|
|
|
|
case DLL_THREAD_ATTACH:
|
|
break;
|
|
|
|
case DLL_THREAD_DETACH:
|
|
break;
|
|
|
|
case DLL_PROCESS_DETACH:
|
|
break;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHAddressToString(
|
|
IN LPSOCKADDR Address,
|
|
IN INT AddressLength,
|
|
IN LPWSAPROTOCOL_INFOW ProtocolInfo OPTIONAL,
|
|
OUT LPWSTR AddressString,
|
|
IN OUT LPDWORD AddressStringLength)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHEnumProtocols(
|
|
IN LPINT lpiProtocols OPTIONAL,
|
|
IN LPWSTR lpTransportKeyName,
|
|
IN OUT LPVOID lpProtocolBuffer,
|
|
IN OUT LPDWORD lpdwBufferLength)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHGetBroadcastSockaddr(
|
|
IN PVOID HelperDllSocketContext,
|
|
OUT PSOCKADDR Sockaddr,
|
|
OUT PINT SockaddrLength)
|
|
{
|
|
INT Size = 2 * sizeof(UINT);
|
|
|
|
if (*SockaddrLength < Size)
|
|
{
|
|
DPRINT1("Socket address length too small: %d\n", *SockaddrLength);
|
|
return WSAEFAULT;
|
|
}
|
|
|
|
RtlZeroMemory(Sockaddr, *SockaddrLength);
|
|
|
|
Sockaddr->sa_family = AF_INET;
|
|
*((PUINT)Sockaddr->sa_data) = INADDR_BROADCAST;
|
|
|
|
/* *SockaddrLength = Size; */
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHGetProviderGuid(
|
|
IN LPWSTR ProviderName,
|
|
OUT LPGUID ProviderGuid)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
/*
|
|
Document from OSR how WSHGetSockaddrType works
|
|
http://www.osronline.com/ddkx/network/37wshfun_5lyq.htm
|
|
*/
|
|
|
|
INT
|
|
EXPORT
|
|
WSHGetSockaddrType(
|
|
IN PSOCKADDR Sockaddr,
|
|
IN DWORD SockaddrLength,
|
|
OUT PSOCKADDR_INFO SockaddrInfo)
|
|
{
|
|
PSOCKADDR_IN ipv4 = (PSOCKADDR_IN)Sockaddr;
|
|
|
|
if (!ipv4 || !SockaddrInfo || SockaddrLength < sizeof(SOCKADDR_IN) ||
|
|
ipv4->sin_family != AF_INET)
|
|
{
|
|
DPRINT1("Invalid parameter: %x %x %d %u\n", ipv4, SockaddrInfo, SockaddrLength, (ipv4 ? ipv4->sin_family : 0));
|
|
return WSAEINVAL;
|
|
}
|
|
|
|
switch (ntohl(ipv4->sin_addr.s_addr))
|
|
{
|
|
case INADDR_ANY:
|
|
SockaddrInfo->AddressInfo = SockaddrAddressInfoWildcard;
|
|
break;
|
|
|
|
case INADDR_BROADCAST:
|
|
SockaddrInfo->AddressInfo = SockaddrAddressInfoBroadcast;
|
|
break;
|
|
|
|
case INADDR_LOOPBACK:
|
|
SockaddrInfo->AddressInfo = SockaddrAddressInfoLoopback;
|
|
break;
|
|
|
|
default:
|
|
SockaddrInfo->AddressInfo = SockaddrAddressInfoNormal;
|
|
break;
|
|
}
|
|
|
|
if (ntohs(ipv4->sin_port) == 0)
|
|
SockaddrInfo->EndpointInfo = SockaddrEndpointInfoWildcard;
|
|
else if (ntohs(ipv4->sin_port) < IPPORT_RESERVED)
|
|
SockaddrInfo->EndpointInfo = SockaddrEndpointInfoReserved;
|
|
else
|
|
SockaddrInfo->EndpointInfo = SockaddrEndpointInfoNormal;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
static
|
|
void
|
|
GetTdiTypeId(
|
|
_In_ INT Level,
|
|
_In_ INT OptionName,
|
|
_Out_ PULONG TdiType,
|
|
_Out_ PULONG TdiId)
|
|
{
|
|
switch (Level)
|
|
{
|
|
case SOL_SOCKET:
|
|
*TdiType = INFO_TYPE_ADDRESS_OBJECT;
|
|
switch (OptionName)
|
|
{
|
|
case SO_KEEPALIVE:
|
|
/* FIXME: Return proper option */
|
|
ASSERT(FALSE);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case IPPROTO_IP:
|
|
*TdiType = INFO_TYPE_ADDRESS_OBJECT;
|
|
switch (OptionName)
|
|
{
|
|
case IP_TTL:
|
|
*TdiId = AO_OPTION_TTL;
|
|
return;
|
|
|
|
case IP_DONTFRAGMENT:
|
|
*TdiId = AO_OPTION_IP_DONTFRAGMENT;
|
|
return;
|
|
|
|
#if 0
|
|
case IP_RECEIVE_BROADCAST:
|
|
*TdiId = AO_OPTION_BROADCAST;
|
|
return;
|
|
#endif
|
|
|
|
case IP_HDRINCL:
|
|
*TdiId = AO_OPTION_IP_HDRINCL;
|
|
return;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case IPPROTO_TCP:
|
|
*TdiType = INFO_TYPE_CONNECTION;
|
|
switch (OptionName)
|
|
{
|
|
case TCP_NODELAY:
|
|
*TdiId = TCP_SOCKET_NODELAY;
|
|
return;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
DPRINT1("Unknown level/option name: %d %d\n", Level, OptionName);
|
|
*TdiType = 0;
|
|
*TdiId = 0;
|
|
}
|
|
|
|
INT
|
|
EXPORT
|
|
WSHGetSocketInformation(
|
|
IN PVOID HelperDllSocketContext,
|
|
IN SOCKET SocketHandle,
|
|
IN HANDLE TdiAddressObjectHandle,
|
|
IN HANDLE TdiConnectionObjectHandle,
|
|
IN INT Level,
|
|
IN INT OptionName,
|
|
OUT PCHAR OptionValue,
|
|
OUT LPINT OptionLength)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
DPRINT1("Get: Unknown level/option name: %d %d\n", Level, OptionName);
|
|
|
|
*OptionLength = 0;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHGetWildcardSockaddr(
|
|
IN PVOID HelperDllSocketContext,
|
|
OUT PSOCKADDR Sockaddr,
|
|
OUT PINT SockaddrLength)
|
|
{
|
|
INT Size = 2 * sizeof(UINT);
|
|
|
|
if (*SockaddrLength < Size)
|
|
{
|
|
DPRINT1("Socket address length too small: %d\n", *SockaddrLength);
|
|
return WSAEFAULT;
|
|
}
|
|
|
|
RtlZeroMemory(Sockaddr, *SockaddrLength);
|
|
|
|
Sockaddr->sa_family = AF_INET;
|
|
*((PUINT)Sockaddr->sa_data) = INADDR_ANY;
|
|
|
|
/* *SockaddrLength = Size; */
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
DWORD
|
|
EXPORT
|
|
WSHGetWinsockMapping(
|
|
OUT PWINSOCK_MAPPING Mapping,
|
|
IN DWORD MappingLength)
|
|
{
|
|
DWORD Rows = 6;
|
|
DWORD Columns = 3;
|
|
DWORD Size = 2 * sizeof(DWORD) + Columns * Rows * sizeof(DWORD);
|
|
|
|
if (MappingLength < Size)
|
|
{
|
|
DPRINT1("Mapping length too small: %d\n", MappingLength);
|
|
return Size;
|
|
}
|
|
|
|
Mapping->Rows = Rows;
|
|
Mapping->Columns = Columns;
|
|
|
|
Mapping->Mapping[0].AddressFamily = AF_INET;
|
|
Mapping->Mapping[0].SocketType = SOCK_STREAM;
|
|
Mapping->Mapping[0].Protocol = 0;
|
|
|
|
Mapping->Mapping[1].AddressFamily = AF_INET;
|
|
Mapping->Mapping[1].SocketType = SOCK_STREAM;
|
|
Mapping->Mapping[1].Protocol = IPPROTO_TCP;
|
|
|
|
Mapping->Mapping[2].AddressFamily = AF_INET;
|
|
Mapping->Mapping[2].SocketType = SOCK_DGRAM;
|
|
Mapping->Mapping[2].Protocol = 0;
|
|
|
|
Mapping->Mapping[3].AddressFamily = AF_INET;
|
|
Mapping->Mapping[3].SocketType = SOCK_DGRAM;
|
|
Mapping->Mapping[3].Protocol = IPPROTO_UDP;
|
|
|
|
Mapping->Mapping[4].AddressFamily = AF_INET;
|
|
Mapping->Mapping[4].SocketType = SOCK_RAW;
|
|
Mapping->Mapping[4].Protocol = 0;
|
|
|
|
Mapping->Mapping[5].AddressFamily = AF_INET;
|
|
Mapping->Mapping[5].SocketType = SOCK_RAW;
|
|
Mapping->Mapping[5].Protocol = IPPROTO_ICMP;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHGetWSAProtocolInfo(
|
|
IN LPWSTR ProviderName,
|
|
OUT LPWSAPROTOCOL_INFOW *ProtocolInfo,
|
|
OUT LPDWORD ProtocolInfoEntries)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHIoctl(
|
|
IN PVOID HelperDllSocketContext,
|
|
IN SOCKET SocketHandle,
|
|
IN HANDLE TdiAddressObjectHandle,
|
|
IN HANDLE TdiConnectionObjectHandle,
|
|
IN DWORD IoControlCode,
|
|
IN LPVOID InputBuffer,
|
|
IN DWORD InputBufferLength,
|
|
IN LPVOID OutputBuffer,
|
|
IN DWORD OutputBufferLength,
|
|
OUT LPDWORD NumberOfBytesReturned,
|
|
IN LPWSAOVERLAPPED Overlapped,
|
|
IN LPWSAOVERLAPPED_COMPLETION_ROUTINE CompletionRoutine,
|
|
OUT LPBOOL NeedsCompletion)
|
|
{
|
|
INT res;
|
|
|
|
if (IoControlCode == SIO_GET_INTERFACE_LIST)
|
|
{
|
|
res = WSHIoctl_GetInterfaceList(
|
|
OutputBuffer,
|
|
OutputBufferLength,
|
|
NumberOfBytesReturned,
|
|
NeedsCompletion);
|
|
return res;
|
|
}
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
DPRINT1("Ioctl: Unknown IOCTL code: %x\n", IoControlCode);
|
|
|
|
return WSAEINVAL;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHJoinLeaf(
|
|
IN PVOID HelperDllSocketContext,
|
|
IN SOCKET SocketHandle,
|
|
IN HANDLE TdiAddressObjectHandle,
|
|
IN HANDLE TdiConnectionObjectHandle,
|
|
IN PVOID LeafHelperDllSocketContext,
|
|
IN SOCKET LeafSocketHandle,
|
|
IN PSOCKADDR Sockaddr,
|
|
IN DWORD SockaddrLength,
|
|
IN LPWSABUF CallerData,
|
|
IN LPWSABUF CalleeData,
|
|
IN LPQOS SocketQOS,
|
|
IN LPQOS GroupQOS,
|
|
IN DWORD Flags)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
INT
|
|
SendRequest(
|
|
IN PVOID Request,
|
|
IN DWORD RequestSize,
|
|
IN DWORD IOCTL)
|
|
{
|
|
BOOLEAN Success;
|
|
HANDLE TcpCC;
|
|
DWORD BytesReturned;
|
|
|
|
if (openTcpFile(&TcpCC, FILE_READ_DATA | FILE_WRITE_DATA) != STATUS_SUCCESS)
|
|
return WSAEINVAL;
|
|
|
|
Success = DeviceIoControl(TcpCC,
|
|
IOCTL,
|
|
Request,
|
|
RequestSize,
|
|
NULL,
|
|
0,
|
|
&BytesReturned,
|
|
NULL);
|
|
|
|
closeTcpFile(TcpCC);
|
|
|
|
DPRINT("DeviceIoControl: %ld\n", (Success ? NO_ERROR : GetLastError()));
|
|
|
|
if (!Success)
|
|
return WSAEINVAL;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
INT
|
|
EXPORT
|
|
WSHNotify(
|
|
IN PVOID HelperDllSocketContext,
|
|
IN SOCKET SocketHandle,
|
|
IN HANDLE TdiAddressObjectHandle,
|
|
IN HANDLE TdiConnectionObjectHandle,
|
|
IN DWORD NotifyEvent)
|
|
{
|
|
PSOCKET_CONTEXT Context = HelperDllSocketContext;
|
|
NTSTATUS Status;
|
|
HANDLE TcpCC;
|
|
TDIEntityID *EntityIDs;
|
|
DWORD EntityCount, i;
|
|
PQUEUED_REQUEST QueuedRequest, NextQueuedRequest;
|
|
|
|
switch (NotifyEvent)
|
|
{
|
|
case WSH_NOTIFY_CLOSE:
|
|
DPRINT("WSHNotify: WSH_NOTIFY_CLOSE\n");
|
|
QueuedRequest = Context->RequestQueue;
|
|
while (QueuedRequest)
|
|
{
|
|
NextQueuedRequest = QueuedRequest->Next;
|
|
|
|
HeapFree(GetProcessHeap(), 0, QueuedRequest->Info);
|
|
HeapFree(GetProcessHeap(), 0, QueuedRequest);
|
|
|
|
QueuedRequest = NextQueuedRequest;
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, HelperDllSocketContext);
|
|
break;
|
|
|
|
|
|
case WSH_NOTIFY_BIND:
|
|
DPRINT("WSHNotify: WSH_NOTIFY_BIND\n");
|
|
Status = openTcpFile(&TcpCC, FILE_READ_DATA);
|
|
if (Status != STATUS_SUCCESS)
|
|
return WSAEINVAL;
|
|
|
|
Status = tdiGetEntityIDSet(TcpCC,
|
|
&EntityIDs,
|
|
&EntityCount);
|
|
|
|
closeTcpFile(TcpCC);
|
|
|
|
if (Status != STATUS_SUCCESS)
|
|
return WSAEINVAL;
|
|
|
|
for (i = 0; i < EntityCount; i++)
|
|
{
|
|
if (EntityIDs[i].tei_entity == CO_TL_ENTITY ||
|
|
EntityIDs[i].tei_entity == CL_TL_ENTITY ||
|
|
EntityIDs[i].tei_entity == ER_ENTITY)
|
|
{
|
|
Context->AddrFileInstance = EntityIDs[i].tei_instance;
|
|
Context->AddrFileEntityType = EntityIDs[i].tei_entity;
|
|
}
|
|
}
|
|
|
|
DPRINT("Instance: %lx Type: %lx\n", Context->AddrFileInstance, Context->AddrFileEntityType);
|
|
|
|
tdiFreeThingSet(EntityIDs);
|
|
|
|
Context->SocketState = SocketStateBound;
|
|
|
|
QueuedRequest = Context->RequestQueue;
|
|
while (QueuedRequest)
|
|
{
|
|
QueuedRequest->Info->ID.toi_entity.tei_entity = Context->AddrFileEntityType;
|
|
QueuedRequest->Info->ID.toi_entity.tei_instance = Context->AddrFileInstance;
|
|
|
|
SendRequest(QueuedRequest->Info,
|
|
sizeof(*QueuedRequest->Info) + QueuedRequest->Info->BufferSize,
|
|
IOCTL_TCP_SET_INFORMATION_EX);
|
|
|
|
NextQueuedRequest = QueuedRequest->Next;
|
|
|
|
HeapFree(GetProcessHeap(), 0, QueuedRequest->Info);
|
|
HeapFree(GetProcessHeap(), 0, QueuedRequest);
|
|
|
|
QueuedRequest = NextQueuedRequest;
|
|
}
|
|
Context->RequestQueue = NULL;
|
|
break;
|
|
|
|
default:
|
|
DPRINT1("Unwanted notification received! (%ld)\n", NotifyEvent);
|
|
break;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHOpenSocket(
|
|
IN OUT PINT AddressFamily,
|
|
IN OUT PINT SocketType,
|
|
IN OUT PINT Protocol,
|
|
OUT PUNICODE_STRING TransportDeviceName,
|
|
OUT PVOID HelperDllSocketContext,
|
|
OUT PDWORD NotificationEvents)
|
|
/*
|
|
* FUNCTION: Opens a socket
|
|
*/
|
|
{
|
|
return WSHOpenSocket2(AddressFamily,
|
|
SocketType,
|
|
Protocol,
|
|
0,
|
|
0,
|
|
TransportDeviceName,
|
|
HelperDllSocketContext,
|
|
NotificationEvents);
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHOpenSocket2(
|
|
OUT PINT AddressFamily,
|
|
IN OUT PINT SocketType,
|
|
IN OUT PINT Protocol,
|
|
IN GROUP Group,
|
|
IN DWORD Flags,
|
|
OUT PUNICODE_STRING TransportDeviceName,
|
|
OUT PVOID *HelperDllSocketContext,
|
|
OUT PDWORD NotificationEvents)
|
|
/*
|
|
* FUNCTION: Opens a socket
|
|
* ARGUMENTS:
|
|
* AddressFamily = Address of buffer with address family (updated)
|
|
* SocketType = Address of buffer with type of socket (updated)
|
|
* Protocol = Address of buffer with protocol number (updated)
|
|
* Group = Socket group
|
|
* Flags = Socket flags
|
|
* TransportDeviceName = Address of buffer to place name of transport device
|
|
* HelperDllSocketContext = Address of buffer to place socket context pointer
|
|
* NotificationEvents = Address of buffer to place flags for event notification
|
|
* RETURNS:
|
|
* Status of operation
|
|
* NOTES:
|
|
* Mapping tripple is returned in an canonicalized form
|
|
*/
|
|
{
|
|
PSOCKET_CONTEXT Context;
|
|
UNICODE_STRING String;
|
|
UNICODE_STRING TcpDeviceName = RTL_CONSTANT_STRING(DD_TCP_DEVICE_NAME);
|
|
UNICODE_STRING UdpDeviceName = RTL_CONSTANT_STRING(DD_UDP_DEVICE_NAME);
|
|
UNICODE_STRING RawDeviceName = RTL_CONSTANT_STRING(DD_RAW_IP_DEVICE_NAME);
|
|
|
|
DPRINT("WSHOpenSocket2 called\n");
|
|
|
|
switch (*SocketType) {
|
|
case SOCK_STREAM:
|
|
String = TcpDeviceName;
|
|
break;
|
|
|
|
case SOCK_DGRAM:
|
|
String = UdpDeviceName;
|
|
break;
|
|
|
|
case SOCK_RAW:
|
|
if ((*Protocol < 0) || (*Protocol > 255))
|
|
return WSAEINVAL;
|
|
|
|
String = RawDeviceName;
|
|
break;
|
|
|
|
default:
|
|
return WSAEINVAL;
|
|
}
|
|
|
|
RtlInitUnicodeString(TransportDeviceName, NULL);
|
|
|
|
TransportDeviceName->MaximumLength = String.Length + /* Transport device name */
|
|
(4 * sizeof(WCHAR) + /* Separator and protocol */
|
|
sizeof(UNICODE_NULL)); /* Terminating null */
|
|
|
|
TransportDeviceName->Buffer = HeapAlloc(
|
|
GetProcessHeap(),
|
|
0,
|
|
TransportDeviceName->MaximumLength);
|
|
|
|
if (!TransportDeviceName->Buffer)
|
|
return WSAENOBUFS;
|
|
|
|
/* Append the transport device name */
|
|
RtlAppendUnicodeStringToString(TransportDeviceName, &String);
|
|
|
|
if (*SocketType == SOCK_RAW) {
|
|
/* Append a separator */
|
|
TransportDeviceName->Buffer[TransportDeviceName->Length / sizeof(WCHAR)] = OBJ_NAME_PATH_SEPARATOR;
|
|
TransportDeviceName->Length += sizeof(WCHAR);
|
|
TransportDeviceName->Buffer[TransportDeviceName->Length / sizeof(WCHAR)] = UNICODE_NULL;
|
|
|
|
/* Append the protocol number */
|
|
String.Buffer = TransportDeviceName->Buffer + (TransportDeviceName->Length / sizeof(WCHAR));
|
|
String.Length = 0;
|
|
String.MaximumLength = TransportDeviceName->MaximumLength - TransportDeviceName->Length;
|
|
|
|
RtlIntegerToUnicodeString((ULONG)*Protocol, 10, &String);
|
|
|
|
TransportDeviceName->Length += String.Length;
|
|
}
|
|
|
|
/* Setup a socket context area */
|
|
|
|
Context = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(SOCKET_CONTEXT));
|
|
if (!Context) {
|
|
RtlFreeUnicodeString(TransportDeviceName);
|
|
return WSAENOBUFS;
|
|
}
|
|
|
|
Context->AddressFamily = *AddressFamily;
|
|
Context->SocketType = *SocketType;
|
|
Context->Protocol = *Protocol;
|
|
Context->Flags = Flags;
|
|
Context->SocketState = SocketStateCreated;
|
|
|
|
*HelperDllSocketContext = Context;
|
|
*NotificationEvents = WSH_NOTIFY_CLOSE | WSH_NOTIFY_BIND;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
INT
|
|
EXPORT
|
|
WSHSetSocketInformation(
|
|
IN PVOID HelperDllSocketContext,
|
|
IN SOCKET SocketHandle,
|
|
IN HANDLE TdiAddressObjectHandle,
|
|
IN HANDLE TdiConnectionObjectHandle,
|
|
IN INT Level,
|
|
IN INT OptionName,
|
|
IN PCHAR OptionValue,
|
|
IN INT OptionLength)
|
|
{
|
|
PSOCKET_CONTEXT Context = HelperDllSocketContext;
|
|
ULONG TdiType, TdiId;
|
|
INT Status;
|
|
PTCP_REQUEST_SET_INFORMATION_EX Info;
|
|
PQUEUED_REQUEST Queued, NextQueued;
|
|
|
|
DPRINT("WSHSetSocketInformation\n");
|
|
|
|
/* FIXME: We only handle address file object here */
|
|
|
|
switch (Level)
|
|
{
|
|
case SOL_SOCKET:
|
|
switch (OptionName)
|
|
{
|
|
case SO_DONTROUTE:
|
|
if (OptionLength < sizeof(BOOL))
|
|
{
|
|
return WSAEFAULT;
|
|
}
|
|
Context->DontRoute = *(BOOL*)OptionValue;
|
|
/* This is silently ignored on Windows */
|
|
return 0;
|
|
|
|
case SO_KEEPALIVE:
|
|
/* FIXME -- We'll send this to TCPIP */
|
|
DPRINT1("Set: SO_KEEPALIVE not yet supported\n");
|
|
return 0;
|
|
|
|
default:
|
|
/* Invalid option */
|
|
DPRINT1("Set: Received unexpected SOL_SOCKET option %d\n", OptionName);
|
|
return WSAENOPROTOOPT;
|
|
}
|
|
break;
|
|
|
|
case IPPROTO_IP:
|
|
switch (OptionName)
|
|
{
|
|
case IP_TTL:
|
|
case IP_DONTFRAGMENT:
|
|
case IP_HDRINCL:
|
|
/* Send these to TCPIP */
|
|
break;
|
|
|
|
default:
|
|
/* Invalid option -- FIXME */
|
|
DPRINT1("Set: Received unsupported IPPROTO_IP option %d\n", OptionName);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
case IPPROTO_TCP:
|
|
switch (OptionName)
|
|
{
|
|
case TCP_NODELAY:
|
|
if (OptionLength < sizeof(CHAR))
|
|
{
|
|
return WSAEFAULT;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
/* Invalid option */
|
|
DPRINT1("Set: Received unexpected IPPROTO_TCP option %d\n", OptionName);
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
DPRINT1("Set: Received unexpected %d option %d\n", Level, OptionName);
|
|
return 0;
|
|
}
|
|
|
|
/* If we get here, GetAddressOption must return something valid */
|
|
GetTdiTypeId(Level, OptionName, &TdiType, &TdiId);
|
|
ASSERT((TdiId != 0) && (TdiType != 0));
|
|
|
|
Info = HeapAlloc(GetProcessHeap(), 0, sizeof(*Info) + OptionLength);
|
|
if (!Info)
|
|
return WSAENOBUFS;
|
|
|
|
Info->ID.toi_entity.tei_entity = Context->AddrFileEntityType;
|
|
Info->ID.toi_entity.tei_instance = Context->AddrFileInstance;
|
|
Info->ID.toi_class = INFO_CLASS_PROTOCOL;
|
|
Info->ID.toi_type = TdiType;
|
|
Info->ID.toi_id = TdiId;
|
|
Info->BufferSize = OptionLength;
|
|
memcpy(Info->Buffer, OptionValue, OptionLength);
|
|
|
|
if (Context->SocketState == SocketStateCreated)
|
|
{
|
|
if (Context->RequestQueue)
|
|
{
|
|
Queued = Context->RequestQueue;
|
|
while ((NextQueued = Queued->Next))
|
|
{
|
|
Queued = NextQueued;
|
|
}
|
|
|
|
Queued->Next = HeapAlloc(GetProcessHeap(), 0, sizeof(QUEUED_REQUEST));
|
|
if (!Queued->Next)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, Info);
|
|
return WSAENOBUFS;
|
|
}
|
|
|
|
NextQueued = Queued->Next;
|
|
NextQueued->Next = NULL;
|
|
NextQueued->Info = Info;
|
|
}
|
|
else
|
|
{
|
|
Context->RequestQueue = HeapAlloc(GetProcessHeap(), 0, sizeof(QUEUED_REQUEST));
|
|
if (!Context->RequestQueue)
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, Info);
|
|
return WSAENOBUFS;
|
|
}
|
|
|
|
Context->RequestQueue->Next = NULL;
|
|
Context->RequestQueue->Info = Info;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
Status = SendRequest(Info, sizeof(*Info) + Info->BufferSize, IOCTL_TCP_SET_INFORMATION_EX);
|
|
|
|
HeapFree(GetProcessHeap(), 0, Info);
|
|
|
|
return Status;
|
|
}
|
|
|
|
|
|
INT
|
|
EXPORT
|
|
WSHStringToAddress(
|
|
IN LPWSTR AddressString,
|
|
IN DWORD AddressFamily,
|
|
IN LPWSAPROTOCOL_INFOW ProtocolInfo OPTIONAL,
|
|
OUT LPSOCKADDR Address,
|
|
IN OUT LPDWORD AddressStringLength)
|
|
{
|
|
UNIMPLEMENTED;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
/* EOF */
|