2004-02-03 05:47:36 +00:00
/*
Unix SMB / CIFS implementation .
process incoming packets - main loop
2005-11-17 08:00:48 +00:00
Copyright ( C ) Andrew Tridgell 2004 - 2005
Copyright ( C ) Stefan Metzmacher 2004 - 2005
2004-02-03 05:47:36 +00:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2004-02-03 05:47:36 +00:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2004-02-03 05:47:36 +00:00
*/
# include "includes.h"
2020-11-20 15:27:17 +01:00
# include "samba/service_task.h"
# include "samba/service_stream.h"
# include "samba/service.h"
2004-11-02 07:18:24 +00:00
# include "smb_server/smb_server.h"
2006-04-28 01:55:17 +00:00
# include "smb_server/service_smb_proto.h"
2005-07-10 01:08:10 +00:00
# include "lib/messaging/irpc.h"
2005-11-09 10:51:26 +00:00
# include "lib/stream/packet.h"
2005-11-18 14:13:49 +00:00
# include "libcli/smb2/smb2.h"
2005-12-28 15:38:36 +00:00
# include "smb_server/smb2/smb2_server.h"
2006-03-07 11:07:23 +00:00
# include "system/network.h"
2006-08-17 13:37:04 +00:00
# include "lib/socket/netif.h"
2006-07-23 18:43:07 +00:00
# include "param/share.h"
2007-07-04 05:16:19 +00:00
# include "dsdb/samdb/samdb.h"
2007-09-08 12:42:09 +00:00
# include "param/param.h"
2004-02-03 05:47:36 +00:00
2009-02-02 08:39:45 +01:00
static NTSTATUS smbsrv_recv_generic_request ( void * private_data , DATA_BLOB blob )
2005-11-18 08:44:36 +00:00
{
NTSTATUS status ;
2009-02-02 08:39:45 +01:00
struct smbsrv_connection * smb_conn = talloc_get_type ( private_data , struct smbsrv_connection ) ;
2005-11-18 08:44:36 +00:00
uint32_t protocol_version ;
/* see if its a special NBT packet */
if ( CVAL ( blob . data , 0 ) ! = 0 ) {
2007-12-03 21:25:17 +01:00
status = smbsrv_init_smb_connection ( smb_conn , smb_conn - > lp_ctx ) ;
2005-11-18 08:44:36 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
return smbsrv_recv_smb_request ( smb_conn , blob ) ;
}
if ( blob . length < ( NBT_HDR_SIZE + MIN_SMB_SIZE ) ) {
2005-11-30 02:08:15 +00:00
DEBUG ( 2 , ( " Invalid SMB packet length count %ld \n " , ( long ) blob . length ) ) ;
2005-11-18 08:44:36 +00: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 :
2007-12-03 21:25:17 +01:00
status = smbsrv_init_smb_connection ( smb_conn , smb_conn - > lp_ctx ) ;
2005-11-18 08:44:36 +00:00
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 14:13:49 +00:00
case SMB2_MAGIC :
2014-02-04 15:09:09 +13:00
if ( lpcfg_server_max_protocol ( smb_conn - > lp_ctx ) < PROTOCOL_SMB2_02 ) break ;
2005-11-18 14:13:49 +00:00
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 08:44:36 +00:00
}
2006-03-06 14:19:11 +00:00
DEBUG ( 2 , ( " Invalid SMB packet: protocol prefix: 0x%08X \n " , protocol_version ) ) ;
2005-11-18 08:44:36 +00:00
smbsrv_terminate_connection ( smb_conn , " NON-SMB packet " ) ;
return NT_STATUS_OK ;
}
2004-02-03 05:47:36 +00:00
/*
close the socket and shutdown a server_context
*/
2004-07-13 21:04:56 +00:00
void smbsrv_terminate_connection ( struct smbsrv_connection * smb_conn , const char * reason )
2004-02-03 05:47:36 +00:00
{
2005-12-08 10:23:56 +00:00
stream_terminate_connection ( smb_conn - > connection , reason ) ;
2004-06-23 23:44:50 +00:00
}
2004-02-03 05:47:36 +00:00
/*
called when a SMB socket becomes readable
*/
2005-02-03 11:25:52 +00:00
static void smbsrv_recv ( struct stream_connection * conn , uint16_t flags )
2004-02-03 05:47:36 +00:00
{
2009-02-02 10:30:03 +01:00
struct smbsrv_connection * smb_conn = talloc_get_type ( conn - > private_data ,
2005-12-08 10:23:56 +00:00
struct smbsrv_connection ) ;
2004-07-13 21:04:56 +00:00
DEBUG ( 10 , ( " smbsrv_recv \n " ) ) ;
2005-11-09 10:51:26 +00:00
packet_recv ( smb_conn - > packet ) ;
2004-07-13 21:04:56 +00:00
}
/*
called when a SMB socket becomes writable
*/
2005-02-03 11:25:52 +00:00
static void smbsrv_send ( struct stream_connection * conn , uint16_t flags )
2004-07-13 21:04:56 +00:00
{
2009-02-02 10:30:03 +01:00
struct smbsrv_connection * smb_conn = talloc_get_type ( conn - > private_data ,
2005-11-09 13:42:56 +00:00
struct smbsrv_connection ) ;
packet_queue_run ( smb_conn - > packet ) ;
2004-07-13 21:04:56 +00:00
}
2005-11-09 10:51:26 +00:00
/*
handle socket recv errors
*/
2009-02-02 08:39:45 +01:00
static void smbsrv_recv_error ( void * private_data , NTSTATUS status )
2005-11-09 10:51:26 +00:00
{
2009-02-02 08:39:45 +01:00
struct smbsrv_connection * smb_conn = talloc_get_type ( private_data , struct smbsrv_connection ) ;
2005-11-09 10:51:26 +00:00
smbsrv_terminate_connection ( smb_conn , nt_errstr ( status ) ) ;
}
2004-02-03 05:47:36 +00:00
/*
initialise a server_context from a open socket and register a event handler
for reading from that socket
*/
2005-01-30 00:54:57 +00:00
static void smbsrv_accept ( struct stream_connection * conn )
2004-02-03 05:47:36 +00:00
{
2004-06-29 07:40:14 +00:00
struct smbsrv_connection * smb_conn ;
2004-02-03 05:47:36 +00:00
2004-07-13 21:04:56 +00:00
DEBUG ( 5 , ( " smbsrv_accept \n " ) ) ;
2004-02-03 05:47:36 +00:00
2005-01-27 07:08:20 +00:00
smb_conn = talloc_zero ( conn , struct smbsrv_connection ) ;
2005-12-08 10:23:56 +00:00
if ( ! smb_conn ) {
stream_terminate_connection ( conn , " out of memory " ) ;
return ;
}
2004-02-03 05:47:36 +00:00
2005-11-09 10:51:26 +00:00
smb_conn - > packet = packet_init ( smb_conn ) ;
2005-12-08 10:23:56 +00:00
if ( ! smb_conn - > packet ) {
smbsrv_terminate_connection ( smb_conn , " out of memory " ) ;
2005-11-10 00:36:53 +00:00
return ;
}
2005-11-09 10:51:26 +00:00
packet_set_private ( smb_conn - > packet , smb_conn ) ;
packet_set_socket ( smb_conn - > packet , conn - > socket ) ;
2005-11-18 08:44:36 +00:00
packet_set_callback ( smb_conn - > packet , smbsrv_recv_generic_request ) ;
2005-11-09 10:51:26 +00: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 03:45:57 +00:00
packet_set_fde ( smb_conn - > packet , conn - > event . fde ) ;
packet_set_serialise ( smb_conn - > packet ) ;
2009-05-19 15:42:39 +01:00
packet_set_initial_read ( smb_conn - > packet , 4 ) ;
2005-11-09 10:51:26 +00:00
2008-02-21 18:09:47 +01:00
smb_conn - > lp_ctx = conn - > lp_ctx ;
2004-07-13 21:04:56 +00:00
smb_conn - > connection = conn ;
2009-02-02 10:30:03 +01:00
conn - > private_data = smb_conn ;
2005-07-10 01:08:10 +00:00
2006-03-26 11:32:27 +00:00
smb_conn - > statistics . connect_time = timeval_current ( ) ;
2005-07-19 03:58:44 +00:00
smbsrv_management_init ( smb_conn ) ;
2006-07-23 18:43:07 +00:00
2008-05-26 05:12:31 +02:00
irpc_add_name ( conn - > msg_ctx , " smb_server " ) ;
2021-03-25 11:50:14 +13:00
if ( ! NT_STATUS_IS_OK ( share_get_context ( smb_conn ,
smb_conn - > lp_ctx ,
& ( smb_conn - > share_context ) ) ) ) {
2006-07-23 18:43:07 +00:00
smbsrv_terminate_connection ( smb_conn , " share_init failed! " ) ;
return ;
}
2004-07-13 21:04:56 +00:00
}
2005-01-30 00:54:57 +00:00
static const struct stream_server_ops smb_stream_ops = {
2006-03-09 17:48:41 +00:00
. name = " smbsrv " ,
2004-07-13 21:04:56 +00:00
. accept_connection = smbsrv_accept ,
. recv_handler = smbsrv_recv ,
. send_handler = smbsrv_send ,
2005-01-14 01:32:56 +00:00
} ;
2005-01-30 00:54:57 +00:00
/*
setup a listening socket on all the SMB ports for a particular address
*/
2010-11-15 10:12:22 +11:00
_PUBLIC_ NTSTATUS smbsrv_add_socket ( TALLOC_CTX * mem_ctx ,
struct tevent_context * event_context ,
2007-12-03 00:28:22 +01:00
struct loadparm_context * lp_ctx ,
2010-11-15 10:12:22 +11:00
const struct model_ops * model_ops ,
2017-09-15 07:09:23 +12:00
const char * address ,
void * process_context )
2005-01-14 01:32:56 +00:00
{
2010-07-16 14:32:42 +10:00
const char * * ports = lpcfg_smb_ports ( lp_ctx ) ;
2005-01-30 00:54:57 +00:00
int i ;
NTSTATUS status ;
for ( i = 0 ; ports [ i ] ; i + + ) {
uint16_t port = atoi ( ports [ i ] ) ;
if ( port = = 0 ) continue ;
2010-11-15 10:12:22 +11:00
status = stream_setup_socket ( mem_ctx , event_context , lp_ctx ,
2008-01-05 19:03:43 -06:00
model_ops , & smb_stream_ops ,
2011-05-12 12:36:33 +02:00
" ip " , address , & port ,
2010-07-16 14:32:42 +10:00
lpcfg_socket_options ( lp_ctx ) ,
2017-09-15 07:09:23 +12:00
NULL , process_context ) ;
2005-01-30 00:54:57 +00:00
NT_STATUS_NOT_OK_RETURN ( status ) ;
}
return NT_STATUS_OK ;
2005-01-14 01:32:56 +00:00
}
2007-07-04 05:16:19 +00:00
2005-01-30 00:54:57 +00:00