2010-06-03 22:04:08 +02:00
/*
* Unix SMB / CIFS implementation .
* RPC Pipe client / server routines
* Copyright ( C ) Andrew Tridgell 1992 - 1998 ,
* Largely re - written : 2005
* Copyright ( C ) Jeremy Allison 1998 - 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 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"
2010-07-02 10:17:44 +02:00
# include "rpc_server/srv_pipe_internal.h"
2010-06-03 22:04:08 +02:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_SRV
static int pipes_open ;
static pipes_struct * InternalPipes ;
/* TODO
* the following prototypes are declared here to avoid
* code being moved about too much for a patch to be
* disrupted / less obvious .
*
* these functions , and associated functions that they
* call , should be moved behind a . so module - loading
* system _anyway_ . so that ' s the next step . . .
*/
static int close_internal_rpc_pipe_hnd ( struct pipes_struct * p ) ;
/****************************************************************************
Internal Pipe iterator functions .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
pipes_struct * get_first_internal_pipe ( void )
{
return InternalPipes ;
}
pipes_struct * get_next_internal_pipe ( pipes_struct * p )
{
return p - > next ;
}
static void free_pipe_rpc_context_internal ( PIPE_RPC_FNS * list )
{
PIPE_RPC_FNS * tmp = list ;
PIPE_RPC_FNS * tmp2 ;
while ( tmp ) {
tmp2 = tmp - > next ;
SAFE_FREE ( tmp ) ;
tmp = tmp2 ;
}
return ;
}
2010-06-07 14:08:05 -04:00
bool check_open_pipes ( void )
{
pipes_struct * p ;
for ( p = InternalPipes ; p ! = NULL ; p = p - > next ) {
if ( num_pipe_handles ( p ) ! = 0 ) {
return true ;
}
}
return false ;
}
2010-06-03 22:04:08 +02:00
/****************************************************************************
Close an rpc pipe .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int close_internal_rpc_pipe_hnd ( struct pipes_struct * p )
{
if ( ! p ) {
DEBUG ( 0 , ( " Invalid pipe in close_internal_rpc_pipe_hnd \n " ) ) ;
return False ;
}
if ( p - > auth . auth_data_free_func ) {
( * p - > auth . auth_data_free_func ) ( & p - > auth ) ;
}
free_pipe_rpc_context_internal ( p - > contexts ) ;
/* Free the handles database. */
close_policy_by_pipe ( p ) ;
DLIST_REMOVE ( InternalPipes , p ) ;
ZERO_STRUCTP ( p ) ;
return 0 ;
}
/****************************************************************************
Make an internal namedpipes structure
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct pipes_struct * make_internal_rpc_pipe_p ( TALLOC_CTX * mem_ctx ,
const struct ndr_syntax_id * syntax ,
const char * client_address ,
struct auth_serversupplied_info * server_info )
{
pipes_struct * p ;
DEBUG ( 4 , ( " Create pipe requested %s \n " ,
get_pipe_name_from_syntax ( talloc_tos ( ) , syntax ) ) ) ;
p = TALLOC_ZERO_P ( mem_ctx , struct pipes_struct ) ;
if ( ! p ) {
DEBUG ( 0 , ( " ERROR! no memory for pipes_struct! \n " ) ) ;
return NULL ;
}
p - > mem_ctx = talloc_named ( p , 0 , " pipe %s %p " ,
get_pipe_name_from_syntax ( talloc_tos ( ) ,
syntax ) , p ) ;
if ( p - > mem_ctx = = NULL ) {
DEBUG ( 0 , ( " open_rpc_pipe_p: talloc_init failed. \n " ) ) ;
TALLOC_FREE ( p ) ;
return NULL ;
}
2010-06-07 17:38:01 -04:00
if ( ! init_pipe_handles ( p , syntax ) ) {
2010-06-03 22:04:08 +02:00
DEBUG ( 0 , ( " open_rpc_pipe_p: init_pipe_handles failed. \n " ) ) ;
TALLOC_FREE ( p ) ;
return NULL ;
}
/*
* Initialize the incoming RPC data buffer with one PDU worth of memory .
* We cheat here and say we ' re marshalling , as we intend to add incoming
* data directly into the prs_struct and we want it to auto grow . We will
* change the type to UNMARSALLING before processing the stream .
*/
if ( ! prs_init ( & p - > in_data . data , 128 , p - > mem_ctx , MARSHALL ) ) {
DEBUG ( 0 , ( " open_rpc_pipe_p: malloc fail for in_data struct. \n " ) ) ;
close_policy_by_pipe ( p ) ;
TALLOC_FREE ( p ) ;
return NULL ;
}
p - > server_info = copy_serverinfo ( p , server_info ) ;
if ( p - > server_info = = NULL ) {
DEBUG ( 0 , ( " open_rpc_pipe_p: copy_serverinfo failed \n " ) ) ;
close_policy_by_pipe ( p ) ;
TALLOC_FREE ( p ) ;
return NULL ;
}
DLIST_ADD ( InternalPipes , p ) ;
2010-06-07 15:07:38 -04:00
strlcpy ( p - > client_address , client_address , sizeof ( p - > client_address ) ) ;
2010-06-03 22:04:08 +02:00
p - > endian = RPC_LITTLE_ENDIAN ;
/*
* Initialize the outgoing RPC data buffer with no memory .
*/
prs_init_empty ( & p - > out_data . rdata , p - > mem_ctx , MARSHALL ) ;
p - > syntax = * syntax ;
DEBUG ( 4 , ( " Created internal pipe %s (pipes_open=%d) \n " ,
get_pipe_name_from_syntax ( talloc_tos ( ) , syntax ) , pipes_open ) ) ;
talloc_set_destructor ( p , close_internal_rpc_pipe_hnd ) ;
return p ;
}
2010-07-02 10:17:44 +02:00
/****************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS internal_ndr_push ( TALLOC_CTX * mem_ctx ,
struct rpc_pipe_client * cli ,
const struct ndr_interface_table * table ,
uint32_t opnum ,
void * r )
{
const struct ndr_interface_call * call ;
struct ndr_push * push ;
enum ndr_err_code ndr_err ;
DATA_BLOB blob ;
bool ret ;
if ( ! ndr_syntax_id_equal ( & table - > syntax_id , & cli - > abstract_syntax ) | |
( opnum > = table - > num_calls ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
call = & table - > calls [ opnum ] ;
if ( DEBUGLEVEL > = 10 ) {
ndr_print_function_debug ( call - > ndr_print ,
call - > name , NDR_IN , r ) ;
}
push = ndr_push_init_ctx ( mem_ctx ) ;
if ( push = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
ndr_err = call - > ndr_push ( push , NDR_IN , r ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
TALLOC_FREE ( push ) ;
return ndr_map_error2ntstatus ( ndr_err ) ;
}
blob = ndr_push_blob ( push ) ;
ret = prs_init_data_blob ( & cli - > pipes_struct - > in_data . data , & blob , mem_ctx ) ;
TALLOC_FREE ( push ) ;
if ( ! ret ) {
return NT_STATUS_NO_MEMORY ;
}
return NT_STATUS_OK ;
}
/****************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS internal_ndr_pull ( TALLOC_CTX * mem_ctx ,
struct rpc_pipe_client * cli ,
const struct ndr_interface_table * table ,
uint32_t opnum ,
void * r )
{
const struct ndr_interface_call * call ;
struct ndr_pull * pull ;
enum ndr_err_code ndr_err ;
DATA_BLOB blob ;
bool ret ;
if ( ! ndr_syntax_id_equal ( & table - > syntax_id , & cli - > abstract_syntax ) | |
( opnum > = table - > num_calls ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
call = & table - > calls [ opnum ] ;
ret = prs_data_blob ( & cli - > pipes_struct - > out_data . rdata , & blob , mem_ctx ) ;
if ( ! ret ) {
return NT_STATUS_NO_MEMORY ;
}
pull = ndr_pull_init_blob ( & blob , mem_ctx ) ;
if ( pull = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
/* have the ndr parser alloc memory for us */
pull - > flags | = LIBNDR_FLAG_REF_ALLOC ;
ndr_err = call - > ndr_pull ( pull , NDR_OUT , r ) ;
TALLOC_FREE ( pull ) ;
if ( ! NDR_ERR_CODE_IS_SUCCESS ( ndr_err ) ) {
return ndr_map_error2ntstatus ( ndr_err ) ;
}
if ( DEBUGLEVEL > = 10 ) {
ndr_print_function_debug ( call - > ndr_print ,
call - > name , NDR_OUT , r ) ;
}
return NT_STATUS_OK ;
}
/****************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS rpc_pipe_internal_dispatch ( struct rpc_pipe_client * cli ,
TALLOC_CTX * mem_ctx ,
const struct ndr_interface_table * table ,
uint32_t opnum , void * r )
{
NTSTATUS status ;
int num_cmds = rpc_srv_get_pipe_num_cmds ( & table - > syntax_id ) ;
const struct api_struct * cmds = rpc_srv_get_pipe_cmds ( & table - > syntax_id ) ;
int i ;
if ( cli - > pipes_struct = = NULL ) {
return NT_STATUS_INVALID_PARAMETER ;
}
2010-07-09 19:35:22 -04:00
/* set opnum */
cli - > pipes_struct - > opnum = opnum ;
2010-07-02 10:17:44 +02:00
for ( i = 0 ; i < num_cmds ; i + + ) {
if ( cmds [ i ] . opnum = = opnum & & cmds [ i ] . fn ! = NULL ) {
break ;
}
}
if ( i = = num_cmds ) {
return NT_STATUS_INVALID_PARAMETER ;
}
prs_init_empty ( & cli - > pipes_struct - > out_data . rdata , cli - > pipes_struct - > mem_ctx , MARSHALL ) ;
status = internal_ndr_push ( mem_ctx , cli , table , opnum , r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( ! cmds [ i ] . fn ( cli - > pipes_struct ) ) {
return NT_STATUS_UNSUCCESSFUL ;
}
status = internal_ndr_pull ( mem_ctx , cli , table , opnum , r ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
prs_mem_free ( & cli - > pipes_struct - > in_data . data ) ;
prs_mem_free ( & cli - > pipes_struct - > out_data . rdata ) ;
return NT_STATUS_OK ;
}
2010-06-03 22:04:08 +02:00
/**
* @ brief Create a new RPC client context which uses a local dispatch function .
*
* @ param [ in ] mem_ctx The memory context to use .
*
* @ param [ in ] abstract_syntax Normally the syntax_id of the autogenerated
* ndr_table_ < name > .
*
* @ param [ in ] dispatch The corresponding autogenerated dispatch function
* rpc_ < name > _dispatch .
*
* @ param [ in ] serversupplied_info The server supplied authentication function .
*
* @ param [ out ] presult A pointer to store the connected rpc client pipe .
*
* @ return NT_STATUS_OK on success , a corresponding NT status if an
* error occured .
*
* @ code
* struct rpc_pipe_client * winreg_pipe ;
* NTSTATUS status ;
*
* status = rpc_pipe_open_internal ( tmp_ctx ,
* & ndr_table_winreg . syntax_id ,
* rpc_winreg_dispatch ,
* p - > server_info ,
* & winreg_pipe ) ;
* @ endcode
*/
NTSTATUS rpc_pipe_open_internal ( TALLOC_CTX * mem_ctx ,
const struct ndr_syntax_id * abstract_syntax ,
struct auth_serversupplied_info * serversupplied_info ,
struct rpc_pipe_client * * presult )
{
struct rpc_pipe_client * result ;
result = TALLOC_ZERO_P ( mem_ctx , struct rpc_pipe_client ) ;
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > abstract_syntax = * abstract_syntax ;
result - > transfer_syntax = ndr_transfer_syntax ;
2010-07-02 10:17:44 +02:00
result - > dispatch = rpc_pipe_internal_dispatch ;
2010-06-03 22:04:08 +02:00
result - > pipes_struct = make_internal_rpc_pipe_p (
result , abstract_syntax , " " , serversupplied_info ) ;
if ( result - > pipes_struct = = NULL ) {
TALLOC_FREE ( result ) ;
return NT_STATUS_NO_MEMORY ;
}
result - > max_xmit_frag = - 1 ;
result - > max_recv_frag = - 1 ;
* presult = result ;
return NT_STATUS_OK ;
}