454 lines
11 KiB
C
454 lines
11 KiB
C
//+----------------------------------------------------------------------------
|
||
//
|
||
// Copyright (C) 1996, Microsoft Corporation
|
||
//
|
||
// File: dominfo.h
|
||
//
|
||
// Contents: Code to figure out domain dfs addresses
|
||
//
|
||
// Classes: None
|
||
//
|
||
// Functions:
|
||
//
|
||
// History: Feb 7, 1996 Milans created
|
||
//
|
||
//-----------------------------------------------------------------------------
|
||
|
||
#include <nt.h>
|
||
#include <ntrtl.h>
|
||
#include <nturtl.h>
|
||
#include <dfsfsctl.h>
|
||
#include <windows.h>
|
||
|
||
#include <dsgetdc.h>
|
||
#include <dsrole.h>
|
||
#include <wsutil.h>
|
||
|
||
#include "dominfo.h"
|
||
|
||
#define MUP_EVENT_NAME TEXT("wkssvc: MUP finished initializing event")
|
||
|
||
HANDLE
|
||
CreateMupEvent(void);
|
||
|
||
NTSTATUS
|
||
DfsGetDomainNameInfo(void);
|
||
|
||
HANDLE hMupEvent = NULL;
|
||
BOOLEAN MupEventSignaled = FALSE;
|
||
BOOLEAN GotDomainNameInfo = FALSE;
|
||
ULONG DfsDebug = 0;
|
||
|
||
//
|
||
// Commonly used strings and characters
|
||
//
|
||
|
||
#define UNICODE_PATH_SEP_STR L"\\"
|
||
#define UNICODE_PATH_SEP L'\\'
|
||
#define DNS_PATH_SEP L'.'
|
||
|
||
|
||
//+----------------------------------------------------------------------------
|
||
//
|
||
// Function: DfsGetDCName
|
||
//
|
||
// Synopsis: Gets the name of a DC we can use for expanded name referrals
|
||
// It will stick this into the driver.
|
||
//
|
||
// Arguments: [Flags] -- TBD
|
||
//
|
||
// Returns: [STATUS_SUCCESS] -- Successfully created domain pkt entry.
|
||
//
|
||
// [STATUS_INSUFFICIENT_RESOURCES] -- Out of memory condition.
|
||
//
|
||
// [STATUS_OBJECT_NAME_NOT_FOUND] -- wszDomain is not a trusted
|
||
// domain.
|
||
//
|
||
// [STATUS_UNEXPECTED_NETWORK_ERROR] -- Unable to get DC for
|
||
// domain.
|
||
//
|
||
//-----------------------------------------------------------------------------
|
||
|
||
NTSTATUS
|
||
DfsGetDCName(
|
||
IN ULONG Flags,
|
||
BOOLEAN *DcNameFailed)
|
||
{
|
||
NTSTATUS Status;
|
||
HANDLE hDfs;
|
||
ULONG cbSize;
|
||
WCHAR *DCName;
|
||
ULONG dwErr;
|
||
ULONG Len;
|
||
PDOMAIN_CONTROLLER_INFO pDomainControllerInfo;
|
||
|
||
*DcNameFailed = TRUE;
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("DfsGetDCName(%d)\n", Flags);
|
||
#endif
|
||
|
||
if( WsInAWorkgroup() == TRUE ) {
|
||
//
|
||
// We are in a workgroup. We will never find a DC!
|
||
//
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("DfsGetDCName exit STATUS_NO_SUCH_DOMAIN\n");
|
||
#endif
|
||
return STATUS_NO_SUCH_DOMAIN;
|
||
}
|
||
|
||
if (hMupEvent == NULL) {
|
||
|
||
hMupEvent = CreateMupEvent();
|
||
|
||
}
|
||
|
||
dwErr = DsGetDcName(
|
||
NULL, // Computername
|
||
NULL, // DomainName
|
||
NULL, // DomainGuid
|
||
NULL, // SiteGuid
|
||
Flags | DS_DIRECTORY_SERVICE_REQUIRED,
|
||
&pDomainControllerInfo);
|
||
|
||
//
|
||
// If DsGetDcName succeeded, try to get the NetBios & Dns domain names.
|
||
//
|
||
|
||
if (dwErr != NO_ERROR) {
|
||
|
||
if (MupEventSignaled == FALSE) {
|
||
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("Signaling mup event\n");
|
||
#endif
|
||
SetEvent(hMupEvent);
|
||
MupEventSignaled = TRUE;
|
||
|
||
}
|
||
|
||
switch (dwErr) {
|
||
case ERROR_NOT_ENOUGH_MEMORY:
|
||
Status = STATUS_INSUFFICIENT_RESOURCES;
|
||
break;
|
||
case ERROR_NETWORK_UNREACHABLE:
|
||
Status = STATUS_NETWORK_UNREACHABLE;
|
||
break;
|
||
default:
|
||
Status = STATUS_UNEXPECTED_NETWORK_ERROR;
|
||
break;
|
||
}
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("DfsGetDCName(1) exit 0x%x\n", Status);
|
||
#endif
|
||
return (Status);
|
||
} else {
|
||
if (pDomainControllerInfo == NULL) {
|
||
DbgBreakPoint();
|
||
}
|
||
}
|
||
|
||
*DcNameFailed = FALSE;
|
||
|
||
DfsGetDomainNameInfo();
|
||
|
||
//
|
||
// Remove leading '\'s
|
||
//
|
||
DCName = pDomainControllerInfo->DomainControllerName;
|
||
while (*DCName == UNICODE_PATH_SEP && *DCName != UNICODE_NULL)
|
||
DCName++;
|
||
|
||
//
|
||
// Remove trailing '.', if present
|
||
//
|
||
|
||
Len = wcslen(DCName);
|
||
|
||
if (Len >= 1 && DCName[Len-1] == DNS_PATH_SEP) {
|
||
|
||
DCName[Len-1] = UNICODE_NULL;
|
||
|
||
}
|
||
|
||
if (wcslen(DCName) <= 0) {
|
||
NetApiBufferFree(pDomainControllerInfo);
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("DfsGetDCName exit STATUS_UNEXPECTED_NETWORK_ERROR\n");
|
||
#endif
|
||
return (STATUS_UNEXPECTED_NETWORK_ERROR);
|
||
}
|
||
|
||
Status = DfsOpen( &hDfs, NULL );
|
||
|
||
if (!NT_SUCCESS(Status)) {
|
||
NetApiBufferFree(pDomainControllerInfo);
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("DfsGetDCName(2) exit 0x%x\n", Status);
|
||
#endif
|
||
return (Status);
|
||
}
|
||
|
||
//
|
||
// Take the name and fscontrol it down to the driver
|
||
//
|
||
|
||
cbSize = wcslen(DCName) * sizeof(WCHAR) + sizeof(WCHAR);
|
||
|
||
Status = DfsFsctl(
|
||
hDfs,
|
||
FSCTL_DFS_PKT_SET_DC_NAME,
|
||
DCName,
|
||
cbSize,
|
||
NULL,
|
||
0L);
|
||
|
||
NetApiBufferFree(pDomainControllerInfo);
|
||
|
||
//
|
||
// Inform anyone waiting that the mup is ready.
|
||
//
|
||
|
||
if (MupEventSignaled == FALSE) {
|
||
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("Signaling mup event\n");
|
||
#endif
|
||
SetEvent(hMupEvent);
|
||
MupEventSignaled = TRUE;
|
||
|
||
}
|
||
|
||
NtClose( hDfs );
|
||
|
||
#if DBG
|
||
if (DfsDebug)
|
||
DbgPrint("DfsGetDCName(3) exit 0x%x\n", Status);
|
||
#endif
|
||
|
||
return (Status);
|
||
|
||
}
|
||
|
||
//+----------------------------------------------------------------------------
|
||
//
|
||
// Function: DfsGetDomainNameInfo
|
||
//
|
||
// Synopsis: Gets the Netbios & Dns name of the domain, then sends them
|
||
// down to the drvier;
|
||
//
|
||
// Returns: [STATUS_SUCCESS] -- Successfully created domain pkt entry.
|
||
// [other] -- return from DfsOpen or
|
||
// DsRoleGetPrimaryDomainInformation
|
||
//
|
||
//-----------------------------------------------------------------------------
|
||
|
||
NTSTATUS
|
||
DfsGetDomainNameInfo(void)
|
||
{
|
||
NTSTATUS Status;
|
||
ULONG dwErr;
|
||
HANDLE hDfs;
|
||
ULONG cbSize;
|
||
PDSROLE_PRIMARY_DOMAIN_INFO_BASIC pPrimaryDomainInfo;
|
||
|
||
Status = DfsOpen( &hDfs, NULL );
|
||
|
||
if (!NT_SUCCESS(Status)) {
|
||
return (Status);
|
||
}
|
||
|
||
//
|
||
// Get our machine name and type/role.
|
||
//
|
||
|
||
dwErr = DsRoleGetPrimaryDomainInformation(
|
||
NULL,
|
||
DsRolePrimaryDomainInfoBasic,
|
||
(PBYTE *)&pPrimaryDomainInfo);
|
||
|
||
if (dwErr == ERROR_SUCCESS) {
|
||
|
||
if (pPrimaryDomainInfo->DomainNameFlat != NULL) {
|
||
|
||
cbSize = wcslen(pPrimaryDomainInfo->DomainNameFlat) * sizeof(WCHAR) + sizeof(WCHAR);
|
||
|
||
Status = DfsFsctl(
|
||
hDfs,
|
||
FSCTL_DFS_PKT_SET_DOMAINNAMEFLAT,
|
||
pPrimaryDomainInfo->DomainNameFlat,
|
||
cbSize,
|
||
NULL,
|
||
0L);
|
||
|
||
}
|
||
|
||
if (pPrimaryDomainInfo->DomainNameDns != NULL) {
|
||
|
||
cbSize = wcslen(pPrimaryDomainInfo->DomainNameDns) * sizeof(WCHAR) + sizeof(WCHAR);
|
||
|
||
Status = DfsFsctl(
|
||
hDfs,
|
||
FSCTL_DFS_PKT_SET_DOMAINNAMEDNS,
|
||
pPrimaryDomainInfo->DomainNameDns,
|
||
cbSize,
|
||
NULL,
|
||
0L);
|
||
|
||
}
|
||
|
||
DsRoleFreeMemory(pPrimaryDomainInfo);
|
||
|
||
GotDomainNameInfo = TRUE;
|
||
|
||
}
|
||
|
||
NtClose( hDfs );
|
||
return (Status);
|
||
|
||
}
|
||
|
||
UNICODE_STRING LocalDfsName = {
|
||
sizeof(DFS_DRIVER_NAME)-sizeof(UNICODE_NULL),
|
||
sizeof(DFS_DRIVER_NAME)-sizeof(UNICODE_NULL),
|
||
DFS_DRIVER_NAME
|
||
};
|
||
|
||
//+-------------------------------------------------------------------------
|
||
//
|
||
// Function: DfsOpen, public
|
||
//
|
||
// Synopsis:
|
||
//
|
||
// Arguments:
|
||
//
|
||
// Returns:
|
||
//
|
||
//--------------------------------------------------------------------------
|
||
|
||
NTSTATUS
|
||
DfsOpen(
|
||
IN OUT PHANDLE DfsHandle,
|
||
IN PUNICODE_STRING DfsName OPTIONAL
|
||
)
|
||
{
|
||
NTSTATUS status;
|
||
OBJECT_ATTRIBUTES objectAttributes;
|
||
IO_STATUS_BLOCK ioStatus;
|
||
PUNICODE_STRING name;
|
||
|
||
if (ARGUMENT_PRESENT(DfsName)) {
|
||
name = DfsName;
|
||
} else {
|
||
name = &LocalDfsName;
|
||
}
|
||
|
||
InitializeObjectAttributes(
|
||
&objectAttributes,
|
||
name,
|
||
OBJ_CASE_INSENSITIVE,
|
||
NULL,
|
||
NULL
|
||
);
|
||
|
||
status = NtCreateFile(
|
||
DfsHandle,
|
||
SYNCHRONIZE | FILE_WRITE_DATA,
|
||
&objectAttributes,
|
||
&ioStatus,
|
||
NULL,
|
||
FILE_ATTRIBUTE_NORMAL,
|
||
FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
|
||
FILE_OPEN_IF,
|
||
FILE_CREATE_TREE_CONNECTION | FILE_SYNCHRONOUS_IO_NONALERT,
|
||
NULL,
|
||
0);
|
||
|
||
if (NT_SUCCESS(status))
|
||
status = ioStatus.Status;
|
||
|
||
return status;
|
||
}
|
||
|
||
|
||
//+-------------------------------------------------------------------------
|
||
//
|
||
// Function: DfsFsctl, public
|
||
//
|
||
// Synopsis:
|
||
//
|
||
// Arguments:
|
||
//
|
||
// Returns:
|
||
//
|
||
//--------------------------------------------------------------------------
|
||
|
||
NTSTATUS
|
||
DfsFsctl(
|
||
IN HANDLE DfsHandle,
|
||
IN ULONG FsControlCode,
|
||
IN PVOID InputBuffer OPTIONAL,
|
||
IN ULONG InputBufferLength,
|
||
OUT PVOID OutputBuffer OPTIONAL,
|
||
IN ULONG OutputBufferLength
|
||
)
|
||
{
|
||
NTSTATUS status;
|
||
IO_STATUS_BLOCK ioStatus;
|
||
|
||
status = NtFsControlFile(
|
||
DfsHandle,
|
||
NULL, // Event,
|
||
NULL, // ApcRoutine,
|
||
NULL, // ApcContext,
|
||
&ioStatus,
|
||
FsControlCode,
|
||
InputBuffer,
|
||
InputBufferLength,
|
||
OutputBuffer,
|
||
OutputBufferLength
|
||
);
|
||
|
||
if(NT_SUCCESS(status))
|
||
status = ioStatus.Status;
|
||
|
||
return status;
|
||
}
|
||
|
||
//+-------------------------------------------------------------------------
|
||
//
|
||
// CreateMUPEvent()
|
||
//
|
||
// Purpose: Creates an event so other processes can check if the
|
||
// MUP is ready yet
|
||
//
|
||
// Parameters: none
|
||
//
|
||
// Note: This handle should never be closed or other processes
|
||
// will fail on the call to OpenEvent().
|
||
//
|
||
// Return: Event handle if successful
|
||
// NULL if an error occurs
|
||
//
|
||
//+-------------------------------------------------------------------------
|
||
|
||
HANDLE
|
||
CreateMupEvent(void)
|
||
{
|
||
HANDLE hEvent;
|
||
|
||
// Use default security descriptor.
|
||
|
||
hEvent = CreateEvent (NULL, TRUE, FALSE, MUP_EVENT_NAME);
|
||
|
||
return hEvent;
|
||
|
||
}
|