2005-11-06 02:46:57 +03:00
/*
Unix SMB / CIFS implementation .
Get a struct wb_dom_info for a domain using DNS , netbios , possibly cldap
etc .
Copyright ( C ) Volker Lendecke 2005
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
2007-07-10 06:07:03 +04:00
the Free Software Foundation ; either version 3 of the License , or
2005-11-06 02:46:57 +03:00
( 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
2007-07-10 06:07:03 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-11-06 02:46:57 +03:00
*/
# include "includes.h"
# include "libcli/composite/composite.h"
2006-03-07 14:07:23 +03:00
# include "libcli/resolve/resolve.h"
2006-04-02 16:02:01 +04:00
# include "libcli/security/security.h"
2005-11-06 02:46:57 +03:00
# include "winbind/wb_server.h"
# include "smbd/service_task.h"
2010-09-14 11:48:52 +04:00
# include "libcli/finddc.h"
2012-02-24 18:58:40 +04:00
# include "lib/socket/netif.h"
# include "param/param.h"
2005-11-06 02:46:57 +03:00
struct get_dom_info_state {
struct composite_context * ctx ;
struct wb_dom_info * info ;
} ;
2010-09-13 10:37:10 +04:00
static void get_dom_info_recv_addrs ( struct tevent_req * req ) ;
2005-11-06 02:46:57 +03:00
struct composite_context * wb_get_dom_info_send ( TALLOC_CTX * mem_ctx ,
struct wbsrv_service * service ,
const char * domain_name ,
2010-09-14 11:36:23 +04:00
const char * dns_domain_name ,
2005-11-06 02:46:57 +03:00
const struct dom_sid * sid )
{
2010-09-13 10:37:10 +04:00
struct composite_context * result ;
struct tevent_req * req ;
2005-11-06 02:46:57 +03:00
struct get_dom_info_state * state ;
2007-07-16 15:27:29 +04:00
struct dom_sid * dom_sid ;
2010-09-14 11:36:23 +04:00
struct finddcs finddcs_io ;
2011-01-11 22:38:27 +03:00
DEBUG ( 5 , ( " wb_get_dom_info_send called \n " ) ) ;
2007-04-30 01:40:48 +04:00
result = composite_create ( mem_ctx , service - > task - > event_ctx ) ;
2005-11-06 02:46:57 +03:00
if ( result = = NULL ) goto failed ;
state = talloc ( result , struct get_dom_info_state ) ;
if ( state = = NULL ) goto failed ;
state - > ctx = result ;
result - > private_data = state ;
state - > info = talloc_zero ( state , struct wb_dom_info ) ;
if ( state - > info = = NULL ) goto failed ;
2007-07-16 15:27:29 +04:00
state - > info - > name = talloc_strdup ( state - > info , domain_name ) ;
if ( state - > info - > name = = NULL ) goto failed ;
state - > info - > sid = dom_sid_dup ( state - > info , sid ) ;
if ( state - > info - > sid = = NULL ) goto failed ;
2013-06-15 17:01:44 +04:00
if ( dom_sid_equal ( sid , & global_sid_Builtin ) | |
( ( lpcfg_server_role ( service - > task - > lp_ctx ) ! = ROLE_DOMAIN_MEMBER ) & &
2012-02-24 18:58:40 +04:00
dom_sid_equal ( sid , service - > primary_sid ) & &
2013-06-15 17:01:44 +04:00
service - > sec_channel_type ! = SEC_CHAN_RODC ) ) {
2012-02-24 18:58:40 +04:00
struct interface * ifaces = NULL ;
load_interface_list ( state , service - > task - > lp_ctx , & ifaces ) ;
state - > info - > dc = talloc ( state - > info , struct nbt_dc_name ) ;
state - > info - > dc - > address = talloc_strdup ( state - > info - > dc ,
iface_list_n_ip ( ifaces , 0 ) ) ;
state - > info - > dc - > name = talloc_strdup ( state - > info - > dc ,
lpcfg_netbios_name ( service - > task - > lp_ctx ) ) ;
composite_done ( state - > ctx ) ;
return result ;
}
2007-07-16 15:27:29 +04:00
dom_sid = dom_sid_dup ( mem_ctx , sid ) ;
if ( dom_sid = = NULL ) goto failed ;
2005-11-06 02:46:57 +03:00
2010-09-14 11:36:23 +04:00
ZERO_STRUCT ( finddcs_io ) ;
2010-09-14 14:10:51 +04:00
finddcs_io . in . domain_name = dns_domain_name ;
2010-09-14 11:36:23 +04:00
finddcs_io . in . domain_sid = dom_sid ;
finddcs_io . in . minimum_dc_flags = NBT_SERVER_LDAP | NBT_SERVER_DS ;
if ( service - > sec_channel_type = = SEC_CHAN_RODC ) {
finddcs_io . in . minimum_dc_flags | = NBT_SERVER_WRITABLE ;
}
req = finddcs_cldap_send ( mem_ctx , & finddcs_io ,
lpcfg_resolve_context ( service - > task - > lp_ctx ) ,
service - > task - > event_ctx ) ;
2010-09-13 10:37:10 +04:00
if ( req = = NULL ) goto failed ;
tevent_req_set_callback ( req , get_dom_info_recv_addrs , state ) ;
2005-11-06 02:46:57 +03:00
return result ;
failed :
talloc_free ( result ) ;
return NULL ;
}
2010-09-13 10:37:10 +04:00
static void get_dom_info_recv_addrs ( struct tevent_req * req )
2005-11-06 02:46:57 +03:00
{
2010-09-13 10:37:10 +04:00
struct get_dom_info_state * state = tevent_req_callback_data ( req , struct get_dom_info_state ) ;
2010-09-14 11:36:23 +04:00
struct finddcs finddcs_io ;
2010-09-16 15:17:54 +04:00
state - > info - > dc = talloc ( state - > info , struct nbt_dc_name ) ;
2005-11-06 02:46:57 +03:00
2010-09-14 11:36:23 +04:00
state - > ctx - > status = finddcs_cldap_recv ( req , state - > info , & finddcs_io ) ;
2005-11-06 02:46:57 +03:00
if ( ! composite_is_ok ( state - > ctx ) ) return ;
2010-09-14 11:36:23 +04:00
if ( finddcs_io . out . netlogon . ntver ! = NETLOGON_NT_VERSION_5EX ) {
/* the finddcs code should have mapped the response to
the type we want */
DEBUG ( 0 , ( __location__ " : unexpected ntver 0x%08x in finddcs response \n " ,
finddcs_io . out . netlogon . ntver ) ) ;
state - > ctx - > status = NT_STATUS_UNEXPECTED_NETWORK_ERROR ;
if ( ! composite_is_ok ( state - > ctx ) ) return ;
}
state - > info - > dc - > address = finddcs_io . out . address ;
state - > info - > dc - > name = finddcs_io . out . netlogon . data . nt5_ex . pdc_dns_name ;
2005-11-06 02:46:57 +03:00
composite_done ( state - > ctx ) ;
}
NTSTATUS wb_get_dom_info_recv ( struct composite_context * ctx ,
TALLOC_CTX * mem_ctx ,
struct wb_dom_info * * result )
{
NTSTATUS status = composite_wait ( ctx ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
struct get_dom_info_state * state =
talloc_get_type ( ctx - > private_data ,
struct get_dom_info_state ) ;
* result = talloc_steal ( mem_ctx , state - > info ) ;
}
talloc_free ( ctx ) ;
return status ;
}
NTSTATUS wb_get_dom_info ( TALLOC_CTX * mem_ctx ,
struct wbsrv_service * service ,
const char * domain_name ,
2010-09-14 11:36:23 +04:00
const char * dns_domain_name ,
2005-11-06 02:46:57 +03:00
const struct dom_sid * sid ,
struct wb_dom_info * * result )
{
struct composite_context * ctx =
2010-09-14 11:36:23 +04:00
wb_get_dom_info_send ( mem_ctx , service , domain_name , dns_domain_name , sid ) ;
2005-11-06 02:46:57 +03:00
return wb_get_dom_info_recv ( ctx , mem_ctx , result ) ;
}