2002-12-12 23:35:55 +00:00
/*
Unix SMB / CIFS implementation .
Winbind daemon connection manager
Copyright ( C ) Tim Potter 2001
Copyright ( C ) Andrew Bartlett 2002
2003-06-30 20:45:14 +00:00
Copyright ( C ) Gerald Carter 2003
2002-12-12 23:35:55 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2006-12-21 00:43:21 +00:00
/**********************************************************************
Is this our primary domain ?
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# ifdef HAVE_KRB5
static BOOL is_our_primary_domain ( const char * domain )
{
int role = lp_server_role ( ) ;
if ( ( role = = ROLE_DOMAIN_MEMBER ) & & strequal ( lp_workgroup ( ) , domain ) ) {
return True ;
} else if ( strequal ( get_global_sam_name ( ) , domain ) ) {
return True ;
}
return False ;
}
# endif
2003-06-30 20:45:14 +00:00
/**************************************************************************
2006-08-30 05:52:31 +00:00
Find the name and IP address for a server in the realm / domain
2003-06-30 20:45:14 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-08-30 05:52:31 +00:00
static BOOL ads_dc_name ( const char * domain ,
const char * realm ,
struct in_addr * dc_ip ,
fstring srv_name )
2003-06-30 20:45:14 +00:00
{
ADS_STRUCT * ads ;
2007-01-18 09:58:57 +00:00
char * sitename ;
2006-08-30 05:52:31 +00:00
int i ;
2003-06-30 20:45:14 +00:00
2006-08-31 01:20:21 +00:00
if ( ! realm & & strequal ( domain , lp_workgroup ( ) ) ) {
2003-06-30 20:45:14 +00:00
realm = lp_realm ( ) ;
2006-08-31 01:20:21 +00:00
}
2003-06-30 20:45:14 +00:00
2007-01-18 09:58:57 +00:00
sitename = sitename_fetch ( realm ) ;
2006-08-30 05:52:31 +00:00
/* Try this 3 times then give up. */
for ( i = 0 ; i < 3 ; i + + ) {
ads = ads_init ( realm , domain , NULL ) ;
if ( ! ads ) {
SAFE_FREE ( sitename ) ;
return False ;
}
2003-06-30 20:45:14 +00:00
2006-08-30 05:52:31 +00:00
DEBUG ( 4 , ( " ads_dc_name: domain=%s \n " , domain ) ) ;
2003-06-30 20:45:14 +00:00
# ifdef HAVE_ADS
2006-08-30 05:52:31 +00:00
/* we don't need to bind, just connect */
ads - > auth . flags | = ADS_AUTH_NO_BIND ;
ads_connect ( ads ) ;
2003-06-30 20:45:14 +00:00
# endif
2006-08-30 05:52:31 +00:00
if ( ! ads - > config . realm ) {
SAFE_FREE ( sitename ) ;
ads_destroy ( & ads ) ;
return False ;
}
/* Now we've found a server, see if our sitename
2006-08-31 01:20:21 +00:00
has changed . If so , we need to re - do the DNS query
2006-08-30 05:52:31 +00:00
to ensure we only find servers in our site . */
2007-01-18 09:58:57 +00:00
if ( stored_sitename_changed ( realm , sitename ) ) {
2006-08-31 01:20:21 +00:00
SAFE_FREE ( sitename ) ;
2007-01-18 09:58:57 +00:00
sitename = sitename_fetch ( realm ) ;
2006-08-31 01:20:21 +00:00
ads_destroy ( & ads ) ;
2006-10-11 18:54:40 +00:00
/* Ensure we don't cache the DC we just connected to. */
namecache_delete ( realm , 0x1C ) ;
namecache_delete ( domain , 0x1C ) ;
2006-08-31 01:20:21 +00:00
continue ;
2006-08-30 05:52:31 +00:00
}
2006-08-31 01:20:21 +00:00
# ifdef HAVE_KRB5
2006-12-21 00:43:21 +00:00
if ( is_our_primary_domain ( domain ) & & ( ads - > config . flags & ADS_KDC ) & & ads_closest_dc ( ads ) ) {
2006-08-31 01:20:21 +00:00
/* We're going to use this KDC for this realm/domain.
If we are using sites , then force the krb5 libs
to use this KDC . */
2006-08-30 05:52:31 +00:00
2006-08-31 01:20:21 +00:00
create_local_private_krb5_conf_for_domain ( realm ,
domain ,
2007-01-17 18:25:35 +00:00
sitename ,
2006-08-31 01:20:21 +00:00
ads - > ldap_ip ) ;
}
# endif
break ;
}
2006-08-30 05:52:31 +00:00
if ( i = = 3 ) {
2006-08-30 16:02:08 +00:00
DEBUG ( 1 , ( " ads_dc_name: sitename (now \" %s \" ) keeps changing ??? \n " ,
sitename ? sitename : " " ) ) ;
2006-08-30 05:52:31 +00:00
SAFE_FREE ( sitename ) ;
2003-06-30 20:45:14 +00:00
return False ;
2004-05-06 23:16:52 +00:00
}
2003-06-30 20:45:14 +00:00
2006-08-30 05:52:31 +00:00
SAFE_FREE ( sitename ) ;
2003-06-30 20:45:14 +00:00
fstrcpy ( srv_name , ads - > config . ldap_server_name ) ;
2003-07-03 19:11:31 +00:00
strupper_m ( srv_name ) ;
2003-06-30 20:45:14 +00:00
* dc_ip = ads - > ldap_ip ;
ads_destroy ( & ads ) ;
DEBUG ( 4 , ( " ads_dc_name: using server='%s' IP=%s \n " ,
srv_name , inet_ntoa ( * dc_ip ) ) ) ;
return True ;
}
2003-06-06 14:11:14 +00:00
/****************************************************************************
2003-06-13 21:03:15 +00:00
Utility function to return the name of a DC . The name is guaranteed to be
valid since we have already done a name_status_find on it
2003-06-06 14:11:14 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-06-13 21:03:15 +00:00
2003-06-30 20:45:14 +00:00
static BOOL rpc_dc_name ( const char * domain , fstring srv_name , struct in_addr * ip_out )
2002-12-12 23:35:55 +00:00
{
2003-06-25 17:41:05 +00:00
struct ip_service * ip_list = NULL ;
struct in_addr dc_ip , exclude_ip ;
2002-12-12 23:35:55 +00:00
int count , i ;
2003-06-06 14:11:14 +00:00
NTSTATUS result ;
2002-12-12 23:35:55 +00:00
zero_ip ( & exclude_ip ) ;
/* get a list of all domain controllers */
2007-01-17 18:25:35 +00:00
if ( ! NT_STATUS_IS_OK ( get_sorted_dc_list ( domain , NULL , & ip_list , & count ,
2006-08-28 09:19:30 +00:00
False ) ) ) {
2002-12-12 23:35:55 +00:00
DEBUG ( 3 , ( " Could not look up dc's for domain %s \n " , domain ) ) ;
return False ;
}
/* Remove the entry we've already failed with (should be the PDC). */
for ( i = 0 ; i < count ; i + + ) {
2003-06-25 17:41:05 +00:00
if ( is_zero_ip ( ip_list [ i ] . ip ) )
2002-12-12 23:35:55 +00:00
continue ;
2003-06-25 17:41:05 +00:00
if ( name_status_find ( domain , 0x1c , 0x20 , ip_list [ i ] . ip , srv_name ) ) {
2003-06-06 14:11:14 +00:00
result = check_negative_conn_cache ( domain , srv_name ) ;
if ( NT_STATUS_IS_OK ( result ) ) {
2003-06-25 17:41:05 +00:00
dc_ip = ip_list [ i ] . ip ;
2003-06-06 14:11:14 +00:00
goto done ;
}
2003-06-23 19:05:23 +00:00
}
2002-12-12 23:35:55 +00:00
}
2003-06-23 19:05:23 +00:00
2002-12-12 23:35:55 +00:00
SAFE_FREE ( ip_list ) ;
2003-06-13 21:03:15 +00:00
/* No-one to talk to )-: */
return False ; /* Boo-hoo */
done :
2002-12-12 23:35:55 +00:00
/* We have the netbios name and IP address of a domain controller.
Ideally we should sent a SAMLOGON request to determine whether
the DC is alive and kicking . If we can catch a dead DC before
performing a cli_connect ( ) we can avoid a 30 - second timeout . */
2003-06-23 19:05:23 +00:00
DEBUG ( 3 , ( " rpc_dc_name: Returning DC %s (%s) for domain %s \n " , srv_name ,
2002-12-12 23:35:55 +00:00
inet_ntoa ( dc_ip ) , domain ) ) ;
* ip_out = dc_ip ;
SAFE_FREE ( ip_list ) ;
return True ;
}
2003-06-30 20:45:14 +00:00
/**********************************************************************
wrapper around ads and rpc methods of finds DC ' s
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-01-05 04:10:28 +00:00
BOOL get_dc_name ( const char * domain , const char * realm , fstring srv_name , struct in_addr * ip_out )
2003-06-30 20:45:14 +00:00
{
struct in_addr dc_ip ;
BOOL ret ;
2003-08-08 23:53:13 +00:00
BOOL our_domain = False ;
2003-06-30 20:45:14 +00:00
zero_ip ( & dc_ip ) ;
ret = False ;
2003-08-08 23:53:13 +00:00
2004-01-05 04:10:28 +00:00
if ( strequal ( lp_workgroup ( ) , domain ) | | strequal ( lp_realm ( ) , realm ) )
2003-08-08 23:53:13 +00:00
our_domain = True ;
2004-01-05 04:10:28 +00:00
/* always try to obey what the admin specified in smb.conf
( for the local domain ) */
2003-08-08 23:53:13 +00:00
2004-01-05 04:10:28 +00:00
if ( ( our_domain & & lp_security ( ) = = SEC_ADS ) | | realm ) {
ret = ads_dc_name ( domain , realm , & dc_ip , srv_name ) ;
2003-08-08 23:53:13 +00:00
}
2003-06-30 20:45:14 +00:00
if ( ! ret ) {
/* fall back on rpc methods if the ADS methods fail */
ret = rpc_dc_name ( domain , srv_name , & dc_ip ) ;
}
2003-08-08 23:53:13 +00:00
2003-06-30 20:45:14 +00:00
* ip_out = dc_ip ;
return ret ;
}