2019-10-03 19:05:04 +03:00
/*
Unix SMB / CIFS implementation .
server side dcerpc core code
Copyright ( C ) Andrew Tridgell 2003 - 2005
Copyright ( C ) Stefan ( metze ) Metzmacher 2004 - 2005
Copyright ( C ) Samuel Cabrero < scabrero @ samba . org > 2019
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"
2019-10-03 20:38:31 +03:00
# include "librpc/rpc/dcesrv_core.h"
# include "librpc/rpc/dcesrv_core_proto.h"
2021-04-02 13:20:38 +03:00
# include "librpc/rpc/dcerpc_util.h"
2019-10-03 20:38:31 +03:00
# include "librpc/gen_ndr/auth.h"
# include "auth/gensec/gensec.h"
# include "lib/util/dlinklist.h"
# include "libcli/security/security.h"
# include "param/param.h"
# include "lib/tsocket/tsocket.h"
# include "librpc/gen_ndr/ndr_dcerpc.h"
# include "lib/util/tevent_ntstatus.h"
2021-03-27 19:17:34 +03:00
# include "system/network.h"
2019-10-03 20:38:31 +03:00
2021-02-02 17:10:38 +03:00
/**
* @ file
* @ brief DCERPC server
*/
2019-11-06 07:27:08 +03:00
2019-02-04 19:37:01 +03:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_SRV
2020-08-07 23:27:39 +03:00
# undef strcasecmp
2019-10-03 20:38:31 +03:00
static NTSTATUS dcesrv_negotiate_contexts ( struct dcesrv_call_state * call ,
const struct dcerpc_bind * b ,
struct dcerpc_ack_ctx * ack_ctx_list ) ;
/*
see if two endpoints match
*/
static bool endpoints_match ( const struct dcerpc_binding * ep1 ,
const struct dcerpc_binding * ep2 )
{
enum dcerpc_transport_t t1 ;
enum dcerpc_transport_t t2 ;
const char * e1 ;
const char * e2 ;
t1 = dcerpc_binding_get_transport ( ep1 ) ;
t2 = dcerpc_binding_get_transport ( ep2 ) ;
e1 = dcerpc_binding_get_string_option ( ep1 , " endpoint " ) ;
e2 = dcerpc_binding_get_string_option ( ep2 , " endpoint " ) ;
if ( t1 ! = t2 ) {
return false ;
}
if ( ! e1 | | ! e2 ) {
return e1 = = e2 ;
}
if ( strcasecmp ( e1 , e2 ) ! = 0 ) {
return false ;
}
return true ;
}
/*
find an endpoint in the dcesrv_context
*/
2019-10-03 20:53:32 +03:00
_PUBLIC_ NTSTATUS dcesrv_find_endpoint ( struct dcesrv_context * dce_ctx ,
2019-10-03 20:51:53 +03:00
const struct dcerpc_binding * ep_description ,
struct dcesrv_endpoint * * _out )
2019-10-03 20:38:31 +03:00
{
2019-10-03 20:51:53 +03:00
struct dcesrv_endpoint * ep = NULL ;
2019-10-03 20:38:31 +03:00
for ( ep = dce_ctx - > endpoint_list ; ep ; ep = ep - > next ) {
if ( endpoints_match ( ep - > ep_description , ep_description ) ) {
2019-10-03 20:51:53 +03:00
* _out = ep ;
return NT_STATUS_OK ;
2019-10-03 20:38:31 +03:00
}
}
2019-10-03 20:51:53 +03:00
return NT_STATUS_NOT_FOUND ;
2019-10-03 20:38:31 +03:00
}
/*
find a registered context_id from a bind or alter_context
*/
static struct dcesrv_connection_context * dcesrv_find_context ( struct dcesrv_connection * conn ,
uint16_t context_id )
{
struct dcesrv_connection_context * c ;
for ( c = conn - > contexts ; c ; c = c - > next ) {
if ( c - > context_id = = context_id ) return c ;
}
return NULL ;
}
/*
find the interface operations on any endpoint with this binding
*/
static const struct dcesrv_interface * find_interface_by_binding ( struct dcesrv_context * dce_ctx ,
struct dcerpc_binding * binding ,
const struct dcesrv_interface * iface )
{
struct dcesrv_endpoint * ep ;
for ( ep = dce_ctx - > endpoint_list ; ep ; ep = ep - > next ) {
if ( endpoints_match ( ep - > ep_description , binding ) ) {
2021-01-27 16:00:06 +03:00
const struct dcesrv_interface * ret = NULL ;
ret = find_interface_by_syntax_id (
ep , & iface - > syntax_id ) ;
if ( ret ! = NULL ) {
return ret ;
2019-10-03 20:38:31 +03:00
}
}
}
return NULL ;
}
/*
find the interface operations on an endpoint by uuid
*/
2021-01-27 15:51:33 +03:00
_PUBLIC_ const struct dcesrv_interface * find_interface_by_syntax_id (
const struct dcesrv_endpoint * endpoint ,
const struct ndr_syntax_id * interface )
2019-10-03 20:38:31 +03:00
{
struct dcesrv_if_list * ifl ;
for ( ifl = endpoint - > interface_list ; ifl ; ifl = ifl - > next ) {
2021-01-27 15:54:25 +03:00
if ( ndr_syntax_id_equal ( & ifl - > iface - > syntax_id , interface ) ) {
2019-10-01 17:48:01 +03:00
return ifl - > iface ;
2019-10-03 20:38:31 +03:00
}
}
return NULL ;
}
/*
2021-01-20 17:00:08 +03:00
find the earlier parts of a fragmented call awaiting reassembly
2019-10-03 20:38:31 +03:00
*/
static struct dcesrv_call_state * dcesrv_find_fragmented_call ( struct dcesrv_connection * dce_conn , uint32_t call_id )
{
struct dcesrv_call_state * c ;
for ( c = dce_conn - > incoming_fragmented_call_list ; c ; c = c - > next ) {
if ( c - > pkt . call_id = = call_id ) {
return c ;
}
}
return NULL ;
}
/*
register an interface on an endpoint
An endpoint is one unix domain socket ( for ncalrpc ) , one TCP port
( for ncacn_ip_tcp ) or one ( forwarded ) named pipe ( for ncacn_np ) .
Each endpoint can have many interfaces such as netlogon , lsa or
samr . Some have essentially the full set .
This is driven from the set of interfaces listed in each IDL file
via the PIDL generated * __op_init_server ( ) functions .
*/
_PUBLIC_ NTSTATUS dcesrv_interface_register ( struct dcesrv_context * dce_ctx ,
const char * ep_name ,
const char * ncacn_np_secondary_endpoint ,
const struct dcesrv_interface * iface ,
const struct security_descriptor * sd )
2021-08-19 13:52:04 +03:00
{
struct dcerpc_binding * binding = NULL ;
struct dcerpc_binding * binding2 = NULL ;
NTSTATUS ret ;
ret = dcerpc_parse_binding ( dce_ctx , ep_name , & binding ) ;
if ( NT_STATUS_IS_ERR ( ret ) ) {
DBG_ERR ( " Trouble parsing binding string '%s' \n " , ep_name ) ;
goto out ;
}
if ( ncacn_np_secondary_endpoint ! = NULL ) {
ret = dcerpc_parse_binding ( dce_ctx ,
ncacn_np_secondary_endpoint ,
& binding2 ) ;
if ( NT_STATUS_IS_ERR ( ret ) ) {
DBG_ERR ( " Trouble parsing 2nd binding string '%s' \n " ,
ncacn_np_secondary_endpoint ) ;
goto out ;
}
}
ret = dcesrv_interface_register_b ( dce_ctx ,
binding ,
binding2 ,
iface ,
sd ) ;
out :
TALLOC_FREE ( binding ) ;
TALLOC_FREE ( binding2 ) ;
return ret ;
}
_PUBLIC_ NTSTATUS dcesrv_interface_register_b ( struct dcesrv_context * dce_ctx ,
struct dcerpc_binding * binding ,
struct dcerpc_binding * binding2 ,
const struct dcesrv_interface * iface ,
const struct security_descriptor * sd )
2019-10-03 20:38:31 +03:00
{
struct dcesrv_endpoint * ep ;
struct dcesrv_if_list * ifl ;
bool add_ep = false ;
NTSTATUS status ;
enum dcerpc_transport_t transport ;
char * ep_string = NULL ;
bool use_single_process = true ;
const char * ep_process_string ;
/*
* If we are not using handles , there is no need for force
* this service into using a single process .
*
* However , due to the way we listen for RPC packets , we can
* only do this if we have a single service per pipe or TCP
* port , so we still force a single combined process for
* ncalrpc .
*/
if ( iface - > flags & DCESRV_INTERFACE_FLAGS_HANDLES_NOT_USED ) {
use_single_process = false ;
}
transport = dcerpc_binding_get_transport ( binding ) ;
if ( transport = = NCACN_IP_TCP ) {
int port ;
/*
* First check if there is already a port specified , eg
* for epmapper on ncacn_ip_tcp : [ 135 ]
*/
const char * endpoint
= dcerpc_binding_get_string_option ( binding ,
" endpoint " ) ;
if ( endpoint = = NULL ) {
port = lpcfg_parm_int ( dce_ctx - > lp_ctx , NULL ,
" rpc server port " , iface - > name , 0 ) ;
/*
* For RPC services that are not set to use a single
* process , we do not default to using the ' rpc server
* port ' because that would cause a double - bind on
* that port .
*/
if ( port = = 0 & & ! use_single_process ) {
port = lpcfg_rpc_server_port ( dce_ctx - > lp_ctx ) ;
}
if ( port ! = 0 ) {
2021-01-10 15:07:54 +03:00
char port_str [ 6 ] ;
2019-10-03 20:38:31 +03:00
snprintf ( port_str , sizeof ( port_str ) , " %u " , port ) ;
status = dcerpc_binding_set_string_option ( binding ,
" endpoint " ,
port_str ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
}
}
}
2021-08-19 13:52:04 +03:00
if ( transport = = NCACN_NP & & binding2 ! = NULL ) {
2019-10-03 20:38:31 +03:00
enum dcerpc_transport_t transport2 ;
transport2 = dcerpc_binding_get_transport ( binding2 ) ;
SMB_ASSERT ( transport2 = = transport ) ;
}
/* see if the interface is already registered on the endpoint */
if ( find_interface_by_binding ( dce_ctx , binding , iface ) ! = NULL ) {
2021-08-19 13:52:04 +03:00
char * binding_string = dcerpc_binding_string ( dce_ctx , binding ) ;
DBG_ERR ( " Interface '%s' already registered on endpoint '%s' \n " ,
iface - > name , binding_string ) ;
TALLOC_FREE ( binding_string ) ;
2019-10-03 20:38:31 +03:00
return NT_STATUS_OBJECT_NAME_COLLISION ;
}
/* check if this endpoint exists
*/
2019-10-03 20:51:53 +03:00
status = dcesrv_find_endpoint ( dce_ctx , binding , & ep ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
2019-10-03 20:38:31 +03:00
/*
* We want a new port on ncacn_ip_tcp for NETLOGON , so
* it can be multi - process . Other processes can also
* listen on distinct ports , if they have one forced
* in the code above with eg ' rpc server port : drsuapi = 1027 '
*
* If we have mulitiple endpoints on port 0 , they each
* get an epemeral port ( currently by walking up from
* 1024 ) .
*
* Because one endpoint can only have one process
* model , we add a new IP_TCP endpoint for each model .
*
* This works in conjunction with the forced overwrite
* of ep - > use_single_process below .
*/
if ( ep - > use_single_process ! = use_single_process
& & transport = = NCACN_IP_TCP ) {
add_ep = true ;
}
}
2019-10-03 20:51:53 +03:00
if ( NT_STATUS_EQUAL ( status , NT_STATUS_NOT_FOUND ) | | add_ep ) {
2019-10-03 20:38:31 +03:00
ep = talloc_zero ( dce_ctx , struct dcesrv_endpoint ) ;
if ( ! ep ) {
return NT_STATUS_NO_MEMORY ;
}
2021-08-19 13:52:04 +03:00
ep - > ep_description = dcerpc_binding_dup ( ep , binding ) ;
if ( transport = = NCACN_NP & & binding2 ! = NULL ) {
ep - > ep_2nd_description =
dcerpc_binding_dup ( ep , binding2 ) ;
}
2019-10-03 20:38:31 +03:00
add_ep = true ;
/* add mgmt interface */
ifl = talloc_zero ( ep , struct dcesrv_if_list ) ;
if ( ! ifl ) {
2021-01-10 15:12:26 +03:00
TALLOC_FREE ( ep ) ;
2019-10-03 20:38:31 +03:00
return NT_STATUS_NO_MEMORY ;
}
2019-10-01 17:48:01 +03:00
ifl - > iface = talloc_memdup ( ifl ,
dcesrv_get_mgmt_interface ( ) ,
sizeof ( struct dcesrv_interface ) ) ;
if ( ifl - > iface = = NULL ) {
talloc_free ( ep ) ;
return NT_STATUS_NO_MEMORY ;
}
2019-10-03 20:38:31 +03:00
DLIST_ADD ( ep - > interface_list , ifl ) ;
2019-10-03 20:51:53 +03:00
} else if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_NOTICE ( " Failed to find endpoint: %s \n " , nt_errstr ( status ) ) ;
return status ;
2019-10-03 20:38:31 +03:00
}
/*
* By default don ' t force into a single process , but if any
* interface on this endpoint on this service uses handles
* ( most do ) , then we must force into single process mode
*
* By overwriting this each time a new interface is added to
* this endpoint , we end up with the most restrictive setting .
*/
if ( use_single_process ) {
ep - > use_single_process = true ;
}
/* talloc a new interface list element */
ifl = talloc_zero ( ep , struct dcesrv_if_list ) ;
if ( ! ifl ) {
return NT_STATUS_NO_MEMORY ;
}
/* copy the given interface struct to the one on the endpoints interface list */
2019-10-01 17:48:01 +03:00
ifl - > iface = talloc_memdup ( ifl ,
iface ,
sizeof ( struct dcesrv_interface ) ) ;
if ( ifl - > iface = = NULL ) {
talloc_free ( ep ) ;
return NT_STATUS_NO_MEMORY ;
}
2019-10-03 20:38:31 +03:00
/* if we have a security descriptor given,
* we should see if we can set it up on the endpoint
*/
if ( sd ! = NULL ) {
/* if there's currently no security descriptor given on the endpoint
* we try to set it
*/
if ( ep - > sd = = NULL ) {
ep - > sd = security_descriptor_copy ( ep , sd ) ;
}
/* if now there's no security descriptor given on the endpoint
* something goes wrong , either we failed to copy the security descriptor
* or there was already one on the endpoint
*/
if ( ep - > sd ! = NULL ) {
2021-08-19 13:52:04 +03:00
char * binding_string =
dcerpc_binding_string ( dce_ctx , binding ) ;
DBG_ERR ( " Interface '%s' failed to setup a security "
" descriptor on endpoint '%s' \n " ,
iface - > name , binding_string ) ;
TALLOC_FREE ( binding_string ) ;
2019-10-03 20:38:31 +03:00
if ( add_ep ) free ( ep ) ;
free ( ifl ) ;
return NT_STATUS_OBJECT_NAME_COLLISION ;
}
}
/* finally add the interface on the endpoint */
DLIST_ADD ( ep - > interface_list , ifl ) ;
/* if it's a new endpoint add it to the dcesrv_context */
if ( add_ep ) {
DLIST_ADD ( dce_ctx - > endpoint_list , ep ) ;
}
/* Re-get the string as we may have set a port */
ep_string = dcerpc_binding_string ( dce_ctx , ep - > ep_description ) ;
if ( use_single_process ) {
ep_process_string = " single process required " ;
} else {
ep_process_string = " multi process compatible " ;
}
DBG_INFO ( " Interface '%s' registered on endpoint '%s' (%s) \n " ,
iface - > name , ep_string , ep_process_string ) ;
TALLOC_FREE ( ep_string ) ;
return NT_STATUS_OK ;
}
static NTSTATUS dcesrv_session_info_session_key ( struct dcesrv_auth * auth ,
DATA_BLOB * session_key )
{
if ( auth - > session_info = = NULL ) {
return NT_STATUS_NO_USER_SESSION_KEY ;
}
if ( auth - > session_info - > session_key . length = = 0 ) {
return NT_STATUS_NO_USER_SESSION_KEY ;
}
* session_key = auth - > session_info - > session_key ;
return NT_STATUS_OK ;
}
static NTSTATUS dcesrv_remote_session_key ( struct dcesrv_auth * auth ,
DATA_BLOB * session_key )
{
if ( auth - > auth_type ! = DCERPC_AUTH_TYPE_NONE ) {
return NT_STATUS_NO_USER_SESSION_KEY ;
}
return dcesrv_session_info_session_key ( auth , session_key ) ;
}
static NTSTATUS dcesrv_local_fixed_session_key ( struct dcesrv_auth * auth ,
DATA_BLOB * session_key )
{
return dcerpc_generic_session_key ( session_key ) ;
}
/*
* Fetch the authentication session key if available .
*
* This is the key generated by a gensec authentication .
*
*/
_PUBLIC_ NTSTATUS dcesrv_auth_session_key ( struct dcesrv_call_state * call ,
DATA_BLOB * session_key )
{
struct dcesrv_auth * auth = call - > auth_state ;
SMB_ASSERT ( auth - > auth_finished ) ;
return dcesrv_session_info_session_key ( auth , session_key ) ;
}
/*
* Fetch the transport session key if available .
* Typically this is the SMB session key
* or a fixed key for local transports .
*
* The key is always truncated to 16 bytes .
*/
_PUBLIC_ NTSTATUS dcesrv_transport_session_key ( struct dcesrv_call_state * call ,
DATA_BLOB * session_key )
{
struct dcesrv_auth * auth = call - > auth_state ;
NTSTATUS status ;
SMB_ASSERT ( auth - > auth_finished ) ;
if ( auth - > session_key_fn = = NULL ) {
return NT_STATUS_NO_USER_SESSION_KEY ;
}
status = auth - > session_key_fn ( auth , session_key ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
session_key - > length = MIN ( session_key - > length , 16 ) ;
return NT_STATUS_OK ;
}
static struct dcesrv_auth * dcesrv_auth_create ( struct dcesrv_connection * conn )
{
const struct dcesrv_endpoint * ep = conn - > endpoint ;
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport ( ep - > ep_description ) ;
struct dcesrv_auth * auth = NULL ;
auth = talloc_zero ( conn , struct dcesrv_auth ) ;
if ( auth = = NULL ) {
return NULL ;
}
switch ( transport ) {
case NCACN_NP :
auth - > session_key_fn = dcesrv_remote_session_key ;
break ;
case NCALRPC :
case NCACN_UNIX_STREAM :
auth - > session_key_fn = dcesrv_local_fixed_session_key ;
break ;
default :
/*
* All other ' s get a NULL pointer , which
* results in NT_STATUS_NO_USER_SESSION_KEY
*/
break ;
}
return auth ;
}
/*
connect to a dcerpc endpoint
*/
_PUBLIC_ NTSTATUS dcesrv_endpoint_connect ( struct dcesrv_context * dce_ctx ,
TALLOC_CTX * mem_ctx ,
const struct dcesrv_endpoint * ep ,
struct auth_session_info * session_info ,
struct tevent_context * event_ctx ,
uint32_t state_flags ,
struct dcesrv_connection * * _p )
{
struct dcesrv_auth * auth = NULL ;
2021-01-20 16:30:57 +03:00
struct dcesrv_connection * p = NULL ;
2019-10-03 20:38:31 +03:00
if ( ! session_info ) {
return NT_STATUS_ACCESS_DENIED ;
}
p = talloc_zero ( mem_ctx , struct dcesrv_connection ) ;
2021-01-20 16:30:57 +03:00
if ( p = = NULL ) {
goto nomem ;
}
2019-10-03 20:38:31 +03:00
p - > dce_ctx = dce_ctx ;
p - > endpoint = ep ;
2019-11-11 05:02:30 +03:00
p - > packet_log_dir = lpcfg_parm_string ( dce_ctx - > lp_ctx ,
NULL ,
" dcesrv " ,
" stubs directory " ) ;
2019-10-03 20:38:31 +03:00
p - > event_ctx = event_ctx ;
p - > state_flags = state_flags ;
p - > allow_bind = true ;
p - > max_recv_frag = 5840 ;
p - > max_xmit_frag = 5840 ;
p - > max_total_request_size = DCERPC_NCACN_REQUEST_DEFAULT_MAX_SIZE ;
p - > support_hdr_signing = lpcfg_parm_bool ( dce_ctx - > lp_ctx ,
NULL ,
" dcesrv " ,
" header signing " ,
true ) ;
p - > max_auth_states = lpcfg_parm_ulong ( dce_ctx - > lp_ctx ,
NULL ,
" dcesrv " ,
" max auth states " ,
2049 ) ;
auth = dcesrv_auth_create ( p ) ;
if ( auth = = NULL ) {
2021-01-20 16:30:57 +03:00
goto nomem ;
2019-10-03 20:38:31 +03:00
}
auth - > session_info = talloc_reference ( auth , session_info ) ;
if ( auth - > session_info = = NULL ) {
2021-01-20 16:30:57 +03:00
goto nomem ;
2019-10-03 20:38:31 +03:00
}
p - > default_auth_state = auth ;
/*
* For now we only support NDR32 .
*/
p - > preferred_transfer = & ndr_transfer_syntax_ndr ;
* _p = p ;
return NT_STATUS_OK ;
2021-01-20 16:30:57 +03:00
nomem :
TALLOC_FREE ( p ) ;
return NT_STATUS_NO_MEMORY ;
2019-10-03 20:38:31 +03:00
}
/*
move a call from an existing linked list to the specified list . This
prevents bugs where we forget to remove the call from a previous
list when moving it .
*/
static void dcesrv_call_set_list ( struct dcesrv_call_state * call ,
enum dcesrv_call_list list )
{
switch ( call - > list ) {
case DCESRV_LIST_NONE :
break ;
case DCESRV_LIST_CALL_LIST :
DLIST_REMOVE ( call - > conn - > call_list , call ) ;
break ;
case DCESRV_LIST_FRAGMENTED_CALL_LIST :
DLIST_REMOVE ( call - > conn - > incoming_fragmented_call_list , call ) ;
break ;
case DCESRV_LIST_PENDING_CALL_LIST :
DLIST_REMOVE ( call - > conn - > pending_call_list , call ) ;
break ;
}
call - > list = list ;
switch ( list ) {
case DCESRV_LIST_NONE :
break ;
case DCESRV_LIST_CALL_LIST :
DLIST_ADD_END ( call - > conn - > call_list , call ) ;
break ;
case DCESRV_LIST_FRAGMENTED_CALL_LIST :
DLIST_ADD_END ( call - > conn - > incoming_fragmented_call_list , call ) ;
break ;
case DCESRV_LIST_PENDING_CALL_LIST :
DLIST_ADD_END ( call - > conn - > pending_call_list , call ) ;
break ;
}
}
static void dcesrv_call_disconnect_after ( struct dcesrv_call_state * call ,
const char * reason )
{
struct dcesrv_auth * a = NULL ;
if ( call - > conn - > terminate ! = NULL ) {
return ;
}
call - > conn - > allow_bind = false ;
call - > conn - > allow_alter = false ;
call - > conn - > default_auth_state - > auth_invalid = true ;
for ( a = call - > conn - > auth_states ; a ! = NULL ; a = a - > next ) {
a - > auth_invalid = true ;
}
call - > terminate_reason = talloc_strdup ( call , reason ) ;
if ( call - > terminate_reason = = NULL ) {
call - > terminate_reason = __location__ ;
}
}
/*
return a dcerpc bind_nak
*/
static NTSTATUS dcesrv_bind_nak ( struct dcesrv_call_state * call , uint32_t reason )
{
struct ncacn_packet pkt ;
struct dcerpc_bind_nak_version version ;
struct data_blob_list_item * rep ;
NTSTATUS status ;
static const uint8_t _pad [ 3 ] = { 0 , } ;
/*
* We add the call to the pending_call_list
* in order to defer the termination .
*/
dcesrv_call_disconnect_after ( call , " dcesrv_bind_nak " ) ;
/* setup a bind_nak */
dcesrv_init_hdr ( & pkt , lpcfg_rpc_big_endian ( call - > conn - > dce_ctx - > lp_ctx ) ) ;
pkt . auth_length = 0 ;
pkt . call_id = call - > pkt . call_id ;
pkt . ptype = DCERPC_PKT_BIND_NAK ;
pkt . pfc_flags = DCERPC_PFC_FLAG_FIRST | DCERPC_PFC_FLAG_LAST ;
pkt . u . bind_nak . reject_reason = reason ;
version . rpc_vers = 5 ;
version . rpc_vers_minor = 0 ;
pkt . u . bind_nak . num_versions = 1 ;
pkt . u . bind_nak . versions = & version ;
pkt . u . bind_nak . _pad = data_blob_const ( _pad , sizeof ( _pad ) ) ;
rep = talloc_zero ( call , struct data_blob_list_item ) ;
if ( ! rep ) {
return NT_STATUS_NO_MEMORY ;
}
status = dcerpc_ncacn_push_auth ( & rep - > blob , call , & pkt , NULL ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
dcerpc_set_frag_length ( & rep - > blob , rep - > blob . length ) ;
DLIST_ADD_END ( call - > replies , rep ) ;
dcesrv_call_set_list ( call , DCESRV_LIST_CALL_LIST ) ;
if ( call - > conn - > call_list & & call - > conn - > call_list - > replies ) {
if ( call - > conn - > transport . report_output_data ) {
call - > conn - > transport . report_output_data ( call - > conn ) ;
}
}
return NT_STATUS_OK ;
}
2020-11-13 13:25:41 +03:00
static NTSTATUS _dcesrv_fault_disconnect_flags ( struct dcesrv_call_state * call ,
uint32_t fault_code ,
uint8_t extra_flags ,
const char * func ,
const char * location )
2019-10-03 20:38:31 +03:00
{
2020-11-13 13:25:41 +03:00
const char * reason = NULL ;
reason = talloc_asprintf ( call , " %s:%s: fault=%u (%s) flags=0x%x " ,
func , location ,
fault_code ,
dcerpc_errstr ( call , fault_code ) ,
extra_flags ) ;
if ( reason = = NULL ) {
reason = location ;
}
2019-10-03 20:38:31 +03:00
/*
* We add the call to the pending_call_list
* in order to defer the termination .
*/
2020-11-13 13:25:41 +03:00
dcesrv_call_disconnect_after ( call , reason ) ;
return dcesrv_fault_with_flags ( call , fault_code , extra_flags ) ;
2019-10-03 20:38:31 +03:00
}
2020-11-13 13:25:41 +03:00
# define dcesrv_fault_disconnect(call, fault_code) \
_dcesrv_fault_disconnect_flags ( call , fault_code , \
DCERPC_PFC_FLAG_DID_NOT_EXECUTE , \
__func__ , __location__ )
2020-11-13 13:27:19 +03:00
# define dcesrv_fault_disconnect0(call, fault_code) \
_dcesrv_fault_disconnect_flags ( call , fault_code , 0 , \
__func__ , __location__ )
2020-11-13 13:25:41 +03:00
2019-10-03 20:38:31 +03:00
static int dcesrv_connection_context_destructor ( struct dcesrv_connection_context * c )
{
DLIST_REMOVE ( c - > conn - > contexts , c ) ;
if ( c - > iface & & c - > iface - > unbind ) {
c - > iface - > unbind ( c , c - > iface ) ;
c - > iface = NULL ;
}
return 0 ;
}
static void dcesrv_prepare_context_auth ( struct dcesrv_call_state * dce_call )
{
struct loadparm_context * lp_ctx = dce_call - > conn - > dce_ctx - > lp_ctx ;
const struct dcesrv_endpoint * endpoint = dce_call - > conn - > endpoint ;
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport ( endpoint - > ep_description ) ;
struct dcesrv_connection_context * context = dce_call - > context ;
const struct dcesrv_interface * iface = context - > iface ;
context - > min_auth_level = DCERPC_AUTH_LEVEL_NONE ;
if ( transport = = NCALRPC ) {
context - > allow_connect = true ;
return ;
}
/*
* allow overwrite per interface
* allow dcerpc auth level connect : < interface >
*/
context - > allow_connect = lpcfg_allow_dcerpc_auth_level_connect ( lp_ctx ) ;
context - > allow_connect = lpcfg_parm_bool ( lp_ctx , NULL ,
" allow dcerpc auth level connect " ,
iface - > name ,
context - > allow_connect ) ;
}
NTSTATUS dcesrv_interface_bind_require_integrity ( struct dcesrv_connection_context * context ,
const struct dcesrv_interface * iface )
{
/*
* For connection oriented DCERPC DCERPC_AUTH_LEVEL_PACKET ( 4 )
* has the same behavior as DCERPC_AUTH_LEVEL_INTEGRITY ( 5 ) .
*/
context - > min_auth_level = DCERPC_AUTH_LEVEL_PACKET ;
return NT_STATUS_OK ;
}
NTSTATUS dcesrv_interface_bind_require_privacy ( struct dcesrv_connection_context * context ,
const struct dcesrv_interface * iface )
{
context - > min_auth_level = DCERPC_AUTH_LEVEL_PRIVACY ;
return NT_STATUS_OK ;
}
_PUBLIC_ NTSTATUS dcesrv_interface_bind_reject_connect ( struct dcesrv_connection_context * context ,
const struct dcesrv_interface * iface )
{
struct loadparm_context * lp_ctx = context - > conn - > dce_ctx - > lp_ctx ;
const struct dcesrv_endpoint * endpoint = context - > conn - > endpoint ;
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport ( endpoint - > ep_description ) ;
if ( transport = = NCALRPC ) {
context - > allow_connect = true ;
return NT_STATUS_OK ;
}
/*
* allow overwrite per interface
* allow dcerpc auth level connect : < interface >
*/
context - > allow_connect = false ;
context - > allow_connect = lpcfg_parm_bool ( lp_ctx , NULL ,
" allow dcerpc auth level connect " ,
iface - > name ,
context - > allow_connect ) ;
return NT_STATUS_OK ;
}
_PUBLIC_ NTSTATUS dcesrv_interface_bind_allow_connect ( struct dcesrv_connection_context * context ,
const struct dcesrv_interface * iface )
{
struct loadparm_context * lp_ctx = context - > conn - > dce_ctx - > lp_ctx ;
const struct dcesrv_endpoint * endpoint = context - > conn - > endpoint ;
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport ( endpoint - > ep_description ) ;
if ( transport = = NCALRPC ) {
context - > allow_connect = true ;
return NT_STATUS_OK ;
}
/*
* allow overwrite per interface
* allow dcerpc auth level connect : < interface >
*/
context - > allow_connect = true ;
context - > allow_connect = lpcfg_parm_bool ( lp_ctx , NULL ,
" allow dcerpc auth level connect " ,
iface - > name ,
context - > allow_connect ) ;
return NT_STATUS_OK ;
}
struct dcesrv_conn_auth_wait_context {
struct tevent_req * req ;
bool done ;
NTSTATUS status ;
} ;
struct dcesrv_conn_auth_wait_state {
uint8_t dummy ;
} ;
static struct tevent_req * dcesrv_conn_auth_wait_send ( TALLOC_CTX * mem_ctx ,
struct tevent_context * ev ,
void * private_data )
{
struct dcesrv_conn_auth_wait_context * auth_wait =
talloc_get_type_abort ( private_data ,
struct dcesrv_conn_auth_wait_context ) ;
struct tevent_req * req = NULL ;
struct dcesrv_conn_auth_wait_state * state = NULL ;
req = tevent_req_create ( mem_ctx , & state ,
struct dcesrv_conn_auth_wait_state ) ;
if ( req = = NULL ) {
return NULL ;
}
auth_wait - > req = req ;
tevent_req_defer_callback ( req , ev ) ;
if ( ! auth_wait - > done ) {
return req ;
}
if ( tevent_req_nterror ( req , auth_wait - > status ) ) {
return tevent_req_post ( req , ev ) ;
}
tevent_req_done ( req ) ;
return tevent_req_post ( req , ev ) ;
}
static NTSTATUS dcesrv_conn_auth_wait_recv ( struct tevent_req * req )
{
return tevent_req_simple_recv_ntstatus ( req ) ;
}
static NTSTATUS dcesrv_conn_auth_wait_setup ( struct dcesrv_connection * conn )
{
struct dcesrv_conn_auth_wait_context * auth_wait = NULL ;
if ( conn - > wait_send ! = NULL ) {
return NT_STATUS_INTERNAL_ERROR ;
}
auth_wait = talloc_zero ( conn , struct dcesrv_conn_auth_wait_context ) ;
if ( auth_wait = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
conn - > wait_private = auth_wait ;
conn - > wait_send = dcesrv_conn_auth_wait_send ;
conn - > wait_recv = dcesrv_conn_auth_wait_recv ;
return NT_STATUS_OK ;
}
static void dcesrv_conn_auth_wait_finished ( struct dcesrv_connection * conn ,
NTSTATUS status )
{
struct dcesrv_conn_auth_wait_context * auth_wait =
talloc_get_type_abort ( conn - > wait_private ,
struct dcesrv_conn_auth_wait_context ) ;
auth_wait - > done = true ;
auth_wait - > status = status ;
if ( auth_wait - > req = = NULL ) {
return ;
}
if ( tevent_req_nterror ( auth_wait - > req , status ) ) {
return ;
}
tevent_req_done ( auth_wait - > req ) ;
}
static NTSTATUS dcesrv_auth_reply ( struct dcesrv_call_state * call ) ;
static void dcesrv_bind_done ( struct tevent_req * subreq ) ;
/*
handle a bind request
*/
static NTSTATUS dcesrv_bind ( struct dcesrv_call_state * call )
{
struct dcesrv_connection * conn = call - > conn ;
2021-01-14 14:20:43 +03:00
struct dcesrv_context * dce_ctx = conn - > dce_ctx ;
2019-10-03 20:38:31 +03:00
struct ncacn_packet * pkt = & call - > ack_pkt ;
NTSTATUS status ;
uint32_t extra_flags = 0 ;
uint16_t max_req = 0 ;
uint16_t max_rep = 0 ;
struct dcerpc_binding * ep_2nd_description = NULL ;
const char * endpoint = NULL ;
struct dcesrv_auth * auth = call - > auth_state ;
struct dcerpc_ack_ctx * ack_ctx_list = NULL ;
struct dcerpc_ack_ctx * ack_features = NULL ;
struct tevent_req * subreq = NULL ;
size_t i ;
status = dcerpc_verify_ncacn_packet_header ( & call - > pkt ,
DCERPC_PKT_BIND ,
call - > pkt . u . bind . auth_info . length ,
0 , /* required flags */
DCERPC_PFC_FLAG_FIRST |
DCERPC_PFC_FLAG_LAST |
DCERPC_PFC_FLAG_SUPPORT_HEADER_SIGN |
0x08 | /* this is not defined, but should be ignored */
DCERPC_PFC_FLAG_CONC_MPX |
DCERPC_PFC_FLAG_DID_NOT_EXECUTE |
DCERPC_PFC_FLAG_MAYBE |
DCERPC_PFC_FLAG_OBJECT_UUID ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return dcesrv_bind_nak ( call ,
DCERPC_BIND_NAK_REASON_PROTOCOL_VERSION_NOT_SUPPORTED ) ;
}
/* max_recv_frag and max_xmit_frag result always in the same value! */
max_req = MIN ( call - > pkt . u . bind . max_xmit_frag ,
call - > pkt . u . bind . max_recv_frag ) ;
/*
* The values are between 2048 and 5840 tested against Windows 2012 R2
* via ncacn_ip_tcp on port 135.
*/
max_req = MAX ( 2048 , max_req ) ;
2021-01-14 14:17:53 +03:00
max_rep = MIN ( max_req , conn - > max_recv_frag ) ;
2019-10-03 20:38:31 +03:00
/* They are truncated to an 8 byte boundary. */
max_rep & = 0xFFF8 ;
/* max_recv_frag and max_xmit_frag result always in the same value! */
2021-01-14 14:17:53 +03:00
conn - > max_recv_frag = max_rep ;
conn - > max_xmit_frag = max_rep ;
2019-10-03 20:38:31 +03:00
2021-02-02 17:07:35 +03:00
status = dce_ctx - > callbacks - > assoc_group . find (
call , dce_ctx - > callbacks - > assoc_group . private_data ) ;
2019-10-03 20:38:31 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_NOTICE ( " Failed to find assoc_group 0x%08x: %s \n " ,
call - > pkt . u . bind . assoc_group_id , nt_errstr ( status ) ) ;
return dcesrv_bind_nak ( call , 0 ) ;
}
if ( call - > pkt . u . bind . num_contexts < 1 ) {
return dcesrv_bind_nak ( call , 0 ) ;
}
ack_ctx_list = talloc_zero_array ( call , struct dcerpc_ack_ctx ,
call - > pkt . u . bind . num_contexts ) ;
if ( ack_ctx_list = = NULL ) {
return dcesrv_bind_nak ( call , 0 ) ;
}
/*
* Set some sane defaults ( required by dcesrv_negotiate_contexts ( ) /
* dcesrv_check_or_create_context ( ) ) and do some protocol validation
* and set sane defaults .
*/
for ( i = 0 ; i < call - > pkt . u . bind . num_contexts ; i + + ) {
const struct dcerpc_ctx_list * c = & call - > pkt . u . bind . ctx_list [ i ] ;
struct dcerpc_ack_ctx * a = & ack_ctx_list [ i ] ;
bool is_feature = false ;
uint64_t features = 0 ;
if ( c - > num_transfer_syntaxes = = 0 ) {
return dcesrv_bind_nak ( call , 0 ) ;
}
a - > result = DCERPC_BIND_ACK_RESULT_PROVIDER_REJECTION ;
a - > reason . value = DCERPC_BIND_ACK_REASON_ABSTRACT_SYNTAX_NOT_SUPPORTED ;
/*
* It ' s only treated as bind time feature request , if the first
* transfer_syntax matches , all others are ignored .
*/
is_feature = dcerpc_extract_bind_time_features ( c - > transfer_syntaxes [ 0 ] ,
& features ) ;
if ( ! is_feature ) {
continue ;
}
if ( ack_features ! = NULL ) {
/*
* Only one bind time feature context is allowed .
*/
return dcesrv_bind_nak ( call , 0 ) ;
}
ack_features = a ;
a - > result = DCERPC_BIND_ACK_RESULT_NEGOTIATE_ACK ;
a - > reason . negotiate = 0 ;
if ( features & DCERPC_BIND_TIME_SECURITY_CONTEXT_MULTIPLEXING ) {
2021-01-14 14:17:53 +03:00
if ( conn - > max_auth_states ! = 0 ) {
2019-10-03 20:38:31 +03:00
a - > reason . negotiate | =
DCERPC_BIND_TIME_SECURITY_CONTEXT_MULTIPLEXING ;
}
}
if ( features & DCERPC_BIND_TIME_KEEP_CONNECTION_ON_ORPHAN ) {
a - > reason . negotiate | =
DCERPC_BIND_TIME_KEEP_CONNECTION_ON_ORPHAN ;
}
2021-01-14 14:17:53 +03:00
conn - > assoc_group - > bind_time_features = a - > reason . negotiate ;
2019-10-03 20:38:31 +03:00
}
/*
* Try to negotiate one new presentation context .
*
* Deep in here we locate the iface ( by uuid ) that the client
* requested , from the list of interfaces on the
* call - > conn - > endpoint , and call iface - > bind ( ) on that iface .
*
* call - > conn was set up at the accept ( ) of the socket , and
* call - > conn - > endpoint has a list of interfaces restricted to
* this port or pipe .
*/
status = dcesrv_negotiate_contexts ( call , & call - > pkt . u . bind , ack_ctx_list ) ;
if ( NT_STATUS_EQUAL ( status , NT_STATUS_RPC_PROTOCOL_ERROR ) ) {
return dcesrv_bind_nak ( call , 0 ) ;
}
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
/*
* At this point we still don ' t know which interface ( eg
* netlogon , lsa , drsuapi ) the caller requested in this bind !
* The most recently added context is available as the first
* element in the linked list at call - > conn - > contexts , that is
* call - > conn - > contexts - > iface , but they may not have
* requested one at all !
*/
if ( ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_CONC_MPX ) & &
( call - > state_flags & DCESRV_CALL_STATE_FLAG_MULTIPLEXED ) ) {
call - > conn - > state_flags | = DCESRV_CALL_STATE_FLAG_MULTIPLEXED ;
extra_flags | = DCERPC_PFC_FLAG_CONC_MPX ;
}
if ( call - > state_flags & DCESRV_CALL_STATE_FLAG_PROCESS_PENDING_CALL ) {
2021-01-14 14:17:53 +03:00
conn - > state_flags | = DCESRV_CALL_STATE_FLAG_PROCESS_PENDING_CALL ;
2019-10-03 20:38:31 +03:00
}
/*
* After finding the interface and setting up the NDR
* transport negotiation etc , handle any authentication that
* is being requested .
*/
if ( ! dcesrv_auth_bind ( call ) ) {
if ( auth - > auth_level = = DCERPC_AUTH_LEVEL_NONE ) {
/*
* With DCERPC_AUTH_LEVEL_NONE , we get the
* reject_reason in auth - > auth_context_id .
*/
return dcesrv_bind_nak ( call , auth - > auth_context_id ) ;
}
/*
* This must a be a temporary failure e . g . talloc or invalid
* configuration , e . g . no machine account .
*/
return dcesrv_bind_nak ( call ,
DCERPC_BIND_NAK_REASON_TEMPORARY_CONGESTION ) ;
}
/* setup a bind_ack */
2021-01-14 14:20:43 +03:00
dcesrv_init_hdr ( pkt , lpcfg_rpc_big_endian ( dce_ctx - > lp_ctx ) ) ;
2019-10-03 20:38:31 +03:00
pkt - > auth_length = 0 ;
pkt - > call_id = call - > pkt . call_id ;
pkt - > ptype = DCERPC_PKT_BIND_ACK ;
pkt - > pfc_flags = DCERPC_PFC_FLAG_FIRST | DCERPC_PFC_FLAG_LAST | extra_flags ;
2021-01-14 14:17:53 +03:00
pkt - > u . bind_ack . max_xmit_frag = conn - > max_xmit_frag ;
pkt - > u . bind_ack . max_recv_frag = conn - > max_recv_frag ;
pkt - > u . bind_ack . assoc_group_id = conn - > assoc_group - > id ;
2019-10-03 20:38:31 +03:00
2021-01-14 14:17:53 +03:00
ep_2nd_description = conn - > endpoint - > ep_2nd_description ;
2019-10-03 20:38:31 +03:00
if ( ep_2nd_description = = NULL ) {
2021-01-14 14:17:53 +03:00
ep_2nd_description = conn - > endpoint - > ep_description ;
2019-10-03 20:38:31 +03:00
}
endpoint = dcerpc_binding_get_string_option (
ep_2nd_description ,
" endpoint " ) ;
if ( endpoint = = NULL ) {
endpoint = " " ;
}
pkt - > u . bind_ack . secondary_address = endpoint ;
pkt - > u . bind_ack . num_results = call - > pkt . u . bind . num_contexts ;
pkt - > u . bind_ack . ctx_list = ack_ctx_list ;
pkt - > u . bind_ack . auth_info = data_blob_null ;
status = dcesrv_auth_prepare_bind_ack ( call , pkt ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return dcesrv_bind_nak ( call , 0 ) ;
}
if ( auth - > auth_finished ) {
return dcesrv_auth_reply ( call ) ;
}
subreq = gensec_update_send ( call , call - > event_ctx ,
auth - > gensec_security ,
call - > in_auth_info . credentials ) ;
if ( subreq = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
tevent_req_set_callback ( subreq , dcesrv_bind_done , call ) ;
return dcesrv_conn_auth_wait_setup ( conn ) ;
}
static void dcesrv_bind_done ( struct tevent_req * subreq )
{
struct dcesrv_call_state * call =
tevent_req_callback_data ( subreq ,
struct dcesrv_call_state ) ;
struct dcesrv_connection * conn = call - > conn ;
NTSTATUS status ;
status = gensec_update_recv ( subreq , call ,
& call - > out_auth_info - > credentials ) ;
TALLOC_FREE ( subreq ) ;
status = dcesrv_auth_complete ( call , status ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
status = dcesrv_bind_nak ( call , 0 ) ;
dcesrv_conn_auth_wait_finished ( conn , status ) ;
return ;
}
status = dcesrv_auth_reply ( call ) ;
dcesrv_conn_auth_wait_finished ( conn , status ) ;
return ;
}
static NTSTATUS dcesrv_auth_reply ( struct dcesrv_call_state * call )
{
struct ncacn_packet * pkt = & call - > ack_pkt ;
struct data_blob_list_item * rep = NULL ;
NTSTATUS status ;
rep = talloc_zero ( call , struct data_blob_list_item ) ;
if ( ! rep ) {
return NT_STATUS_NO_MEMORY ;
}
status = dcerpc_ncacn_push_auth ( & rep - > blob ,
call ,
pkt ,
call - > out_auth_info ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
dcerpc_set_frag_length ( & rep - > blob , rep - > blob . length ) ;
DLIST_ADD_END ( call - > replies , rep ) ;
dcesrv_call_set_list ( call , DCESRV_LIST_CALL_LIST ) ;
if ( call - > conn - > call_list & & call - > conn - > call_list - > replies ) {
if ( call - > conn - > transport . report_output_data ) {
call - > conn - > transport . report_output_data ( call - > conn ) ;
}
}
return NT_STATUS_OK ;
}
static void dcesrv_auth3_done ( struct tevent_req * subreq ) ;
/*
handle a auth3 request
*/
static NTSTATUS dcesrv_auth3 ( struct dcesrv_call_state * call )
{
struct dcesrv_connection * conn = call - > conn ;
struct dcesrv_auth * auth = call - > auth_state ;
struct tevent_req * subreq = NULL ;
NTSTATUS status ;
if ( ! auth - > auth_started ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
if ( auth - > auth_finished ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
status = dcerpc_verify_ncacn_packet_header ( & call - > pkt ,
DCERPC_PKT_AUTH3 ,
call - > pkt . u . auth3 . auth_info . length ,
0 , /* required flags */
DCERPC_PFC_FLAG_FIRST |
DCERPC_PFC_FLAG_LAST |
DCERPC_PFC_FLAG_SUPPORT_HEADER_SIGN |
0x08 | /* this is not defined, but should be ignored */
DCERPC_PFC_FLAG_CONC_MPX |
DCERPC_PFC_FLAG_DID_NOT_EXECUTE |
DCERPC_PFC_FLAG_MAYBE |
DCERPC_PFC_FLAG_OBJECT_UUID ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
/* handle the auth3 in the auth code */
if ( ! dcesrv_auth_prepare_auth3 ( call ) ) {
/*
* we don ' t send a reply to a auth3 request ,
* except by a fault .
*
* In anycase we mark the connection as
* invalid .
*/
auth - > auth_invalid = true ;
if ( call - > fault_code ! = 0 ) {
return dcesrv_fault_disconnect ( call , call - > fault_code ) ;
}
TALLOC_FREE ( call ) ;
return NT_STATUS_OK ;
}
subreq = gensec_update_send ( call , call - > event_ctx ,
auth - > gensec_security ,
call - > in_auth_info . credentials ) ;
if ( subreq = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
tevent_req_set_callback ( subreq , dcesrv_auth3_done , call ) ;
return dcesrv_conn_auth_wait_setup ( conn ) ;
}
static void dcesrv_auth3_done ( struct tevent_req * subreq )
{
struct dcesrv_call_state * call =
tevent_req_callback_data ( subreq ,
struct dcesrv_call_state ) ;
struct dcesrv_connection * conn = call - > conn ;
struct dcesrv_auth * auth = call - > auth_state ;
NTSTATUS status ;
status = gensec_update_recv ( subreq , call ,
& call - > out_auth_info - > credentials ) ;
TALLOC_FREE ( subreq ) ;
status = dcesrv_auth_complete ( call , status ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
/*
* we don ' t send a reply to a auth3 request ,
* except by a fault .
*
* In anycase we mark the connection as
* invalid .
*/
auth - > auth_invalid = true ;
if ( call - > fault_code ! = 0 ) {
status = dcesrv_fault_disconnect ( call , call - > fault_code ) ;
dcesrv_conn_auth_wait_finished ( conn , status ) ;
return ;
}
TALLOC_FREE ( call ) ;
dcesrv_conn_auth_wait_finished ( conn , NT_STATUS_OK ) ;
return ;
}
/*
* we don ' t send a reply to a auth3 request .
*/
TALLOC_FREE ( call ) ;
dcesrv_conn_auth_wait_finished ( conn , NT_STATUS_OK ) ;
return ;
}
static NTSTATUS dcesrv_check_or_create_context ( struct dcesrv_call_state * call ,
const struct dcerpc_bind * b ,
const struct dcerpc_ctx_list * ctx ,
struct dcerpc_ack_ctx * ack ,
bool validate_only ,
const struct ndr_syntax_id * supported_transfer )
{
struct dcesrv_connection_context * context ;
const struct dcesrv_interface * iface ;
NTSTATUS status ;
const struct ndr_syntax_id * selected_transfer = NULL ;
size_t i ;
bool ok ;
if ( b = = NULL ) {
return NT_STATUS_INTERNAL_ERROR ;
}
if ( ctx = = NULL ) {
return NT_STATUS_INTERNAL_ERROR ;
}
if ( ctx - > num_transfer_syntaxes < 1 ) {
return NT_STATUS_INTERNAL_ERROR ;
}
if ( ack = = NULL ) {
return NT_STATUS_INTERNAL_ERROR ;
}
if ( supported_transfer = = NULL ) {
return NT_STATUS_INTERNAL_ERROR ;
}
switch ( ack - > result ) {
case DCERPC_BIND_ACK_RESULT_ACCEPTANCE :
case DCERPC_BIND_ACK_RESULT_NEGOTIATE_ACK :
/*
* We is already completed .
*/
return NT_STATUS_OK ;
default :
break ;
}
ack - > result = DCERPC_BIND_ACK_RESULT_PROVIDER_REJECTION ;
ack - > reason . value = DCERPC_BIND_ACK_REASON_ABSTRACT_SYNTAX_NOT_SUPPORTED ;
2021-01-27 15:51:33 +03:00
iface = find_interface_by_syntax_id (
call - > conn - > endpoint , & ctx - > abstract_syntax ) ;
2019-10-03 20:38:31 +03:00
if ( iface = = NULL ) {
2021-01-27 11:49:33 +03:00
struct ndr_syntax_id_buf buf ;
DBG_NOTICE ( " Request for unknown dcerpc interface %s \n " ,
ndr_syntax_id_buf_string (
& ctx - > abstract_syntax , & buf ) ) ;
2019-10-03 20:38:31 +03:00
/*
* We report this only via ack - > result
*/
return NT_STATUS_OK ;
}
ack - > result = DCERPC_BIND_ACK_RESULT_PROVIDER_REJECTION ;
ack - > reason . value = DCERPC_BIND_ACK_REASON_TRANSFER_SYNTAXES_NOT_SUPPORTED ;
if ( validate_only ) {
/*
* We report this only via ack - > result
*/
return NT_STATUS_OK ;
}
for ( i = 0 ; i < ctx - > num_transfer_syntaxes ; i + + ) {
/*
* we only do NDR encoded dcerpc for now .
*/
ok = ndr_syntax_id_equal ( & ctx - > transfer_syntaxes [ i ] ,
supported_transfer ) ;
if ( ok ) {
selected_transfer = supported_transfer ;
break ;
}
}
context = dcesrv_find_context ( call - > conn , ctx - > context_id ) ;
if ( context ! = NULL ) {
ok = ndr_syntax_id_equal ( & context - > iface - > syntax_id ,
& ctx - > abstract_syntax ) ;
if ( ! ok ) {
return NT_STATUS_RPC_PROTOCOL_ERROR ;
}
if ( selected_transfer ! = NULL ) {
ok = ndr_syntax_id_equal ( & context - > transfer_syntax ,
selected_transfer ) ;
if ( ! ok ) {
return NT_STATUS_RPC_PROTOCOL_ERROR ;
}
ack - > result = DCERPC_BIND_ACK_RESULT_ACCEPTANCE ;
ack - > reason . value = DCERPC_BIND_ACK_REASON_NOT_SPECIFIED ;
ack - > syntax = context - > transfer_syntax ;
}
/*
* We report this only via ack - > result
*/
return NT_STATUS_OK ;
}
if ( selected_transfer = = NULL ) {
/*
* We report this only via ack - > result
*/
return NT_STATUS_OK ;
}
ack - > result = DCERPC_BIND_ACK_RESULT_USER_REJECTION ;
ack - > reason . value = DCERPC_BIND_ACK_REASON_LOCAL_LIMIT_EXCEEDED ;
/* add this context to the list of available context_ids */
context = talloc_zero ( call - > conn , struct dcesrv_connection_context ) ;
if ( context = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
context - > conn = call - > conn ;
context - > context_id = ctx - > context_id ;
context - > iface = iface ;
context - > transfer_syntax = * selected_transfer ;
DLIST_ADD ( call - > conn - > contexts , context ) ;
call - > context = context ;
talloc_set_destructor ( context , dcesrv_connection_context_destructor ) ;
dcesrv_prepare_context_auth ( call ) ;
/*
* Multiplex is supported by default
*/
call - > state_flags | = DCESRV_CALL_STATE_FLAG_MULTIPLEXED ;
status = iface - > bind ( context , iface ) ;
call - > context = NULL ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
/* we don't want to trigger the iface->unbind() hook */
context - > iface = NULL ;
talloc_free ( context ) ;
/*
* We report this only via ack - > result
*/
return NT_STATUS_OK ;
}
ack - > result = DCERPC_BIND_ACK_RESULT_ACCEPTANCE ;
ack - > reason . value = DCERPC_BIND_ACK_REASON_NOT_SPECIFIED ;
ack - > syntax = context - > transfer_syntax ;
return NT_STATUS_OK ;
}
static NTSTATUS dcesrv_negotiate_contexts ( struct dcesrv_call_state * call ,
const struct dcerpc_bind * b ,
struct dcerpc_ack_ctx * ack_ctx_list )
{
NTSTATUS status ;
size_t i ;
bool validate_only = false ;
bool preferred_ndr32 ;
/*
* Try to negotiate one new presentation context ,
* using our preferred transfer syntax .
*/
for ( i = 0 ; i < b - > num_contexts ; i + + ) {
const struct dcerpc_ctx_list * c = & b - > ctx_list [ i ] ;
struct dcerpc_ack_ctx * a = & ack_ctx_list [ i ] ;
status = dcesrv_check_or_create_context ( call , b , c , a ,
validate_only ,
call - > conn - > preferred_transfer ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( a - > result = = DCERPC_BIND_ACK_RESULT_ACCEPTANCE ) {
/*
* We managed to negotiate one context .
*
* = > we ' re done .
*/
validate_only = true ;
}
}
preferred_ndr32 = ndr_syntax_id_equal ( & ndr_transfer_syntax_ndr ,
call - > conn - > preferred_transfer ) ;
if ( preferred_ndr32 ) {
/*
* We ' re done .
*/
return NT_STATUS_OK ;
}
/*
* Try to negotiate one new presentation context ,
* using NDR 32 as fallback .
*/
for ( i = 0 ; i < b - > num_contexts ; i + + ) {
const struct dcerpc_ctx_list * c = & b - > ctx_list [ i ] ;
struct dcerpc_ack_ctx * a = & ack_ctx_list [ i ] ;
status = dcesrv_check_or_create_context ( call , b , c , a ,
validate_only ,
& ndr_transfer_syntax_ndr ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( a - > result = = DCERPC_BIND_ACK_RESULT_ACCEPTANCE ) {
/*
* We managed to negotiate one context .
*
* = > we ' re done .
*/
validate_only = true ;
}
}
return NT_STATUS_OK ;
}
static void dcesrv_alter_done ( struct tevent_req * subreq ) ;
/*
handle a alter context request
*/
static NTSTATUS dcesrv_alter ( struct dcesrv_call_state * call )
{
struct dcesrv_connection * conn = call - > conn ;
NTSTATUS status ;
bool auth_ok = false ;
struct ncacn_packet * pkt = & call - > ack_pkt ;
uint32_t extra_flags = 0 ;
struct dcesrv_auth * auth = call - > auth_state ;
struct dcerpc_ack_ctx * ack_ctx_list = NULL ;
struct tevent_req * subreq = NULL ;
size_t i ;
if ( ! call - > conn - > allow_alter ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
status = dcerpc_verify_ncacn_packet_header ( & call - > pkt ,
DCERPC_PKT_ALTER ,
call - > pkt . u . alter . auth_info . length ,
0 , /* required flags */
DCERPC_PFC_FLAG_FIRST |
DCERPC_PFC_FLAG_LAST |
DCERPC_PFC_FLAG_SUPPORT_HEADER_SIGN |
0x08 | /* this is not defined, but should be ignored */
DCERPC_PFC_FLAG_CONC_MPX |
DCERPC_PFC_FLAG_DID_NOT_EXECUTE |
DCERPC_PFC_FLAG_MAYBE |
DCERPC_PFC_FLAG_OBJECT_UUID ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
auth_ok = dcesrv_auth_alter ( call ) ;
if ( ! auth_ok ) {
if ( call - > fault_code ! = 0 ) {
return dcesrv_fault_disconnect ( call , call - > fault_code ) ;
}
}
if ( call - > pkt . u . alter . num_contexts < 1 ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
ack_ctx_list = talloc_zero_array ( call , struct dcerpc_ack_ctx ,
call - > pkt . u . alter . num_contexts ) ;
if ( ack_ctx_list = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
/*
* Set some sane defaults ( required by dcesrv_negotiate_contexts ( ) /
* dcesrv_check_or_create_context ( ) ) and do some protocol validation
* and set sane defaults .
*/
for ( i = 0 ; i < call - > pkt . u . alter . num_contexts ; i + + ) {
const struct dcerpc_ctx_list * c = & call - > pkt . u . alter . ctx_list [ i ] ;
struct dcerpc_ack_ctx * a = & ack_ctx_list [ i ] ;
if ( c - > num_transfer_syntaxes = = 0 ) {
return dcesrv_fault_disconnect ( call ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
a - > result = DCERPC_BIND_ACK_RESULT_PROVIDER_REJECTION ;
a - > reason . value = DCERPC_BIND_ACK_REASON_ABSTRACT_SYNTAX_NOT_SUPPORTED ;
}
/*
* Try to negotiate one new presentation context .
*/
status = dcesrv_negotiate_contexts ( call , & call - > pkt . u . alter , ack_ctx_list ) ;
if ( NT_STATUS_EQUAL ( status , NT_STATUS_RPC_PROTOCOL_ERROR ) ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_CONC_MPX ) & &
( call - > state_flags & DCESRV_CALL_STATE_FLAG_MULTIPLEXED ) ) {
call - > conn - > state_flags | = DCESRV_CALL_STATE_FLAG_MULTIPLEXED ;
extra_flags | = DCERPC_PFC_FLAG_CONC_MPX ;
}
if ( call - > state_flags & DCESRV_CALL_STATE_FLAG_PROCESS_PENDING_CALL ) {
call - > conn - > state_flags | = DCESRV_CALL_STATE_FLAG_PROCESS_PENDING_CALL ;
}
/* handle any authentication that is being requested */
if ( ! auth_ok ) {
if ( call - > in_auth_info . auth_type ! = auth - > auth_type ) {
return dcesrv_fault_disconnect ( call ,
DCERPC_FAULT_SEC_PKG_ERROR ) ;
}
return dcesrv_fault_disconnect ( call , DCERPC_FAULT_ACCESS_DENIED ) ;
}
dcesrv_init_hdr ( pkt , lpcfg_rpc_big_endian ( call - > conn - > dce_ctx - > lp_ctx ) ) ;
pkt - > auth_length = 0 ;
pkt - > call_id = call - > pkt . call_id ;
pkt - > ptype = DCERPC_PKT_ALTER_RESP ;
pkt - > pfc_flags = DCERPC_PFC_FLAG_FIRST | DCERPC_PFC_FLAG_LAST | extra_flags ;
pkt - > u . alter_resp . max_xmit_frag = call - > conn - > max_xmit_frag ;
pkt - > u . alter_resp . max_recv_frag = call - > conn - > max_recv_frag ;
pkt - > u . alter_resp . assoc_group_id = call - > conn - > assoc_group - > id ;
pkt - > u . alter_resp . secondary_address = " " ;
pkt - > u . alter_resp . num_results = call - > pkt . u . alter . num_contexts ;
pkt - > u . alter_resp . ctx_list = ack_ctx_list ;
pkt - > u . alter_resp . auth_info = data_blob_null ;
status = dcesrv_auth_prepare_alter_ack ( call , pkt ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return dcesrv_fault_disconnect ( call , DCERPC_FAULT_SEC_PKG_ERROR ) ;
}
if ( auth - > auth_finished ) {
return dcesrv_auth_reply ( call ) ;
}
subreq = gensec_update_send ( call , call - > event_ctx ,
auth - > gensec_security ,
call - > in_auth_info . credentials ) ;
if ( subreq = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
tevent_req_set_callback ( subreq , dcesrv_alter_done , call ) ;
return dcesrv_conn_auth_wait_setup ( conn ) ;
}
static void dcesrv_alter_done ( struct tevent_req * subreq )
{
struct dcesrv_call_state * call =
tevent_req_callback_data ( subreq ,
struct dcesrv_call_state ) ;
struct dcesrv_connection * conn = call - > conn ;
NTSTATUS status ;
status = gensec_update_recv ( subreq , call ,
& call - > out_auth_info - > credentials ) ;
TALLOC_FREE ( subreq ) ;
status = dcesrv_auth_complete ( call , status ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
status = dcesrv_fault_disconnect ( call , DCERPC_FAULT_SEC_PKG_ERROR ) ;
dcesrv_conn_auth_wait_finished ( conn , status ) ;
return ;
}
status = dcesrv_auth_reply ( call ) ;
dcesrv_conn_auth_wait_finished ( conn , status ) ;
return ;
}
/*
possibly save the call for inspection with ndrdump
*/
static void dcesrv_save_call ( struct dcesrv_call_state * call , const char * why )
{
# ifdef DEVELOPER
2019-11-11 05:02:30 +03:00
dcerpc_log_packet ( call - > conn - > packet_log_dir ,
call - > context - > iface - > name ,
call - > pkt . u . request . opnum ,
NDR_IN ,
& call - > pkt . u . request . stub_and_verifier ,
why ) ;
2019-10-03 20:38:31 +03:00
# endif
}
2019-11-06 07:27:08 +03:00
# ifdef DEVELOPER
/*
Save the call for use as a seed for fuzzing .
This is only enabled in a developer build , and only has effect if the
" dcesrv fuzz directory " param is set .
*/
void _dcesrv_save_ndr_fuzz_seed ( DATA_BLOB call_blob ,
struct dcesrv_call_state * call ,
int flags )
{
const char * dump_dir = lpcfg_parm_string ( call - > conn - > dce_ctx - > lp_ctx ,
NULL ,
" dcesrv " , " fuzz directory " ) ;
dcerpc_save_ndr_fuzz_seed ( call ,
call_blob ,
dump_dir ,
call - > context - > iface - > name ,
flags ,
call - > pkt . u . request . opnum ,
call - > ndr_pull - > flags & LIBNDR_FLAG_NDR64 ) ;
}
# endif /*if DEVELOPER, enveloping _dcesrv_save_ndr_fuzz_seed() */
2019-10-03 20:38:31 +03:00
static NTSTATUS dcesrv_check_verification_trailer ( struct dcesrv_call_state * call )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
const uint32_t bitmask1 = call - > conn - > client_hdr_signing ?
DCERPC_SEC_VT_CLIENT_SUPPORTS_HEADER_SIGNING : 0 ;
const struct dcerpc_sec_vt_pcontext pcontext = {
. abstract_syntax = call - > context - > iface - > syntax_id ,
. transfer_syntax = call - > context - > transfer_syntax ,
} ;
const struct dcerpc_sec_vt_header2 header2 =
dcerpc_sec_vt_header2_from_ncacn_packet ( & call - > pkt ) ;
enum ndr_err_code ndr_err ;
struct dcerpc_sec_verification_trailer * vt = NULL ;
NTSTATUS status = NT_STATUS_OK ;
bool ok ;
SMB_ASSERT ( call - > pkt . ptype = = DCERPC_PKT_REQUEST ) ;
ndr_err = ndr_pop_dcerpc_sec_verification_trailer ( call - > ndr_pull ,
frame , & vt ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
status = ndr_map_error2ntstatus ( ndr_err ) ;
goto done ;
}
ok = dcerpc_sec_verification_trailer_check ( vt , & bitmask1 ,
& pcontext , & header2 ) ;
if ( ! ok ) {
status = NT_STATUS_ACCESS_DENIED ;
goto done ;
}
done :
TALLOC_FREE ( frame ) ;
return status ;
}
/*
handle a dcerpc request packet
*/
static NTSTATUS dcesrv_request ( struct dcesrv_call_state * call )
{
const struct dcesrv_endpoint * endpoint = call - > conn - > endpoint ;
struct dcesrv_auth * auth = call - > auth_state ;
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport ( endpoint - > ep_description ) ;
struct ndr_pull * pull ;
NTSTATUS status ;
2020-11-16 16:15:06 +03:00
if ( auth - > auth_invalid ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
2019-10-03 20:38:31 +03:00
if ( ! auth - > auth_finished ) {
return dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
/* if authenticated, and the mech we use can't do async replies, don't use them... */
if ( auth - > gensec_security ! = NULL & &
! gensec_have_feature ( auth - > gensec_security , GENSEC_FEATURE_ASYNC_REPLIES ) ) {
call - > state_flags & = ~ DCESRV_CALL_STATE_FLAG_MAY_ASYNC ;
}
if ( call - > context = = NULL ) {
return dcesrv_fault_with_flags ( call , DCERPC_NCA_S_UNKNOWN_IF ,
DCERPC_PFC_FLAG_DID_NOT_EXECUTE ) ;
}
switch ( auth - > auth_level ) {
case DCERPC_AUTH_LEVEL_NONE :
case DCERPC_AUTH_LEVEL_PACKET :
case DCERPC_AUTH_LEVEL_INTEGRITY :
case DCERPC_AUTH_LEVEL_PRIVACY :
break ;
default :
if ( ! call - > context - > allow_connect ) {
char * addr ;
addr = tsocket_address_string ( call - > conn - > remote_address ,
call ) ;
DEBUG ( 2 , ( " %s: restrict auth_level_connect access "
" to [%s] with auth[type=0x%x,level=0x%x] "
" on [%s] from [%s] \n " ,
__func__ , call - > context - > iface - > name ,
auth - > auth_type ,
auth - > auth_level ,
derpc_transport_string_by_transport ( transport ) ,
addr ) ) ;
return dcesrv_fault ( call , DCERPC_FAULT_ACCESS_DENIED ) ;
}
break ;
}
if ( auth - > auth_level < call - > context - > min_auth_level ) {
char * addr ;
addr = tsocket_address_string ( call - > conn - > remote_address , call ) ;
DEBUG ( 2 , ( " %s: restrict access by min_auth_level[0x%x] "
" to [%s] with auth[type=0x%x,level=0x%x] "
" on [%s] from [%s] \n " ,
__func__ ,
call - > context - > min_auth_level ,
call - > context - > iface - > name ,
auth - > auth_type ,
auth - > auth_level ,
derpc_transport_string_by_transport ( transport ) ,
addr ) ) ;
return dcesrv_fault ( call , DCERPC_FAULT_ACCESS_DENIED ) ;
}
pull = ndr_pull_init_blob ( & call - > pkt . u . request . stub_and_verifier , call ) ;
NT_STATUS_HAVE_NO_MEMORY ( pull ) ;
pull - > flags | = LIBNDR_FLAG_REF_ALLOC ;
call - > ndr_pull = pull ;
if ( ! ( call - > pkt . drep [ 0 ] & DCERPC_DREP_LE ) ) {
pull - > flags | = LIBNDR_FLAG_BIGENDIAN ;
}
status = dcesrv_check_verification_trailer ( call ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
uint32_t faultcode = DCERPC_FAULT_OTHER ;
if ( NT_STATUS_EQUAL ( status , NT_STATUS_ACCESS_DENIED ) ) {
faultcode = DCERPC_FAULT_ACCESS_DENIED ;
}
DEBUG ( 10 , ( " dcesrv_check_verification_trailer failed: %s \n " ,
nt_errstr ( status ) ) ) ;
return dcesrv_fault ( call , faultcode ) ;
}
/* unravel the NDR for the packet */
status = call - > context - > iface - > ndr_pull ( call , call , pull , & call - > r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
uint8_t extra_flags = 0 ;
if ( call - > fault_code = = DCERPC_FAULT_OP_RNG_ERROR ) {
/* we got an unknown call */
DEBUG ( 3 , ( __location__ " : Unknown RPC call %u on %s \n " ,
call - > pkt . u . request . opnum ,
call - > context - > iface - > name ) ) ;
dcesrv_save_call ( call , " unknown " ) ;
extra_flags | = DCERPC_PFC_FLAG_DID_NOT_EXECUTE ;
} else {
dcesrv_save_call ( call , " pullfail " ) ;
}
2019-11-06 07:27:08 +03:00
2019-10-03 20:38:31 +03:00
return dcesrv_fault_with_flags ( call , call - > fault_code , extra_flags ) ;
}
2019-11-06 07:27:08 +03:00
dcesrv_save_ndr_fuzz_seed ( call - > pkt . u . request . stub_and_verifier ,
call ,
NDR_IN ) ;
2019-10-03 20:38:31 +03:00
if ( pull - > offset ! = pull - > data_size ) {
dcesrv_save_call ( call , " extrabytes " ) ;
DEBUG ( 3 , ( " Warning: %d extra bytes in incoming RPC request \n " ,
pull - > data_size - pull - > offset ) ) ;
}
/* call the dispatch function */
status = call - > context - > iface - > dispatch ( call , call , call - > r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " dcerpc fault in call %s:%02x - %s \n " ,
call - > context - > iface - > name ,
call - > pkt . u . request . opnum ,
dcerpc_errstr ( pull , call - > fault_code ) ) ) ;
return dcesrv_fault ( call , call - > fault_code ) ;
}
/* add the call to the pending list */
dcesrv_call_set_list ( call , DCESRV_LIST_PENDING_CALL_LIST ) ;
if ( call - > state_flags & DCESRV_CALL_STATE_FLAG_ASYNC ) {
return NT_STATUS_OK ;
}
return dcesrv_reply ( call ) ;
}
/*
remove the call from the right list when freed
*/
static int dcesrv_call_dequeue ( struct dcesrv_call_state * call )
{
dcesrv_call_set_list ( call , DCESRV_LIST_NONE ) ;
return 0 ;
}
_PUBLIC_ const struct tsocket_address * dcesrv_connection_get_local_address ( struct dcesrv_connection * conn )
{
return conn - > local_address ;
}
_PUBLIC_ const struct tsocket_address * dcesrv_connection_get_remote_address ( struct dcesrv_connection * conn )
{
return conn - > remote_address ;
}
/*
process some input to a dcerpc endpoint server .
*/
static NTSTATUS dcesrv_process_ncacn_packet ( struct dcesrv_connection * dce_conn ,
struct ncacn_packet * pkt ,
DATA_BLOB blob )
{
NTSTATUS status ;
struct dcesrv_call_state * call ;
struct dcesrv_call_state * existing = NULL ;
size_t num_auth_ctx = 0 ;
enum dcerpc_AuthType auth_type = 0 ;
enum dcerpc_AuthLevel auth_level = 0 ;
uint32_t auth_context_id = 0 ;
2020-11-16 16:15:06 +03:00
bool auth_invalid = false ;
2019-10-03 20:38:31 +03:00
call = talloc_zero ( dce_conn , struct dcesrv_call_state ) ;
if ( ! call ) {
data_blob_free ( & blob ) ;
talloc_free ( pkt ) ;
return NT_STATUS_NO_MEMORY ;
}
call - > conn = dce_conn ;
call - > event_ctx = dce_conn - > event_ctx ;
call - > state_flags = call - > conn - > state_flags ;
call - > time = timeval_current ( ) ;
call - > list = DCESRV_LIST_NONE ;
talloc_steal ( call , pkt ) ;
talloc_steal ( call , blob . data ) ;
call - > pkt = * pkt ;
if ( dce_conn - > max_auth_states = = 0 ) {
call - > auth_state = dce_conn - > default_auth_state ;
} else if ( call - > pkt . auth_length = = 0 ) {
if ( call - > pkt . ptype = = DCERPC_PKT_REQUEST & &
dce_conn - > default_auth_level_connect ! = NULL )
{
call - > auth_state = dce_conn - > default_auth_level_connect ;
} else {
call - > auth_state = dce_conn - > default_auth_state ;
}
}
if ( call - > auth_state = = NULL ) {
struct dcesrv_auth * a = NULL ;
2020-11-16 16:15:06 +03:00
bool check_type_level = true ;
2019-10-03 20:38:31 +03:00
auth_type = dcerpc_get_auth_type ( & blob ) ;
auth_level = dcerpc_get_auth_level ( & blob ) ;
auth_context_id = dcerpc_get_auth_context_id ( & blob ) ;
if ( call - > pkt . ptype = = DCERPC_PKT_REQUEST ) {
2020-11-16 16:15:06 +03:00
if ( ! ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_FIRST ) ) {
check_type_level = false ;
}
2019-10-03 20:38:31 +03:00
dce_conn - > default_auth_level_connect = NULL ;
if ( auth_level = = DCERPC_AUTH_LEVEL_CONNECT ) {
dce_conn - > got_explicit_auth_level_connect = true ;
}
}
for ( a = dce_conn - > auth_states ; a ! = NULL ; a = a - > next ) {
num_auth_ctx + + ;
2020-11-16 16:15:06 +03:00
if ( a - > auth_context_id ! = auth_context_id ) {
2019-10-03 20:38:31 +03:00
continue ;
}
2020-11-16 16:15:06 +03:00
if ( a - > auth_type ! = auth_type ) {
auth_invalid = true ;
2019-10-03 20:38:31 +03:00
}
2020-11-16 16:15:06 +03:00
if ( a - > auth_level ! = auth_level ) {
auth_invalid = true ;
}
if ( check_type_level & & auth_invalid ) {
a - > auth_invalid = true ;
2019-10-03 20:38:31 +03:00
}
DLIST_PROMOTE ( dce_conn - > auth_states , a ) ;
call - > auth_state = a ;
break ;
}
}
if ( call - > auth_state = = NULL ) {
struct dcesrv_auth * a = NULL ;
if ( num_auth_ctx > = dce_conn - > max_auth_states ) {
return dcesrv_fault_disconnect ( call ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
a = dcesrv_auth_create ( dce_conn ) ;
if ( a = = NULL ) {
talloc_free ( call ) ;
return NT_STATUS_NO_MEMORY ;
}
DLIST_ADD ( dce_conn - > auth_states , a ) ;
if ( call - > pkt . ptype = = DCERPC_PKT_REQUEST ) {
/*
* This can never be valid .
*/
2020-11-16 16:15:06 +03:00
auth_invalid = true ;
2019-10-03 20:38:31 +03:00
a - > auth_invalid = true ;
}
call - > auth_state = a ;
}
talloc_set_destructor ( call , dcesrv_call_dequeue ) ;
if ( call - > conn - > allow_bind ) {
/*
* Only one bind is possible per connection
*/
call - > conn - > allow_bind = false ;
return dcesrv_bind ( call ) ;
}
/* we have to check the signing here, before combining the
pdus */
if ( call - > pkt . ptype = = DCERPC_PKT_REQUEST ) {
dcesrv_default_auth_state_prepare_request ( call ) ;
if ( call - > auth_state - > auth_started & &
! call - > auth_state - > auth_finished ) {
return dcesrv_fault_disconnect ( call ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
status = dcerpc_verify_ncacn_packet_header ( & call - > pkt ,
DCERPC_PKT_REQUEST ,
call - > pkt . u . request . stub_and_verifier . length ,
0 , /* required_flags */
DCERPC_PFC_FLAG_FIRST |
DCERPC_PFC_FLAG_LAST |
DCERPC_PFC_FLAG_PENDING_CANCEL |
0x08 | /* this is not defined, but should be ignored */
DCERPC_PFC_FLAG_CONC_MPX |
DCERPC_PFC_FLAG_DID_NOT_EXECUTE |
DCERPC_PFC_FLAG_MAYBE |
DCERPC_PFC_FLAG_OBJECT_UUID ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return dcesrv_fault_disconnect ( call ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
if ( call - > pkt . frag_length > DCERPC_FRAG_MAX_SIZE ) {
/*
* We don ' t use dcesrv_fault_disconnect ( )
* here , because we don ' t want to set
* DCERPC_PFC_FLAG_DID_NOT_EXECUTE
*
* Note that we don ' t check against the negotiated
* max_recv_frag , but a hard coded value .
*/
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
2019-10-03 20:38:31 +03:00
}
if ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_FIRST ) {
if ( dce_conn - > pending_call_list ! = NULL ) {
/*
* concurrent requests are only allowed
* if DCERPC_PFC_FLAG_CONC_MPX was negotiated .
*/
if ( ! ( dce_conn - > state_flags & DCESRV_CALL_STATE_FLAG_MULTIPLEXED ) ) {
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( call ,
2019-10-03 20:38:31 +03:00
DCERPC_NCA_S_PROTO_ERROR ) ;
}
}
/* only one request is possible in the fragmented list */
if ( dce_conn - > incoming_fragmented_call_list ! = NULL ) {
2020-11-16 16:15:06 +03:00
call - > fault_code = DCERPC_NCA_S_PROTO_ERROR ;
existing = dcesrv_find_fragmented_call ( dce_conn ,
call - > pkt . call_id ) ;
if ( existing ! = NULL & & call - > auth_state ! = existing - > auth_state ) {
call - > context = dcesrv_find_context ( call - > conn ,
call - > pkt . u . request . context_id ) ;
if ( call - > pkt . auth_length ! = 0 & & existing - > context = = call - > context ) {
call - > fault_code = DCERPC_FAULT_SEC_PKG_ERROR ;
}
}
2019-10-03 20:38:31 +03:00
if ( ! ( dce_conn - > state_flags & DCESRV_CALL_STATE_FLAG_MULTIPLEXED ) ) {
/*
* Without DCERPC_PFC_FLAG_CONC_MPX
* we need to return the FAULT on the
* already existing call .
*
* This is important to get the
* call_id and context_id right .
*/
2020-11-16 16:15:06 +03:00
dce_conn - > incoming_fragmented_call_list - > fault_code = call - > fault_code ;
2019-10-03 20:38:31 +03:00
TALLOC_FREE ( call ) ;
call = dce_conn - > incoming_fragmented_call_list ;
}
2020-11-16 16:15:06 +03:00
if ( existing ! = NULL ) {
call - > context = existing - > context ;
}
return dcesrv_fault_disconnect0 ( call , call - > fault_code ) ;
2019-10-03 20:38:31 +03:00
}
if ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_PENDING_CANCEL ) {
return dcesrv_fault_disconnect ( call ,
DCERPC_FAULT_NO_CALL_ACTIVE ) ;
}
call - > context = dcesrv_find_context ( call - > conn ,
call - > pkt . u . request . context_id ) ;
if ( call - > context = = NULL ) {
return dcesrv_fault_with_flags ( call , DCERPC_NCA_S_UNKNOWN_IF ,
DCERPC_PFC_FLAG_DID_NOT_EXECUTE ) ;
}
} else {
int cmp ;
existing = dcesrv_find_fragmented_call ( dce_conn ,
call - > pkt . call_id ) ;
if ( existing = = NULL ) {
2020-11-16 16:15:06 +03:00
if ( ! ( dce_conn - > state_flags & DCESRV_CALL_STATE_FLAG_MULTIPLEXED ) ) {
/*
* Without DCERPC_PFC_FLAG_CONC_MPX
* we need to return the FAULT on the
* already existing call .
*
* This is important to get the
* call_id and context_id right .
*/
if ( dce_conn - > incoming_fragmented_call_list ! = NULL ) {
TALLOC_FREE ( call ) ;
call = dce_conn - > incoming_fragmented_call_list ;
}
return dcesrv_fault_disconnect0 ( call ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
if ( dce_conn - > incoming_fragmented_call_list ! = NULL ) {
return dcesrv_fault_disconnect0 ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
}
call - > context = dcesrv_find_context ( call - > conn ,
call - > pkt . u . request . context_id ) ;
if ( call - > context = = NULL ) {
return dcesrv_fault_with_flags ( call , DCERPC_NCA_S_UNKNOWN_IF ,
DCERPC_PFC_FLAG_DID_NOT_EXECUTE ) ;
}
if ( auth_invalid ) {
return dcesrv_fault_disconnect0 ( call ,
DCERPC_FAULT_ACCESS_DENIED ) ;
}
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( call ,
2019-10-03 20:38:31 +03:00
DCERPC_NCA_S_PROTO_ERROR ) ;
}
if ( call - > pkt . ptype ! = existing - > pkt . ptype ) {
/* trying to play silly buggers are we? */
return dcesrv_fault_disconnect ( existing ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
cmp = memcmp ( call - > pkt . drep , existing - > pkt . drep ,
sizeof ( pkt - > drep ) ) ;
if ( cmp ! = 0 ) {
return dcesrv_fault_disconnect ( existing ,
DCERPC_NCA_S_PROTO_ERROR ) ;
}
2020-11-16 16:15:06 +03:00
call - > auth_state = existing - > auth_state ;
call - > context = existing - > context ;
2019-10-03 20:38:31 +03:00
}
}
if ( call - > pkt . ptype = = DCERPC_PKT_REQUEST ) {
bool ok ;
uint8_t payload_offset = DCERPC_REQUEST_LENGTH ;
if ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_OBJECT_UUID ) {
payload_offset + = 16 ;
}
ok = dcesrv_auth_pkt_pull ( call , & blob ,
0 , /* required_flags */
DCERPC_PFC_FLAG_FIRST |
DCERPC_PFC_FLAG_LAST |
DCERPC_PFC_FLAG_PENDING_CANCEL |
0x08 | /* this is not defined, but should be ignored */
DCERPC_PFC_FLAG_CONC_MPX |
DCERPC_PFC_FLAG_DID_NOT_EXECUTE |
DCERPC_PFC_FLAG_MAYBE |
DCERPC_PFC_FLAG_OBJECT_UUID ,
payload_offset ,
& call - > pkt . u . request . stub_and_verifier ) ;
if ( ! ok ) {
/*
* We don ' t use dcesrv_fault_disconnect ( )
* here , because we don ' t want to set
* DCERPC_PFC_FLAG_DID_NOT_EXECUTE
*/
if ( call - > fault_code = = 0 ) {
call - > fault_code = DCERPC_FAULT_ACCESS_DENIED ;
}
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( call , call - > fault_code ) ;
2019-10-03 20:38:31 +03:00
}
}
/* see if this is a continued packet */
if ( existing ! = NULL ) {
struct dcerpc_request * er = & existing - > pkt . u . request ;
const struct dcerpc_request * nr = & call - > pkt . u . request ;
size_t available ;
size_t alloc_size ;
size_t alloc_hint ;
/*
* Up to 4 MByte are allowed by all fragments
*/
available = dce_conn - > max_total_request_size ;
if ( er - > stub_and_verifier . length > available ) {
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( existing ,
DCERPC_FAULT_ACCESS_DENIED ) ;
2019-10-03 20:38:31 +03:00
}
available - = er - > stub_and_verifier . length ;
if ( nr - > alloc_hint > available ) {
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( existing ,
DCERPC_FAULT_ACCESS_DENIED ) ;
2019-10-03 20:38:31 +03:00
}
if ( nr - > stub_and_verifier . length > available ) {
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( existing ,
DCERPC_FAULT_ACCESS_DENIED ) ;
2019-10-03 20:38:31 +03:00
}
alloc_hint = er - > stub_and_verifier . length + nr - > alloc_hint ;
/* allocate at least 1 byte */
alloc_hint = MAX ( alloc_hint , 1 ) ;
alloc_size = er - > stub_and_verifier . length +
nr - > stub_and_verifier . length ;
alloc_size = MAX ( alloc_size , alloc_hint ) ;
er - > stub_and_verifier . data =
talloc_realloc ( existing ,
er - > stub_and_verifier . data ,
uint8_t , alloc_size ) ;
if ( er - > stub_and_verifier . data = = NULL ) {
TALLOC_FREE ( call ) ;
return dcesrv_fault_with_flags ( existing ,
DCERPC_FAULT_OUT_OF_RESOURCES ,
DCERPC_PFC_FLAG_DID_NOT_EXECUTE ) ;
}
memcpy ( er - > stub_and_verifier . data +
er - > stub_and_verifier . length ,
nr - > stub_and_verifier . data ,
nr - > stub_and_verifier . length ) ;
er - > stub_and_verifier . length + = nr - > stub_and_verifier . length ;
existing - > pkt . pfc_flags | = ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_LAST ) ;
TALLOC_FREE ( call ) ;
call = existing ;
}
/* this may not be the last pdu in the chain - if its isn't then
just put it on the incoming_fragmented_call_list and wait for the rest */
if ( call - > pkt . ptype = = DCERPC_PKT_REQUEST & &
! ( call - > pkt . pfc_flags & DCERPC_PFC_FLAG_LAST ) ) {
/*
* Up to 4 MByte are allowed by all fragments
*/
if ( call - > pkt . u . request . alloc_hint > dce_conn - > max_total_request_size ) {
2020-11-13 13:27:19 +03:00
return dcesrv_fault_disconnect0 ( call ,
DCERPC_FAULT_ACCESS_DENIED ) ;
2019-10-03 20:38:31 +03:00
}
dcesrv_call_set_list ( call , DCESRV_LIST_FRAGMENTED_CALL_LIST ) ;
return NT_STATUS_OK ;
}
/* This removes any fragments we may have had stashed away */
dcesrv_call_set_list ( call , DCESRV_LIST_NONE ) ;
switch ( call - > pkt . ptype ) {
case DCERPC_PKT_BIND :
status = dcesrv_bind_nak ( call ,
DCERPC_BIND_NAK_REASON_NOT_SPECIFIED ) ;
break ;
case DCERPC_PKT_AUTH3 :
status = dcesrv_auth3 ( call ) ;
break ;
case DCERPC_PKT_ALTER :
status = dcesrv_alter ( call ) ;
break ;
case DCERPC_PKT_REQUEST :
status = dcesrv_request ( call ) ;
break ;
case DCERPC_PKT_CO_CANCEL :
case DCERPC_PKT_ORPHANED :
/*
* Window just ignores CO_CANCEL and ORPHANED ,
* so we do . . .
*/
status = NT_STATUS_OK ;
TALLOC_FREE ( call ) ;
break ;
case DCERPC_PKT_BIND_ACK :
case DCERPC_PKT_BIND_NAK :
case DCERPC_PKT_ALTER_RESP :
case DCERPC_PKT_RESPONSE :
case DCERPC_PKT_FAULT :
case DCERPC_PKT_SHUTDOWN :
default :
status = dcesrv_fault_disconnect ( call , DCERPC_NCA_S_PROTO_ERROR ) ;
break ;
}
/* if we are going to be sending a reply then add
it to the list of pending calls . We add it to the end to keep the call
list in the order we will answer */
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( call ) ;
}
return status ;
}
_PUBLIC_ NTSTATUS dcesrv_init_context ( TALLOC_CTX * mem_ctx ,
struct loadparm_context * lp_ctx ,
struct dcesrv_context_callbacks * cb ,
struct dcesrv_context * * _dce_ctx )
{
struct dcesrv_context * dce_ctx ;
2021-02-02 17:07:35 +03:00
if ( cb = = NULL ) {
return NT_STATUS_INVALID_PARAMETER ;
}
2019-10-03 20:38:31 +03:00
dce_ctx = talloc_zero ( mem_ctx , struct dcesrv_context ) ;
NT_STATUS_HAVE_NO_MEMORY ( dce_ctx ) ;
if ( uid_wrapper_enabled ( ) ) {
setenv ( " UID_WRAPPER_MYUID " , " 1 " , 1 ) ;
}
dce_ctx - > initial_euid = geteuid ( ) ;
if ( uid_wrapper_enabled ( ) ) {
unsetenv ( " UID_WRAPPER_MYUID " ) ;
}
dce_ctx - > endpoint_list = NULL ;
dce_ctx - > lp_ctx = lp_ctx ;
dce_ctx - > assoc_groups_idr = idr_init ( dce_ctx ) ;
2020-12-26 11:27:10 +03:00
if ( dce_ctx - > assoc_groups_idr = = NULL ) {
TALLOC_FREE ( dce_ctx ) ;
return NT_STATUS_NO_MEMORY ;
}
2019-10-03 20:38:31 +03:00
dce_ctx - > broken_connections = NULL ;
2021-02-02 17:07:35 +03:00
dce_ctx - > callbacks = cb ;
2019-10-03 20:38:31 +03:00
2019-02-05 20:54:02 +03:00
* _dce_ctx = dce_ctx ;
return NT_STATUS_OK ;
}
2019-10-01 17:59:07 +03:00
_PUBLIC_ NTSTATUS dcesrv_reinit_context ( struct dcesrv_context * dce_ctx )
{
NTSTATUS status ;
status = dcesrv_shutdown_registered_ep_servers ( dce_ctx ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
/* Clear endpoints */
while ( dce_ctx - > endpoint_list ! = NULL ) {
struct dcesrv_endpoint * e = dce_ctx - > endpoint_list ;
DLIST_REMOVE ( dce_ctx - > endpoint_list , e ) ;
TALLOC_FREE ( e ) ;
}
/* Remove broken connections */
dcesrv_cleanup_broken_connections ( dce_ctx ) ;
/* Reinit assoc group idr */
TALLOC_FREE ( dce_ctx - > assoc_groups_idr ) ;
dce_ctx - > assoc_groups_idr = idr_init ( dce_ctx ) ;
if ( dce_ctx - > assoc_groups_idr = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
return NT_STATUS_OK ;
}
2021-02-02 17:10:38 +03:00
/**
* @ brief Set callback functions on an existing dcesrv_context
*
* This allows to reset callbacks initially set via
* dcesrv_init_context ( )
*
* @ param [ in ] dce_ctx The context to set the callbacks on
* @ param [ in ] cb The callbacks to set on dce_ctx
*/
_PUBLIC_ void dcesrv_context_set_callbacks (
struct dcesrv_context * dce_ctx ,
struct dcesrv_context_callbacks * cb )
{
dce_ctx - > callbacks = cb ;
}
2019-02-05 20:54:02 +03:00
_PUBLIC_ NTSTATUS dcesrv_init_ep_servers ( struct dcesrv_context * dce_ctx ,
const char * * endpoint_servers )
{
NTSTATUS status ;
int i ;
if ( endpoint_servers = = NULL ) {
DBG_ERR ( " No endpoint servers configured \n " ) ;
return NT_STATUS_INTERNAL_ERROR ;
}
2019-10-03 20:38:31 +03:00
for ( i = 0 ; endpoint_servers [ i ] ; i + + ) {
2019-09-06 15:38:29 +03:00
status = dcesrv_init_ep_server ( dce_ctx , endpoint_servers [ i ] ) ;
2019-10-03 20:38:31 +03:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2019-09-06 15:38:29 +03:00
DBG_ERR ( " failed to init endpoint server = '%s': %s \n " ,
endpoint_servers [ i ] , nt_errstr ( status ) ) ;
2019-10-03 20:38:31 +03:00
return status ;
}
}
return NT_STATUS_OK ;
}
/* the list of currently registered DCERPC endpoint servers.
*/
static struct ep_server {
struct dcesrv_endpoint_server * ep_server ;
} * ep_servers = NULL ;
2019-10-03 20:44:10 +03:00
static int num_ep_servers = 0 ;
2019-10-03 20:38:31 +03:00
2019-09-06 15:38:29 +03:00
_PUBLIC_ NTSTATUS dcesrv_init_registered_ep_servers (
struct dcesrv_context * dce_ctx )
{
NTSTATUS status ;
int i ;
for ( i = 0 ; i < num_ep_servers ; i + + ) {
status = dcesrv_init_ep_server ( dce_ctx ,
ep_servers [ i ] . ep_server - > name ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
}
return NT_STATUS_OK ;
}
_PUBLIC_ NTSTATUS dcesrv_init_ep_server ( struct dcesrv_context * dce_ctx ,
const char * ep_server_name )
{
struct dcesrv_endpoint_server * ep_server = NULL ;
NTSTATUS status ;
ep_server = discard_const_p ( struct dcesrv_endpoint_server ,
dcesrv_ep_server_byname ( ep_server_name ) ) ;
if ( ep_server = = NULL ) {
DBG_ERR ( " Failed to find endpoint server '%s' \n " ,
ep_server_name ) ;
return NT_STATUS_INTERNAL_ERROR ;
}
if ( ep_server - > initialized ) {
return NT_STATUS_OK ;
}
status = ep_server - > init_server ( dce_ctx , ep_server ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_ERR ( " Failed to init endpoint server '%s': %s \n " ,
ep_server_name , nt_errstr ( status ) ) ;
return status ;
}
ep_server - > initialized = true ;
return NT_STATUS_OK ;
}
2019-09-06 16:16:01 +03:00
_PUBLIC_ NTSTATUS dcesrv_shutdown_registered_ep_servers (
struct dcesrv_context * dce_ctx )
{
NTSTATUS status ;
int i ;
for ( i = 0 ; i < num_ep_servers ; i + + ) {
status = dcesrv_shutdown_ep_server ( dce_ctx ,
ep_servers [ i ] . ep_server - > name ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
}
return NT_STATUS_OK ;
}
_PUBLIC_ NTSTATUS dcesrv_shutdown_ep_server ( struct dcesrv_context * dce_ctx ,
const char * ep_server_name )
{
struct dcesrv_endpoint_server * ep_server = NULL ;
NTSTATUS status ;
ep_server = discard_const_p ( struct dcesrv_endpoint_server ,
dcesrv_ep_server_byname ( ep_server_name ) ) ;
if ( ep_server = = NULL ) {
DBG_ERR ( " Failed to find endpoint server '%s' \n " ,
ep_server_name ) ;
return NT_STATUS_INTERNAL_ERROR ;
}
if ( ! ep_server - > initialized ) {
return NT_STATUS_OK ;
}
DBG_INFO ( " Shutting down DCE/RPC endpoint server '%s' \n " ,
ep_server_name ) ;
status = ep_server - > shutdown_server ( dce_ctx , ep_server ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DBG_ERR ( " Failed to shutdown endpoint server '%s': %s \n " ,
ep_server_name , nt_errstr ( status ) ) ;
return status ;
}
ep_server - > initialized = false ;
return NT_STATUS_OK ;
}
2019-10-03 20:38:31 +03:00
/*
register a DCERPC endpoint server .
The ' name ' can be later used by other backends to find the operations
structure for this backend .
*/
_PUBLIC_ NTSTATUS dcerpc_register_ep_server ( const struct dcesrv_endpoint_server * ep_server )
{
if ( dcesrv_ep_server_byname ( ep_server - > name ) ! = NULL ) {
/* its already registered! */
DEBUG ( 0 , ( " DCERPC endpoint server '%s' already registered \n " ,
ep_server - > name ) ) ;
return NT_STATUS_OBJECT_NAME_COLLISION ;
}
ep_servers = realloc_p ( ep_servers , struct ep_server , num_ep_servers + 1 ) ;
if ( ! ep_servers ) {
smb_panic ( " out of memory in dcerpc_register " ) ;
}
ep_servers [ num_ep_servers ] . ep_server = smb_xmemdup ( ep_server , sizeof ( * ep_server ) ) ;
ep_servers [ num_ep_servers ] . ep_server - > name = smb_xstrdup ( ep_server - > name ) ;
num_ep_servers + + ;
DEBUG ( 3 , ( " DCERPC endpoint server '%s' registered \n " ,
ep_server - > name ) ) ;
return NT_STATUS_OK ;
}
/*
return the operations structure for a named backend of the specified type
*/
_PUBLIC_ const struct dcesrv_endpoint_server * dcesrv_ep_server_byname ( const char * name )
{
int i ;
for ( i = 0 ; i < num_ep_servers ; i + + ) {
if ( strcmp ( ep_servers [ i ] . ep_server - > name , name ) = = 0 ) {
return ep_servers [ i ] . ep_server ;
}
}
return NULL ;
}
/*
return the DCERPC module version , and the size of some critical types
This can be used by endpoint server modules to either detect compilation errors , or provide
multiple implementations for different smbd compilation options in one module
*/
const struct dcesrv_critical_sizes * dcerpc_module_version ( void )
{
static const struct dcesrv_critical_sizes critical_sizes = {
DCERPC_MODULE_VERSION ,
sizeof ( struct dcesrv_context ) ,
sizeof ( struct dcesrv_endpoint ) ,
sizeof ( struct dcesrv_endpoint_server ) ,
sizeof ( struct dcesrv_interface ) ,
sizeof ( struct dcesrv_if_list ) ,
sizeof ( struct dcesrv_connection ) ,
sizeof ( struct dcesrv_call_state ) ,
sizeof ( struct dcesrv_auth ) ,
sizeof ( struct dcesrv_handle )
} ;
return & critical_sizes ;
}
_PUBLIC_ void dcesrv_terminate_connection ( struct dcesrv_connection * dce_conn , const char * reason )
{
struct dcesrv_context * dce_ctx = dce_conn - > dce_ctx ;
struct dcesrv_auth * a = NULL ;
dce_conn - > wait_send = NULL ;
dce_conn - > wait_recv = NULL ;
dce_conn - > wait_private = NULL ;
dce_conn - > allow_bind = false ;
dce_conn - > allow_alter = false ;
dce_conn - > default_auth_state - > auth_invalid = true ;
for ( a = dce_conn - > auth_states ; a ! = NULL ; a = a - > next ) {
a - > auth_invalid = true ;
}
if ( dce_conn - > pending_call_list = = NULL ) {
char * full_reason = talloc_asprintf ( dce_conn , " dcesrv: %s " , reason ) ;
DLIST_REMOVE ( dce_ctx - > broken_connections , dce_conn ) ;
dce_conn - > transport . terminate_connection ( dce_conn ,
full_reason ? full_reason : reason ) ;
return ;
}
if ( dce_conn - > terminate ! = NULL ) {
return ;
}
DEBUG ( 3 , ( " dcesrv: terminating connection due to '%s' deferred due to pending calls \n " ,
reason ) ) ;
dce_conn - > terminate = talloc_strdup ( dce_conn , reason ) ;
if ( dce_conn - > terminate = = NULL ) {
dce_conn - > terminate = " dcesrv: deferred terminating connection - no memory " ;
}
DLIST_ADD_END ( dce_ctx - > broken_connections , dce_conn ) ;
}
_PUBLIC_ void dcesrv_cleanup_broken_connections ( struct dcesrv_context * dce_ctx )
{
struct dcesrv_connection * cur , * next ;
next = dce_ctx - > broken_connections ;
while ( next ! = NULL ) {
cur = next ;
next = cur - > next ;
if ( cur - > state_flags & DCESRV_CALL_STATE_FLAG_PROCESS_PENDING_CALL ) {
struct dcesrv_connection_context * context_cur , * context_next ;
context_next = cur - > contexts ;
while ( context_next ! = NULL ) {
context_cur = context_next ;
context_next = context_cur - > next ;
dcesrv_connection_context_destructor ( context_cur ) ;
}
}
dcesrv_terminate_connection ( cur , cur - > terminate ) ;
}
}
struct dcesrv_sock_reply_state {
struct dcesrv_connection * dce_conn ;
struct dcesrv_call_state * call ;
struct iovec iov ;
} ;
static void dcesrv_sock_reply_done ( struct tevent_req * subreq ) ;
static void dcesrv_call_terminate_step1 ( struct tevent_req * subreq ) ;
_PUBLIC_ void dcesrv_sock_report_output_data ( struct dcesrv_connection * dce_conn )
{
struct dcesrv_call_state * call ;
call = dce_conn - > call_list ;
if ( ! call | | ! call - > replies ) {
return ;
}
while ( call - > replies ) {
struct data_blob_list_item * rep = call - > replies ;
struct dcesrv_sock_reply_state * substate ;
struct tevent_req * subreq ;
substate = talloc_zero ( call , struct dcesrv_sock_reply_state ) ;
if ( ! substate ) {
dcesrv_terminate_connection ( dce_conn , " no memory " ) ;
return ;
}
substate - > dce_conn = dce_conn ;
substate - > call = NULL ;
DLIST_REMOVE ( call - > replies , rep ) ;
if ( call - > replies = = NULL & & call - > terminate_reason = = NULL ) {
substate - > call = call ;
}
substate - > iov . iov_base = ( void * ) rep - > blob . data ;
substate - > iov . iov_len = rep - > blob . length ;
subreq = tstream_writev_queue_send ( substate ,
dce_conn - > event_ctx ,
dce_conn - > stream ,
dce_conn - > send_queue ,
& substate - > iov , 1 ) ;
if ( ! subreq ) {
dcesrv_terminate_connection ( dce_conn , " no memory " ) ;
return ;
}
tevent_req_set_callback ( subreq , dcesrv_sock_reply_done ,
substate ) ;
}
if ( call - > terminate_reason ! = NULL ) {
struct tevent_req * subreq ;
subreq = tevent_queue_wait_send ( call ,
dce_conn - > event_ctx ,
dce_conn - > send_queue ) ;
if ( ! subreq ) {
dcesrv_terminate_connection ( dce_conn , __location__ ) ;
return ;
}
tevent_req_set_callback ( subreq , dcesrv_call_terminate_step1 ,
call ) ;
}
DLIST_REMOVE ( call - > conn - > call_list , call ) ;
call - > list = DCESRV_LIST_NONE ;
}
static void dcesrv_sock_reply_done ( struct tevent_req * subreq )
{
struct dcesrv_sock_reply_state * substate = tevent_req_callback_data ( subreq ,
struct dcesrv_sock_reply_state ) ;
int ret ;
int sys_errno ;
NTSTATUS status ;
struct dcesrv_call_state * call = substate - > call ;
ret = tstream_writev_queue_recv ( subreq , & sys_errno ) ;
TALLOC_FREE ( subreq ) ;
if ( ret = = - 1 ) {
status = map_nt_error_from_unix_common ( sys_errno ) ;
dcesrv_terminate_connection ( substate - > dce_conn , nt_errstr ( status ) ) ;
return ;
}
talloc_free ( substate ) ;
if ( call ) {
talloc_free ( call ) ;
}
}
static void dcesrv_call_terminate_step2 ( struct tevent_req * subreq ) ;
static void dcesrv_call_terminate_step1 ( struct tevent_req * subreq )
{
struct dcesrv_call_state * call = tevent_req_callback_data ( subreq ,
struct dcesrv_call_state ) ;
bool ok ;
struct timeval tv ;
/* make sure we stop send queue before removing subreq */
tevent_queue_stop ( call - > conn - > send_queue ) ;
ok = tevent_queue_wait_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! ok ) {
dcesrv_terminate_connection ( call - > conn , __location__ ) ;
return ;
}
/* disconnect after 200 usecs */
tv = timeval_current_ofs_usec ( 200 ) ;
subreq = tevent_wakeup_send ( call , call - > conn - > event_ctx , tv ) ;
if ( subreq = = NULL ) {
dcesrv_terminate_connection ( call - > conn , __location__ ) ;
return ;
}
tevent_req_set_callback ( subreq , dcesrv_call_terminate_step2 ,
call ) ;
}
static void dcesrv_call_terminate_step2 ( struct tevent_req * subreq )
{
struct dcesrv_call_state * call = tevent_req_callback_data ( subreq ,
struct dcesrv_call_state ) ;
bool ok ;
ok = tevent_wakeup_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! ok ) {
dcesrv_terminate_connection ( call - > conn , __location__ ) ;
return ;
}
dcesrv_terminate_connection ( call - > conn , call - > terminate_reason ) ;
}
static void dcesrv_conn_wait_done ( struct tevent_req * subreq ) ;
static void dcesrv_read_fragment_done ( struct tevent_req * subreq )
{
struct dcesrv_connection * dce_conn = tevent_req_callback_data ( subreq ,
struct dcesrv_connection ) ;
struct dcesrv_context * dce_ctx = dce_conn - > dce_ctx ;
struct ncacn_packet * pkt ;
DATA_BLOB buffer ;
NTSTATUS status ;
if ( dce_conn - > terminate ) {
/*
* if the current connection is broken
* we need to clean it up before any other connection
*/
dcesrv_terminate_connection ( dce_conn , dce_conn - > terminate ) ;
dcesrv_cleanup_broken_connections ( dce_ctx ) ;
return ;
}
dcesrv_cleanup_broken_connections ( dce_ctx ) ;
status = dcerpc_read_ncacn_packet_recv ( subreq , dce_conn ,
& pkt , & buffer ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
dcesrv_terminate_connection ( dce_conn , nt_errstr ( status ) ) ;
return ;
}
status = dcesrv_process_ncacn_packet ( dce_conn , pkt , buffer ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
dcesrv_terminate_connection ( dce_conn , nt_errstr ( status ) ) ;
return ;
}
/*
* This is used to block the connection during
* pending authentication .
*/
if ( dce_conn - > wait_send ! = NULL ) {
subreq = dce_conn - > wait_send ( dce_conn ,
dce_conn - > event_ctx ,
dce_conn - > wait_private ) ;
if ( ! subreq ) {
status = NT_STATUS_NO_MEMORY ;
dcesrv_terminate_connection ( dce_conn , nt_errstr ( status ) ) ;
return ;
}
tevent_req_set_callback ( subreq , dcesrv_conn_wait_done , dce_conn ) ;
return ;
}
subreq = dcerpc_read_ncacn_packet_send ( dce_conn ,
dce_conn - > event_ctx ,
dce_conn - > stream ) ;
if ( ! subreq ) {
status = NT_STATUS_NO_MEMORY ;
dcesrv_terminate_connection ( dce_conn , nt_errstr ( status ) ) ;
return ;
}
tevent_req_set_callback ( subreq , dcesrv_read_fragment_done , dce_conn ) ;
}
static void dcesrv_conn_wait_done ( struct tevent_req * subreq )
{
struct dcesrv_connection * dce_conn = tevent_req_callback_data ( subreq ,
struct dcesrv_connection ) ;
struct dcesrv_context * dce_ctx = dce_conn - > dce_ctx ;
NTSTATUS status ;
if ( dce_conn - > terminate ) {
/*
* if the current connection is broken
* we need to clean it up before any other connection
*/
dcesrv_terminate_connection ( dce_conn , dce_conn - > terminate ) ;
dcesrv_cleanup_broken_connections ( dce_ctx ) ;
return ;
}
dcesrv_cleanup_broken_connections ( dce_ctx ) ;
status = dce_conn - > wait_recv ( subreq ) ;
dce_conn - > wait_send = NULL ;
dce_conn - > wait_recv = NULL ;
dce_conn - > wait_private = NULL ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
dcesrv_terminate_connection ( dce_conn , nt_errstr ( status ) ) ;
return ;
}
status = dcesrv_connection_loop_start ( dce_conn ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
dcesrv_terminate_connection ( dce_conn , nt_errstr ( status ) ) ;
return ;
}
}
/**
* retrieve credentials from a dce_call
*/
_PUBLIC_ struct cli_credentials * dcesrv_call_credentials ( struct dcesrv_call_state * dce_call )
{
struct dcesrv_auth * auth = dce_call - > auth_state ;
SMB_ASSERT ( auth - > auth_finished ) ;
return auth - > session_info - > credentials ;
}
/**
* returns true if this is an authenticated call
*/
_PUBLIC_ bool dcesrv_call_authenticated ( struct dcesrv_call_state * dce_call )
{
struct dcesrv_auth * auth = dce_call - > auth_state ;
enum security_user_level level ;
SMB_ASSERT ( auth - > auth_finished ) ;
level = security_session_user_level ( auth - > session_info , NULL ) ;
return level > = SECURITY_USER ;
}
/**
* retrieve account_name for a dce_call
*/
_PUBLIC_ const char * dcesrv_call_account_name ( struct dcesrv_call_state * dce_call )
{
struct dcesrv_auth * auth = dce_call - > auth_state ;
SMB_ASSERT ( auth - > auth_finished ) ;
return auth - > session_info - > info - > account_name ;
}
/**
* retrieve session_info from a dce_call
*/
_PUBLIC_ struct auth_session_info * dcesrv_call_session_info ( struct dcesrv_call_state * dce_call )
{
struct dcesrv_auth * auth = dce_call - > auth_state ;
SMB_ASSERT ( auth - > auth_finished ) ;
return auth - > session_info ;
}
/**
* retrieve auth type / level from a dce_call
*/
_PUBLIC_ void dcesrv_call_auth_info ( struct dcesrv_call_state * dce_call ,
enum dcerpc_AuthType * auth_type ,
enum dcerpc_AuthLevel * auth_level )
{
struct dcesrv_auth * auth = dce_call - > auth_state ;
SMB_ASSERT ( auth - > auth_finished ) ;
if ( auth_type ! = NULL ) {
* auth_type = auth - > auth_type ;
}
if ( auth_level ! = NULL ) {
* auth_level = auth - > auth_level ;
}
}
_PUBLIC_ NTSTATUS dcesrv_connection_loop_start ( struct dcesrv_connection * conn )
{
struct tevent_req * subreq ;
subreq = dcerpc_read_ncacn_packet_send ( conn ,
conn - > event_ctx ,
conn - > stream ) ;
if ( subreq = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
tevent_req_set_callback ( subreq , dcesrv_read_fragment_done , conn ) ;
return NT_STATUS_OK ;
}
2020-10-23 12:42:14 +03:00
_PUBLIC_ NTSTATUS dcesrv_call_dispatch_local ( struct dcesrv_call_state * call )
{
NTSTATUS status ;
struct ndr_pull * pull = NULL ;
struct ndr_push * push = NULL ;
struct data_blob_list_item * rep = NULL ;
pull = ndr_pull_init_blob ( & call - > pkt . u . request . stub_and_verifier ,
call ) ;
if ( pull = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
pull - > flags | = LIBNDR_FLAG_REF_ALLOC ;
call - > ndr_pull = pull ;
/* unravel the NDR for the packet */
status = call - > context - > iface - > ndr_pull ( call , call , pull , & call - > r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2021-03-01 19:26:24 +03:00
DBG_INFO ( " DCE/RPC fault in call %s:%02X - %s \n " ,
call - > context - > iface - > name ,
call - > pkt . u . request . opnum ,
dcerpc_errstr ( call , call - > fault_code ) ) ;
2020-10-23 12:52:57 +03:00
return dcerpc_fault_to_nt_status ( call - > fault_code ) ;
2020-10-23 12:42:14 +03:00
}
status = call - > context - > iface - > local ( call , call , call - > r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2021-03-01 19:26:24 +03:00
DBG_INFO ( " DCE/RPC fault in call %s:%02X - %s \n " ,
call - > context - > iface - > name ,
call - > pkt . u . request . opnum ,
dcerpc_errstr ( call , call - > fault_code ) ) ;
2020-10-23 12:52:57 +03:00
return dcerpc_fault_to_nt_status ( call - > fault_code ) ;
2020-10-23 12:42:14 +03:00
}
2020-10-23 12:54:29 +03:00
/* This can never go async for now! */
SMB_ASSERT ( ! ( call - > state_flags & DCESRV_CALL_STATE_FLAG_ASYNC ) ) ;
2020-10-23 12:56:19 +03:00
/* call the reply function */
status = call - > context - > iface - > reply ( call , call , call - > r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2021-03-01 19:26:24 +03:00
DBG_INFO ( " DCE/RPC fault in call %s:%02X - %s \n " ,
call - > context - > iface - > name ,
call - > pkt . u . request . opnum ,
dcerpc_errstr ( call , call - > fault_code ) ) ;
2020-10-23 12:56:19 +03:00
return dcerpc_fault_to_nt_status ( call - > fault_code ) ;
}
2020-10-23 12:42:14 +03:00
push = ndr_push_init_ctx ( call ) ;
if ( push = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
push - > ptr_count = call - > ndr_pull - > ptr_count ;
status = call - > context - > iface - > ndr_push ( call , call , push , call - > r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2021-03-01 19:26:24 +03:00
DBG_INFO ( " DCE/RPC fault in call %s:%02X - %s \n " ,
call - > context - > iface - > name ,
call - > pkt . u . request . opnum ,
dcerpc_errstr ( call , call - > fault_code ) ) ;
2020-10-23 12:52:57 +03:00
return dcerpc_fault_to_nt_status ( call - > fault_code ) ;
2020-10-23 12:42:14 +03:00
}
rep = talloc_zero ( call , struct data_blob_list_item ) ;
if ( rep = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
rep - > blob = ndr_push_blob ( push ) ;
DLIST_ADD_END ( call - > replies , rep ) ;
return NT_STATUS_OK ;
}