mirror of https://github.com/tongzx/nt5src
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.
2960 lines
67 KiB
2960 lines
67 KiB
/*++
|
|
|
|
Copyright (c) 1996-2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
cnapi.c
|
|
|
|
Abstract:
|
|
|
|
Cluster Network configuration APIs
|
|
|
|
Author:
|
|
|
|
Mike Massa (mikemas) 18-Mar-1996
|
|
|
|
Environment:
|
|
|
|
User Mode - Win32
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
#include <nt.h>
|
|
#include <ntrtl.h>
|
|
#include <nturtl.h>
|
|
#include <windows.h>
|
|
#include <clusapi.h>
|
|
#include <clusdef.h>
|
|
#include <ntddcnet.h>
|
|
#include <cnettest.h>
|
|
#include <cnetapi.h>
|
|
#include <clusrtl.h>
|
|
#include <winsock2.h>
|
|
#include <tdi.h>
|
|
#include <align.h>
|
|
|
|
|
|
//
|
|
// Private Support Routines.
|
|
//
|
|
static NTSTATUS
|
|
OpenDevice(
|
|
HANDLE *Handle,
|
|
LPWSTR DeviceName,
|
|
ULONG ShareAccess
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function opens a specified IO device.
|
|
|
|
Arguments:
|
|
|
|
Handle - pointer to location where the opened device Handle is
|
|
returned.
|
|
|
|
DriverName - name of the device to be opened.
|
|
|
|
Return Value:
|
|
|
|
Windows Error Code.
|
|
|
|
--*/
|
|
{
|
|
OBJECT_ATTRIBUTES objectAttributes;
|
|
IO_STATUS_BLOCK ioStatusBlock;
|
|
UNICODE_STRING nameString;
|
|
NTSTATUS status;
|
|
|
|
*Handle = NULL;
|
|
|
|
//
|
|
// Open a Handle to the device.
|
|
//
|
|
|
|
RtlInitUnicodeString(&nameString, DeviceName);
|
|
|
|
InitializeObjectAttributes(
|
|
&objectAttributes,
|
|
&nameString,
|
|
OBJ_CASE_INSENSITIVE,
|
|
(HANDLE) NULL,
|
|
(PSECURITY_DESCRIPTOR) NULL
|
|
);
|
|
|
|
|
|
status = NtCreateFile(
|
|
Handle,
|
|
SYNCHRONIZE | FILE_READ_DATA | FILE_WRITE_DATA,
|
|
&objectAttributes,
|
|
&ioStatusBlock,
|
|
NULL,
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
ShareAccess,
|
|
FILE_OPEN_IF,
|
|
0,
|
|
NULL,
|
|
0
|
|
);
|
|
|
|
return(status);
|
|
|
|
} // OpenDevice
|
|
|
|
|
|
NTSTATUS
|
|
DoIoctl(
|
|
IN HANDLE Handle,
|
|
IN DWORD IoctlCode,
|
|
IN PVOID Request,
|
|
IN DWORD RequestSize,
|
|
IN PVOID Response,
|
|
IN OUT PDWORD ResponseSize,
|
|
IN LPOVERLAPPED Overlapped
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Packages and issues an ioctl.
|
|
|
|
Arguments:
|
|
|
|
Handle - An open file Handle on which to issue the request.
|
|
|
|
IoctlCode - The IOCTL opcode.
|
|
|
|
Request - A pointer to the input buffer.
|
|
|
|
RequestSize - Size of the input buffer.
|
|
|
|
Response - A pointer to the output buffer.
|
|
|
|
ResponseSize - On input, the size in bytes of the output buffer.
|
|
On output, the number of bytes returned in the output buffer.
|
|
|
|
Return Value:
|
|
|
|
NT Status Code.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
|
|
|
|
if (ARGUMENT_PRESENT(Overlapped)) {
|
|
Overlapped->Internal = (ULONG_PTR) STATUS_PENDING;
|
|
|
|
status = NtDeviceIoControlFile(
|
|
Handle,
|
|
Overlapped->hEvent,
|
|
NULL,
|
|
(((DWORD_PTR) Overlapped->hEvent) & 1) ? NULL : Overlapped,
|
|
(PIO_STATUS_BLOCK) &(Overlapped->Internal),
|
|
IoctlCode,
|
|
Request,
|
|
RequestSize,
|
|
Response,
|
|
*ResponseSize
|
|
);
|
|
|
|
}
|
|
else {
|
|
IO_STATUS_BLOCK ioStatusBlock = {0, 0};
|
|
HANDLE event = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
|
|
if (event != NULL) {
|
|
|
|
//
|
|
// Prevent operation from completing to a completion port.
|
|
//
|
|
event = (HANDLE) (((ULONG_PTR) event) | 1);
|
|
|
|
status = NtDeviceIoControlFile(
|
|
Handle,
|
|
event,
|
|
NULL,
|
|
NULL,
|
|
&ioStatusBlock,
|
|
IoctlCode,
|
|
Request,
|
|
RequestSize,
|
|
Response,
|
|
*ResponseSize
|
|
);
|
|
|
|
if (status == STATUS_PENDING) {
|
|
status = NtWaitForSingleObject(
|
|
event,
|
|
TRUE,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
status = ioStatusBlock.Status;
|
|
|
|
// NOTENOTE: on 64 bit this truncates might want to add > code
|
|
|
|
*ResponseSize = (ULONG)ioStatusBlock.Information;
|
|
}
|
|
else {
|
|
*ResponseSize = 0;
|
|
}
|
|
|
|
CloseHandle(event);
|
|
}
|
|
else {
|
|
status = GetLastError();
|
|
}
|
|
}
|
|
|
|
return(status);
|
|
|
|
} // DoIoctl
|
|
|
|
|
|
#define FACILITY_CODE_MASK 0x0FFF0000
|
|
#define FACILITY_CODE_SHIFT 16
|
|
#define SHIFTED_FACILITY_CLUSTER (FACILITY_CLUSTER_ERROR_CODE << FACILITY_CODE_SHIFT)
|
|
|
|
|
|
DWORD
|
|
NtStatusToClusnetError(
|
|
NTSTATUS Status
|
|
)
|
|
{
|
|
DWORD dosStatus;
|
|
|
|
if ( !((Status & FACILITY_CODE_MASK) == SHIFTED_FACILITY_CLUSTER) ) {
|
|
dosStatus = RtlNtStatusToDosError(Status);
|
|
}
|
|
else {
|
|
//dosStatus = (DWORD) Status;
|
|
switch ( Status ) {
|
|
|
|
case STATUS_CLUSTER_INVALID_NODE:
|
|
dosStatus = ERROR_CLUSTER_INVALID_NODE;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_EXISTS:
|
|
dosStatus = ERROR_CLUSTER_NODE_EXISTS;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_JOIN_IN_PROGRESS:
|
|
dosStatus = ERROR_CLUSTER_JOIN_IN_PROGRESS;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_NOT_FOUND:
|
|
dosStatus = ERROR_CLUSTER_NODE_NOT_FOUND;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_LOCAL_NODE_NOT_FOUND:
|
|
dosStatus = ERROR_CLUSTER_LOCAL_NODE_NOT_FOUND;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETWORK_EXISTS:
|
|
dosStatus = ERROR_CLUSTER_NETWORK_EXISTS;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETWORK_NOT_FOUND:
|
|
dosStatus = ERROR_CLUSTER_NETWORK_NOT_FOUND;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETINTERFACE_EXISTS:
|
|
dosStatus = ERROR_CLUSTER_NETINTERFACE_EXISTS;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETINTERFACE_NOT_FOUND:
|
|
dosStatus =ERROR_CLUSTER_NETINTERFACE_NOT_FOUND;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_INVALID_REQUEST:
|
|
dosStatus = ERROR_CLUSTER_INVALID_REQUEST;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_INVALID_NETWORK_PROVIDER:
|
|
dosStatus = ERROR_CLUSTER_INVALID_NETWORK_PROVIDER;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_DOWN:
|
|
dosStatus = ERROR_CLUSTER_NODE_DOWN;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_UNREACHABLE:
|
|
dosStatus = ERROR_CLUSTER_NODE_UNREACHABLE;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_NOT_MEMBER:
|
|
dosStatus = ERROR_CLUSTER_NODE_NOT_MEMBER;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_JOIN_NOT_IN_PROGRESS:
|
|
dosStatus = ERROR_CLUSTER_JOIN_NOT_IN_PROGRESS;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_INVALID_NETWORK:
|
|
dosStatus = ERROR_CLUSTER_INVALID_NETWORK;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_UP:
|
|
dosStatus = ERROR_CLUSTER_NODE_UP;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_NOT_PAUSED:
|
|
dosStatus = ERROR_CLUSTER_NODE_NOT_PAUSED;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NO_SECURITY_CONTEXT:
|
|
dosStatus = ERROR_CLUSTER_NO_SECURITY_CONTEXT;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETWORK_NOT_INTERNAL:
|
|
dosStatus = ERROR_CLUSTER_NETWORK_NOT_INTERNAL;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_ALREADY_UP:
|
|
dosStatus = ERROR_CLUSTER_NODE_ALREADY_UP;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_ALREADY_DOWN:
|
|
dosStatus = ERROR_CLUSTER_NODE_ALREADY_DOWN;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETWORK_ALREADY_ONLINE:
|
|
dosStatus = ERROR_CLUSTER_NETWORK_ALREADY_ONLINE;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NETWORK_ALREADY_OFFLINE:
|
|
dosStatus = ERROR_CLUSTER_NETWORK_ALREADY_OFFLINE;
|
|
break;
|
|
|
|
case STATUS_CLUSTER_NODE_ALREADY_MEMBER:
|
|
dosStatus = ERROR_CLUSTER_NODE_ALREADY_MEMBER;
|
|
break;
|
|
|
|
default:
|
|
dosStatus = (DWORD)Status;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return(dosStatus);
|
|
}
|
|
|
|
|
|
//
|
|
// Public Routines
|
|
//
|
|
HANDLE
|
|
ClusnetOpenControlChannel(
|
|
IN ULONG ShareAccess
|
|
)
|
|
{
|
|
HANDLE handle = NULL;
|
|
DWORD status;
|
|
|
|
status = OpenDevice(&handle, L"\\Device\\ClusterNetwork", ShareAccess);
|
|
|
|
if (status != ERROR_SUCCESS) {
|
|
SetLastError(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
return(handle);
|
|
|
|
} // ClusnetOpenControlChannel
|
|
|
|
|
|
DWORD
|
|
ClusnetEnableShutdownOnClose(
|
|
IN HANDLE ControlChannel
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
ULONG responseSize = 0;
|
|
CLUSNET_SHUTDOWN_ON_CLOSE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
|
|
request.ProcessId = GetCurrentProcessId();
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_ENABLE_SHUTDOWN_ON_CLOSE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetEnableShutdownOnClose
|
|
|
|
|
|
DWORD
|
|
ClusnetDisableShutdownOnClose(
|
|
IN HANDLE ControlChannel
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
ULONG responseSize = 0;
|
|
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_DISABLE_SHUTDOWN_ON_CLOSE,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetEnableShutdownOnClose
|
|
|
|
|
|
DWORD
|
|
ClusnetInitialize(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID LocalNodeId,
|
|
IN ULONG MaxNodes,
|
|
IN CLUSNET_NODE_UP_ROUTINE NodeUpRoutine,
|
|
IN CLUSNET_NODE_DOWN_ROUTINE NodeDownRoutine,
|
|
IN CLUSNET_CHECK_QUORUM_ROUTINE CheckQuorumRoutine,
|
|
IN CLUSNET_HOLD_IO_ROUTINE HoldIoRoutine,
|
|
IN CLUSNET_RESUME_IO_ROUTINE ResumeIoRoutine,
|
|
IN CLUSNET_HALT_ROUTINE HaltRoutine
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CLUSNET_INITIALIZE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.LocalNodeId = LocalNodeId;
|
|
request.MaxNodes = MaxNodes;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_INITIALIZE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetInitialize
|
|
|
|
|
|
DWORD
|
|
ClusnetShutdown(
|
|
IN HANDLE ControlChannel
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
DWORD requestSize = 0;
|
|
DWORD responseSize = 0;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_SHUTDOWN,
|
|
NULL,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetShutdown
|
|
|
|
|
|
DWORD
|
|
ClusnetRegisterNode(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_NODE_REG_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_REGISTER_NODE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetRegisterNode
|
|
|
|
|
|
DWORD
|
|
ClusnetDeregisterNode(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_NODE_DEREG_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_DEREGISTER_NODE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetDeregisterNode
|
|
|
|
|
|
DWORD
|
|
ClusnetRegisterNetwork(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN ULONG Priority,
|
|
IN BOOLEAN Restricted
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_NETWORK_REG_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NetworkId;
|
|
request.Priority = Priority;
|
|
request.Restricted = Restricted;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_REGISTER_NETWORK,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetRegisterNetwork
|
|
|
|
|
|
DWORD
|
|
ClusnetDeregisterNetwork(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_NETWORK_DEREG_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_DEREGISTER_NETWORK,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetDeregisterNetwork
|
|
|
|
|
|
DWORD
|
|
ClusnetRegisterInterface(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN ULONG Priority,
|
|
IN PWSTR AdapterId,
|
|
IN ULONG AdapterIdLength,
|
|
IN PVOID TdiAddress,
|
|
IN ULONG TdiAddressLength,
|
|
OUT PULONG MediaStatus
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Registers a node's interface on a network.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
NodeId - The ID of the node for which to register the interface.
|
|
|
|
NetworkId - The ID of the network for which to register the interface.
|
|
|
|
Priority - The priority value assigned to the interface. If a value of
|
|
zero is specified, the interface will inherit its priority
|
|
from the network.
|
|
|
|
AdapterId - ID of adapter associated with interface
|
|
|
|
AdapterIdLength - Length of buffer holding adapter ID, not including
|
|
terminating UNICODE_NULL character
|
|
|
|
TdiAddress - A pointer to a TDI TRANSPORT_ADDRESS structure containing
|
|
the transport address of the interface.
|
|
|
|
TdiAddressLength - The length, in bytes, of the TdiAddress buffer.
|
|
|
|
MediaStatus - returned current status of media (e.g. cable disconnected)
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_INTERFACE_REG_REQUEST request;
|
|
DWORD requestSize;
|
|
CX_INTERFACE_REG_RESPONSE response;
|
|
DWORD responseSize
|
|
= sizeof(CX_INTERFACE_REG_RESPONSE);
|
|
DWORD adapterIdOffset;
|
|
|
|
|
|
// calculate the size of the request structure without the adapter
|
|
// id string.
|
|
requestSize = FIELD_OFFSET(CX_INTERFACE_REG_REQUEST, TdiAddress) +
|
|
TdiAddressLength;
|
|
|
|
// round request to type alignment for adapter id string
|
|
requestSize = ROUND_UP_COUNT(requestSize, TYPE_ALIGNMENT(PWSTR));
|
|
|
|
// add buffer for interface name. null-terminate to be safe.
|
|
if (AdapterId == NULL) {
|
|
AdapterIdLength = 0;
|
|
}
|
|
adapterIdOffset = requestSize;
|
|
requestSize += AdapterIdLength + sizeof(UNICODE_NULL);
|
|
|
|
if (requestSize < sizeof(CX_INTERFACE_REG_REQUEST)) {
|
|
requestSize = sizeof(CX_INTERFACE_REG_REQUEST);
|
|
}
|
|
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
ZeroMemory(request, requestSize);
|
|
|
|
request->NodeId = NodeId;
|
|
request->NetworkId = NetworkId;
|
|
request->Priority = Priority;
|
|
request->TdiAddressLength = TdiAddressLength;
|
|
|
|
MoveMemory(
|
|
&(request->TdiAddress[0]),
|
|
TdiAddress,
|
|
TdiAddressLength
|
|
);
|
|
|
|
request->AdapterIdLength = AdapterIdLength;
|
|
request->AdapterIdOffset = adapterIdOffset;
|
|
|
|
if (AdapterId != NULL) {
|
|
CopyMemory(
|
|
(PUWSTR)((PUCHAR)request + adapterIdOffset),
|
|
AdapterId,
|
|
AdapterIdLength
|
|
);
|
|
}
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_REGISTER_INTERFACE,
|
|
request,
|
|
requestSize,
|
|
&response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
LocalFree(request);
|
|
|
|
if (MediaStatus != NULL) {
|
|
*MediaStatus = response.MediaStatus;
|
|
}
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetRegisterInterface
|
|
|
|
|
|
DWORD
|
|
ClusnetDeregisterInterface(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_INTERFACE_DEREG_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.NodeId = NodeId;
|
|
request.NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_DEREGISTER_INTERFACE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetDeregisterInterface
|
|
|
|
|
|
DWORD
|
|
ClusnetOnlineNodeComm(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Enables communication to the specified node.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open control channel handle to the Cluster Network
|
|
driver.
|
|
|
|
NodeId - The ID of the node to which to enable communication.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_ONLINE_NODE_COMM_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_ONLINE_NODE_COMM,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetOnlineNodeCommunication
|
|
|
|
|
|
DWORD
|
|
ClusnetOfflineNodeComm(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Disable communication to the specified node.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open control channel handle to the Cluster Network
|
|
driver.
|
|
|
|
NodeId - The ID of the node to which to disable communication.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_OFFLINE_NODE_COMM_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_OFFLINE_NODE_COMM,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetOfflineNodeCommunication
|
|
|
|
|
|
DWORD
|
|
ClusnetOnlineNetwork(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN PWCHAR TdiProviderName,
|
|
IN PVOID TdiBindAddress,
|
|
IN ULONG TdiBindAddressLength,
|
|
IN LPWSTR AdapterName,
|
|
OUT PVOID TdiBindAddressInfo,
|
|
IN PULONG TdiBindAddressInfoLength
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Brings a cluster network online using the specified TDI transport
|
|
provider and local TDI transport address.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
NetworkId - The ID of the network to bring online.
|
|
|
|
TdiProviderName - The name of the transport provider device that
|
|
this network should open (e.g. \Device\Udp).
|
|
|
|
TdiAddress - A pointer to a TDI TRANSPORT_ADDRESS structure containing
|
|
the transport address of the local interface to which
|
|
the network should be bound.
|
|
|
|
TdiAddressLength - The length, in bytes, of the TdiAddress buffer.
|
|
|
|
AdapterName - name of the adapter on which this network is associated
|
|
|
|
TdiBindAddressInfo - A pointer to a TDI_ADDRESS_INFO structure. On output,
|
|
this structure contains the actual address that
|
|
the provider opened.
|
|
|
|
TdiBindAddressInfoLength - On input, a pointer to the size, in bytes,
|
|
of the TdiBindAddressInfo parameter. On
|
|
output, the variable is updated to the
|
|
amount of date returned in the
|
|
TdiBindAddressInfo structure.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_ONLINE_NETWORK_REQUEST request;
|
|
DWORD requestSize;
|
|
PVOID response;
|
|
ULONG tdiProviderNameLength;
|
|
ULONG adapterNameLength;
|
|
|
|
|
|
tdiProviderNameLength = (wcslen(TdiProviderName) + 1) * sizeof(WCHAR);
|
|
adapterNameLength = (wcslen(AdapterName) + 1) * sizeof(WCHAR);
|
|
|
|
//
|
|
// The request size is based on the size and required alignment
|
|
// of each field of data following the structure.
|
|
//
|
|
requestSize = sizeof(CX_ONLINE_NETWORK_REQUEST);
|
|
|
|
// Provider Name
|
|
requestSize = ROUND_UP_COUNT(requestSize, TYPE_ALIGNMENT(PWSTR))
|
|
+ tdiProviderNameLength;
|
|
|
|
// Bind Address
|
|
requestSize = ROUND_UP_COUNT(requestSize, TYPE_ALIGNMENT(PWSTR))
|
|
+ TdiBindAddressLength;
|
|
|
|
// Adapter Name
|
|
requestSize = ROUND_UP_COUNT(requestSize, TYPE_ALIGNMENT(PWSTR))
|
|
+ adapterNameLength;
|
|
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->Id = NetworkId;
|
|
request->TdiProviderNameLength = tdiProviderNameLength;
|
|
request->TdiProviderNameOffset =
|
|
ROUND_UP_COUNT(sizeof(CX_ONLINE_NETWORK_REQUEST),
|
|
TYPE_ALIGNMENT(PWSTR));
|
|
|
|
MoveMemory(
|
|
(((PUCHAR) request) + request->TdiProviderNameOffset),
|
|
TdiProviderName,
|
|
tdiProviderNameLength
|
|
);
|
|
|
|
request->TdiBindAddressLength = TdiBindAddressLength;
|
|
request->TdiBindAddressOffset =
|
|
ROUND_UP_COUNT((request->TdiProviderNameOffset +
|
|
tdiProviderNameLength),
|
|
TYPE_ALIGNMENT(TRANSPORT_ADDRESS));
|
|
|
|
|
|
MoveMemory(
|
|
(((PUCHAR) request) + request->TdiBindAddressOffset),
|
|
TdiBindAddress,
|
|
TdiBindAddressLength
|
|
);
|
|
|
|
request->AdapterNameLength = adapterNameLength;
|
|
request->AdapterNameOffset =
|
|
ROUND_UP_COUNT((request->TdiBindAddressOffset +
|
|
TdiBindAddressLength),
|
|
TYPE_ALIGNMENT(PWSTR));
|
|
|
|
MoveMemory(
|
|
(((PUCHAR) request) + request->AdapterNameOffset),
|
|
AdapterName,
|
|
adapterNameLength
|
|
);
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_ONLINE_NETWORK,
|
|
request,
|
|
requestSize,
|
|
TdiBindAddressInfo,
|
|
TdiBindAddressInfoLength,
|
|
NULL
|
|
);
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetOnlineNetwork
|
|
|
|
|
|
DWORD
|
|
ClusnetOfflineNetwork(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_OFFLINE_NETWORK_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_OFFLINE_NETWORK,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetOfflineNetwork
|
|
|
|
|
|
DWORD
|
|
ClusnetSetNetworkRestriction(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN BOOLEAN Restricted,
|
|
IN ULONG NewPriority
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SET_NETWORK_RESTRICTION_REQUEST request;
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NetworkId;
|
|
request.Restricted = Restricted;
|
|
request.NewPriority = NewPriority;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SET_NETWORK_RESTRICTION,
|
|
&request,
|
|
sizeof(CX_SET_NETWORK_RESTRICTION_REQUEST),
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetSetNetworkRestriction
|
|
|
|
|
|
DWORD
|
|
ClusnetGetNetworkPriority(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
OUT PULONG Priority
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_GET_NETWORK_PRIORITY_REQUEST request;
|
|
PCX_GET_NETWORK_PRIORITY_RESPONSE response;
|
|
DWORD requestSize;
|
|
DWORD responseSize;
|
|
|
|
|
|
requestSize = sizeof(CX_GET_NETWORK_PRIORITY_REQUEST);
|
|
responseSize = sizeof(CX_GET_NETWORK_PRIORITY_RESPONSE);
|
|
|
|
if (requestSize > responseSize) {
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
response = (PCX_GET_NETWORK_PRIORITY_RESPONSE) request;
|
|
}
|
|
else {
|
|
response = LocalAlloc(LMEM_FIXED, responseSize);
|
|
request = (PCX_GET_NETWORK_PRIORITY_REQUEST) response;
|
|
}
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->Id = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_NETWORK_PRIORITY,
|
|
request,
|
|
requestSize,
|
|
response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
if (responseSize != sizeof(CX_GET_NETWORK_PRIORITY_RESPONSE)) {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
else {
|
|
*Priority = response->Priority;
|
|
}
|
|
}
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetNetworkPriority
|
|
|
|
|
|
DWORD
|
|
ClusnetSetNetworkPriority(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN ULONG Priority
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SET_NETWORK_PRIORITY_REQUEST request;
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.Id = NetworkId;
|
|
request.Priority = Priority;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SET_NETWORK_PRIORITY,
|
|
&request,
|
|
sizeof(CX_SET_NETWORK_PRIORITY_REQUEST),
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetSetNetworkPriority
|
|
|
|
|
|
DWORD
|
|
ClusnetGetInterfacePriority(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
OUT PULONG InterfacePriority,
|
|
OUT PULONG NetworkPriority
|
|
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_GET_INTERFACE_PRIORITY_REQUEST request;
|
|
PCX_GET_INTERFACE_PRIORITY_RESPONSE response;
|
|
DWORD requestSize;
|
|
DWORD responseSize;
|
|
|
|
|
|
requestSize = sizeof(CX_GET_INTERFACE_PRIORITY_REQUEST);
|
|
responseSize = sizeof(CX_GET_INTERFACE_PRIORITY_RESPONSE);
|
|
|
|
if (requestSize > responseSize) {
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
response = (PCX_GET_INTERFACE_PRIORITY_RESPONSE) request;
|
|
}
|
|
else {
|
|
response = LocalAlloc(LMEM_FIXED, responseSize);
|
|
request = (PCX_GET_INTERFACE_PRIORITY_REQUEST) response;
|
|
}
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->NodeId = NodeId;
|
|
request->NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_INTERFACE_PRIORITY,
|
|
request,
|
|
requestSize,
|
|
response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
if (responseSize != sizeof(CX_GET_INTERFACE_PRIORITY_RESPONSE)) {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
else {
|
|
*InterfacePriority = response->InterfacePriority;
|
|
*NetworkPriority = response->NetworkPriority;
|
|
}
|
|
}
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetInterfacePriority
|
|
|
|
|
|
DWORD
|
|
ClusnetSetInterfacePriority(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN ULONG Priority
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SET_INTERFACE_PRIORITY_REQUEST request;
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.NodeId = NodeId;
|
|
request.NetworkId = NetworkId;
|
|
request.Priority = Priority;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SET_INTERFACE_PRIORITY,
|
|
&request,
|
|
sizeof(CX_SET_INTERFACE_PRIORITY_REQUEST),
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetInterfacePriority
|
|
|
|
|
|
DWORD
|
|
ClusnetGetNodeCommState(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
OUT PCLUSNET_NODE_COMM_STATE State
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_GET_NODE_STATE_REQUEST request;
|
|
PCX_GET_NODE_STATE_RESPONSE response;
|
|
DWORD requestSize;
|
|
DWORD responseSize;
|
|
|
|
|
|
requestSize = sizeof(CX_GET_NODE_STATE_REQUEST);
|
|
responseSize = sizeof(CX_GET_NODE_STATE_RESPONSE);
|
|
|
|
if (requestSize > responseSize) {
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
response = (PCX_GET_NODE_STATE_RESPONSE) request;
|
|
}
|
|
else {
|
|
response = LocalAlloc(LMEM_FIXED, responseSize);
|
|
request = (PCX_GET_NODE_STATE_REQUEST) response;
|
|
}
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_NODE_STATE,
|
|
request,
|
|
requestSize,
|
|
response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
if (responseSize != sizeof(CX_GET_NODE_STATE_RESPONSE)) {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
else {
|
|
*State = response->State;
|
|
}
|
|
}
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetNodeState
|
|
|
|
|
|
DWORD
|
|
ClusnetGetNetworkState(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
OUT PCLUSNET_NETWORK_STATE State
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_GET_NETWORK_STATE_REQUEST request;
|
|
PCX_GET_NETWORK_STATE_RESPONSE response;
|
|
DWORD requestSize;
|
|
DWORD responseSize;
|
|
|
|
|
|
requestSize = sizeof(CX_GET_NETWORK_STATE_REQUEST);
|
|
responseSize = sizeof(CX_GET_NETWORK_STATE_RESPONSE);
|
|
|
|
if (requestSize > responseSize) {
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
response = (PCX_GET_NETWORK_STATE_RESPONSE) request;
|
|
}
|
|
else {
|
|
response = LocalAlloc(LMEM_FIXED, responseSize);
|
|
request = (PCX_GET_NETWORK_STATE_REQUEST) response;
|
|
}
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->Id = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_NETWORK_STATE,
|
|
request,
|
|
requestSize,
|
|
response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
if (responseSize != sizeof(CX_GET_NETWORK_STATE_RESPONSE)) {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
else {
|
|
*State = response->State;
|
|
}
|
|
}
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetNetworkState
|
|
|
|
|
|
DWORD
|
|
ClusnetGetInterfaceState(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
OUT PCLUSNET_INTERFACE_STATE State
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_GET_INTERFACE_STATE_REQUEST request;
|
|
PCX_GET_INTERFACE_STATE_RESPONSE response;
|
|
DWORD requestSize;
|
|
DWORD responseSize;
|
|
|
|
|
|
requestSize = sizeof(CX_GET_INTERFACE_STATE_REQUEST);
|
|
responseSize = sizeof(CX_GET_INTERFACE_STATE_RESPONSE);
|
|
|
|
if (requestSize > responseSize) {
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
response = (PCX_GET_INTERFACE_STATE_RESPONSE) request;
|
|
}
|
|
else {
|
|
response = LocalAlloc(LMEM_FIXED, responseSize);
|
|
request = (PCX_GET_INTERFACE_STATE_REQUEST) response;
|
|
}
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->NodeId = NodeId;
|
|
request->NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_INTERFACE_STATE,
|
|
request,
|
|
requestSize,
|
|
response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
if (responseSize != sizeof(CX_GET_INTERFACE_STATE_RESPONSE)) {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
else {
|
|
*State = response->State;
|
|
}
|
|
}
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetInterfaceState
|
|
|
|
|
|
#ifdef MM_IN_CLUSNSET
|
|
|
|
DWORD
|
|
ClusnetFormCluster(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG ClockPeriod,
|
|
IN ULONG SendHBRate,
|
|
IN ULONG RecvHBRate
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CMM_FORM_CLUSTER_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.ClockPeriod = ClockPeriod;
|
|
request.SendHBRate = SendHBRate;
|
|
request.RecvHBRate = RecvHBRate;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CMM_FORM_CLUSTER,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetFormCluster
|
|
|
|
|
|
DWORD
|
|
ClusnetJoinCluster(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID JoiningNodeId,
|
|
IN CLUSNET_JOIN_PHASE Phase,
|
|
IN ULONG JoinTimeout,
|
|
IN OUT PVOID * MessageToSend,
|
|
OUT PULONG MessageLength,
|
|
OUT PULONG DestNodeMask
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CMM_JOIN_CLUSTER_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
PCMM_JOIN_CLUSTER_RESPONSE response;
|
|
ULONG IoctlCode;
|
|
DWORD responseSize;
|
|
|
|
|
|
//
|
|
// Parse the input parameters
|
|
//
|
|
if ( Phase == ClusnetJoinPhase1 )
|
|
IoctlCode = IOCTL_CMM_JOIN_CLUSTER_PHASE1;
|
|
else if ( Phase == ClusnetJoinPhase2 )
|
|
IoctlCode = IOCTL_CMM_JOIN_CLUSTER_PHASE2;
|
|
else if ( Phase == ClusnetJoinPhase3 )
|
|
IoctlCode = IOCTL_CMM_JOIN_CLUSTER_PHASE3;
|
|
else if ( Phase == ClusnetJoinPhase4 )
|
|
IoctlCode = IOCTL_CMM_JOIN_CLUSTER_PHASE4;
|
|
else if ( Phase == ClusnetJoinPhaseAbort )
|
|
IoctlCode = IOCTL_CMM_JOIN_CLUSTER_ABORT;
|
|
else
|
|
return(ERROR_INVALID_PARAMETER);
|
|
|
|
|
|
request.JoiningNode = JoiningNodeId;
|
|
request.JoinTimeout = JoinTimeout;
|
|
|
|
//
|
|
// allocate space for the response buffer and a message space at the back
|
|
// of the struct. Current RGP message requirements are 80 bytes
|
|
// (sizeof(rgp_msgbuf)).
|
|
//
|
|
|
|
responseSize = sizeof(*response) + 200;
|
|
|
|
if (*MessageToSend != NULL) {
|
|
//
|
|
// recycle old message buffer
|
|
//
|
|
response = CONTAINING_RECORD(
|
|
*MessageToSend,
|
|
CMM_JOIN_CLUSTER_RESPONSE,
|
|
SendData
|
|
);
|
|
}
|
|
else {
|
|
response = LocalAlloc(LMEM_FIXED, responseSize);
|
|
}
|
|
|
|
if ( response == NULL ) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IoctlCode,
|
|
&request,
|
|
requestSize,
|
|
response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (NT_SUCCESS(status)) {
|
|
*MessageToSend = &(response->SendData[0]);
|
|
*MessageLength = response->SizeOfSendData;
|
|
*DestNodeMask = response->SendNodeMask;
|
|
|
|
return(ERROR_SUCCESS);
|
|
}
|
|
|
|
LocalFree( response );
|
|
*MessageToSend = NULL;
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetJoinCluster
|
|
|
|
|
|
VOID
|
|
ClusnetEndJoinCluster(
|
|
IN HANDLE ControlChannel,
|
|
IN PVOID LastSentMessage
|
|
)
|
|
{
|
|
ULONG responseSize = 0;
|
|
PCMM_JOIN_CLUSTER_RESPONSE response;
|
|
|
|
|
|
if (LastSentMessage != NULL) {
|
|
response = CONTAINING_RECORD(
|
|
LastSentMessage,
|
|
CMM_JOIN_CLUSTER_RESPONSE,
|
|
SendData
|
|
);
|
|
|
|
LocalFree(response);
|
|
}
|
|
|
|
(VOID) DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CMM_JOIN_CLUSTER_END,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return;
|
|
|
|
} // ClusnetEndJoinCluster
|
|
|
|
|
|
DWORD
|
|
ClusnetDeliverJoinMessage(
|
|
IN HANDLE ControlChannel,
|
|
IN PVOID Message,
|
|
IN ULONG MessageLength
|
|
)
|
|
{
|
|
NTSTATUS status;
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CMM_DELIVER_JOIN_MESSAGE,
|
|
Message,
|
|
MessageLength,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetDeliverJoinMessage
|
|
|
|
DWORD
|
|
ClusnetLeaveCluster(
|
|
IN HANDLE ControlChannel
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CMM_LEAVE_CLUSTER,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetLeaveCluster
|
|
|
|
|
|
DWORD
|
|
ClusnetEvictNode(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG NodeId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CMM_EJECT_CLUSTER_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
request.Node = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CMM_EJECT_CLUSTER,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetEvictNode
|
|
|
|
|
|
#endif // MM_IN_CLUSNSET
|
|
|
|
DWORD
|
|
ClusnetGetNodeMembershipState(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG NodeId,
|
|
OUT CLUSNET_NODE_STATE * State
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_GET_NODE_MMSTATE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
CX_GET_NODE_MMSTATE_RESPONSE response;
|
|
DWORD responseSize = sizeof(response);
|
|
|
|
request.Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_NODE_MMSTATE,
|
|
&request,
|
|
requestSize,
|
|
&response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
|
|
*State = response.State;
|
|
}
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetNodeMembershipState
|
|
|
|
DWORD
|
|
ClusnetSetNodeMembershipState(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG NodeId,
|
|
IN CLUSNET_NODE_STATE State
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Set the internal node membership state to the indicated value
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SET_NODE_MMSTATE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize;
|
|
|
|
request.NodeId = NodeId;
|
|
request.State = State;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SET_NODE_MMSTATE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetSetNodeMembershipState
|
|
|
|
DWORD
|
|
ClusnetSetEventMask(
|
|
IN HANDLE ControlChannel,
|
|
IN CLUSNET_EVENT_TYPE EventMask
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Based on the supplied callback pointers, set the mask of events
|
|
generated in kernel mode in which this file handle is interested
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
EventMask - bit mask of interested events
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CLUSNET_SET_EVENT_MASK_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
request.EventMask = EventMask;
|
|
request.KmodeEventCallback = NULL;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_SET_EVENT_MASK,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetSetEventMask
|
|
|
|
|
|
DWORD
|
|
ClusnetGetNextEvent(
|
|
IN HANDLE ControlChannel,
|
|
OUT PCLUSNET_EVENT Event,
|
|
IN LPOVERLAPPED Overlapped OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Wait for the next event to be completed.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
Event - handle to event that is set when IO is complete
|
|
Response - pointer to structure that is filled in when IRP completes
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
ULONG ResponseSize = sizeof( CLUSNET_EVENT );
|
|
|
|
//
|
|
// if no event passed in, then assume the caller wants to block.
|
|
// we still need an event to block on while waiting...
|
|
//
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_GET_NEXT_EVENT,
|
|
NULL,
|
|
0,
|
|
Event,
|
|
&ResponseSize,
|
|
Overlapped
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetNextEvent
|
|
|
|
DWORD
|
|
ClusnetHalt(
|
|
IN HANDLE ControlChannel
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Tell clusnet that we need to halt immediately
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
DWORD responseSize;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_HALT,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetHalt
|
|
|
|
DWORD
|
|
ClusnetSetMemLogging(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG NumberOfEntries
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Turn in-memory logging in clusnet on or off.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
NumberOfEntires - # of entries to allocate for the log. Zero turns off logging
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CLUSNET_SET_MEM_LOGGING_REQUEST request;
|
|
DWORD requestSize = sizeof( request );
|
|
DWORD responseSize;
|
|
|
|
request.NumberOfEntries = NumberOfEntries;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_SET_MEMORY_LOGGING,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetSetMemLogging
|
|
|
|
DWORD
|
|
ClusnetSendPoisonPacket(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Send a poison packet to the indicated node
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SEND_POISON_PKT_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
request.Id = NodeId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SEND_POISON_PACKET,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
DWORD
|
|
ClusnetSetOuterscreen(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG Outerscreen
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
set the cluster member outerscreen
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SET_OUTERSCREEN_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
request.Outerscreen = Outerscreen;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SET_OUTERSCREEN,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
DWORD
|
|
ClusnetRegroupFinished(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG NewEpoch
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
inform clusnet that regroup has finished
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
NewEpoch - new event epoch used to detect stale events
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_REGROUP_FINISHED_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
request.NewEpoch = NewEpoch;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_REGROUP_FINISHED,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
DWORD
|
|
ClusnetImportSecurityContexts(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID JoiningNodeId,
|
|
IN PWCHAR PackageName,
|
|
IN ULONG SignatureSize,
|
|
IN PVOID ServerContext,
|
|
IN PVOID ClientContext
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
inform clusnet that regroup has finished
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
NewEpoch - new event epoch used to detect stale events
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS status;
|
|
CX_IMPORT_SECURITY_CONTEXT_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
request.JoiningNodeId = JoiningNodeId;
|
|
request.PackageName = PackageName;
|
|
request.PackageNameSize = sizeof(WCHAR) * ( wcslen( PackageName ) + 1 );
|
|
request.SignatureSize = SignatureSize;
|
|
request.ServerContext = ServerContext;
|
|
request.ClientContext = ClientContext;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_IMPORT_SECURITY_CONTEXTS,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
DWORD
|
|
ClusnetReserveEndpoint(
|
|
IN HANDLE ControlChannel,
|
|
IN PWSTR EndpointString
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Tell clusnet to tell TCP/IP to reserve the port number in
|
|
EndpointString.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
EndpointString - string containing port number assigned to clusnet
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
USHORT port;
|
|
DWORD err;
|
|
DWORD responseSize = 0;
|
|
NTSTATUS status;
|
|
|
|
err = ClRtlTcpipStringToEndpoint(EndpointString, &port);
|
|
|
|
if (err == ERROR_SUCCESS) {
|
|
|
|
// TCP/IP needs the port in host byte-order format.
|
|
// ClRtlTcpipStringToEndpoint returns it in network
|
|
// byte-order format.
|
|
port = ntohs(port);
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_RESERVE_ENDPOINT,
|
|
&port,
|
|
sizeof(port),
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
err = NtStatusToClusnetError(status);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
DWORD
|
|
ClusnetConfigureMulticast(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
IN ULONG MulticastNetworkBrand,
|
|
IN PVOID MulticastAddress,
|
|
IN ULONG MulticastAddressLength,
|
|
IN PVOID Key,
|
|
IN ULONG KeyLength,
|
|
IN PVOID Salt,
|
|
IN ULONG SaltLength
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Configures multicast parameters for the specified network.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
PCX_CONFIGURE_MULTICAST_REQUEST request;
|
|
DWORD requestSize;
|
|
DWORD responseSize;
|
|
|
|
//
|
|
// The request size is based on the size and required alignment
|
|
// of each field of data following the structure. If there is no
|
|
// data following the structure, only the structure is required.
|
|
//
|
|
requestSize = sizeof(CX_CONFIGURE_MULTICAST_REQUEST);
|
|
|
|
if (MulticastAddressLength != 0) {
|
|
requestSize = ROUND_UP_COUNT(requestSize,
|
|
TYPE_ALIGNMENT(TRANSPORT_ADDRESS)
|
|
) +
|
|
MulticastAddressLength;
|
|
}
|
|
|
|
if (KeyLength != 0) {
|
|
requestSize = ROUND_UP_COUNT(requestSize,
|
|
TYPE_ALIGNMENT(PVOID)
|
|
) +
|
|
KeyLength;
|
|
}
|
|
|
|
if (SaltLength != 0) {
|
|
requestSize = ROUND_UP_COUNT(requestSize,
|
|
TYPE_ALIGNMENT(PVOID)
|
|
) +
|
|
SaltLength;
|
|
}
|
|
|
|
|
|
//
|
|
// Allocate the request buffer.
|
|
//
|
|
request = LocalAlloc(LMEM_FIXED, requestSize);
|
|
|
|
if (request == NULL) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
request->NetworkId = NetworkId;
|
|
request->MulticastNetworkBrand = MulticastNetworkBrand;
|
|
|
|
if (MulticastAddress != NULL) {
|
|
request->MulticastAddress =
|
|
ROUND_UP_COUNT(sizeof(CX_CONFIGURE_MULTICAST_REQUEST),
|
|
TYPE_ALIGNMENT(TRANSPORT_ADDRESS));
|
|
MoveMemory(
|
|
(((PUCHAR) request) + request->MulticastAddress),
|
|
MulticastAddress,
|
|
MulticastAddressLength
|
|
);
|
|
request->MulticastAddressLength = MulticastAddressLength;
|
|
} else {
|
|
request->MulticastAddress = 0;
|
|
request->MulticastAddressLength = 0;
|
|
}
|
|
|
|
if (Key != NULL) {
|
|
request->Key =
|
|
ROUND_UP_COUNT((request->MulticastAddress
|
|
+ request->MulticastAddressLength),
|
|
TYPE_ALIGNMENT(PVOID));
|
|
MoveMemory(
|
|
(((PUCHAR) request) + request->Key),
|
|
Key,
|
|
KeyLength
|
|
);
|
|
request->KeyLength = KeyLength;
|
|
} else {
|
|
request->Key = 0;
|
|
request->KeyLength = 0;
|
|
}
|
|
|
|
if (Salt != NULL) {
|
|
request->Salt =
|
|
ROUND_UP_COUNT((request->Key + request->KeyLength),
|
|
TYPE_ALIGNMENT(PVOID));
|
|
MoveMemory(
|
|
(((PUCHAR) request) + request->Salt),
|
|
Salt,
|
|
SaltLength
|
|
);
|
|
request->SaltLength = SaltLength;
|
|
} else {
|
|
request->Salt = 0;
|
|
request->SaltLength = 0;
|
|
}
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_CONFIGURE_MULTICAST,
|
|
request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
LocalFree(request);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetConfigureMulticast
|
|
|
|
|
|
DWORD
|
|
ClusnetGetMulticastReachableSet(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NETWORK_ID NetworkId,
|
|
OUT ULONG * NodeScreen
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Queries the current set of nodes considered reachable by
|
|
a multicast on the specified network.
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - open clusnet control channel
|
|
|
|
NetworkId - multicast network
|
|
|
|
NodeScreen - mask of nodes
|
|
|
|
Return value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_GET_MULTICAST_REACHABLE_SET_REQUEST request;
|
|
CX_GET_MULTICAST_REACHABLE_SET_RESPONSE response;
|
|
DWORD responseSize = sizeof(response);
|
|
|
|
request.Id = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_GET_MULTICAST_REACHABLE_SET,
|
|
&request,
|
|
sizeof(request),
|
|
&response,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
if (status == STATUS_SUCCESS) {
|
|
|
|
*NodeScreen = response.NodeScreen;
|
|
}
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
|
|
} // ClusnetGetMulticastReachableSet
|
|
|
|
|
|
#if DBG
|
|
|
|
DWORD
|
|
ClusnetSetDebugMask(
|
|
IN HANDLE ControlChannel,
|
|
IN ULONG Mask
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CLUSNET_SET_DEBUG_MASK_REQUEST request;
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.DebugMask = Mask;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CLUSNET_SET_DEBUG_MASK,
|
|
&request,
|
|
sizeof(CLUSNET_SET_DEBUG_MASK_REQUEST),
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
|
|
DWORD
|
|
ClusnetOnlinePendingInterface(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_ONLINE_PENDING_INTERFACE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.NodeId = NodeId;
|
|
request.NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_ONLINE_PENDING_INTERFACE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
|
|
DWORD
|
|
ClusnetOnlineInterface(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_ONLINE_INTERFACE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.NodeId = NodeId;
|
|
request.NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_ONLINE_INTERFACE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
|
|
DWORD
|
|
ClusnetOfflineInterface(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_OFFLINE_INTERFACE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.NodeId = NodeId;
|
|
request.NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_OFFLINE_INTERFACE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
|
|
DWORD
|
|
ClusnetFailInterface(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN CL_NETWORK_ID NetworkId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_FAIL_INTERFACE_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
|
|
|
|
request.NodeId = NodeId;
|
|
request.NetworkId = NetworkId;
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_FAIL_INTERFACE,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
|
|
DWORD
|
|
ClusnetSendMmMsg(
|
|
IN HANDLE ControlChannel,
|
|
IN CL_NODE_ID NodeId,
|
|
IN ULONG Pattern
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
|
|
Arguments:
|
|
|
|
ControlChannel - An open handle to the Cluster Network control device.
|
|
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if the operation was successful.
|
|
A Windows error code otherwise.
|
|
|
|
--*/
|
|
{
|
|
NTSTATUS status;
|
|
CX_SEND_MM_MSG_REQUEST request;
|
|
DWORD requestSize = sizeof(request);
|
|
DWORD responseSize = 0;
|
|
DWORD i;
|
|
|
|
|
|
request.DestNodeId = NodeId;
|
|
|
|
for (i=0; i < CX_MM_MSG_DATA_LEN; i++) {
|
|
request.MessageData[i] = Pattern;
|
|
}
|
|
|
|
status = DoIoctl(
|
|
ControlChannel,
|
|
IOCTL_CX_SEND_MM_MSG,
|
|
&request,
|
|
requestSize,
|
|
NULL,
|
|
&responseSize,
|
|
NULL
|
|
);
|
|
|
|
return(NtStatusToClusnetError(status));
|
|
}
|
|
|
|
|
|
#endif // DBG
|
|
|
|
|
|
|