2004-02-03 08:47:36 +03:00
/*
Unix SMB / CIFS implementation .
process incoming packets - main loop
2005-11-17 11:00:48 +03:00
Copyright ( C ) Andrew Tridgell 2004 - 2005
Copyright ( C ) Stefan Metzmacher 2004 - 2005
2004-02-03 08:47:36 +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
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 .
*/
# include "includes.h"
2005-02-03 14:56:03 +03:00
# include "lib/events/events.h"
2004-11-02 03:24:21 +03:00
# include "system/time.h"
2004-11-02 09:42:15 +03:00
# include "dlinklist.h"
2005-01-30 03:54:57 +03:00
# include "smbd/service_stream.h"
2004-11-02 10:18:24 +03:00
# include "smb_server/smb_server.h"
2005-07-10 05:08:10 +04:00
# include "lib/messaging/irpc.h"
2005-11-09 13:51:26 +03:00
# include "lib/stream/packet.h"
2005-11-18 17:13:49 +03:00
# include "libcli/smb2/smb2.h"
2004-02-03 08:47:36 +03:00
2005-11-18 11:44:36 +03:00
static NTSTATUS smbsrv_recv_generic_request ( void * private , DATA_BLOB blob )
{
NTSTATUS status ;
struct smbsrv_connection * smb_conn = talloc_get_type ( private , struct smbsrv_connection ) ;
uint32_t protocol_version ;
/* see if its a special NBT packet */
if ( CVAL ( blob . data , 0 ) ! = 0 ) {
status = smbsrv_init_smb_connection ( smb_conn ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
packet_set_callback ( smb_conn - > packet , smbsrv_recv_smb_request ) ;
return smbsrv_recv_smb_request ( smb_conn , blob ) ;
}
if ( blob . length < ( NBT_HDR_SIZE + MIN_SMB_SIZE ) ) {
2005-11-30 05:08:15 +03:00
DEBUG ( 2 , ( " Invalid SMB packet length count %ld \n " , ( long ) blob . length ) ) ;
2005-11-18 11:44:36 +03:00
smbsrv_terminate_connection ( smb_conn , " Invalid SMB packet " ) ;
return NT_STATUS_OK ;
}
protocol_version = IVAL ( blob . data , NBT_HDR_SIZE ) ;
switch ( protocol_version ) {
case SMB_MAGIC :
status = smbsrv_init_smb_connection ( smb_conn ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
packet_set_callback ( smb_conn - > packet , smbsrv_recv_smb_request ) ;
return smbsrv_recv_smb_request ( smb_conn , blob ) ;
2005-11-18 17:13:49 +03:00
case SMB2_MAGIC :
if ( ! lp_parm_bool ( - 1 , " smbsrv " , " enable smb2 " , False ) ) break ;
status = smbsrv_init_smb2_connection ( smb_conn ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
packet_set_callback ( smb_conn - > packet , smbsrv_recv_smb2_request ) ;
return smbsrv_recv_smb2_request ( smb_conn , blob ) ;
2005-11-18 11:44:36 +03:00
}
DEBUG ( 2 , ( " Invalid SMB packet: protocl prefix: 0x%08X \n " , protocol_version ) ) ;
smbsrv_terminate_connection ( smb_conn , " NON-SMB packet " ) ;
return NT_STATUS_OK ;
}
2004-02-03 08:47:36 +03:00
/*
close the socket and shutdown a server_context
*/
2004-07-14 01:04:56 +04:00
void smbsrv_terminate_connection ( struct smbsrv_connection * smb_conn , const char * reason )
2004-02-03 08:47:36 +03:00
{
2005-12-08 13:23:56 +03:00
stream_terminate_connection ( smb_conn - > connection , reason ) ;
2004-06-24 03:44:50 +04:00
}
2004-02-03 08:47:36 +03:00
/*
called when a SMB socket becomes readable
*/
2005-02-03 14:25:52 +03:00
static void smbsrv_recv ( struct stream_connection * conn , uint16_t flags )
2004-02-03 08:47:36 +03:00
{
2005-12-08 13:23:56 +03:00
struct smbsrv_connection * smb_conn = talloc_get_type ( conn - > private ,
struct smbsrv_connection ) ;
2004-07-14 01:04:56 +04:00
DEBUG ( 10 , ( " smbsrv_recv \n " ) ) ;
2005-11-09 13:51:26 +03:00
packet_recv ( smb_conn - > packet ) ;
2004-02-03 08:47:36 +03:00
/* free up temporary memory */
lp_talloc_free ( ) ;
2004-07-14 01:04:56 +04:00
}
/*
called when a SMB socket becomes writable
*/
2005-02-03 14:25:52 +03:00
static void smbsrv_send ( struct stream_connection * conn , uint16_t flags )
2004-07-14 01:04:56 +04:00
{
2005-11-09 16:42:56 +03:00
struct smbsrv_connection * smb_conn = talloc_get_type ( conn - > private ,
struct smbsrv_connection ) ;
packet_queue_run ( smb_conn - > packet ) ;
2004-07-14 01:04:56 +04:00
}
2005-11-09 13:51:26 +03:00
/*
handle socket recv errors
*/
static void smbsrv_recv_error ( void * private , NTSTATUS status )
{
struct smbsrv_connection * smb_conn = talloc_get_type ( private , struct smbsrv_connection ) ;
smbsrv_terminate_connection ( smb_conn , nt_errstr ( status ) ) ;
}
2004-02-03 08:47:36 +03:00
/*
initialise a server_context from a open socket and register a event handler
for reading from that socket
*/
2005-01-30 03:54:57 +03:00
static void smbsrv_accept ( struct stream_connection * conn )
2004-02-03 08:47:36 +03:00
{
2004-06-29 11:40:14 +04:00
struct smbsrv_connection * smb_conn ;
2004-02-03 08:47:36 +03:00
2004-07-14 01:04:56 +04:00
DEBUG ( 5 , ( " smbsrv_accept \n " ) ) ;
2004-02-03 08:47:36 +03:00
2005-01-27 10:08:20 +03:00
smb_conn = talloc_zero ( conn , struct smbsrv_connection ) ;
2005-12-08 13:23:56 +03:00
if ( ! smb_conn ) {
stream_terminate_connection ( conn , " out of memory " ) ;
return ;
}
2004-02-03 08:47:36 +03:00
2005-11-09 13:51:26 +03:00
smb_conn - > packet = packet_init ( smb_conn ) ;
2005-12-08 13:23:56 +03:00
if ( ! smb_conn - > packet ) {
smbsrv_terminate_connection ( smb_conn , " out of memory " ) ;
2005-11-10 03:36:53 +03:00
return ;
}
2005-11-09 13:51:26 +03:00
packet_set_private ( smb_conn - > packet , smb_conn ) ;
packet_set_socket ( smb_conn - > packet , conn - > socket ) ;
2005-11-18 11:44:36 +03:00
packet_set_callback ( smb_conn - > packet , smbsrv_recv_generic_request ) ;
2005-11-09 13:51:26 +03:00
packet_set_full_request ( smb_conn - > packet , packet_full_request_nbt ) ;
packet_set_error_handler ( smb_conn - > packet , smbsrv_recv_error ) ;
packet_set_event_context ( smb_conn - > packet , conn - > event . ctx ) ;
2005-11-14 06:45:57 +03:00
packet_set_fde ( smb_conn - > packet , conn - > event . fde ) ;
packet_set_serialise ( smb_conn - > packet ) ;
2005-11-09 13:51:26 +03:00
2004-07-14 01:04:56 +04:00
smb_conn - > connection = conn ;
2005-01-30 03:54:57 +03:00
conn - > private = smb_conn ;
2005-07-10 05:08:10 +04:00
irpc_add_name ( conn - > msg_ctx , " smb_server " ) ;
2005-07-19 07:58:44 +04:00
smbsrv_management_init ( smb_conn ) ;
2004-07-14 01:04:56 +04:00
}
2005-01-30 03:54:57 +03:00
static const struct stream_server_ops smb_stream_ops = {
2004-07-14 01:04:56 +04:00
. name = " smb " ,
. accept_connection = smbsrv_accept ,
. recv_handler = smbsrv_recv ,
. send_handler = smbsrv_send ,
2005-01-14 04:32:56 +03:00
} ;
2005-01-30 03:54:57 +03:00
/*
setup a listening socket on all the SMB ports for a particular address
*/
static NTSTATUS smb_add_socket ( struct event_context * event_context ,
const struct model_ops * model_ops ,
const char * address )
2005-01-14 04:32:56 +03:00
{
2005-01-30 03:54:57 +03:00
const char * * ports = lp_smb_ports ( ) ;
int i ;
NTSTATUS status ;
for ( i = 0 ; ports [ i ] ; i + + ) {
uint16_t port = atoi ( ports [ i ] ) ;
if ( port = = 0 ) continue ;
status = stream_setup_socket ( event_context , model_ops , & smb_stream_ops ,
" ipv4 " , address , & port , NULL ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
return NT_STATUS_OK ;
2005-01-14 04:32:56 +03:00
}
2005-01-30 03:54:57 +03:00
/*
called on startup of the smb server service It ' s job is to start
listening on all configured SMB server sockets
*/
static NTSTATUS smbsrv_init ( struct event_context * event_context , const struct model_ops * model_ops )
{
NTSTATUS status ;
2004-07-14 01:04:56 +04:00
2005-01-30 03:54:57 +03:00
if ( lp_interfaces ( ) & & lp_bind_interfaces_only ( ) ) {
int num_interfaces = iface_count ( ) ;
int i ;
/* We have been given an interfaces line, and been
told to only bind to those interfaces . Create a
socket per interface and bind to only these .
*/
for ( i = 0 ; i < num_interfaces ; i + + ) {
2005-02-10 06:22:47 +03:00
const char * address = iface_n_ip ( i ) ;
2005-01-30 03:54:57 +03:00
status = smb_add_socket ( event_context , model_ops , address ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
} else {
/* Just bind to lp_socket_address() (usually 0.0.0.0) */
status = smb_add_socket ( event_context , model_ops , lp_socket_address ( ) ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
return NT_STATUS_OK ;
2004-07-14 01:04:56 +04:00
}
2005-01-30 03:54:57 +03:00
/* called at smbd startup - register ourselves as a server service */
2004-07-14 01:04:56 +04:00
NTSTATUS server_service_smb_init ( void )
{
2005-01-30 03:54:57 +03:00
return register_server_service ( " smb " , smbsrv_init ) ;
2004-02-03 08:47:36 +03:00
}