2005-01-16 01:28:11 +00:00
/*
Unix SMB / CIFS implementation .
Copyright ( C ) Andrew Tridgell 2005
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
a composite API for making a full SMB connection
*/
# include "includes.h"
# include "libcli/raw/libcliraw.h"
# include "libcli/composite/composite.h"
/* the stages of this call */
enum connect_stage { CONNECT_SOCKET ,
CONNECT_SESSION_REQUEST ,
CONNECT_NEGPROT ,
CONNECT_SESSION_SETUP ,
CONNECT_TCON } ;
struct connect_state {
struct smbcli_socket * sock ;
struct smbcli_transport * transport ;
struct smbcli_session * session ;
2005-01-16 21:58:28 +00:00
struct smb_composite_connect * io ;
union smb_tcon * io_tcon ;
struct smb_composite_sesssetup * io_setup ;
2005-01-16 01:28:11 +00:00
} ;
static void request_handler ( struct smbcli_request * ) ;
static void composite_handler ( struct smbcli_composite * ) ;
/*
setup a negprot send
*/
static NTSTATUS connect_send_negprot ( struct smbcli_composite * c ,
struct smb_composite_connect * io )
{
struct connect_state * state = c - > private ;
struct smbcli_request * req ;
req = smb_raw_negotiate_send ( state - > transport , lp_maxprotocol ( ) ) ;
NT_STATUS_HAVE_NO_MEMORY ( req ) ;
req - > async . fn = request_handler ;
req - > async . private = c ;
c - > stage = CONNECT_NEGPROT ;
c - > req = req ;
return NT_STATUS_OK ;
}
/*
a tree connect request has competed
*/
static NTSTATUS connect_tcon ( struct smbcli_composite * c ,
struct smb_composite_connect * io )
{
2005-01-16 21:58:28 +00:00
struct connect_state * state = c - > private ;
2005-01-16 01:28:11 +00:00
struct smbcli_request * req = c - > req ;
NTSTATUS status ;
2005-01-16 21:58:28 +00:00
status = smb_tree_connect_recv ( req , c , state - > io_tcon ) ;
2005-01-16 01:28:11 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
2005-01-16 21:58:28 +00:00
io - > out . tree - > tid = state - > io_tcon - > tconx . out . tid ;
if ( state - > io_tcon - > tconx . out . dev_type ) {
2005-01-16 01:28:11 +00:00
io - > out . tree - > device = talloc_strdup ( io - > out . tree ,
2005-01-16 21:58:28 +00:00
state - > io_tcon - > tconx . out . dev_type ) ;
2005-01-16 01:28:11 +00:00
}
2005-01-16 21:58:28 +00:00
if ( state - > io_tcon - > tconx . out . fs_type ) {
2005-01-16 01:28:11 +00:00
io - > out . tree - > fs_type = talloc_strdup ( io - > out . tree ,
2005-01-16 21:58:28 +00:00
state - > io_tcon - > tconx . out . fs_type ) ;
2005-01-16 01:28:11 +00:00
}
/* all done! */
c - > state = SMBCLI_REQUEST_DONE ;
if ( c - > async . fn ) {
c - > async . fn ( c ) ;
}
return NT_STATUS_OK ;
}
/*
a session setup request has competed
*/
static NTSTATUS connect_session_setup ( struct smbcli_composite * c ,
struct smb_composite_connect * io )
{
struct connect_state * state = c - > private ;
2005-01-16 11:15:08 +00:00
struct smbcli_composite * req = c - > req ;
struct smbcli_request * req2 ;
2005-01-16 01:28:11 +00:00
NTSTATUS status ;
2005-01-16 11:15:08 +00:00
status = smb_composite_sesssetup_recv ( req ) ;
2005-01-16 01:28:11 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
2005-01-16 21:58:28 +00:00
state - > session - > vuid = state - > io_setup - > out . vuid ;
2005-01-16 01:28:11 +00:00
/* setup for a tconx */
io - > out . tree = smbcli_tree_init ( state - > session ) ;
NT_STATUS_HAVE_NO_MEMORY ( io - > out . tree ) ;
2005-01-16 21:58:28 +00:00
state - > io_tcon = talloc ( c , union smb_tcon ) ;
NT_STATUS_HAVE_NO_MEMORY ( state - > io_tcon ) ;
2005-01-16 01:28:11 +00:00
/* connect to a share using a tree connect */
2005-01-16 21:58:28 +00:00
state - > io_tcon - > generic . level = RAW_TCON_TCONX ;
state - > io_tcon - > tconx . in . flags = 0 ;
state - > io_tcon - > tconx . in . password = data_blob ( NULL , 0 ) ;
2005-01-16 01:28:11 +00:00
2005-01-16 21:58:28 +00:00
state - > io_tcon - > tconx . in . path = talloc_asprintf ( state - > io_tcon ,
2005-01-16 01:28:11 +00:00
" \\ \\ %s \\ %s " ,
io - > in . called_name ,
io - > in . service ) ;
2005-01-16 21:58:28 +00:00
NT_STATUS_HAVE_NO_MEMORY ( state - > io_tcon - > tconx . in . path ) ;
2005-01-16 01:28:11 +00:00
if ( ! io - > in . service_type ) {
2005-01-16 21:58:28 +00:00
state - > io_tcon - > tconx . in . device = " ????? " ;
2005-01-16 01:28:11 +00:00
} else {
2005-01-16 21:58:28 +00:00
state - > io_tcon - > tconx . in . device = io - > in . service_type ;
2005-01-16 01:28:11 +00:00
}
2005-01-16 21:58:28 +00:00
req2 = smb_tree_connect_send ( io - > out . tree , state - > io_tcon ) ;
2005-01-16 11:15:08 +00:00
NT_STATUS_HAVE_NO_MEMORY ( req2 ) ;
2005-01-16 01:28:11 +00:00
2005-01-16 11:15:08 +00:00
req2 - > async . fn = request_handler ;
req2 - > async . private = c ;
c - > req = req2 ;
2005-01-16 01:28:11 +00:00
c - > stage = CONNECT_TCON ;
return NT_STATUS_OK ;
}
/*
a negprot request has competed
*/
static NTSTATUS connect_negprot ( struct smbcli_composite * c ,
struct smb_composite_connect * io )
{
struct connect_state * state = c - > private ;
struct smbcli_request * req = c - > req ;
2005-01-16 11:15:08 +00:00
struct smbcli_composite * req2 ;
2005-01-16 01:28:11 +00:00
NTSTATUS status ;
status = smb_raw_negotiate_recv ( req ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
/* next step is a session setup */
state - > session = smbcli_session_init ( state - > transport ) ;
NT_STATUS_HAVE_NO_MEMORY ( state - > session ) ;
/* get rid of the extra reference to the transport */
talloc_free ( state - > transport ) ;
2005-01-16 21:58:28 +00:00
state - > io_setup = talloc ( c , struct smb_composite_sesssetup ) ;
NT_STATUS_HAVE_NO_MEMORY ( state - > io_setup ) ;
2005-01-16 01:28:11 +00:00
/* prepare a session setup to establish a security context */
2005-01-16 21:58:28 +00:00
state - > io_setup - > in . sesskey = state - > transport - > negotiate . sesskey ;
state - > io_setup - > in . capabilities = state - > transport - > negotiate . capabilities ;
state - > io_setup - > in . domain = io - > in . domain ;
state - > io_setup - > in . user = io - > in . user ;
state - > io_setup - > in . password = io - > in . password ;
2005-01-16 01:28:11 +00:00
2005-01-16 21:58:28 +00:00
req2 = smb_composite_sesssetup_send ( state - > session , state - > io_setup ) ;
2005-01-16 11:15:08 +00:00
NT_STATUS_HAVE_NO_MEMORY ( req2 ) ;
2005-01-16 01:28:11 +00:00
2005-01-16 11:15:08 +00:00
req2 - > async . fn = composite_handler ;
req2 - > async . private = c ;
c - > req = req2 ;
2005-01-16 01:28:11 +00:00
c - > stage = CONNECT_SESSION_SETUP ;
return NT_STATUS_OK ;
}
/*
a session request operation has competed
*/
static NTSTATUS connect_session_request ( struct smbcli_composite * c ,
struct smb_composite_connect * io )
{
struct smbcli_request * req = c - > req ;
NTSTATUS status ;
status = smbcli_transport_connect_recv ( req ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
/* next step is a negprot */
return connect_send_negprot ( c , io ) ;
}
/*
a socket connection operation has competed
*/
static NTSTATUS connect_socket ( struct smbcli_composite * c ,
struct smb_composite_connect * io )
{
struct connect_state * state = c - > private ;
NTSTATUS status ;
struct smbcli_request * req ;
struct nmb_name calling , called ;
status = smbcli_sock_connect_recv ( c - > req ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
/* the socket is up - we can initialise the smbcli transport layer */
state - > transport = smbcli_transport_init ( state - > sock ) ;
NT_STATUS_HAVE_NO_MEMORY ( state - > transport ) ;
/* we have a connected socket - next step is a session
request , if needed . Port 445 doesn ' t need it , so it goes
straight to the negprot */
if ( state - > sock - > port = = 445 ) {
return connect_send_negprot ( c , io ) ;
}
make_nmb_name ( & calling , io - > in . calling_name , 0x0 ) ;
choose_called_name ( & called , io - > in . called_name , 0x20 ) ;
req = smbcli_transport_connect_send ( state - > transport , & calling , & called ) ;
NT_STATUS_HAVE_NO_MEMORY ( req ) ;
req - > async . fn = request_handler ;
req - > async . private = c ;
c - > stage = CONNECT_SESSION_REQUEST ;
c - > req = req ;
return NT_STATUS_OK ;
}
/*
handle and dispatch state transitions
*/
static void state_handler ( struct smbcli_composite * c )
{
2005-01-16 21:58:28 +00:00
struct connect_state * state = c - > private ;
2005-01-16 01:28:11 +00:00
switch ( c - > stage ) {
case CONNECT_SOCKET :
2005-01-16 21:58:28 +00:00
c - > status = connect_socket ( c , state - > io ) ;
2005-01-16 01:28:11 +00:00
break ;
case CONNECT_SESSION_REQUEST :
2005-01-16 21:58:28 +00:00
c - > status = connect_session_request ( c , state - > io ) ;
2005-01-16 01:28:11 +00:00
break ;
case CONNECT_NEGPROT :
2005-01-16 21:58:28 +00:00
c - > status = connect_negprot ( c , state - > io ) ;
2005-01-16 01:28:11 +00:00
break ;
case CONNECT_SESSION_SETUP :
2005-01-16 21:58:28 +00:00
c - > status = connect_session_setup ( c , state - > io ) ;
2005-01-16 01:28:11 +00:00
break ;
case CONNECT_TCON :
2005-01-16 21:58:28 +00:00
c - > status = connect_tcon ( c , state - > io ) ;
2005-01-16 01:28:11 +00:00
break ;
}
if ( ! NT_STATUS_IS_OK ( c - > status ) ) {
c - > state = SMBCLI_REQUEST_ERROR ;
if ( c - > async . fn ) {
c - > async . fn ( c ) ;
}
}
}
/*
handler for completion of a smbcli_request sub - request
*/
static void request_handler ( struct smbcli_request * req )
{
struct smbcli_composite * c = req - > async . private ;
return state_handler ( c ) ;
}
/*
handler for completion of a smbcli_composite sub - request
*/
static void composite_handler ( struct smbcli_composite * req )
{
struct smbcli_composite * c = req - > async . private ;
return state_handler ( c ) ;
}
/*
a function to establish a smbcli_tree from scratch
*/
struct smbcli_composite * smb_composite_connect_send ( struct smb_composite_connect * io )
{
struct smbcli_composite * c , * req ;
struct connect_state * state ;
c = talloc_zero ( NULL , struct smbcli_composite ) ;
if ( c = = NULL ) goto failed ;
state = talloc ( c , struct connect_state ) ;
if ( state = = NULL ) goto failed ;
state - > sock = smbcli_sock_init ( state ) ;
if ( state - > sock = = NULL ) goto failed ;
2005-01-16 21:58:28 +00:00
state - > io = io ;
2005-01-16 01:28:11 +00:00
c - > state = SMBCLI_REQUEST_SEND ;
c - > stage = CONNECT_SOCKET ;
c - > event_ctx = state - > sock - > event . ctx ;
c - > private = state ;
req = smbcli_sock_connect_send ( state - > sock , io - > in . dest_host , io - > in . port ) ;
if ( req = = NULL ) goto failed ;
req - > async . private = c ;
req - > async . fn = composite_handler ;
c - > req = req ;
return c ;
failed :
talloc_free ( c ) ;
return NULL ;
}
/*
recv half of async composite connect code
*/
NTSTATUS smb_composite_connect_recv ( struct smbcli_composite * c , TALLOC_CTX * mem_ctx )
{
NTSTATUS status ;
status = smb_composite_wait ( c ) ;
if ( NT_STATUS_IS_OK ( status ) ) {
2005-01-16 21:58:28 +00:00
struct connect_state * state = c - > private ;
talloc_steal ( mem_ctx , state - > io - > out . tree ) ;
2005-01-16 01:28:11 +00:00
}
talloc_free ( c ) ;
return status ;
}
/*
sync version of smb_composite_connect
*/
NTSTATUS smb_composite_connect ( struct smb_composite_connect * io , TALLOC_CTX * mem_ctx )
{
struct smbcli_composite * c = smb_composite_connect_send ( io ) ;
return smb_composite_connect_recv ( c , mem_ctx ) ;
}