2009-08-26 11:33:16 +02:00
/*
Unix SMB / CIFS implementation .
async dsgetdcname
Copyright ( C ) Volker Lendecke 2009
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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "winbindd.h"
2014-05-08 12:17:32 +12:00
# include "librpc/gen_ndr/ndr_winbind_c.h"
2016-03-17 14:27:32 +01:00
# include "librpc/gen_ndr/ndr_netlogon.h"
2018-10-18 21:53:36 +02:00
# include "lib/gencache.h"
2009-08-26 11:33:16 +02:00
struct wb_dsgetdcname_state {
struct netr_DsRGetDCNameInfo * dcinfo ;
} ;
static void wb_dsgetdcname_done ( struct tevent_req * subreq ) ;
struct tevent_req * wb_dsgetdcname_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
const char * domain_name ,
const struct GUID * domain_guid ,
const char * site_name ,
uint32_t flags )
{
struct tevent_req * req , * subreq ;
struct wb_dsgetdcname_state * state ;
2018-02-14 13:24:54 +01:00
struct dcerpc_binding_handle * child_binding_handle = NULL ;
2009-08-26 11:33:16 +02:00
struct GUID guid ;
struct GUID * guid_ptr = NULL ;
req = tevent_req_create ( mem_ctx , & state , struct wb_dsgetdcname_state ) ;
if ( req = = NULL ) {
return NULL ;
}
2014-06-30 14:23:58 +12:00
if ( strequal ( domain_name , " BUILTIN " ) ) {
2009-08-26 11:33:16 +02:00
/*
2014-06-30 14:23:58 +12:00
* This makes no sense
2009-08-26 11:33:16 +02:00
*/
tevent_req_nterror ( req , NT_STATUS_DOMAIN_CONTROLLER_NOT_FOUND ) ;
return tevent_req_post ( req , ev ) ;
}
2014-06-30 14:23:58 +12:00
if ( strequal ( domain_name , get_global_sam_name ( ) ) ) {
int role = lp_server_role ( ) ;
if ( role ! = ROLE_ACTIVE_DIRECTORY_DC ) {
/*
* Two options here : Give back our own address , or say there ' s
* nobody around . Right now opting for the latter , one measure
* to prevent the loopback connects . This might change if
* needed .
*/
tevent_req_nterror ( req , NT_STATUS_DOMAIN_CONTROLLER_NOT_FOUND ) ;
return tevent_req_post ( req , ev ) ;
}
}
2009-08-26 11:33:16 +02:00
if ( IS_DC ) {
/*
* We have to figure out the DC ourselves
*/
2018-02-14 13:24:54 +01:00
child_binding_handle = locator_child_handle ( ) ;
2009-08-26 11:33:16 +02:00
} else {
struct winbindd_domain * domain = find_our_domain ( ) ;
2018-02-14 13:24:54 +01:00
child_binding_handle = dom_child_handle ( domain ) ;
2009-08-26 11:33:16 +02:00
}
if ( domain_guid ! = NULL ) {
/* work around a const issue in rpccli_ autogenerated code */
guid = * domain_guid ;
guid_ptr = & guid ;
}
2010-08-12 15:13:54 +02:00
subreq = dcerpc_wbint_DsGetDcName_send (
2018-02-14 13:24:54 +01:00
state , ev , child_binding_handle , domain_name , guid_ptr , site_name ,
2009-08-26 11:33:16 +02:00
flags , & state - > dcinfo ) ;
if ( tevent_req_nomem ( subreq , req ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_set_callback ( subreq , wb_dsgetdcname_done , req ) ;
return req ;
}
static void wb_dsgetdcname_done ( struct tevent_req * subreq )
{
struct tevent_req * req = tevent_req_callback_data (
subreq , struct tevent_req ) ;
struct wb_dsgetdcname_state * state = tevent_req_data (
req , struct wb_dsgetdcname_state ) ;
NTSTATUS status , result ;
2010-08-12 15:13:54 +02:00
status = dcerpc_wbint_DsGetDcName_recv ( subreq , state , & result ) ;
2009-08-26 11:33:16 +02:00
TALLOC_FREE ( subreq ) ;
2010-10-18 10:10:43 +02:00
if ( any_nt_status_not_ok ( status , result , & status ) ) {
2009-08-26 11:33:16 +02:00
tevent_req_nterror ( req , status ) ;
return ;
}
tevent_req_done ( req ) ;
}
NTSTATUS wb_dsgetdcname_recv ( struct tevent_req * req , TALLOC_CTX * mem_ctx ,
struct netr_DsRGetDCNameInfo * * pdcinfo )
{
struct wb_dsgetdcname_state * state = tevent_req_data (
req , struct wb_dsgetdcname_state ) ;
NTSTATUS status ;
if ( tevent_req_is_nterror ( req , & status ) ) {
return status ;
}
* pdcinfo = talloc_move ( mem_ctx , & state - > dcinfo ) ;
return NT_STATUS_OK ;
}
2016-03-17 14:27:32 +01:00
NTSTATUS wb_dsgetdcname_gencache_set ( const char * domname ,
struct netr_DsRGetDCNameInfo * dcinfo )
{
DATA_BLOB blob ;
enum ndr_err_code ndr_err ;
char * key ;
bool ok ;
key = talloc_asprintf_strupper_m ( talloc_tos ( ) , " DCINFO/%s " , domname ) ;
if ( key = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
if ( DEBUGLEVEL > = 10 ) {
NDR_PRINT_DEBUG ( netr_DsRGetDCNameInfo , dcinfo ) ;
}
ndr_err = ndr_push_struct_blob (
& blob , key , dcinfo ,
( ndr_push_flags_fn_t ) ndr_push_netr_DsRGetDCNameInfo ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
NTSTATUS status = ndr_map_error2ntstatus ( ndr_err ) ;
DBG_WARNING ( " ndr_push_struct_blob failed: %s \n " ,
ndr_errstr ( ndr_err ) ) ;
TALLOC_FREE ( key ) ;
return status ;
}
2017-08-02 17:52:40 +02:00
ok = gencache_set_data_blob ( key , blob , time ( NULL ) + 3600 ) ;
2016-03-17 14:27:32 +01:00
if ( ! ok ) {
DBG_WARNING ( " gencache_set_data_blob for key %s failed \n " , key ) ;
TALLOC_FREE ( key ) ;
return NT_STATUS_UNSUCCESSFUL ;
}
TALLOC_FREE ( key ) ;
return NT_STATUS_OK ;
}
struct dcinfo_parser_state {
NTSTATUS status ;
TALLOC_CTX * mem_ctx ;
struct netr_DsRGetDCNameInfo * dcinfo ;
} ;
2018-10-13 13:41:59 +02:00
static void dcinfo_parser ( const struct gencache_timeout * timeout ,
DATA_BLOB blob ,
void * private_data )
2016-03-17 14:27:32 +01:00
{
struct dcinfo_parser_state * state = private_data ;
enum ndr_err_code ndr_err ;
2018-10-13 13:41:59 +02:00
if ( gencache_timeout_expired ( timeout ) ) {
2016-04-26 08:43:26 +02:00
return ;
}
2016-03-17 14:27:32 +01:00
state - > dcinfo = talloc ( state - > mem_ctx , struct netr_DsRGetDCNameInfo ) ;
if ( state - > dcinfo = = NULL ) {
state - > status = NT_STATUS_NO_MEMORY ;
return ;
}
ndr_err = ndr_pull_struct_blob_all (
& blob , state - > dcinfo , state - > dcinfo ,
( ndr_pull_flags_fn_t ) ndr_pull_netr_DsRGetDCNameInfo ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
DBG_ERR ( " ndr_pull_struct_blob failed \n " ) ;
state - > status = ndr_map_error2ntstatus ( ndr_err ) ;
2018-10-15 08:32:33 +02:00
TALLOC_FREE ( state - > dcinfo ) ;
2016-03-17 14:27:32 +01:00
return ;
}
state - > status = NT_STATUS_OK ;
}
NTSTATUS wb_dsgetdcname_gencache_get ( TALLOC_CTX * mem_ctx ,
const char * domname ,
struct netr_DsRGetDCNameInfo * * dcinfo )
{
struct dcinfo_parser_state state ;
char * key ;
bool ok ;
key = talloc_asprintf_strupper_m ( mem_ctx , " DCINFO/%s " , domname ) ;
if ( key = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
state = ( struct dcinfo_parser_state ) {
. status = NT_STATUS_DOMAIN_CONTROLLER_NOT_FOUND ,
. mem_ctx = mem_ctx ,
} ;
ok = gencache_parse ( key , dcinfo_parser , & state ) ;
TALLOC_FREE ( key ) ;
if ( ! ok ) {
return NT_STATUS_DOMAIN_CONTROLLER_NOT_FOUND ;
}
if ( ! NT_STATUS_IS_OK ( state . status ) ) {
return state . status ;
}
if ( DEBUGLEVEL > = 10 ) {
NDR_PRINT_DEBUG ( netr_DsRGetDCNameInfo , state . dcinfo ) ;
}
* dcinfo = state . dcinfo ;
return NT_STATUS_OK ;
}