1998-03-12 00:11:04 +03:00
/*
2002-01-30 09:08:46 +03:00
* Unix SMB / CIFS implementation .
1998-03-12 00:11:04 +03:00
* RPC Pipe client / server routines
* Copyright ( C ) Andrew Tridgell 1992 - 1997 ,
* Copyright ( C ) Luke Kenneth Casson Leighton 1996 - 1997 ,
2001-03-11 03:32:10 +03:00
* Copyright ( C ) Jeremy Allison 2001.
1998-03-12 00:11:04 +03:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
2007-07-09 23:25:36 +04:00
* the Free Software Foundation ; either version 3 of the License , or
1998-03-12 00:11:04 +03:00
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
2007-07-10 09:23:25 +04:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
1998-03-12 00:11:04 +03:00
*/
# include "includes.h"
2010-05-05 03:39:16 +04:00
# include "../librpc/gen_ndr/ndr_lsa.h"
2010-05-28 04:18:21 +04:00
# include "../librpc/gen_ndr/ndr_samr.h"
2011-03-24 14:08:15 +03:00
# include "auth.h"
1998-03-12 00:11:04 +03:00
2002-07-15 14:35:28 +04:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_SRV
2009-04-18 18:10:57 +04:00
/*
* Handle database - stored per pipe .
*/
2010-06-08 01:38:01 +04:00
struct dcesrv_handle {
struct dcesrv_handle * prev , * next ;
struct policy_handle wire_handle ;
2009-04-18 18:46:53 +04:00
uint32_t access_granted ;
2010-06-08 01:38:01 +04:00
void * data ;
2009-04-18 18:10:57 +04:00
} ;
struct handle_list {
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * handles ; /* List of pipe handles. */
2009-04-18 18:10:57 +04:00
size_t count ; /* Current number of handles. */
2010-06-08 01:38:01 +04:00
size_t pipe_ref_count ; /* Number of pipe handles referring
* to this tree . */
2009-04-18 18:10:57 +04:00
} ;
2001-03-13 23:18:45 +03:00
/* This is the max handles across all instances of a pipe name. */
1998-03-12 00:11:04 +03:00
# ifndef MAX_OPEN_POLS
2009-05-29 14:41:41 +04:00
# define MAX_OPEN_POLS 2048
1998-03-12 00:11:04 +03:00
# endif
2002-03-30 03:45:26 +03:00
/****************************************************************************
Hack as handles need to be persisant over lsa pipe closes so long as a samr
pipe is open . JRA .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-02-01 14:03:31 +03:00
static bool is_samr_lsa_pipe ( const struct ndr_syntax_id * syntax )
2002-03-30 03:45:26 +03:00
{
2009-02-01 14:03:31 +03:00
return ( ndr_syntax_id_equal ( syntax , & ndr_table_samr . syntax_id )
| | ndr_syntax_id_equal ( syntax , & ndr_table_lsarpc . syntax_id ) ) ;
2002-03-30 03:45:26 +03:00
}
2010-07-28 11:46:43 +04:00
size_t num_pipe_handles ( struct pipes_struct * p )
2009-04-18 18:10:57 +04:00
{
2010-06-08 00:02:14 +04:00
if ( p - > pipe_handles = = NULL ) {
2009-04-18 18:10:57 +04:00
return 0 ;
}
2010-06-08 00:02:14 +04:00
return p - > pipe_handles - > count ;
2009-04-18 18:10:57 +04:00
}
1998-03-12 00:11:04 +03:00
/****************************************************************************
2001-03-13 23:18:45 +03:00
Initialise a policy handle list on a pipe . Handle list is shared between all
pipes of the same name .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-03-11 03:32:10 +03:00
2010-07-28 11:46:43 +04:00
bool init_pipe_handles ( struct pipes_struct * p , const struct ndr_syntax_id * syntax )
1998-03-12 00:11:04 +03:00
{
2010-07-28 11:46:43 +04:00
struct pipes_struct * plist ;
2009-01-07 19:36:54 +03:00
struct handle_list * hl ;
2001-03-13 23:18:45 +03:00
2009-01-07 19:36:54 +03:00
for ( plist = get_first_internal_pipe ( ) ;
plist ;
plist = get_next_internal_pipe ( plist ) ) {
2009-11-08 21:37:42 +03:00
if ( ndr_syntax_id_equal ( syntax , & plist - > syntax ) ) {
2009-01-07 19:36:54 +03:00
break ;
}
2009-11-08 21:37:42 +03:00
if ( is_samr_lsa_pipe ( & plist - > syntax )
2009-02-01 14:03:31 +03:00
& & is_samr_lsa_pipe ( syntax ) ) {
2009-01-07 19:36:54 +03:00
/*
* samr and lsa share a handle space ( same process
* under Windows ? )
*/
2001-03-13 23:18:45 +03:00
break ;
}
}
2009-01-07 19:36:54 +03:00
if ( plist ! = NULL ) {
hl = plist - > pipe_handles ;
if ( hl = = NULL ) {
return false ;
}
} else {
2001-03-13 23:18:45 +03:00
/*
2009-01-07 19:36:54 +03:00
* First open , we have to create the handle list
2001-03-13 23:18:45 +03:00
*/
2010-07-14 16:57:47 +04:00
hl = talloc_zero ( NULL , struct handle_list ) ;
2009-01-07 19:36:54 +03:00
if ( hl = = NULL ) {
return false ;
}
2001-03-13 23:18:45 +03:00
2010-01-31 17:38:16 +03:00
DEBUG ( 10 , ( " init_pipe_handle_list: created handle list for "
2009-10-31 21:28:16 +03:00
" pipe %s \n " ,
get_pipe_name_from_syntax ( talloc_tos ( ) , syntax ) ) ) ;
2001-03-13 23:18:45 +03:00
}
/*
* One more pipe is using this list .
*/
hl - > pipe_ref_count + + ;
/*
* Point this pipe at this list .
*/
p - > pipe_handles = hl ;
2010-01-31 17:38:16 +03:00
DEBUG ( 10 , ( " init_pipe_handle_list: pipe_handles ref count = %lu for "
" pipe %s \n " , ( unsigned long ) p - > pipe_handles - > pipe_ref_count ,
2009-10-31 21:28:16 +03:00
get_pipe_name_from_syntax ( talloc_tos ( ) , syntax ) ) ) ;
2001-03-13 23:18:45 +03:00
return True ;
1998-03-12 00:11:04 +03:00
}
/****************************************************************************
find first available policy slot . creates a policy handle for you .
2009-01-07 20:44:52 +03:00
If " data_ptr " is given , this must be a talloc ' ed object , create_policy_hnd
talloc_moves this into the handle . If the policy_hnd is closed ,
data_ptr is TALLOC_FREE ( ) ' ed
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-03-11 03:32:10 +03:00
2010-07-28 11:46:43 +04:00
static struct dcesrv_handle * create_rpc_handle_internal ( struct pipes_struct * p ,
2010-06-08 01:38:01 +04:00
struct policy_handle * hnd , void * data_ptr )
1998-03-12 00:11:04 +03:00
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * rpc_hnd ;
2001-03-11 03:32:10 +03:00
static uint32 pol_hnd_low = 0 ;
static uint32 pol_hnd_high = 0 ;
2006-09-12 02:02:34 +04:00
time_t t = time ( NULL ) ;
1998-03-12 00:11:04 +03:00
2001-03-13 23:18:45 +03:00
if ( p - > pipe_handles - > count > MAX_OPEN_POLS ) {
DEBUG ( 0 , ( " create_policy_hnd: ERROR: too many handles (%d) on this pipe. \n " ,
( int ) p - > pipe_handles - > count ) ) ;
2009-04-18 18:46:53 +04:00
return NULL ;
1998-08-17 08:54:06 +04:00
}
1998-03-12 00:11:04 +03:00
2010-06-08 01:38:01 +04:00
rpc_hnd = talloc_zero ( p - > pipe_handles , struct dcesrv_handle ) ;
if ( ! rpc_hnd ) {
2001-03-11 03:32:10 +03:00
DEBUG ( 0 , ( " create_policy_hnd: ERROR: out of memory! \n " ) ) ;
2009-04-18 18:46:53 +04:00
return NULL ;
1998-03-12 00:11:04 +03:00
}
2009-01-07 20:44:52 +03:00
if ( data_ptr ! = NULL ) {
2010-06-08 01:38:01 +04:00
rpc_hnd - > data = talloc_move ( rpc_hnd , & data_ptr ) ;
2009-01-07 20:44:52 +03:00
}
1998-03-12 00:11:04 +03:00
2002-01-22 22:45:17 +03:00
pol_hnd_low + + ;
2010-06-08 01:38:01 +04:00
if ( pol_hnd_low = = 0 ) {
pol_hnd_high + + ;
}
2002-01-22 22:45:17 +03:00
2010-06-08 01:38:01 +04:00
/* first bit must be null */
SIVAL ( & rpc_hnd - > wire_handle . handle_type , 0 , 0 ) ;
2006-09-12 02:02:34 +04:00
2010-06-08 01:38:01 +04:00
/* second bit is incrementing */
SIVAL ( & rpc_hnd - > wire_handle . uuid . time_low , 0 , pol_hnd_low ) ;
SSVAL ( & rpc_hnd - > wire_handle . uuid . time_mid , 0 , pol_hnd_high ) ;
SSVAL ( & rpc_hnd - > wire_handle . uuid . time_hi_and_version , 0 , ( pol_hnd_high > > 16 ) ) ;
2006-09-12 02:02:34 +04:00
2010-06-08 01:38:01 +04:00
/* split the current time into two 16 bit values */
2006-09-12 02:02:34 +04:00
2010-06-08 01:38:01 +04:00
/* something random */
SSVAL ( rpc_hnd - > wire_handle . uuid . clock_seq , 0 , ( t > > 16 ) ) ;
/* something random */
SSVAL ( rpc_hnd - > wire_handle . uuid . node , 0 , t ) ;
/* something more random */
SIVAL ( rpc_hnd - > wire_handle . uuid . node , 2 , sys_getpid ( ) ) ;
1998-08-17 08:54:06 +04:00
2010-06-08 01:38:01 +04:00
DLIST_ADD ( p - > pipe_handles - > handles , rpc_hnd ) ;
2001-03-13 23:18:45 +03:00
p - > pipe_handles - > count + + ;
1998-08-17 08:54:06 +04:00
2010-06-08 01:38:01 +04:00
* hnd = rpc_hnd - > wire_handle ;
DEBUG ( 4 , ( " Opened policy hnd[%d] " , ( int ) p - > pipe_handles - > count ) ) ;
dump_data ( 4 , ( uint8_t * ) hnd , sizeof ( * hnd ) ) ;
1998-08-17 08:54:06 +04:00
2010-06-08 01:38:01 +04:00
return rpc_hnd ;
2009-04-18 18:46:53 +04:00
}
2010-07-28 11:46:43 +04:00
bool create_policy_hnd ( struct pipes_struct * p , struct policy_handle * hnd ,
2009-04-18 18:46:53 +04:00
void * data_ptr )
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * rpc_hnd ;
rpc_hnd = create_rpc_handle_internal ( p , hnd , data_ptr ) ;
if ( rpc_hnd = = NULL ) {
return false ;
}
return true ;
1998-03-12 00:11:04 +03:00
}
/****************************************************************************
2001-03-11 03:32:10 +03:00
find policy by handle - internal version .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-07-28 11:46:43 +04:00
static struct dcesrv_handle * find_policy_by_hnd_internal ( struct pipes_struct * p ,
2010-06-08 01:38:01 +04:00
const struct policy_handle * hnd , void * * data_p )
2001-03-11 03:32:10 +03:00
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * h ;
unsigned int count ;
2001-03-11 03:32:10 +03:00
2010-06-08 01:38:01 +04:00
if ( data_p ) {
2001-03-11 03:32:10 +03:00
* data_p = NULL ;
2010-06-08 01:38:01 +04:00
}
2001-03-11 03:32:10 +03:00
2010-06-08 01:38:01 +04:00
count = 0 ;
for ( h = p - > pipe_handles - > handles ; h ! = NULL ; h = h - > next ) {
if ( memcmp ( & h - > wire_handle , hnd , sizeof ( * hnd ) ) = = 0 ) {
DEBUG ( 4 , ( " Found policy hnd[%u] " , count ) ) ;
2007-03-28 17:34:59 +04:00
dump_data ( 4 , ( uint8 * ) hnd , sizeof ( * hnd ) ) ;
2010-06-08 01:38:01 +04:00
if ( data_p ) {
* data_p = h - > data ;
}
return h ;
1998-03-12 00:11:04 +03:00
}
2010-06-08 01:38:01 +04:00
count + + ;
1998-03-12 00:11:04 +03:00
}
DEBUG ( 4 , ( " Policy not found: " ) ) ;
2010-06-08 01:38:01 +04:00
dump_data ( 4 , ( uint8_t * ) hnd , sizeof ( * hnd ) ) ;
1998-03-12 00:11:04 +03:00
2010-06-08 01:38:01 +04:00
p - > bad_handle_fault_state = true ;
2002-01-25 08:28:37 +03:00
1998-08-17 08:54:06 +04:00
return NULL ;
}
/****************************************************************************
2001-03-11 03:32:10 +03:00
find policy by handle
1998-08-17 08:54:06 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-07-28 11:46:43 +04:00
bool find_policy_by_hnd ( struct pipes_struct * p , const struct policy_handle * hnd ,
2009-04-18 15:30:38 +04:00
void * * data_p )
1998-03-12 00:11:04 +03:00
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * rpc_hnd ;
rpc_hnd = find_policy_by_hnd_internal ( p , hnd , data_p ) ;
if ( rpc_hnd = = NULL ) {
return false ;
}
return true ;
1998-03-12 00:11:04 +03:00
}
/****************************************************************************
2001-03-11 03:32:10 +03:00
Close a policy .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-08-17 08:54:06 +04:00
2010-07-28 11:46:43 +04:00
bool close_policy_hnd ( struct pipes_struct * p , struct policy_handle * hnd )
1998-10-22 02:36:26 +04:00
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * rpc_hnd ;
1998-10-22 02:36:26 +04:00
2010-06-08 01:38:01 +04:00
rpc_hnd = find_policy_by_hnd_internal ( p , hnd , NULL ) ;
if ( rpc_hnd = = NULL ) {
DEBUG ( 3 , ( " Error closing policy (policy not found) \n " ) ) ;
return false ;
1998-10-22 02:36:26 +04:00
}
2001-03-11 03:32:10 +03:00
DEBUG ( 3 , ( " Closed policy \n " ) ) ;
1998-10-22 02:36:26 +04:00
2001-04-03 02:27:40 +04:00
p - > pipe_handles - > count - - ;
1998-03-12 00:11:04 +03:00
2010-06-08 01:38:01 +04:00
DLIST_REMOVE ( p - > pipe_handles - > handles , rpc_hnd ) ;
TALLOC_FREE ( rpc_hnd ) ;
1999-11-19 01:03:47 +03:00
2010-06-08 01:38:01 +04:00
return true ;
1999-11-24 21:09:33 +03:00
}
1998-03-12 00:11:04 +03:00
/****************************************************************************
2010-06-08 01:38:01 +04:00
Close a pipe - free the handle set if it was the last pipe reference .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-07-28 11:46:43 +04:00
void close_policy_by_pipe ( struct pipes_struct * p )
2001-03-11 03:32:10 +03:00
{
2001-03-13 23:18:45 +03:00
p - > pipe_handles - > pipe_ref_count - - ;
if ( p - > pipe_handles - > pipe_ref_count = = 0 ) {
/*
* Last pipe open on this list - free the list .
*/
2010-06-08 01:38:01 +04:00
TALLOC_FREE ( p - > pipe_handles ) ;
2001-03-13 23:18:45 +03:00
2009-02-01 14:03:31 +03:00
DEBUG ( 10 , ( " close_policy_by_pipe: deleted handle list for "
2009-11-08 21:37:53 +03:00
" pipe %s \n " ,
get_pipe_name_from_syntax ( talloc_tos ( ) , & p - > syntax ) ) ) ;
2001-03-13 23:18:45 +03:00
}
1998-03-12 00:11:04 +03:00
}
2002-07-15 14:35:28 +04:00
/*******************************************************************
Shall we allow access to this rpc ? Currently this function
implements the ' restrict anonymous ' setting by denying access to
anonymous users if the restrict anonymous level is > 0. Further work
will be checking a security descriptor to determine whether a user
token has enough access to access the pipe .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-07-28 11:46:43 +04:00
bool pipe_access_check ( struct pipes_struct * p )
2002-07-15 14:35:28 +04:00
{
/* Don't let anonymous users access this RPC if restrict
anonymous > 0 */
if ( lp_restrict_anonymous ( ) > 0 ) {
2005-02-19 14:09:52 +03:00
/* schannel, so we must be ok */
2010-07-20 21:26:36 +04:00
if ( p - > pipe_bound & &
( p - > auth . auth_type = = DCERPC_AUTH_TYPE_SCHANNEL ) ) {
2005-02-19 14:09:52 +03:00
return True ;
2005-09-30 21:13:37 +04:00
}
2005-02-19 14:09:52 +03:00
2011-02-21 12:25:52 +03:00
if ( p - > session_info - > guest ) {
2002-07-15 14:35:28 +04:00
return False ;
2005-09-30 21:13:37 +04:00
}
2002-07-15 14:35:28 +04:00
}
return True ;
}
2009-04-18 15:31:20 +04:00
2009-04-18 18:46:53 +04:00
void * _policy_handle_create ( struct pipes_struct * p , struct policy_handle * hnd ,
uint32_t access_granted , size_t data_size ,
const char * type , NTSTATUS * pstatus )
2009-04-18 15:31:20 +04:00
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * rpc_hnd ;
2009-04-18 15:31:20 +04:00
void * data ;
if ( p - > pipe_handles - > count > MAX_OPEN_POLS ) {
DEBUG ( 0 , ( " policy_handle_create: ERROR: too many handles (%d) "
" on pipe %s. \n " , ( int ) p - > pipe_handles - > count ,
2009-11-08 21:37:53 +03:00
get_pipe_name_from_syntax ( talloc_tos ( ) , & p - > syntax ) ) ) ;
2009-04-18 18:46:53 +04:00
* pstatus = NT_STATUS_INSUFFICIENT_RESOURCES ;
return NULL ;
2009-04-18 15:31:20 +04:00
}
data = talloc_size ( talloc_tos ( ) , data_size ) ;
if ( data = = NULL ) {
2009-04-18 18:46:53 +04:00
* pstatus = NT_STATUS_NO_MEMORY ;
return NULL ;
2009-04-18 15:31:20 +04:00
}
2009-04-20 18:50:14 +04:00
talloc_set_name_const ( data , type ) ;
2009-04-18 15:31:20 +04:00
2010-06-08 01:38:01 +04:00
rpc_hnd = create_rpc_handle_internal ( p , hnd , data ) ;
if ( rpc_hnd = = NULL ) {
2009-04-18 15:31:20 +04:00
TALLOC_FREE ( data ) ;
2009-04-18 18:46:53 +04:00
* pstatus = NT_STATUS_NO_MEMORY ;
return NULL ;
2009-04-18 15:31:20 +04:00
}
2010-06-08 01:38:01 +04:00
rpc_hnd - > access_granted = access_granted ;
2009-04-18 18:46:53 +04:00
* pstatus = NT_STATUS_OK ;
return data ;
2009-04-18 15:31:20 +04:00
}
void * _policy_handle_find ( struct pipes_struct * p ,
const struct policy_handle * hnd ,
2009-04-18 18:46:53 +04:00
uint32_t access_required ,
uint32_t * paccess_granted ,
const char * name , const char * location ,
NTSTATUS * pstatus )
2009-04-18 15:31:20 +04:00
{
2010-06-08 01:38:01 +04:00
struct dcesrv_handle * rpc_hnd ;
2009-04-18 15:31:20 +04:00
void * data ;
2010-06-08 01:38:01 +04:00
rpc_hnd = find_policy_by_hnd_internal ( p , hnd , & data ) ;
if ( rpc_hnd = = NULL ) {
2009-04-18 18:46:53 +04:00
* pstatus = NT_STATUS_INVALID_HANDLE ;
2009-04-18 15:31:20 +04:00
return NULL ;
}
if ( strcmp ( name , talloc_get_name ( data ) ) ! = 0 ) {
DEBUG ( 10 , ( " expected %s, got %s \n " , name ,
talloc_get_name ( data ) ) ) ;
2009-04-18 18:46:53 +04:00
* pstatus = NT_STATUS_INVALID_HANDLE ;
2009-04-18 15:31:20 +04:00
return NULL ;
}
2010-06-08 01:38:01 +04:00
if ( ( access_required & rpc_hnd - > access_granted ) ! = access_required ) {
2009-04-18 18:46:53 +04:00
if ( geteuid ( ) = = sec_initial_uid ( ) ) {
DEBUG ( 4 , ( " %s: ACCESS should be DENIED (granted: "
" %#010x; required: %#010x) \n " , location ,
2010-06-08 01:38:01 +04:00
rpc_hnd - > access_granted , access_required ) ) ;
2009-04-18 18:46:53 +04:00
DEBUGADD ( 4 , ( " but overwritten by euid == 0 \n " ) ) ;
goto okay ;
}
DEBUG ( 2 , ( " %s: ACCESS DENIED (granted: %#010x; required: "
2010-06-08 01:38:01 +04:00
" %#010x) \n " , location , rpc_hnd - > access_granted ,
2009-04-18 18:46:53 +04:00
access_required ) ) ;
* pstatus = NT_STATUS_ACCESS_DENIED ;
return NULL ;
}
okay :
2009-04-18 15:31:20 +04:00
DEBUG ( 10 , ( " found handle of type %s \n " , talloc_get_name ( data ) ) ) ;
2009-04-18 18:46:53 +04:00
if ( paccess_granted ! = NULL ) {
2010-06-08 01:38:01 +04:00
* paccess_granted = rpc_hnd - > access_granted ;
2009-04-18 18:46:53 +04:00
}
* pstatus = NT_STATUS_OK ;
2009-04-18 15:31:20 +04:00
return data ;
}