1998-03-11 21:11:04 +00:00
/*
2002-01-30 06:08:46 +00:00
* Unix SMB / CIFS implementation .
1998-03-11 21:11:04 +00:00
* RPC Pipe client / server routines
* Copyright ( C ) Andrew Tridgell 1992 - 1997 ,
* Copyright ( C ) Luke Kenneth Casson Leighton 1996 - 1997 ,
2001-03-11 00:32:10 +00:00
* Copyright ( C ) Jeremy Allison 2001.
1998-03-11 21:11:04 +00: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 19:25:36 +00:00
* the Free Software Foundation ; either version 3 of the License , or
1998-03-11 21:11:04 +00: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 05:23:25 +00:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
1998-03-11 21:11:04 +00:00
*/
# include "includes.h"
2002-07-15 10:35:28 +00:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_RPC_SRV
2009-04-18 16:10:57 +02:00
/*
* Handle database - stored per pipe .
*/
struct policy {
struct policy * next , * prev ;
struct policy_handle pol_hnd ;
2009-04-18 16:46:53 +02:00
uint32_t access_granted ;
2009-04-18 16:10:57 +02:00
void * data_ptr ;
} ;
struct handle_list {
struct policy * Policy ; /* List of policies. */
size_t count ; /* Current number of handles. */
size_t pipe_ref_count ; /* Number of pipe handles referring to this list. */
} ;
2001-03-13 20:18:45 +00:00
/* This is the max handles across all instances of a pipe name. */
1998-03-11 21:11:04 +00:00
# ifndef MAX_OPEN_POLS
2009-05-29 12:41:41 +02:00
# define MAX_OPEN_POLS 2048
1998-03-11 21:11:04 +00:00
# endif
2002-03-30 00:45:26 +00:00
/****************************************************************************
Hack as handles need to be persisant over lsa pipe closes so long as a samr
pipe is open . JRA .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-02-01 12:03:31 +01:00
static bool is_samr_lsa_pipe ( const struct ndr_syntax_id * syntax )
2002-03-30 00:45:26 +00:00
{
2009-02-01 12:03:31 +01: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 00:45:26 +00:00
}
2009-04-18 16:10:57 +02:00
size_t num_pipe_handles ( struct handle_list * list )
{
if ( list = = NULL ) {
return 0 ;
}
return list - > count ;
}
1998-03-11 21:11:04 +00:00
/****************************************************************************
2001-03-13 20:18:45 +00:00
Initialise a policy handle list on a pipe . Handle list is shared between all
pipes of the same name .
1998-03-11 21:11:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-03-11 00:32:10 +00:00
2009-02-01 12:03:31 +01:00
bool init_pipe_handle_list ( pipes_struct * p , const struct ndr_syntax_id * syntax )
1998-03-11 21:11:04 +00:00
{
2009-01-07 17:36:54 +01:00
pipes_struct * plist ;
struct handle_list * hl ;
2001-03-13 20:18:45 +00:00
2009-01-07 17:36:54 +01:00
for ( plist = get_first_internal_pipe ( ) ;
plist ;
plist = get_next_internal_pipe ( plist ) ) {
2009-11-08 19:37:42 +01:00
if ( ndr_syntax_id_equal ( syntax , & plist - > syntax ) ) {
2009-01-07 17:36:54 +01:00
break ;
}
2009-11-08 19:37:42 +01:00
if ( is_samr_lsa_pipe ( & plist - > syntax )
2009-02-01 12:03:31 +01:00
& & is_samr_lsa_pipe ( syntax ) ) {
2009-01-07 17:36:54 +01:00
/*
* samr and lsa share a handle space ( same process
* under Windows ? )
*/
2001-03-13 20:18:45 +00:00
break ;
}
}
2009-01-07 17:36:54 +01:00
if ( plist ! = NULL ) {
hl = plist - > pipe_handles ;
if ( hl = = NULL ) {
return false ;
}
} else {
2001-03-13 20:18:45 +00:00
/*
2009-01-07 17:36:54 +01:00
* First open , we have to create the handle list
2001-03-13 20:18:45 +00:00
*/
2009-01-07 17:36:54 +01:00
hl = SMB_MALLOC_P ( struct handle_list ) ;
if ( hl = = NULL ) {
return false ;
}
2001-03-13 20:18:45 +00:00
ZERO_STRUCTP ( hl ) ;
2010-01-31 15:38:16 +01:00
DEBUG ( 10 , ( " init_pipe_handle_list: created handle list for "
2009-10-31 19:28:16 +01:00
" pipe %s \n " ,
get_pipe_name_from_syntax ( talloc_tos ( ) , syntax ) ) ) ;
2001-03-13 20:18:45 +00: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 15:38:16 +01: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 19:28:16 +01:00
get_pipe_name_from_syntax ( talloc_tos ( ) , syntax ) ) ) ;
2001-03-13 20:18:45 +00:00
return True ;
1998-03-11 21:11:04 +00:00
}
/****************************************************************************
find first available policy slot . creates a policy handle for you .
2009-01-07 18:44:52 +01: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-11 21:11:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-03-11 00:32:10 +00:00
2009-04-18 16:46:53 +02:00
static struct policy * create_policy_hnd_internal ( pipes_struct * p ,
struct policy_handle * hnd ,
void * data_ptr )
1998-03-11 21:11:04 +00:00
{
2001-03-11 00:32:10 +00:00
static uint32 pol_hnd_low = 0 ;
static uint32 pol_hnd_high = 0 ;
2006-09-11 22:02:34 +00:00
time_t t = time ( NULL ) ;
1998-03-11 21:11:04 +00:00
2001-03-11 00:32:10 +00:00
struct policy * pol ;
1998-03-11 21:11:04 +00:00
2001-03-13 20:18:45 +00: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 16:46:53 +02:00
return NULL ;
1998-08-17 04:54:06 +00:00
}
1998-03-11 21:11:04 +00:00
2009-01-07 18:44:52 +01:00
pol = TALLOC_ZERO_P ( NULL , struct policy ) ;
2001-03-11 00:32:10 +00:00
if ( ! pol ) {
DEBUG ( 0 , ( " create_policy_hnd: ERROR: out of memory! \n " ) ) ;
2009-04-18 16:46:53 +02:00
return NULL ;
1998-03-11 21:11:04 +00:00
}
2009-01-07 18:44:52 +01:00
if ( data_ptr ! = NULL ) {
pol - > data_ptr = talloc_move ( pol , & data_ptr ) ;
}
1998-03-11 21:11:04 +00:00
2002-01-22 19:45:17 +00:00
pol_hnd_low + + ;
if ( pol_hnd_low = = 0 )
( pol_hnd_high ) + + ;
2006-09-11 22:02:34 +00:00
SIVAL ( & pol - > pol_hnd . handle_type , 0 , 0 ) ; /* first bit must be null */
SIVAL ( & pol - > pol_hnd . uuid . time_low , 0 , pol_hnd_low ) ; /* second bit is incrementing */
SSVAL ( & pol - > pol_hnd . uuid . time_mid , 0 , pol_hnd_high ) ; /* second bit is incrementing */
SSVAL ( & pol - > pol_hnd . uuid . time_hi_and_version , 0 , ( pol_hnd_high > > 16 ) ) ; /* second bit is incrementing */
/* split the current time into two 16 bit values */
SSVAL ( pol - > pol_hnd . uuid . clock_seq , 0 , ( t > > 16 ) ) ; /* something random */
SSVAL ( pol - > pol_hnd . uuid . node , 0 , t ) ; /* something random */
SIVAL ( pol - > pol_hnd . uuid . node , 2 , sys_getpid ( ) ) ; /* something more random */
1998-08-17 04:54:06 +00:00
2001-03-13 20:18:45 +00:00
DLIST_ADD ( p - > pipe_handles - > Policy , pol ) ;
p - > pipe_handles - > count + + ;
1998-08-17 04:54:06 +00:00
2001-03-11 00:32:10 +00:00
* hnd = pol - > pol_hnd ;
1998-08-17 04:54:06 +00:00
2001-03-13 20:18:45 +00:00
DEBUG ( 4 , ( " Opened policy hnd[%d] " , ( int ) p - > pipe_handles - > count ) ) ;
2007-03-28 13:34:59 +00:00
dump_data ( 4 , ( uint8 * ) hnd , sizeof ( * hnd ) ) ;
1998-08-17 04:54:06 +00:00
2009-04-18 16:46:53 +02:00
return pol ;
}
bool create_policy_hnd ( pipes_struct * p , struct policy_handle * hnd ,
void * data_ptr )
{
return create_policy_hnd_internal ( p , hnd , data_ptr ) ! = NULL ;
1998-03-11 21:11:04 +00:00
}
/****************************************************************************
2001-03-11 00:32:10 +00:00
find policy by handle - internal version .
1998-03-11 21:11:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-04-18 13:30:38 +02:00
static struct policy * find_policy_by_hnd_internal ( pipes_struct * p ,
const struct policy_handle * hnd ,
void * * data_p )
2001-03-11 00:32:10 +00:00
{
struct policy * pol ;
size_t i ;
if ( data_p )
* data_p = NULL ;
2001-03-13 20:18:45 +00:00
for ( i = 0 , pol = p - > pipe_handles - > Policy ; pol ; pol = pol - > next , i + + ) {
2001-03-11 00:32:10 +00:00
if ( memcmp ( & pol - > pol_hnd , hnd , sizeof ( * hnd ) ) = = 0 ) {
DEBUG ( 4 , ( " Found policy hnd[%d] " , ( int ) i ) ) ;
2007-03-28 13:34:59 +00:00
dump_data ( 4 , ( uint8 * ) hnd , sizeof ( * hnd ) ) ;
2001-03-11 00:32:10 +00:00
if ( data_p )
* data_p = pol - > data_ptr ;
return pol ;
1998-03-11 21:11:04 +00:00
}
}
DEBUG ( 4 , ( " Policy not found: " ) ) ;
2007-03-28 13:34:59 +00:00
dump_data ( 4 , ( uint8 * ) hnd , sizeof ( * hnd ) ) ;
1998-03-11 21:11:04 +00:00
2002-01-25 05:28:37 +00:00
p - > bad_handle_fault_state = True ;
1998-08-17 04:54:06 +00:00
return NULL ;
}
/****************************************************************************
2001-03-11 00:32:10 +00:00
find policy by handle
1998-08-17 04:54:06 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-04-18 13:30:38 +02:00
bool find_policy_by_hnd ( pipes_struct * p , const struct policy_handle * hnd ,
void * * data_p )
1998-03-11 21:11:04 +00:00
{
2001-03-11 00:32:10 +00:00
return find_policy_by_hnd_internal ( p , hnd , data_p ) = = NULL ? False : True ;
1998-03-11 21:11:04 +00:00
}
/****************************************************************************
2001-03-11 00:32:10 +00:00
Close a policy .
1998-03-11 21:11:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-08-17 04:54:06 +00:00
2009-03-18 22:49:41 +01:00
bool close_policy_hnd ( pipes_struct * p , struct policy_handle * hnd )
1998-10-21 22:36:26 +00:00
{
2001-03-11 00:32:10 +00:00
struct policy * pol = find_policy_by_hnd_internal ( p , hnd , NULL ) ;
1998-10-21 22:36:26 +00:00
2001-03-11 00:32:10 +00:00
if ( ! pol ) {
DEBUG ( 3 , ( " Error closing policy \n " ) ) ;
return False ;
1998-10-21 22:36:26 +00:00
}
2001-03-11 00:32:10 +00:00
DEBUG ( 3 , ( " Closed policy \n " ) ) ;
1998-10-21 22:36:26 +00:00
2001-04-02 22:27:40 +00:00
p - > pipe_handles - > count - - ;
1998-03-11 21:11:04 +00:00
2001-04-02 22:27:40 +00:00
DLIST_REMOVE ( p - > pipe_handles - > Policy , pol ) ;
1998-08-17 04:54:06 +00:00
2009-01-07 18:44:52 +01:00
TALLOC_FREE ( pol ) ;
1999-11-18 22:03:47 +00:00
2001-03-11 00:32:10 +00:00
return True ;
1999-11-24 18:09:33 +00:00
}
1998-03-11 21:11:04 +00:00
/****************************************************************************
2001-03-13 20:18:45 +00:00
Close a pipe - free the handle list if it was the last pipe reference .
1998-03-11 21:11:04 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-03-11 00:32:10 +00:00
void close_policy_by_pipe ( pipes_struct * p )
{
2001-03-13 20:18:45 +00:00
p - > pipe_handles - > pipe_ref_count - - ;
if ( p - > pipe_handles - > pipe_ref_count = = 0 ) {
/*
* Last pipe open on this list - free the list .
*/
while ( p - > pipe_handles - > Policy )
close_policy_hnd ( p , & p - > pipe_handles - > Policy - > pol_hnd ) ;
1998-08-17 04:54:06 +00:00
2001-03-13 20:18:45 +00:00
p - > pipe_handles - > Policy = NULL ;
p - > pipe_handles - > count = 0 ;
2001-09-17 10:26:23 +00:00
SAFE_FREE ( p - > pipe_handles ) ;
2009-02-01 12:03:31 +01:00
DEBUG ( 10 , ( " close_policy_by_pipe: deleted handle list for "
2009-11-08 19:37:53 +01:00
" pipe %s \n " ,
get_pipe_name_from_syntax ( talloc_tos ( ) , & p - > syntax ) ) ) ;
2001-03-13 20:18:45 +00:00
}
1998-03-11 21:11:04 +00:00
}
2002-07-15 10:35:28 +00: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 .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool pipe_access_check ( pipes_struct * p )
2002-07-15 10:35:28 +00:00
{
/* Don't let anonymous users access this RPC if restrict
anonymous > 0 */
if ( lp_restrict_anonymous ( ) > 0 ) {
2005-02-19 11:09:52 +00:00
/* schannel, so we must be ok */
2005-09-30 17:13:37 +00:00
if ( p - > pipe_bound & & ( p - > auth . auth_type = = PIPE_AUTH_TYPE_SCHANNEL ) ) {
2005-02-19 11:09:52 +00:00
return True ;
2005-09-30 17:13:37 +00:00
}
2005-02-19 11:09:52 +00:00
2008-06-24 15:24:08 +02:00
if ( p - > server_info - > guest ) {
2002-07-15 10:35:28 +00:00
return False ;
2005-09-30 17:13:37 +00:00
}
2002-07-15 10:35:28 +00:00
}
return True ;
}
2009-04-18 13:31:20 +02:00
2009-04-18 16:46:53 +02: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 13:31:20 +02:00
{
2009-04-18 16:46:53 +02:00
struct policy * pol ;
2009-04-18 13:31:20 +02: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 19:37:53 +01:00
get_pipe_name_from_syntax ( talloc_tos ( ) , & p - > syntax ) ) ) ;
2009-04-18 16:46:53 +02:00
* pstatus = NT_STATUS_INSUFFICIENT_RESOURCES ;
return NULL ;
2009-04-18 13:31:20 +02:00
}
data = talloc_size ( talloc_tos ( ) , data_size ) ;
if ( data = = NULL ) {
2009-04-18 16:46:53 +02:00
* pstatus = NT_STATUS_NO_MEMORY ;
return NULL ;
2009-04-18 13:31:20 +02:00
}
2009-04-20 16:50:14 +02:00
talloc_set_name_const ( data , type ) ;
2009-04-18 13:31:20 +02:00
2009-04-18 16:46:53 +02:00
pol = create_policy_hnd_internal ( p , hnd , data ) ;
if ( pol = = NULL ) {
2009-04-18 13:31:20 +02:00
TALLOC_FREE ( data ) ;
2009-04-18 16:46:53 +02:00
* pstatus = NT_STATUS_NO_MEMORY ;
return NULL ;
2009-04-18 13:31:20 +02:00
}
2009-04-18 16:46:53 +02:00
pol - > access_granted = access_granted ;
* pstatus = NT_STATUS_OK ;
return data ;
2009-04-18 13:31:20 +02:00
}
void * _policy_handle_find ( struct pipes_struct * p ,
const struct policy_handle * hnd ,
2009-04-18 16:46:53 +02:00
uint32_t access_required ,
uint32_t * paccess_granted ,
const char * name , const char * location ,
NTSTATUS * pstatus )
2009-04-18 13:31:20 +02:00
{
2009-04-18 16:46:53 +02:00
struct policy * pol ;
2009-04-18 13:31:20 +02:00
void * data ;
2009-04-18 16:46:53 +02:00
pol = find_policy_by_hnd_internal ( p , hnd , & data ) ;
if ( pol = = NULL ) {
* pstatus = NT_STATUS_INVALID_HANDLE ;
2009-04-18 13:31:20 +02: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 16:46:53 +02:00
* pstatus = NT_STATUS_INVALID_HANDLE ;
2009-04-18 13:31:20 +02:00
return NULL ;
}
2009-04-18 16:46:53 +02:00
if ( ( access_required & pol - > access_granted ) ! = access_required ) {
if ( geteuid ( ) = = sec_initial_uid ( ) ) {
DEBUG ( 4 , ( " %s: ACCESS should be DENIED (granted: "
" %#010x; required: %#010x) \n " , location ,
pol - > access_granted , access_required ) ) ;
DEBUGADD ( 4 , ( " but overwritten by euid == 0 \n " ) ) ;
goto okay ;
}
DEBUG ( 2 , ( " %s: ACCESS DENIED (granted: %#010x; required: "
" %#010x) \n " , location , pol - > access_granted ,
access_required ) ) ;
* pstatus = NT_STATUS_ACCESS_DENIED ;
return NULL ;
}
okay :
2009-04-18 13:31:20 +02:00
DEBUG ( 10 , ( " found handle of type %s \n " , talloc_get_name ( data ) ) ) ;
2009-04-18 16:46:53 +02:00
if ( paccess_granted ! = NULL ) {
* paccess_granted = pol - > access_granted ;
}
* pstatus = NT_STATUS_OK ;
2009-04-18 13:31:20 +02:00
return data ;
}