2005-06-04 03:10:26 +04:00
/*
Unix SMB / CIFS implementation .
Copyright ( C ) Rafal Szczesniak 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
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 .
*/
/*
2006-05-16 01:49:27 +04:00
a composite function for domain handling on samr pipe
2005-06-04 03:10:26 +04:00
*/
# include "includes.h"
# include "libcli/composite/composite.h"
2006-05-16 01:49:27 +04:00
# include "libnet/libnet.h"
2006-03-15 02:35:30 +03:00
# include "librpc/gen_ndr/ndr_samr_c.h"
2005-06-04 03:10:26 +04:00
static void domain_open_handler ( struct rpc_request * ) ;
2006-05-16 01:49:27 +04:00
enum domain_open_stage { DOMOPEN_CONNECT , DOMOPEN_LOOKUP , DOMOPEN_OPEN ,
DOMOPEN_CLOSE_EXISTING , DOMOPEN_RPC_CONNECT } ;
2005-06-04 03:10:26 +04:00
struct domain_open_state {
enum domain_open_stage stage ;
2006-05-16 01:49:27 +04:00
struct libnet_context * ctx ;
2005-06-04 03:10:26 +04:00
struct dcerpc_pipe * pipe ;
struct rpc_request * req ;
2006-05-16 01:49:27 +04:00
struct composite_context * rpcconn_req ;
2005-06-04 03:10:26 +04:00
struct samr_Connect connect ;
struct samr_LookupDomain lookup ;
struct samr_OpenDomain open ;
2006-05-16 01:49:27 +04:00
struct samr_Close close ;
struct libnet_RpcConnect rpcconn ;
2005-07-08 12:09:02 +04:00
struct lsa_String domain_name ;
2005-06-04 03:10:26 +04:00
uint32_t access_mask ;
2005-06-06 12:59:19 +04:00
struct policy_handle connect_handle ;
2005-06-04 03:10:26 +04:00
struct policy_handle domain_handle ;
} ;
2006-05-16 01:49:27 +04:00
/**
* Stage 0.5 ( optional ) : Connect to samr rpc pipe
*/
static void domain_open_rpc_connect ( struct composite_context * ctx )
{
struct composite_context * c ;
struct domain_open_state * s ;
c = talloc_get_type ( ctx - > async . private_data , struct composite_context ) ;
s = talloc_get_type ( c - > private_data , struct domain_open_state ) ;
c - > status = libnet_RpcConnect_recv ( ctx , s - > ctx , c , & s - > rpcconn ) ;
if ( ! composite_is_ok ( c ) ) return ;
s - > pipe = s - > rpcconn . out . dcerpc_pipe ;
/* preparing parameters for samr_Connect rpc call */
s - > connect . in . system_name = 0 ;
s - > connect . in . access_mask = s - > access_mask ;
s - > connect . out . connect_handle = & s - > connect_handle ;
/* send request */
s - > req = dcerpc_samr_Connect_send ( s - > pipe , c , & s - > connect ) ;
if ( composite_nomem ( s - > req , c ) ) return ;
/* callback handler */
s - > req - > async . callback = domain_open_handler ;
s - > req - > async . private = c ;
s - > stage = DOMOPEN_CONNECT ;
}
/**
* Stage 0.5 ( optional ) : Close existing ( in libnet context ) domain
* handle
*/
static NTSTATUS domain_open_close ( struct composite_context * c ,
struct domain_open_state * s )
{
/* receive samr_Close reply */
c - > status = dcerpc_ndr_request_recv ( s - > req ) ;
NT_STATUS_NOT_OK_RETURN ( c - > status ) ;
/* reset domain handle and associated data in libnet_context */
s - > ctx - > domain . name = NULL ;
s - > ctx - > domain . access_mask = 0 ;
ZERO_STRUCT ( s - > ctx - > domain . handle ) ;
/* preparing parameters for samr_Connect rpc call */
s - > connect . in . system_name = 0 ;
s - > connect . in . access_mask = s - > access_mask ;
s - > connect . out . connect_handle = & s - > connect_handle ;
/* send request */
s - > req = dcerpc_samr_Connect_send ( s - > pipe , c , & s - > connect ) ;
if ( s - > req = = NULL ) return NT_STATUS_NO_MEMORY ;
/* callback handler */
s - > req - > async . callback = domain_open_handler ;
s - > req - > async . private = c ;
s - > stage = DOMOPEN_CONNECT ;
return NT_STATUS_OK ;
}
2005-06-15 02:08:00 +04:00
/**
* Stage 1 : Connect to SAM server .
*/
2005-06-06 12:59:19 +04:00
static NTSTATUS domain_open_connect ( struct composite_context * c ,
struct domain_open_state * s )
{
struct samr_LookupDomain * r = & s - > lookup ;
/* receive samr_Connect reply */
c - > status = dcerpc_ndr_request_recv ( s - > req ) ;
NT_STATUS_NOT_OK_RETURN ( c - > status ) ;
/* prepare for samr_LookupDomain call */
r - > in . connect_handle = & s - > connect_handle ;
r - > in . domain_name = & s - > domain_name ;
s - > req = dcerpc_samr_LookupDomain_send ( s - > pipe , c , r ) ;
if ( s - > req = = NULL ) goto failure ;
s - > req - > async . callback = domain_open_handler ;
s - > req - > async . private = c ;
s - > stage = DOMOPEN_LOOKUP ;
return NT_STATUS_OK ;
failure :
return NT_STATUS_UNSUCCESSFUL ;
}
2005-06-15 02:08:00 +04:00
/**
* Stage 2 : Lookup domain by name .
*/
2005-06-06 12:59:19 +04:00
static NTSTATUS domain_open_lookup ( struct composite_context * c ,
struct domain_open_state * s )
{
struct samr_OpenDomain * r = & s - > open ;
2005-06-15 02:08:00 +04:00
/* receive samr_LookupDomain reply */
2005-06-06 12:59:19 +04:00
c - > status = dcerpc_ndr_request_recv ( s - > req ) ;
NT_STATUS_NOT_OK_RETURN ( c - > status ) ;
2005-06-15 02:08:00 +04:00
/* prepare for samr_OpenDomain call */
2005-06-06 12:59:19 +04:00
r - > in . connect_handle = & s - > connect_handle ;
r - > in . access_mask = SEC_FLAG_MAXIMUM_ALLOWED ;
r - > in . sid = s - > lookup . out . sid ;
r - > out . domain_handle = & s - > domain_handle ;
s - > req = dcerpc_samr_OpenDomain_send ( s - > pipe , c , r ) ;
if ( s - > req = = NULL ) goto failure ;
s - > req - > async . callback = domain_open_handler ;
s - > req - > async . private = c ;
s - > stage = DOMOPEN_OPEN ;
return NT_STATUS_OK ;
failure :
return NT_STATUS_UNSUCCESSFUL ;
}
2005-06-15 02:08:00 +04:00
/*
* Stage 3 : Open domain .
*/
2005-06-06 12:59:19 +04:00
static NTSTATUS domain_open_open ( struct composite_context * c ,
struct domain_open_state * s )
{
2005-06-15 02:08:00 +04:00
/* receive samr_OpenDomain reply */
2005-06-06 12:59:19 +04:00
c - > status = dcerpc_ndr_request_recv ( s - > req ) ;
NT_STATUS_NOT_OK_RETURN ( c - > status ) ;
2005-09-26 15:47:55 +04:00
c - > state = COMPOSITE_STATE_DONE ;
2005-06-06 12:59:19 +04:00
return NT_STATUS_OK ;
}
2005-06-15 02:08:00 +04:00
/**
* Event handler for asynchronous request . Handles transition through
* intermediate stages of the call .
*
* @ param req rpc call context
*/
2005-06-04 03:10:26 +04:00
static void domain_open_handler ( struct rpc_request * req )
{
struct composite_context * c = req - > async . private ;
2005-09-26 15:47:55 +04:00
struct domain_open_state * s = talloc_get_type ( c - > private_data , struct domain_open_state ) ;
2005-06-04 03:10:26 +04:00
/* Stages of the call */
switch ( s - > stage ) {
case DOMOPEN_CONNECT :
2005-06-06 12:59:19 +04:00
c - > status = domain_open_connect ( c , s ) ;
2005-06-04 03:10:26 +04:00
break ;
case DOMOPEN_LOOKUP :
2005-06-06 12:59:19 +04:00
c - > status = domain_open_lookup ( c , s ) ;
2005-06-04 03:10:26 +04:00
break ;
case DOMOPEN_OPEN :
2005-06-06 12:59:19 +04:00
c - > status = domain_open_open ( c , s ) ;
2005-06-04 03:10:26 +04:00
break ;
2006-05-16 01:49:27 +04:00
case DOMOPEN_CLOSE_EXISTING :
c - > status = domain_open_close ( c , s ) ;
break ;
case DOMOPEN_RPC_CONNECT :
/* this state shouldn't be handled here */
c - > status = NT_STATUS_UNSUCCESSFUL ;
break ;
2005-06-04 03:10:26 +04:00
}
if ( ! NT_STATUS_IS_OK ( c - > status ) ) {
2005-09-26 15:47:55 +04:00
c - > state = COMPOSITE_STATE_ERROR ;
2005-06-04 03:10:26 +04:00
}
2006-05-16 01:49:27 +04:00
if ( c - > state = = COMPOSITE_STATE_DONE ) {
composite_done ( c ) ;
}
2005-06-04 03:10:26 +04:00
}
2005-06-15 02:08:00 +04:00
/**
2006-06-28 02:16:36 +04:00
* Sends asynchronous DomainOpen request
2005-06-15 02:08:00 +04:00
*
2006-05-16 01:49:27 +04:00
* @ param ctx initialised libnet context
2005-06-15 02:08:00 +04:00
* @ param io arguments and results of the call
2006-05-16 01:49:27 +04:00
* @ param monitor pointer to monitor function that is passed monitor message
2005-06-15 02:08:00 +04:00
*/
2006-05-16 01:49:27 +04:00
struct composite_context * libnet_DomainOpen_send ( struct libnet_context * ctx ,
struct libnet_DomainOpen * io ,
void ( * monitor ) ( struct monitor_msg * ) )
2005-06-04 03:10:26 +04:00
{
struct composite_context * c ;
struct domain_open_state * s ;
2006-05-16 01:49:27 +04:00
c = talloc_zero ( ctx , struct composite_context ) ;
if ( c = = NULL ) return NULL ;
2005-06-04 03:10:26 +04:00
s = talloc_zero ( c , struct domain_open_state ) ;
2006-05-16 01:49:27 +04:00
if ( composite_nomem ( s , c ) ) return c ;
2005-06-04 03:10:26 +04:00
2006-05-16 01:49:27 +04:00
c - > state = COMPOSITE_STATE_IN_PROGRESS ;
c - > private_data = s ;
c - > event_ctx = ctx - > event_ctx ;
2005-06-04 03:10:26 +04:00
2006-05-16 01:49:27 +04:00
s - > ctx = ctx ;
s - > pipe = ctx - > samr_pipe ;
2005-06-06 12:59:19 +04:00
s - > access_mask = io - > in . access_mask ;
s - > domain_name . string = io - > in . domain_name ;
2006-05-16 01:49:27 +04:00
if ( ctx - > samr_pipe = = NULL ) {
s - > rpcconn . level = LIBNET_RPC_CONNECT_DC ;
s - > rpcconn . in . name = io - > in . domain_name ;
s - > rpcconn . in . dcerpc_iface = & dcerpc_table_samr ;
s - > rpcconn_req = libnet_RpcConnect_send ( ctx , c , & s - > rpcconn ) ;
if ( composite_nomem ( s - > rpcconn_req , c ) ) return c ;
s - > rpcconn_req - > async . fn = domain_open_rpc_connect ;
s - > rpcconn_req - > async . private_data = c ;
s - > stage = DOMOPEN_RPC_CONNECT ;
return c ;
}
/* libnet context's domain handle is not empty, so check out what
was opened first , before doing anything */
if ( ! policy_handle_empty ( & ctx - > domain . handle ) ) {
if ( strequal ( ctx - > domain . name , io - > in . domain_name ) & &
ctx - > domain . access_mask = = io - > in . access_mask ) {
/* this domain is already opened */
composite_done ( c ) ;
return c ;
} else {
/* another domain or access rights have been
requested - close the existing handle first */
s - > close . in . handle = & ctx - > domain . handle ;
/* send request to close domain handle */
s - > req = dcerpc_samr_Close_send ( s - > pipe , c , & s - > close ) ;
if ( composite_nomem ( s - > req , c ) ) return c ;
/* callback handler */
s - > req - > async . callback = domain_open_handler ;
s - > req - > async . private = c ;
s - > stage = DOMOPEN_CLOSE_EXISTING ;
return c ;
}
}
/* preparing parameters for samr_Connect rpc call */
2005-06-06 12:59:19 +04:00
s - > connect . in . system_name = 0 ;
s - > connect . in . access_mask = s - > access_mask ;
s - > connect . out . connect_handle = & s - > connect_handle ;
2005-06-04 03:10:26 +04:00
/* send request */
2006-05-16 01:49:27 +04:00
s - > req = dcerpc_samr_Connect_send ( s - > pipe , c , & s - > connect ) ;
if ( composite_nomem ( s - > req , c ) ) return c ;
2005-06-04 03:10:26 +04:00
/* callback handler */
s - > req - > async . callback = domain_open_handler ;
s - > req - > async . private = c ;
s - > stage = DOMOPEN_CONNECT ;
return c ;
}
2005-06-06 12:59:19 +04:00
2005-06-15 02:08:00 +04:00
/**
2006-06-28 02:16:36 +04:00
* Waits for and receives result of asynchronous DomainOpen call
2005-06-15 02:08:00 +04:00
*
2006-06-28 02:16:36 +04:00
* @ param c composite context returned by asynchronous DomainOpen call
2006-05-16 01:49:27 +04:00
* @ param ctx initialised libnet context
2005-06-15 02:08:00 +04:00
* @ param mem_ctx memory context of the call
* @ param io pointer to results ( and arguments ) of the call
* @ return nt status code of execution
*/
2006-05-16 01:49:27 +04:00
NTSTATUS libnet_DomainOpen_recv ( struct composite_context * c , struct libnet_context * ctx ,
TALLOC_CTX * mem_ctx , struct libnet_DomainOpen * io )
2005-06-06 12:59:19 +04:00
{
NTSTATUS status ;
struct domain_open_state * s ;
2005-06-15 02:08:00 +04:00
/* wait for results of sending request */
2005-06-06 12:59:19 +04:00
status = composite_wait ( c ) ;
if ( NT_STATUS_IS_OK ( status ) & & io ) {
2005-09-26 15:47:55 +04:00
s = talloc_get_type ( c - > private_data , struct domain_open_state ) ;
2005-06-06 12:59:19 +04:00
io - > out . domain_handle = s - > domain_handle ;
2006-05-16 01:49:27 +04:00
/* store the resulting handle and related data for use by other
libnet functions */
ctx - > domain . handle = s - > domain_handle ;
ctx - > domain . name = talloc_strdup ( ctx , s - > domain_name . string ) ;
ctx - > domain . access_mask = s - > access_mask ;
2005-06-06 12:59:19 +04:00
}
talloc_free ( c ) ;
return status ;
}
2005-06-15 02:08:00 +04:00
/**
2006-06-28 02:16:36 +04:00
* Synchronous version of DomainOpen call
2005-06-15 02:08:00 +04:00
*
2006-05-16 01:49:27 +04:00
* @ param ctx initialised libnet context
2005-06-15 02:08:00 +04:00
* @ param mem_ctx memory context for the call
* @ param io arguments and results of the call
* @ return nt status code of execution
*/
2006-05-16 01:49:27 +04:00
NTSTATUS libnet_DomainOpen ( struct libnet_context * ctx ,
TALLOC_CTX * mem_ctx ,
struct libnet_DomainOpen * io )
2005-06-06 12:59:19 +04:00
{
2006-05-16 01:49:27 +04:00
struct composite_context * c = libnet_DomainOpen_send ( ctx , io , NULL ) ;
return libnet_DomainOpen_recv ( c , ctx , mem_ctx , io ) ;
2005-06-06 12:59:19 +04:00
}