2020-09-30 16:53:55 +02:00

8081 lines
237 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1998 - 1998 Microsoft Corporation
Module Name:
ldapjoin.c
Abstract:
NetJoin support functions for accessing the DS via LDAP, validating names, and handling LSA
functionality
Author:
Mac McLain (MacM) 27-Jan-1998 Name validation code based on ui\common\lmobj\lmobj code
by ThomasPa
Environment:
User mode only.
Revision History:
--*/
// Netlib uses DsGetDcName AND is linked into netapi32 where DsGetDcName is
// implemented. So define that we aren't importing the API.
#define _DSGETDCAPI_
#include <netsetp.h>
#include <lmaccess.h>
#include <wincrypt.h>
#define WKSTA_NETLOGON
#define NETSETUP_JOIN
#include <confname.h>
#include <winldap.h>
#include <nb30.h>
#include <msgrutil.h>
#include <lmaccess.h>
#include <lmuse.h>
#include <lmwksta.h>
#include <stdio.h>
#include <ntddbrow.h>
#include <netlibnt.h>
#include <ntddnfs.h>
#include <remboot.h>
#include <dns.h>
#include <ntsam.h>
#include <rpc.h>
#include <ntdsapi.h>
#include <netlogon.h>
#include <logonp.h>
#include <wchar.h>
#include <icanon.h> // NetpNameCanonicalize
#include <tstring.h> // STRLEN
#include <autoenr.h> // Autoenrol routine
#include "joinp.h"
#define NETSETUPP_WINLOGON_PATH L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon\\"
#define NETSETUPP_WINLOGON_CAD L"DisableCAD"
#define NETSETUPP_ALL_FILTER L"(ObjectClass=*)"
#define NETSETUPP_OU_FILTER L"(ObjectClass=OrganizationalUnit)"
#define NETSETUPP_RETURNED_ATTR L"AllowedChildClassesEffective"
#define NETSETUPP_DN_ATTR L"DistinguishedName"
#define NETSETUPP_WELL_KNOWN L"WellKnownObjects"
#define NETSETUPP_COMPUTER_OBJECT L"Computer"
#define NETSETUPP_OBJ_PREFIX L"CN="
#define NETSETUPP_ACCNT_TYPE_ENABLED L"4096"
#define NETSETUPP_ACCNT_TYPE_DISABLED L"4098"
//
// DNS registration removal function prototype
//
typedef DWORD (APIENTRY *DNS_REGISTRATION_REMOVAL_FN) ( VOID );
typedef DWORD (APIENTRY *DNS_REGISTRATION_ADDITION_FN) ( LPWSTR );
//
// Locally defined macros
//
#define clearncb(x) memset((char *)x,'\0',sizeof(NCB))
NTSTATUS
NetpGetLsaHandle(
IN LPWSTR lpServer, OPTIONAL
IN PLSA_HANDLE pPolicyHandleIn, OPTIONAL
OUT PLSA_HANDLE pPolicyHandleOut
)
/*++
Routine Description:
Either returns the given LSA handle if it's valid, or opens a new one
Arguments:
lpServer -- server name : NULL == local policy
pPolicyHandleIn -- Potentially open policy handle
pPolicyHandleOut -- Open policy handle returned here
Returns:
STATUS_SUCCESS -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
OBJECT_ATTRIBUTES OA;
UNICODE_STRING Server, *pServer = NULL;
if ( pPolicyHandleIn == NULL || *pPolicyHandleIn == NULL )
{
if ( lpServer != NULL )
{
RtlInitUnicodeString( &Server, lpServer );
pServer = &Server;
}
//
// Open the local policy
//
InitializeObjectAttributes( &OA, NULL, 0, NULL, NULL );
Status = LsaOpenPolicy( pServer, &OA, MAXIMUM_ALLOWED, pPolicyHandleOut );
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "NetpGetLsaHandle: LsaOpenPolicy on %ws failed: 0x%lx\n",
GetStrPtr(lpServer), Status ));
}
}
else
{
*pPolicyHandleOut = *pPolicyHandleIn;
}
return( Status );
}
VOID
NetpSetLsaHandle(
IN LSA_HANDLE OpenHandle,
OUT PLSA_HANDLE pReturnedHandle
)
/*++
Routine Description:
Either closes the opened handle or returns it
Arguments:
OpenHandle -- Handle returned from NetpGetLsaHandle
pReturnedHandle -- handle is passed back to the caller if requested
Returns:
VOID
--*/
{
if ( pReturnedHandle == NULL )
{
if ( OpenHandle != NULL )
{
LsaClose( OpenHandle );
}
}
else
{
*pReturnedHandle = OpenHandle;
}
}
NET_API_STATUS
NET_API_FUNCTION
NetpSetLsaPrimaryDomain(
IN LPWSTR lpDomain,
IN PSID pDomainSid, OPTIONAL
IN PPOLICY_DNS_DOMAIN_INFO pPolicyDns, OPTIONAL
OUT PLSA_HANDLE pPolicyHandle OPTIONAL
)
/*++
Routine Description:
Sets the primary domain in the local LSA policy
Arguments:
lpDomain -- Name of the domain to join
pDomainSid -- Primary domain sid to be set
pPolicyDns -- DNS domain info
pPolicyHandle -- handle returned if non-null
Returns:
NERR_Success -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
LSA_HANDLE LocalPolicy = NULL;
POLICY_PRIMARY_DOMAIN_INFO PolicyPDI;
Status = NetpGetLsaHandle( NULL, pPolicyHandle, &LocalPolicy );
//
// Now, build the primary domain info, and set it
//
if ( NT_SUCCESS( Status ) )
{
RtlInitUnicodeString( &(PolicyPDI.Name), lpDomain );
PolicyPDI.Sid = pDomainSid;
Status = LsaSetInformationPolicy( LocalPolicy,
PolicyPrimaryDomainInformation,
( PVOID )&PolicyPDI );
if ( NT_SUCCESS( Status ) && pPolicyDns )
{
Status = LsaSetInformationPolicy( LocalPolicy,
PolicyDnsDomainInformation,
( PVOID )pPolicyDns );
}
}
NetpSetLsaHandle( LocalPolicy, pPolicyHandle );
NetpLog(( "NetpSetLsaPrimaryDomain: for '%ws' status: 0x%lx\n", GetStrPtr(lpDomain), Status ));
return( RtlNtStatusToDosError( Status ) );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetLsaPrimaryDomain(
IN LPWSTR lpServer, OPTIONAL
OUT PPOLICY_PRIMARY_DOMAIN_INFO *ppPolicyPDI,
OUT PPOLICY_DNS_DOMAIN_INFO *ppPolicyDns,
OUT PLSA_HANDLE pPolicyHandle OPTIONAL
)
/*++
Routine Description:
Gets the primary domain info in the local LSA policy
Arguments:
PolicyHandle -- Handle to the open policy. If NULL, a new handle is
opened.
lpServer -- Optional server name on which to read the policy
ppPolicyPDI -- Primary domain policy returned here
ppPolicyDNS -- Dns domain information is returned here if it exists
pPolicyHandle -- Optional. Policy handle returned here if not null
Returns:
NERR_Success -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
LSA_HANDLE LocalPolicy = NULL;
UNICODE_STRING Server, *pServer = NULL;
//
// Initialization
//
*ppPolicyPDI = NULL;
*ppPolicyDns = NULL;
if ( lpServer != NULL )
{
RtlInitUnicodeString( &Server, lpServer );
pServer = &Server;
}
Status = NetpGetLsaHandle( lpServer, pPolicyHandle, &LocalPolicy );
//
// Now, get the primary domain info
//
if ( NT_SUCCESS( Status ) )
{
Status = LsaQueryInformationPolicy( LocalPolicy,
PolicyDnsDomainInformation,
( PVOID * )ppPolicyDns );
if ( Status == RPC_NT_PROCNUM_OUT_OF_RANGE )
{
Status = STATUS_SUCCESS;
*ppPolicyDns = NULL;
}
if ( NT_SUCCESS( Status ) )
{
Status = LsaQueryInformationPolicy( LocalPolicy,
PolicyPrimaryDomainInformation,
(PVOID *)ppPolicyPDI);
if ( !NT_SUCCESS( Status ) && (*ppPolicyDns) != NULL )
{
LsaFreeMemory( *ppPolicyDns );
*ppPolicyDns = NULL;
}
}
}
NetpSetLsaHandle( LocalPolicy, pPolicyHandle );
NetpLog(( "NetpGetLsaPrimaryDomain: status: 0x%lx\n", Status ));
return( RtlNtStatusToDosError( Status ) );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetLsaDcRole(
IN LPWSTR lpMachine,
OUT BOOL *pfIsDC
)
/*++
Routine Description:
Gets the role of the DC in the domain
Arguments:
lpMachine -- Machine to connect to
pfIsDC -- If TRUE, this is a DC.
Returns:
NERR_Success -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
PBYTE pBuff;
LSA_HANDLE hPolicy = NULL;
Status = NetpGetLsaHandle( lpMachine, NULL, &hPolicy );
//
// Now, get the server role info
//
if ( NT_SUCCESS( Status ) ) {
Status = LsaQueryInformationPolicy( hPolicy,
PolicyLsaServerRoleInformation,
&pBuff);
if ( NT_SUCCESS(Status) ) {
if ( *(PPOLICY_LSA_SERVER_ROLE)pBuff == PolicyServerRoleBackup ||
*(PPOLICY_LSA_SERVER_ROLE)pBuff == PolicyServerRolePrimary ) {
*pfIsDC = TRUE;
} else {
*pfIsDC = FALSE;
}
LsaFreeMemory( pBuff );
}
LsaClose( hPolicy );
}
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "NetpGetLsaDcRole failed with 0x%lx\n", Status ));
}
return( RtlNtStatusToDosError( Status ) );
}
NTSTATUS
NetpLsaOpenSecret(
IN LSA_HANDLE hLsa,
IN PUNICODE_STRING pusSecretName,
IN ACCESS_MASK DesiredAccess,
OUT PLSA_HANDLE phSecret
)
/*++
Routine Description:
Open the specified LSA secret as self.
LsaQuerySecret fails for a network client whent the client is not
trusted (see lsa\server\dbsecret.c). This causes remote join
operation to fail. To get around this, this function temporarily
un-impersonates, opens the secrets and impersonates again.
Thus the open secret occurrs in LocalSystem context.
$ REVIEW kumarp 15-July-1999
This is obviously not a good design. This should be changed post NT5.
Arguments:
same as those for LsaOpenSecret
Returns:
NTSTATUS, see help for LsaOpenSecret
--*/
{
NTSTATUS Status = STATUS_UNSUCCESSFUL;
HANDLE hToken=NULL;
__try
{
if (OpenThreadToken(GetCurrentThread(), TOKEN_IMPERSONATE,
TRUE, &hToken))
{
if (SetThreadToken(NULL, NULL))
{
Status = LsaOpenSecret(hLsa, pusSecretName,
DesiredAccess, phSecret);
}
}
}
__finally
{
if (hToken)
{
if ( !SetThreadToken(NULL, hToken) ) {
NetpLog(( "NetpLsaOpenSecret: Couldn't reset the user token 0x%lx\n",
GetLastError() ));
Status = NetpApiStatusToNtStatus( GetLastError() );
}
}
}
if ( hToken ) {
CloseHandle( hToken );
}
NetpLog(( "NetpLsaOpenSecret: status: 0x%lx\n", Status ));
return Status;
}
NET_API_STATUS
NET_API_FUNCTION
NetpManageMachineSecret(
IN LPWSTR lpMachine,
IN LPWSTR lpPassword,
IN DWORD Action,
IN BOOL UseDefaultForOldPwd,
OUT PLSA_HANDLE pPolicyHandle OPTIONAL
)
/*++
Routine Description:
Create/delete the machine secret
Arguments:
lpMachine -- Machine to add/delete the secret for
lpPassword -- Machine password to use.
Action -- Action to take
UseDefaultForOldPwd - if TRUE, the default password should be set
for the old password value. Used only if
secret is created.
pPolicyHandle -- If present, the opened policy handle is returned here
Returns:
NERR_Success -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
LSA_HANDLE LocalPolicy = NULL, SecretHandle = NULL;
UNICODE_STRING Key, Data, *CurrentValue = NULL;
BOOLEAN SecretCreated = FALSE;
WCHAR MachinePasswordBuffer[PWLEN + 1];
UNICODE_STRING MachinePassword;
BOOLEAN FreeCurrentValue = FALSE;
if( Action == NETSETUPP_CREATE )
{
ASSERT( lpPassword );
}
Status = NetpGetLsaHandle( NULL, pPolicyHandle, &LocalPolicy );
//
// open/create the secret
//
if ( NT_SUCCESS( Status ) )
{
RtlInitUnicodeString( &Key, L"$MACHINE.ACC" );
RtlInitUnicodeString( &Data, lpPassword );
Status = NetpLsaOpenSecret( LocalPolicy, &Key,
Action == NETSETUPP_CREATE ?
SECRET_SET_VALUE | SECRET_QUERY_VALUE : DELETE,
&SecretHandle );
if ( Status == STATUS_OBJECT_NAME_NOT_FOUND )
{
if ( Action == NETSETUPP_DELETE )
{
Status = STATUS_SUCCESS;
}
else
{
Status = LsaCreateSecret( LocalPolicy, &Key,
SECRET_SET_VALUE, &SecretHandle );
if ( NT_SUCCESS( Status ) )
{
SecretCreated = TRUE;
}
}
}
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "NetpManageMachineSecret: Open/Create secret failed: 0x%lx\n", Status ));
}
if ( NT_SUCCESS( Status ) )
{
if ( Action == NETSETUPP_CREATE )
{
//
// First, read the current value, so we can save it as the old value
//
if ( !UseDefaultForOldPwd ) {
if ( SecretCreated )
{
CurrentValue = &Data;
}
else
{
Status = LsaQuerySecret( SecretHandle, &CurrentValue,
NULL, NULL, NULL );
FreeCurrentValue = TRUE;
}
//
// If we are to use the default value for old password,
// generate the default value
//
} else {
NetpGenerateDefaultPassword(lpMachine, MachinePasswordBuffer);
RtlInitUnicodeString( &MachinePassword, MachinePasswordBuffer );
CurrentValue = &MachinePassword;
}
if ( NT_SUCCESS( Status ) )
{
//
// Now, store both the new password and the old
//
Status = LsaSetSecret( SecretHandle, &Data, CurrentValue );
if ( FreeCurrentValue )
{
LsaFreeMemory( CurrentValue );
}
}
}
else
{
//
// No secret handle means we failed earlier in
// some intermediate state. That's ok, just press on.
//
if ( SecretHandle != NULL )
{
Status = LsaDelete( SecretHandle );
if ( NT_SUCCESS( Status ) )
{
SecretHandle = NULL;
}
}
}
}
if ( SecretHandle )
{
LsaClose( SecretHandle );
}
}
NetpSetLsaHandle( LocalPolicy, pPolicyHandle );
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "NetpManageMachineSecret: '%s' operation failed: 0x%lx\n",
Action == NETSETUPP_CREATE ? "CREATE" : "DELETE", Status ));
}
return( RtlNtStatusToDosError( Status ) );
}
NET_API_STATUS
NET_API_FUNCTION
NetpReadCurrentSecret(
OUT LPWSTR *lpCurrentSecret,
OUT PLSA_HANDLE pPolicyHandle OPTIONAL
)
/*++
Routine Description:
Reads the value of the current secret
Arguments:
lpCurrentSecret -- Where the current secret is returned
pPolicyHandle -- If present, the opened policy handle is returned here
Returns:
NERR_Success -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
LSA_HANDLE LocalPolicy = NULL, SecretHandle;
UNICODE_STRING Secret, *Data = NULL;
Status = NetpGetLsaHandle( NULL, pPolicyHandle, &LocalPolicy );
//
// Now, read the secret
//
if ( NT_SUCCESS( Status ) ) {
RtlInitUnicodeString( &Secret, L"$MACHINE.ACC" );
// Status = LsaRetrievePrivateData( LocalPolicy, &Key, &Data );
Status = NetpLsaOpenSecret( LocalPolicy,
&Secret,
SECRET_QUERY_VALUE,
&SecretHandle );
if ( NT_SUCCESS(Status) ) {
Status = LsaQuerySecret( SecretHandle,
&Data,
NULL,
NULL,
NULL );
LsaClose( SecretHandle );
}
if ( NT_SUCCESS( Status ) ) {
if( NetApiBufferAllocate( Data->Length + sizeof( WCHAR ),
( PBYTE * )lpCurrentSecret ) != NERR_Success ) {
Status = STATUS_INSUFFICIENT_RESOURCES;
} else {
RtlCopyMemory( ( PVOID )*lpCurrentSecret,
Data->Buffer,
Data->Length );
( *lpCurrentSecret )[ Data->Length / sizeof( WCHAR ) ] = UNICODE_NULL;
}
}
}
NetpSetLsaHandle( LocalPolicy, pPolicyHandle );
if ( Data != NULL ) {
LsaFreeMemory( Data );
}
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "NetpReadCurrentSecret: failed: 0x%lx\n", Status ));
}
return( RtlNtStatusToDosError( Status ) );
}
NET_API_STATUS
NET_API_FUNCTION
NetpSetNetlogonDomainCache(
IN LPWSTR lpDc
)
/*++
Routine Description:
Initializes NetLogons trusted domain cache, using the trusted
domain list on the DC.
Arguments:
lpDc -- Name of a DC in the domain
The caller should already have an valid connection to IPC$
Returns:
NERR_Success -- Success
--*/
{
DWORD dwErr = ERROR_SUCCESS;
PDS_DOMAIN_TRUSTSW TrustedDomains=NULL;
ULONG TrustedDomainCount=0;
//
// Get the trusted domain list from the DC.
//
dwErr = DsEnumerateDomainTrustsW( lpDc, DS_DOMAIN_VALID_FLAGS,
&TrustedDomains, &TrustedDomainCount );
//
// If the server does not support returning all trust types
// (i.e. the server is an NT4 machine) ask for only those
// which it can return.
//
if ( dwErr == ERROR_NOT_SUPPORTED ) {
NetpLog(( "NetpSetNetlogonDomainCache: DsEnumerateDomainTrustsW for all trusts failed with ERROR_NOT_SUPPORTED -- retry\n"));
dwErr = DsEnumerateDomainTrustsW(
lpDc,
DS_DOMAIN_PRIMARY | DS_DOMAIN_DIRECT_OUTBOUND,
&TrustedDomains,
&TrustedDomainCount );
if ( dwErr == ERROR_NOT_SUPPORTED ) {
//
// looks like the DC is running NT3.51. In this case, we do not want
// to fail the join operation because we could not write
// the netlogon cache. reset the error code.
//
// see bug "359684 Win2k workstation unable to join NT3.51Domain"
//
NetpLog(( "NetpSetNetlogonDomainCache: DsEnumerateDomainTrustsW for some trusts failed with ERROR_NOT_SUPPORTED -- ignore\n"));
dwErr = ERROR_SUCCESS;
} else if ( dwErr != ERROR_SUCCESS ) {
NetpLog(( "NetpSetNetlogonDomainCache: DsEnumerateDomainTrustsW for some trusts failed with 0x%lx\n", dwErr ));
}
} else if ( dwErr != ERROR_SUCCESS ) {
NetpLog(( "NetpSetNetlogonDomainCache: DsEnumerateDomainTrustsW failed 0x%lx\n", dwErr ));
}
if ( dwErr == ERROR_SUCCESS ) {
//
// Write the trusted domain list to a file where netlogon will find it.
//
if ( TrustedDomainCount > 0 ) {
dwErr = NlWriteFileForestTrustList (
NL_FOREST_BINARY_LOG_FILE_JOIN,
TrustedDomains,
TrustedDomainCount );
} else {
NetpLog(( "NetpSetNetlogonDomainCache: No trusts to write\n" ));
}
}
//
// Disable the no Ctrl-Alt-Del from the winlogon side of things
//
if ( dwErr == ERROR_SUCCESS ) {
HKEY hWinlogon;
dwErr = RegOpenKey( HKEY_LOCAL_MACHINE,
NETSETUPP_WINLOGON_PATH, &hWinlogon );
if ( dwErr == ERROR_SUCCESS ) {
DWORD Value;
Value = 0;
dwErr = RegSetValueEx( hWinlogon, NETSETUPP_WINLOGON_CAD, 0,
REG_DWORD, (PBYTE)&Value, sizeof( ULONG ) );
RegCloseKey( hWinlogon );
}
//
// Failing to set this never causes failure
//
if ( dwErr != ERROR_SUCCESS ) {
NetpLog(( "Setting Winlogon DisableCAD failed with %lu\n", dwErr ));
dwErr = ERROR_SUCCESS;
}
}
//
// Free locally used resources
//
if ( TrustedDomains != NULL ) {
NetApiBufferFree( TrustedDomains );
}
return dwErr;
}
/*++
Routine Description:
Is this a terminal-server-application-server?
Arguments:
Args - none
Return Value:
TRUE or FALSE
--*/
BOOL IsAppServer(void)
{
OSVERSIONINFOEX osVersionInfo;
DWORDLONG dwlConditionMask = 0;
BOOL fIsWTS;
osVersionInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEX);
fIsWTS = GetVersionEx((OSVERSIONINFO *)&osVersionInfo) &&
(osVersionInfo.wSuiteMask & VER_SUITE_TERMINAL) &&
!(osVersionInfo.wSuiteMask & VER_SUITE_SINGLEUSERTS);
return fIsWTS;
}
NET_API_STATUS
NET_API_FUNCTION
NetpManageLocalGroups(
IN PSID pDomainSid,
IN BOOL fDelete
)
/*++
Routine Description:
Performs SAM account handling to either add or remove the DomainAdmins,
etc groups from the local groups.
Arguments:
pDomainSid -- SID of the domain being joined/left
fDelete -- Whether to add or remove the admin alias
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
//
// Keep these in synch with the rids and Sids below
//
ULONG LocalRids[] =
{
DOMAIN_ALIAS_RID_ADMINS,
DOMAIN_ALIAS_RID_USERS
};
PWSTR ppwszLocalGroups[ sizeof( LocalRids ) / sizeof( ULONG )] =
{
NULL,
NULL
};
ULONG Rids[] =
{
DOMAIN_GROUP_RID_ADMINS,
DOMAIN_GROUP_RID_USERS
};
BOOLEAN GroupMustExist[ sizeof( LocalRids ) / sizeof( ULONG )] =
{
TRUE,
TRUE
};
static SID_IDENTIFIER_AUTHORITY BultinAuth = SECURITY_NT_AUTHORITY;
DWORD Sids[sizeof( SID )/sizeof( DWORD ) + SID_MAX_SUB_AUTHORITIES][sizeof(Rids) / sizeof(ULONG)];
DWORD cDSidSize, *pLastSub, i, j;
PUCHAR pSubAuthCnt;
PWSTR LocalGroupName = NULL;
PWCHAR DomainName = NULL;
ULONG Size, DomainSize;
SID_NAME_USE SNE;
ULONG numOfGroups;
cDSidSize = RtlLengthSid( pDomainSid );
// number of groups to process
numOfGroups = sizeof(Rids) / sizeof(ULONG);
for ( i = 0 ; i < numOfGroups && NetStatus == NERR_Success; i++)
{
Size = 0;
DomainSize = 0;
if ( DomainName != NULL ) {
NetApiBufferFree( DomainName );
DomainName = NULL;
}
//
// Get the name of the local group first...
//
RtlInitializeSid( ( PSID )Sids[ i ], &BultinAuth, 2 );
*(RtlSubAuthoritySid(( PSID )Sids[ i ], 0)) = SECURITY_BUILTIN_DOMAIN_RID;
*(RtlSubAuthoritySid(( PSID )Sids[ i ], 1)) = LocalRids[ i ];
LookupAccountSidW( NULL, ( PSID )Sids[ i ], NULL, &Size,
DomainName, &DomainSize, &SNE );
if ( GetLastError() == ERROR_INSUFFICIENT_BUFFER )
{
NetStatus = NetApiBufferAllocate( Size * sizeof(WCHAR),
&LocalGroupName );
if ( NetStatus == NERR_Success ) {
NetStatus = NetApiBufferAllocate( DomainSize * sizeof(WCHAR),
&DomainName );
}
if ( NetStatus == NERR_Success )
{
if ( !LookupAccountSid( NULL, ( PSID )Sids[ i ], LocalGroupName,
&Size, DomainName, &DomainSize, &SNE ) )
{
NetStatus = GetLastError();
if ( NetStatus == ERROR_NONE_MAPPED && GroupMustExist[ i ] == FALSE )
{
NetStatus = NERR_Success;
continue;
}
else
{
#ifdef NETSETUP_VERBOSE_LOGGING
UNICODE_STRING DisplaySid;
NTSTATUS Status2;
RtlZeroMemory( &DisplaySid, sizeof( UNICODE_STRING ) );
Status2 = RtlConvertSidToUnicodeString( &DisplaySid,
( PSID )Sids[ i ], TRUE );
if ( NT_SUCCESS( Status2 ) )
{
NetpLog(( "LookupAccounSid on %wZ failed with %lu\n",
&DisplaySid,
NetStatus ));
RtlFreeUnicodeString(&DisplaySid);
}
else
{
NetpLog(( "LookupAccounSid on <undisplayable sid> "
"failed with %lu\n",
NetStatus ));
}
#endif
}
}
else
{
ppwszLocalGroups[ i ] = LocalGroupName;
}
}
else
{
break;
}
}
RtlCopyMemory( (PBYTE)Sids[i], pDomainSid, cDSidSize );
//
// Now, add the new domain relative rid
//
pSubAuthCnt = GetSidSubAuthorityCount( (PSID)Sids[i] );
(*pSubAuthCnt)++;
pLastSub = GetSidSubAuthority( (PSID)Sids[i], (*pSubAuthCnt) - 1 );
*pLastSub = Rids[i];
if ( fDelete == NETSETUPP_CREATE)
{
NetStatus = NetLocalGroupAddMember( NULL,
ppwszLocalGroups[i],
(PSID)Sids[i] );
if ( NetStatus == ERROR_MEMBER_IN_ALIAS )
{
NetStatus = NERR_Success;
}
}
else
{
NetStatus = NetLocalGroupDelMember( NULL,
ppwszLocalGroups[i],
(PSID)Sids[i] );
if ( NetStatus == ERROR_MEMBER_NOT_IN_ALIAS )
{
NetStatus = NERR_Success;
}
}
}
//
// If something failed, try to restore what was deleted
//
if ( NetStatus != NERR_Success )
{
for ( j = 0; j < i; j++ ) {
if ( fDelete == NETSETUPP_DELETE)
{
NetLocalGroupAddMember( NULL,
ppwszLocalGroups[j],
(PSID)Sids[j] );
}
else
{
NetLocalGroupDelMember( NULL,
ppwszLocalGroups[j],
(PSID)Sids[j] );
}
}
}
if ( DomainName != NULL ) {
NetApiBufferFree( DomainName );
}
for ( i = 0; i < numOfGroups ; i++ )
{
if ( ppwszLocalGroups[ i ] )
{
NetApiBufferFree( ppwszLocalGroups[ i ] );
}
}
if ( NetStatus != NERR_Success )
{
NetpLog(( "NetpManageLocalGroups failed with %lu\n", NetStatus ));
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpHandleJoinedStateInfo(
IN PNETSETUP_SAVED_JOIN_STATE SavedState,
IN BOOLEAN Save,
OUT PLSA_HANDLE ReturnedPolicyHandle OPTIONAL
)
/*++
Routine Description:
Saves or restores the join state info.
Arguments:
SavedState -- join state info
This includes:
- machine account secret value
- primary domain info
- dns domain info
Save -- TRUE == save state, FALSE == restore state
ReturnedPolicyHandle -- local LSA handle returned in this
Returns:
NERR_Success -- Success
--*/
{
NTSTATUS Status = STATUS_SUCCESS;
LSA_HANDLE LocalPolicy = NULL, SecretHandle;
UNICODE_STRING Secret;
if ( Save )
{
RtlZeroMemory( SavedState, sizeof( NETSETUP_SAVED_JOIN_STATE ) );
}
//
// get handle to local LSA policy
//
Status = NetpGetLsaHandle( NULL, ReturnedPolicyHandle, &LocalPolicy );
if ( NT_SUCCESS( Status ) )
{
//
// First, read the machine account secret
//
RtlInitUnicodeString( &Secret, L"$MACHINE.ACC" );
Status = NetpLsaOpenSecret( LocalPolicy,
&Secret,
SECRET_QUERY_VALUE | SECRET_SET_VALUE,
&SecretHandle );
if ( NT_SUCCESS( Status ) )
{
if ( Save )
{
SavedState->MachineSecret = TRUE;
Status = LsaQuerySecret( SecretHandle,
&( SavedState->CurrentValue ),
NULL,
&( SavedState->PreviousValue ),
NULL );
}
else
{
if ( SavedState ->MachineSecret )
{
Status = LsaSetSecret( SecretHandle,
SavedState->CurrentValue,
SavedState->PreviousValue );
}
}
LsaClose( SecretHandle );
}
//
// If machine secret is not present, it is not an error.
//
if ( Status == STATUS_OBJECT_NAME_NOT_FOUND )
{
if ( Save )
{
SavedState->MachineSecret = FALSE;
}
Status = STATUS_SUCCESS;
}
//
// Now, save/restore the policy information
//
if ( NT_SUCCESS( Status ) )
{
if ( Save )
{
Status = NetpGetLsaPrimaryDomain( NULL,
&( SavedState->PrimaryDomainInfo ),
&( SavedState->DnsDomainInfo ),
&LocalPolicy );
}
else
{
Status = LsaSetInformationPolicy( LocalPolicy,
PolicyPrimaryDomainInformation,
SavedState->PrimaryDomainInfo );
if ( NT_SUCCESS( Status ) )
{
Status = LsaSetInformationPolicy( LocalPolicy,
PolicyDnsDomainInformation,
SavedState->DnsDomainInfo );
}
}
}
}
NetpSetLsaHandle( LocalPolicy, ReturnedPolicyHandle );
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "NetpHandleJoinedStateInfo: '%s' operation failed: 0x%lx\n",
Save ? "Save" : "Restore", Status ));
}
return( RtlNtStatusToDosError( Status ) );
}
NET_API_STATUS
MsgFmtNcbName(
OUT PCHAR DestBuf,
IN LPTSTR Name,
IN DWORD Type)
/*++
Routine Description:
FmtNcbName - format a name NCB-style
Given a name, a name type, and a destination address, this
function copies the name and the type to the destination in
the format used in the name fields of a Network Control
Block.
SIDE EFFECTS
Modifies 16 bytes starting at the destination address.
Arguments:
DestBuf - Pointer to the destination buffer.
Name - Unicode NUL-terminated name string
Type - Name type number (0, 3, 5, or 32) (3=NON_FWD, 5=FWD)
Return Value:
NERR_Success - The operation was successful
Translated Return Code from the Rtl Translate routine.
--*/
{
DWORD i; // Counter
NTSTATUS ntStatus;
NET_API_STATUS status;
OEM_STRING ansiString;
UNICODE_STRING unicodeString;
PCHAR pAnsiString;
//
// Force the name to be upper case.
//
status = NetpNameCanonicalize(
NULL,
Name,
Name,
STRSIZE(Name),
NAMETYPE_MESSAGEDEST,
0);
if (status != NERR_Success) {
return(status);
}
//
// Convert the unicode name string into an ansi string - using the
// current locale.
//
#ifdef UNICODE
unicodeString.Length = (USHORT)(STRLEN(Name)*sizeof(WCHAR));
unicodeString.MaximumLength = (USHORT)((STRLEN(Name)+1) * sizeof(WCHAR));
unicodeString.Buffer = Name;
ntStatus = RtlUnicodeStringToOemString(
&ansiString,
&unicodeString,
TRUE); // Allocate the ansiString Buffer.
if (!NT_SUCCESS(ntStatus))
{
NetpLog(( "FmtNcbName: RtlUnicodeStringToOemString failed 0x%lx\n",
ntStatus ));
return NetpNtStatusToApiStatus(ntStatus);
}
pAnsiString = ansiString.Buffer;
*(pAnsiString+ansiString.Length) = '\0';
#else
UNUSED(ntStatus);
UNUSED(unicodeString);
UNUSED(ansiString);
pAnsiString = Name;
#endif // UNICODE
//
// copy each character until a NUL is reached, or until NCBNAMSZ-1
// characters have been copied.
//
for (i=0; i < NCBNAMSZ - 1; ++i) {
if (*pAnsiString == '\0') {
break;
}
//
// Copy the Name
//
*DestBuf++ = *pAnsiString++;
}
//
// Free the buffer that RtlUnicodeStringToOemString created for us.
// NOTE: only the ansiString.Buffer portion is free'd.
//
#ifdef UNICODE
RtlFreeOemString( &ansiString);
#endif // UNICODE
//
// Pad the name field with spaces
//
for(; i < NCBNAMSZ - 1; ++i) {
*DestBuf++ = ' ';
}
//
// Set the name type.
//
NetpAssert( Type!=5 ); // 5 is not valid for NT.
*DestBuf = (CHAR) Type; // Set name type
return(NERR_Success);
}
NET_API_STATUS
NET_API_FUNCTION
NetpCheckNetBiosNameNotInUse(
IN LPWSTR pszName,
IN BOOLEAN MachineName,
IN BOOLEAN Unique
)
{
NCB ncb;
LANA_ENUM lanaBuffer;
unsigned char i;
unsigned char nbStatus;
NET_API_STATUS NetStatus = NERR_Success;
WCHAR szMachineNameBuf[MAX_COMPUTERNAME_LENGTH + 1];
LPWSTR szMachineName=szMachineNameBuf;
//
// Find the number of networks by sending an enum request via Netbios.
//
clearncb(&ncb);
ncb.ncb_command = NCBENUM; // Enumerate LANA nums (wait)
ncb.ncb_buffer = (PUCHAR)&lanaBuffer;
ncb.ncb_length = sizeof(LANA_ENUM);
nbStatus = Netbios (&ncb);
if (nbStatus != NRC_GOODRET)
{
NetStatus = NetpNetBiosStatusToApiStatus( nbStatus );
goto Cleanup;
}
clearncb(&ncb);
NetStatus = MsgFmtNcbName( (char *)ncb.ncb_name, pszName,
MachineName ? 0 : 0x1c );
if ( NetStatus != NERR_Success )
{
goto Cleanup;
}
//
// Move the Adapter Numbers (lana) into the array that will contain them.
//
for ( i = 0; i < lanaBuffer.length && NetStatus == NERR_Success; i++ )
{
NetpNetBiosReset( lanaBuffer.lana[i] );
if ( Unique )
{
ncb.ncb_command = NCBADDNAME;
}
else
{
ncb.ncb_command = NCBADDGRNAME;
}
ncb.ncb_lana_num = lanaBuffer.lana[i];
nbStatus = Netbios( &ncb );
switch ( nbStatus )
{
case NRC_DUPNAME:
// NRC_DUPNAME ==
// "A duplicate name existed in the local name table"
//
// In this case, we need to check if the name being checked
// is the same as the local computer name. If so,
// the name is expected to be in the local table therefore
// we convert this errcode to a success code
//
NetStatus = NetpGetComputerNameAllocIfReqd(
&szMachineName, MAX_COMPUTERNAME_LENGTH+1);
if (NetStatus == NERR_Success)
{
if (!_wcsicmp(szMachineName, pszName))
{
NetStatus = NERR_Success;
}
else
{
NetStatus = ERROR_DUP_NAME;
}
}
break;
case NRC_INUSE:
NetStatus = ERROR_DUP_NAME;
break;
case NRC_GOODRET:
// Delete the name
ncb.ncb_command = NCBDELNAME;
ncb.ncb_lana_num = lanaBuffer.lana[i];
// Not much we can do if this fails.
Netbios( &ncb );
// fall through
default:
NetStatus = NetpNetBiosStatusToApiStatus( nbStatus );
break;
}
}
Cleanup:
if ( NetStatus != NERR_Success )
{
NetpLog(( "NetpCheckNetBiosNameNotInUse: for '%ws' returned: 0x%lx\n",
pszName, NetStatus ));
}
if (szMachineName != szMachineNameBuf)
{
NetApiBufferFree(szMachineName);
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpIsValidDomainName(
IN LPWSTR lpName,
IN LPWSTR lpServer,
IN LPWSTR lpAccount,
IN LPWSTR lpPassword
)
/*++
Routine Description:
Determines if a name is a DC name or not. Copied from
ui\net\common\src\lmboj\lmobj\lmodom.cxx
Arguments:
lpName -- Name to check
lpServer -- Name of a server within that domain
Returns:
NERR_Success -- Success
ERROR_DUP_NAME -- The domain name is in use
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWKSTA_INFO_100 pWKI100 = NULL;
BOOL fIsDC;
POLICY_LSA_SERVER_ROLE Role;
NetStatus = NetpManageIPCConnect( lpServer, lpAccount,
lpPassword,
NETSETUPP_CONNECT_IPC | NETSETUPP_NULL_SESSION_IPC );
if ( NetStatus == NERR_Success ) {
//
// Now, get the info from the server
//
NetStatus = NetWkstaGetInfo( lpServer, 100, (LPBYTE *)&pWKI100 );
if ( NetStatus == NERR_Success ) {
if (_wcsicmp( lpName, pWKI100->wki100_langroup ) == 0 ) {
//
// Ok, it's a match... Determine the domain role.
//
NetStatus = NetpGetLsaDcRole( lpServer, &fIsDC );
if ( ( NetStatus == NERR_Success ) && ( fIsDC == FALSE ) )
{
NetStatus = NERR_DCNotFound;
}
}
}
NetpManageIPCConnect( lpServer, lpAccount,
lpPassword, NETSETUPP_DISCONNECT_IPC );
}
if ( NetStatus != NERR_Success ) {
NetpLog((
"NetpIsValidDomainName for %ws returned 0x%lx\n",
lpName, NetStatus ));
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpCheckDomainNameIsValid(
IN LPWSTR lpName,
IN LPWSTR lpAccount,
IN LPWSTR lpPassword,
IN BOOL fShouldExist
)
/*++
Routine Description:
Checks to see if the given name is in use by a domain
Arguments:
lpName -- Name to check
Returns:
NERR_Success -- The domain is found and valid
ERROR_NO_SUCH_DOMAIN -- Domain name not found
--*/
{
NET_API_STATUS NetStatus;
PBYTE pbDC;
DWORD cDCs, i, j;
PUNICODE_STRING pDCList;
LPWSTR pwszDomain;
#if(_WIN32_WINNT >= 0x0500)
PDOMAIN_CONTROLLER_INFO pDCInfo = NULL;
#else
PBYTE pDCInfo = NULL;
#endif
UNREFERENCED_PARAMETER( lpAccount );
UNREFERENCED_PARAMETER( lpPassword );
//
// Start with NetGetAnyDCName
//
#if(_WIN32_WINNT >= 0x0500)
NetStatus = DsGetDcName( NULL, lpName, NULL, NULL,
DS_FORCE_REDISCOVERY, &pDCInfo );
#else
NetStatus = NetGetAnyDCName( NULL,
( LPCWSTR )lpName,
&pDCInfo );
#endif
if ( NetStatus != NERR_Success ) {
if ( NetStatus == ERROR_NO_SUCH_USER ) {
NetStatus = NERR_Success;
}
} else {
NetApiBufferFree( pDCInfo );
}
//
// Map our error codes so we only return success if we validated the
// domain name
//
if ( fShouldExist ) {
if ( NetStatus == NERR_Success || NetStatus == ERROR_NO_LOGON_SERVERS ) {
NetStatus = NERR_Success;
} else {
NetStatus = ERROR_NO_SUCH_DOMAIN;
}
} else {
if ( NetStatus == NERR_Success || NetStatus == ERROR_NO_LOGON_SERVERS ) {
NetStatus = ERROR_DUP_NAME;
} else if ( NetStatus == NERR_DCNotFound || NetStatus == ERROR_NO_SUCH_DOMAIN ) {
NetStatus = NERR_Success;
}
}
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpCheckDomainNameIsValid for %ws returned 0x%lx\n",
lpName, NetStatus ));
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpManageIPCConnect(
IN LPWSTR lpServer,
IN LPWSTR lpAccount,
IN LPWSTR lpPassword,
IN ULONG fOptions
)
/*++
Routine Description:
Manages the connections to the servers IPC share
Arguments:
lpServer -- Server to connect to
lpAccount -- Account to use
lpPassword -- Password to use. The password has been NOT been encoded
fOptions -- Flags to determine operation/connect/disconnect
Returns:
NERR_Success -- The domain is found and valid
--*/
{
NET_API_STATUS NetStatus;
#if(_WIN32_WINNT >= 0x0500)
WCHAR wszPath[2 + DNS_MAX_NAME_LENGTH + 1 + NNLEN + 1];
#else
WCHAR wszPath[2 + 256 + 1 + NNLEN + 1];
#endif
PWSTR pwszPath = wszPath;
USE_INFO_2 NetUI2;
PWSTR pwszUser, pwszDomain, pwszReset;
DWORD BadParm = 0;
DWORD ForceLevel = USE_NOFORCE;
//
// Guard against buffer overrun: the server name
// length has to be no more than max DNS name
// length plus 2 ( for "\\").
//
if ( wcslen(lpServer) > DNS_MAX_NAME_LENGTH + 2 ) {
NetpLog(( "NetpManageIPCConnect: server name %ws too long - error out\n", lpServer ));
return ERROR_INVALID_PARAMETER;
}
//
// Build the path...
//
if (*lpServer != L'\\') {
wcscpy(wszPath, L"\\\\");
pwszPath += 2;
}
if ( FLAG_ON( fOptions, NETSETUPP_USE_LOTS_FORCE ) )
{
ASSERT( FLAG_ON(fOptions, NETSETUPP_DISCONNECT_IPC ) );
ForceLevel = USE_LOTS_OF_FORCE;
}
swprintf( pwszPath, L"%ws\\IPC$", lpServer );
pwszPath = wszPath;
if ( FLAG_ON( fOptions, NETSETUPP_DISCONNECT_IPC ) )
{
NetStatus = NetUseDel( NULL, pwszPath, ForceLevel );
if ( NetStatus != NERR_Success )
{
NetpKdPrint(( PREFIX_NETJOIN "NetUseDel on %ws failed with %d\n", pwszPath, NetStatus ));
NetpLog(( "NetUseDel on %ws failed with %d\n", pwszPath, NetStatus ));
if ( (NetStatus != NERR_UseNotFound)
&& (ForceLevel != USE_LOTS_OF_FORCE) )
{
NetStatus = NetUseDel( NULL, pwszPath, USE_LOTS_OF_FORCE );
if ( NetStatus != NERR_Success )
{
ASSERT( NetStatus == NERR_Success );
NetpKdPrint(( PREFIX_NETJOIN "NetUseDel with force on %ws failed with %d\n",
pwszPath, NetStatus ));
NetpLog(( "NetUseDel with force on %ws failed with %d\n",
pwszPath, NetStatus ));
}
}
}
}
else
{
if ( lpAccount != NULL )
{
pwszReset = wcschr( lpAccount, L'\\' );
if (pwszReset != NULL)
{
pwszUser = pwszReset + 1;
pwszDomain = lpAccount;
*pwszReset = UNICODE_NULL;
}
else
{
pwszUser = lpAccount;
//
// First, assume it's a UPN, so we pass in an empty string
//
pwszDomain = L"";
}
}
else
{
pwszUser = NULL;
pwszDomain = NULL;
pwszReset = NULL;
}
RtlZeroMemory(&NetUI2, sizeof(USE_INFO_2) );
NetUI2.ui2_local = NULL;
NetUI2.ui2_remote = pwszPath;
NetUI2.ui2_asg_type = USE_IPC;
NetUI2.ui2_username = pwszUser;
NetUI2.ui2_domainname = pwszDomain;
NetUI2.ui2_password = lpPassword;
NetStatus = NetUseAdd( NULL, 2, (PBYTE)&NetUI2, &BadParm );
if ( NetStatus == ERROR_LOGON_FAILURE )
{
//
// If we passed in an empty domain name, try it again with a NULL one
//
if ( pwszReset == NULL && pwszUser != NULL )
{
NetUI2.ui2_domainname = NULL;
NetStatus = NetUseAdd( NULL, 2, (PBYTE)&NetUI2, &BadParm );
}
}
if ( NetStatus != NERR_Success )
{
NetpKdPrint((PREFIX_NETJOIN "NetUseAdd to %ws returned %lu\n", pwszPath, NetStatus ));
NetpLog(( "NetUseAdd to %ws returned %lu\n", pwszPath, NetStatus ));
if ( NetStatus == ERROR_INVALID_PARAMETER && BadParm != 0 )
{
NetpLog(( "NetUseAdd bad parameter is %lu\n", BadParm ));
}
}
if ( pwszReset != NULL )
{
*pwszReset = L'\\';
}
if ( ( NetStatus == ERROR_NOLOGON_WORKSTATION_TRUST_ACCOUNT ||
NetStatus == ERROR_NOLOGON_SERVER_TRUST_ACCOUNT ||
NetStatus == ERROR_SESSION_CREDENTIAL_CONFLICT ||
NetStatus == ERROR_ACCESS_DENIED ||
NetStatus == ERROR_LOGON_FAILURE ) &&
FLAG_ON( fOptions, NETSETUPP_NULL_SESSION_IPC ) )
{
NetpLog(( "Trying add to %ws using NULL Session\n", pwszPath ));
//
// Try it again with the null session
//
NetUI2.ui2_username = L"";
NetUI2.ui2_domainname = L"";
NetUI2.ui2_password = L"";
NetStatus = NetUseAdd( NULL, 2, (PBYTE)&NetUI2, NULL );
if ( NetStatus != NERR_Success ) {
NetpLog(( "NullSession NetUseAdd to %ws returned %lu\n",
pwszPath, NetStatus ));
}
}
}
return( NetStatus );
}
NET_API_STATUS
NetpBrowserCheckDomain(
IN LPWSTR NewDomainName
)
/*++
Routine Description:
Tell the browser to check a domain/workgroup name
Note: this routine is currently not in use.
Arguments:
NewDomainName - new name of the domain.
Return Value:
Status of the operation.
--*/
{
NET_API_STATUS NetStatus;
NTSTATUS Status;
UNICODE_STRING DeviceName;
IO_STATUS_BLOCK IoStatusBlock;
OBJECT_ATTRIBUTES ObjectAttributes;
HANDLE BrowserHandle = NULL;
LPBYTE Where;
DWORD BytesReturned;
UCHAR PacketBuffer[sizeof(LMDR_REQUEST_PACKET)+2*(DNLEN+1)*sizeof(WCHAR)];
PLMDR_REQUEST_PACKET RequestPacket = (PLMDR_REQUEST_PACKET)PacketBuffer;
//
// Open the browser driver.
//
//
// Open the browser device.
//
RtlInitUnicodeString(&DeviceName, DD_BROWSER_DEVICE_NAME_U);
InitializeObjectAttributes(
&ObjectAttributes,
&DeviceName,
OBJ_CASE_INSENSITIVE,
NULL,
NULL
);
Status = NtOpenFile(
&BrowserHandle,
SYNCHRONIZE,
&ObjectAttributes,
&IoStatusBlock,
0,
0
);
if (NT_SUCCESS(Status)) {
Status = IoStatusBlock.Status;
}
if (!NT_SUCCESS(Status)) {
NetStatus = NetpNtStatusToApiStatus( Status );
goto Cleanup;
}
//
// Build the request packet.
//
RequestPacket->Version = LMDR_REQUEST_PACKET_VERSION_DOM;
RtlInitUnicodeString( &RequestPacket->TransportName, NULL );
RequestPacket->Parameters.DomainRename.ValidateOnly = TRUE;
RtlInitUnicodeString( &RequestPacket->EmulatedDomainName, NULL );
//
// Copy the new domain name into the packet.
//
Where = (LPBYTE) RequestPacket->Parameters.DomainRename.DomainName;
RequestPacket->Parameters.DomainRename.DomainNameLength = wcslen( NewDomainName ) * sizeof(WCHAR);
wcscpy( (LPWSTR)Where, NewDomainName );
Where += RequestPacket->Parameters.DomainRename.DomainNameLength + sizeof(WCHAR);
//
// Send the request to the Datagram Receiver device driver.
//
if ( !DeviceIoControl(
BrowserHandle,
IOCTL_LMDR_RENAME_DOMAIN,
RequestPacket,
(DWORD)(Where - (LPBYTE)RequestPacket),
NULL,
0,
&BytesReturned,
NULL )) {
NetStatus = GetLastError();
goto Cleanup;
}
NetStatus = NO_ERROR;
Cleanup:
if ( BrowserHandle != NULL ) {
NtClose( BrowserHandle );
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpCreateAuthIdentForCreds(
IN PWSTR Account,
IN PWSTR Password,
OUT SEC_WINNT_AUTH_IDENTITY *AuthIdent
)
/*++
Routine Description:
Internal routine to create an AuthIdent structure for the given creditentials
Arguments:
Account - Account name
Password - Password for the account
AuthIdent - AuthIdentity struct to fill in
Returns:
ERROR_SUCCESS - Success
ERROR_NOT_ENOUGH_MEMORY - A memory allocation failed.
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR UserCredentialString = NULL;
PWSTR szUser=NULL;
PWSTR szDomain=NULL;
RtlZeroMemory( AuthIdent, sizeof( SEC_WINNT_AUTH_IDENTITY ) );
//
// If there are no creds, just return
//
if ( Account == NULL )
{
return NERR_Success;
}
NetStatus = NetpSeparateUserAndDomain(Account, &szUser, &szDomain);
if ( NetStatus == NERR_Success )
{
if ( szUser )
{
AuthIdent->User = szUser;
AuthIdent->UserLength = wcslen( szUser );
}
if ( szDomain )
{
AuthIdent->Domain = szDomain;
AuthIdent->DomainLength = wcslen( szDomain );
}
if ( Password )
{
AuthIdent->Password = Password;
AuthIdent->PasswordLength = wcslen( Password );
}
AuthIdent->Flags = SEC_WINNT_AUTH_IDENTITY_UNICODE;
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetSeparatedSubstrings(
IN LPCWSTR szString,
IN WCHAR chSeparator,
OUT LPWSTR* pszS1,
OUT LPWSTR* pszS2
)
{
NET_API_STATUS NetStatus = ERROR_FILE_NOT_FOUND;
LPWSTR szT=NULL;
LPWSTR szS1=NULL;
LPWSTR szS2=NULL;
*pszS1 = NULL;
*pszS2 = NULL;
if (szString && wcschr( szString, chSeparator ))
{
NetStatus = NetpDuplicateString(szString, -1, &szS1);
if ( NetStatus == NERR_Success )
{
szT = wcschr( szS1, chSeparator );
*szT = UNICODE_NULL;
szT++;
NetStatus = NetpDuplicateString(szT, -1, &szS2);
if (NetStatus == NERR_Success)
{
*pszS1 = szS1;
*pszS2 = szS2;
}
}
}
if (NetStatus != NERR_Success)
{
NetApiBufferFree(szS1);
NetApiBufferFree(szS2);
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpSeparateUserAndDomain(
IN LPCWSTR szUserAndDomain,
OUT LPWSTR* pszUser,
OUT LPWSTR* pszDomain
)
{
NET_API_STATUS NetStatus = NERR_Success;
*pszUser = NULL;
*pszDomain = NULL;
//
// check for domain\user format
//
NetStatus = NetpGetSeparatedSubstrings(szUserAndDomain, L'\\',
pszDomain, pszUser);
if (NetStatus == ERROR_FILE_NOT_FOUND)
{
//
// check for user@domain format
//
//NetStatus = NetpGetSeparatedSubstrings(szUserAndDomain, L'@',
// pszUser, pszDomain);
//if (NetStatus == ERROR_FILE_NOT_FOUND)
//{
//
// domain not specified, szUserAndDomain specifies a user
// (may be in the UPN format)
//
NetStatus = NetpDuplicateString(szUserAndDomain, -1, pszUser);
//}
}
return NetStatus;
}
VOID
NET_API_FUNCTION
NetpFreeAuthIdentForCreds(
IN PSEC_WINNT_AUTH_IDENTITY AuthIdent
)
/*++
Routine Description:
Free the authident structure allocated above
Arguments:
AuthIdent - AuthIdentity struct to free
Returns:
VOID
--*/
{
if ( AuthIdent )
{
NetApiBufferFree( AuthIdent->User );
NetApiBufferFree( AuthIdent->Domain );
}
}
NET_API_STATUS
NET_API_FUNCTION
NetpLdapUnbind(
IN PLDAP Ldap
)
/*++
Routine Description:
Unbinds a current ldap connection
Arguments:
Ldap -- Connection to be severed
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
if ( Ldap != NULL ) {
NetStatus = LdapMapErrorToWin32( ldap_unbind( Ldap ) );
}
NetpLog(( "ldap_unbind status: 0x%lx\n", NetStatus ));
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpLdapBind(
IN LPWSTR szUncDcName,
IN LPWSTR szUser,
IN LPWSTR szPassword,
OUT PLDAP *pLdap
)
/*++
Routine Description:
Binds to the named server using the given credentials
Arguments:
szUncDcName -- DC to connect to
szUser -- User name to bind with
szPassword -- Password to use for bind
pLdap -- Where the connection handle is returned
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
SEC_WINNT_AUTH_IDENTITY AuthId = {0}, *pAuthId = NULL;
LONG LdapOption;
ULONG LdapStatus = LDAP_SUCCESS;
//
// Initialization
//
*pLdap = NULL;
if ( szUser ) {
NetStatus = NetpCreateAuthIdentForCreds( szUser, szPassword, &AuthId );
pAuthId = &AuthId;
}
if ( NetStatus == NERR_Success ) {
//
// Open an LDAP connection to the DC and set useful options
//
*pLdap = ldap_initW( szUncDcName + 2, LDAP_PORT );
if ( *pLdap ) {
//
// Tell LDAP we are passing an explicit DC name
// to avoid the DC discovery
//
LdapOption = PtrToLong( LDAP_OPT_ON );
LdapStatus = ldap_set_optionW( *pLdap,
LDAP_OPT_AREC_EXCLUSIVE,
&LdapOption );
if ( LdapStatus != LDAP_SUCCESS ) {
NetpLog(( "NetpLdapBind: ldap_set_option LDAP_OPT_AREC_EXCLUSIVE failed on %ws: %ld: %s\n",
szUncDcName,
LdapStatus,
ldap_err2stringA(LdapStatus) ));
NetStatus = LdapMapErrorToWin32( LdapStatus );
//
// Do the bind
//
} else {
LdapStatus = ldap_bind_sW( *pLdap,
NULL,
(PWSTR) pAuthId,
LDAP_AUTH_NEGOTIATE );
if ( LdapStatus != LDAP_SUCCESS ) {
NetpLog(( "NetpLdapBind: ldap_bind failed on %ws: %ld: %s\n",
szUncDcName,
LdapStatus,
ldap_err2stringA(LdapStatus) ));
NetStatus = LdapMapErrorToWin32( LdapStatus );
}
}
if ( NetStatus != NERR_Success ) {
NetpLdapUnbind( *pLdap );
*pLdap = NULL;
}
} else {
LdapStatus = LdapGetLastError();
NetpLog(( "NetpLdapBind: ldap_init to %ws failed: %lu\n",
szUncDcName,
LdapStatus ));
NetStatus = LdapMapErrorToWin32( LdapStatus );
}
NetpFreeAuthIdentForCreds( pAuthId );
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetNCRoot(
IN PLDAP Ldap,
OUT LPWSTR *NCRoot,
OUT PBOOLEAN SupportsPageable
)
/*++
Routine Description:
This routine determines the DS root for the given domain and determines whether this
server supports pageable searches
Arguments:
Ldap -- Connection to the server
NCRoot -- Where the root is returned. Must be freed via NetApiBufferFree
SupportsPageable -- if TRUE, this server supports pageable searches
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR Attribs[3] = {
L"defaultNamingContext",
L"supportedControl",
NULL
};
PWSTR *Values = NULL;
LDAPMessage *Message=NULL, *Entry;
ULONG Items, i;
NetStatus = LdapMapErrorToWin32( ldap_search_s( Ldap, NULL, LDAP_SCOPE_BASE,
NETSETUPP_ALL_FILTER, Attribs, 0, &Message ) );
if ( NetStatus == NERR_Success ) {
Entry = ldap_first_entry( Ldap, Message );
if ( Entry ) {
//
// Now, we'll have to get the values
//
Values = ldap_get_values( Ldap, Entry, Attribs[ 0 ] );
if ( Values ) {
NetStatus = NetpDuplicateString(Values[ 0 ], -1, NCRoot);
ldap_value_free( Values );
} else {
NetStatus = LdapMapErrorToWin32( Ldap->ld_errno );
}
//
// Now, see if we have the right control bits to do pageable stuff
//
if ( NetStatus == NERR_Success ) {
Values = ldap_get_values( Ldap, Entry, Attribs[ 1 ] );
if ( Values ) {
Items = ldap_count_values( Values );
for ( i = 0; i < Items ; i++ ) {
if ( _wcsicmp( Values[ i ], LDAP_PAGED_RESULT_OID_STRING_W ) == 0 ) {
*SupportsPageable = TRUE;
break;
}
}
ldap_value_free( Values );
} else {
NetStatus = LdapMapErrorToWin32( Ldap->ld_errno );
}
}
} else {
NetStatus = LdapMapErrorToWin32( Ldap->ld_errno );
}
}
if ( NetStatus != NERR_Success ) {
NetpLog(( "Failed to find the root NC: %lu\n", NetStatus ));
}
//Cleanup:
if (Message)
{
ldap_msgfree( Message );
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetDefaultJoinableOu(
IN LPWSTR Root,
IN PLDAP Ldap,
OUT PWSTR *DefaultOu
)
/*++
Routine Description:
This routine searches for all the OUs under the given domain root under which the bound
user has the rights to create a computer object
This routine does pageable searches
Arguments:
Root -- Root NC path
Ldap -- Connection to the server
DefaultOu - Where the default joinable ou is returned. NULL if no default joinable ou was
found
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR Attribs[] = {
NETSETUPP_WELL_KNOWN,
NULL
};
ULONG Count, Status, i, StringLength;
PWSTR *WKOs = NULL, *Classes = NULL;
LDAPMessage *Message = NULL, *Entry, *Message2 = NULL, *Entry2;
PWSTR ParseString, End, DN = NULL;
BOOLEAN MatchFound = FALSE;
*DefaultOu = NULL;
NetpLog(( "Default OU search\n" ));
//
// Ok, first, read the list of WellKnownObjects off of the root
//
Status = ldap_search_s( Ldap,
Root,
LDAP_SCOPE_BASE,
NETSETUPP_ALL_FILTER,
Attribs,
0,
&Message );
if ( Message ) {
Entry = ldap_first_entry( Ldap, Message );
while ( Status == LDAP_SUCCESS && Entry ) {
//
// Read the list of objects that the current user is allowed to
// create under this OU and make sure that we can create a computer
// object
//
WKOs = ldap_get_values( Ldap, Entry, Attribs[ 0 ] );
if ( WKOs ) {
i = 0;
while ( WKOs[ i ] ) {
if ( !toupper( WKOs[ i ][ 0 ] ) == L'B' ) {
NetpLog(( "Unexpected object string %ws\n",
WKOs[ i ] ));
i++;
continue;
}
ParseString = WKOs[ i ] + 2;
StringLength = wcstoul( ParseString, &End, 10 );
ParseString = End + 1; // Skip over the ':'
if ( _wcsnicmp( ParseString,
L"AA312825768811D1ADED00C04FD8D5CD",
StringLength ) == 0 ) {
MatchFound = TRUE;
ParseString += StringLength + 1;
//
// Now, see if it is accessible or not
//
Attribs[ 0 ] = NETSETUPP_RETURNED_ATTR;
Status = ldap_search_s( Ldap,
Root,
LDAP_SCOPE_BASE,
NETSETUPP_ALL_FILTER,
Attribs,
0,
&Message2 );
if ( Message2 ) {
Entry2 = ldap_first_entry( Ldap, Message2 );
while ( Status == LDAP_SUCCESS && Entry2 ) {
//
// Read the list of objects that the current user is allowed to
// create under this OU and make sure that we can create a computer
// object
//
Classes = ldap_get_values( Ldap, Entry2, Attribs[ 0 ] );
if ( Classes ) {
i = 0;
while ( Classes[ i ] ) {
if ( _wcsicmp( Classes[ i ],
NETSETUPP_COMPUTER_OBJECT ) == 0 ) {
DN = ldap_get_dn( Ldap, Entry2 );
if ( DN != NULL ) {
NetStatus = NetpDuplicateString(DN, -1,
DefaultOu);
ldap_memfree( DN );
DN = NULL;
break;
}
}
i++;
}
ldap_value_free( Classes );
}
//
// If we found the entry or failed to allocate memory,
// we are done
//
if ( *DefaultOu != NULL || NetStatus != NERR_Success ) {
break;
}
Entry2 = ldap_next_entry( Ldap, Entry2 );
}
Status = Ldap->ld_errno;
ldap_msgfree( Message2 );
}
if ( NetStatus != NERR_Success || MatchFound ) {
break;
}
}
i++;
}
ldap_value_free( WKOs );
}
Entry = ldap_next_entry( Ldap, Entry );
}
Status = Ldap->ld_errno;
ldap_msgfree( Message );
}
if ( NetStatus == NERR_Success ) {
NetStatus = LdapMapErrorToWin32( Status );
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetListOfJoinableOUsPaged(
IN LPWSTR Root,
IN PLDAP Ldap,
OUT PULONG OUCount,
OUT PWSTR **OUs
)
/*++
Routine Description:
This routine searches for all the OUs under the given domain root under which the bound
user has the rights to create a computer object
This routine does pageable searches
Arguments:
Root -- Root NC path
Ldap -- Connection to the server
OUCount -- Where the count of strings is returned
OUs -- Where the list of OUs is returned
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PLDAPSearch SearchHandle = NULL;
PWSTR Attribs[] = {
NETSETUPP_RETURNED_ATTR,
NULL
};
ULONG Count, i;
ULONG Status = LDAP_SUCCESS;
PWSTR *Classes = NULL;
LDAPMessage *Message = NULL, *Entry;
PWSTR DN;
PWSTR *DnList = NULL, *NewList = NULL;
ULONG CurrentIndex = 0, ListCount = 0;
PWSTR DefaultOu = NULL;
NetpLog(( "PAGED OU search\n" ));
//
// Initialize the pageable search
//
SearchHandle = ldap_search_init_pageW( Ldap,
Root,
LDAP_SCOPE_SUBTREE,
NETSETUPP_OU_FILTER,
Attribs,
FALSE,
NULL,
NULL,
0,
2000,
NULL );
if ( SearchHandle == NULL ) {
NetStatus = LdapMapErrorToWin32( LdapGetLastError( ) );
} else {
while ( NetStatus == NERR_Success ) {
Count = 0;
//
// Get the next page
//
Status = ldap_get_next_page_s( Ldap,
SearchHandle,
NULL,
100,
&Count,
&Message );
if ( Message ) {
//
// Process all of the entries
//
Entry = ldap_first_entry( Ldap, Message );
while ( Status == LDAP_SUCCESS && Entry ) {
//
// Read the list of classes that the current user is allowed to
// create under this OU and make sure that we can create a computer
// object
//
Classes = ldap_get_values( Ldap, Entry, Attribs[ 0 ] );
if ( Classes ) {
i = 0;
while ( Classes[ i ] ) {
if ( _wcsicmp( Classes[ i ], NETSETUPP_COMPUTER_OBJECT ) == 0 ) {
DN = ldap_get_dn( Ldap, Entry );
NetpKdPrint(( PREFIX_NETJOIN "DN = %ws\n", DN ));
//
// We'll allocate the return list in blocks of 10 to cut
// down on the number of allocations
//
if ( DN != NULL ) {
if ( CurrentIndex >= ListCount ) {
if ( NetApiBufferAllocate( ( ListCount + 10 ) * sizeof( PWSTR ),
( PVOID * )&NewList ) != NERR_Success ) {
Status = LDAP_NO_MEMORY;
} else {
RtlZeroMemory( NewList, ( ListCount + 10 ) * sizeof( PWSTR ) );
RtlCopyMemory( NewList,
DnList,
ListCount * sizeof( PWSTR ) );
ListCount += 10;
NetApiBufferFree( DnList );
DnList = NewList;
}
}
//
// Copy the string
//
if ( Status == LDAP_SUCCESS ) {
if (NERR_Success ==
NetpDuplicateString(DN, -1,
&NewList[CurrentIndex]))
{
CurrentIndex++;
}
else
{
Status = LDAP_NO_MEMORY;
}
}
ldap_memfree( DN );
DN = NULL;
}
break;
}
i++;
}
ldap_value_free( Classes );
}
Entry = ldap_next_entry( Ldap, Entry );
}
Status = Ldap->ld_errno;
ldap_msgfree( Message );
Message = NULL;
}
if ( Status == LDAP_NO_RESULTS_RETURNED ) {
Status = LDAP_SUCCESS;
break;
}
}
ldap_search_abandon_page( Ldap,
SearchHandle );
NetStatus = LdapMapErrorToWin32( Status );
}
//
// Check the computers container
//
if ( NetStatus == NERR_Success ) {
NetStatus = NetpGetDefaultJoinableOu( Root,
Ldap,
&DefaultOu );
if ( NetStatus == NERR_Success && DefaultOu ) {
//
// We'll allocate the return list in blocks of 10 to cut
// down on the number of allocations
//
if ( CurrentIndex >= ListCount ) {
if ( NetApiBufferAllocate( ( ListCount + 10 ) * sizeof( PWSTR ),
( PVOID * )&NewList ) != NERR_Success ) {
Status = LDAP_NO_MEMORY;
} else {
RtlZeroMemory( NewList, ( ListCount + 10 ) * sizeof( PWSTR ) );
RtlCopyMemory( NewList,
DnList,
ListCount * sizeof( PWSTR ) );
ListCount += 10;
NetApiBufferFree( DnList );
DnList = NewList;
}
}
//
// Copy the string
//
if ( Status == LDAP_SUCCESS ) {
if (NERR_Success ==
NetpDuplicateString(DefaultOu, -1, &NewList[CurrentIndex]))
{
CurrentIndex++;
}
else
{
Status = LDAP_NO_MEMORY;
}
}
NetApiBufferFree( DefaultOu );
}
}
//
// If there was an error, free everyting
//
if ( NetStatus != NERR_Success ) {
for ( i = 0; i < ListCount; i++ ) {
NetApiBufferFree( DnList[ i ] );
}
NetApiBufferFree( DnList );
} else {
*OUs = DnList;
*OUCount = CurrentIndex;
}
if ( NetStatus == NERR_Success ) {
NetpLog(( "Found %lu OUs\n", *OUs ));
} else {
NetpLog(( "Failed to obtain the list of joinable OUs: %lu\n",
NetStatus ));
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetListOfJoinableOUsNonPaged(
IN LPWSTR Root,
IN PLDAP Ldap,
OUT PULONG OUCount,
OUT PWSTR **OUs
)
/*++
Routine Description:
This routine searches for all the OUs under the given domain root under which the bound
user has the rights to create a computer object
This routine does not use pageable searchs and will return only the max_search count
of entries
Arguments:
Root -- Root NC path
Ldap -- Connection to the server
OUCount -- Where the count of strings is returned
OUs -- Where the list of OUs is returned
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR Attribs[] = {
NETSETUPP_RETURNED_ATTR,
NULL
};
ULONG Count, Status, i;
PWSTR *Classes = NULL;
LDAPMessage *Message = NULL, *Entry;
PWSTR DN;
PWSTR *DnList = NULL, *NewList = NULL;
ULONG CurrentIndex = 0, ListCount = 0;
PWSTR DefaultOu = NULL;
NetpLog(( "Normal OU search\n" ));
Status = ldap_search_s( Ldap,
Root,
LDAP_SCOPE_SUBTREE,
NETSETUPP_OU_FILTER,
Attribs,
0,
&Message );
if ( Message ) {
Entry = ldap_first_entry( Ldap, Message );
while ( Status == LDAP_SUCCESS && Entry ) {
//
// Read the list of classes that the current user is allowed to
// create under this OU and make sure that we can create a computer
// object
//
Classes = ldap_get_values( Ldap, Entry, Attribs[ 0 ] );
if ( Classes ) {
i = 0;
while ( Classes[ i ] ) {
if ( _wcsicmp( Classes[ i ], NETSETUPP_COMPUTER_OBJECT ) == 0 ) {
DN = ldap_get_dn( Ldap, Entry );
//
// We'll allocate the return list in blocks of 10 to cut
// down on the number of allocations
//
if ( CurrentIndex >= ListCount ) {
if ( NetApiBufferAllocate( ( ListCount + 10 ) * sizeof( PWSTR ),
( PVOID * )&NewList ) != NERR_Success ) {
Status = LDAP_NO_MEMORY;
} else {
RtlCopyMemory( NewList,
DnList,
ListCount * sizeof( PWSTR ) );
ListCount += 10;
DnList = NewList;
}
}
//
// Copy the string
//
if ( Status == LDAP_SUCCESS ) {
if (NERR_Success ==
NetpDuplicateString(DN, -1, &NewList[CurrentIndex]))
{
CurrentIndex++;
}
else
{
Status = LDAP_NO_MEMORY;
}
}
ldap_memfree( DN );
break;
}
i++;
}
ldap_value_free( Classes );
}
Entry = ldap_next_entry( Ldap, Entry );
}
Status = Ldap->ld_errno;
ldap_msgfree( Message );
}
NetStatus = LdapMapErrorToWin32( Status );
//
// Check the computers container
//
if ( NetStatus == NERR_Success ) {
NetStatus = NetpGetDefaultJoinableOu( Root,
Ldap,
&DefaultOu );
if ( NetStatus == NERR_Success && DefaultOu ) {
//
// We'll allocate the return list in blocks of 10 to cut
// down on the number of allocations
//
if ( CurrentIndex >= ListCount ) {
if ( NetApiBufferAllocate( ( ListCount + 10 ) * sizeof( PWSTR ),
( PVOID * )&NewList ) != NERR_Success ) {
Status = LDAP_NO_MEMORY;
} else {
RtlCopyMemory( NewList,
DnList,
ListCount * sizeof( PWSTR ) );
ListCount += 10;
DnList = NewList;
}
}
//
// Copy the string
//
if ( Status == LDAP_SUCCESS ) {
if (NERR_Success ==
NetpDuplicateString(DefaultOu, -1, &NewList[CurrentIndex]))
{
CurrentIndex++;
}
else
{
Status = LDAP_NO_MEMORY;
}
}
NetApiBufferFree( DefaultOu );
}
}
//
// If there was an error, free everyting
//
if ( NetStatus != NERR_Success ) {
for ( i = 0; i < ListCount; i++ ) {
NetApiBufferFree( DnList[ i ] );
}
NetApiBufferFree( DnList );
} else {
*OUs = DnList;
*OUCount = CurrentIndex;
}
if ( NetStatus == NERR_Success ) {
NetpLog(( "Found %lu OUs\n", *OUs ));
} else {
NetpLog(( "Failed to obtain the list of joinable OUs: %lu\n",
NetStatus ));
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetListOfJoinableOUs(
IN LPWSTR Domain,
IN LPWSTR Account,
IN LPWSTR Password,
OUT PULONG Count,
OUT PWSTR **OUs
)
/*++
Routine Description:
This routine searches for all the OUs under the given domain root under which the bound
user has the rights to create a computer object
Arguments:
Domain -- Domain under which to find all of the OUs under which a computer object can be
created
Account -- Account to use for the LDAP bind
Password -- Password to used for the bind. The password is encoded. The first WCHAR of the
password is the seed.
OUCount -- Where the count of strings is returned
OUs -- Where the list of OUs is returned
Returns:
NERR_Success -- Success
NERR_DefaultJoinRequired -- The servers for this domain do not support the DS so the computer
account can only be created under the default container (for NT4, this is the SAM account
database)
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR DomainControllerName = NULL;
ULONG DcFlags = 0;
PLDAP Ldap = NULL;
PWSTR NCRoot;
BOOLEAN Pageable = FALSE;
UCHAR Seed;
UNICODE_STRING EncodedPassword;
if ( Password ) {
if ( wcslen( Password ) < 1 ) {
return( ERROR_INVALID_PARAMETER );
}
Seed = ( UCHAR )*Password;
RtlInitUnicodeString( &EncodedPassword, Password + 1 );
} else {
RtlZeroMemory( &EncodedPassword, sizeof( UNICODE_STRING ) );
Seed = 0;
}
NetSetuppOpenLog();
//
// First, find a DC in the destination domain
//
NetStatus = NetpDsGetDcName( NULL,
Domain,
NULL,
NETSETUPP_DSGETDC_FLAGS,
&DcFlags,
&DomainControllerName
,NULL
);
if ( NetStatus == NERR_Success ) {
//
// Try and bind to the server
//
RtlRunDecodeUnicodeString( Seed, &EncodedPassword );
NetStatus = NetpLdapBind( DomainControllerName,
Account,
EncodedPassword.Buffer,
&Ldap );
RtlRunEncodeUnicodeString( &Seed, &EncodedPassword );
if ( NetStatus == NERR_Success ) {
//
// Get the X500 domain name
//
NetStatus = NetpGetNCRoot( Ldap,
&NCRoot,
&Pageable );
if ( NetStatus == NERR_Success ) {
//
// Get the list of OUs
//
if ( Pageable ) {
NetStatus = NetpGetListOfJoinableOUsPaged( NCRoot,
Ldap,
Count,
OUs );
} else {
NetStatus = NetpGetListOfJoinableOUsNonPaged( NCRoot,
Ldap,
Count,
OUs );
}
NetApiBufferFree( NCRoot );
}
NetpLdapUnbind( Ldap );
} else if ( NetStatus == ERROR_BAD_NET_RESP ) {
NetStatus = NERR_DefaultJoinRequired;
}
NetApiBufferFree( DomainControllerName );
}
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpGetListOfJoinableOUs failed with %lu\n",
NetStatus ));
}
NetSetuppCloseLog( );
return( NetStatus );
}
NET_API_STATUS
NetpGetDnsHostName(
IN LPWSTR PassedHostName OPTIONAL,
IN PUNICODE_STRING DnsDomainName,
IN BOOL UseGpSuffix,
OUT LPWSTR *DnsHostName
)
/*++
Routine Description:
This routine determines the value of DnsHostName attribute to be set on the
computer object in the DS. DnsHostName is <HostName.PrimaryDnsSuffix>.
Here HostName is a computer name which may be different from the Netbios name;
Netbios name is at most 15 characters of HostName. PrimaryDnsSuffix can be
set through Policy or through the UI or can be defaulted to the DNS name of the
domain being joined; policy setting takes preference.
This routine determines *new* values for HostName and PrimaryDnsSuffix which
will be applied after the reboot. Thus DnsHostName will have the correct value
after the machine reboots.
Arguments:
PassedHostName - The host name of this machine (can be longer than 15 chars).
If NULL, the host name is read from the registry.
DnsDomainName - DNS name of the domain being joined
UseGpSuffix - If TRUE, the primary DNS suffix that comes down through
policy will be used.
DnsHostname - Returns the value of DnsHostName. Must be freed by calling
NetApiBufferFree.
Returns:
NO_ERROR - Success
ERROR_NOT_ENOUGH_MEMORY - There was not enough memory to read the data from
registry
ERROR_INVALID_COMPUTERNAME - It was not possible to determine DnsHostName from
the registry
--*/
{
LONG RegStatus;
HKEY Key = NULL;
DWORD Type;
NET_API_STATUS NetStatus;
PWSTR HostName = NULL;
PWSTR PrimaryDnsSuffix = NULL;
LPWSTR LocalDnsHostName;
DWORD Size = 0;
//
// First detemine HostName.
//
// If it's passed, use it
//
if ( PassedHostName != NULL ) {
HostName = PassedHostName;
//
// Otherwise, read it from teh registry
//
} else {
RegStatus = RegOpenKeyExW( HKEY_LOCAL_MACHINE,
L"System\\CurrentControlSet\\Services\\Tcpip\\Parameters",
0,
KEY_QUERY_VALUE,
&Key );
//
// Not having host name is critical -- error out in such case
//
if ( RegStatus != ERROR_SUCCESS ) {
NetpLog(( "NetpGetDnsHostName: Cannot open TCPIP parameters: 0x%lx\n", RegStatus ));
} else {
//
// First try to read the new value
//
RegStatus = RegQueryValueExW( Key,
L"NV Hostname",
0,
&Type,
NULL,
&Size );
if ( RegStatus == ERROR_SUCCESS && Size != 0 ) {
HostName = LocalAlloc( 0, Size );
if ( HostName == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
RegStatus = RegQueryValueExW( Key,
L"NV Hostname",
0,
&Type,
(PUCHAR) HostName,
&Size );
if ( RegStatus != ERROR_SUCCESS ) {
NetpLog(( "NetpGetDnsHostName: Cannot read NV Hostname: 0x%lx\n", RegStatus ));
NetStatus = ERROR_INVALID_COMPUTERNAME;
goto Cleanup;
} else {
NetpLog(( "NetpGetDnsHostName: Read NV Hostname: %ws\n", HostName ));
}
}
//
// If the new value does not exist for some reason,
// try to read the currently active one
//
if ( HostName == NULL ) {
RegStatus = RegQueryValueExW( Key,
L"Hostname",
0,
&Type,
NULL,
&Size );
if ( RegStatus == ERROR_SUCCESS && Size != 0 ) {
HostName = LocalAlloc( 0, Size );
if ( HostName == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
RegStatus = RegQueryValueExW( Key,
L"Hostname",
0,
&Type,
(PUCHAR) HostName,
&Size );
if ( RegStatus != ERROR_SUCCESS ) {
NetpLog(( "NetpGetDnsHostName: Cannot read Hostname: 0x%lx\n", RegStatus ));
NetStatus = ERROR_INVALID_COMPUTERNAME;
goto Cleanup;
} else {
NetpLog(( "NetpGetDnsHostName: Read Hostname: %ws\n", HostName ));
}
}
}
}
}
//
// If we couldn't get HostName, something's really bad
//
if ( HostName == NULL ) {
NetpLog(( "NetpGetDnsHostName: Could not get Hostname\n" ));
NetStatus = ERROR_INVALID_COMPUTERNAME;
goto Cleanup;
}
if ( Key != NULL ) {
RegCloseKey( Key );
Key = NULL;
}
//
// Second read primary DNS suffix of this machine.
//
// Try the suffix that comes down through policy first
//
if ( UseGpSuffix ) {
RegStatus = RegOpenKeyExW( HKEY_LOCAL_MACHINE,
L"Software\\Policies\\Microsoft\\System\\DNSclient",
0,
KEY_QUERY_VALUE,
&Key );
if ( RegStatus == 0 ) {
//
// Read only the new value; if it doesn't exist the
// current value will be deleted after the reboot
//
RegStatus = RegQueryValueExW( Key,
L"NV PrimaryDnsSuffix",
0,
&Type,
NULL,
&Size );
if ( RegStatus == ERROR_SUCCESS && Size != 0 ) {
PrimaryDnsSuffix = LocalAlloc( 0, Size );
if ( PrimaryDnsSuffix == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
RegStatus = RegQueryValueExW( Key,
L"NV PrimaryDnsSuffix",
0,
&Type,
(PUCHAR) PrimaryDnsSuffix,
&Size );
if ( RegStatus != ERROR_SUCCESS ) {
NetpLog(( "NetpGetDnsHostName: Cannot read NV PrimaryDnsSuffix: 0x%lx\n", RegStatus ));
NetStatus = ERROR_INVALID_COMPUTERNAME;
goto Cleanup;
} else {
NetpLog(( "NetpGetDnsHostName: Read NV PrimaryDnsSuffix: %ws\n", PrimaryDnsSuffix ));
}
}
}
}
//
// If there is no policy setting for PrimaryDnsSuffix,
// get it from the TCPIP setting
//
if ( Key != NULL ) {
RegCloseKey( Key );
Key = NULL;
}
if ( PrimaryDnsSuffix == NULL ) {
RegStatus = RegOpenKeyExW( HKEY_LOCAL_MACHINE,
L"System\\CurrentControlSet\\Services\\Tcpip\\Parameters",
0,
KEY_QUERY_VALUE,
&Key );
if ( RegStatus == ERROR_SUCCESS ) {
ULONG SyncValue;
Size = sizeof( ULONG );
RegStatus = RegQueryValueEx( Key,
L"SyncDomainWithMembership",
0,
&Type,
(PUCHAR)&SyncValue,
&Size );
//
// If we are not to sync DNS suffix with the name of the
// domain that we join, get the configured suffix
//
if ( RegStatus == ERROR_SUCCESS && SyncValue == 0 ) {
//
// Read the new value
//
RegStatus = RegQueryValueExW( Key,
L"NV Domain",
0,
&Type,
NULL,
&Size );
if ( RegStatus == ERROR_SUCCESS && Size != 0 ) {
PrimaryDnsSuffix = LocalAlloc( 0, Size );
if ( PrimaryDnsSuffix == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
RegStatus = RegQueryValueExW( Key,
L"NV Domain",
0,
&Type,
(PUCHAR) PrimaryDnsSuffix,
&Size );
if ( RegStatus != ERROR_SUCCESS ) {
NetpLog(( "NetpGetDnsHostName: Cannot read NV Domain: 0x%lx\n", RegStatus ));
NetStatus = ERROR_INVALID_COMPUTERNAME;
goto Cleanup;
} else {
NetpLog(( "NetpGetDnsHostName: Read NV Domain: %ws\n", PrimaryDnsSuffix ));
}
}
//
// If the new value does not exist for some reason,
// read the currently active one
//
if ( PrimaryDnsSuffix == NULL ) {
RegStatus = RegQueryValueExW( Key,
L"Domain",
0,
&Type,
NULL,
&Size );
if ( RegStatus == ERROR_SUCCESS && Size != 0 ) {
PrimaryDnsSuffix = LocalAlloc( 0, Size );
if ( PrimaryDnsSuffix == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
RegStatus = RegQueryValueExW( Key,
L"Domain",
0,
&Type,
(PUCHAR) PrimaryDnsSuffix,
&Size );
if ( RegStatus != ERROR_SUCCESS ) {
NetpLog(( "NetpGetDnsHostName: Cannot read Domain: 0x%lx\n", RegStatus ));
NetStatus = ERROR_INVALID_COMPUTERNAME;
goto Cleanup;
} else {
NetpLog(( "NetpGetDnsHostName: Read Domain: %ws\n", PrimaryDnsSuffix ));
}
}
}
}
}
}
//
// If we still have no PrimaryDnsSuffix, use DNS name of the domain we join
//
if ( PrimaryDnsSuffix == NULL ) {
NetpLog(( "NetpGetDnsHostName: PrimaryDnsSuffix defaulted to DNS domain name: %wZ\n", DnsDomainName ));
PrimaryDnsSuffix = LocalAlloc( 0, DnsDomainName->Length + sizeof(WCHAR) );
if ( PrimaryDnsSuffix == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
RtlCopyMemory( PrimaryDnsSuffix,
DnsDomainName->Buffer,
DnsDomainName->Length );
PrimaryDnsSuffix[ (DnsDomainName->Length)/sizeof(WCHAR) ] = UNICODE_NULL;
}
//
// Now we have Hostname and Primary DNS suffix.
// Connect them with . to form DnsHostName.
//
NetStatus = NetApiBufferAllocate(
(wcslen(HostName) + 1 + wcslen(PrimaryDnsSuffix) + 1) * sizeof(WCHAR),
&LocalDnsHostName );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
wcscpy( LocalDnsHostName, HostName );
wcscat( LocalDnsHostName, L"." );
wcscat( LocalDnsHostName, PrimaryDnsSuffix );
//
// If we are here, it's a success
//
*DnsHostName = LocalDnsHostName;
NetStatus = NO_ERROR;
Cleanup:
if ( Key != NULL ) {
RegCloseKey( Key );
}
if ( HostName != NULL && HostName != PassedHostName ) {
LocalFree( HostName );
}
if ( PrimaryDnsSuffix != NULL ) {
LocalFree( PrimaryDnsSuffix );
}
return NetStatus;
}
VOID
NetpRemoveDuplicateStrings(
IN PWCHAR *Source,
IN OUT PWCHAR *Target
)
/*++
Routine Description:
This routine accepts two pointer arrays and removes those entries
from the target array which point to strings that are identical to
one of the strings pointed to by the entries in the source array.
On return, the target array entries which precede the NULL terminator
will point to strings which are different from any of the strings
pointed to by the source array elements.
Arguments:
Source -- The NULL terminated array of pointes to source strings.
For example:
Source[0] = L"abc";
Source[1] = L"def";
Source[2] = NULL;
Target -- The NULL terminated array of pointes to target strings.
For example:
Target[0] = L"abc";
Target[1] = L"ghi";
Target[2] = L"def";
Target[3] = NULL;
On return, the Target array will be, for our example:
Target[0] = L"ghi";
Target[1] = NULL;
Target[2] = L"def";
Target[3] = NULL;
Note that, on return, the target array has size of 1 and
contains only one valid pointer.
Returns:
VOID
--*/
{
PWCHAR *TargetPtr, *TargetNextPtr, *SourcePtr;
BOOL KeepEntry;
//
// Sanity check
//
if ( Source == NULL || *Source == NULL ||
Target == NULL || *Target == NULL ) {
return;
}
//
// Loop through the target and compare with the source
//
for ( TargetPtr = TargetNextPtr = Target;
*TargetNextPtr != NULL;
TargetNextPtr++ ) {
KeepEntry = TRUE;
for ( SourcePtr = Source; *SourcePtr != NULL; SourcePtr++ ) {
if ( _wcsicmp( *SourcePtr, *TargetNextPtr ) == 0 ) {
KeepEntry = FALSE;
break;
}
}
if ( KeepEntry ) {
*TargetPtr = *TargetNextPtr;
TargetPtr ++;
}
}
//
// Terminate the target array
//
*TargetPtr = NULL;
return;
}
DWORD
NetpCrackNamesStatus2Win32Error(
DWORD dwStatus
)
{
switch (dwStatus) {
case DS_NAME_ERROR_RESOLVING:
return ERROR_DS_NAME_ERROR_RESOLVING;
case DS_NAME_ERROR_NOT_FOUND:
return ERROR_DS_NAME_ERROR_NOT_FOUND;
case DS_NAME_ERROR_NOT_UNIQUE:
return ERROR_DS_NAME_ERROR_NOT_UNIQUE;
case DS_NAME_ERROR_NO_MAPPING:
return ERROR_DS_NAME_ERROR_NO_MAPPING;
case DS_NAME_ERROR_DOMAIN_ONLY:
return ERROR_DS_NAME_ERROR_DOMAIN_ONLY;
case DS_NAME_ERROR_NO_SYNTACTICAL_MAPPING:
return ERROR_DS_NAME_ERROR_NO_SYNTACTICAL_MAPPING;
}
return ERROR_FILE_NOT_FOUND;
}
//
// Machine account attributes in the DS
//
#define NETSETUPP_OBJECTCLASS L"objectClass"
#define NETSETUPP_SAMACCOUNTNAME L"SamAccountName"
#define NETSETUPP_DNSHOSTNAME L"DnsHostName"
#define NETSETUPP_SERVICEPRINCIPALNAME L"ServicePrincipalName"
#define NETSETUPP_USERACCOUNTCONTROL L"userAccountControl"
#define NETSETUPP_UNICODEPWD L"unicodePwd"
#define NETSETUPP_ORGANIZATIONALUNIT L"OrganizationalUnit"
#define NETSETUPP_HOST_SPN_PREFIX L"HOST/"
#define NETSETUPP_COMP_OBJ_ATTR_COUNT 6
#define NETSETUPP_MULTIVAL_ATTRIB 0x01
#define NETSETUPP_COMPUTER_CONTAINER_GUID_IN_B32_FORM L"B:32:" GUID_COMPUTRS_CONTAINER_W L":"
typedef struct _NETSETUPP_MACH_ACC_ATTRIBUTE {
PWSTR AttribType; // Type of the attribute
ULONG AttribFlags; // Attribute flags
PWSTR *AttribValues; // Values of the attribute
} NETSETUPP_MACH_ACC_ATTRIBUTE, *PNETSETUPP_MACH_ACC_ATTRIBUTE;
NET_API_STATUS
NET_API_FUNCTION
NetpGetComputerObjectDn(
IN PDOMAIN_CONTROLLER_INFO DcInfo,
IN LPWSTR Account,
IN LPWSTR Password,
IN PLDAP Ldap,
IN LPWSTR ComputerName,
IN LPWSTR OU OPTIONAL,
OUT LPWSTR *ComputerObjectDn
)
/*++
Routine Description:
Get the DN for the computer account in the specified OU.
The algorithm is as follows.
First try to get the DN of the pre-existing account (if any)
by cracking the account name into a DN. If that succeeds, verify
that the passed OU (if any) matches the cracked DN. If the OU
matches, return success, otherwise return error (ERROR_FILE_EXISTS).
If no OU is not passed, simply return the cracked DN.
If the account does not exist, verify that the passed OU
(if any) exists. If so, build the DN from the computer name and
the OU and return it. If no OU is passed, get the default computer
container name (by reading the WellKnownObjects attribute) and build
the DN using the computer name and the default computer container DN.
Arguments:
DcInfo - Domain controller on which to create the object
Account - Account to use for the LDAP bind
Password - Password to used for the bind
Ldap - Ldap binding to the DC
ComputerName - Name of the computer being joined
OU - OU under which to create the object.
The name must be a fully qualified name
e.g.: "ou=test,dc=ntdev,dc=microsoft,dc=com"
NULL indicates to use the default computer container
ComputerObjectDn - Returns the DN of the computer object.
The retuned buffer must be freed using NetApiBufferFree
Returns:
NO_ERROR -- Success
ERROR_DS_NAME_ERROR_NOT_UNIQUE -- One of names being cracked
(the Netbios domain name or the pre-existing account name
or the root DN) is not unique.
ERROR_FILE_EXISTS -- The OU passed does not match the cracked DN
of the pre-existing account.
ERROR_FILE_NOT_FOUND -- The specified OU does not exist or
Could not get/read the WellKnownObjects attribute or
Could not get the default computer container name from the
WellKnownObjects attribute.
ERROR_NOT_ENOUGH_MEMORY -- Could not allocated memory required.
One of the errors returned by DsCrackNames.
(see NetpCrackNamesStatus2Win32Error())
--*/
{
NET_API_STATUS NetStatus = NO_ERROR;
ULONG LdapStatus;
HANDLE hDs = NULL;
PWCHAR AccountUserName = NULL;
PWCHAR AccountDomainName = NULL;
LPWSTR NetbiosDomainNameWithBackslash = NULL;
PWCHAR ComputerContainerDn = NULL;
PWCHAR NameToCrack = NULL;
RPC_AUTH_IDENTITY_HANDLE AuthId = 0;
PDS_NAME_RESULTW CrackedName = NULL;
PWCHAR WellKnownObjectsAttr[2];
PWSTR *WellKnownObjectValues = NULL;
LDAPMessage *LdapMessage = NULL, *LdapEntry = NULL;
LPWSTR LocalComputerObjectDn = NULL;
ULONG Index;
//
// First check whether the account already exists for the computer
//
// If account is passed, prepare the corresponding credentials.
// Otherwise, use the default creds of the user running this routine.
//
if ( Account != NULL ) {
NetStatus = NetpSeparateUserAndDomain( Account, &AccountUserName, &AccountDomainName );
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpGetComputerObjectDn: Cannot NetpSeparateUserAndDomain 0x%lx\n", NetStatus ));
goto Cleanup;
}
NetStatus = DsMakePasswordCredentials( AccountUserName,
AccountDomainName,
Password,
&AuthId);
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpGetComputerObjectDn: Cannot DsMakePasswordCredentials 0x%lx\n", NetStatus ));
goto Cleanup;
}
}
//
// Bind to the DS on the DC.
//
NetStatus = DsBindWithCredW( DcInfo->DomainControllerName, NULL, AuthId, &hDs);
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpGetComputerObjectDn: Unable to bind to DS on '%ws': 0x%lx\n",
DcInfo->DomainControllerName, NetStatus ));
goto Cleanup ;
}
//
// Attempt to crack the account name into a DN.
//
// We need to have the Netbios domain name to
// form an NT4 style account name since DsCrackNames
// doesn't accept DNS domain names for cracking accounts.
// So, if we have a DNS domain name, we need to crack it
// into a Netbios domain name first.
//
if ( (DcInfo->Flags & DS_DNS_DOMAIN_FLAG) == 0 ) {
NetbiosDomainNameWithBackslash = LocalAlloc( 0, (wcslen(DcInfo->DomainName) + 2) * sizeof(WCHAR) );
if ( NetbiosDomainNameWithBackslash == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( NetbiosDomainNameWithBackslash, L"%ws\\", DcInfo->DomainName );
} else {
NameToCrack = LocalAlloc( 0, (wcslen(DcInfo->DomainName) + 1 + 1) * sizeof(WCHAR) );
if ( NameToCrack == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( NameToCrack, L"%ws/", DcInfo->DomainName );
//
// Be verbose
//
NetpLog(( "NetpGetComputerObjectDn: Cracking DNS domain name %ws into Netbios on %ws\n",
NameToCrack,
DcInfo->DomainControllerName ));
if ( CrackedName != NULL ) {
DsFreeNameResultW( CrackedName );
CrackedName = NULL;
}
//
// Crack the DNS domain name into a Netbios domain name
//
NetStatus = DsCrackNamesW( hDs,
0,
DS_CANONICAL_NAME,
DS_NT4_ACCOUNT_NAME,
1,
&NameToCrack,
&CrackedName );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpGetComputerObjectDn: CrackNames failed for %ws: 0x%lx\n",
NameToCrack,
NetStatus ));
goto Cleanup ;
}
//
// Check for consistency
//
if ( CrackedName->cItems != 1 ) {
NetStatus = ERROR_DS_NAME_ERROR_NOT_UNIQUE;
NetpLog(( "NetpGetComputerObjectDn: Cracked Name %ws is not unique: %lu\n",
NameToCrack,
CrackedName->cItems ));
goto Cleanup ;
}
if ( CrackedName->rItems[0].status != DS_NAME_NO_ERROR ) {
NetpLog(( "NetpGetComputerObjectDn: CrackNames failed for %ws: substatus 0x%lx\n",
NameToCrack,
CrackedName->rItems[0].status ));
NetStatus = NetpCrackNamesStatus2Win32Error( CrackedName->rItems[0].status );
goto Cleanup ;
}
//
// Be verbose
//
NetpLog(( "NetpGetComputerObjectDn: Crack results: \tname = %ws\n",
CrackedName->rItems[0].pName ));
//
// We've got the Netbios domain name
// (the cracked name already includes the trailing backslash)
//
NetbiosDomainNameWithBackslash = LocalAlloc( 0, (wcslen(CrackedName->rItems[0].pName) + 1) * sizeof(WCHAR) );
if ( NetbiosDomainNameWithBackslash == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
wcscpy( NetbiosDomainNameWithBackslash, CrackedName->rItems[0].pName );
}
//
// Form the NT4 account name given the Netbios domain name
//
if ( NameToCrack != NULL ) {
LocalFree( NameToCrack );
NameToCrack = NULL;
}
NameToCrack = LocalAlloc( 0,
(wcslen(NetbiosDomainNameWithBackslash) + wcslen(ComputerName) + 1 + 1) * sizeof(WCHAR) );
if ( NameToCrack == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( NameToCrack, L"%ws%ws$", NetbiosDomainNameWithBackslash, ComputerName );
//
// Crack the account name into a DN
//
if ( CrackedName != NULL ) {
DsFreeNameResultW( CrackedName );
CrackedName = NULL;
}
//
// Be verbose
//
NetpLog(( "NetpGetComputerObjectDn: Cracking account name %ws on %ws\n",
NameToCrack,
DcInfo->DomainControllerName ));
NetStatus = DsCrackNamesW( hDs,
0,
DS_NT4_ACCOUNT_NAME,
DS_FQDN_1779_NAME,
1,
&NameToCrack,
&CrackedName );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpGetComputerObjectDn: CrackNames failed for %ws: 0x%lx\n",
NameToCrack,
NetStatus ));
goto Cleanup ;
}
//
// Check for consistency
//
if ( CrackedName->cItems > 1 ) {
NetStatus = ERROR_DS_NAME_ERROR_NOT_UNIQUE;
NetpLog(( "NetpGetComputerObjectDn: Cracked Name %ws is not unique: %lu\n",
NameToCrack,
CrackedName->cItems ));
goto Cleanup ;
}
//
// If the account alredy exists, verify that the passed OU (if any)
// matches that of the account DN
//
if ( CrackedName->rItems[0].status == DS_NAME_NO_ERROR ) {
ULONG DnSize;
NetpLog(( "NetpGetComputerObjectDn: Crack results: \t(Account already exists) DN = %ws\n",
CrackedName->rItems[0].pName ));
DnSize = ( wcslen(CrackedName->rItems[0].pName) + 1 ) * sizeof(WCHAR);
//
// Allocate storage for the computer object DN
//
NetStatus = NetApiBufferAllocate( DnSize, &LocalComputerObjectDn );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
//
// If the OU is passed, verify that it matches the cracked name
//
if ( OU != NULL ) {
ULONG DnSizeFromOu;
DnSizeFromOu = ( wcslen(NETSETUPP_OBJ_PREFIX) +
wcslen(ComputerName) + 1 + wcslen(OU) + 1 ) * sizeof(WCHAR);
if ( DnSizeFromOu != DnSize ) {
NetpLog(( "NetpGetComputerObjectDn: Passed OU doesn't match in size cracked DN: %lu %lu\n",
DnSizeFromOu,
DnSize ));
NetStatus = ERROR_FILE_EXISTS;
goto Cleanup;
}
swprintf( LocalComputerObjectDn, L"%ws%ws,%ws", NETSETUPP_OBJ_PREFIX, ComputerName, OU );
if ( _wcsicmp(LocalComputerObjectDn, CrackedName->rItems[0].pName) != 0 ) {
NetpLog(( "NetpGetComputerObjectDn: Passed OU doesn't match cracked DN: %ws %ws\n",
LocalComputerObjectDn,
CrackedName->rItems[0].pName ));
NetStatus = ERROR_FILE_EXISTS;
goto Cleanup;
}
//
// Otherwise, just use the cracked name
//
} else {
wcscpy( LocalComputerObjectDn, CrackedName->rItems[0].pName );
}
//
// We've got the computer object DN from the existing account
//
NetStatus = NO_ERROR;
goto Cleanup;
}
//
// Be verbose
//
NetpLog(( "NetpGetComputerObjectDn: Crack results: \tAccount does not exist\n" ));
//
// At this point, we know that the account does not exist
// If OU is passed, simply verify it
//
if ( OU != NULL ) {
LdapStatus = ldap_compare_s( Ldap,
OU,
NETSETUPP_OBJECTCLASS,
NETSETUPP_ORGANIZATIONALUNIT );
if ( LdapStatus == LDAP_COMPARE_FALSE ) {
NetStatus = ERROR_FILE_NOT_FOUND;
NetpLog(( "NetpGetComputerObjectDn: Specified path '%ws' is not an OU\n", OU ));
goto Cleanup;
} else if ( LdapStatus != LDAP_COMPARE_TRUE ) {
NetStatus = LdapMapErrorToWin32( LdapStatus );
NetpLog(( "NetpGetComputerObjectDn: ldap_compare_s failed: 0x%lx 0x%lx\n",
LdapStatus, NetStatus ));
goto Cleanup;
}
//
// OU has been verified.
// Allocate the computer object DN.
//
NetStatus = NetApiBufferAllocate(
( wcslen(NETSETUPP_OBJ_PREFIX) +
wcslen(ComputerName) + 1 + wcslen(OU) + 1 ) * sizeof(WCHAR),
&LocalComputerObjectDn );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
//
// We've got the computer object DN from the OU passed
//
swprintf( LocalComputerObjectDn, L"%ws%ws,%ws", NETSETUPP_OBJ_PREFIX, ComputerName, OU );
NetpLog(( "NetpGetComputerObjectDn: Got DN %ws from the passed OU\n", LocalComputerObjectDn ));
NetStatus = NO_ERROR;
goto Cleanup;
}
//
// At this point, the account does not exist
// and no OU was specified. So get the default
// computer container DN.
//
if ( CrackedName != NULL ) {
DsFreeNameResultW( CrackedName );
CrackedName = NULL;
}
//
// Be verbose
//
NetpLog(( "NetpGetComputerObjectDn: Cracking Netbios domain name %ws into root DN on %ws\n",
NetbiosDomainNameWithBackslash,
DcInfo->DomainControllerName ));
NetStatus = DsCrackNamesW( hDs,
0,
DS_NT4_ACCOUNT_NAME,
DS_FQDN_1779_NAME,
1,
&NetbiosDomainNameWithBackslash,
&CrackedName );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpGetComputerObjectDn: CrackNames failed for %ws: 0x%lx\n",
NetbiosDomainNameWithBackslash,
NetStatus ));
goto Cleanup ;
}
//
// Check for consistency
//
if ( CrackedName->cItems != 1 ) {
NetStatus = ERROR_DS_NAME_ERROR_NOT_UNIQUE;
NetpLog(( "NetpGetComputerObjectDn: Cracked Name %ws is not unique: %lu\n",
NetbiosDomainNameWithBackslash,
CrackedName->cItems ));
goto Cleanup ;
}
if ( CrackedName->rItems[0].status != DS_NAME_NO_ERROR ) {
NetpLog(( "NetpGetComputerObjectDn: CrackNames failed for %ws: substatus 0x%lx\n",
NetbiosDomainNameWithBackslash,
CrackedName->rItems[0].status ));
NetStatus = NetpCrackNamesStatus2Win32Error( CrackedName->rItems[0].status );
goto Cleanup ;
}
//
// Be verbose
//
NetpLog(( "NetpGetComputerObjectDn: Crack results: \tname = %ws\n",
CrackedName->rItems[0].pName ));
//
// Now get the computer container DN given the root DN.
// The DN of the computer container is part of the wellKnownObjects
// attribute in the root of the domain. So, look it up.
//
WellKnownObjectsAttr[0] = L"wellKnownObjects";
WellKnownObjectsAttr[1] = NULL;
LdapStatus = ldap_search_s( Ldap,
CrackedName->rItems[0].pName, // Root DN
LDAP_SCOPE_BASE,
L"objectclass=*",
WellKnownObjectsAttr,
0,
&LdapMessage);
if ( LdapStatus != LDAP_SUCCESS ) {
NetStatus = LdapMapErrorToWin32( LdapStatus );
NetpLog(( "NetpGetComputerObjectDn: ldap_search_s failed 0x%lx 0x%lx\n",
LdapStatus,
NetStatus ));
goto Cleanup;
}
if ( ldap_count_entries(Ldap, LdapMessage) == 0 ) {
NetStatus = ERROR_FILE_NOT_FOUND;
NetpLog(( "NetpGetComputerObjectDn: ldap_search_s returned no entries\n" ));
goto Cleanup;
}
LdapEntry = ldap_first_entry( Ldap, LdapMessage );
if ( LdapEntry == NULL ) {
NetStatus = ERROR_FILE_NOT_FOUND;
NetpLog(( "NetpGetComputerObjectDn: ldap_first_entry returned NULL\n" ));
goto Cleanup;
}
WellKnownObjectValues = ldap_get_valuesW( Ldap,
LdapEntry,
L"wellKnownObjects" );
if ( WellKnownObjectValues == NULL ) {
NetStatus = ERROR_FILE_NOT_FOUND;
NetpLog(( "NetpGetComputerObjectDn: ldap_get_valuesW returned NULL\n" ));
goto Cleanup;
}
//
// Lookup the default computer container
//
for ( Index = 0; WellKnownObjectValues[Index] != NULL; Index++ ) {
//
// The structure of this particular field is:
// L"B:32:GUID:DN" where GUID is AA312825768811D1ADED00C04FD8D5CD
//
if ( _wcsnicmp( WellKnownObjectValues[Index],
NETSETUPP_COMPUTER_CONTAINER_GUID_IN_B32_FORM,
wcslen(NETSETUPP_COMPUTER_CONTAINER_GUID_IN_B32_FORM) ) == 0 ) {
ComputerContainerDn = WellKnownObjectValues[Index] +
wcslen(NETSETUPP_COMPUTER_CONTAINER_GUID_IN_B32_FORM);
break;
}
}
//
// If we couldn't get the computer container DN, error out
//
if ( ComputerContainerDn == NULL || *ComputerContainerDn == L'\0' ) {
NetpLog(( "NetpGetComputerObjectDn: Couldn't get computer container DN\n" ));
NetStatus = ERROR_FILE_NOT_FOUND;
goto Cleanup;
}
//
// Allocate the computer object DN
//
NetStatus = NetApiBufferAllocate(
( wcslen(NETSETUPP_OBJ_PREFIX) +
wcslen(ComputerName) + 1 + wcslen(ComputerContainerDn) + 1 ) * sizeof(WCHAR),
&LocalComputerObjectDn );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
//
// We've got the computer object DN from the default computer container
//
swprintf( LocalComputerObjectDn, L"%ws%ws,%ws", NETSETUPP_OBJ_PREFIX, ComputerName, ComputerContainerDn );
NetpLog(( "NetpGetComputerObjectDn: Got DN %ws from the default computer container\n", LocalComputerObjectDn ));
NetStatus = NO_ERROR;
//
// Free locally used resources
//
Cleanup:
if ( hDs ) {
DsUnBind( &hDs );
}
if ( CrackedName ) {
DsFreeNameResultW( CrackedName );
}
if ( AuthId ) {
DsFreePasswordCredentials( AuthId );
}
if ( WellKnownObjectValues != NULL ) {
ldap_value_free( WellKnownObjectValues );
}
if ( NameToCrack != NULL ) {
LocalFree( NameToCrack );
}
if ( AccountUserName != NULL ) {
NetApiBufferFree( AccountUserName );
}
if ( AccountDomainName != NULL ) {
NetApiBufferFree( AccountDomainName );
}
if ( NetbiosDomainNameWithBackslash != NULL ) {
LocalFree( NetbiosDomainNameWithBackslash );
}
if ( LdapMessage != NULL ) {
ldap_msgfree( LdapMessage );
}
if ( NetStatus == NO_ERROR ) {
*ComputerObjectDn = LocalComputerObjectDn;
} else if ( LocalComputerObjectDn != NULL ) {
NetApiBufferFree( LocalComputerObjectDn );
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpModifyComputerObjectInDs(
IN LPWSTR DC,
IN PLDAP Ldap,
IN LPWSTR ComputerName,
IN LPWSTR ComputerObjectDn,
IN ULONG NumberOfAttributes,
IN OUT PNETSETUPP_MACH_ACC_ATTRIBUTE Attrib
)
/*++
Routine Description:
Create a computer account in the specified OU.
Arguments:
DC -- Domain controller on which to create the object
Ldap -- Ldap binding to the DC
ComputerName -- Name of the computer being joined
ComputerObjectDn -- DN of computer object being modified
NumberOfAttributes -- Number of attributes passed
Attrib -- List of attribute structures. The list may
be modified on return so that only those entries
that were not already set in the DS will be preserved.
NOTE: If the machine password (unicodePwd) is passed as one of the attributes,
it must be the last entry in the attribute list because this order is assumed
by the fail-over code below.
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
ULONG LdapStatus;
PWSTR *AttribTypesList = NULL;
LDAPMod *ModList = NULL;
PLDAPMod *Mods = NULL;
LDAPMessage *Message = NULL, *Entry;
ULONG Index;
ULONG ModIndex = 0;
BOOL NewAccount = FALSE;
BOOL AccountBeingEnabled = FALSE;
PWSTR SamAccountName = NULL;
USER_INFO_1 *CurrentUI1 = NULL;
//
// Allocate storage for the attribute list and the modifications block
//
NetStatus = NetApiBufferAllocate( (NumberOfAttributes+1)*sizeof(PWSTR),
(PVOID *) &AttribTypesList );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
NetStatus = NetApiBufferAllocate( NumberOfAttributes * sizeof(LDAPMod),
(PVOID *) &ModList );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
NetStatus = NetApiBufferAllocate( (NumberOfAttributes+1)*sizeof(PLDAPMod),
(PVOID *) &Mods );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
//
// Build modification list given the list of attributes
//
NetpLog(( "NetpModifyComputerObjectInDs: Initial attribute values:\n" ));
for ( Index = 0; Index < NumberOfAttributes; Index++ ) {
ModList[Index].mod_op = LDAP_MOD_ADD; // Set to add. We may adjust this below.
ModList[Index].mod_type = Attrib[Index].AttribType;
ModList[Index].mod_values = Attrib[Index].AttribValues;
//
// See whether we enable the account
//
if ( _wcsicmp(ModList[Index].mod_type, NETSETUPP_USERACCOUNTCONTROL) == 0 &&
_wcsicmp(*(ModList[Index].mod_values), NETSETUPP_ACCNT_TYPE_ENABLED) == 0 ) {
AccountBeingEnabled = TRUE;
}
//
// Be verbose - output all values of each attribute
//
NetpLog(( "\t\t%ws =", Attrib[Index].AttribType ));
//
// Don't leak sensitive info!
//
if ( _wcsicmp( Attrib[Index].AttribType, NETSETUPP_UNICODEPWD ) == 0 ) {
NetpLog(( " <SomePassword>" ));
} else {
PWSTR *CurrentValues;
for ( CurrentValues = Attrib[Index].AttribValues; *CurrentValues != NULL; CurrentValues++ ) {
NetpLog(( " %ws", *CurrentValues ));
}
}
NetpLog(( "\n" ));
}
//
// Now check which attribute values are already set in the DS
//
for ( Index = 0; Index < NumberOfAttributes; Index++ ) {
AttribTypesList[Index] = Attrib[Index].AttribType;
}
AttribTypesList[Index] = NULL; // Terminate the list
LdapStatus = ldap_search_s( Ldap,
ComputerObjectDn,
LDAP_SCOPE_BASE,
NULL,
AttribTypesList,
0,
&Message );
//
// If the computer object does not exist,
// we need to add all attributes
//
if ( LdapStatus == LDAP_NO_SUCH_OBJECT ) {
NetpLog(( "NetpModifyComputerObjectInDs: Computer Object does not exist in OU\n" ));
NewAccount = TRUE;
for ( ModIndex = 0; ModIndex < NumberOfAttributes; ModIndex++ ) {
Mods[ModIndex] = &ModList[ModIndex];
}
//
// Terminate the modification list
//
Mods[ModIndex] = NULL;
//
// Otherwise see which attribute values need modification
//
} else if ( LdapStatus == LDAP_SUCCESS ) {
NetpLog(( "NetpModifyComputerObjectInDs: Computer Object already exists in OU:\n" ));
//
// Get the first entry (there should be only one)
//
Entry = ldap_first_entry( Ldap, Message );
//
// Loop through the attributes and weed out those values
// which are already set.
//
for ( Index = 0; Index < NumberOfAttributes; Index++ ) {
PWSTR *AttribValueRet = NULL;
//
// Be verbose - output the values returned for each type
//
NetpLog(( "\t\t%ws =", Attrib[Index].AttribType ));
AttribValueRet = ldap_get_values( Ldap, Entry, Attrib[Index].AttribType );
if ( AttribValueRet != NULL ) {
//
// Don't leak sensitive info!
//
if ( _wcsicmp( Attrib[Index].AttribType, NETSETUPP_UNICODEPWD ) == 0 ) {
NetpLog(( " <SomePassword>" ));
} else {
PWSTR *CurrentValueRet;
for ( CurrentValueRet = AttribValueRet; *CurrentValueRet != NULL; CurrentValueRet++ ) {
NetpLog(( " %ws", *CurrentValueRet ));
}
}
//
// Remove those values from the modification which are alredy set
//
NetpRemoveDuplicateStrings( AttribValueRet, Attrib[Index].AttribValues );
ldap_value_free( AttribValueRet );
//
// If this is a single valued attribute, we need to
// replace (not add) its value since it already exists in the DS
//
if ( (Attrib[Index].AttribFlags & NETSETUPP_MULTIVAL_ATTRIB) == 0 ) {
ModList[Index].mod_op = LDAP_MOD_REPLACE;
}
}
NetpLog(( "\n" ));
//
// If there are any attribute values which are
// not already set, add them to the modification.
//
if ( *(Attrib[Index].AttribValues) != NULL ) {
Mods[ModIndex] = &ModList[Index];
ModIndex ++;
}
}
//
// Terminate the modification list
//
Mods[ModIndex] = NULL;
//
// Otherwise, error out
//
} else {
NetStatus = LdapMapErrorToWin32( LdapStatus );
NetpLog(( "NetpModifyComputerObjectInDs: ldap_search_s failed: 0x%lx 0x%lx\n",
LdapStatus, NetStatus ));
goto Cleanup;
}
//
// Do the modifications if there are any
//
if ( ModIndex == 0 ) {
NetpLog(( "NetpModifyComputerObjectInDs: There are _NO_ modifications to do\n" ));
NetStatus = NERR_Success;
} else {
//
// Be verbose - output the attribute values to be set
//
NetpLog(( "NetpModifyComputerObjectInDs: Attribute values to set:\n" ));
for ( Index = 0; Mods[Index] != NULL; Index++ ) {
NetpLog(( "\t\t%ws =", (*(Mods[Index])).mod_type ));
//
// Don't leak sensitive info!
//
if ( _wcsicmp( (*(Mods[Index])).mod_type, NETSETUPP_UNICODEPWD ) == 0 ) {
NetpLog(( " <SomePassword>" ));
} else {
ULONG ValIndex;
for ( ValIndex = 0; ((*(Mods[Index])).mod_values)[ValIndex] != NULL; ValIndex++ ) {
NetpLog(( " %ws", ((*(Mods[Index])).mod_values)[ValIndex] ));
}
}
NetpLog(( "\n" ));
}
//
// Now, add the missing attributes
//
if ( NewAccount ) {
LdapStatus = ldap_add_s( Ldap, ComputerObjectDn, Mods );
} else {
LdapStatus = ldap_modify_s( Ldap, ComputerObjectDn, Mods );
}
if ( LdapStatus != LDAP_SUCCESS ) {
//
// Return the error code the user understands
//
if ( LdapStatus == LDAP_ALREADY_EXISTS ) {
NetStatus = NERR_UserExists;
} else {
NetStatus = LdapMapErrorToWin32( LdapStatus );
}
NetpLog(( "NetpModifyComputerObjectInDs: %s failed: 0x%lx 0x%lx\n",
NewAccount ?
"ldap_add_s" : "ldap_modify_s",
LdapStatus, NetStatus ));
goto Cleanup;
}
}
//
// If we enable the account, toggle the account type property.
// See comment in NetpSetMachineAccountPasswordAndTypeEx() for
// an explanation of why this is needed. (Search for USN).
//
if ( AccountBeingEnabled ) {
Mods[0] = NULL;
for ( Index = 0; Index < NumberOfAttributes; Index++ ) {
if ( _wcsicmp( ModList[Index].mod_type, NETSETUPP_USERACCOUNTCONTROL ) == 0 ) {
Mods[0] = &ModList[Index];
//
// If this is a single valued attribute (it is, but...), we need to
// replace (not add) its value since it already exists in the DS
//
if ( (Attrib[Index].AttribFlags & NETSETUPP_MULTIVAL_ATTRIB) == 0 ) {
ModList[Index].mod_op = LDAP_MOD_REPLACE;
}
break;
}
}
Mods[1] = NULL;
if ( Mods[0] != NULL ) {
//
// Disable the account
//
*(Mods[0]->mod_values) = NETSETUPP_ACCNT_TYPE_DISABLED;
LdapStatus = ldap_modify_s( Ldap, ComputerObjectDn, Mods );
if ( LdapStatus != LDAP_SUCCESS ) {
NetStatus = LdapMapErrorToWin32( LdapStatus );
NetpLog(( "NetpModifyComputerObjectInDs: set UserAccountControl (1) on '%ws' failed: 0x%lx 0x%lx\n",
ComputerObjectDn, LdapStatus, NetStatus ));
goto Cleanup;
}
//
// Re-enable the account
//
*(Mods[0]->mod_values) = NETSETUPP_ACCNT_TYPE_ENABLED;
LdapStatus = ldap_modify_s( Ldap, ComputerObjectDn, Mods );
if ( LdapStatus != LDAP_SUCCESS ) {
NetStatus = LdapMapErrorToWin32( LdapStatus );
NetpLog(( "NetpModifyComputerObjectInDs: set UserAccountControl (2) on '%ws' failed: 0x%lx 0x%lx\n",
ComputerObjectDn, LdapStatus, NetStatus ));
goto Cleanup;
}
NetpLog(( "NetpModifyComputerObjectInDs: Toggled UserAccountControl successfully\n" ));
}
}
//
// If we've made up to this point, it's success!
//
NetStatus = NERR_Success;
//
// REVIEW: On error, consider using ldap_get_option to retrieve
// a human readable string describing the error that happend.
// Use LDAP_OPT_SERVER_ERROR as the option value. Return the
// string to the caller who may want to expose it to the user.
//
Cleanup:
if ( AttribTypesList != NULL ) {
NetApiBufferFree( AttribTypesList );
}
if ( ModList != NULL ) {
NetApiBufferFree( ModList );
}
if ( Mods != NULL ) {
NetApiBufferFree( Mods );
}
if ( Message != NULL ) {
ldap_msgfree( Message );
}
if ( SamAccountName != NULL ) {
NetApiBufferFree( SamAccountName );
}
if ( CurrentUI1 != NULL ) {
NetApiBufferFree( CurrentUI1 );
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpCreateComputerObjectInDs(
IN PDOMAIN_CONTROLLER_INFO DcInfo,
IN LPWSTR Account OPTIONAL,
IN LPWSTR Password OPTIONAL,
IN LPWSTR ComputerName,
IN LPWSTR MachinePassword OPTIONAL,
IN LPWSTR DnsHostName OPTIONAL,
IN LPWSTR OU OPTIONAL
)
/*++
Routine Description:
Create a computer account in the specified OU.
Arguments:
DcInfo -- Domain controller on which to create the object
Account -- Account to use for the LDAP and DS binds.
If NULL, the default creds of the current user
context are used.
Password -- Password to used for the binds. Ignored if
Account is NULL.
ComputerName -- (Netbios) Name of the computer being joined
MachinePassword -- Password to set on the machine object
DnsHostName -- DNS host name of the computer being joined
OU -- OU under which to create the object.
The name must be a fully qualified name
e.g.: "ou=test,dc=ntdev,dc=microsoft,dc=com"
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus;
PLDAP Ldap = NULL;
PWSTR ComputerObjectDn = NULL;
PWSTR SamAccountName = NULL;
PWSTR DnsSpn = NULL;
PWSTR NetbiosSpn = NULL;
ULONG AttribCount;
PWSTR ClassValues[ 2 ];
PWSTR AccntNameValues[ 2 ];
PWSTR DnsHostNameValues[ 2 ];
PWSTR SpnValues[ 3 ];
PWSTR PasswordValues[ 2 ];
PWSTR AccntTypeValues[ 2 ];
NETSETUPP_MACH_ACC_ATTRIBUTE Attributes[NETSETUPP_COMP_OBJ_ATTR_COUNT];
USER_INFO_1003 UserInfo1003 = {NULL};
//
// Validate parameters
//
if ( DcInfo == NULL ) {
NetpLog(( "NetpCreateComputerObjectInDs: No DcInfo passed\n" ));
NetStatus = ERROR_INVALID_PARAMETER;
goto Cleanup;
}
if ( ComputerName == NULL ) {
NetpLog(( "NetpCreateComputerObjectInDs: No ComputerName passed\n" ));
NetStatus = ERROR_INVALID_PARAMETER;
goto Cleanup;
}
//
// Verify that the DC runs DS
//
if ( (DcInfo->Flags & DS_DS_FLAG) == 0 ||
(DcInfo->Flags & DS_WRITABLE_FLAG) == 0 ) {
NetpLog(( "NetpCreateComputerObjectInDs: DC passed '%ws' doesn't have writable DS 0x%lx\n",
DcInfo->DomainControllerName,
DcInfo->Flags ));
NetStatus = ERROR_NOT_SUPPORTED;
goto Cleanup;
}
//
// First, try to bind to the server
//
NetStatus = NetpLdapBind( DcInfo->DomainControllerName, Account, Password, &Ldap );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpCreateComputerObjectInDs: NetpLdapBind failed: 0x%lx\n", NetStatus ));
goto Cleanup;
}
//
// Next get the computer object DN
//
NetStatus = NetpGetComputerObjectDn( DcInfo,
Account,
Password,
Ldap,
ComputerName,
OU,
&ComputerObjectDn );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpCreateComputerObjectInDs: NetpGetComputerObjectDn failed: 0x%lx\n", NetStatus ));
//
// Return meaningful error
//
if ( NetStatus == ERROR_FILE_EXISTS ) {
NetStatus = NERR_UserExists;
}
goto Cleanup;
}
//
// Get SAM account name
//
NetStatus = NetpGetMachineAccountName( ComputerName, &SamAccountName );
if ( NetStatus != NO_ERROR ) {
goto Cleanup;
}
//
// Build SPN values
//
if ( DnsHostName != NULL ) {
DnsSpn = LocalAlloc( 0, (wcslen(NETSETUPP_HOST_SPN_PREFIX) + wcslen(DnsHostName) + 1) * sizeof(WCHAR) );
if ( DnsSpn == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( DnsSpn, L"%ws%ws", NETSETUPP_HOST_SPN_PREFIX, DnsHostName );
NetbiosSpn = LocalAlloc( 0, (wcslen(NETSETUPP_HOST_SPN_PREFIX) + wcslen(ComputerName) + 1) * sizeof(WCHAR) );
if ( Netbios == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( NetbiosSpn, L"%ws%ws", NETSETUPP_HOST_SPN_PREFIX, ComputerName );
}
//
// Prepare the list of attributes that need to be set in the DS
//
// Always keep unicodePwd as the last entry because this order is
// assumed by the API called below.
//
AttribCount = 0;
Attributes[AttribCount].AttribType = NETSETUPP_OBJECTCLASS; //
Attributes[AttribCount].AttribFlags = NETSETUPP_MULTIVAL_ATTRIB; //
Attributes[AttribCount].AttribValues = ClassValues; // ObjectClass
ClassValues[ 0 ] = NETSETUPP_COMPUTER_OBJECT; //
ClassValues[ 1 ] = NULL; //
AttribCount ++;
Attributes[AttribCount].AttribType = NETSETUPP_SAMACCOUNTNAME; //
Attributes[AttribCount].AttribFlags = 0; //
Attributes[AttribCount].AttribValues = AccntNameValues; // SamAccountName
AccntNameValues[ 0 ] = SamAccountName; //
AccntNameValues[ 1 ] = NULL; //
AttribCount ++;
Attributes[AttribCount].AttribType = NETSETUPP_USERACCOUNTCONTROL; //
Attributes[AttribCount].AttribFlags = 0; //
Attributes[AttribCount].AttribValues = AccntTypeValues; // userAccountControl
AccntTypeValues[ 0 ] = NETSETUPP_ACCNT_TYPE_ENABLED; //
AccntTypeValues[ 1 ] = NULL; //
AttribCount ++;
if ( DnsHostName != NULL ) {
Attributes[AttribCount].AttribType = NETSETUPP_DNSHOSTNAME; //
Attributes[AttribCount].AttribFlags = 0; //
Attributes[AttribCount].AttribValues = DnsHostNameValues; // DnsHostName
DnsHostNameValues[ 0 ] = DnsHostName; //
DnsHostNameValues[ 1 ] = NULL; //
AttribCount ++;
Attributes[AttribCount].AttribType = NETSETUPP_SERVICEPRINCIPALNAME; //
Attributes[AttribCount].AttribFlags = NETSETUPP_MULTIVAL_ATTRIB; //
Attributes[AttribCount].AttribValues = SpnValues; // ServicePrincipalName
SpnValues[ 0 ] = DnsSpn; //
SpnValues[ 1 ] = NetbiosSpn; //
SpnValues[ 2 ] = NULL; //
AttribCount ++;
}
//
// The following attribute is the machine password. We avoid
// updating it through ldap because it is hard to ensure that
// the ldap session uses the 128-bit encryption required by
// SAM on the DC for password updates.
//
// To enforce the encryption, we would need to set an option
// LDAP_OPT_ENCRYPT via a ldap_set_option call following ldap_open
// before calling ldap_bind_s. However, there is no guarantee that
// the established connection will use 128 bit encryption; it may
// use 56 bit encryption if either side does not support strong
// encryption. We could, in principle, find out the resulting encryption
// strength using some QueryContextAttribute call, but it's just too much
// trouble. So, we will just create the account without the password and
// we will then update the password using good old Net/SAM API.
//
#if 0
Attributes[AttribCount].AttribType = NETSETUPP_UNICODEPWD; //
Attributes[AttribCount].AttribFlags = 0; //
Attributes[AttribCount].AttribValues = PasswordValues; // unicodePwd
PasswordValues[ 0 ] = MachinePassword; //
PasswordValues[ 1 ] = NULL; //
AttribCount ++;
#endif
//
// Modify the computer object given the list of attributes
//
NetStatus = NetpModifyComputerObjectInDs( DcInfo->DomainControllerName,
Ldap,
ComputerName,
ComputerObjectDn,
AttribCount,
Attributes );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpCreateComputerObjectInDs: NetpModifyComputerObjectInDs failed: 0x%lx\n", NetStatus ));
goto Cleanup;
}
//
// Now set the password using good old Net/SAM API.
//
UserInfo1003.usri1003_password = MachinePassword;
NetStatus = NetUserSetInfo( DcInfo->DomainControllerName,
SamAccountName,
1003,
(PBYTE) &UserInfo1003,
NULL );
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpCreateComputerObjectInDs: NetUserSetInfo (level 1003) failed on '%ws' for '%ws': 0x%lx."
" Deleting the account.\n",
DcInfo->DomainControllerName,
SamAccountName,
NetStatus ));
}
//
// Delete the account if we couldn't set the password.
// Ignore the failure if we cannot delete the account for some reason.
//
if ( NetStatus != NO_ERROR ) {
ULONG LdapStatus;
LdapStatus = ldap_delete_s( Ldap, ComputerObjectDn );
if ( LdapStatus != LDAP_SUCCESS ) {
NetpLog(( "NetpCreateComputerObjectInDs: Failed to delete '%ws': 0x%lx 0x%lx\n",
ComputerObjectDn, LdapStatus, LdapMapErrorToWin32( LdapStatus ) ));
}
}
//
// Tell Netlogon that it should avoid setting
// DnsHostName and SPN until the reboot
//
if ( NetStatus == NO_ERROR && DnsHostName != NULL ) {
NetpAvoidNetlogonSpnSet( TRUE );
}
Cleanup:
if ( Ldap != NULL ) {
NetpLdapUnbind( Ldap );
}
if ( ComputerObjectDn != NULL ) {
NetApiBufferFree( ComputerObjectDn );
}
if ( SamAccountName != NULL ) {
NetApiBufferFree( SamAccountName );
}
if ( DnsSpn != NULL ) {
LocalFree( DnsSpn );
}
if ( NetbiosSpn != NULL ) {
LocalFree( NetbiosSpn );
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpSetDnsHostNameAndSpn(
IN PDOMAIN_CONTROLLER_INFO DcInfo,
IN LPWSTR Account,
IN LPWSTR Password,
IN LPWSTR ComputerName,
IN LPWSTR DnsHostName
)
/*++
Routine Description:
Set DnsHostName and HOST SPN (ServicePrincipalName) attributes on the
computer object in the DS.
Arguments:
DcInfo -- Domain controller on which to create the object
Account -- Account to use for the LDAP bind
Password -- Password to used for the bind
ComputerName -- Name of the computer being joined
DnsHostName -- DNS host name of the machine
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus;
HANDLE hToken = NULL;
PLDAP Ldap = NULL;
PWSTR ComputerObjectDn = NULL;
PWSTR DnsSpn = NULL;
PWSTR NetbiosSpn = NULL;
PWSTR DnsHostNameValues[ 2 ];
PWSTR SpnValues[ 3 ] = {NULL};
NETSETUPP_MACH_ACC_ATTRIBUTE Attributes[ 2 ];
//
// REVIEW: Kerberos has a bug such that if this server is joined remotely
// and the impersonated client connected to this server using NTLM (as is
// the case if this server is not a member of a domain before the join),
// explicit credentials supplied to ldap_bind or DsBindWithCredW will not
// work (AcquireCredentialsHandle call will fail). To get around this, we
// temporarily un-impersonates, bind to the DC, and then impersonate again
// at the end of this routine.
//
if ( OpenThreadToken( GetCurrentThread(),
TOKEN_IMPERSONATE,
TRUE,
&hToken ) ) {
if ( RevertToSelf() == 0 ) {
NetpLog(( "NetpSetDnsHostNameAndSpn: RevertToSelf failed: 0x%lx\n",
GetLastError() ));
}
} else {
NetpLog(( "NetpSetDnsHostNameAndSpn: OpenThreadToken failed: 0x%lx\n",
GetLastError() ));
}
//
// Bind to the DC
//
NetStatus = NetpLdapBind( DcInfo->DomainControllerName, Account, Password, &Ldap );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpSetDnsHostNameAndSpn: NetpLdapBind failed: 0x%lx\n", NetStatus ));
goto Cleanup;
}
//
// Next get the computer object DN
//
NetStatus = NetpGetComputerObjectDn( DcInfo,
Account,
Password,
Ldap,
ComputerName,
NULL, // Default computer container
&ComputerObjectDn );
if ( NetStatus != NO_ERROR ) {
NetpLog(( "NetpSetDnsHostNameAndSpn: NetpGetComputerObjectDn failed: 0x%lx\n", NetStatus ));
//
// Return meaningful error
//
if ( NetStatus == ERROR_FILE_EXISTS ) {
NetStatus = NERR_UserExists;
}
goto Cleanup;
}
//
// Build DnsHostName values
//
DnsHostNameValues[ 0 ] = DnsHostName;
DnsHostNameValues[ 1 ] = NULL;
//
// Build SPN values
//
DnsSpn = LocalAlloc( 0,
(wcslen(NETSETUPP_HOST_SPN_PREFIX) + wcslen(DnsHostName) + 1) * sizeof(WCHAR) );
if ( DnsSpn == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( DnsSpn, L"%ws%ws", NETSETUPP_HOST_SPN_PREFIX, DnsHostName );
NetbiosSpn = LocalAlloc( 0,
(wcslen(NETSETUPP_HOST_SPN_PREFIX) + wcslen(ComputerName) + 1) * sizeof(WCHAR) );
if ( NetbiosSpn == NULL ) {
NetStatus = ERROR_NOT_ENOUGH_MEMORY;
goto Cleanup;
}
swprintf( NetbiosSpn, L"%ws%ws", NETSETUPP_HOST_SPN_PREFIX, ComputerName );
SpnValues[0] = DnsSpn;
SpnValues[1] = NetbiosSpn;
SpnValues[2] = NULL;
//
// Prepare the list of attributes that need to be set in the DS
//
Attributes[0].AttribType = NETSETUPP_DNSHOSTNAME; //
Attributes[0].AttribFlags = 0; // DnsHostName
Attributes[0].AttribValues = DnsHostNameValues; //
Attributes[1].AttribType = NETSETUPP_SERVICEPRINCIPALNAME; //
Attributes[1].AttribFlags = NETSETUPP_MULTIVAL_ATTRIB; // ServicePrincipalName
Attributes[1].AttribValues = SpnValues; //
//
// Modify the computer object given the list of attributes
//
NetStatus = NetpModifyComputerObjectInDs( DcInfo->DomainControllerName,
Ldap,
ComputerName,
ComputerObjectDn,
2,
Attributes );
//
// Tell Netlogon that it should avoid setting
// DnsHostName and SPN until the reboot
//
if ( NetStatus == NO_ERROR ) {
NetpAvoidNetlogonSpnSet( TRUE );
}
Cleanup:
if ( Ldap != NULL ) {
NetpLdapUnbind( Ldap );
}
//
// REVIEW: Revert the impersonation
//
if ( hToken != NULL ) {
if ( SetThreadToken( NULL, hToken ) == 0 ) {
NetpLog(( "NetpSetDnsHostNameAndSpn: SetThreadToken failed: 0x%lx\n",
GetLastError() ));
}
CloseHandle( hToken );
}
//
// Free locally allocated memory
//
if ( ComputerObjectDn != NULL ) {
NetApiBufferFree( ComputerObjectDn );
}
if ( DnsSpn != NULL ) {
LocalFree( DnsSpn );
}
if ( NetbiosSpn != NULL ) {
LocalFree( NetbiosSpn );
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpDeleteComputerObjectInOU(
IN LPWSTR DC,
IN LPWSTR OU,
IN LPWSTR ComputerName,
IN LPWSTR Account,
IN LPWSTR Password
)
/*++
Routine Description:
This routine will actually create a computer account in the specified OU.
Arguments:
DC -- Domain controller on which to create the object
OU -- OU under which to create the object
ComputerName -- Name of the computer being joined
Account -- Account to use for the LDAP bind
Password -- Password to used for the bind
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR ObjectName = NULL, SamAccountName = NULL;
PLDAP Ldap = NULL;
ULONG Len;
Len = wcslen( ComputerName );
NetStatus = NetApiBufferAllocate( sizeof( NETSETUPP_OBJ_PREFIX ) + ( wcslen( OU ) + Len + 1 ) * sizeof( WCHAR ),
( PVOID * ) &ObjectName );
if ( NetStatus == NERR_Success ) {
swprintf( ObjectName, L"%ws%ws,%ws", NETSETUPP_OBJ_PREFIX, ComputerName, OU );
NetStatus = NetApiBufferAllocate( ( Len + 2 ) * sizeof( WCHAR ),
( PVOID * )&SamAccountName );
if ( NetStatus == NERR_Success ) {
swprintf( SamAccountName, L"%ws$", ComputerName );
}
}
if ( NetStatus == NERR_Success ) {
//
// Try and bind to the server
//
NetStatus = NetpLdapBind( DC,
Account,
Password,
&Ldap );
if ( NetStatus == NERR_Success ) {
//
// Now, do the delete..
//
NetStatus = LdapMapErrorToWin32( ldap_delete_s( Ldap, ObjectName ) );
NetpLdapUnbind( Ldap );
}
}
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpCreateComputerObjectInOU failed with %lu\n",
NetStatus ));
}
NetApiBufferFree( ObjectName );
NetApiBufferFree( SamAccountName );
if ( NetStatus != NERR_Success ) {
NetpLog(( "NetpDeleteComputerObjectInOU failed with %lu\n",
NetStatus ));
}
return( NetStatus );
}
#if defined(REMOTE_BOOT)
NET_API_STATUS
NetpGetRemoteBootMachinePassword(
OUT LPWSTR Password
)
/*++
Routine Description:
Determine if this is a remote boot client, and if so return
the machine account password.
This information is obtained via an IOCTL to the redirector.
Arguments:
Password - returns the password. Should be at least PWLEN WCHARs long.
Return Value:
NERR_Success if the password is found.
An error if this is not a remote boot machine.
--*/
{
NET_API_STATUS NetStatus;
NTSTATUS Status;
UNICODE_STRING DeviceName;
IO_STATUS_BLOCK IoStatusBlock;
OBJECT_ATTRIBUTES ObjectAttributes;
HANDLE RedirHandle = NULL;
UCHAR PacketBuffer[sizeof(ULONG)+64];
PLMMR_RB_CHECK_FOR_NEW_PASSWORD RequestPacket = (PLMMR_RB_CHECK_FOR_NEW_PASSWORD)PacketBuffer;
//
// Open the redirector device.
//
RtlInitUnicodeString(&DeviceName, DD_NFS_DEVICE_NAME_U);
InitializeObjectAttributes(
&ObjectAttributes,
&DeviceName,
OBJ_CASE_INSENSITIVE,
NULL,
NULL
);
Status = NtOpenFile(
&RedirHandle,
SYNCHRONIZE,
&ObjectAttributes,
&IoStatusBlock,
0,
0
);
if (NT_SUCCESS(Status)) {
Status = IoStatusBlock.Status;
}
if (!NT_SUCCESS(Status)) {
NetpLog(( "Could not open redirector device %lx\n",
Status ));
NetStatus = NetpNtStatusToApiStatus( Status );
goto Cleanup;
}
//
// Send the request to the redir.
//
Status = NtFsControlFile(
RedirHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
FSCTL_LMMR_RB_CHECK_FOR_NEW_PASSWORD,
NULL, // no input buffer
0,
PacketBuffer,
sizeof(PacketBuffer));
if (NT_SUCCESS(Status)) {
Status = IoStatusBlock.Status;
}
//
// We expect this to work on a disked machine, since we need the password
// to join.
//
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "Could not open FSCTL_LMMR_RB_CHECK_FOR_NEW_PASSWORD %lx\n",
Status ));
NetStatus = NetpNtStatusToApiStatus( Status );
goto Cleanup;
}
//
// Copy the result back to the caller's buffer.
// Guard against buffer overrun.
//
if ( RequestPacket->Length > PWLEN*sizeof(WCHAR) ) {
NetStatus = ERROR_INVALID_PARAMETER;
NetpLog(( "NetpGetRemoteBootMachinePassword: Password too long %d\n",
RequestPacket->Length ));
} else {
RtlCopyMemory(Password, RequestPacket->Data, RequestPacket->Length);
Password[RequestPacket->Length / 2] = L'\0';
NetStatus = NO_ERROR;
}
Cleanup:
if ( RedirHandle != NULL ) {
NtClose( RedirHandle );
}
return NetStatus;
}
#endif // REMOTE_BOOT
NET_API_STATUS
NET_API_FUNCTION
NetpSetMachineAccountPasswordAndType(
IN LPWSTR lpDcName,
IN PSID DomainSid,
IN LPWSTR lpAccountName,
IN LPWSTR lpPassword
)
{
return( NetpSetMachineAccountPasswordAndTypeEx(
lpDcName,
DomainSid,
lpAccountName,
lpPassword,
0,
TRUE
) );
}
NET_API_STATUS
NET_API_FUNCTION
NetpSetMachineAccountPasswordAndTypeEx(
IN LPWSTR lpDcName,
IN PSID DomainSid,
IN LPWSTR lpAccountName,
IN OUT OPTIONAL LPWSTR lpPassword,
IN OPTIONAL UCHAR AccountState,
IN BOOL fIsNt4Dc
)
/*++
Routine Description:
Due to a few strange reasons, we cannot use the supported, documented Net apis for
managing the machine account, so we have to use the undocumented Sam apis. This routine
will set the password and account type on an account that alread exists.
Arguments:
lpDcName - Name of the DC on which the account lives
DomainSid - Sid of the domain on which the account lives
lpAccountName - Name of the account
lpPassword - Password to be set on the account.
This function gets a strong password to begin with.
If the dc refuses to accept this password, this fn
can weaken the password by making it shorter.
The caller of this function should check if the length
of the supplied password was changed.
This function should preferably return a BOOL to
indicate this.
AccountState - if specified, the account will be set to this state.
possible values:
ACCOUNT_STATE_ENABLED, ACCOUNT_STATE_DISABLED
fIsNt4Dc - TRUE if the DC is NT4 or earlier.
Return Value:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus=NERR_Success;
NTSTATUS Status = STATUS_SUCCESS;
UNICODE_STRING DcName, AccountName;
OBJECT_ATTRIBUTES ObjectAttributes;
SAM_HANDLE SamHandle = NULL, DomainHandle = NULL, AccountHandle = NULL;
ULONG UserRid;
PULONG RidList = NULL;
PSID_NAME_USE NameUseList = NULL;
PUSER_CONTROL_INFORMATION UserAccountControl = NULL;
USER_SET_PASSWORD_INFORMATION PasswordInfo;
ULONG OldUserInfo;
BOOL fAccountControlModified = FALSE;
LPWSTR lpSamAccountName=lpAccountName;
ULONG AccountNameLen=0;
AccountNameLen = wcslen( lpAccountName );
//
// if caller has not passed in sam-account name,
// generate it from machine name ==> append $ at the end
//
if (lpAccountName[AccountNameLen-1] != L'$')
{
NetStatus = NetpGetMachineAccountName(lpAccountName,
&lpSamAccountName);
if (NetStatus != NERR_Success)
{
Status = STATUS_INSUFFICIENT_RESOURCES;
goto SetPasswordError;
}
}
RtlInitUnicodeString( &DcName, lpDcName );
RtlZeroMemory( &ObjectAttributes, sizeof( OBJECT_ATTRIBUTES ) );
Status = SamConnect( &DcName,
&SamHandle,
SAM_SERVER_CONNECT | SAM_SERVER_LOOKUP_DOMAIN,
&ObjectAttributes );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamConnect to %wZ failed with 0x%lx\n", &DcName, Status ));
goto SetPasswordError;
}
//
// Open the domain
//
Status = SamOpenDomain( SamHandle,
DOMAIN_LOOKUP,
DomainSid,
&DomainHandle );
if ( !NT_SUCCESS( Status ) ) {
#ifdef NETSETUP_VERBOSE_LOGGING
UNICODE_STRING DisplaySid;
NTSTATUS Status2;
RtlZeroMemory( &DisplaySid, sizeof( UNICODE_STRING ) );
Status2 = RtlConvertSidToUnicodeString( &DisplaySid, DomainSid, TRUE );
if ( NT_SUCCESS( Status2 ) ) {
NetpLog(( "SamOpenDomain on %wZ failed with 0x%lx\n",
&DisplaySid, Status ));
RtlFreeUnicodeString(&DisplaySid);
} else {
NetpLog(( "SamOpenDomain on <undisplayable sid> failed with 0x%lx\n",
Status ));
}
#endif
goto SetPasswordError;
}
//
// Get the RID of the user account
//
RtlInitUnicodeString( &AccountName, lpSamAccountName );
Status = SamLookupNamesInDomain( DomainHandle,
1,
&AccountName,
&RidList,
&NameUseList );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamLookupNamesInDomain on %wZ failed with 0x%lx\n",
&AccountName, Status ));
goto SetPasswordError;
}
UserRid = RidList[ 0 ];
SamFreeMemory( RidList );
SamFreeMemory( NameUseList );
//
// Finally, open the user account
//
Status = SamOpenUser( DomainHandle,
USER_FORCE_PASSWORD_CHANGE | USER_READ_ACCOUNT | USER_WRITE_ACCOUNT,
UserRid,
&AccountHandle );
if ( !NT_SUCCESS( Status ) ) {
Status = SamOpenUser( DomainHandle,
USER_FORCE_PASSWORD_CHANGE | USER_READ_ACCOUNT,
UserRid,
&AccountHandle );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamOpenUser on %lu failed with 0x%lx\n",
UserRid,
Status ));
goto SetPasswordError;
}
}
//
// Now, read the current user account type and see if it needs to be modified
//
Status = SamQueryInformationUser( AccountHandle,
UserControlInformation,
( PVOID * )&UserAccountControl );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamQueryInformationUser for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
}
OldUserInfo = UserAccountControl->UserAccountControl;
//
// Avoid whacking the account if it's anything other than a workstation account
//
if ( (OldUserInfo & USER_MACHINE_ACCOUNT_MASK) != USER_WORKSTATION_TRUST_ACCOUNT ) {
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: Broken account type 0x%lx -- error out\n",
OldUserInfo ));
Status = STATUS_USER_EXISTS;
goto SetPasswordError;
}
//
// Determine if the account control changes. If the account is being enabled,
// we want to perform the following sequence of operations for NT5: enable, disable,
// and enable again. This is needed to increase the USN (Universal Sequence
// Number) of this attribute so that the enabled value will win if the DS
// replication resolves colliding changes, as the following example shows.
// Suppose we have two DCs in the domain we join, A abd B. Suppose the account
// is currently disabled on A (because the user unjoined using that DC),
// but it is still enabled on B (because the replication hasn't happened yet).
// Suppose the user performs now joining to the domain. Then we have discovered
// B and so we proceed with setting up the changes to the existing account. If
// we don't toggle the account control attribute, then the USN of this attribute
// will not change on B (since attribute's value doesn't change) while it was
// incremented on A as the result of unjoin. At the replication time the data
// from A will rule and the account will be incorrectly marked as diabled.
//
// NOTE: This design may fail for the case of unjoining a domain that has
// three (or more) DCs, A, B, and C if the following sequence of operations
// happens. Suppose that the account is originally enabled on all DCs (state [1]
// in the bellow diagram). Then the user unjoins using DC A (state [2]). Then the
// user joins using B where the account is still enabled (state [3]). Then the user
// unjoins using C where the account is still enabled (state [4]). The final
// operation is unjoin, so the user expects that his account is disabled. We've
// assumed here that for some reason no replication was happening when these
// operations were performed. Then at the replication time the value from B will
// win (because of the additional toggling performed at the join time). But the
// account state on B is Enabled, so the final result will be that the account is
// enabled on all DCs which is not what the user expects.
//
// A B C
// Enabled [1] Enabled [1] Enabled [1]
// Disabled [2] Enabled (no-op)+Disabled (1 op) Disabled [4]
// Enabled [3]
//
if ( AccountState != ACCOUNT_STATE_IGNORE ) {
if ( ( AccountState == ACCOUNT_STATE_ENABLED ) &&
( (OldUserInfo & USER_ACCOUNT_DISABLED) || !fIsNt4Dc ) ) {
fAccountControlModified = TRUE;
UserAccountControl->UserAccountControl &= ~USER_ACCOUNT_DISABLED;
}
if ( ( AccountState == ACCOUNT_STATE_DISABLED ) &&
!( OldUserInfo & USER_ACCOUNT_DISABLED ) ) {
fAccountControlModified = TRUE;
UserAccountControl->UserAccountControl |= USER_ACCOUNT_DISABLED;
}
}
if ( fAccountControlModified == FALSE ) {
SamFreeMemory( UserAccountControl );
UserAccountControl = NULL;
}
//
// First, set the account type if required
//
if ( UserAccountControl ) {
Status = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamSetInformationUser for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
//
// If we are enabling the account, disable and re-enable it to
// make the two additional account state toggles.
//
} else if ( AccountState == ACCOUNT_STATE_ENABLED ) {
UserAccountControl->UserAccountControl |= USER_ACCOUNT_DISABLED;
Status = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS(Status) ) {
NetpLog(( "SamSetInformationUser (second) for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
}
UserAccountControl->UserAccountControl &= ~USER_ACCOUNT_DISABLED;
Status = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS(Status) ) {
NetpLog(( "SamSetInformationUser (third) for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
}
}
}
//
// If requested, set the password on the account
//
if ( lpPassword != NULL )
{
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
PasswordInfo.PasswordExpired = FALSE;
//
// Ok, then, set the password on the account
//
// The caller has passed in a strong password, try that first
// NT5 dcs will always accept a strong password.
//
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
if ( !NT_SUCCESS( Status ) )
{
if ( (Status == STATUS_PASSWORD_RESTRICTION) &&
!NetpIsDefaultPassword( lpAccountName, lpPassword ))
{
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: STATUS_PASSWORD_RESTRICTION error setting password. retrying...\n" ));
//
// SAM did not accpet a long password, try LM20_PWLEN
//
// This is probably because the dc is NT4 dc.
// NT4 dcs will not accept a password longer than LM20_PWLEN
//
lpPassword[LM20_PWLEN] = UNICODE_NULL;
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
if ( Status == STATUS_PASSWORD_RESTRICTION )
{
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: STATUS_PASSWORD_RESTRICTION error setting password. retrying...\n" ));
//
// SAM did not accpet a LM20_PWLEN password, try shorter one
//
// SAM uses RtlUpcaseUnicodeStringToOemString internally.
// In this process it is possible that in the worst case,
// n unicode char password will get mapped to 2*n dbcs
// char password. This will make it exceed LM20_PWLEN.
// To guard against this worst case, try a password
// with LM20_PWLEN/2 length
//
// One might say that LM20_PWLEN/2 length password
// is not really secure. I agree, but it is definitely
// better than the default password which we will have
// to fall back to otherwise.
//
lpPassword[LM20_PWLEN/2] = UNICODE_NULL;
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
if ( Status == STATUS_PASSWORD_RESTRICTION )
{
//
// SAM did not accpet a short pwd, try default pwd
//
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: STATUS_PASSWORD_RESTRICTION error setting password. retrying...\n" ));
NetpGenerateDefaultPassword(lpAccountName, lpPassword);
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
}
}
}
if ( NT_SUCCESS( Status ) )
{
NetpLog(( "NetpGenerateDefaultPassword: successfully set password\n" ));
}
else
{
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: SamSetInformationUser for UserSetPasswordInformation failed: 0x%lx\n", Status ));
//
// Make sure we try to restore the account control
//
if ( UserAccountControl )
{
NTSTATUS Status2;
UserAccountControl->UserAccountControl = OldUserInfo;
Status2 = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS( Status2 ) )
{
NetpLog(( "SamSetInformationUser for UserControlInformation (RESTORE) failed with 0x%lx\n", Status2 ));
}
}
goto SetPasswordError;
}
}
}
SetPasswordError:
if ( lpSamAccountName != lpAccountName )
{
NetApiBufferFree( lpSamAccountName );
}
if ( AccountHandle ) {
SamCloseHandle( AccountHandle );
}
if ( DomainHandle ) {
SamCloseHandle( DomainHandle );
}
if ( SamHandle ) {
SamCloseHandle( SamHandle );
}
NetStatus = RtlNtStatusToDosError( Status );
SamFreeMemory( UserAccountControl );
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpUpdateDnsRegistrations (
IN BOOL AddRegistrations
)
/*++
Routine Description:
This function removes or adds DNS registration entries via an entrypoint defined
in DHCPCSVC.DLL called DhcpRemoveDNSRegistrations or DhcpStaticRefreshParams,
respectively.
Arguments:
AddRegistrations -- TRUE if records should added. FALSE if records
should be deleted.
Return Value:
Error in the dll load. Otherwise the DHCP API return code.
--*/
{
NET_API_STATUS NetStatus = NO_ERROR;
HMODULE hModule = NULL;
DNS_REGISTRATION_REMOVAL_FN pfnRemove = NULL;
DNS_REGISTRATION_ADDITION_FN pfnAdd = NULL;
hModule = LoadLibraryW( L"dhcpcsvc.dll" );
if ( hModule != NULL ) {
if ( AddRegistrations ) {
pfnAdd = (DNS_REGISTRATION_ADDITION_FN)GetProcAddress(
hModule,
"DhcpStaticRefreshParams"
);
if ( pfnAdd != NULL ) {
NetStatus = ( *pfnAdd )( NULL ); // register on all adapters
} else {
NetStatus = ERROR_INVALID_DLL;
}
} else {
pfnRemove = (DNS_REGISTRATION_REMOVAL_FN)GetProcAddress(
hModule,
"DhcpRemoveDNSRegistrations"
);
if ( pfnRemove != NULL ) {
NetStatus = ( *pfnRemove )();
} else {
NetStatus = ERROR_INVALID_DLL;
}
}
FreeLibrary( hModule );
} else {
NetStatus = ERROR_DLL_NOT_FOUND;
}
return NetStatus;
}
//
// Helper functions
//
LPWSTR
GetStrPtr(IN LPWSTR szString OPTIONAL)
{
return szString ? szString : L"(NULL)";
}
NET_API_STATUS
NET_API_FUNCTION
NetpDuplicateString(IN LPCWSTR szSrc,
IN LONG cchSrc,
OUT LPWSTR* pszDst)
{
NET_API_STATUS NetStatus;
if (cchSrc < 0)
{
cchSrc = wcslen(szSrc);
}
++cchSrc;
NetStatus = NetApiBufferAllocate(cchSrc * sizeof( WCHAR ),
pszDst);
if ( NetStatus == NERR_Success )
{
wcsncpy(*pszDst, szSrc, cchSrc);
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpConcatStrings(IN LPCWSTR szSrc1,
IN LONG cchSrc1,
IN LPCWSTR szSrc2,
IN LONG cchSrc2,
OUT LPWSTR* pszDst)
{
NET_API_STATUS NetStatus;
if (cchSrc1 < 0)
{
cchSrc1 = wcslen(szSrc1);
}
if (cchSrc2 < 0)
{
cchSrc2 = wcslen(szSrc2);
}
NetStatus = NetApiBufferAllocate((cchSrc1 + cchSrc2 + 1) * sizeof( WCHAR ),
pszDst);
if ( NetStatus == NERR_Success )
{
wcsncpy(*pszDst, szSrc1, cchSrc1);
wcsncpy(*pszDst + cchSrc1, szSrc2, cchSrc2+1);
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpConcatStrings3(IN LPCWSTR szSrc1,
IN LONG cchSrc1,
IN LPCWSTR szSrc2,
IN LONG cchSrc2,
IN LPCWSTR szSrc3,
IN LONG cchSrc3,
OUT LPWSTR* pszDst)
{
NET_API_STATUS NetStatus;
if (cchSrc1 < 0)
{
cchSrc1 = wcslen(szSrc1);
}
if (cchSrc2 < 0)
{
cchSrc2 = wcslen(szSrc2);
}
if (cchSrc3 < 0)
{
cchSrc3 = wcslen(szSrc3);
}
NetStatus = NetApiBufferAllocate((cchSrc1 + cchSrc2 + cchSrc3 + 1) *
sizeof( WCHAR ), pszDst);
if ( NetStatus == NERR_Success )
{
wcsncpy(*pszDst, szSrc1, cchSrc1);
wcsncpy(*pszDst + cchSrc1, szSrc2, cchSrc2);
wcsncpy(*pszDst + cchSrc1 + cchSrc2, szSrc3, cchSrc3+1);
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetMachineAccountName(
IN LPCWSTR szMachineName,
OUT LPWSTR* pszMachineAccountName
)
/*++
Routine Description:
Get machine account name from machine name.
Arguments:
szMachineName -- name of a computer
pszMachineAccountName -- receives the name of computer account
Returns:
NERR_Success -- Success
Notes:
Caller must free the allocated memory using NetApiBufferFree.
--*/
{
NET_API_STATUS NetStatus;
ULONG ulLen;
LPWSTR szMachineAccountName;
ulLen = wcslen(szMachineName);
NetStatus = NetApiBufferAllocate( (ulLen + 2) * sizeof(WCHAR),
(PBYTE *) &szMachineAccountName );
if ( NetStatus == NERR_Success )
{
wcscpy(szMachineAccountName, szMachineName);
_wcsupr(szMachineAccountName);
szMachineAccountName[ulLen] = L'$';
szMachineAccountName[ulLen+1] = UNICODE_NULL;
*pszMachineAccountName = szMachineAccountName;
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpGeneratePassword(
IN LPCWSTR szMachine,
IN BOOL fRandomPwdPreferred,
IN LPCWSTR szDcName,
IN BOOL fIsNt4Dc,
OUT LPWSTR szPassword
)
/*++
Routine Description:
Arguments:
szMachine -- name of a computer
szPassword -- receives the generated password this buffer must be
atleast PWLEN+1 char long.
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
BOOL fUseDefaultPwd = FALSE;
// The default password is used if we are joining an NT4 DC
// that has RefusePasswordChange set. This is determined by
// remotely reading the appropriate netlogon regval.
// If the key cannot be read, it is assumed that the value is not set
//
if ( fIsNt4Dc )
{
//
// we are joining an NT4 domain, see if RefusePasswordChange is set
//
NetStatus = NetpGetNt4RefusePasswordChangeStatus( szDcName,
&fUseDefaultPwd );
}
if ( NetStatus == NERR_Success )
{
//
// if we are explicitly asked to use a default password, generate one
//
if ( fUseDefaultPwd )
{
NetpGenerateDefaultPassword(szMachine, szPassword);
}
//
// otherwise if the caller prefers a random password, generate one
//
else if ( fRandomPwdPreferred )
{
NetStatus = NetpGenerateRandomPassword(szPassword);
}
#if defined(REMOTE_BOOT)
//
// If it's a remote boot machine, then this will return the
// current machine account password, so use that.
//
else if (NERR_Success ==
NetpGetRemoteBootMachinePassword(szPassword))
{
// do nothing since the above already generated the password
}
#endif
else
{
//
// if none of the above apply,
// we end up generating a default password
//
NetpGenerateDefaultPassword(szMachine, szPassword);
NetStatus = NERR_Success;
}
}
return NetStatus;
}
void
NetpGenerateDefaultPassword(
IN LPCWSTR szMachine,
OUT LPWSTR szPassword
)
/*++
Routine Description:
Generate the default password from machine name.
This is simply the first 14 characters of the machine name lower cased.
Arguments:
szMachine -- name of a computer
szPassword -- receives the generated password
Returns:
NERR_Success -- Success
--*/
{
wcsncpy( szPassword, szMachine, LM20_PWLEN );
szPassword[LM20_PWLEN] = UNICODE_NULL;
_wcslwr( szPassword );
}
BOOL
NetpIsDefaultPassword(
IN LPCWSTR szMachine,
IN LPWSTR szPassword
)
/*++
Routine Description:
Determine if szPassword is the default password for szMachine
Arguments:
szMachine -- name of a computer
szPassword -- machine password
Returns:
TRUE if szPassword is the default password,
FALSE otherwise
--*/
{
WCHAR szPassword2[LM20_PWLEN+1];
NetpGenerateDefaultPassword(szMachine, szPassword2);
return (wcscmp(szPassword, szPassword2) == 0);
}
NET_API_STATUS
NET_API_FUNCTION
NetpGenerateRandomPassword(
OUT LPWSTR szPassword
)
{
NET_API_STATUS NetStatus=NERR_Success;
ULONG Length, i;
BYTE n;
HCRYPTPROV CryptProvider = 0;
LPWSTR szPwd=szPassword;
BOOL fStatus;
#define PWD_CHAR_MIN 32 // ' ' space
#define PWD_CHAR_MAX 122 // 'z'
//
// there is a reason behind this number
//
Length = 120;
szPassword[Length] = UNICODE_NULL;
//
// Generate a random password.
//
// the password is made of english printable chars. when w2k client
// joins NT4 dc. SAM on the dc calls RRtlUpcaseUnicodeStringToOemString
// the password length will remain unchanged. If we do not do this,
// the dc returns STATUS_PASSWORD_RESTRICTION and we have to
// fall back to default password.
//
if ( CryptAcquireContext( &CryptProvider, NULL, NULL,
PROV_RSA_FULL, CRYPT_VERIFYCONTEXT ) )
{
for ( i = 0; i < Length; i++, szPwd++ )
{
//
// the method we use here is not very efficient.
// This does not matter much in the context of NetJoin apis
// but it should not be used where perf is a criterion
//
while ( ( fStatus = CryptGenRandom( CryptProvider, sizeof(BYTE),
(LPBYTE) &n ) ) &&
( ( n < PWD_CHAR_MIN ) || ( n > PWD_CHAR_MAX ) ) )
{
// try till we get a non-zero random number
}
if ( fStatus )
{
*szPwd = (WCHAR) n;
}
else
{
NetStatus = GetLastError();
break;
}
}
CryptReleaseContext( CryptProvider, 0 );
}
else
{
NetStatus = GetLastError();
}
if ( NetStatus != NERR_Success )
{
NetpLog(( "NetpGenerateRandomPassword: failed: 0x%lx\n", NetStatus ));
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpStoreIntialDcRecord(
IN PDOMAIN_CONTROLLER_INFO DcInfo
)
/*++
Routine Description:
This function will cache the name of the domain controller on which we successfully
created/modified the machine account, so that the auth packages will know which dc to
try first
Arguments:
lpDcName - Name of the DC on which the account was created/modified
CreateNetlogonStoppedKey - If TRUE, a volatile key will be created
in the Netlogon registry section. The presence of this key
will instruct the client side of DsGetDcName( ) and the MSV1
package not to wait on netlogon to start.
Return Value:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
HKEY hNetLogon, hJoinKey = NULL;
ULONG Disp;
NetStatus = RegOpenKey( HKEY_LOCAL_MACHINE,
NETSETUPP_NETLOGON_JD_PATH,
&hNetLogon );
if ( NetStatus == NERR_Success ) {
NetStatus = RegCreateKeyEx( hNetLogon,
NETSETUPP_NETLOGON_JD,
0,
NULL,
REG_OPTION_NON_VOLATILE,
KEY_WRITE,
NULL,
&hJoinKey,
&Disp );
//
// Now, start creating all of the values. Ignore any failures, and don't write out
// NULL values
//
if ( NetStatus == NERR_Success ) {
PWSTR String = DcInfo->DomainControllerName;
//
// DomainControllerName
//
if ( String ) {
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DC,
0,
REG_SZ,
( const PBYTE )String,
( wcslen( String ) + 1 ) * sizeof( WCHAR ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_DC, String, NetStatus ));
}
}
//
// DomainControllerAddress
//
String = DcInfo->DomainControllerAddress;
if ( String ) {
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DCA,
0,
REG_SZ,
( const PBYTE )String,
( wcslen( String ) + 1 ) * sizeof( WCHAR ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_DCA, String, NetStatus ));
}
}
//
// DomainControllerType
//
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DCAT,
0,
REG_DWORD,
( const PBYTE )&DcInfo->DomainControllerAddressType,
sizeof( ULONG ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %lu failed with %lu\n",
NETSETUPP_NETLOGON_JD_DCAT,
DcInfo->DomainControllerAddressType, NetStatus ));
}
//
// DomainControllerType
//
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DG,
0,
REG_BINARY,
( const PBYTE )&DcInfo->DomainGuid,
sizeof( GUID ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_DG, NetStatus ));
}
//
// DomainName
//
String = DcInfo->DomainName;
if ( String ) {
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DN,
0,
REG_SZ,
( const PBYTE )String,
( wcslen( String ) + 1 ) * sizeof( WCHAR ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_DN, String, NetStatus ));
}
}
//
// DnsForestName
//
String = DcInfo->DnsForestName;
if ( String ) {
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DFN,
0,
REG_SZ,
( const PBYTE )String,
( wcslen( String ) + 1 ) * sizeof( WCHAR ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_DFN, String, NetStatus ));
}
}
//
// Flags
//
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_F,
0,
REG_DWORD,
( const PBYTE )&DcInfo->Flags,
sizeof( ULONG ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %lu failed with %lu\n",
NETSETUPP_NETLOGON_JD_F, DcInfo->Flags, NetStatus ));
}
//
// DcSiteName
//
String = DcInfo->DcSiteName;
if ( String ) {
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_DSN,
0,
REG_SZ,
( const PBYTE )String,
( wcslen( String ) + 1 ) * sizeof( WCHAR ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_DSN, String, NetStatus ));
}
}
//
// DcSiteName
//
String = DcInfo->ClientSiteName;
if ( String ) {
NetStatus = RegSetValueEx( hJoinKey,
NETSETUPP_NETLOGON_JD_CSN,
0,
REG_SZ,
( const PBYTE )String,
( wcslen( String ) + 1 ) * sizeof( WCHAR ) );
if ( NetStatus != NERR_Success ) {
NetpLog(( "Set of value %ws to %ws failed with %lu\n",
NETSETUPP_NETLOGON_JD_CSN, String, NetStatus ));
}
}
RegCloseKey( hJoinKey );
}
RegCloseKey( hNetLogon );
}
return( NetStatus );
}
VOID
NetpAvoidNetlogonSpnSet(
BOOL AvoidSet
)
/*++
Routine Description:
This function will write into Netlogon reg key to instruct Netlogon
not to register DnsHostName and SPNs. This is needed because
Netlogon could otherwise set incorrect values based on the old computer
name. The registry key that this function writes is volatile so that
Netlogon will notice it before the reboot but it will not exist after
the reboot when Netlogon will have the new computer name.
Arguments:
AvoidSet - If TRUE, this routine will inform netlogon to not write SPNs
Otherwise, it will delete the reg key which we may have set previously.
Return Value:
None
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
HKEY hNetLogon = NULL;
HKEY hNetLogonAvoidSpnSet = NULL;
ULONG Disp;
NetStatus = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
NETSETUPP_NETLOGON_JD_PATH,
0,
KEY_ALL_ACCESS,
&hNetLogon );
if ( NetStatus == NERR_Success ) {
//
// If we are to avoid SPN setting by netlogon,
// write the appropriate reg key to inform Netlogon accordingly
//
if ( AvoidSet ) {
NetStatus = RegCreateKeyEx( hNetLogon,
NETSETUPP_NETLOGON_AVOID_SPN,
0,
NULL,
REG_OPTION_VOLATILE,
KEY_WRITE,
NULL,
&hNetLogonAvoidSpnSet,
&Disp );
if ( NetStatus == NERR_Success ) {
RegCloseKey( hNetLogonAvoidSpnSet );
}
//
// Otherwise, delete the reg key which we may have set previously.
//
} else {
RegDeleteKey( hNetLogon,
NETSETUPP_NETLOGON_AVOID_SPN );
}
RegCloseKey( hNetLogon );
}
}
NET_API_STATUS
NET_API_FUNCTION
NetpWaitForNetlogonSc(
IN LPCWSTR szDomainName
)
{
NET_API_STATUS NetStatus = NERR_Success;
NTSTATUS NlSubStatus=STATUS_SUCCESS;
LPBYTE pNetlogonInfo=NULL;
UINT cAttempts=0;
BOOLEAN fScSetup=FALSE;
PNETLOGON_INFO_2 pNetlogonInfo2;
#define NL_SC_WAIT_INTERVAL 2000
#define NL_SC_WAIT_NUM_ATTEMPTS 60
NetpLog(( "NetpWaitForNetlogonSc: waiting for netlogon secure channel setup...\n"));
while (!fScSetup && (cAttempts < NL_SC_WAIT_NUM_ATTEMPTS))
{
cAttempts++;
NetStatus = I_NetLogonControl2( NULL, NETLOGON_CONTROL_TC_QUERY,
2, (LPBYTE) &szDomainName,
(LPBYTE *) &pNetlogonInfo );
if (NetStatus == NERR_Success)
{
pNetlogonInfo2 = (PNETLOGON_INFO_2) pNetlogonInfo;
NlSubStatus = pNetlogonInfo2->netlog2_tc_connection_status;
fScSetup = NlSubStatus == NERR_Success;
NetApiBufferFree(pNetlogonInfo);
}
if (!fScSetup)
{
Sleep(NL_SC_WAIT_INTERVAL);
}
}
NetpLog(( "NetpWaitForNetlogonSc: status: 0x%lx, sub-status: 0x%lx\n",
NetStatus, NlSubStatus));
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetDefaultLcidOnMachine(
IN LPCWSTR szMachine,
OUT LCID* plcidMachine
)
{
NET_API_STATUS NetStatus = NERR_Success;
HKEY hkeyRemoteMachine, hkeyLanguage;
WCHAR szLocale[16];
DWORD dwLocaleSize=0;
DWORD dwType;
static WCHAR c_szRegKeySystemLanguage[] =
L"System\\CurrentControlSet\\Control\\Nls\\Locale";
static WCHAR c_szRegValDefault[] = L"(Default)";
//
// Connect to the remote registry
//
if ( NetStatus == NERR_Success )
{
NetStatus = RegConnectRegistry( szMachine,
HKEY_LOCAL_MACHINE,
&hkeyRemoteMachine );
//
// Now, open the system language key
//
if ( NetStatus == NERR_Success )
{
NetStatus = RegOpenKeyEx( hkeyRemoteMachine,
c_szRegKeySystemLanguage,
0, KEY_READ, &hkeyLanguage);
//
// get default locale
//
if ( NetStatus == NERR_Success )
{
dwLocaleSize = sizeof( szLocale );
NetStatus = RegQueryValueEx( hkeyLanguage,
c_szRegValDefault,
NULL, &dwType,
(LPBYTE) szLocale,
&dwLocaleSize );
if ( NetStatus == NERR_Success)
{
if ((dwType == REG_SZ) &&
(swscanf(szLocale, L"%lx", plcidMachine) != 1))
{
//$ REVIEW kumarp 29-May-1999
// better errorcode?
NetStatus = ERROR_INVALID_PARAMETER;
}
}
RegCloseKey( hkeyLanguage );
}
RegCloseKey( hkeyRemoteMachine );
}
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpVerifyStrOemCompatibleInLocale(
IN LPCWSTR szString,
IN LCID lcidRemote
)
{
NET_API_STATUS NetStatus = NERR_Success;
NTSTATUS NtStatus=STATUS_SUCCESS;
OEM_STRING osLocal = { 0 };
OEM_STRING osRemote = { 0 };
UNICODE_STRING sString;
LCID lcidLocal;
lcidLocal = GetThreadLocale();
RtlInitUnicodeString(&sString, szString);
NtStatus = RtlUnicodeStringToOemString(&osLocal, &sString, TRUE);
__try
{
if (NtStatus == STATUS_SUCCESS)
{
if (SetThreadLocale(lcidRemote))
{
NtStatus = RtlUnicodeStringToOemString(&osRemote,
&sString, TRUE);
if (NtStatus == STATUS_SUCCESS)
{
if (!RtlEqualMemory(osLocal.Buffer, osRemote.Buffer,
osLocal.Length))
{
NetStatus = NERR_NameUsesIncompatibleCodePage;
}
}
else
{
NetStatus = RtlNtStatusToDosError(NtStatus);
}
}
else
{
NetStatus = GetLastError();
}
}
else
{
NetStatus = RtlNtStatusToDosError(NtStatus);
}
}
__finally
{
if (!SetThreadLocale(lcidLocal))
{
NetStatus = GetLastError();
}
// RtlFreeOemString checks for NULL Buffer
RtlFreeOemString(&osLocal);
RtlFreeOemString(&osRemote);
}
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpVerifyStrOemCompatibleOnMachine(
IN LPCWSTR szRemoteMachine,
IN LPCWSTR szString
)
{
NET_API_STATUS NetStatus = NERR_Success;
LCID lcidRemoteMachine;
NetStatus = NetpGetDefaultLcidOnMachine(szRemoteMachine,
&lcidRemoteMachine);
if (NetStatus == NERR_Success)
{
NetStatus = NetpVerifyStrOemCompatibleInLocale(szString,
lcidRemoteMachine);
}
return NetStatus;
}
#define NETP_NETLOGON_PATH L"System\\CurrentControlSet\\services\\Netlogon\\parameters\\"
#define NETP_NETLOGON_RPC L"RefusePasswordChange"
NET_API_STATUS
NET_API_FUNCTION
NetpGetNt4RefusePasswordChangeStatus(
IN LPCWSTR Nt4Dc,
OUT BOOL* RefusePasswordChangeSet
)
/*++
Routine Description:
Read the regkey NETP_NETLOGON_PATH\NETP_NETLOGON_RPC on Nt4Dc.
Return the value read in the out parameter.
Arguments:
Nt4Dc -- name of machine to read reg. from
RefusePasswordChangeSet -- value returned
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
PWSTR FullComputerName = NULL;
HKEY NetlogonRootKey, DcKey;
ULONG Length, Type;
DWORD Value;
*RefusePasswordChangeSet = FALSE;
//
// Build the full computer name if necessary
//
if ( *Nt4Dc != L'\\' )
{
NetStatus = NetApiBufferAllocate( ( wcslen( Nt4Dc ) + 3 ) * sizeof( WCHAR ),
( LPVOID * )&FullComputerName );
if ( NetStatus == NERR_Success )
{
swprintf( FullComputerName, L"\\\\%ws", Nt4Dc );
}
}
else
{
FullComputerName = (LPWSTR) Nt4Dc;
}
NetpLog(( "NetpGetNt4RefusePasswordChangeStatus: trying to read from '%ws'\n", FullComputerName));
//
// Connect to the remote registry
//
if ( NetStatus == NERR_Success )
{
NetStatus = RegConnectRegistry( FullComputerName,
HKEY_LOCAL_MACHINE,
&DcKey );
//
// Now, open the netlogon parameters section
//
if ( NetStatus == NERR_Success )
{
NetStatus = RegOpenKeyEx( DcKey,
NETP_NETLOGON_PATH,
0,
KEY_READ,
&NetlogonRootKey);
//
// Now, see if the key actually exists...
//
if ( NetStatus == NERR_Success )
{
Length = sizeof( Value );
NetStatus = RegQueryValueEx( NetlogonRootKey,
NETP_NETLOGON_RPC,
NULL,
&Type,
( LPBYTE )&Value,
&Length );
if ( NetStatus == NERR_Success)
{
NetpLog(( "NetpGetNt4RefusePasswordChangeStatus: RefusePasswordChange == %d\n", Value));
if ( Value != 0 )
{
*RefusePasswordChangeSet = TRUE;
}
}
RegCloseKey( NetlogonRootKey );
}
RegCloseKey( DcKey );
}
}
if ( FullComputerName != Nt4Dc )
{
NetApiBufferFree( FullComputerName );
}
//
// If anything went wrong, ignore it...
//
if ( NetStatus != NERR_Success )
{
NetpLog(( "NetpGetNt4RefusePasswordChangeStatus: failed but ignored the failure: 0x%lx\n", NetStatus ));
NetStatus = NERR_Success;
}
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpGetComputerNameAllocIfReqd(
OUT LPWSTR* ppwszMachine,
IN UINT cLen
)
/*++
Routine Description:
Get name of the computer on which this runs. Alloc a buffer
if the name is longer than cLen.
Arguments:
ppwszMachine -- pointer to buffer. this receives a buffer if allocated.
cLen -- length of buffer pointed to by *ppwszMachine.
If the computer name to be returned is longer than this
a new buffer is allocated.
Returns:
NERR_Success -- Success
--*/
{
NET_API_STATUS NetStatus=NERR_Success;
if ( GetComputerName( *ppwszMachine, &cLen ) == FALSE )
{
NetStatus = GetLastError();
if ( (NetStatus == ERROR_INSUFFICIENT_BUFFER) ||
(NetStatus == ERROR_BUFFER_OVERFLOW) )
{
// allocate an extra char for the append-$ case
NetStatus = NetApiBufferAllocate( (cLen + 1 + 1) * sizeof(WCHAR),
(PBYTE *) ppwszMachine );
if ( NetStatus == NERR_Success )
{
if ( GetComputerName( *ppwszMachine, &cLen ) == FALSE )
{
NetStatus = GetLastError();
}
}
}
}
return NetStatus;
}
// ======================================================================
//
// Note: all code below this has been added as helper code for
// NetpSetComputerAccountPassword. this function is used by
// netdom.exe to fix a dc that was rendered unusable because
// of a ds restore resulting into 2+ password mismatch on
// machine account.
//
// This entire code is temporary and should be removed and
// rewritten post w2k.
//
static
NET_API_STATUS
NET_API_FUNCTION
NetpEncodePassword(
IN LPWSTR lpPassword,
IN OUT PUCHAR Seed,
OUT LPWSTR *EncodedPassword,
OUT PULONG EncodedPasswordLength
)
{
NET_API_STATUS status = NERR_Success;
UNICODE_STRING EncodedPasswordU;
PWSTR PasswordPart;
ULONG PwdLen;
*EncodedPassword = NULL;
*EncodedPasswordLength = 0;
if ( lpPassword ) {
PwdLen = wcslen( ( LPWSTR )lpPassword ) * sizeof( WCHAR );
PwdLen += sizeof( WCHAR ) + sizeof( WCHAR );
status = NetApiBufferAllocate( PwdLen,
( PVOID * )EncodedPassword );
if ( status == NERR_Success ) {
//
// We'll put the encode byte as the first character in the string
//
PasswordPart = ( *EncodedPassword ) + 1;
wcscpy( PasswordPart, ( LPWSTR )lpPassword );
RtlInitUnicodeString( &EncodedPasswordU, PasswordPart );
*Seed = 0;
RtlRunEncodeUnicodeString( Seed, &EncodedPasswordU );
*( PWCHAR )( *EncodedPassword ) = ( WCHAR )*Seed;
//
// Encode the old password as well...
//
RtlInitUnicodeString( &EncodedPasswordU, lpPassword );
RtlRunEncodeUnicodeString( Seed, &EncodedPasswordU );
*EncodedPasswordLength = PwdLen;
}
}
return( status );
}
NTSTATUS
NetpLsaOpenSecret2(
IN LSA_HANDLE hLsa,
IN PUNICODE_STRING pusSecretName,
IN ACCESS_MASK DesiredAccess,
OUT PLSA_HANDLE phSecret
)
{
NTSTATUS Status = STATUS_UNSUCCESSFUL;
HANDLE hToken=NULL;
__try
{
if (OpenThreadToken(GetCurrentThread(), TOKEN_IMPERSONATE,
TRUE, &hToken))
{
if (SetThreadToken(NULL, NULL))
{
Status = STATUS_SUCCESS;
}
}
else
{
if (GetLastError() == ERROR_NO_TOKEN)
{
Status = STATUS_SUCCESS;
}
}
if ( NT_SUCCESS(Status) )
{
Status = LsaOpenSecret(hLsa, pusSecretName,
DesiredAccess, phSecret);
}
}
__finally
{
if (hToken)
{
if ( !SetThreadToken(NULL, hToken) ) {
NetpLog(( "NetpLsaOpenSecret2: Couldn't reset the user token 0x%lx\n",
GetLastError() ));
Status = NetpApiStatusToNtStatus( GetLastError() );
}
}
}
if ( hToken ) {
CloseHandle( hToken );
}
NetpLog(( "NetpLsaOpenSecret: status: 0x%lx\n", Status ));
return Status;
}
NET_API_STATUS
NET_API_FUNCTION
NetpManageMachineSecret2(
IN LPWSTR lpMachine,
IN LPWSTR lpPassword,
IN DWORD Action,
OUT PLSA_HANDLE pPolicyHandle OPTIONAL
)
{
NTSTATUS Status = STATUS_SUCCESS;
LSA_HANDLE LocalPolicy = NULL, SecretHandle = NULL;
UNICODE_STRING Key, Data, *CurrentValue = NULL;
BOOLEAN SecretCreated = FALSE;
if( Action == NETSETUPP_CREATE )
{
ASSERT( lpPassword );
}
UNREFERENCED_PARAMETER( lpMachine );
Status = NetpGetLsaHandle( NULL, pPolicyHandle, &LocalPolicy );
//
// open/create the secret
//
if ( NT_SUCCESS( Status ) )
{
RtlInitUnicodeString( &Key, L"$MACHINE.ACC" );
RtlInitUnicodeString( &Data, lpPassword );
Status = NetpLsaOpenSecret2( LocalPolicy, &Key,
Action == NETSETUPP_CREATE ?
SECRET_SET_VALUE | SECRET_QUERY_VALUE : DELETE,
&SecretHandle );
if ( Status == STATUS_OBJECT_NAME_NOT_FOUND )
{
if ( Action == NETSETUPP_DELETE )
{
Status = STATUS_SUCCESS;
}
else
{
Status = LsaCreateSecret( LocalPolicy, &Key,
SECRET_SET_VALUE, &SecretHandle );
if ( NT_SUCCESS( Status ) )
{
SecretCreated = TRUE;
}
}
}
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "NetpManageMachineSecret: Open/Create secret failed: 0x%lx\n", Status ));
}
if ( NT_SUCCESS( Status ) )
{
if ( Action == NETSETUPP_CREATE )
{
//
// First, read the current value, so we can save it as the old value
//
if ( SecretCreated )
{
CurrentValue = &Data;
}
else
{
Status = LsaQuerySecret( SecretHandle, &CurrentValue,
NULL, NULL, NULL );
}
if ( NT_SUCCESS( Status ) )
{
//
// Now, store both the new password and the old
//
Status = LsaSetSecret( SecretHandle, &Data, CurrentValue );
if ( !SecretCreated )
{
LsaFreeMemory( CurrentValue );
}
}
}
else
{
//
// No secret handle means we failed earlier in
// some intermediate state. That's ok, just press on.
//
if ( SecretHandle != NULL )
{
Status = LsaDelete( SecretHandle );
if ( NT_SUCCESS( Status ) )
{
SecretHandle = NULL;
}
}
}
}
if ( SecretHandle )
{
LsaClose( SecretHandle );
}
}
NetpSetLsaHandle( LocalPolicy, pPolicyHandle );
if ( !NT_SUCCESS( Status ) )
{
NetpLog(( "NetpManageMachineSecret: '%s' operation failed: 0x%lx\n",
Action == NETSETUPP_CREATE ? "CREATE" : "DELETE", Status ));
}
return( RtlNtStatusToDosError( Status ) );
}
NET_API_STATUS
NET_API_FUNCTION
NetpSetMachineAccountPasswordAndTypeEx2(
IN LPWSTR lpDcName,
IN PSID DomainSid,
IN LPWSTR lpAccountName,
IN OUT OPTIONAL LPWSTR lpPassword,
IN OPTIONAL UCHAR AccountState
)
{
NET_API_STATUS NetStatus=NERR_Success;
NTSTATUS Status = STATUS_SUCCESS;
UNICODE_STRING DcName, AccountName;
OBJECT_ATTRIBUTES ObjectAttributes;
SAM_HANDLE SamHandle = NULL, DomainHandle = NULL, AccountHandle = NULL;
ULONG UserRid;
PULONG RidList = NULL;
PSID_NAME_USE NameUseList = NULL;
PUSER_CONTROL_INFORMATION UserAccountControl = NULL;
USER_SET_PASSWORD_INFORMATION PasswordInfo;
ULONG OldUserInfo;
BOOL fAccountControlModified = FALSE;
LPWSTR lpSamAccountName=lpAccountName;
ULONG AccountNameLen=0;
AccountNameLen = wcslen( lpAccountName );
//
// if caller has not passed in sam-account name,
// generate it from machine name ==> append $ at the end
//
if (lpAccountName[AccountNameLen-1] != L'$')
{
NetStatus = NetpGetMachineAccountName(lpAccountName,
&lpSamAccountName);
if (NetStatus != NERR_Success)
{
Status = STATUS_INSUFFICIENT_RESOURCES;
goto SetPasswordError;
}
}
RtlInitUnicodeString( &DcName, lpDcName );
RtlZeroMemory( &ObjectAttributes, sizeof( OBJECT_ATTRIBUTES ) );
Status = SamConnect( &DcName,
&SamHandle,
SAM_SERVER_CONNECT | SAM_SERVER_LOOKUP_DOMAIN,
&ObjectAttributes );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamConnect to %wZ failed with 0x%lx\n", &DcName, Status ));
goto SetPasswordError;
}
//
// Open the domain
//
Status = SamOpenDomain( SamHandle,
DOMAIN_LOOKUP,
DomainSid,
&DomainHandle );
if ( !NT_SUCCESS( Status ) ) {
#ifdef NETSETUP_VERBOSE_LOGGING
UNICODE_STRING DisplaySid;
NTSTATUS Status2;
RtlZeroMemory( &DisplaySid, sizeof( UNICODE_STRING ) );
Status2 = RtlConvertSidToUnicodeString( &DisplaySid, DomainSid, TRUE );
if ( NT_SUCCESS( Status2 ) ) {
NetpLog(( "SamOpenDomain on %wZ failed with 0x%lx\n",
&DisplaySid, Status ));
RtlFreeUnicodeString(&DisplaySid);
} else {
NetpLog(( "SamOpenDomain on <undisplayable sid> failed with 0x%lx\n",
Status ));
}
#endif
goto SetPasswordError;
}
//
// Get the RID of the user account
//
RtlInitUnicodeString( &AccountName, lpSamAccountName );
Status = SamLookupNamesInDomain( DomainHandle,
1,
&AccountName,
&RidList,
&NameUseList );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamLookupNamesInDomain on %wZ failed with 0x%lx\n",
&AccountName, Status ));
goto SetPasswordError;
}
UserRid = RidList[ 0 ];
SamFreeMemory( RidList );
SamFreeMemory( NameUseList );
//
// Finally, open the user account
//
Status = SamOpenUser( DomainHandle,
USER_FORCE_PASSWORD_CHANGE | USER_READ_ACCOUNT | USER_WRITE_ACCOUNT,
UserRid,
&AccountHandle );
if ( !NT_SUCCESS( Status ) ) {
Status = SamOpenUser( DomainHandle,
USER_FORCE_PASSWORD_CHANGE | USER_READ_ACCOUNT,
UserRid,
&AccountHandle );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamOpenUser on %lu failed with 0x%lx\n",
UserRid,
Status ));
goto SetPasswordError;
}
}
//
// Now, read the current user account type and see if it needs to be modified
//
Status = SamQueryInformationUser( AccountHandle,
UserControlInformation,
( PVOID * )&UserAccountControl );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamQueryInformationUser for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
}
OldUserInfo = UserAccountControl->UserAccountControl;
//
// Determine if the account control changes. If the account is being enabled,
// we want to perform the following sequence of operations: enable, disable,
// and enable again. This is needed to increase the USN (Universal Sequence
// Number) of this attribute so that the enabled value will win if the DS
// replication resolves colliding changes, as the following example shows.
// Suppose we have two DCs in the domain we join, A abd B. Suppose the account
// is currently disabled on A (because the user unjoined using that DC),
// but it is still enabled on B (because the replication hasn't happened yet).
// Suppose the user performs now joining to the domain. Then we have discovered
// B and so we proceed with setting up the changes to the existing account. If
// we don't toggle the account control attribute, then the USN of this attribute
// will not change on B (since attribute's value doesn't change) while it was
// incremented on A as the result of unjoin. At the replication time the data
// from A will rule and the account will be incorrectly marked as diabled.
//
// NOTE: This design may fail for the case of unjoining a domain that has
// three (or more) DCs, A, B, and C if the following sequence of operations
// happens. Suppose that the account is originally enabled on all DCs (state [1]
// in the bellow diagram). Then the user unjoins using DC A (state [2]). Then the
// user joins using B where the account is still enabled (state [3]). Then the user
// unjoins using C where the account is still enabled (state [4]). The final
// operation is unjoin, so the user expects that his account is disabled. We've
// assumed here that for some reason no replication was happening when these
// operations were performed. Then at the replication time the value from B will
// win (because of the additional toggling performed at the join time). But the
// account state on B is Enabled, so the final result will be that the account is
// enabled on all DCs which is not what the user expects.
//
// A B C
// Enabled [1] Enabled [1] Enabled [1]
// Disabled [2] Enabled (no-op)+Disabled (1 op) Disabled [4]
// Enabled [3]
//
if ( AccountState != ACCOUNT_STATE_IGNORE ) {
if ( AccountState == ACCOUNT_STATE_ENABLED ) {
fAccountControlModified = TRUE;
UserAccountControl->UserAccountControl &= ~USER_ACCOUNT_DISABLED;
}
if ( ( AccountState == ACCOUNT_STATE_DISABLED ) &&
!( OldUserInfo & USER_ACCOUNT_DISABLED ) ) {
fAccountControlModified = TRUE;
UserAccountControl->UserAccountControl |= USER_ACCOUNT_DISABLED;
}
}
if ( fAccountControlModified == FALSE ) {
SamFreeMemory( UserAccountControl );
UserAccountControl = NULL;
}
//
// First, set the account type if required
//
if ( UserAccountControl ) {
Status = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS( Status ) ) {
NetpLog(( "SamSetInformationUser for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
//
// If we are enabling the account, disable and re-enable it to
// make the two additional account state toggles.
//
} else if ( AccountState == ACCOUNT_STATE_ENABLED ) {
UserAccountControl->UserAccountControl |= USER_ACCOUNT_DISABLED;
Status = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS(Status) ) {
NetpLog(( "SamSetInformationUser (second) for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
}
UserAccountControl->UserAccountControl &= ~USER_ACCOUNT_DISABLED;
Status = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS(Status) ) {
NetpLog(( "SamSetInformationUser (third) for UserControlInformation "
"failed with 0x%lx\n", Status ));
goto SetPasswordError;
}
}
}
//
// If requested, set the password on the account
//
if ( lpPassword != NULL )
{
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
PasswordInfo.PasswordExpired = FALSE;
//
// Ok, then, set the password on the account
//
// The caller has passed in a strong password, try that first
// NT5 dcs will always accept a strong password.
//
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
if ( !NT_SUCCESS( Status ) )
{
if ( (Status == STATUS_PASSWORD_RESTRICTION) &&
!NetpIsDefaultPassword( lpAccountName, lpPassword ))
{
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: STATUS_PASSWORD_RESTRICTION error setting password. retrying...\n" ));
//
// SAM did not accpet a long password, try LM20_PWLEN
//
// This is probably because the dc is NT4 dc.
// NT4 dcs will not accept a password longer than LM20_PWLEN
//
lpPassword[LM20_PWLEN] = UNICODE_NULL;
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
if ( Status == STATUS_PASSWORD_RESTRICTION )
{
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: STATUS_PASSWORD_RESTRICTION error setting password. retrying...\n" ));
//
// SAM did not accpet a LM20_PWLEN password, try shorter one
//
// SAM uses RtlUpcaseUnicodeStringToOemString internally.
// In this process it is possible that in the worst case,
// n unicode char password will get mapped to 2*n dbcs
// char password. This will make it exceed LM20_PWLEN.
// To guard against this worst case, try a password
// with LM20_PWLEN/2 length
//
// One might say that LM20_PWLEN/2 length password
// is not really secure. I agree, but it is definitely
// better than the default password which we will have
// to fall back to otherwise.
//
lpPassword[LM20_PWLEN/2] = UNICODE_NULL;
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
if ( Status == STATUS_PASSWORD_RESTRICTION )
{
//
// SAM did not accpet a short pwd, try default pwd
//
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: STATUS_PASSWORD_RESTRICTION error setting password. retrying...\n" ));
NetpGenerateDefaultPassword(lpAccountName, lpPassword);
RtlInitUnicodeString( &PasswordInfo.Password, lpPassword );
Status = SamSetInformationUser( AccountHandle,
UserSetPasswordInformation,
( PVOID )&PasswordInfo );
}
}
}
if ( NT_SUCCESS( Status ) )
{
NetpLog(( "NetpGenerateDefaultPassword: successfully set password\n" ));
}
else
{
NetpLog(( "NetpSetMachineAccountPasswordAndTypeEx: SamSetInformationUser for UserSetPasswordInformation failed: 0x%lx\n", Status ));
//
// Make sure we try to restore the account control
//
if ( UserAccountControl )
{
NTSTATUS Status2;
UserAccountControl->UserAccountControl = OldUserInfo;
Status2 = SamSetInformationUser( AccountHandle,
UserControlInformation,
( PVOID )UserAccountControl );
if ( !NT_SUCCESS( Status2 ) )
{
NetpLog(( "SamSetInformationUser for UserControlInformation (RESTORE) failed with 0x%lx\n", Status2 ));
}
}
goto SetPasswordError;
}
}
}
SetPasswordError:
if ( lpSamAccountName != lpAccountName )
{
NetApiBufferFree( lpSamAccountName );
}
if ( AccountHandle ) {
SamCloseHandle( AccountHandle );
}
if ( DomainHandle ) {
SamCloseHandle( DomainHandle );
}
if ( SamHandle ) {
SamCloseHandle( SamHandle );
}
NetStatus = RtlNtStatusToDosError( Status );
SamFreeMemory( UserAccountControl );
return( NetStatus );
}
NET_API_STATUS
NET_API_FUNCTION
NetpSetComputerAccountPassword(
IN PWSTR szMachine,
IN PWSTR szDomainController,
IN PWSTR szUser,
IN PWSTR szUserPassword,
IN PVOID Reserved
)
{
NET_API_STATUS NetStatus=NERR_Success;
NET_API_STATUS NetStatus2=NERR_Success;
BOOL fIpcConnected = FALSE;
BYTE bSeed;
PPOLICY_PRIMARY_DOMAIN_INFO pPolicyPDI = NULL;
PPOLICY_DNS_DOMAIN_INFO pPolicyDns = NULL;
LSA_HANDLE hDC = NULL;
WCHAR szMachinePassword[ PWLEN + 1];
WCHAR szMachineNameBuf[MAX_COMPUTERNAME_LENGTH + 1];
PWSTR szMachineName=szMachineNameBuf;
NetSetuppOpenLog();
NetpLog(( "NetpSetComputerAccountPassword: for '%ws' on '%ws' using '%ws' creds\n", GetStrPtr(szMachine), GetStrPtr(szDomainController), GetStrPtr(szUser) ));
if ( ( szDomainController == NULL ) ||
( szUser == NULL ) ||
( szUserPassword == NULL ) )
{
NetStatus = ERROR_INVALID_PARAMETER;
goto Cleanup;
}
if ( szMachine == NULL )
{
NetStatus = NetpGetComputerNameAllocIfReqd(&szMachineName,
MAX_COMPUTERNAME_LENGTH);
}
else
{
szMachineName = szMachine;
}
NetStatus = NetpManageIPCConnect( szDomainController,
szUser, szUserPassword,
NETSETUPP_CONNECT_IPC );
NetpLog(( "NetpSetComputerAccountPassword: status of connecting to dc '%ws': 0x%lx\n", szDomainController, NetStatus ));
//
// get the lsa domain info on the DC
//
if ( NetStatus == NERR_Success )
{
fIpcConnected = TRUE;
NetStatus = NetpGetLsaPrimaryDomain(szDomainController,
&pPolicyPDI, &pPolicyDns, &hDC);
}
if (NetStatus == NERR_Success)
{
//
// Generate the password to use on the machine account.
//
NetStatus = NetpGeneratePassword( szMachineName,
TRUE, // fRandomPwdPreferred
szDomainController,
FALSE, // fIsNt4Dc
szMachinePassword );
NetpLog(( "NetpSetComputerAccountPassword: status of generating machine password: 0x%lx\n", NetStatus ));
}
if (NetStatus == NERR_Success)
{
NetStatus = NetpSetMachineAccountPasswordAndTypeEx2(
szDomainController, pPolicyPDI->Sid,
szMachineName, szMachinePassword,
ACCOUNT_STATE_IGNORE
);
NetpLog(( "NetpSetComputerAccountPassword: status of setting machine password on %ws: 0x%lx\n", GetStrPtr(szDomainController), NetStatus ));
}
if (NetStatus == NERR_Success)
{
// if we are not creating the machine account,
// just set the password
NetStatus = NetpSetMachineAccountPasswordAndTypeEx2(
szMachineName, pPolicyPDI->Sid,
szMachineName, szMachinePassword,
ACCOUNT_STATE_IGNORE
);
NetpLog(( "NetpSetComputerAccountPassword: status of setting machine password on %ws: 0x%lx\n", GetStrPtr(szMachineName), NetStatus ));
}
//
// set the local machine secret
//
if ( NetStatus == NERR_Success )
{
if ( NetStatus == NERR_Success )
{
NetStatus = NetpManageMachineSecret2( szMachineName,
szMachinePassword,
NETSETUPP_CREATE, NULL );
}
NetpLog(( "NetpSetComputerAccountPassword: status of setting local secret: 0x%lx\n", NetStatus ));
}
//
// Now, we no longer need our session to our dc
//
if ( fIpcConnected )
{
//RtlRunDecodeUnicodeString( bSeed, &usEncodedPassword );
NetStatus2 = NetpManageIPCConnect( szDomainController, szUser,
//usEncodedPassword.Buffer,
NULL,
NETSETUPP_DISCONNECT_IPC );
//RtlRunEncodeUnicodeString( &bSeed, &usEncodedPassword );
NetpLog(( "NetpJoinDomain: status of disconnecting from '%ws': 0x%lx\n", szDomainController, NetStatus2));
}
Cleanup:
if ( (szMachineName != szMachine) &&
(szMachineName != szMachineNameBuf) )
{
NetApiBufferFree( szMachineName );
}
if ( pPolicyPDI != NULL ) {
LsaFreeMemory( pPolicyPDI );
}
if ( pPolicyDns != NULL ) {
LsaFreeMemory( pPolicyDns );
}
if ( hDC != NULL ) {
LsaClose( hDC );
}
NetpLog(( "NetpSetComputerAccountPassword: status: 0x%lx\n", NetStatus ));
NetSetuppCloseLog();
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpUpdateW32timeConfig(
IN PCSTR szW32timeJoinConfigFuncName
)
/*++
Routine Description:
Call entry point in w32time service so that it can update
its internal info after a domain join/unjoin
Arguments:
szW32timeJoinConfigFuncName - name of entry point to call
(must be W32TimeVerifyJoinConfig or W32TimeVerifyUnjoinConfig)
Return Value:
NERR_Success -- on Success
otherwise win32 error codes returned by LoadLibrary, GetProcAddress
Notes:
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
HANDLE hDll = NULL;
typedef VOID (*PW32TimeUpdateJoinConfig)( VOID );
PW32TimeUpdateJoinConfig pfnW32timeUpdateJoinConfig = NULL;
//
// Call into the time service to allow it initialize itself properly
//
hDll = LoadLibraryA( "w32Time" );
if ( hDll != NULL )
{
pfnW32timeUpdateJoinConfig =
(PW32TimeUpdateJoinConfig) GetProcAddress(hDll,
szW32timeJoinConfigFuncName);
if ( pfnW32timeUpdateJoinConfig != NULL )
{
pfnW32timeUpdateJoinConfig();
}
else
{
NetStatus = GetLastError();
NetpLog(( "NetpUpdateW32timeConfig: Failed to get proc address for %s: 0x%lx\n", szW32timeJoinConfigFuncName, NetStatus ));
}
}
else
{
NetStatus = GetLastError();
NetpLog(( "NetpUpdateW32timeConfig: Failed to load w32time: 0x%lx\n", NetStatus ));
}
if ( hDll != NULL )
{
FreeLibrary( hDll );
}
NetpLog(( "NetpUpdateW32timeConfig: 0x%lx\n", NetStatus ));
return NetStatus;
}
NET_API_STATUS
NET_API_FUNCTION
NetpUpdateAutoenrolConfig(
IN BOOL UnjoinDomain
)
/*++
Routine Description:
Call entry point in pautoenr service so that it can update
its internal info after a domain join/unjoin
Arguments:
UnjoinDomain - If TRUE, we are unjoining from a domain.
Otherwise, we are roling back from unsuccessful domain
unjoin.
Return Value:
NERR_Success -- on Success
otherwise win32 error codes returned by LoadLibrary, GetProcAddress
Notes:
--*/
{
NET_API_STATUS NetStatus = NERR_Success;
HANDLE hDll = NULL;
ULONG Flags = 0;
typedef BOOL (*PCertAutoRemove)( DWORD );
PCertAutoRemove pfnCertAutoRemove = NULL;
//
// Prepare the flags to pass to autoenrol routine
//
if ( UnjoinDomain ) {
Flags = CERT_AUTO_REMOVE_COMMIT;
} else {
Flags = CERT_AUTO_REMOVE_ROLL_BACK;
}
//
// Call into the time service to allow it initialize itself properly
//
hDll = LoadLibraryA( "pautoenr" );
if ( hDll != NULL ) {
pfnCertAutoRemove =
(PCertAutoRemove) GetProcAddress( hDll, "CertAutoRemove" );
if ( pfnCertAutoRemove != NULL ) {
if ( !pfnCertAutoRemove(Flags) ) {
NetStatus = GetLastError();
NetpLog(( "NetpUpdateAutoenrolConfig: CertAutoRemove failed 0x%lx\n",
NetStatus ));
}
} else {
NetStatus = GetLastError();
NetpLog(( "NetpUpdateAutoenrolConfig: Failed to get CertAutoRemove proc address 0x%lx\n",
NetStatus ));
}
} else {
NetStatus = GetLastError();
NetpLog(( "NetpUpdateAutoenrolConfig: Failed to load pautoenr: 0x%lx\n", NetStatus ));
}
if ( hDll != NULL ) {
FreeLibrary( hDll );
}
return NetStatus;
}