LPWSPEVENTSELECT callback function (ws2spi.h)

The LPWSPEventSelect function specifies an event object to be associated with the supplied set of network events.

Syntax

LPWSPEVENTSELECT Lpwspeventselect;

int Lpwspeventselect(
  [in]  SOCKET s,
  [in]  WSAEVENT hEventObject,
  [in]  long lNetworkEvents,
  [out] LPINT lpErrno
)
{...}

Parameters

[in] s

A descriptor identifying the socket.

[in] hEventObject

The handle identifying the event object to be associated with the supplied set of network events.

[in] lNetworkEvents

A bitmask that specifies the combination of network events in which the Windows Sockets SPI client has interest. Constructed by using the bitwise OR operator with any of these values.

Value Meaning
FD_READ
Issues notification of readiness for reading.
FD_WRITE
Issues notification of readiness for writing.
FD_OOB
Issues notification of the arrival of OOB data.
FD_ACCEPT
Issues notification of incoming connections.
FD_CONNECT
Issues notification of completed connection.
FD_CLOSE
Issues notification of socket closure.
FD_QOS
Issues notification of socket (QoS) changes.
FD_GROUP_QOS
Reserved.
FD_ROUTING_INTERFACE_CHANGE
Issues notification of routing interface changes for the specified destination(s).
FD_ADDRESS_LIST_CHANGE
Issues notification of local address list changes for the socket's address family.

[out] lpErrno

A pointer to the error code. See the Return value section for more info.

Return value

The return value is zero if the Windows Sockets SPI client's specification of the network events and the associated event object was successful. Otherwise, the value SOCKET_ERROR is returned, and a specific error number is available in lpErrno.

Error Code Meaning
WSAENETDOWN
The network subsystem has failed.
WSAEINVAL
Indicates that one of the specified parameters was invalid, or the specified socket is in an invalid state.
WSAEINPROGRESS
Blocking Windows Sockets call is in progress or the service provider is still processing a callback function.
WSAENOTSOCK
The descriptor is not a socket.

Remarks

This function is used to specify an event object, hEventObject, to be associated with the selected network events, lNetworkEvents. The socket for which an event object is specified is identified by s. The event object is set when any of the nominated network events occur.

LPWSPEventSelect operates very similarly to LPWSPAsyncSelect, the difference being in the actions taken when a nominated network event occurs. Whereas WSPAsyncSelect causes a Windows Sockets SPI client-specified Windows message to be posted, LPWSPEventSelect sets the associated event object and records the occurrence of this event in an internal network event record. A Windows Sockets SPI client can use LPWSPEnumNetworkEvents to retrieve the contents of the internal network event record, and thus determine which of the nominated network events have occurred.

LPWSPEventSelect is the only function that causes network activity and errors to be recorded and retrievable through LPWSPEnumNetworkEvents. See the descriptions of LPWSPSelect and LPWSPAsyncSelect to find out how those functions report network activity and errors.

This function automatically sets socket s to nonblocking mode, regardless of the value of lNetworkEvents.

Issuing an LPWSPEventSelect for a socket cancels any previous LPWSPAsyncSelect or LPWSPEventSelect for the same socket, and clears the internal network event record. For example, to associate an event object with both reading and writing network events, the Windows Sockets SPI client must call LPWSPEventSelect with both FD_READ and FD_WRITE, like this.

rc = WSPEventSelect(s, hEventObject, FD_READ | FD_WRITE);

It's not possible to specify different event objects for different network events. The following code won't work; the second call cancels the effects of the first, and the only association will be the FD_WRITE network event associated with hEventObject2.

// Incorrect example.
rc = WSPEventSelect(s, hEventObject1, FD_READ);
rc = WSPEventSelect(s, hEventObject2, FD_WRITE);

To cancel the association and selection of network events on a socket, you should set lNetworkEvents to zero, in which case the hEventObject parameter is ignored.

rc = WSPEventSelect(s, hEventObject, 0);

Closing a socket with LPWSPCloseSocket also cancels the association and selection of network events specified in LPWSPEventSelect for the socket. The Windows Sockets SPI client, however, still must call WSACloseEvent to explicitly close the event object, and free any resources.

Since an LPWSPAccept'ed socket has the same properties as the listening socket used to accept it, any LPWSPEventSelect association and network events selection set for the listening socket apply to the accepted socket. For example, if a listening socket has LPWSPEventSelect association of hEventObject with FD_ACCEPT, FD_READ, and FD_WRITE, then any socket accepted on that listening socket will also have FD_ACCEPT, FD_READ, and FD_WRITE network events associated with the same hEventObject. If a different hEventObject or network events are desired, then the Windows Sockets SPI client should call LPWSPEventSelect, passing the accepted socket, and the desired new information.

Having successfully recorded the occurrence of the network event and signaled the associated event object, no further actions are taken for that network event until the Windows Sockets SPI client makes the function call that implicitly re-enables the setting of that network event and signaling of the associated event object.

Network event Re-enabling function
FD_READ LPWSPRecv or LPWSPRecvFrom
FD_WRITE LPWSPSend or LPWSPSendTo
FD_OOB LPWSPRecv or LPWSPRecvFrom
FD_ACCEPT LPWSPAccept, unless the error code returned is WSATRY_AGAIN indicating that the condition function returned CF_DEFER
FD_CONNECT NONE
FD_CLOSE NONE
FD_QOS LPWSPIoctl with SIO_GET_QOS
FD_GROUP_QOS Reserved for future use with socket groups: LPWSPIoctl with SIO_GET_GROUP_QOS
FD_ROUTING_INTERFACE_CHANGE LPWSPIoctl with command SIO_ROUTING_INTERFACE_CHANGE
FD_ADDRESS_LIST_CHANGE LPWSPIoctl with command SIO_ADDRESS_LIST_CHANGE

Any call to the re-enabling routine, even one that fails, results in re-enabling of recording and signaling for the relevant network event and event object, respectively.

For FD_READ, FD_OOB, and FD_ACCEPT network events, network event recording and event object signaling are level-triggered. This means that if the re-enabling routine is called, and the relevant network condition is still valid after the call, then the network event is recorded and the associated event object is signaled. This allows a Windows Sockets SPI client to be event-driven while also being unconcerned with the amount of data that arrives at any one time. Consider the following sequence.

  1. The service provider receives 100 bytes of data on socket s, records the FD_READ network event, and signals the associated event object.
  2. The Windows Sockets SPI client issues WSPRecv(s, buffptr, 50, 0) to read 50 bytes.
  3. The service provider records the FD_READ network event, and signals the associated event object again since there is still data to be read.

With these semantics, a Windows Sockets SPI client need not read all available data in response to an FD_READ network event. Rather, a single LPWSPRecv in response to each FD_READ network event is appropriate.

The FD_QOS and FD_GROUP_QOS events are considered edge-triggered. A message will be posted exactly once when a quality of service (QOS) change occurs. Further indications won't be issued until either the service provider detects a further change in QOS, or the Windows Sockets SPI client renegotiates the QOS for the socket.

The FD_ROUTING_INTERFACE_CHANGE and FD_ADDRESS_LIST_CHANGE events are considered edge-triggered as well. A message will be posted exactly once when a change occurs after the Windows Sockets SPI client has request the notification by issuing WSAIoctl with SIO_ROUTING_INTERFACE_CHANGE or SIO_ADDRESS_LIST_CHANGE correspondingly. Further messages won't be forthcoming until the Windows Sockets SPI client reissues the IOCTL and another change is detected since the IOCTL was issued.

If a network event has already happened when the Windows Sockets SPI client calls LPWSPEventSelect, or when the re-enabling function is called, then a network event is recorded and the associated event object is signaled, as appropriate. For example, consider the following sequence.

  1. A Windows Sockets SPI client calls LPWSPListen.
  2. A connect request is received, but not yet accepted.
  3. The Windows Sockets SPI client calls LPWSPEventSelect specifying that it is interested in the FD_ACCEPT network event for the socket. The service provider records the FD_ACCEPT network event, and signals the associated event object immediately.

The FD_WRITE network event is handled slightly differently. An FD_WRITE network event is recorded when a socket is first connected with LPWSPConnect or accepted with LPWSPAccept, and then after an LPWSPSend or LPWSPSendTo fails with WSAEWOULDBLOCK and buffer space becomes available. Therefore, a Windows Sockets SPI client can assume that sends are possible starting from the first FD_WRITE network event setting, and lasting until a send returns WSAEWOULDBLOCK. After such a failure, the Windows Sockets SPI client will find that sends are again possible when an FD_WRITE network event is recorded and the associated event object is signaled.

The FD_OOB network event is used only when a socket is configured to receive out-of-band data separately. If the socket is configured to receive out-of-band data in-line, then the out-of-band (expedited) data is treated as normal data, and the Windows Sockets SPI client should register an interest in, and will get, FD_READ network event, not FD_OOB network event. A Windows Sockets SPI client may set or inspect the way in which out-of-band data is to be handled by using LPWSPSetSockOpt or LPWSPGetSockOpt for the SO_OOBINLINE option.

The error code in an FD_CLOSE network event indicates whether the socket close was graceful, or abortive. If the error code is 0, then the close was graceful; if the error code is WSAECONNRESET, then the socket's virtual circuit was reset. This applies only to connection-oriented sockets such as SOCK_STREAM.

The FD_CLOSE network event is recorded when a close indication is received for the virtual circuit corresponding to the socket. In TCP terms, this means that the FD_CLOSE is recorded when the connection goes into the FIN WAIT or CLOSE WAIT states. This results from the remote end performing an LPWSPShutdown on the send side, or an LPWSPCloseSocket.

A service providers should record only an FD_CLOSE network event to indicate closure of a virtual circuit; it should not record an FD_READ network event to indicate that condition.

The FD_QOS or FD_GROUP_QOS network event is recorded when there has been a change to any field in the flow spec associated with socket s, or the socket group that s belongs to, respectively. This change must be made available to Windows Sockets SPI clients via the LPWSPIoctl function with SIO_GET_QOS and/or SIO_GET_GROUP_QOS to retrieve the current QOS for socket s, or for the socket group s belongs to, respectively.

The FD_ROUTING_INTERFACE_CHANGE network event is recorded when the local interface that should be used to reach the destination specified in WSAIoctl with SIO_ROUTING_INTERFACE_CHANGE changes after such IOCTL has been issued.

The FD_ADDRESS_LIST_CHANGE network event is recorded when the list of addresses of sockets' protocol family to which the Windows Sockets SPI client can bind changes after WSAIoctl with SIO_ADDRESS_LIST_CHANGE has been issued.

Requirements

Requirement Value
Minimum supported client Windows 10 Build 20348
Minimum supported server Windows 10 Build 20348
Header ws2spi.h

See also

LPWSPEnumNetworkEvents