2 Unix SMB/CIFS implementation.
4 process incoming connections and fork a samba3 in inetd mode
6 Copyright (C) Stefan Metzmacher 2008
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>.
23 #include "smbd/service_task.h"
24 #include "smbd/service_stream.h"
25 #include "smbd/service.h"
26 #include "lib/messaging/irpc.h"
27 #include "lib/stream/packet.h"
28 #include "lib/socket/socket.h"
29 #include "libcli/smb2/smb2.h"
30 #include "smb_server/smb2/smb2_server.h"
31 #include "system/network.h"
32 #include "lib/socket/netif.h"
33 #include "param/share.h"
34 #include "param/param.h"
35 #include "dynconfig/dynconfig.h"
36 #include "smbd/process_model.h"
39 initialise a server_context from a open socket and register a event handler
40 for reading from that socket
42 static void samba3_smb_accept(struct stream_connection
*conn
)
45 int fd
= socket_get_fd(conn
->socket
);
49 extern char **environ
;
60 prog
= lp_parm_string(conn
->lp_ctx
, NULL
, "samba3", "smbd");
63 argv
[0] = talloc_asprintf(conn
, "%s/%s", dyn_BINDIR
, "smbd3");
66 argv
[0] = talloc_strdup(conn
, prog
);
69 if (argv
[0] == NULL
) {
70 stream_terminate_connection(conn
, "out of memory");
75 execve(argv
[0], argv
, environ
);
78 * Should never get here
80 reason
= talloc_asprintf(conn
, "Could not execute %s", argv
[0]);
82 stream_terminate_connection(conn
, "out of memory");
85 stream_terminate_connection(conn
, reason
);
89 static const struct stream_server_ops samba3_smb_stream_ops
= {
91 .accept_connection
= samba3_smb_accept
,
95 setup a listening socket on all the SMB ports for a particular address
97 static NTSTATUS
samba3_add_socket(struct tevent_context
*event_context
,
98 struct loadparm_context
*lp_ctx
,
99 const struct model_ops
*model_ops
,
102 const char **ports
= lp_smb_ports(lp_ctx
);
106 for (i
=0;ports
[i
];i
++) {
107 uint16_t port
= atoi(ports
[i
]);
108 if (port
== 0) continue;
109 status
= stream_setup_socket(event_context
, lp_ctx
,
110 model_ops
, &samba3_smb_stream_ops
,
111 "ip", address
, &port
,
112 lp_socket_options(lp_ctx
),
114 NT_STATUS_NOT_OK_RETURN(status
);
122 open the smb server sockets
124 static void samba3_smb_task_init(struct task_server
*task
)
127 const struct model_ops
*model_ops
;
129 model_ops
= process_model_startup(task
->event_ctx
, "standard");
131 if (model_ops
== NULL
) {
135 task_server_set_title(task
, "task[samba3_smb]");
137 if (lp_interfaces(task
->lp_ctx
)
138 && lp_bind_interfaces_only(task
->lp_ctx
)) {
141 struct interface
*ifaces
;
143 load_interfaces(task
, lp_interfaces(task
->lp_ctx
), &ifaces
);
145 num_interfaces
= iface_count(ifaces
);
147 /* We have been given an interfaces line, and been
148 told to only bind to those interfaces. Create a
149 socket per interface and bind to only these.
151 for(i
= 0; i
< num_interfaces
; i
++) {
152 const char *address
= iface_n_ip(ifaces
, i
);
153 status
= samba3_add_socket(task
->event_ctx
,
156 if (!NT_STATUS_IS_OK(status
)) goto failed
;
159 /* Just bind to lp_socket_address() (usually 0.0.0.0) */
160 status
= samba3_add_socket(task
->event_ctx
, task
->lp_ctx
,
162 lp_socket_address(task
->lp_ctx
));
163 if (!NT_STATUS_IS_OK(status
)) goto failed
;
168 task_server_terminate(task
, "Failed to startup samba3 smb task");
171 /* called at smbd startup - register ourselves as a server service */
172 NTSTATUS
server_service_samba3_smb_init(void)
174 return register_server_service("samba3_smb", samba3_smb_task_init
);