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
2005-09-30 21:13:37 +04:00
* Largely rewritten by Jeremy Allison 2005.
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"
2008-05-20 20:08:41 +04:00
# include "librpc/gen_ndr/cli_epmapper.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_CLI
2008-07-19 22:40:27 +04:00
/*******************************************************************
interface / version dce / rpc pipe identification
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-19 23:07:14 +04:00
# define PIPE_SRVSVC "\\PIPE\\srvsvc"
# define PIPE_SAMR "\\PIPE\\samr"
# define PIPE_WINREG "\\PIPE\\winreg"
# define PIPE_WKSSVC "\\PIPE\\wkssvc"
# define PIPE_NETLOGON "\\PIPE\\NETLOGON"
# define PIPE_NTLSA "\\PIPE\\ntlsa"
# define PIPE_NTSVCS "\\PIPE\\ntsvcs"
# define PIPE_LSASS "\\PIPE\\lsass"
# define PIPE_LSARPC "\\PIPE\\lsarpc"
# define PIPE_SPOOLSS "\\PIPE\\spoolss"
# define PIPE_NETDFS "\\PIPE\\netdfs"
# define PIPE_ECHO "\\PIPE\\rpcecho"
# define PIPE_SHUTDOWN "\\PIPE\\initshutdown"
# define PIPE_EPM "\\PIPE\\epmapper"
# define PIPE_SVCCTL "\\PIPE\\svcctl"
# define PIPE_EVENTLOG "\\PIPE\\eventlog"
# define PIPE_EPMAPPER "\\PIPE\\epmapper"
# define PIPE_DRSUAPI "\\PIPE\\drsuapi"
2008-07-19 22:40:27 +04:00
/*
* IMPORTANT ! ! If you update this structure , make sure to
* update the index # defines in smb . h .
*/
2008-07-20 12:00:46 +04:00
static const struct pipe_id_info {
/* the names appear not to matter: the syntaxes _do_ matter */
const char * client_pipe ;
const RPC_IFACE * abstr_syntax ; /* this one is the abstract syntax id */
} pipe_names [ ] =
2008-07-19 22:40:27 +04:00
{
2008-07-20 12:00:46 +04:00
{ PIPE_LSARPC , & ndr_table_lsarpc . syntax_id } ,
{ PIPE_LSARPC , & ndr_table_dssetup . syntax_id } ,
{ PIPE_SAMR , & ndr_table_samr . syntax_id } ,
{ PIPE_NETLOGON , & ndr_table_netlogon . syntax_id } ,
{ PIPE_SRVSVC , & ndr_table_srvsvc . syntax_id } ,
{ PIPE_WKSSVC , & ndr_table_wkssvc . syntax_id } ,
{ PIPE_WINREG , & ndr_table_winreg . syntax_id } ,
{ PIPE_SPOOLSS , & syntax_spoolss } ,
{ PIPE_NETDFS , & ndr_table_netdfs . syntax_id } ,
{ PIPE_ECHO , & ndr_table_rpcecho . syntax_id } ,
{ PIPE_SHUTDOWN , & ndr_table_initshutdown . syntax_id } ,
{ PIPE_SVCCTL , & ndr_table_svcctl . syntax_id } ,
{ PIPE_EVENTLOG , & ndr_table_eventlog . syntax_id } ,
{ PIPE_NTSVCS , & ndr_table_ntsvcs . syntax_id } ,
{ PIPE_EPMAPPER , & ndr_table_epmapper . syntax_id } ,
{ PIPE_DRSUAPI , & ndr_table_drsuapi . syntax_id } ,
{ NULL , NULL }
2008-07-19 22:40:27 +04:00
} ;
/****************************************************************************
2008-07-20 20:48:44 +04:00
Return the pipe name from the interface .
2008-07-19 22:40:27 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 19:59:30 +04:00
const char * cli_get_pipe_name_from_iface ( TALLOC_CTX * mem_ctx ,
struct cli_state * cli ,
const struct ndr_syntax_id * interface )
2008-07-20 12:56:36 +04:00
{
int i ;
for ( i = 0 ; pipe_names [ i ] . client_pipe ; i + + ) {
if ( ndr_syntax_id_equal ( pipe_names [ i ] . abstr_syntax ,
interface ) ) {
2008-07-20 20:48:44 +04:00
return & pipe_names [ i ] . client_pipe [ 5 ] ;
2008-07-20 12:56:36 +04:00
}
}
/*
* Here we should ask \ \ epmapper , but for now our code is only
* interested in the known pipes mentioned in pipe_names [ ]
*/
return NULL ;
}
2005-09-30 21:13:37 +04:00
/********************************************************************
Map internal value to wire value .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-10-02 01:18:32 +04:00
2005-09-30 21:13:37 +04:00
static int map_pipe_auth_type_to_rpc_auth_type ( enum pipe_auth_type auth_type )
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
{
2005-09-30 21:13:37 +04:00
switch ( auth_type ) {
case PIPE_AUTH_TYPE_NONE :
return RPC_ANONYMOUS_AUTH_TYPE ;
case PIPE_AUTH_TYPE_NTLMSSP :
return RPC_NTLMSSP_AUTH_TYPE ;
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
case PIPE_AUTH_TYPE_SPNEGO_KRB5 :
return RPC_SPNEGO_AUTH_TYPE ;
case PIPE_AUTH_TYPE_SCHANNEL :
return RPC_SCHANNEL_AUTH_TYPE ;
case PIPE_AUTH_TYPE_KRB5 :
return RPC_KRB5_AUTH_TYPE ;
default :
DEBUG ( 0 , ( " map_pipe_auth_type_to_rpc_type: unknown pipe "
" auth type %u \n " ,
( unsigned int ) auth_type ) ) ;
break ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
2005-09-30 21:13:37 +04:00
return - 1 ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
2008-04-24 22:24:37 +04:00
/********************************************************************
Pipe description for a DEBUG
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static char * rpccli_pipe_txt ( TALLOC_CTX * mem_ctx , struct rpc_pipe_client * cli )
{
char * result ;
2008-05-13 01:37:07 +04:00
switch ( cli - > transport_type ) {
case NCACN_NP :
result = talloc_asprintf ( mem_ctx , " host %s, pipe %s, "
" fnum 0x%x " ,
cli - > desthost ,
cli - > trans . np . pipe_name ,
( unsigned int ) ( cli - > trans . np . fnum ) ) ;
break ;
case NCACN_IP_TCP :
case NCACN_UNIX_STREAM :
result = talloc_asprintf ( mem_ctx , " host %s, fd %d " ,
cli - > desthost , cli - > trans . sock . fd ) ;
break ;
default :
result = talloc_asprintf ( mem_ctx , " host %s " , cli - > desthost ) ;
break ;
}
2008-04-24 22:24:37 +04:00
SMB_ASSERT ( result ! = NULL ) ;
return result ;
}
1998-03-12 00:11:04 +03:00
/********************************************************************
1999-12-13 16:27:58 +03:00
Rpc pipe call id .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
1998-09-05 09:07:05 +04:00
static uint32 get_rpc_call_id ( void )
1998-03-12 00:11:04 +03:00
{
1999-12-13 16:27:58 +03:00
static uint32 call_id = 0 ;
return + + call_id ;
1998-03-12 00:11:04 +03:00
}
2009-01-14 00:50:43 +03:00
/*
* Realloc pdu to have a least " size " bytes
*/
static bool rpc_grow_buffer ( prs_struct * pdu , size_t size )
{
size_t extra_size ;
if ( prs_data_size ( pdu ) > = size ) {
return true ;
}
extra_size = size - prs_data_size ( pdu ) ;
if ( ! prs_force_grow ( pdu , extra_size ) ) {
DEBUG ( 0 , ( " rpc_grow_buffer: Failed to grow parse struct by "
" %d bytes. \n " , ( int ) extra_size ) ) ;
return false ;
}
DEBUG ( 5 , ( " rpc_grow_buffer: grew buffer by %d bytes to %u \n " ,
( int ) extra_size , prs_data_size ( pdu ) ) ) ;
return true ;
}
2008-04-25 00:24:20 +04:00
1998-03-12 00:11:04 +03:00
/*******************************************************************
1999-12-13 16:27:58 +03:00
Use SMBreadX to get rest of one fragment ' s worth of rpc data .
2009-01-14 18:06:10 +03:00
Reads the whole size or give an error message
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-04-24 02:45:53 +04:00
2009-01-15 23:56:03 +03:00
struct rpc_read_state {
struct event_context * ev ;
struct rpc_pipe_client * cli ;
char * data ;
size_t size ;
size_t num_read ;
} ;
static void rpc_read_np_done ( struct async_req * subreq ) ;
static void rpc_read_sock_done ( struct async_req * subreq ) ;
static struct async_req * rpc_read_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_pipe_client * cli ,
char * data , size_t size )
1998-03-12 00:11:04 +03:00
{
2009-01-15 23:56:03 +03:00
struct async_req * result , * subreq ;
struct rpc_read_state * state ;
1998-03-12 00:11:04 +03:00
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct rpc_read_state ) ) {
2009-01-15 23:56:03 +03:00
return NULL ;
}
state - > ev = ev ;
state - > cli = cli ;
state - > data = data ;
state - > size = size ;
state - > num_read = 0 ;
DEBUG ( 5 , ( " rpc_read_send: data_to_read: %u \n " , ( unsigned int ) size ) ) ;
if ( cli - > transport_type = = NCACN_NP ) {
subreq = cli_read_andx_send (
state , ev , cli - > trans . np . cli ,
cli - > trans . np . fnum , 0 , size ) ;
if ( subreq = = NULL ) {
DEBUG ( 10 , ( " cli_read_andx_send failed \n " ) ) ;
goto fail ;
2009-01-14 18:43:37 +03:00
}
2009-01-15 23:56:03 +03:00
subreq - > async . fn = rpc_read_np_done ;
subreq - > async . priv = result ;
return result ;
}
if ( ( cli - > transport_type = = NCACN_IP_TCP )
| | ( cli - > transport_type = = NCACN_UNIX_STREAM ) ) {
subreq = recvall_send ( state , ev , cli - > trans . sock . fd ,
data , size , 0 ) ;
if ( subreq = = NULL ) {
DEBUG ( 10 , ( " recvall_send failed \n " ) ) ;
goto fail ;
2009-01-14 18:51:43 +03:00
}
2009-01-15 23:56:03 +03:00
subreq - > async . fn = rpc_read_sock_done ;
subreq - > async . priv = result ;
return result ;
}
if ( async_post_status ( result , ev , NT_STATUS_INVALID_PARAMETER ) ) {
return result ;
}
fail :
TALLOC_FREE ( result ) ;
return NULL ;
}
2009-01-14 18:43:37 +03:00
2009-01-15 23:56:03 +03:00
static void rpc_read_np_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_read_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_read_state ) ;
NTSTATUS status ;
ssize_t received ;
uint8_t * rcvbuf ;
status = cli_read_andx_recv ( subreq , & received , & rcvbuf ) ;
/*
* We can ' t TALLOC_FREE ( subreq ) as usual here , as rcvbuf still is a
* child of that .
*/
if ( NT_STATUS_EQUAL ( status , NT_STATUS_BUFFER_TOO_SMALL ) ) {
status = NT_STATUS_OK ;
}
if ( ! NT_STATUS_IS_OK ( status ) ) {
TALLOC_FREE ( subreq ) ;
async_req_error ( req , status ) ;
return ;
}
memcpy ( state - > data + state - > num_read , rcvbuf , received ) ;
TALLOC_FREE ( subreq ) ;
state - > num_read + = received ;
if ( state - > num_read = = state - > size ) {
async_req_done ( req ) ;
return ;
}
subreq = cli_read_andx_send (
state , state - > ev , state - > cli - > trans . np . cli ,
state - > cli - > trans . np . fnum , 0 ,
state - > size - state - > num_read ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = rpc_read_np_done ;
subreq - > async . priv = req ;
}
static void rpc_read_sock_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
NTSTATUS status ;
status = recvall_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
static NTSTATUS rpc_read_recv ( struct async_req * req )
{
return async_req_simple_recv ( req ) ;
}
2009-01-17 17:07:52 +03:00
struct rpc_write_state {
struct event_context * ev ;
struct rpc_pipe_client * cli ;
const char * data ;
size_t size ;
size_t num_written ;
} ;
static void rpc_write_np_done ( struct async_req * subreq ) ;
static void rpc_write_sock_done ( struct async_req * subreq ) ;
static struct async_req * rpc_write_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_pipe_client * cli ,
const char * data , size_t size )
{
struct async_req * result , * subreq ;
struct rpc_write_state * state ;
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct rpc_write_state ) ) {
2009-01-17 17:07:52 +03:00
return NULL ;
}
state - > ev = ev ;
state - > cli = cli ;
state - > data = data ;
state - > size = size ;
state - > num_written = 0 ;
DEBUG ( 5 , ( " rpc_write_send: data_to_write: %u \n " , ( unsigned int ) size ) ) ;
if ( cli - > transport_type = = NCACN_NP ) {
subreq = cli_write_andx_send (
state , ev , cli - > trans . np . cli ,
cli - > trans . np . fnum , 8 , /* 8 means message mode. */
( uint8_t * ) data , 0 , size ) ;
if ( subreq = = NULL ) {
DEBUG ( 10 , ( " cli_write_andx_send failed \n " ) ) ;
goto fail ;
}
subreq - > async . fn = rpc_write_np_done ;
subreq - > async . priv = result ;
return result ;
}
if ( ( cli - > transport_type = = NCACN_IP_TCP )
| | ( cli - > transport_type = = NCACN_UNIX_STREAM ) ) {
subreq = sendall_send ( state , ev , cli - > trans . sock . fd ,
data , size , 0 ) ;
if ( subreq = = NULL ) {
DEBUG ( 10 , ( " sendall_send failed \n " ) ) ;
goto fail ;
}
subreq - > async . fn = rpc_write_sock_done ;
subreq - > async . priv = result ;
return result ;
}
if ( async_post_status ( result , ev , NT_STATUS_INVALID_PARAMETER ) ) {
return result ;
}
fail :
TALLOC_FREE ( result ) ;
return NULL ;
}
static void rpc_write_np_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_write_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_write_state ) ;
NTSTATUS status ;
size_t written ;
status = cli_write_andx_recv ( subreq , & written ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
state - > num_written + = written ;
if ( state - > num_written = = state - > size ) {
async_req_done ( req ) ;
return ;
}
subreq = cli_write_andx_send (
state , state - > ev , state - > cli - > trans . np . cli ,
state - > cli - > trans . np . fnum , 8 ,
( uint8_t * ) ( state - > data + state - > num_written ) ,
0 , state - > size - state - > num_written ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = rpc_write_np_done ;
subreq - > async . priv = req ;
}
static void rpc_write_sock_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
NTSTATUS status ;
status = sendall_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
static NTSTATUS rpc_write_recv ( struct async_req * req )
{
return async_req_simple_recv ( req ) ;
}
2009-01-16 16:46:41 +03:00
static NTSTATUS parse_rpc_header ( struct rpc_pipe_client * cli ,
struct rpc_hdr_info * prhdr ,
prs_struct * pdu )
{
/*
* This next call sets the endian bit correctly in current_pdu . We
* will propagate this to rbuf later .
*/
if ( ! smb_io_rpc_hdr ( " rpc_hdr " , prhdr , pdu , 0 ) ) {
DEBUG ( 0 , ( " get_current_pdu: Failed to unmarshall RPC_HDR. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
if ( prhdr - > frag_len > cli - > max_recv_frag ) {
DEBUG ( 0 , ( " cli_pipe_get_current_pdu: Server sent fraglen %d, "
" we only allow %d \n " , ( int ) prhdr - > frag_len ,
( int ) cli - > max_recv_frag ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
return NT_STATUS_OK ;
}
1998-03-12 00:11:04 +03:00
/****************************************************************************
2005-09-30 21:13:37 +04:00
Try and get a PDU ' s worth of data from current_pdu . If not , then read more
from the wire .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
2009-01-17 14:50:02 +03:00
struct get_complete_frag_state {
2009-01-16 16:46:41 +03:00
struct event_context * ev ;
struct rpc_pipe_client * cli ;
struct rpc_hdr_info * prhdr ;
prs_struct * pdu ;
} ;
2009-01-17 14:50:02 +03:00
static void get_complete_frag_got_header ( struct async_req * subreq ) ;
static void get_complete_frag_got_rest ( struct async_req * subreq ) ;
2009-01-16 16:46:41 +03:00
2009-01-17 14:50:02 +03:00
static struct async_req * get_complete_frag_send ( TALLOC_CTX * mem_ctx ,
2009-01-16 16:46:41 +03:00
struct event_context * ev ,
struct rpc_pipe_client * cli ,
struct rpc_hdr_info * prhdr ,
prs_struct * pdu )
1998-03-12 00:11:04 +03:00
{
2009-01-16 16:46:41 +03:00
struct async_req * result , * subreq ;
2009-01-17 14:50:02 +03:00
struct get_complete_frag_state * state ;
2009-01-16 16:46:41 +03:00
uint32_t pdu_len ;
NTSTATUS status ;
2005-09-30 21:13:37 +04:00
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct get_complete_frag_state ) ) {
2009-01-16 16:46:41 +03:00
return NULL ;
2005-09-30 21:13:37 +04:00
}
2009-01-16 16:46:41 +03:00
state - > ev = ev ;
state - > cli = cli ;
state - > prhdr = prhdr ;
state - > pdu = pdu ;
pdu_len = prs_data_size ( pdu ) ;
if ( pdu_len < RPC_HEADER_LEN ) {
if ( ! rpc_grow_buffer ( pdu , RPC_HEADER_LEN ) ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq = rpc_read_send ( state , state - > ev , state - > cli ,
prs_data_p ( state - > pdu ) + pdu_len ,
RPC_HEADER_LEN - pdu_len ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
2009-01-17 14:50:02 +03:00
subreq - > async . fn = get_complete_frag_got_header ;
2009-01-16 16:46:41 +03:00
subreq - > async . priv = result ;
return result ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
}
1998-03-12 00:11:04 +03:00
2009-01-16 16:46:41 +03:00
status = parse_rpc_header ( cli , prhdr , pdu ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto post_status ;
2009-01-14 14:58:13 +03:00
}
2009-01-16 16:46:41 +03:00
/*
* Ensure we have frag_len bytes of data .
*/
if ( pdu_len < prhdr - > frag_len ) {
if ( ! rpc_grow_buffer ( pdu , prhdr - > frag_len ) ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
2009-01-14 00:50:43 +03:00
}
2009-01-16 16:46:41 +03:00
subreq = rpc_read_send ( state , state - > ev , state - > cli ,
prs_data_p ( pdu ) + pdu_len ,
prhdr - > frag_len - pdu_len ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
2005-09-30 21:13:37 +04:00
}
2009-01-17 14:50:02 +03:00
subreq - > async . fn = get_complete_frag_got_rest ;
2009-01-16 16:46:41 +03:00
subreq - > async . priv = result ;
return result ;
1999-12-13 16:27:58 +03:00
}
1998-04-21 06:36:37 +04:00
2009-01-16 16:46:41 +03:00
status = NT_STATUS_OK ;
post_status :
if ( async_post_status ( result , ev , status ) ) {
return result ;
}
TALLOC_FREE ( result ) ;
return NULL ;
}
2009-01-17 14:50:02 +03:00
static void get_complete_frag_got_header ( struct async_req * subreq )
2009-01-16 16:46:41 +03:00
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
2009-01-17 14:50:02 +03:00
struct get_complete_frag_state * state = talloc_get_type_abort (
req - > private_data , struct get_complete_frag_state ) ;
2009-01-16 16:46:41 +03:00
NTSTATUS status ;
status = rpc_read_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
status = parse_rpc_header ( state - > cli , state - > prhdr , state - > pdu ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
if ( ! rpc_grow_buffer ( state - > pdu , state - > prhdr - > frag_len ) ) {
async_req_error ( req , NT_STATUS_NO_MEMORY ) ;
return ;
}
/*
* We ' re here in this piece of code because we ' ve read exactly
* RPC_HEADER_LEN bytes into state - > pdu .
*/
subreq = rpc_read_send ( state , state - > ev , state - > cli ,
prs_data_p ( state - > pdu ) + RPC_HEADER_LEN ,
state - > prhdr - > frag_len - RPC_HEADER_LEN ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
2009-01-17 14:50:02 +03:00
subreq - > async . fn = get_complete_frag_got_rest ;
2009-01-16 16:46:41 +03:00
subreq - > async . priv = req ;
}
2009-01-17 14:50:02 +03:00
static void get_complete_frag_got_rest ( struct async_req * subreq )
2009-01-16 16:46:41 +03:00
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
NTSTATUS status ;
status = rpc_read_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
2009-01-17 14:50:02 +03:00
static NTSTATUS get_complete_frag_recv ( struct async_req * req )
2009-01-16 16:46:41 +03:00
{
return async_req_simple_recv ( req ) ;
}
1998-10-14 10:29:20 +04:00
/****************************************************************************
2005-09-30 21:13:37 +04:00
NTLMSSP specific sign / seal .
Virtually identical to rpc_server / srv_pipe . c : api_pipe_ntlmssp_auth_process .
In fact I should probably abstract these into identical pieces of code . . . JRA .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
static NTSTATUS cli_pipe_verify_ntlmssp ( struct rpc_pipe_client * cli , RPC_HDR * prhdr ,
prs_struct * current_pdu ,
uint8 * p_ss_padding_len )
{
RPC_HDR_AUTH auth_info ;
uint32 save_offset = prs_offset ( current_pdu ) ;
uint32 auth_len = prhdr - > auth_len ;
2008-04-21 12:39:37 +04:00
NTLMSSP_STATE * ntlmssp_state = cli - > auth - > a_u . ntlmssp_state ;
2005-09-30 21:13:37 +04:00
unsigned char * data = NULL ;
size_t data_len ;
unsigned char * full_packet_data = NULL ;
size_t full_packet_data_len ;
DATA_BLOB auth_blob ;
NTSTATUS status ;
2008-04-21 12:39:37 +04:00
if ( cli - > auth - > auth_level = = PIPE_AUTH_LEVEL_NONE
| | cli - > auth - > auth_level = = PIPE_AUTH_LEVEL_CONNECT ) {
2005-09-30 21:13:37 +04:00
return NT_STATUS_OK ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
if ( ! ntlmssp_state ) {
return NT_STATUS_INVALID_PARAMETER ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* Ensure there's enough data for an authenticated response. */
if ( ( auth_len > RPC_MAX_SIGN_SIZE ) | |
( RPC_HEADER_LEN + RPC_HDR_RESP_LEN + RPC_HDR_AUTH_LEN + auth_len > prhdr - > frag_len ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_ntlmssp: auth_len %u is too large. \n " ,
( unsigned int ) auth_len ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/*
* We need the full packet data + length ( minus auth stuff ) as well as the packet data + length
* after the RPC header .
* We need to pass in the full packet ( minus auth len ) to the NTLMSSP sign and check seal
* functions as NTLMv2 checks the rpc headers also .
*/
data = ( unsigned char * ) ( prs_data_p ( current_pdu ) + RPC_HEADER_LEN + RPC_HDR_RESP_LEN ) ;
data_len = ( size_t ) ( prhdr - > frag_len - RPC_HEADER_LEN - RPC_HDR_RESP_LEN - RPC_HDR_AUTH_LEN - auth_len ) ;
2005-10-18 07:24:00 +04:00
full_packet_data = ( unsigned char * ) prs_data_p ( current_pdu ) ;
2005-09-30 21:13:37 +04:00
full_packet_data_len = prhdr - > frag_len - auth_len ;
/* Pull the auth header and the following data into a blob. */
if ( ! prs_set_offset ( current_pdu , RPC_HEADER_LEN + RPC_HDR_RESP_LEN + data_len ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_ntlmssp: cannot move offset to %u. \n " ,
( unsigned int ) RPC_HEADER_LEN + ( unsigned int ) RPC_HDR_RESP_LEN + ( unsigned int ) data_len ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & auth_info , current_pdu , 0 ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_ntlmssp: failed to unmarshall RPC_HDR_AUTH. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
2005-10-18 07:24:00 +04:00
auth_blob . data = ( unsigned char * ) prs_data_p ( current_pdu ) + prs_offset ( current_pdu ) ;
2005-09-30 21:13:37 +04:00
auth_blob . length = auth_len ;
2008-04-21 12:39:37 +04:00
switch ( cli - > auth - > auth_level ) {
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_LEVEL_PRIVACY :
/* Data is encrypted. */
status = ntlmssp_unseal_packet ( ntlmssp_state ,
data , data_len ,
full_packet_data ,
full_packet_data_len ,
& auth_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_ntlmssp: failed to unseal "
2008-04-24 22:24:37 +04:00
" packet from %s. Error was %s. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ,
2005-09-30 21:13:37 +04:00
nt_errstr ( status ) ) ) ;
return status ;
}
break ;
case PIPE_AUTH_LEVEL_INTEGRITY :
/* Data is signed. */
status = ntlmssp_check_packet ( ntlmssp_state ,
data , data_len ,
full_packet_data ,
full_packet_data_len ,
& auth_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_ntlmssp: check signing failed on "
2008-04-24 22:24:37 +04:00
" packet from %s. Error was %s. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ,
2005-09-30 21:13:37 +04:00
nt_errstr ( status ) ) ) ;
return status ;
}
break ;
default :
2008-04-21 12:39:37 +04:00
DEBUG ( 0 , ( " cli_pipe_verify_ntlmssp: unknown internal "
" auth level %d \n " , cli - > auth - > auth_level ) ) ;
2005-09-30 21:13:37 +04:00
return NT_STATUS_INVALID_INFO_CLASS ;
}
1998-10-14 10:29:20 +04:00
1999-12-13 16:27:58 +03:00
/*
2005-09-30 21:13:37 +04:00
* Return the current pointer to the data offset .
1999-12-13 16:27:58 +03:00
*/
2005-09-30 21:13:37 +04:00
if ( ! prs_set_offset ( current_pdu , save_offset ) ) {
DEBUG ( 0 , ( " api_pipe_auth_process: failed to set offset back to %u \n " ,
( unsigned int ) save_offset ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
1999-12-13 16:27:58 +03:00
/*
2005-09-30 21:13:37 +04:00
* Remember the padding length . We must remove it from the real data
* stream once the sign / seal is done .
1999-12-13 16:27:58 +03:00
*/
1998-10-14 10:29:20 +04:00
2005-09-30 21:13:37 +04:00
* p_ss_padding_len = auth_info . auth_pad_len ;
return NT_STATUS_OK ;
}
/****************************************************************************
schannel specific sign / seal .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-10-14 10:29:20 +04:00
2005-09-30 21:13:37 +04:00
static NTSTATUS cli_pipe_verify_schannel ( struct rpc_pipe_client * cli , RPC_HDR * prhdr ,
prs_struct * current_pdu ,
uint8 * p_ss_padding_len )
{
RPC_HDR_AUTH auth_info ;
RPC_AUTH_SCHANNEL_CHK schannel_chk ;
uint32 auth_len = prhdr - > auth_len ;
uint32 save_offset = prs_offset ( current_pdu ) ;
2008-04-21 12:39:37 +04:00
struct schannel_auth_struct * schannel_auth =
cli - > auth - > a_u . schannel_auth ;
2005-09-30 21:13:37 +04:00
uint32 data_len ;
2008-04-21 12:39:37 +04:00
if ( cli - > auth - > auth_level = = PIPE_AUTH_LEVEL_NONE
| | cli - > auth - > auth_level = = PIPE_AUTH_LEVEL_CONNECT ) {
2005-09-30 21:13:37 +04:00
return NT_STATUS_OK ;
}
2003-07-16 07:22:43 +04:00
2005-09-30 21:13:37 +04:00
if ( auth_len ! = RPC_AUTH_SCHANNEL_SIGN_OR_SEAL_CHK_LEN ) {
DEBUG ( 0 , ( " cli_pipe_verify_schannel: auth_len %u. \n " , ( unsigned int ) auth_len ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
2003-04-17 01:09:48 +04:00
2005-09-30 21:13:37 +04:00
if ( ! schannel_auth ) {
return NT_STATUS_INVALID_PARAMETER ;
}
1998-10-14 10:29:20 +04:00
2005-09-30 21:13:37 +04:00
/* Ensure there's enough data for an authenticated response. */
if ( ( auth_len > RPC_MAX_SIGN_SIZE ) | |
( RPC_HEADER_LEN + RPC_HDR_RESP_LEN + RPC_HDR_AUTH_LEN + auth_len > prhdr - > frag_len ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_schannel: auth_len %u is too large. \n " ,
( unsigned int ) auth_len ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
1998-10-14 10:29:20 +04:00
}
2005-09-30 21:13:37 +04:00
data_len = prhdr - > frag_len - RPC_HEADER_LEN - RPC_HDR_RESP_LEN - RPC_HDR_AUTH_LEN - auth_len ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
if ( ! prs_set_offset ( current_pdu , RPC_HEADER_LEN + RPC_HDR_RESP_LEN + data_len ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_schannel: cannot move offset to %u. \n " ,
( unsigned int ) RPC_HEADER_LEN + RPC_HDR_RESP_LEN + data_len ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
2008-04-20 18:23:48 +04:00
2005-09-30 21:13:37 +04:00
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & auth_info , current_pdu , 0 ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_schannel: failed to unmarshall RPC_HDR_AUTH. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
2003-07-16 07:22:43 +04:00
}
2001-03-10 02:48:58 +03:00
2005-09-30 21:13:37 +04:00
if ( auth_info . auth_type ! = RPC_SCHANNEL_AUTH_TYPE ) {
DEBUG ( 0 , ( " cli_pipe_verify_schannel: Invalid auth info %d on schannel \n " ,
auth_info . auth_type ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
if ( ! smb_io_rpc_auth_schannel_chk ( " " , RPC_AUTH_SCHANNEL_SIGN_OR_SEAL_CHK_LEN ,
& schannel_chk , current_pdu , 0 ) ) {
DEBUG ( 0 , ( " cli_pipe_verify_schannel: failed to unmarshal RPC_AUTH_SCHANNEL_CHK. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
if ( ! schannel_decode ( schannel_auth ,
2008-04-21 12:39:37 +04:00
cli - > auth - > auth_level ,
2005-09-30 21:13:37 +04:00
SENDER_IS_ACCEPTOR ,
& schannel_chk ,
prs_data_p ( current_pdu ) + RPC_HEADER_LEN + RPC_HDR_RESP_LEN ,
data_len ) ) {
DEBUG ( 3 , ( " cli_pipe_verify_schannel: failed to decode PDU "
2008-04-24 22:24:37 +04:00
" Connection to %s. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ) ) ;
2005-09-30 21:13:37 +04:00
return NT_STATUS_INVALID_PARAMETER ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* The sequence number gets incremented on both send and receive. */
schannel_auth - > seq_num + + ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
/*
* Return the current pointer to the data offset .
*/
if ( ! prs_set_offset ( current_pdu , save_offset ) ) {
DEBUG ( 0 , ( " api_pipe_auth_process: failed to set offset back to %u \n " ,
( unsigned int ) save_offset ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
1998-10-19 21:32:10 +04:00
2005-09-30 21:13:37 +04:00
/*
* Remember the padding length . We must remove it from the real data
* stream once the sign / seal is done .
*/
* p_ss_padding_len = auth_info . auth_pad_len ;
return NT_STATUS_OK ;
}
2003-07-16 07:22:43 +04:00
2005-09-30 21:13:37 +04:00
/****************************************************************************
Do the authentication checks on an incoming pdu . Check sign and unseal etc .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
Changes all over the shop, but all towards:
- NTLM2 support in the server
- KEY_EXCH support in the server
- variable length session keys.
In detail:
- NTLM2 is an extension of NTLMv1, that is compatible with existing
domain controllers (unlike NTLMv2, which requires a DC upgrade).
* This is known as 'NTLMv2 session security' *
(This is not yet implemented on the RPC pipes however, so there may
well still be issues for PDC setups, particuarly around password
changes. We do not fully understand the sign/seal implications of
NTLM2 on RPC pipes.)
This requires modifications to our authentication subsystem, as we
must handle the 'challege' input into the challenge-response algorithm
being changed. This also needs to be turned off for
'security=server', which does not support this.
- KEY_EXCH is another 'security' mechanism, whereby the session key
actually used by the server is sent by the client, rather than being
the shared-secret directly or indirectly.
- As both these methods change the session key, the auth subsystem
needed to be changed, to 'override' session keys provided by the
backend.
- There has also been a major overhaul of the NTLMSSP subsystem, to merge the 'client' and 'server' functions, so they both operate on a single structure. This should help the SPNEGO implementation.
- The 'names blob' in NTLMSSP is always in unicode - never in ascii.
Don't make an ascii version ever.
- The other big change is to allow variable length session keys. We
have always assumed that session keys are 16 bytes long - and padded
to this length if shorter. However, Kerberos session keys are 8 bytes
long, when the krb5 login uses DES.
* This fix allows SMB signging on machines not yet running MIT KRB5 1.3.1. *
- Add better DEBUG() messages to ntlm_auth, warning administrators of
misconfigurations that prevent access to the privileged pipe. This
should help reduce some of the 'it just doesn't work' issues.
- Fix data_blob_talloc() to behave the same way data_blob() does when
passed a NULL data pointer. (just allocate)
REMEMBER to make clean after this commit - I have changed plenty of data structures...
(This used to be commit f3bbc87b0dac63426cda6fac7a295d3aad810ecc)
2003-11-22 16:19:38 +03:00
2005-09-30 21:13:37 +04:00
static NTSTATUS cli_pipe_validate_rpc_response ( struct rpc_pipe_client * cli , RPC_HDR * prhdr ,
prs_struct * current_pdu ,
uint8 * p_ss_padding_len )
{
NTSTATUS ret = NT_STATUS_OK ;
2003-07-16 07:22:43 +04:00
2005-09-30 21:13:37 +04:00
/* Paranioa checks for auth_len. */
if ( prhdr - > auth_len ) {
if ( prhdr - > auth_len > prhdr - > frag_len ) {
return NT_STATUS_INVALID_PARAMETER ;
}
2003-07-16 07:22:43 +04:00
2008-06-27 06:46:18 +04:00
if ( prhdr - > auth_len + ( unsigned int ) RPC_HDR_AUTH_LEN < prhdr - > auth_len | |
prhdr - > auth_len + ( unsigned int ) RPC_HDR_AUTH_LEN < ( unsigned int ) RPC_HDR_AUTH_LEN ) {
2005-09-30 21:13:37 +04:00
/* Integer wrap attempt. */
return NT_STATUS_INVALID_PARAMETER ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
}
2005-09-30 21:13:37 +04:00
/*
* Now we have a complete RPC request PDU fragment , try and verify any auth data .
*/
2008-04-21 12:39:37 +04:00
switch ( cli - > auth - > auth_type ) {
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_NONE :
if ( prhdr - > auth_len ) {
2008-04-24 22:24:37 +04:00
DEBUG ( 3 , ( " cli_pipe_validate_rpc_response: "
" Connection to %s - got non-zero "
" auth len %u. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ,
2005-09-30 21:13:37 +04:00
( unsigned int ) prhdr - > auth_len ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
2005-09-30 21:13:37 +04:00
break ;
2001-03-10 02:48:58 +03:00
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_NTLMSSP :
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
ret = cli_pipe_verify_ntlmssp ( cli , prhdr , current_pdu , p_ss_padding_len ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
return ret ;
}
break ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_SCHANNEL :
ret = cli_pipe_verify_schannel ( cli , prhdr , current_pdu , p_ss_padding_len ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
return ret ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
2005-09-30 21:13:37 +04:00
break ;
case PIPE_AUTH_TYPE_KRB5 :
case PIPE_AUTH_TYPE_SPNEGO_KRB5 :
default :
2008-04-21 12:39:37 +04:00
DEBUG ( 3 , ( " cli_pipe_validate_rpc_response: Connection "
2008-04-24 22:24:37 +04:00
" to %s - unknown internal auth type %u. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ,
2008-04-21 12:39:37 +04:00
cli - > auth - > auth_type ) ) ;
2005-09-30 21:13:37 +04:00
return NT_STATUS_INVALID_INFO_CLASS ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
return NT_STATUS_OK ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/****************************************************************************
Do basic authentication checks on an incoming pdu .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS cli_pipe_validate_current_pdu ( struct rpc_pipe_client * cli , RPC_HDR * prhdr ,
prs_struct * current_pdu ,
uint8 expected_pkt_type ,
char * * ppdata ,
uint32 * pdata_len ,
prs_struct * return_data )
{
NTSTATUS ret = NT_STATUS_OK ;
uint32 current_pdu_len = prs_data_size ( current_pdu ) ;
if ( current_pdu_len ! = prhdr - > frag_len ) {
DEBUG ( 5 , ( " cli_pipe_validate_current_pdu: incorrect pdu length %u, expected %u \n " ,
( unsigned int ) current_pdu_len , ( unsigned int ) prhdr - > frag_len ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
1998-10-14 10:29:20 +04:00
}
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
/*
* Point the return values at the real data including the RPC
* header . Just in case the caller wants it .
*/
* ppdata = prs_data_p ( current_pdu ) ;
* pdata_len = current_pdu_len ;
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
/* Ensure we have the correct type. */
switch ( prhdr - > pkt_type ) {
case RPC_ALTCONTRESP :
case RPC_BINDACK :
/* Alter context and bind ack share the same packet definitions. */
break ;
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
case RPC_RESPONSE :
2004-05-14 00:32:21 +04:00
{
2005-09-30 21:13:37 +04:00
RPC_HDR_RESP rhdr_resp ;
uint8 ss_padding_len = 0 ;
if ( ! smb_io_rpc_hdr_resp ( " rpc_hdr_resp " , & rhdr_resp , current_pdu , 0 ) ) {
DEBUG ( 5 , ( " cli_pipe_validate_current_pdu: failed to unmarshal RPC_HDR_RESP. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
/* Here's where we deal with incoming sign/seal. */
ret = cli_pipe_validate_rpc_response ( cli , prhdr ,
current_pdu , & ss_padding_len ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
return ret ;
}
/* Point the return values at the NDR data. Remember to remove any ss padding. */
* ppdata = prs_data_p ( current_pdu ) + RPC_HEADER_LEN + RPC_HDR_RESP_LEN ;
if ( current_pdu_len < RPC_HEADER_LEN + RPC_HDR_RESP_LEN + ss_padding_len ) {
return NT_STATUS_BUFFER_TOO_SMALL ;
}
* pdata_len = current_pdu_len - RPC_HEADER_LEN - RPC_HDR_RESP_LEN - ss_padding_len ;
/* Remember to remove the auth footer. */
if ( prhdr - > auth_len ) {
/* We've already done integer wrap tests on auth_len in
cli_pipe_validate_rpc_response ( ) . */
if ( * pdata_len < RPC_HDR_AUTH_LEN + prhdr - > auth_len ) {
return NT_STATUS_BUFFER_TOO_SMALL ;
}
* pdata_len - = ( RPC_HDR_AUTH_LEN + prhdr - > auth_len ) ;
}
DEBUG ( 10 , ( " cli_pipe_validate_current_pdu: got pdu len %u, data_len %u, ss_len %u \n " ,
current_pdu_len , * pdata_len , ss_padding_len ) ) ;
/*
* If this is the first reply , and the allocation hint is reasonably , try and
* set up the return_data parse_struct to the correct size .
*/
if ( ( prs_data_size ( return_data ) = = 0 ) & & rhdr_resp . alloc_hint & & ( rhdr_resp . alloc_hint < 15 * 1024 * 1024 ) ) {
if ( ! prs_set_buffer_size ( return_data , rhdr_resp . alloc_hint ) ) {
DEBUG ( 0 , ( " cli_pipe_validate_current_pdu: reply alloc hint %u "
" too large to allocate \n " ,
( unsigned int ) rhdr_resp . alloc_hint ) ) ;
return NT_STATUS_NO_MEMORY ;
}
}
break ;
2004-05-14 00:32:21 +04:00
}
2005-09-30 21:13:37 +04:00
case RPC_BINDNACK :
2008-04-24 22:24:37 +04:00
DEBUG ( 1 , ( " cli_pipe_validate_current_pdu: Bind NACK "
" received from %s! \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ) ) ;
2005-09-30 21:13:37 +04:00
/* Use this for now... */
return NT_STATUS_NETWORK_ACCESS_DENIED ;
case RPC_FAULT :
2004-05-14 00:32:21 +04:00
{
2005-09-30 21:13:37 +04:00
RPC_HDR_RESP rhdr_resp ;
RPC_HDR_FAULT fault_resp ;
if ( ! smb_io_rpc_hdr_resp ( " rpc_hdr_resp " , & rhdr_resp , current_pdu , 0 ) ) {
DEBUG ( 5 , ( " cli_pipe_validate_current_pdu: failed to unmarshal RPC_HDR_RESP. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
if ( ! smb_io_rpc_hdr_fault ( " fault " , & fault_resp , current_pdu , 0 ) ) {
DEBUG ( 5 , ( " cli_pipe_validate_current_pdu: failed to unmarshal RPC_HDR_FAULT. \n " ) ) ;
return NT_STATUS_BUFFER_TOO_SMALL ;
}
2003-04-09 19:54:17 +04:00
2008-04-24 22:24:37 +04:00
DEBUG ( 1 , ( " cli_pipe_validate_current_pdu: RPC fault "
" code %s received from %s! \n " ,
2008-10-30 01:21:06 +03:00
dcerpc_errstr ( debug_ctx ( ) , NT_STATUS_V ( fault_resp . status ) ) ,
2008-04-24 22:24:37 +04:00
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ) ) ;
2005-09-30 21:13:37 +04:00
if ( NT_STATUS_IS_OK ( fault_resp . status ) ) {
return NT_STATUS_UNSUCCESSFUL ;
} else {
return fault_resp . status ;
}
2003-04-09 19:54:17 +04:00
}
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
default :
DEBUG ( 0 , ( " cli_pipe_validate_current_pdu: unknown packet type %u received "
2008-04-24 22:24:37 +04:00
" from %s! \n " ,
2005-09-30 21:13:37 +04:00
( unsigned int ) prhdr - > pkt_type ,
2008-04-24 22:24:37 +04:00
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ) ) ;
2005-09-30 21:13:37 +04:00
return NT_STATUS_INVALID_INFO_CLASS ;
}
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
if ( prhdr - > pkt_type ! = expected_pkt_type ) {
2008-04-24 22:24:37 +04:00
DEBUG ( 3 , ( " cli_pipe_validate_current_pdu: Connection to %s "
" got an unexpected RPC packet type - %u, not %u \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ,
2005-09-30 21:13:37 +04:00
prhdr - > pkt_type ,
expected_pkt_type ) ) ;
return NT_STATUS_INVALID_INFO_CLASS ;
2003-04-09 19:54:17 +04:00
}
2005-09-30 21:13:37 +04:00
/* Do this just before return - we don't want to modify any rpc header
data before now as we may have needed to do cryptographic actions on
it before . */
if ( ( prhdr - > pkt_type = = RPC_BINDACK ) & & ! ( prhdr - > flags & RPC_FLG_LAST ) ) {
DEBUG ( 5 , ( " cli_pipe_validate_current_pdu: bug in server (AS/U?), "
" setting fragment first/last ON. \n " ) ) ;
prhdr - > flags | = RPC_FLG_FIRST | RPC_FLG_LAST ;
}
return NT_STATUS_OK ;
1998-10-14 10:29:20 +04:00
}
2005-09-30 21:13:37 +04:00
/****************************************************************************
Ensure we eat the just processed pdu from the current_pdu prs_struct .
Normally the frag_len and buffer size will match , but on the first trans
reply there is a theoretical chance that buffer size > frag_len , so we must
deal with that .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS cli_pipe_reset_current_pdu ( struct rpc_pipe_client * cli , RPC_HDR * prhdr , prs_struct * current_pdu )
{
uint32 current_pdu_len = prs_data_size ( current_pdu ) ;
if ( current_pdu_len < prhdr - > frag_len ) {
return NT_STATUS_BUFFER_TOO_SMALL ;
}
/* Common case. */
if ( current_pdu_len = = ( uint32 ) prhdr - > frag_len ) {
prs_mem_free ( current_pdu ) ;
2008-03-15 01:26:28 +03:00
prs_init_empty ( current_pdu , prs_get_mem_context ( current_pdu ) , UNMARSHALL ) ;
2005-09-30 21:13:37 +04:00
/* Make current_pdu dynamic with no memory. */
prs_give_memory ( current_pdu , 0 , 0 , True ) ;
return NT_STATUS_OK ;
}
/*
* Oh no ! More data in buffer than we processed in current pdu .
* Cheat . Move the data down and shrink the buffer .
*/
memcpy ( prs_data_p ( current_pdu ) , prs_data_p ( current_pdu ) + prhdr - > frag_len ,
current_pdu_len - prhdr - > frag_len ) ;
/* Remember to set the read offset back to zero. */
prs_set_offset ( current_pdu , 0 ) ;
/* Shrink the buffer. */
if ( ! prs_set_buffer_size ( current_pdu , current_pdu_len - prhdr - > frag_len ) ) {
return NT_STATUS_BUFFER_TOO_SMALL ;
}
return NT_STATUS_OK ;
}
1998-10-14 10:29:20 +04:00
2009-01-11 13:17:26 +03:00
/****************************************************************************
Call a remote api on an arbitrary pipe . takes param , data and setup buffers .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-01-16 19:07:52 +03:00
struct cli_api_pipe_state {
struct event_context * ev ;
struct rpc_pipe_client * cli ;
uint32_t max_rdata_len ;
uint8_t * rdata ;
uint32_t rdata_len ;
} ;
static void cli_api_pipe_np_trans_done ( struct async_req * subreq ) ;
static void cli_api_pipe_sock_send_done ( struct async_req * subreq ) ;
static void cli_api_pipe_sock_read_done ( struct async_req * subreq ) ;
static struct async_req * cli_api_pipe_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_pipe_client * cli ,
uint8_t * data , size_t data_len ,
uint32_t max_rdata_len )
{
struct async_req * result , * subreq ;
struct cli_api_pipe_state * state ;
NTSTATUS status ;
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct cli_api_pipe_state ) ) {
2009-01-16 19:07:52 +03:00
return NULL ;
}
state - > ev = ev ;
state - > cli = cli ;
state - > max_rdata_len = max_rdata_len ;
if ( state - > max_rdata_len < RPC_HEADER_LEN ) {
/*
* For a RPC reply we always need at least RPC_HEADER_LEN
* bytes . We check this here because we will receive
* RPC_HEADER_LEN bytes in cli_trans_sock_send_done .
*/
status = NT_STATUS_INVALID_PARAMETER ;
goto post_status ;
}
if ( cli - > transport_type = = NCACN_NP ) {
uint16_t setup [ 2 ] ;
SSVAL ( setup + 0 , 0 , TRANSACT_DCERPCCMD ) ;
SSVAL ( setup + 1 , 0 , cli - > trans . np . fnum ) ;
subreq = cli_trans_send (
state , ev , cli - > trans . np . cli , SMBtrans ,
" \\ PIPE \\ " , 0 , 0 , 0 , setup , 2 , 0 ,
NULL , 0 , 0 , data , data_len , max_rdata_len ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq - > async . fn = cli_api_pipe_np_trans_done ;
subreq - > async . priv = result ;
return result ;
}
if ( ( cli - > transport_type = = NCACN_IP_TCP )
| | ( cli - > transport_type = = NCACN_UNIX_STREAM ) ) {
subreq = sendall_send ( state , ev , cli - > trans . sock . fd ,
data , data_len , 0 ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq - > async . fn = cli_api_pipe_sock_send_done ;
subreq - > async . priv = result ;
return result ;
}
status = NT_STATUS_INVALID_PARAMETER ;
post_status :
if ( async_post_status ( result , ev , status ) ) {
return result ;
}
TALLOC_FREE ( result ) ;
return NULL ;
}
static void cli_api_pipe_np_trans_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct cli_api_pipe_state * state = talloc_get_type_abort (
req - > private_data , struct cli_api_pipe_state ) ;
NTSTATUS status ;
status = cli_trans_recv ( subreq , state , NULL , NULL , NULL , NULL ,
& state - > rdata , & state - > rdata_len ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
static void cli_api_pipe_sock_send_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct cli_api_pipe_state * state = talloc_get_type_abort (
req - > private_data , struct cli_api_pipe_state ) ;
NTSTATUS status ;
status = sendall_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
state - > rdata = TALLOC_ARRAY ( state , uint8_t , RPC_HEADER_LEN ) ;
if ( async_req_nomem ( state - > rdata , req ) ) {
return ;
}
state - > rdata_len = RPC_HEADER_LEN ;
subreq = recvall_send ( state , state - > ev , state - > cli - > trans . sock . fd ,
state - > rdata , RPC_HEADER_LEN , 0 ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = cli_api_pipe_sock_read_done ;
subreq - > async . priv = req ;
}
static void cli_api_pipe_sock_read_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
NTSTATUS status ;
status = recvall_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
static NTSTATUS cli_api_pipe_recv ( struct async_req * req , TALLOC_CTX * mem_ctx ,
uint8_t * * prdata , uint32_t * prdata_len )
{
struct cli_api_pipe_state * state = talloc_get_type_abort (
req - > private_data , struct cli_api_pipe_state ) ;
NTSTATUS status ;
if ( async_req_is_error ( req , & status ) ) {
return status ;
}
* prdata = talloc_move ( mem_ctx , & state - > rdata ) ;
* prdata_len = state - > rdata_len ;
return NT_STATUS_OK ;
}
1998-03-12 00:11:04 +03:00
/****************************************************************************
2005-09-30 21:13:37 +04:00
Send data on an rpc pipe via trans . The prs_struct data must be the last
pdu fragment of an NDR data stream .
Receive response data from an rpc pipe , which may be large . . .
1998-03-12 00:11:04 +03:00
1999-12-13 16:27:58 +03:00
Read the first fragment : unfortunately have to use SMBtrans for the first
1998-03-12 00:11:04 +03:00
bit , then SMBreadX for subsequent bits .
1999-12-13 16:27:58 +03:00
If first fragment received also wasn ' t the last fragment , continue
1998-03-12 00:11:04 +03:00
getting fragments until we _do_ receive the last fragment .
1999-12-13 16:27:58 +03:00
Request / Response PDU ' s look like the following . . .
| < - - - - - - - - - - - - - - - - - - PDU len - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - > |
| < - HDR_LEN - - > | < - - REQ LEN - - - - - - > | . . . . . . . . . . . . . | < - AUTH_HDRLEN - > | < - AUTH_LEN - - > |
+ - - - - - - - - - - - - + - - - - - - - - - - - - - - - - - + - - - - - - - - - - - - - + - - - - - - - - - - - - - - - + - - - - - - - - - - - - - +
| RPC HEADER | REQ / RESP HEADER | DATA . . . . . . | AUTH_HDR | AUTH DATA |
+ - - - - - - - - - - - - + - - - - - - - - - - - - - - - - - + - - - - - - - - - - - - - + - - - - - - - - - - - - - - - + - - - - - - - - - - - - - +
2005-09-30 21:13:37 +04:00
Where the presence of the AUTH_HDR and AUTH DATA are dependent on the
2003-04-16 19:39:57 +04:00
signing & sealing being negotiated .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-01-17 15:33:34 +03:00
struct rpc_api_pipe_state {
struct event_context * ev ;
struct rpc_pipe_client * cli ;
uint8_t expected_pkt_type ;
prs_struct incoming_frag ;
struct rpc_hdr_info rhdr ;
prs_struct incoming_pdu ; /* Incoming reply */
uint32_t incoming_pdu_offset ;
} ;
static int rpc_api_pipe_state_destructor ( struct rpc_api_pipe_state * state )
{
prs_mem_free ( & state - > incoming_frag ) ;
prs_mem_free ( & state - > incoming_pdu ) ;
return 0 ;
}
static void rpc_api_pipe_trans_done ( struct async_req * subreq ) ;
static void rpc_api_pipe_got_pdu ( struct async_req * subreq ) ;
static struct async_req * rpc_api_pipe_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_pipe_client * cli ,
prs_struct * data , /* Outgoing PDU */
uint8_t expected_pkt_type )
{
struct async_req * result , * subreq ;
struct rpc_api_pipe_state * state ;
2009-01-18 14:22:14 +03:00
uint16_t max_recv_frag ;
2009-01-17 15:33:34 +03:00
NTSTATUS status ;
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct rpc_api_pipe_state ) ) {
2009-01-17 15:33:34 +03:00
return NULL ;
}
state - > ev = ev ;
state - > cli = cli ;
state - > expected_pkt_type = expected_pkt_type ;
state - > incoming_pdu_offset = 0 ;
prs_init_empty ( & state - > incoming_frag , state , UNMARSHALL ) ;
prs_init_empty ( & state - > incoming_pdu , state , UNMARSHALL ) ;
/* Make incoming_pdu dynamic with no memory. */
prs_give_memory ( & state - > incoming_pdu , 0 , 0 , true ) ;
talloc_set_destructor ( state , rpc_api_pipe_state_destructor ) ;
/*
* Ensure we ' re not sending too much .
*/
if ( prs_offset ( data ) > cli - > max_xmit_frag ) {
status = NT_STATUS_INVALID_PARAMETER ;
goto post_status ;
}
DEBUG ( 5 , ( " rpc_api_pipe: %s \n " , rpccli_pipe_txt ( debug_ctx ( ) , cli ) ) ) ;
2009-01-18 14:22:14 +03:00
max_recv_frag = cli - > max_recv_frag ;
# ifdef DEVELOPER
max_recv_frag = RPC_HEADER_LEN + 10 + ( sys_random ( ) % 32 ) ;
# endif
subreq = cli_api_pipe_send ( state , ev , cli , ( uint8_t * ) prs_data_p ( data ) ,
prs_offset ( data ) , max_recv_frag ) ;
2009-01-17 15:33:34 +03:00
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq - > async . fn = rpc_api_pipe_trans_done ;
subreq - > async . priv = result ;
return result ;
post_status :
if ( async_post_status ( result , ev , status ) ) {
return result ;
}
TALLOC_FREE ( result ) ;
return NULL ;
}
static void rpc_api_pipe_trans_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_api_pipe_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_api_pipe_state ) ;
NTSTATUS status ;
uint8_t * rdata = NULL ;
uint32_t rdata_len = 0 ;
char * rdata_copy ;
status = cli_api_pipe_recv ( subreq , state , & rdata , & rdata_len ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " cli_api_pipe failed: %s \n " , nt_errstr ( status ) ) ) ;
async_req_error ( req , status ) ;
return ;
}
if ( rdata = = NULL ) {
DEBUG ( 3 , ( " rpc_api_pipe: %s failed to return data. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , state - > cli ) ) ) ;
async_req_done ( req ) ;
return ;
}
/*
* Give the memory received from cli_trans as dynamic to the current
* pdu . Duplicating it sucks , but prs_struct doesn ' t know about talloc
* : - (
*/
rdata_copy = ( char * ) memdup ( rdata , rdata_len ) ;
TALLOC_FREE ( rdata ) ;
if ( async_req_nomem ( rdata_copy , req ) ) {
return ;
}
prs_give_memory ( & state - > incoming_frag , rdata_copy , rdata_len , true ) ;
/* Ensure we have enough data for a pdu. */
subreq = get_complete_frag_send ( state , state - > ev , state - > cli ,
& state - > rhdr , & state - > incoming_frag ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = rpc_api_pipe_got_pdu ;
subreq - > async . priv = req ;
}
static void rpc_api_pipe_got_pdu ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_api_pipe_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_api_pipe_state ) ;
NTSTATUS status ;
char * rdata = NULL ;
uint32_t rdata_len = 0 ;
status = get_complete_frag_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 5 , ( " get_complete_frag failed: %s \n " ,
nt_errstr ( status ) ) ) ;
async_req_error ( req , status ) ;
return ;
}
status = cli_pipe_validate_current_pdu (
state - > cli , & state - > rhdr , & state - > incoming_frag ,
state - > expected_pkt_type , & rdata , & rdata_len ,
& state - > incoming_pdu ) ;
DEBUG ( 10 , ( " rpc_api_pipe: got frag len of %u at offset %u: %s \n " ,
( unsigned ) prs_data_size ( & state - > incoming_frag ) ,
( unsigned ) state - > incoming_pdu_offset ,
nt_errstr ( status ) ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
if ( ( state - > rhdr . flags & RPC_FLG_FIRST )
& & ( state - > rhdr . pack_type [ 0 ] = = 0 ) ) {
/*
* Set the data type correctly for big - endian data on the
* first packet .
*/
DEBUG ( 10 , ( " rpc_api_pipe: On %s PDU data format is "
" big-endian. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , state - > cli ) ) ) ;
prs_set_endian_data ( & state - > incoming_pdu , RPC_BIG_ENDIAN ) ;
}
/*
* Check endianness on subsequent packets .
*/
if ( state - > incoming_frag . bigendian_data
! = state - > incoming_pdu . bigendian_data ) {
DEBUG ( 0 , ( " rpc_api_pipe: Error : Endianness changed from %s to "
" %s \n " ,
state - > incoming_pdu . bigendian_data ? " big " : " little " ,
state - > incoming_frag . bigendian_data ? " big " : " little " ) ) ;
async_req_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
return ;
}
/* Now copy the data portion out of the pdu into rbuf. */
if ( ! prs_force_grow ( & state - > incoming_pdu , rdata_len ) ) {
async_req_error ( req , NT_STATUS_NO_MEMORY ) ;
return ;
}
memcpy ( prs_data_p ( & state - > incoming_pdu ) + state - > incoming_pdu_offset ,
rdata , ( size_t ) rdata_len ) ;
state - > incoming_pdu_offset + = rdata_len ;
status = cli_pipe_reset_current_pdu ( state - > cli , & state - > rhdr ,
& state - > incoming_frag ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
if ( state - > rhdr . flags & RPC_FLG_LAST ) {
DEBUG ( 10 , ( " rpc_api_pipe: %s returned %u bytes. \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , state - > cli ) ,
( unsigned ) prs_data_size ( & state - > incoming_pdu ) ) ) ;
async_req_done ( req ) ;
return ;
}
subreq = get_complete_frag_send ( state , state - > ev , state - > cli ,
& state - > rhdr , & state - > incoming_frag ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = rpc_api_pipe_got_pdu ;
subreq - > async . priv = req ;
}
static NTSTATUS rpc_api_pipe_recv ( struct async_req * req , TALLOC_CTX * mem_ctx ,
prs_struct * reply_pdu )
{
struct rpc_api_pipe_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_api_pipe_state ) ;
NTSTATUS status ;
if ( async_req_is_error ( req , & status ) ) {
return status ;
}
* reply_pdu = state - > incoming_pdu ;
reply_pdu - > mem_ctx = mem_ctx ;
/*
* Prevent state - > incoming_pdu from being freed in
* rpc_api_pipe_state_destructor ( )
*/
prs_init_empty ( & state - > incoming_pdu , state , UNMARSHALL ) ;
return NT_STATUS_OK ;
}
2005-09-30 21:13:37 +04:00
/*******************************************************************
Creates krb5 auth bind .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
static NTSTATUS create_krb5_auth_bind_req ( struct rpc_pipe_client * cli ,
enum pipe_auth_level auth_level ,
RPC_HDR_AUTH * pauth_out ,
prs_struct * auth_data )
{
# ifdef HAVE_KRB5
int ret ;
2008-04-21 12:39:37 +04:00
struct kerberos_auth_struct * a = cli - > auth - > a_u . kerberos_auth ;
2007-05-14 16:16:20 +04:00
DATA_BLOB tkt = data_blob_null ;
DATA_BLOB tkt_wrapped = data_blob_null ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
/* We may change the pad length before marshalling. */
init_rpc_hdr_auth ( pauth_out , RPC_KRB5_AUTH_TYPE , ( int ) auth_level , 0 , 1 ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " create_krb5_auth_bind_req: creating a service ticket for principal %s \n " ,
a - > service_principal ) ) ;
1998-10-14 10:29:20 +04:00
2005-09-30 21:13:37 +04:00
/* Create the ticket for the service principal and return it in a gss-api wrapped blob. */
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
ret = cli_krb5_get_ticket ( a - > service_principal , 0 , & tkt ,
2007-02-08 20:02:39 +03:00
& a - > session_key , ( uint32 ) AP_OPTS_MUTUAL_REQUIRED , NULL , NULL ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
if ( ret ) {
DEBUG ( 1 , ( " create_krb5_auth_bind_req: cli_krb5_get_ticket for principal %s "
" failed with %s \n " ,
a - > service_principal ,
error_message ( ret ) ) ) ;
2001-03-10 02:48:58 +03:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & tkt ) ;
prs_mem_free ( auth_data ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* wrap that up in a nice GSS-API wrapping */
tkt_wrapped = spnego_gen_krb5_wrap ( tkt , TOK_ID_KRB_AP_REQ ) ;
2001-03-10 02:48:58 +03:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & tkt ) ;
2001-03-10 02:48:58 +03:00
2005-09-30 21:13:37 +04:00
/* Auth len in the rpc header doesn't include auth_header. */
if ( ! prs_copy_data_in ( auth_data , ( char * ) tkt_wrapped . data , tkt_wrapped . length ) ) {
data_blob_free ( & tkt_wrapped ) ;
prs_mem_free ( auth_data ) ;
return NT_STATUS_NO_MEMORY ;
}
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " create_krb5_auth_bind_req: Created krb5 GSS blob : \n " ) ) ;
2007-03-28 17:34:59 +04:00
dump_data ( 5 , tkt_wrapped . data , tkt_wrapped . length ) ;
1998-10-14 10:29:20 +04:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & tkt_wrapped ) ;
return NT_STATUS_OK ;
# else
return NT_STATUS_INVALID_PARAMETER ;
# endif
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/*******************************************************************
Creates SPNEGO NTLMSSP auth bind .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS create_spnego_ntlmssp_auth_rpc_bind_req ( struct rpc_pipe_client * cli ,
enum pipe_auth_level auth_level ,
RPC_HDR_AUTH * pauth_out ,
prs_struct * auth_data )
{
NTSTATUS nt_status ;
2007-05-14 16:16:20 +04:00
DATA_BLOB null_blob = data_blob_null ;
DATA_BLOB request = data_blob_null ;
DATA_BLOB spnego_msg = data_blob_null ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* We may change the pad length before marshalling. */
init_rpc_hdr_auth ( pauth_out , RPC_SPNEGO_AUTH_TYPE , ( int ) auth_level , 0 , 1 ) ;
2003-04-17 01:09:48 +04:00
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " create_spnego_ntlmssp_auth_rpc_bind_req: Processing NTLMSSP Negotiate \n " ) ) ;
2008-04-21 12:39:37 +04:00
nt_status = ntlmssp_update ( cli - > auth - > a_u . ntlmssp_state ,
2005-09-30 21:13:37 +04:00
null_blob ,
& request ) ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
if ( ! NT_STATUS_EQUAL ( nt_status , NT_STATUS_MORE_PROCESSING_REQUIRED ) ) {
data_blob_free ( & request ) ;
prs_mem_free ( auth_data ) ;
return nt_status ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
}
2005-09-30 21:13:37 +04:00
/* Wrap this in SPNEGO. */
spnego_msg = gen_negTokenInit ( OID_NTLMSSP , request ) ;
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & request ) ;
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
/* Auth len in the rpc header doesn't include auth_header. */
if ( ! prs_copy_data_in ( auth_data , ( char * ) spnego_msg . data , spnego_msg . length ) ) {
data_blob_free ( & spnego_msg ) ;
prs_mem_free ( auth_data ) ;
return NT_STATUS_NO_MEMORY ;
}
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " create_spnego_ntlmssp_auth_rpc_bind_req: NTLMSSP Negotiate: \n " ) ) ;
2007-03-28 17:34:59 +04:00
dump_data ( 5 , spnego_msg . data , spnego_msg . length ) ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & spnego_msg ) ;
return NT_STATUS_OK ;
}
/*******************************************************************
Creates NTLMSSP auth bind .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS create_ntlmssp_auth_rpc_bind_req ( struct rpc_pipe_client * cli ,
enum pipe_auth_level auth_level ,
RPC_HDR_AUTH * pauth_out ,
prs_struct * auth_data )
1998-03-12 00:11:04 +03:00
{
2005-09-30 21:13:37 +04:00
NTSTATUS nt_status ;
2007-05-14 16:16:20 +04:00
DATA_BLOB null_blob = data_blob_null ;
DATA_BLOB request = data_blob_null ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
/* We may change the pad length before marshalling. */
init_rpc_hdr_auth ( pauth_out , RPC_NTLMSSP_AUTH_TYPE , ( int ) auth_level , 0 , 1 ) ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " create_ntlmssp_auth_rpc_bind_req: Processing NTLMSSP Negotiate \n " ) ) ;
2008-04-21 12:39:37 +04:00
nt_status = ntlmssp_update ( cli - > auth - > a_u . ntlmssp_state ,
2005-09-30 21:13:37 +04:00
null_blob ,
& request ) ;
1998-10-07 02:03:04 +04:00
2005-09-30 21:13:37 +04:00
if ( ! NT_STATUS_EQUAL ( nt_status , NT_STATUS_MORE_PROCESSING_REQUIRED ) ) {
data_blob_free ( & request ) ;
prs_mem_free ( auth_data ) ;
return nt_status ;
}
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
/* Auth len in the rpc header doesn't include auth_header. */
if ( ! prs_copy_data_in ( auth_data , ( char * ) request . data , request . length ) ) {
data_blob_free ( & request ) ;
prs_mem_free ( auth_data ) ;
return NT_STATUS_NO_MEMORY ;
}
1998-10-07 02:03:04 +04:00
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " create_ntlmssp_auth_rpc_bind_req: NTLMSSP Negotiate: \n " ) ) ;
2007-03-28 17:34:59 +04:00
dump_data ( 5 , request . data , request . length ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & request ) ;
return NT_STATUS_OK ;
}
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
/*******************************************************************
Creates schannel auth bind .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
static NTSTATUS create_schannel_auth_rpc_bind_req ( struct rpc_pipe_client * cli ,
enum pipe_auth_level auth_level ,
RPC_HDR_AUTH * pauth_out ,
prs_struct * auth_data )
{
RPC_AUTH_SCHANNEL_NEG schannel_neg ;
2003-05-12 11:16:21 +04:00
2005-09-30 21:13:37 +04:00
/* We may change the pad length before marshalling. */
init_rpc_hdr_auth ( pauth_out , RPC_SCHANNEL_AUTH_TYPE , ( int ) auth_level , 0 , 1 ) ;
2003-05-12 11:16:21 +04:00
2005-09-30 21:13:37 +04:00
/* Use lp_workgroup() if domain not specified */
2003-05-12 11:16:21 +04:00
2008-04-22 00:27:29 +04:00
if ( ! cli - > auth - > domain | | ! cli - > auth - > domain [ 0 ] ) {
cli - > auth - > domain = talloc_strdup ( cli , lp_workgroup ( ) ) ;
if ( cli - > auth - > domain = = NULL ) {
2008-04-20 15:51:46 +04:00
return NT_STATUS_NO_MEMORY ;
}
2005-09-30 21:13:37 +04:00
}
2003-04-09 19:54:17 +04:00
2008-04-22 00:27:29 +04:00
init_rpc_auth_schannel_neg ( & schannel_neg , cli - > auth - > domain ,
global_myname ( ) ) ;
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
/*
* Now marshall the data into the auth parse_struct .
*/
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
if ( ! smb_io_rpc_auth_schannel_neg ( " schannel_neg " ,
& schannel_neg , auth_data , 0 ) ) {
DEBUG ( 0 , ( " Failed to marshall RPC_AUTH_SCHANNEL_NEG. \n " ) ) ;
prs_mem_free ( auth_data ) ;
return NT_STATUS_NO_MEMORY ;
2003-04-09 19:54:17 +04:00
}
2003-07-31 03:33:56 +04:00
2005-09-30 21:13:37 +04:00
return NT_STATUS_OK ;
}
/*******************************************************************
Creates the internals of a DCE / RPC bind request or alter context PDU .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-02-27 21:45:44 +03:00
static NTSTATUS create_bind_or_alt_ctx_internal ( enum RPC_PKT_TYPE pkt_type ,
2005-09-30 21:13:37 +04:00
prs_struct * rpc_out ,
uint32 rpc_call_id ,
2008-04-20 13:45:41 +04:00
const RPC_IFACE * abstract ,
const RPC_IFACE * transfer ,
2005-09-30 21:13:37 +04:00
RPC_HDR_AUTH * phdr_auth ,
prs_struct * pauth_info )
{
RPC_HDR hdr ;
RPC_HDR_RB hdr_rb ;
RPC_CONTEXT rpc_ctx ;
uint16 auth_len = prs_offset ( pauth_info ) ;
uint8 ss_padding_len = 0 ;
uint16 frag_len = 0 ;
2005-06-08 07:48:40 +04:00
/* create the RPC context. */
2005-09-30 21:13:37 +04:00
init_rpc_context ( & rpc_ctx , 0 /* context id */ , abstract , transfer ) ;
2005-06-08 07:48:40 +04:00
/* create the bind request RPC_HDR_RB */
2005-09-30 21:13:37 +04:00
init_rpc_hdr_rb ( & hdr_rb , RPC_MAX_PDU_FRAG_LEN , RPC_MAX_PDU_FRAG_LEN , 0x0 , & rpc_ctx ) ;
/* Start building the frag length. */
frag_len = RPC_HEADER_LEN + RPC_HDR_RB_LEN ( & hdr_rb ) ;
/* Do we need to pad ? */
if ( auth_len ) {
uint16 data_len = RPC_HEADER_LEN + RPC_HDR_RB_LEN ( & hdr_rb ) ;
if ( data_len % 8 ) {
ss_padding_len = 8 - ( data_len % 8 ) ;
phdr_auth - > auth_pad_len = ss_padding_len ;
}
frag_len + = RPC_HDR_AUTH_LEN + auth_len + ss_padding_len ;
}
2005-06-08 07:48:40 +04:00
2003-07-31 03:33:56 +04:00
/* Create the request RPC_HDR */
2005-09-30 21:13:37 +04:00
init_rpc_hdr ( & hdr , pkt_type , RPC_FLG_FIRST | RPC_FLG_LAST , rpc_call_id , frag_len , auth_len ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
2005-06-08 07:48:40 +04:00
/* Marshall the RPC header */
1999-12-13 16:27:58 +03:00
if ( ! smb_io_rpc_hdr ( " hdr " , & hdr , rpc_out , 0 ) ) {
2005-09-30 21:13:37 +04:00
DEBUG ( 0 , ( " create_bind_or_alt_ctx_internal: failed to marshall RPC_HDR. \n " ) ) ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
return NT_STATUS_NO_MEMORY ;
1999-12-13 16:27:58 +03:00
}
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
1999-12-13 16:27:58 +03:00
/* Marshall the bind request data */
if ( ! smb_io_rpc_hdr_rb ( " " , & hdr_rb , rpc_out , 0 ) ) {
2005-09-30 21:13:37 +04:00
DEBUG ( 0 , ( " create_bind_or_alt_ctx_internal: failed to marshall RPC_HDR_RB. \n " ) ) ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
return NT_STATUS_NO_MEMORY ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
}
1999-12-13 16:27:58 +03:00
/*
* Grow the outgoing buffer to store any auth info .
*/
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
if ( auth_len ! = 0 ) {
2005-09-30 21:13:37 +04:00
if ( ss_padding_len ) {
2005-10-18 07:24:00 +04:00
char pad [ 8 ] ;
2005-09-30 21:13:37 +04:00
memset ( pad , ' \0 ' , 8 ) ;
if ( ! prs_copy_data_in ( rpc_out , pad , ss_padding_len ) ) {
DEBUG ( 0 , ( " create_bind_or_alt_ctx_internal: failed to marshall padding. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
}
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , phdr_auth , rpc_out , 0 ) ) {
DEBUG ( 0 , ( " create_bind_or_alt_ctx_internal: failed to marshall RPC_HDR_AUTH. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
if ( ! prs_append_prs_data ( rpc_out , pauth_info ) ) {
DEBUG ( 0 , ( " create_bind_or_alt_ctx_internal: failed to grow parse struct to add auth. \n " ) ) ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
return NT_STATUS_NO_MEMORY ;
1999-12-13 16:27:58 +03:00
}
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
}
2005-09-30 21:13:37 +04:00
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
return NT_STATUS_OK ;
1998-03-12 00:11:04 +03:00
}
1998-10-09 03:57:46 +04:00
/*******************************************************************
2005-09-30 21:13:37 +04:00
Creates a DCE / RPC bind request .
1998-10-09 03:57:46 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
static NTSTATUS create_rpc_bind_req ( struct rpc_pipe_client * cli ,
prs_struct * rpc_out ,
uint32 rpc_call_id ,
2008-04-20 13:45:41 +04:00
const RPC_IFACE * abstract ,
const RPC_IFACE * transfer ,
2005-09-30 21:13:37 +04:00
enum pipe_auth_type auth_type ,
enum pipe_auth_level auth_level )
1998-10-09 03:57:46 +04:00
{
2005-07-28 00:25:04 +04:00
RPC_HDR_AUTH hdr_auth ;
2005-09-30 21:13:37 +04:00
prs_struct auth_info ;
NTSTATUS ret = NT_STATUS_OK ;
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
ZERO_STRUCT ( hdr_auth ) ;
2008-03-15 01:26:28 +03:00
if ( ! prs_init ( & auth_info , RPC_HDR_AUTH_LEN , prs_get_mem_context ( rpc_out ) , MARSHALL ) )
return NT_STATUS_NO_MEMORY ;
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
switch ( auth_type ) {
case PIPE_AUTH_TYPE_SCHANNEL :
ret = create_schannel_auth_rpc_bind_req ( cli , auth_level , & hdr_auth , & auth_info ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
prs_mem_free ( & auth_info ) ;
return ret ;
}
break ;
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_NTLMSSP :
ret = create_ntlmssp_auth_rpc_bind_req ( cli , auth_level , & hdr_auth , & auth_info ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
prs_mem_free ( & auth_info ) ;
return ret ;
}
break ;
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
ret = create_spnego_ntlmssp_auth_rpc_bind_req ( cli , auth_level , & hdr_auth , & auth_info ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
prs_mem_free ( & auth_info ) ;
return ret ;
}
break ;
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_KRB5 :
ret = create_krb5_auth_bind_req ( cli , auth_level , & hdr_auth , & auth_info ) ;
if ( ! NT_STATUS_IS_OK ( ret ) ) {
prs_mem_free ( & auth_info ) ;
return ret ;
}
break ;
case PIPE_AUTH_TYPE_NONE :
break ;
default :
/* "Can't" happen. */
return NT_STATUS_INVALID_INFO_CLASS ;
1999-12-13 16:27:58 +03:00
}
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
ret = create_bind_or_alt_ctx_internal ( RPC_BIND ,
rpc_out ,
rpc_call_id ,
abstract ,
transfer ,
& hdr_auth ,
& auth_info ) ;
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
prs_mem_free ( & auth_info ) ;
return ret ;
}
1998-10-09 03:57:46 +04:00
1998-03-12 00:11:04 +03:00
/*******************************************************************
2005-09-30 21:13:37 +04:00
Create and add the NTLMSSP sign / seal auth header and data .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-04-24 02:45:53 +04:00
2005-09-30 21:13:37 +04:00
static NTSTATUS add_ntlmssp_auth_footer ( struct rpc_pipe_client * cli ,
RPC_HDR * phdr ,
uint32 ss_padding_len ,
prs_struct * outgoing_pdu )
1998-03-12 00:11:04 +03:00
{
2005-09-30 21:13:37 +04:00
RPC_HDR_AUTH auth_info ;
NTSTATUS status ;
2007-05-14 16:16:20 +04:00
DATA_BLOB auth_blob = data_blob_null ;
2005-09-30 21:13:37 +04:00
uint16 data_and_pad_len = prs_offset ( outgoing_pdu ) - RPC_HEADER_LEN - RPC_HDR_RESP_LEN ;
1998-03-12 00:11:04 +03:00
2008-04-21 12:39:37 +04:00
if ( ! cli - > auth - > a_u . ntlmssp_state ) {
2005-09-30 21:13:37 +04:00
return NT_STATUS_INVALID_PARAMETER ;
}
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
/* Init and marshall the auth header. */
init_rpc_hdr_auth ( & auth_info ,
2008-04-21 12:39:37 +04:00
map_pipe_auth_type_to_rpc_auth_type (
cli - > auth - > auth_type ) ,
cli - > auth - > auth_level ,
2005-09-30 21:13:37 +04:00
ss_padding_len ,
1 /* context id. */ ) ;
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & auth_info , outgoing_pdu , 0 ) ) {
DEBUG ( 0 , ( " add_ntlmssp_auth_footer: failed to marshall RPC_HDR_AUTH. \n " ) ) ;
data_blob_free ( & auth_blob ) ;
return NT_STATUS_NO_MEMORY ;
}
2008-04-21 12:39:37 +04:00
switch ( cli - > auth - > auth_level ) {
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_LEVEL_PRIVACY :
/* Data portion is encrypted. */
2008-04-21 12:39:37 +04:00
status = ntlmssp_seal_packet ( cli - > auth - > a_u . ntlmssp_state ,
2005-10-18 07:24:00 +04:00
( unsigned char * ) prs_data_p ( outgoing_pdu ) + RPC_HEADER_LEN + RPC_HDR_RESP_LEN ,
2005-09-30 21:13:37 +04:00
data_and_pad_len ,
2005-10-18 07:24:00 +04:00
( unsigned char * ) prs_data_p ( outgoing_pdu ) ,
2005-09-30 21:13:37 +04:00
( size_t ) prs_offset ( outgoing_pdu ) ,
& auth_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
data_blob_free ( & auth_blob ) ;
return status ;
}
break ;
case PIPE_AUTH_LEVEL_INTEGRITY :
/* Data is signed. */
2008-04-21 12:39:37 +04:00
status = ntlmssp_sign_packet ( cli - > auth - > a_u . ntlmssp_state ,
2005-10-18 07:24:00 +04:00
( unsigned char * ) prs_data_p ( outgoing_pdu ) + RPC_HEADER_LEN + RPC_HDR_RESP_LEN ,
2005-09-30 21:13:37 +04:00
data_and_pad_len ,
2005-10-18 07:24:00 +04:00
( unsigned char * ) prs_data_p ( outgoing_pdu ) ,
2005-09-30 21:13:37 +04:00
( size_t ) prs_offset ( outgoing_pdu ) ,
& auth_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
data_blob_free ( & auth_blob ) ;
return status ;
}
break ;
default :
/* Can't happen. */
smb_panic ( " bad auth level " ) ;
/* Notreached. */
return NT_STATUS_INVALID_PARAMETER ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* Finally marshall the blob. */
2008-04-20 18:23:48 +04:00
2005-10-18 07:24:00 +04:00
if ( ! prs_copy_data_in ( outgoing_pdu , ( const char * ) auth_blob . data , NTLMSSP_SIG_SIZE ) ) {
2005-09-30 21:13:37 +04:00
DEBUG ( 0 , ( " add_ntlmssp_auth_footer: failed to add %u bytes auth blob. \n " ,
( unsigned int ) NTLMSSP_SIG_SIZE ) ) ;
data_blob_free ( & auth_blob ) ;
return NT_STATUS_NO_MEMORY ;
}
2008-04-20 18:23:48 +04:00
2005-09-30 21:13:37 +04:00
data_blob_free ( & auth_blob ) ;
return NT_STATUS_OK ;
}
1998-10-09 03:57:46 +04:00
2005-09-30 21:13:37 +04:00
/*******************************************************************
Create and add the schannel sign / seal auth header and data .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
static NTSTATUS add_schannel_auth_footer ( struct rpc_pipe_client * cli ,
RPC_HDR * phdr ,
uint32 ss_padding_len ,
prs_struct * outgoing_pdu )
{
RPC_HDR_AUTH auth_info ;
RPC_AUTH_SCHANNEL_CHK verf ;
2008-04-21 12:39:37 +04:00
struct schannel_auth_struct * sas = cli - > auth - > a_u . schannel_auth ;
2005-09-30 21:13:37 +04:00
char * data_p = prs_data_p ( outgoing_pdu ) + RPC_HEADER_LEN + RPC_HDR_RESP_LEN ;
size_t data_and_pad_len = prs_offset ( outgoing_pdu ) - RPC_HEADER_LEN - RPC_HDR_RESP_LEN ;
if ( ! sas ) {
return NT_STATUS_INVALID_PARAMETER ;
}
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
/* Init and marshall the auth header. */
init_rpc_hdr_auth ( & auth_info ,
2008-04-21 12:39:37 +04:00
map_pipe_auth_type_to_rpc_auth_type ( cli - > auth - > auth_type ) ,
cli - > auth - > auth_level ,
2005-09-30 21:13:37 +04:00
ss_padding_len ,
1 /* context id. */ ) ;
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & auth_info , outgoing_pdu , 0 ) ) {
DEBUG ( 0 , ( " add_schannel_auth_footer: failed to marshall RPC_HDR_AUTH. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
1998-03-12 00:11:04 +03:00
2008-04-21 12:39:37 +04:00
switch ( cli - > auth - > auth_level ) {
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_LEVEL_PRIVACY :
case PIPE_AUTH_LEVEL_INTEGRITY :
DEBUG ( 10 , ( " add_schannel_auth_footer: SCHANNEL seq_num=%d \n " ,
sas - > seq_num ) ) ;
schannel_encode ( sas ,
2008-04-21 12:39:37 +04:00
cli - > auth - > auth_level ,
2005-09-30 21:13:37 +04:00
SENDER_IS_INITIATOR ,
& verf ,
data_p ,
data_and_pad_len ) ;
sas - > seq_num + + ;
break ;
default :
/* Can't happen. */
smb_panic ( " bad auth level " ) ;
/* Notreached. */
return NT_STATUS_INVALID_PARAMETER ;
}
2002-11-07 21:54:02 +03:00
2005-09-30 21:13:37 +04:00
/* Finally marshall the blob. */
smb_io_rpc_auth_schannel_chk ( " " ,
RPC_AUTH_SCHANNEL_SIGN_OR_SEAL_CHK_LEN ,
& verf ,
outgoing_pdu ,
0 ) ;
2008-04-20 18:23:48 +04:00
2005-09-30 21:13:37 +04:00
return NT_STATUS_OK ;
2002-11-07 21:54:02 +03:00
}
/*******************************************************************
2005-09-30 21:13:37 +04:00
Calculate how much data we ' re going to send in this packet , also
work out any sign / seal padding length .
2003-04-09 19:54:17 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-09-30 21:13:37 +04:00
static uint32 calculate_data_len_tosend ( struct rpc_pipe_client * cli ,
uint32 data_left ,
uint16 * p_frag_len ,
uint16 * p_auth_len ,
uint32 * p_ss_padding )
2003-04-09 19:54:17 +04:00
{
2005-09-30 21:13:37 +04:00
uint32 data_space , data_len ;
2009-01-18 14:22:14 +03:00
# ifdef DEVELOPER
if ( ( data_left > 0 ) & & ( sys_random ( ) % 2 ) ) {
data_left = MAX ( data_left / 2 , 1 ) ;
}
# endif
2008-04-21 12:39:37 +04:00
switch ( cli - > auth - > auth_level ) {
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_LEVEL_NONE :
case PIPE_AUTH_LEVEL_CONNECT :
data_space = cli - > max_xmit_frag - RPC_HEADER_LEN - RPC_HDR_REQ_LEN ;
data_len = MIN ( data_space , data_left ) ;
* p_ss_padding = 0 ;
* p_auth_len = 0 ;
* p_frag_len = RPC_HEADER_LEN + RPC_HDR_REQ_LEN + data_len ;
return data_len ;
case PIPE_AUTH_LEVEL_INTEGRITY :
case PIPE_AUTH_LEVEL_PRIVACY :
/* Treat the same for all authenticated rpc requests. */
2008-04-21 12:39:37 +04:00
switch ( cli - > auth - > auth_type ) {
2005-09-30 21:13:37 +04:00
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
case PIPE_AUTH_TYPE_NTLMSSP :
* p_auth_len = NTLMSSP_SIG_SIZE ;
break ;
case PIPE_AUTH_TYPE_SCHANNEL :
* p_auth_len = RPC_AUTH_SCHANNEL_SIGN_OR_SEAL_CHK_LEN ;
break ;
default :
smb_panic ( " bad auth type " ) ;
break ;
}
2003-04-09 19:54:17 +04:00
2005-09-30 21:13:37 +04:00
data_space = cli - > max_xmit_frag - RPC_HEADER_LEN - RPC_HDR_REQ_LEN -
RPC_HDR_AUTH_LEN - * p_auth_len ;
data_len = MIN ( data_space , data_left ) ;
2009-01-17 21:19:04 +03:00
* p_ss_padding = 0 ;
2005-09-30 21:13:37 +04:00
if ( data_len % 8 ) {
* p_ss_padding = 8 - ( data_len % 8 ) ;
}
* p_frag_len = RPC_HEADER_LEN + RPC_HDR_REQ_LEN + /* Normal headers. */
data_len + * p_ss_padding + /* data plus padding. */
RPC_HDR_AUTH_LEN + * p_auth_len ; /* Auth header and auth data. */
return data_len ;
default :
smb_panic ( " bad auth level " ) ;
/* Notreached. */
return 0 ;
2003-04-09 19:54:17 +04:00
}
}
2005-09-30 21:13:37 +04:00
/*******************************************************************
External interface .
Does an rpc request on a pipe . Incoming data is NDR encoded in in_data .
Reply is NDR encoded in out_data . Splits the data stream into RPC PDU ' s
and deals with signing / sealing details .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
2009-01-17 19:52:35 +03:00
struct rpc_api_pipe_req_state {
struct event_context * ev ;
struct rpc_pipe_client * cli ;
uint8_t op_num ;
uint32_t call_id ;
prs_struct * req_data ;
uint32_t req_data_sent ;
prs_struct outgoing_frag ;
prs_struct reply_pdu ;
} ;
static int rpc_api_pipe_req_state_destructor ( struct rpc_api_pipe_req_state * s )
{
prs_mem_free ( & s - > outgoing_frag ) ;
prs_mem_free ( & s - > reply_pdu ) ;
return 0 ;
}
static void rpc_api_pipe_req_write_done ( struct async_req * subreq ) ;
static void rpc_api_pipe_req_done ( struct async_req * subreq ) ;
static NTSTATUS prepare_next_frag ( struct rpc_api_pipe_req_state * state ,
bool * is_last_frag ) ;
struct async_req * rpc_api_pipe_req_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_pipe_client * cli ,
uint8_t op_num ,
prs_struct * req_data )
{
struct async_req * result , * subreq ;
struct rpc_api_pipe_req_state * state ;
NTSTATUS status ;
bool is_last_frag ;
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct rpc_api_pipe_req_state ) ) {
2009-01-17 19:52:35 +03:00
return NULL ;
}
state - > ev = ev ;
state - > cli = cli ;
state - > op_num = op_num ;
state - > req_data = req_data ;
state - > req_data_sent = 0 ;
state - > call_id = get_rpc_call_id ( ) ;
if ( cli - > max_xmit_frag
< RPC_HEADER_LEN + RPC_HDR_REQ_LEN + RPC_MAX_SIGN_SIZE ) {
/* Server is screwed up ! */
status = NT_STATUS_INVALID_PARAMETER ;
goto post_status ;
}
prs_init_empty ( & state - > reply_pdu , state , UNMARSHALL ) ;
if ( ! prs_init ( & state - > outgoing_frag , cli - > max_xmit_frag ,
state , MARSHALL ) ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
talloc_set_destructor ( state , rpc_api_pipe_req_state_destructor ) ;
status = prepare_next_frag ( state , & is_last_frag ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto post_status ;
}
if ( is_last_frag ) {
subreq = rpc_api_pipe_send ( state , ev , state - > cli ,
& state - > outgoing_frag ,
RPC_RESPONSE ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq - > async . fn = rpc_api_pipe_req_done ;
subreq - > async . priv = result ;
} else {
subreq = rpc_write_send ( state , ev , cli ,
prs_data_p ( & state - > outgoing_frag ) ,
prs_offset ( & state - > outgoing_frag ) ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq - > async . fn = rpc_api_pipe_req_write_done ;
subreq - > async . priv = result ;
}
return result ;
post_status :
if ( async_post_status ( result , ev , status ) ) {
return result ;
}
TALLOC_FREE ( result ) ;
return NULL ;
}
static NTSTATUS prepare_next_frag ( struct rpc_api_pipe_req_state * state ,
bool * is_last_frag )
{
RPC_HDR hdr ;
RPC_HDR_REQ hdr_req ;
uint32_t data_sent_thistime ;
uint16_t auth_len ;
uint16_t frag_len ;
uint8_t flags = 0 ;
uint32_t ss_padding ;
uint32_t data_left ;
char pad [ 8 ] = { 0 , } ;
NTSTATUS status ;
data_left = prs_offset ( state - > req_data ) - state - > req_data_sent ;
data_sent_thistime = calculate_data_len_tosend (
state - > cli , data_left , & frag_len , & auth_len , & ss_padding ) ;
if ( state - > req_data_sent = = 0 ) {
flags = RPC_FLG_FIRST ;
}
if ( data_sent_thistime = = data_left ) {
flags | = RPC_FLG_LAST ;
}
if ( ! prs_set_offset ( & state - > outgoing_frag , 0 ) ) {
return NT_STATUS_NO_MEMORY ;
}
/* Create and marshall the header and request header. */
init_rpc_hdr ( & hdr , RPC_REQUEST , flags , state - > call_id , frag_len ,
auth_len ) ;
if ( ! smb_io_rpc_hdr ( " hdr " , & hdr , & state - > outgoing_frag , 0 ) ) {
return NT_STATUS_NO_MEMORY ;
}
/* Create the rpc request RPC_HDR_REQ */
init_rpc_hdr_req ( & hdr_req , prs_offset ( state - > req_data ) ,
state - > op_num ) ;
if ( ! smb_io_rpc_hdr_req ( " hdr_req " , & hdr_req ,
& state - > outgoing_frag , 0 ) ) {
return NT_STATUS_NO_MEMORY ;
}
/* Copy in the data, plus any ss padding. */
if ( ! prs_append_some_prs_data ( & state - > outgoing_frag ,
state - > req_data , state - > req_data_sent ,
data_sent_thistime ) ) {
return NT_STATUS_NO_MEMORY ;
}
/* Copy the sign/seal padding data. */
if ( ! prs_copy_data_in ( & state - > outgoing_frag , pad , ss_padding ) ) {
return NT_STATUS_NO_MEMORY ;
}
/* Generate any auth sign/seal and add the auth footer. */
switch ( state - > cli - > auth - > auth_type ) {
case PIPE_AUTH_TYPE_NONE :
status = NT_STATUS_OK ;
break ;
case PIPE_AUTH_TYPE_NTLMSSP :
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
status = add_ntlmssp_auth_footer ( state - > cli , & hdr , ss_padding ,
& state - > outgoing_frag ) ;
break ;
case PIPE_AUTH_TYPE_SCHANNEL :
status = add_schannel_auth_footer ( state - > cli , & hdr , ss_padding ,
& state - > outgoing_frag ) ;
break ;
default :
status = NT_STATUS_INVALID_PARAMETER ;
break ;
}
state - > req_data_sent + = data_sent_thistime ;
* is_last_frag = ( ( flags & RPC_FLG_LAST ) ! = 0 ) ;
return status ;
}
static void rpc_api_pipe_req_write_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_api_pipe_req_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_api_pipe_req_state ) ;
NTSTATUS status ;
bool is_last_frag ;
status = rpc_write_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
status = prepare_next_frag ( state , & is_last_frag ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
if ( is_last_frag ) {
subreq = rpc_api_pipe_send ( state , state - > ev , state - > cli ,
& state - > outgoing_frag ,
RPC_RESPONSE ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = rpc_api_pipe_req_done ;
subreq - > async . priv = req ;
} else {
subreq = rpc_write_send ( state , state - > ev , state - > cli ,
prs_data_p ( & state - > outgoing_frag ) ,
prs_offset ( & state - > outgoing_frag ) ) ;
if ( async_req_nomem ( subreq , req ) ) {
return ;
}
subreq - > async . fn = rpc_api_pipe_req_write_done ;
subreq - > async . priv = req ;
}
}
static void rpc_api_pipe_req_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_api_pipe_req_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_api_pipe_req_state ) ;
NTSTATUS status ;
status = rpc_api_pipe_recv ( subreq , state , & state - > reply_pdu ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
NTSTATUS rpc_api_pipe_req_recv ( struct async_req * req , TALLOC_CTX * mem_ctx ,
prs_struct * reply_pdu )
{
struct rpc_api_pipe_req_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_api_pipe_req_state ) ;
NTSTATUS status ;
if ( async_req_is_error ( req , & status ) ) {
return status ;
}
* reply_pdu = state - > reply_pdu ;
reply_pdu - > mem_ctx = mem_ctx ;
/*
* Prevent state - > req_pdu from being freed in
* rpc_api_pipe_req_state_destructor ( )
*/
prs_init_empty ( & state - > reply_pdu , state , UNMARSHALL ) ;
return NT_STATUS_OK ;
}
NTSTATUS rpc_api_pipe_req ( TALLOC_CTX * mem_ctx , struct rpc_pipe_client * cli ,
uint8 op_num ,
prs_struct * in_data ,
prs_struct * out_data )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct event_context * ev ;
struct async_req * req ;
NTSTATUS status = NT_STATUS_NO_MEMORY ;
ev = event_context_init ( frame ) ;
if ( ev = = NULL ) {
goto fail ;
}
req = rpc_api_pipe_req_send ( frame , ev , cli , op_num , in_data ) ;
if ( req = = NULL ) {
goto fail ;
}
while ( req - > state < ASYNC_REQ_DONE ) {
event_loop_once ( ev ) ;
}
status = rpc_api_pipe_req_recv ( req , mem_ctx , out_data ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}
2005-09-30 21:13:37 +04:00
#if 0
1998-03-12 00:11:04 +03:00
/****************************************************************************
1999-12-13 16:27:58 +03:00
Set the handle state .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-19 04:40:25 +04:00
static bool rpc_pipe_set_hnd_state ( struct rpc_pipe_client * cli ,
2005-06-09 02:10:34 +04:00
const char * pipe_name , uint16 device_state )
1998-04-24 02:45:53 +04:00
{
2007-10-19 04:40:25 +04:00
bool state_set = False ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
char param [ 2 ] ;
uint16 setup [ 2 ] ; /* only need 2 uint16 setup parameters */
char * rparam = NULL ;
char * rdata = NULL ;
uint32 rparam_len , rdata_len ;
1999-12-13 16:27:58 +03:00
if ( pipe_name = = NULL )
return False ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
DEBUG ( 5 , ( " Set Handle state Pipe[%x]: %s - device state:%x \n " ,
2005-06-09 02:10:34 +04:00
cli - > fnum , pipe_name , device_state ) ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
/* create parameters: device state */
SSVAL ( param , 0 , device_state ) ;
/* create setup parameters. */
setup [ 0 ] = 0x0001 ;
2005-06-09 02:10:34 +04:00
setup [ 1 ] = cli - > fnum ; /* pipe file handle. got this from an SMBOpenX. */
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
/* send the data on \PIPE\ */
2005-06-09 02:10:34 +04:00
if ( cli_api_pipe ( cli - > cli , " \\ PIPE \\ " ,
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
setup , 2 , 0 , /* setup, length, max */
param , 2 , 0 , /* param, length, max */
NULL , 0 , 1024 , /* data, length, max */
& rparam , & rparam_len , /* return param, length */
& rdata , & rdata_len ) ) /* return data, length */
{
DEBUG ( 5 , ( " Set Handle state: return OK \n " ) ) ;
state_set = True ;
}
2001-09-17 13:41:30 +04:00
SAFE_FREE ( rparam ) ;
SAFE_FREE ( rdata ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
return state_set ;
1998-03-12 00:11:04 +03:00
}
2005-09-30 21:13:37 +04:00
# endif
1998-03-12 00:11:04 +03:00
/****************************************************************************
2005-09-30 21:13:37 +04:00
Check the rpc bind acknowledge response .
1998-03-12 00:11:04 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-04-24 02:45:53 +04:00
2008-04-20 13:45:41 +04:00
static bool check_bind_response ( RPC_HDR_BA * hdr_ba , const RPC_IFACE * transfer )
1998-03-12 00:11:04 +03:00
{
2004-07-07 22:14:16 +04:00
if ( hdr_ba - > addr . len = = 0 ) {
DEBUG ( 4 , ( " Ignoring length check -- ASU bug (server didn't fill in the pipe name correctly) " ) ) ;
}
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
/* check the transfer syntax */
2008-04-15 22:26:52 +04:00
if ( ( hdr_ba - > transfer . if_version ! = transfer - > if_version ) | |
1999-12-13 16:27:58 +03:00
( memcmp ( & hdr_ba - > transfer . uuid , & transfer - > uuid , sizeof ( transfer - > uuid ) ) ! = 0 ) ) {
2002-11-01 03:37:45 +03:00
DEBUG ( 2 , ( " bind_rpc_pipe: transfer syntax differs \n " ) ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
return False ;
}
1999-12-13 16:27:58 +03:00
if ( hdr_ba - > res . num_results ! = 0x1 | | hdr_ba - > res . result ! = 0 ) {
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
DEBUG ( 2 , ( " bind_rpc_pipe: bind denied results: %d reason: %x \n " ,
hdr_ba - > res . num_results , hdr_ba - > res . reason ) ) ;
}
2005-09-30 21:13:37 +04:00
DEBUG ( 5 , ( " check_bind_response: accepted! \n " ) ) ;
Makefile.in :
- added srvsvc client files
clientgen.c :
- replaced cli_error(cli, int *cls, int *err) with
cli_error(cli, uint8 cls, uint32 *err). this version detects
32 bit status messages. the DOS error "MORE_DATA", the
equivalent of the 32 bit *warning* 0x8000 0005
(STATUS_BUFFER_OVERFLOW), was being processed as an error,
terminating the cli_receive_trans() call.
cli_pipe.c :
- replaced calls that had been incorrectly modified from
32 bit warnings (0x8000 0005 - STATUS_BUFFER_OVERFLOW)
to 8 bit DOS errors (0x01 0xEA - MORE_DATA).
the use of the old version of cli_error (DOS only)
instead of the new one (DOS and 32 bit) caused the
dce/rpc client code to fail.
- replaced 2 space indentation with tab indentation in all functions.
cli_srvsvc.c :
cmd_srvsvc.c :
- added these files back in, fixing them up to use jeremy's
modified versions of the dce/rpc client functions.
parse_srv.c :
- added back in some "unused" functions required by dce/rpc
client-side code. it would be helpful if all such "unused"
functions could be added back in.
rpcclient.c :
- added "session", "file", "share", "connection" enumeration
functions back in. these are equivalent to nt's "NetXXXXXEnum"
Win32 (MSDN) functions.
- added "srvinfo" function back in. this is equivalent to
nt's NetServerGetInfo Win32 (MSDN) function.
(This used to be commit bcf39ffdcc64e049bca2d70a394a99976291e81d)
1998-10-03 01:09:23 +04:00
return True ;
1998-03-12 00:11:04 +03:00
}
2005-09-30 21:13:37 +04:00
/*******************************************************************
Creates a DCE / RPC bind authentication response .
This is the packet that is sent back to the server once we
have received a BIND - ACK , to finish the third leg of
the authentication handshake .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-04-24 02:45:53 +04:00
2005-09-30 21:13:37 +04:00
static NTSTATUS create_rpc_bind_auth3 ( struct rpc_pipe_client * cli ,
uint32 rpc_call_id ,
enum pipe_auth_type auth_type ,
enum pipe_auth_level auth_level ,
DATA_BLOB * pauth_blob ,
prs_struct * rpc_out )
1998-03-12 00:11:04 +03:00
{
2005-09-30 21:13:37 +04:00
RPC_HDR hdr ;
RPC_HDR_AUTH hdr_auth ;
uint32 pad = 0 ;
/* Create the request RPC_HDR */
init_rpc_hdr ( & hdr , RPC_AUTH3 , RPC_FLG_FIRST | RPC_FLG_LAST , rpc_call_id ,
RPC_HEADER_LEN + 4 /* pad */ + RPC_HDR_AUTH_LEN + pauth_blob - > length ,
pauth_blob - > length ) ;
2008-04-20 18:23:48 +04:00
2005-09-30 21:13:37 +04:00
/* Marshall it. */
if ( ! smb_io_rpc_hdr ( " hdr " , & hdr , rpc_out , 0 ) ) {
DEBUG ( 0 , ( " create_rpc_bind_auth3: failed to marshall RPC_HDR. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/*
I ' m puzzled about this - seems to violate the DCE RPC auth rules ,
about padding - shouldn ' t this pad to length 8 ? JRA .
*/
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* 4 bytes padding. */
if ( ! prs_uint32 ( " pad " , rpc_out , 0 , & pad ) ) {
DEBUG ( 0 , ( " create_rpc_bind_auth3: failed to marshall 4 byte pad. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
2003-10-02 01:18:32 +04:00
}
1999-12-13 16:27:58 +03:00
2005-09-30 21:13:37 +04:00
/* Create the request RPC_HDR_AUTHA */
init_rpc_hdr_auth ( & hdr_auth ,
map_pipe_auth_type_to_rpc_auth_type ( auth_type ) ,
auth_level , 0 , 1 ) ;
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & hdr_auth , rpc_out , 0 ) ) {
DEBUG ( 0 , ( " create_rpc_bind_auth3: failed to marshall RPC_HDR_AUTHA. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
1998-10-09 03:57:46 +04:00
}
1998-03-12 00:11:04 +03:00
2005-09-30 21:13:37 +04:00
/*
* Append the auth data to the outgoing buffer .
*/
if ( ! prs_copy_data_in ( rpc_out , ( char * ) pauth_blob - > data , pauth_blob - > length ) ) {
DEBUG ( 0 , ( " create_rpc_bind_auth3: failed to marshall auth blob. \n " ) ) ;
return NT_STATUS_NO_MEMORY ;
}
return NT_STATUS_OK ;
1999-12-13 16:27:58 +03:00
}
2005-09-30 21:13:37 +04:00
/*******************************************************************
Creates a DCE / RPC bind alter context authentication request which
may contain a spnego auth blobl
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
static NTSTATUS create_rpc_alter_context ( uint32 rpc_call_id ,
2008-04-20 13:45:41 +04:00
const RPC_IFACE * abstract ,
const RPC_IFACE * transfer ,
2005-09-30 21:13:37 +04:00
enum pipe_auth_level auth_level ,
const DATA_BLOB * pauth_blob , /* spnego auth blob already created. */
prs_struct * rpc_out )
{
RPC_HDR_AUTH hdr_auth ;
prs_struct auth_info ;
NTSTATUS ret = NT_STATUS_OK ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
ZERO_STRUCT ( hdr_auth ) ;
2008-03-15 01:26:28 +03:00
if ( ! prs_init ( & auth_info , RPC_HDR_AUTH_LEN , prs_get_mem_context ( rpc_out ) , MARSHALL ) )
return NT_STATUS_NO_MEMORY ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
/* We may change the pad length before marshalling. */
init_rpc_hdr_auth ( & hdr_auth , RPC_SPNEGO_AUTH_TYPE , ( int ) auth_level , 0 , 1 ) ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
2005-09-30 21:13:37 +04:00
if ( pauth_blob - > length ) {
if ( ! prs_copy_data_in ( & auth_info , ( const char * ) pauth_blob - > data , pauth_blob - > length ) ) {
prs_mem_free ( & auth_info ) ;
return NT_STATUS_NO_MEMORY ;
Jeremy requested that I get my NTLMSSP patch into CVS. He didn't request
the schannel code, but I've included that anyway. :-)
This patch revives the client-side NTLMSSP support for RPC named pipes
in Samba, and cleans up the client and server schannel code. The use of the
new code is enabled by the 'sign', 'seal' and 'schannel' commands in
rpcclient.
The aim was to prove that our separate NTLMSSP client library actually
implements NTLMSSP signing and sealing as per Microsoft's NTLMv1 implementation,
in the hope that knowing this will assist us in correctly implementing
NTLMSSP signing for SMB packets. (Still not yet functional)
This patch replaces the NTLMSSP implementation in rpc_client/cli_pipe.c with
calls to libsmb/ntlmssp.c. In the process, we have gained the ability to
use the more secure NT password, and the ability to sign-only, instead of
having to seal the pipe connection. (Previously we were limited to sealing,
and could only use the LM-password derived key).
Our new client-side NTLMSSP code also needed alteration to cope with our
comparatively simple server-side implementation. A future step is to replace
it with calls to the same NTLMSSP library.
Also included in this patch is the schannel 'sign only' patch I submitted to
the team earlier. While not enabled (and not functional, at this stage) the
work in this patch makes the code paths *much* easier to follow. I have also
included similar hooks in rpccleint to allow the use of schannel on *any* pipe.
rpcclient now defaults to not using schannel (or any other extra per-pipe
authenticiation) for any connection. The 'schannel' command enables schannel
for all pipes until disabled.
This code is also much more secure than the previous code, as changes to our
cli_pipe routines ensure that the authentication footer cannot be removed
by an attacker, and more error states are correctly handled.
(The same needs to be done to our server)
Andrew Bartlett
(This used to be commit 5472ddc9eaf4e79c5b2e1c8ee8c7f190dc285f19)
2003-07-14 12:46:32 +04:00
}
}
2005-09-30 21:13:37 +04:00
ret = create_bind_or_alt_ctx_internal ( RPC_ALTCONT ,
rpc_out ,
rpc_call_id ,
abstract ,
transfer ,
& hdr_auth ,
& auth_info ) ;
prs_mem_free ( & auth_info ) ;
return ret ;
}
1998-10-15 09:47:29 +04:00
2003-04-09 19:54:17 +04:00
/****************************************************************************
2005-09-30 21:13:37 +04:00
Do an rpc bind .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-04-09 19:54:17 +04:00
2009-01-18 14:12:15 +03:00
struct rpc_pipe_bind_state {
struct event_context * ev ;
struct rpc_pipe_client * cli ;
prs_struct rpc_out ;
uint32_t rpc_call_id ;
} ;
static int rpc_pipe_bind_state_destructor ( struct rpc_pipe_bind_state * state )
{
prs_mem_free ( & state - > rpc_out ) ;
return 0 ;
}
static void rpc_pipe_bind_step_one_done ( struct async_req * subreq ) ;
static NTSTATUS rpc_finish_auth3_bind_send ( struct async_req * req ,
struct rpc_pipe_bind_state * state ,
struct rpc_hdr_info * phdr ,
prs_struct * reply_pdu ) ;
static void rpc_bind_auth3_write_done ( struct async_req * subreq ) ;
static NTSTATUS rpc_finish_spnego_ntlmssp_bind_send ( struct async_req * req ,
struct rpc_pipe_bind_state * state ,
struct rpc_hdr_info * phdr ,
prs_struct * reply_pdu ) ;
static void rpc_bind_ntlmssp_api_done ( struct async_req * subreq ) ;
struct async_req * rpc_pipe_bind_send ( TALLOC_CTX * mem_ctx ,
struct event_context * ev ,
struct rpc_pipe_client * cli ,
struct cli_pipe_auth_data * auth )
{
struct async_req * result , * subreq ;
struct rpc_pipe_bind_state * state ;
NTSTATUS status ;
2009-01-18 18:38:30 +03:00
if ( ! async_req_setup ( mem_ctx , & result , & state ,
struct rpc_pipe_bind_state ) ) {
2009-01-18 14:12:15 +03:00
return NULL ;
}
DEBUG ( 5 , ( " Bind RPC Pipe: %s auth_type %u, auth_level %u \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , cli ) ,
( unsigned int ) auth - > auth_type ,
( unsigned int ) auth - > auth_level ) ) ;
state - > ev = ev ;
state - > cli = cli ;
state - > rpc_call_id = get_rpc_call_id ( ) ;
prs_init_empty ( & state - > rpc_out , state , MARSHALL ) ;
talloc_set_destructor ( state , rpc_pipe_bind_state_destructor ) ;
cli - > auth = talloc_move ( cli , & auth ) ;
/* Marshall the outgoing data. */
status = create_rpc_bind_req ( cli , & state - > rpc_out ,
state - > rpc_call_id ,
& cli - > abstract_syntax ,
& cli - > transfer_syntax ,
cli - > auth - > auth_type ,
cli - > auth - > auth_level ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto post_status ;
}
subreq = rpc_api_pipe_send ( state , ev , cli , & state - > rpc_out ,
RPC_BINDACK ) ;
if ( subreq = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto post_status ;
}
subreq - > async . fn = rpc_pipe_bind_step_one_done ;
subreq - > async . priv = result ;
return result ;
post_status :
if ( async_post_status ( result , ev , status ) ) {
return result ;
}
TALLOC_FREE ( result ) ;
return NULL ;
}
static void rpc_pipe_bind_step_one_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_pipe_bind_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_pipe_bind_state ) ;
prs_struct reply_pdu ;
struct rpc_hdr_info hdr ;
struct rpc_hdr_ba_info hdr_ba ;
NTSTATUS status ;
status = rpc_api_pipe_recv ( subreq , talloc_tos ( ) , & reply_pdu ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 3 , ( " rpc_pipe_bind: %s bind request returned %s \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , state - > cli ) ,
nt_errstr ( status ) ) ) ;
async_req_error ( req , status ) ;
return ;
}
/* Unmarshall the RPC header */
if ( ! smb_io_rpc_hdr ( " hdr " , & hdr , & reply_pdu , 0 ) ) {
DEBUG ( 0 , ( " rpc_pipe_bind: failed to unmarshall RPC_HDR. \n " ) ) ;
async_req_error ( req , NT_STATUS_BUFFER_TOO_SMALL ) ;
return ;
}
if ( ! smb_io_rpc_hdr_ba ( " " , & hdr_ba , & reply_pdu , 0 ) ) {
DEBUG ( 0 , ( " rpc_pipe_bind: Failed to unmarshall "
" RPC_HDR_BA. \n " ) ) ;
async_req_error ( req , NT_STATUS_BUFFER_TOO_SMALL ) ;
return ;
}
if ( ! check_bind_response ( & hdr_ba , & state - > cli - > transfer_syntax ) ) {
DEBUG ( 2 , ( " rpc_pipe_bind: check_bind_response failed. \n " ) ) ;
async_req_error ( req , NT_STATUS_BUFFER_TOO_SMALL ) ;
return ;
}
state - > cli - > max_xmit_frag = hdr_ba . bba . max_tsize ;
state - > cli - > max_recv_frag = hdr_ba . bba . max_rsize ;
/*
* For authenticated binds we may need to do 3 or 4 leg binds .
*/
switch ( state - > cli - > auth - > auth_type ) {
case PIPE_AUTH_TYPE_NONE :
case PIPE_AUTH_TYPE_SCHANNEL :
/* Bind complete. */
async_req_done ( req ) ;
break ;
case PIPE_AUTH_TYPE_NTLMSSP :
/* Need to send AUTH3 packet - no reply. */
status = rpc_finish_auth3_bind_send ( req , state , & hdr ,
& reply_pdu ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
}
break ;
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
/* Need to send alter context request and reply. */
status = rpc_finish_spnego_ntlmssp_bind_send ( req , state , & hdr ,
& reply_pdu ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
}
break ;
case PIPE_AUTH_TYPE_KRB5 :
/* */
default :
DEBUG ( 0 , ( " cli_finish_bind_auth: unknown auth type %u \n " ,
( unsigned int ) state - > cli - > auth - > auth_type ) ) ;
async_req_error ( req , NT_STATUS_INTERNAL_ERROR ) ;
}
}
static NTSTATUS rpc_finish_auth3_bind_send ( struct async_req * req ,
struct rpc_pipe_bind_state * state ,
struct rpc_hdr_info * phdr ,
prs_struct * reply_pdu )
{
DATA_BLOB server_response = data_blob_null ;
DATA_BLOB client_reply = data_blob_null ;
struct rpc_hdr_auth_info hdr_auth ;
struct async_req * subreq ;
NTSTATUS status ;
if ( ( phdr - > auth_len = = 0 )
| | ( phdr - > frag_len < phdr - > auth_len + RPC_HDR_AUTH_LEN ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
if ( ! prs_set_offset (
reply_pdu ,
phdr - > frag_len - phdr - > auth_len - RPC_HDR_AUTH_LEN ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & hdr_auth , reply_pdu , 0 ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
/* TODO - check auth_type/auth_level match. */
server_response = data_blob_talloc ( talloc_tos ( ) , NULL , phdr - > auth_len ) ;
prs_copy_data_out ( ( char * ) server_response . data , reply_pdu ,
phdr - > auth_len ) ;
status = ntlmssp_update ( state - > cli - > auth - > a_u . ntlmssp_state ,
server_response , & client_reply ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " rpc_finish_auth3_bind: NTLMSSP update using server "
" blob failed: %s. \n " , nt_errstr ( status ) ) ) ;
return status ;
}
prs_init_empty ( & state - > rpc_out , talloc_tos ( ) , MARSHALL ) ;
status = create_rpc_bind_auth3 ( state - > cli , state - > rpc_call_id ,
state - > cli - > auth - > auth_type ,
state - > cli - > auth - > auth_level ,
& client_reply , & state - > rpc_out ) ;
data_blob_free ( & client_reply ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
subreq = rpc_write_send ( state , state - > ev , state - > cli ,
prs_data_p ( & state - > rpc_out ) ,
prs_offset ( & state - > rpc_out ) ) ;
if ( subreq = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
subreq - > async . fn = rpc_bind_auth3_write_done ;
subreq - > async . priv = req ;
return NT_STATUS_OK ;
}
static void rpc_bind_auth3_write_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
NTSTATUS status ;
status = rpc_write_recv ( subreq ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
async_req_done ( req ) ;
}
static NTSTATUS rpc_finish_spnego_ntlmssp_bind_send ( struct async_req * req ,
struct rpc_pipe_bind_state * state ,
struct rpc_hdr_info * phdr ,
prs_struct * reply_pdu )
{
DATA_BLOB server_spnego_response = data_blob_null ;
DATA_BLOB server_ntlm_response = data_blob_null ;
DATA_BLOB client_reply = data_blob_null ;
DATA_BLOB tmp_blob = data_blob_null ;
RPC_HDR_AUTH hdr_auth ;
struct async_req * subreq ;
NTSTATUS status ;
if ( ( phdr - > auth_len = = 0 )
| | ( phdr - > frag_len < phdr - > auth_len + RPC_HDR_AUTH_LEN ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
/* Process the returned NTLMSSP blob first. */
if ( ! prs_set_offset (
reply_pdu ,
phdr - > frag_len - phdr - > auth_len - RPC_HDR_AUTH_LEN ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & hdr_auth , reply_pdu , 0 ) ) {
return NT_STATUS_INVALID_PARAMETER ;
}
server_spnego_response = data_blob ( NULL , phdr - > auth_len ) ;
prs_copy_data_out ( ( char * ) server_spnego_response . data ,
reply_pdu , phdr - > auth_len ) ;
/*
* The server might give us back two challenges - tmp_blob is for the
* second .
*/
if ( ! spnego_parse_challenge ( server_spnego_response ,
& server_ntlm_response , & tmp_blob ) ) {
data_blob_free ( & server_spnego_response ) ;
data_blob_free ( & server_ntlm_response ) ;
data_blob_free ( & tmp_blob ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
/* We're finished with the server spnego response and the tmp_blob. */
data_blob_free ( & server_spnego_response ) ;
data_blob_free ( & tmp_blob ) ;
status = ntlmssp_update ( state - > cli - > auth - > a_u . ntlmssp_state ,
server_ntlm_response , & client_reply ) ;
/* Finished with the server_ntlm response */
data_blob_free ( & server_ntlm_response ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " rpc_finish_spnego_ntlmssp_bind: NTLMSSP update "
" using server blob failed. \n " ) ) ;
data_blob_free ( & client_reply ) ;
return status ;
}
/* SPNEGO wrap the client reply. */
tmp_blob = spnego_gen_auth ( client_reply ) ;
data_blob_free ( & client_reply ) ;
client_reply = tmp_blob ;
tmp_blob = data_blob_null ;
/* Now prepare the alter context pdu. */
prs_init_empty ( & state - > rpc_out , state , MARSHALL ) ;
status = create_rpc_alter_context ( state - > rpc_call_id ,
& state - > cli - > abstract_syntax ,
& state - > cli - > transfer_syntax ,
state - > cli - > auth - > auth_level ,
& client_reply ,
& state - > rpc_out ) ;
data_blob_free ( & client_reply ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
subreq = rpc_api_pipe_send ( state , state - > ev , state - > cli ,
& state - > rpc_out , RPC_ALTCONTRESP ) ;
if ( subreq = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
subreq - > async . fn = rpc_bind_ntlmssp_api_done ;
subreq - > async . priv = req ;
return NT_STATUS_OK ;
}
static void rpc_bind_ntlmssp_api_done ( struct async_req * subreq )
{
struct async_req * req = talloc_get_type_abort (
subreq - > async . priv , struct async_req ) ;
struct rpc_pipe_bind_state * state = talloc_get_type_abort (
req - > private_data , struct rpc_pipe_bind_state ) ;
DATA_BLOB server_spnego_response = data_blob_null ;
DATA_BLOB tmp_blob = data_blob_null ;
prs_struct reply_pdu ;
struct rpc_hdr_info hdr ;
struct rpc_hdr_auth_info hdr_auth ;
NTSTATUS status ;
status = rpc_api_pipe_recv ( subreq , talloc_tos ( ) , & reply_pdu ) ;
TALLOC_FREE ( subreq ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
async_req_error ( req , status ) ;
return ;
}
/* Get the auth blob from the reply. */
if ( ! smb_io_rpc_hdr ( " rpc_hdr " , & hdr , & reply_pdu , 0 ) ) {
DEBUG ( 0 , ( " rpc_finish_spnego_ntlmssp_bind: Failed to "
" unmarshall RPC_HDR. \n " ) ) ;
async_req_error ( req , NT_STATUS_BUFFER_TOO_SMALL ) ;
return ;
}
if ( ! prs_set_offset (
& reply_pdu ,
hdr . frag_len - hdr . auth_len - RPC_HDR_AUTH_LEN ) ) {
async_req_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
return ;
}
if ( ! smb_io_rpc_hdr_auth ( " hdr_auth " , & hdr_auth , & reply_pdu , 0 ) ) {
async_req_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
return ;
}
server_spnego_response = data_blob ( NULL , hdr . auth_len ) ;
prs_copy_data_out ( ( char * ) server_spnego_response . data , & reply_pdu ,
hdr . auth_len ) ;
/* Check we got a valid auth response. */
if ( ! spnego_parse_auth_response ( server_spnego_response , NT_STATUS_OK ,
OID_NTLMSSP , & tmp_blob ) ) {
data_blob_free ( & server_spnego_response ) ;
data_blob_free ( & tmp_blob ) ;
async_req_error ( req , NT_STATUS_INVALID_PARAMETER ) ;
return ;
}
data_blob_free ( & server_spnego_response ) ;
data_blob_free ( & tmp_blob ) ;
DEBUG ( 5 , ( " rpc_finish_spnego_ntlmssp_bind: alter context request to "
" %s. \n " , rpccli_pipe_txt ( debug_ctx ( ) , state - > cli ) ) ) ;
async_req_done ( req ) ;
}
NTSTATUS rpc_pipe_bind_recv ( struct async_req * req )
{
return async_req_simple_recv ( req ) ;
}
NTSTATUS rpc_pipe_bind ( struct rpc_pipe_client * cli ,
struct cli_pipe_auth_data * auth )
{
TALLOC_CTX * frame = talloc_stackframe ( ) ;
struct event_context * ev ;
struct async_req * req ;
NTSTATUS status = NT_STATUS_NO_MEMORY ;
ev = event_context_init ( frame ) ;
if ( ev = = NULL ) {
goto fail ;
}
req = rpc_pipe_bind_send ( frame , ev , cli , auth ) ;
if ( req = = NULL ) {
goto fail ;
}
while ( req - > state < ASYNC_REQ_DONE ) {
event_loop_once ( ev ) ;
}
status = rpc_pipe_bind_recv ( req ) ;
fail :
TALLOC_FREE ( frame ) ;
return status ;
}
2003-07-25 05:26:19 +04:00
2008-04-20 01:27:35 +04:00
unsigned int rpccli_set_timeout ( struct rpc_pipe_client * cli ,
unsigned int timeout )
{
2008-04-24 22:42:32 +04:00
return cli_set_timeout ( cli - > trans . np . cli , timeout ) ;
2008-04-20 01:27:35 +04:00
}
2008-04-21 10:01:51 +04:00
bool rpccli_get_pwd_hash ( struct rpc_pipe_client * cli , uint8_t nt_hash [ 16 ] )
{
2008-04-24 22:42:32 +04:00
if ( ( cli - > auth - > auth_type = = PIPE_AUTH_TYPE_NTLMSSP )
| | ( cli - > auth - > auth_type = = PIPE_AUTH_TYPE_SPNEGO_NTLMSSP ) ) {
memcpy ( nt_hash , cli - > auth - > a_u . ntlmssp_state - > nt_hash , 16 ) ;
2008-04-21 10:01:51 +04:00
return true ;
}
2008-04-24 22:42:32 +04:00
if ( cli - > transport_type = = NCACN_NP ) {
E_md4hash ( cli - > trans . np . cli - > pwd . password , nt_hash ) ;
return true ;
}
return false ;
2008-04-21 10:01:51 +04:00
}
2008-04-20 16:05:25 +04:00
struct cli_state * rpc_pipe_np_smb_conn ( struct rpc_pipe_client * p )
{
2008-04-24 22:42:32 +04:00
if ( p - > transport_type = = NCACN_NP ) {
return p - > trans . np . cli ;
}
return NULL ;
2008-04-20 16:05:25 +04:00
}
2008-04-20 15:51:46 +04:00
static int rpc_pipe_destructor ( struct rpc_pipe_client * p )
{
2008-04-24 22:42:32 +04:00
if ( p - > transport_type = = NCACN_NP ) {
bool ret ;
ret = cli_close ( p - > trans . np . cli , p - > trans . np . fnum ) ;
if ( ! ret ) {
DEBUG ( 1 , ( " rpc_pipe_destructor: cli_close failed on "
" pipe %s. Error was %s \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , p ) ,
cli_errstr ( p - > trans . np . cli ) ) ) ;
}
2008-04-20 15:51:46 +04:00
2008-04-24 22:42:32 +04:00
DEBUG ( 10 , ( " rpc_pipe_destructor: closed %s \n " ,
rpccli_pipe_txt ( debug_ctx ( ) , p ) ) ) ;
2008-04-20 15:51:46 +04:00
2008-04-24 22:42:32 +04:00
DLIST_REMOVE ( p - > trans . np . cli - > pipe_list , p ) ;
return ret ? - 1 : 0 ;
}
2008-04-20 15:51:46 +04:00
2008-04-24 22:42:32 +04:00
return - 1 ;
2008-04-20 15:51:46 +04:00
}
2008-04-22 15:03:06 +04:00
NTSTATUS rpccli_anon_bind_data ( TALLOC_CTX * mem_ctx ,
struct cli_pipe_auth_data * * presult )
{
struct cli_pipe_auth_data * result ;
result = talloc ( mem_ctx , struct cli_pipe_auth_data ) ;
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > auth_type = PIPE_AUTH_TYPE_NONE ;
result - > auth_level = PIPE_AUTH_LEVEL_NONE ;
result - > user_name = talloc_strdup ( result , " " ) ;
result - > domain = talloc_strdup ( result , " " ) ;
if ( ( result - > user_name = = NULL ) | | ( result - > domain = = NULL ) ) {
TALLOC_FREE ( result ) ;
return NT_STATUS_NO_MEMORY ;
}
* presult = result ;
return NT_STATUS_OK ;
}
static int cli_auth_ntlmssp_data_destructor ( struct cli_pipe_auth_data * auth )
{
ntlmssp_end ( & auth - > a_u . ntlmssp_state ) ;
return 0 ;
}
NTSTATUS rpccli_ntlmssp_bind_data ( TALLOC_CTX * mem_ctx ,
enum pipe_auth_type auth_type ,
enum pipe_auth_level auth_level ,
const char * domain ,
const char * username ,
const char * password ,
struct cli_pipe_auth_data * * presult )
{
struct cli_pipe_auth_data * result ;
NTSTATUS status ;
result = talloc ( mem_ctx , struct cli_pipe_auth_data ) ;
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > auth_type = auth_type ;
result - > auth_level = auth_level ;
result - > user_name = talloc_strdup ( result , username ) ;
result - > domain = talloc_strdup ( result , domain ) ;
if ( ( result - > user_name = = NULL ) | | ( result - > domain = = NULL ) ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
status = ntlmssp_client_start ( & result - > a_u . ntlmssp_state ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
talloc_set_destructor ( result , cli_auth_ntlmssp_data_destructor ) ;
status = ntlmssp_set_username ( result - > a_u . ntlmssp_state , username ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
status = ntlmssp_set_domain ( result - > a_u . ntlmssp_state , domain ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
status = ntlmssp_set_password ( result - > a_u . ntlmssp_state , password ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto fail ;
}
/*
* Turn off sign + seal to allow selected auth level to turn it back on .
*/
result - > a_u . ntlmssp_state - > neg_flags & =
~ ( NTLMSSP_NEGOTIATE_SIGN | NTLMSSP_NEGOTIATE_SEAL ) ;
if ( auth_level = = PIPE_AUTH_LEVEL_INTEGRITY ) {
result - > a_u . ntlmssp_state - > neg_flags | = NTLMSSP_NEGOTIATE_SIGN ;
} else if ( auth_level = = PIPE_AUTH_LEVEL_PRIVACY ) {
result - > a_u . ntlmssp_state - > neg_flags
| = NTLMSSP_NEGOTIATE_SEAL | NTLMSSP_NEGOTIATE_SIGN ;
}
* presult = result ;
return NT_STATUS_OK ;
fail :
TALLOC_FREE ( result ) ;
return status ;
}
NTSTATUS rpccli_schannel_bind_data ( TALLOC_CTX * mem_ctx , const char * domain ,
enum pipe_auth_level auth_level ,
2008-05-13 01:35:27 +04:00
const uint8_t sess_key [ 16 ] ,
2008-04-22 15:03:06 +04:00
struct cli_pipe_auth_data * * presult )
{
struct cli_pipe_auth_data * result ;
result = talloc ( mem_ctx , struct cli_pipe_auth_data ) ;
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > auth_type = PIPE_AUTH_TYPE_SCHANNEL ;
result - > auth_level = auth_level ;
result - > user_name = talloc_strdup ( result , " " ) ;
result - > domain = talloc_strdup ( result , domain ) ;
if ( ( result - > user_name = = NULL ) | | ( result - > domain = = NULL ) ) {
goto fail ;
}
result - > a_u . schannel_auth = talloc ( result ,
struct schannel_auth_struct ) ;
if ( result - > a_u . schannel_auth = = NULL ) {
goto fail ;
}
2008-05-13 01:35:27 +04:00
memcpy ( result - > a_u . schannel_auth - > sess_key , sess_key ,
sizeof ( result - > a_u . schannel_auth - > sess_key ) ) ;
2008-04-22 15:03:06 +04:00
result - > a_u . schannel_auth - > seq_num = 0 ;
* presult = result ;
return NT_STATUS_OK ;
fail :
TALLOC_FREE ( result ) ;
return NT_STATUS_NO_MEMORY ;
}
2008-06-24 17:53:23 +04:00
# ifdef HAVE_KRB5
2008-04-22 15:03:06 +04:00
static int cli_auth_kerberos_data_destructor ( struct kerberos_auth_struct * auth )
{
data_blob_free ( & auth - > session_key ) ;
return 0 ;
}
2008-06-24 17:53:23 +04:00
# endif
2008-04-22 15:03:06 +04:00
2008-04-22 16:31:35 +04:00
NTSTATUS rpccli_kerberos_bind_data ( TALLOC_CTX * mem_ctx ,
enum pipe_auth_level auth_level ,
const char * service_princ ,
const char * username ,
const char * password ,
struct cli_pipe_auth_data * * presult )
2008-04-22 15:03:06 +04:00
{
2008-04-25 14:36:39 +04:00
# ifdef HAVE_KRB5
2008-04-22 15:03:06 +04:00
struct cli_pipe_auth_data * result ;
if ( ( username ! = NULL ) & & ( password ! = NULL ) ) {
int ret = kerberos_kinit_password ( username , password , 0 , NULL ) ;
if ( ret ! = 0 ) {
return NT_STATUS_ACCESS_DENIED ;
}
}
result = talloc ( mem_ctx , struct cli_pipe_auth_data ) ;
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > auth_type = PIPE_AUTH_TYPE_KRB5 ;
result - > auth_level = auth_level ;
/*
* Username / domain need fixing !
*/
result - > user_name = talloc_strdup ( result , " " ) ;
result - > domain = talloc_strdup ( result , " " ) ;
if ( ( result - > user_name = = NULL ) | | ( result - > domain = = NULL ) ) {
goto fail ;
}
result - > a_u . kerberos_auth = TALLOC_ZERO_P (
result , struct kerberos_auth_struct ) ;
if ( result - > a_u . kerberos_auth = = NULL ) {
goto fail ;
}
talloc_set_destructor ( result - > a_u . kerberos_auth ,
cli_auth_kerberos_data_destructor ) ;
result - > a_u . kerberos_auth - > service_principal = talloc_strdup (
result , service_princ ) ;
if ( result - > a_u . kerberos_auth - > service_principal = = NULL ) {
goto fail ;
}
* presult = result ;
return NT_STATUS_OK ;
fail :
TALLOC_FREE ( result ) ;
return NT_STATUS_NO_MEMORY ;
2008-04-25 14:36:39 +04:00
# else
return NT_STATUS_NOT_SUPPORTED ;
# endif
2008-04-22 15:03:06 +04:00
}
2008-05-20 20:25:42 +04:00
static int rpc_pipe_sock_destructor ( struct rpc_pipe_client * p )
{
close ( p - > trans . sock . fd ) ;
return 0 ;
}
2008-05-20 20:08:41 +04:00
/**
* Create an rpc pipe client struct , connecting to a tcp port .
*/
2008-05-22 13:16:57 +04:00
static NTSTATUS rpc_pipe_open_tcp_port ( TALLOC_CTX * mem_ctx , const char * host ,
uint16_t port ,
const struct ndr_syntax_id * abstract_syntax ,
struct rpc_pipe_client * * presult )
2008-04-25 00:24:20 +04:00
{
struct rpc_pipe_client * result ;
struct sockaddr_storage addr ;
NTSTATUS status ;
2008-06-11 12:44:47 +04:00
result = TALLOC_ZERO_P ( mem_ctx , struct rpc_pipe_client ) ;
2008-04-25 00:24:20 +04:00
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > transport_type = NCACN_IP_TCP ;
2008-07-20 12:21:14 +04:00
result - > abstract_syntax = * abstract_syntax ;
result - > transfer_syntax = ndr_transfer_syntax ;
2009-01-21 16:05:51 +03:00
result - > dispatch = cli_do_rpc_ndr ;
2008-04-25 00:24:20 +04:00
result - > desthost = talloc_strdup ( result , host ) ;
result - > srv_name_slash = talloc_asprintf_strupper_m (
result , " \\ \\ %s " , result - > desthost ) ;
if ( ( result - > desthost = = NULL ) | | ( result - > srv_name_slash = = NULL ) ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
result - > max_xmit_frag = RPC_MAX_PDU_FRAG_LEN ;
result - > max_recv_frag = RPC_MAX_PDU_FRAG_LEN ;
if ( ! resolve_name ( host , & addr , 0 ) ) {
status = NT_STATUS_NOT_FOUND ;
goto fail ;
}
2009-01-03 21:50:05 +03:00
status = open_socket_out ( & addr , port , 60 , & result - > trans . sock . fd ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-25 00:24:20 +04:00
goto fail ;
}
2008-05-20 20:25:42 +04:00
talloc_set_destructor ( result , rpc_pipe_sock_destructor ) ;
2008-04-25 00:24:20 +04:00
* presult = result ;
return NT_STATUS_OK ;
fail :
TALLOC_FREE ( result ) ;
return status ;
}
2008-05-20 20:08:41 +04:00
/**
* Determine the tcp port on which a dcerpc interface is listening
* for the ncacn_ip_tcp transport via the endpoint mapper of the
* target host .
*/
2008-05-22 13:16:57 +04:00
static NTSTATUS rpc_pipe_get_tcp_port ( const char * host ,
const struct ndr_syntax_id * abstract_syntax ,
uint16_t * pport )
2008-05-20 20:08:41 +04:00
{
NTSTATUS status ;
struct rpc_pipe_client * epm_pipe = NULL ;
struct cli_pipe_auth_data * auth = NULL ;
struct dcerpc_binding * map_binding = NULL ;
struct dcerpc_binding * res_binding = NULL ;
struct epm_twr_t * map_tower = NULL ;
struct epm_twr_t * res_towers = NULL ;
struct policy_handle * entry_handle = NULL ;
uint32_t num_towers = 0 ;
uint32_t max_towers = 1 ;
struct epm_twr_p_t towers ;
TALLOC_CTX * tmp_ctx = talloc_stackframe ( ) ;
if ( pport = = NULL ) {
status = NT_STATUS_INVALID_PARAMETER ;
goto done ;
}
/* open the connection to the endpoint mapper */
status = rpc_pipe_open_tcp_port ( tmp_ctx , host , 135 ,
& ndr_table_epmapper . syntax_id ,
& epm_pipe ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
status = rpccli_anon_bind_data ( tmp_ctx , & auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
status = rpc_pipe_bind ( epm_pipe , auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
/* create tower for asking the epmapper */
map_binding = TALLOC_ZERO_P ( tmp_ctx , struct dcerpc_binding ) ;
if ( map_binding = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
map_binding - > transport = NCACN_IP_TCP ;
map_binding - > object = * abstract_syntax ;
map_binding - > host = host ; /* needed? */
map_binding - > endpoint = " 0 " ; /* correct? needed? */
map_tower = TALLOC_ZERO_P ( tmp_ctx , struct epm_twr_t ) ;
if ( map_tower = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
status = dcerpc_binding_build_tower ( tmp_ctx , map_binding ,
& ( map_tower - > tower ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
/* allocate further parameters for the epm_Map call */
res_towers = TALLOC_ARRAY ( tmp_ctx , struct epm_twr_t , max_towers ) ;
if ( res_towers = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
towers . twr = res_towers ;
entry_handle = TALLOC_ZERO_P ( tmp_ctx , struct policy_handle ) ;
if ( entry_handle = = NULL ) {
status = NT_STATUS_NO_MEMORY ;
goto done ;
}
/* ask the endpoint mapper for the port */
status = rpccli_epm_Map ( epm_pipe ,
tmp_ctx ,
2008-05-25 15:44:35 +04:00
CONST_DISCARD ( struct GUID * ,
& ( abstract_syntax - > uuid ) ) ,
2008-05-20 20:08:41 +04:00
map_tower ,
entry_handle ,
max_towers ,
& num_towers ,
& towers ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
if ( num_towers ! = 1 ) {
status = NT_STATUS_UNSUCCESSFUL ;
goto done ;
}
/* extract the port from the answer */
status = dcerpc_binding_from_tower ( tmp_ctx ,
& ( towers . twr - > tower ) ,
& res_binding ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
/* are further checks here necessary? */
if ( res_binding - > transport ! = NCACN_IP_TCP ) {
status = NT_STATUS_UNSUCCESSFUL ;
goto done ;
}
* pport = ( uint16_t ) atoi ( res_binding - > endpoint ) ;
done :
TALLOC_FREE ( tmp_ctx ) ;
return status ;
}
/**
* Create a rpc pipe client struct , connecting to a host via tcp .
* The port is determined by asking the endpoint mapper on the given
* host .
*/
NTSTATUS rpc_pipe_open_tcp ( TALLOC_CTX * mem_ctx , const char * host ,
const struct ndr_syntax_id * abstract_syntax ,
struct rpc_pipe_client * * presult )
{
NTSTATUS status ;
2008-05-23 17:09:21 +04:00
uint16_t port = 0 ;
2008-05-20 20:08:41 +04:00
2008-06-11 12:44:47 +04:00
* presult = NULL ;
2008-05-20 20:08:41 +04:00
status = rpc_pipe_get_tcp_port ( host , abstract_syntax , & port ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
goto done ;
}
status = rpc_pipe_open_tcp_port ( mem_ctx , host , port ,
abstract_syntax , presult ) ;
done :
return status ;
}
2008-05-13 01:37:07 +04:00
/********************************************************************
Create a rpc pipe client struct , connecting to a unix domain socket
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
NTSTATUS rpc_pipe_open_ncalrpc ( TALLOC_CTX * mem_ctx , const char * socket_path ,
const struct ndr_syntax_id * abstract_syntax ,
struct rpc_pipe_client * * presult )
{
struct rpc_pipe_client * result ;
struct sockaddr_un addr ;
NTSTATUS status ;
2008-07-20 12:40:43 +04:00
result = talloc_zero ( mem_ctx , struct rpc_pipe_client ) ;
2008-05-13 01:37:07 +04:00
if ( result = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
result - > transport_type = NCACN_UNIX_STREAM ;
2008-07-20 12:21:14 +04:00
result - > abstract_syntax = * abstract_syntax ;
result - > transfer_syntax = ndr_transfer_syntax ;
2009-01-21 16:05:51 +03:00
result - > dispatch = cli_do_rpc_ndr ;
2008-05-13 01:37:07 +04:00
2008-10-17 14:48:19 +04:00
result - > desthost = talloc_get_myname ( result ) ;
2008-05-13 01:37:07 +04:00
result - > srv_name_slash = talloc_asprintf_strupper_m (
result , " \\ \\ %s " , result - > desthost ) ;
if ( ( result - > desthost = = NULL ) | | ( result - > srv_name_slash = = NULL ) ) {
status = NT_STATUS_NO_MEMORY ;
goto fail ;
}
result - > max_xmit_frag = RPC_MAX_PDU_FRAG_LEN ;
result - > max_recv_frag = RPC_MAX_PDU_FRAG_LEN ;
result - > trans . sock . fd = socket ( AF_UNIX , SOCK_STREAM , 0 ) ;
if ( result - > trans . sock . fd = = - 1 ) {
status = map_nt_error_from_unix ( errno ) ;
goto fail ;
}
2008-05-20 20:25:42 +04:00
talloc_set_destructor ( result , rpc_pipe_sock_destructor ) ;
2008-05-13 01:37:07 +04:00
ZERO_STRUCT ( addr ) ;
addr . sun_family = AF_UNIX ;
strncpy ( addr . sun_path , socket_path , sizeof ( addr . sun_path ) ) ;
if ( sys_connect ( result - > trans . sock . fd ,
( struct sockaddr * ) & addr ) = = - 1 ) {
DEBUG ( 0 , ( " connect(%s) failed: %s \n " , socket_path ,
strerror ( errno ) ) ) ;
close ( result - > trans . sock . fd ) ;
return map_nt_error_from_unix ( errno ) ;
}
* presult = result ;
return NT_STATUS_OK ;
fail :
TALLOC_FREE ( result ) ;
return status ;
}
2008-04-25 00:24:20 +04:00
2005-09-30 21:13:37 +04:00
/****************************************************************************
Open a named pipe over SMB to a remote server .
2005-12-14 07:00:58 +03:00
*
* CAVEAT CALLER OF THIS FUNCTION :
* The returned rpc_pipe_client saves a copy of the cli_state cli pointer ,
* so be sure that this function is called AFTER any structure ( vs pointer )
* assignment of the cli . In particular , libsmbclient does structure
* assignments of cli , which invalidates the data in the returned
* rpc_pipe_client if this function is called before the structure assignment
* of cli .
*
2005-09-30 21:13:37 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-01-02 08:39:49 +03:00
2008-07-20 12:56:36 +04:00
static NTSTATUS rpc_pipe_open_np ( struct cli_state * cli ,
const struct ndr_syntax_id * abstract_syntax ,
struct rpc_pipe_client * * presult )
2005-06-09 02:10:34 +04:00
{
struct rpc_pipe_client * result ;
int fnum ;
2006-03-15 17:58:39 +03:00
/* sanity check to protect against crashes */
if ( ! cli ) {
2008-07-20 12:56:36 +04:00
return NT_STATUS_INVALID_HANDLE ;
2006-03-15 17:58:39 +03:00
}
2008-04-19 20:17:13 +04:00
result = TALLOC_ZERO_P ( NULL , struct rpc_pipe_client ) ;
2005-09-30 21:13:37 +04:00
if ( result = = NULL ) {
2008-07-20 12:56:36 +04:00
return NT_STATUS_NO_MEMORY ;
2005-09-30 21:13:37 +04:00
}
2005-06-09 02:10:34 +04:00
2008-04-24 22:42:32 +04:00
result - > transport_type = NCACN_NP ;
2008-07-20 12:56:36 +04:00
result - > trans . np . pipe_name = cli_get_pipe_name_from_iface (
result , cli , abstract_syntax ) ;
if ( result - > trans . np . pipe_name = = NULL ) {
DEBUG ( 1 , ( " Could not find pipe for interface \n " ) ) ;
TALLOC_FREE ( result ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
2005-09-30 21:13:37 +04:00
2008-04-24 22:42:32 +04:00
result - > trans . np . cli = cli ;
2008-07-20 12:56:36 +04:00
result - > abstract_syntax = * abstract_syntax ;
2008-07-20 12:21:14 +04:00
result - > transfer_syntax = ndr_transfer_syntax ;
2009-01-21 16:05:51 +03:00
result - > dispatch = cli_do_rpc_ndr ;
2008-04-19 23:56:43 +04:00
result - > desthost = talloc_strdup ( result , cli - > desthost ) ;
2008-04-20 01:03:16 +04:00
result - > srv_name_slash = talloc_asprintf_strupper_m (
result , " \\ \\ %s " , result - > desthost ) ;
2008-04-20 14:19:27 +04:00
2009-01-14 14:58:13 +03:00
result - > max_xmit_frag = RPC_MAX_PDU_FRAG_LEN ;
result - > max_recv_frag = RPC_MAX_PDU_FRAG_LEN ;
2008-04-22 16:31:35 +04:00
if ( ( result - > desthost = = NULL ) | | ( result - > srv_name_slash = = NULL ) ) {
2008-04-20 01:03:16 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 12:56:36 +04:00
return NT_STATUS_NO_MEMORY ;
2008-04-20 01:03:16 +04:00
}
2008-04-24 22:42:32 +04:00
fnum = cli_nt_create ( cli , result - > trans . np . pipe_name ,
DESIRED_ACCESS_PIPE ) ;
2008-04-20 14:19:27 +04:00
if ( fnum = = - 1 ) {
2008-10-16 05:07:15 +04:00
DEBUG ( 3 , ( " rpc_pipe_open_np: cli_nt_create failed on pipe %s "
2008-04-20 14:19:27 +04:00
" to machine %s. Error was %s \n " ,
2008-04-24 22:42:32 +04:00
result - > trans . np . pipe_name , cli - > desthost ,
2008-04-20 14:19:27 +04:00
cli_errstr ( cli ) ) ) ;
2008-07-20 12:56:36 +04:00
TALLOC_FREE ( result ) ;
return cli_get_nt_error ( cli ) ;
2008-04-20 14:19:27 +04:00
}
2008-04-24 22:42:32 +04:00
result - > trans . np . fnum = fnum ;
2008-04-20 14:19:27 +04:00
2005-09-30 21:13:37 +04:00
DLIST_ADD ( cli - > pipe_list , result ) ;
2008-04-20 15:51:46 +04:00
talloc_set_destructor ( result , rpc_pipe_destructor ) ;
2008-07-20 12:56:36 +04:00
* presult = result ;
return NT_STATUS_OK ;
2005-06-09 02:10:34 +04:00
}
2008-05-24 03:29:13 +04:00
/****************************************************************************
Open a pipe to a remote server .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
static NTSTATUS cli_rpc_pipe_open ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
struct rpc_pipe_client * * presult )
2008-05-24 03:29:13 +04:00
{
2008-07-20 13:04:31 +04:00
if ( ndr_syntax_id_equal ( interface , & ndr_table_drsuapi . syntax_id ) ) {
/*
* We should have a better way to figure out this drsuapi
* speciality . . .
*/
return rpc_pipe_open_tcp ( NULL , cli - > desthost , interface ,
presult ) ;
2008-05-24 03:29:13 +04:00
}
2008-07-20 13:04:31 +04:00
return rpc_pipe_open_np ( cli , interface , presult ) ;
2008-05-24 03:29:13 +04:00
}
2005-09-30 21:13:37 +04:00
/****************************************************************************
Open a named pipe to an SMB server and bind anonymously .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_noauth ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
struct rpc_pipe_client * * presult )
2005-06-09 02:10:34 +04:00
{
struct rpc_pipe_client * result ;
2008-04-22 16:31:35 +04:00
struct cli_pipe_auth_data * auth ;
2008-07-20 13:04:31 +04:00
NTSTATUS status ;
2005-06-09 02:10:34 +04:00
2008-07-20 13:04:31 +04:00
status = cli_rpc_pipe_open ( cli , interface , & result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2005-09-30 21:13:37 +04:00
}
2005-06-09 02:10:34 +04:00
2008-07-20 13:04:31 +04:00
status = rpccli_anon_bind_data ( result , & auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-22 16:31:35 +04:00
DEBUG ( 0 , ( " rpccli_anon_bind_data returned %s \n " ,
2008-07-20 13:04:31 +04:00
nt_errstr ( status ) ) ) ;
2008-04-22 16:31:35 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2008-04-22 16:31:35 +04:00
}
2008-04-22 20:17:01 +04:00
/*
* This is a bit of an abstraction violation due to the fact that an
* anonymous bind on an authenticated SMB inherits the user / domain
* from the enclosing SMB creds
*/
TALLOC_FREE ( auth - > user_name ) ;
TALLOC_FREE ( auth - > domain ) ;
auth - > user_name = talloc_strdup ( auth , cli - > user_name ) ;
auth - > domain = talloc_strdup ( auth , cli - > domain ) ;
2008-12-04 20:23:45 +03:00
auth - > user_session_key = data_blob_talloc ( auth ,
cli - > user_session_key . data ,
cli - > user_session_key . length ) ;
2008-04-22 20:17:01 +04:00
if ( ( auth - > user_name = = NULL ) | | ( auth - > domain = = NULL ) ) {
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return NT_STATUS_NO_MEMORY ;
2008-04-22 20:17:01 +04:00
}
2008-07-20 13:04:31 +04:00
status = rpc_pipe_bind ( result , auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2007-02-16 16:30:19 +03:00
int lvl = 0 ;
2008-07-20 13:04:31 +04:00
if ( ndr_syntax_id_equal ( interface ,
& ndr_table_dssetup . syntax_id ) ) {
2007-02-16 16:30:19 +03:00
/* non AD domains just don't have this pipe, avoid
* level 0 statement in that case - gd */
lvl = 3 ;
}
2008-07-20 13:04:31 +04:00
DEBUG ( lvl , ( " cli_rpc_pipe_open_noauth: rpc_pipe_bind for pipe "
" %s failed with error %s \n " ,
cli_get_pipe_name_from_iface ( debug_ctx ( ) , cli ,
interface ) ,
nt_errstr ( status ) ) ) ;
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-06-09 02:10:34 +04:00
}
2008-04-24 22:42:32 +04:00
DEBUG ( 10 , ( " cli_rpc_pipe_open_noauth: opened pipe %s to machine "
" %s and bound anonymously. \n " , result - > trans . np . pipe_name ,
cli - > desthost ) ) ;
2005-09-30 21:13:37 +04:00
2008-07-20 13:04:31 +04:00
* presult = result ;
return NT_STATUS_OK ;
2005-06-09 02:10:34 +04:00
}
2005-09-30 21:13:37 +04:00
/****************************************************************************
Open a named pipe to an SMB server and bind using NTLMSSP or SPNEGO NTLMSSP
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
static NTSTATUS cli_rpc_pipe_open_ntlmssp_internal ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_type auth_type ,
enum pipe_auth_level auth_level ,
const char * domain ,
const char * username ,
const char * password ,
struct rpc_pipe_client * * presult )
2005-06-09 02:10:34 +04:00
{
struct rpc_pipe_client * result ;
2008-04-22 16:31:35 +04:00
struct cli_pipe_auth_data * auth ;
2008-07-20 13:04:31 +04:00
NTSTATUS status ;
2005-06-09 02:10:34 +04:00
2008-07-20 13:04:31 +04:00
status = cli_rpc_pipe_open ( cli , interface , & result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2005-09-30 21:13:37 +04:00
}
2008-04-20 18:23:48 +04:00
2008-07-20 13:04:31 +04:00
status = rpccli_ntlmssp_bind_data (
2008-04-22 16:31:35 +04:00
result , auth_type , auth_level , domain , username ,
cli - > pwd . null_pwd ? NULL : password , & auth ) ;
2008-07-20 13:04:31 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-22 16:31:35 +04:00
DEBUG ( 0 , ( " rpccli_ntlmssp_bind_data returned %s \n " ,
2008-07-20 13:04:31 +04:00
nt_errstr ( status ) ) ) ;
goto err ;
2005-09-30 21:13:37 +04:00
}
2008-04-20 18:23:48 +04:00
2008-07-20 13:04:31 +04:00
status = rpc_pipe_bind ( result , auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-09-30 21:13:37 +04:00
DEBUG ( 0 , ( " cli_rpc_pipe_open_ntlmssp_internal: cli_rpc_pipe_bind failed with error %s \n " ,
2008-07-20 13:04:31 +04:00
nt_errstr ( status ) ) ) ;
2005-09-30 21:13:37 +04:00
goto err ;
}
2007-12-18 09:58:22 +03:00
DEBUG ( 10 , ( " cli_rpc_pipe_open_ntlmssp_internal: opened pipe %s to "
" machine %s and bound NTLMSSP as user %s \\ %s. \n " ,
2008-04-24 22:42:32 +04:00
result - > trans . np . pipe_name , cli - > desthost ,
2005-09-30 21:13:37 +04:00
domain , username ) ) ;
2008-07-20 13:04:31 +04:00
* presult = result ;
return NT_STATUS_OK ;
2005-09-30 21:13:37 +04:00
err :
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-06-09 02:10:34 +04:00
}
2005-09-30 21:13:37 +04:00
/****************************************************************************
External interface .
Open a named pipe to an SMB server and bind using NTLMSSP ( bind type 10 )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_ntlmssp ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_level auth_level ,
const char * domain ,
const char * username ,
const char * password ,
struct rpc_pipe_client * * presult )
2005-09-30 21:13:37 +04:00
{
return cli_rpc_pipe_open_ntlmssp_internal ( cli ,
2008-07-20 13:04:31 +04:00
interface ,
2005-09-30 21:13:37 +04:00
PIPE_AUTH_TYPE_NTLMSSP ,
auth_level ,
domain ,
username ,
password ,
2008-07-20 13:04:31 +04:00
presult ) ;
2005-09-30 21:13:37 +04:00
}
/****************************************************************************
External interface .
Open a named pipe to an SMB server and bind using spnego NTLMSSP ( bind type 9 )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_spnego_ntlmssp ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_level auth_level ,
const char * domain ,
const char * username ,
const char * password ,
struct rpc_pipe_client * * presult )
2005-09-30 21:13:37 +04:00
{
return cli_rpc_pipe_open_ntlmssp_internal ( cli ,
2008-07-20 13:04:31 +04:00
interface ,
2005-09-30 21:13:37 +04:00
PIPE_AUTH_TYPE_SPNEGO_NTLMSSP ,
auth_level ,
domain ,
username ,
password ,
2008-07-20 13:04:31 +04:00
presult ) ;
2005-09-30 21:13:37 +04:00
}
/****************************************************************************
2007-12-11 16:36:11 +03:00
Get a the schannel session key out of an already opened netlogon pipe .
2005-09-30 21:13:37 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 18:27:45 +04:00
static NTSTATUS get_schannel_session_key_common ( struct rpc_pipe_client * netlogon_pipe ,
struct cli_state * cli ,
const char * domain ,
uint32 * pneg_flags )
2005-09-30 21:13:37 +04:00
{
uint32 sec_chan_type = 0 ;
2005-10-18 07:24:00 +04:00
unsigned char machine_pwd [ 16 ] ;
2007-12-11 15:05:44 +03:00
const char * machine_account ;
2008-07-20 18:27:45 +04:00
NTSTATUS status ;
2005-09-30 21:13:37 +04:00
/* Get the machine account credentials from secrets.tdb. */
2007-12-11 16:07:32 +03:00
if ( ! get_trust_pw_hash ( domain , machine_pwd , & machine_account ,
& sec_chan_type ) )
2007-12-11 15:05:44 +03:00
{
2005-09-30 21:13:37 +04:00
DEBUG ( 0 , ( " get_schannel_session_key: could not fetch "
" trust account password for domain '%s' \n " ,
domain ) ) ;
2008-07-20 18:27:45 +04:00
return NT_STATUS_CANT_ACCESS_DOMAIN_INFO ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 18:27:45 +04:00
status = rpccli_netlogon_setup_creds ( netlogon_pipe ,
2005-11-04 03:03:55 +03:00
cli - > desthost , /* server name */
domain , /* domain */
global_myname ( ) , /* client name */
machine_account , /* machine account name */
2005-09-30 21:13:37 +04:00
machine_pwd ,
sec_chan_type ,
2006-02-23 00:18:23 +03:00
pneg_flags ) ;
2005-09-30 21:13:37 +04:00
2008-07-20 18:27:45 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 3 , ( " get_schannel_session_key_common: "
" rpccli_netlogon_setup_creds failed with result %s "
" to server %s, domain %s, machine account %s. \n " ,
nt_errstr ( status ) , cli - > desthost , domain ,
machine_account ) ) ;
return status ;
2005-09-30 21:13:37 +04:00
}
2006-02-23 00:18:23 +03:00
if ( ( ( * pneg_flags ) & NETLOGON_NEG_SCHANNEL ) = = 0 ) {
2005-09-30 21:13:37 +04:00
DEBUG ( 3 , ( " get_schannel_session_key: Server %s did not offer schannel \n " ,
cli - > desthost ) ) ;
2008-07-20 18:27:45 +04:00
return NT_STATUS_INVALID_NETWORK_RESPONSE ;
2007-12-11 16:36:11 +03:00
}
2008-07-20 18:27:45 +04:00
return NT_STATUS_OK ; ;
2007-12-11 16:36:11 +03:00
}
/****************************************************************************
Open a netlogon pipe and get the schannel session key .
Now exposed to external callers .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 18:33:26 +04:00
NTSTATUS get_schannel_session_key ( struct cli_state * cli ,
const char * domain ,
uint32 * pneg_flags ,
struct rpc_pipe_client * * presult )
2007-12-11 16:36:11 +03:00
{
struct rpc_pipe_client * netlogon_pipe = NULL ;
2008-07-20 18:33:26 +04:00
NTSTATUS status ;
2007-12-11 16:36:11 +03:00
2008-07-20 18:33:26 +04:00
status = cli_rpc_pipe_open_noauth ( cli , & ndr_table_netlogon . syntax_id ,
& netlogon_pipe ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2007-12-11 16:36:11 +03:00
}
2008-07-20 18:33:26 +04:00
status = get_schannel_session_key_common ( netlogon_pipe , cli , domain ,
pneg_flags ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( netlogon_pipe ) ;
2008-07-20 18:33:26 +04:00
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 18:33:26 +04:00
* presult = netlogon_pipe ;
return NT_STATUS_OK ;
2005-09-30 21:13:37 +04:00
}
/****************************************************************************
External interface .
Open a named pipe to an SMB server and bind using schannel ( bind type 68 )
using session_key . sign and seal .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_schannel_with_key ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_level auth_level ,
const char * domain ,
const struct dcinfo * pdc ,
struct rpc_pipe_client * * presult )
2005-06-09 02:10:34 +04:00
{
struct rpc_pipe_client * result ;
2008-04-22 16:31:35 +04:00
struct cli_pipe_auth_data * auth ;
2008-07-20 13:04:31 +04:00
NTSTATUS status ;
2005-06-09 02:10:34 +04:00
2008-07-20 13:04:31 +04:00
status = cli_rpc_pipe_open ( cli , interface , & result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 13:04:31 +04:00
status = rpccli_schannel_bind_data ( result , domain , auth_level ,
pdc - > sess_key , & auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-22 16:31:35 +04:00
DEBUG ( 0 , ( " rpccli_schannel_bind_data returned %s \n " ,
2008-07-20 13:04:31 +04:00
nt_errstr ( status ) ) ) ;
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 13:04:31 +04:00
status = rpc_pipe_bind ( result , auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " cli_rpc_pipe_open_schannel_with_key: "
" cli_rpc_pipe_bind failed with error %s \n " ,
nt_errstr ( status ) ) ) ;
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-06-09 02:10:34 +04:00
}
2008-07-20 12:40:43 +04:00
/*
* The credentials on a new netlogon pipe are the ones we are passed
* in - copy them over .
*/
result - > dc = ( struct dcinfo * ) talloc_memdup ( result , pdc , sizeof ( * pdc ) ) ;
if ( result - > dc = = NULL ) {
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return NT_STATUS_NO_MEMORY ;
2005-09-30 21:13:37 +04:00
}
DEBUG ( 10 , ( " cli_rpc_pipe_open_schannel_with_key: opened pipe %s to machine %s "
" for domain %s "
" and bound using schannel. \n " ,
2008-04-24 22:42:32 +04:00
result - > trans . np . pipe_name , cli - > desthost , domain ) ) ;
2005-09-30 21:13:37 +04:00
2008-07-20 13:04:31 +04:00
* presult = result ;
return NT_STATUS_OK ;
2005-06-09 02:10:34 +04:00
}
2005-10-06 20:51:10 +04:00
/****************************************************************************
Open a named pipe to an SMB server and bind using schannel ( bind type 68 ) .
Fetch the session key ourselves using a temporary netlogon pipe . This
version uses an ntlmssp auth bound netlogon pipe to get the key .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 18:36:55 +04:00
static NTSTATUS get_schannel_session_key_auth_ntlmssp ( struct cli_state * cli ,
const char * domain ,
const char * username ,
const char * password ,
uint32 * pneg_flags ,
struct rpc_pipe_client * * presult )
2005-10-06 20:51:10 +04:00
{
struct rpc_pipe_client * netlogon_pipe = NULL ;
2008-07-20 18:36:55 +04:00
NTSTATUS status ;
2005-10-06 20:51:10 +04:00
2008-07-20 18:36:55 +04:00
status = cli_rpc_pipe_open_spnego_ntlmssp (
cli , & ndr_table_netlogon . syntax_id , PIPE_AUTH_LEVEL_PRIVACY ,
domain , username , password , & netlogon_pipe ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2005-10-06 20:51:10 +04:00
}
2008-07-20 18:36:55 +04:00
status = get_schannel_session_key_common ( netlogon_pipe , cli , domain ,
pneg_flags ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( netlogon_pipe ) ;
2008-07-20 18:36:55 +04:00
return status ;
2005-10-06 20:51:10 +04:00
}
2008-07-20 18:36:55 +04:00
* presult = netlogon_pipe ;
return NT_STATUS_OK ;
2005-10-06 20:51:10 +04:00
}
/****************************************************************************
Open a named pipe to an SMB server and bind using schannel ( bind type 68 ) .
Fetch the session key ourselves using a temporary netlogon pipe . This version
uses an ntlmssp bind to get the session key .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_ntlmssp_auth_schannel ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_level auth_level ,
const char * domain ,
const char * username ,
const char * password ,
struct rpc_pipe_client * * presult )
2005-10-06 20:51:10 +04:00
{
2008-04-02 04:29:48 +04:00
uint32_t neg_flags = NETLOGON_NEG_AUTH2_ADS_FLAGS ;
2005-10-06 20:51:10 +04:00
struct rpc_pipe_client * netlogon_pipe = NULL ;
struct rpc_pipe_client * result = NULL ;
2008-07-20 13:04:31 +04:00
NTSTATUS status ;
2005-10-06 20:51:10 +04:00
2008-07-20 13:04:31 +04:00
status = get_schannel_session_key_auth_ntlmssp (
2008-07-20 18:36:55 +04:00
cli , domain , username , password , & neg_flags , & netlogon_pipe ) ;
2008-07-20 13:04:31 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-10-06 20:51:10 +04:00
DEBUG ( 0 , ( " cli_rpc_pipe_open_ntlmssp_auth_schannel: failed to get schannel session "
" key from server %s for domain %s. \n " ,
cli - > desthost , domain ) ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-10-06 20:51:10 +04:00
}
2008-07-20 13:04:31 +04:00
status = cli_rpc_pipe_open_schannel_with_key (
cli , interface , auth_level , domain , netlogon_pipe - > dc ,
& result ) ;
2005-10-06 20:51:10 +04:00
/* Now we've bound using the session key we can close the netlog pipe. */
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( netlogon_pipe ) ;
2005-10-06 20:51:10 +04:00
2008-07-20 13:04:31 +04:00
if ( NT_STATUS_IS_OK ( status ) ) {
* presult = result ;
}
return status ;
2005-10-06 20:51:10 +04:00
}
2005-09-30 21:13:37 +04:00
/****************************************************************************
Open a named pipe to an SMB server and bind using schannel ( bind type 68 ) .
Fetch the session key ourselves using a temporary netlogon pipe .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_schannel ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_level auth_level ,
const char * domain ,
struct rpc_pipe_client * * presult )
2005-06-09 02:10:34 +04:00
{
2008-04-02 04:29:48 +04:00
uint32_t neg_flags = NETLOGON_NEG_AUTH2_ADS_FLAGS ;
2005-09-30 21:13:37 +04:00
struct rpc_pipe_client * netlogon_pipe = NULL ;
struct rpc_pipe_client * result = NULL ;
2008-07-20 13:04:31 +04:00
NTSTATUS status ;
2005-09-30 21:13:37 +04:00
2008-07-20 13:04:31 +04:00
status = get_schannel_session_key ( cli , domain , & neg_flags ,
& netlogon_pipe ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2005-09-30 21:13:37 +04:00
DEBUG ( 0 , ( " cli_rpc_pipe_open_schannel: failed to get schannel session "
" key from server %s for domain %s. \n " ,
cli - > desthost , domain ) ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 13:04:31 +04:00
status = cli_rpc_pipe_open_schannel_with_key (
cli , interface , auth_level , domain , netlogon_pipe - > dc ,
& result ) ;
2005-09-30 21:13:37 +04:00
/* Now we've bound using the session key we can close the netlog pipe. */
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( netlogon_pipe ) ;
2005-09-30 21:13:37 +04:00
2008-07-20 13:04:31 +04:00
if ( NT_STATUS_IS_OK ( status ) ) {
* presult = result ;
}
return NT_STATUS_OK ;
2005-09-30 21:13:37 +04:00
}
/****************************************************************************
Open a named pipe to an SMB server and bind using krb5 ( bind type 16 ) .
2005-10-06 03:02:59 +04:00
The idea is this can be called with service_princ , username and password all
NULL so long as the caller has a TGT .
2005-09-30 21:13:37 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-07-20 13:04:31 +04:00
NTSTATUS cli_rpc_pipe_open_krb5 ( struct cli_state * cli ,
const struct ndr_syntax_id * interface ,
enum pipe_auth_level auth_level ,
const char * service_princ ,
const char * username ,
const char * password ,
struct rpc_pipe_client * * presult )
2005-09-30 21:13:37 +04:00
{
# ifdef HAVE_KRB5
struct rpc_pipe_client * result ;
2008-04-22 16:31:35 +04:00
struct cli_pipe_auth_data * auth ;
2008-07-20 13:04:31 +04:00
NTSTATUS status ;
2005-09-30 21:13:37 +04:00
2008-07-20 13:04:31 +04:00
status = cli_rpc_pipe_open ( cli , interface , & result ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 13:04:31 +04:00
status = rpccli_kerberos_bind_data ( result , auth_level , service_princ ,
username , password , & auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
2008-04-22 16:31:35 +04:00
DEBUG ( 0 , ( " rpccli_kerberos_bind_data returned %s \n " ,
2008-07-20 13:04:31 +04:00
nt_errstr ( status ) ) ) ;
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 13:04:31 +04:00
status = rpc_pipe_bind ( result , auth ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
DEBUG ( 0 , ( " cli_rpc_pipe_open_krb5: cli_rpc_pipe_bind failed "
" with error %s \n " , nt_errstr ( status ) ) ) ;
2008-04-20 15:51:46 +04:00
TALLOC_FREE ( result ) ;
2008-07-20 13:04:31 +04:00
return status ;
2005-09-30 21:13:37 +04:00
}
2008-07-20 13:04:31 +04:00
* presult = result ;
return NT_STATUS_OK ;
2005-09-30 21:13:37 +04:00
# else
DEBUG ( 0 , ( " cli_rpc_pipe_open_krb5: kerberos not found at compile time. \n " ) ) ;
2008-07-20 13:04:31 +04:00
return NT_STATUS_NOT_IMPLEMENTED ;
2005-09-30 21:13:37 +04:00
# endif
2005-06-09 02:10:34 +04:00
}
2008-06-10 23:35:34 +04:00
2008-06-25 02:21:37 +04:00
NTSTATUS cli_get_session_key ( TALLOC_CTX * mem_ctx ,
struct rpc_pipe_client * cli ,
2008-06-10 23:35:34 +04:00
DATA_BLOB * session_key )
{
if ( ! session_key | | ! cli ) {
return NT_STATUS_INVALID_PARAMETER ;
}
if ( ! cli - > auth ) {
return NT_STATUS_INVALID_PARAMETER ;
}
switch ( cli - > auth - > auth_type ) {
case PIPE_AUTH_TYPE_SCHANNEL :
2008-06-25 02:21:37 +04:00
* session_key = data_blob_talloc ( mem_ctx ,
cli - > auth - > a_u . schannel_auth - > sess_key , 16 ) ;
2008-06-10 23:35:34 +04:00
break ;
case PIPE_AUTH_TYPE_NTLMSSP :
case PIPE_AUTH_TYPE_SPNEGO_NTLMSSP :
2008-06-25 02:21:37 +04:00
* session_key = data_blob_talloc ( mem_ctx ,
cli - > auth - > a_u . ntlmssp_state - > session_key . data ,
cli - > auth - > a_u . ntlmssp_state - > session_key . length ) ;
2008-06-10 23:35:34 +04:00
break ;
case PIPE_AUTH_TYPE_KRB5 :
case PIPE_AUTH_TYPE_SPNEGO_KRB5 :
2008-06-25 02:21:37 +04:00
* session_key = data_blob_talloc ( mem_ctx ,
cli - > auth - > a_u . kerberos_auth - > session_key . data ,
cli - > auth - > a_u . kerberos_auth - > session_key . length ) ;
2008-06-10 23:35:34 +04:00
break ;
case PIPE_AUTH_TYPE_NONE :
2008-12-04 20:23:45 +03:00
* session_key = data_blob_talloc ( mem_ctx ,
cli - > auth - > user_session_key . data ,
cli - > auth - > user_session_key . length ) ;
break ;
2008-06-10 23:35:34 +04:00
default :
return NT_STATUS_NO_USER_SESSION_KEY ;
}
return NT_STATUS_OK ;
}
2009-01-21 16:05:51 +03:00
/**
* Create a new RPC client context which uses a local dispatch function .
*/
NTSTATUS rpc_pipe_open_internal ( TALLOC_CTX * mem_ctx , const struct ndr_syntax_id * abstract_syntax ,
NTSTATUS ( * dispatch ) ( struct rpc_pipe_client * cli , TALLOC_CTX * mem_ctx , const struct ndr_interface_table * table , uint32_t opnum , void * r ) ,
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 - > transport_type = NCACN_INTERNAL ;
result - > abstract_syntax = * abstract_syntax ;
result - > transfer_syntax = ndr_transfer_syntax ;
result - > dispatch = dispatch ;
result - > pipes_struct = TALLOC_ZERO_P ( mem_ctx , pipes_struct ) ;
if ( result - > pipes_struct = = NULL ) {
TALLOC_FREE ( result ) ;
return NT_STATUS_NO_MEMORY ;
}
result - > pipes_struct - > mem_ctx = mem_ctx ;
result - > pipes_struct - > server_info = serversupplied_info ;
result - > pipes_struct - > pipe_bound = true ;
result - > max_xmit_frag = - 1 ;
result - > max_recv_frag = - 1 ;
* presult = result ;
return NT_STATUS_OK ;
}