1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-13 13:18:06 +03:00
samba-mirror/source3/rpc_server/rpc_service_setup.c
Samuel Cabrero 82020a3f62 s3:rpc_server: Drop dcerpc_binding_vector usage in the server side
The endpoint mapper entry is built using the dcesrv_endpoint and the
interfaces registered into it instead of using the
dcerpc_binding_vector.

Signed-off-by: Samuel Cabrero <scabrero@samba.org>
Reviewed-by: Andrew Bartlett <abartlet@samba.org>
2020-03-20 15:36:34 +00:00

867 lines
23 KiB
C

/*
* Unix SMB/CIFS implementation.
*
* SMBD RPC service callbacks
*
* Copyright (c) 2011 Andreas Schneider <asn@samba.org>
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*/
#include "includes.h"
#include "ntdomain.h"
#include "librpc/gen_ndr/ndr_winreg_scompat.h"
#include "librpc/gen_ndr/ndr_srvsvc_scompat.h"
#include "librpc/gen_ndr/ndr_lsa_scompat.h"
#include "librpc/gen_ndr/ndr_samr_scompat.h"
#include "librpc/gen_ndr/ndr_netlogon_scompat.h"
#include "librpc/gen_ndr/ndr_dfs_scompat.h"
#include "librpc/gen_ndr/ndr_echo_scompat.h"
#include "librpc/gen_ndr/ndr_dssetup_scompat.h"
#include "librpc/gen_ndr/ndr_wkssvc_scompat.h"
#include "librpc/gen_ndr/ndr_spoolss_scompat.h"
#include "librpc/gen_ndr/ndr_svcctl_scompat.h"
#include "librpc/gen_ndr/ndr_ntsvcs_scompat.h"
#include "librpc/gen_ndr/ndr_eventlog_scompat.h"
#include "librpc/gen_ndr/ndr_initshutdown_scompat.h"
#include "printing/nt_printing_migrate_internal.h"
#include "rpc_server/eventlog/srv_eventlog_reg.h"
#include "rpc_server/svcctl/srv_svcctl_reg.h"
#include "rpc_server/spoolss/srv_spoolss_nt.h"
#include "rpc_server/svcctl/srv_svcctl_nt.h"
#include "lib/server_prefork.h"
#include "librpc/rpc/dcesrv_core.h"
#include "librpc/rpc/dcerpc_ep.h"
#include "rpc_server/rpc_sock_helper.h"
#include "rpc_server/rpc_service_setup.h"
#include "rpc_server/rpc_ep_register.h"
#include "rpc_server/rpc_server.h"
#include "rpc_server/rpc_config.h"
#include "rpc_server/rpc_modules.h"
#include "rpc_server/epmapper/srv_epmapper.h"
#undef DBGC_CLASS
#define DBGC_CLASS DBGC_RPC_SRV
static_decl_rpc;
/* Common routine for embedded RPC servers */
NTSTATUS rpc_setup_embedded(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx,
struct dcesrv_context *dce_ctx,
const struct dcesrv_interface *iface)
{
enum rpc_service_mode_e epm_mode = rpc_epmapper_mode();
NTSTATUS status;
/* Registration of ncacn_np services is problematic. The
* ev_ctx passed in here is passed down to all children of the
* smbd process, and if the end point mapper ever goes away,
* they will all attempt to re-register. But we want to test
* the code for now, so it is enabled in on environment in
* make test */
if (epm_mode != RPC_SERVICE_MODE_DISABLED &&
(lp_parm_bool(-1, "rpc_server", "register_embedded_np", false))) {
status = rpc_ep_register(ev_ctx, msg_ctx, dce_ctx, iface);
if (!NT_STATUS_IS_OK(status)) {
return status;
}
}
return NT_STATUS_OK;
}
NTSTATUS dcesrv_create_endpoint_sockets(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx,
struct dcesrv_context *dce_ctx,
struct dcesrv_endpoint *e,
struct pf_listen_fd *listen_fds,
int *listen_fds_size)
{
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport(e->ep_description);
char *binding = NULL;
NTSTATUS status;
int out_fd;
binding = dcerpc_binding_string(dce_ctx, e->ep_description);
if (binding == NULL) {
return NT_STATUS_NO_MEMORY;
}
DBG_DEBUG("Creating endpoint '%s'\n", binding);
switch (transport) {
case NCALRPC:
status = dcesrv_create_ncalrpc_socket(e, &out_fd);
if (NT_STATUS_IS_OK(status)) {
listen_fds[*listen_fds_size].fd = out_fd;
listen_fds[*listen_fds_size].fd_data = e;
(*listen_fds_size)++;
}
break;
case NCACN_IP_TCP:
status = dcesrv_create_ncacn_ip_tcp_sockets(e,
listen_fds,
listen_fds_size);
break;
case NCACN_NP:
status = dcesrv_create_ncacn_np_socket(e, &out_fd);
if (NT_STATUS_IS_OK(status)) {
listen_fds[*listen_fds_size].fd = out_fd;
listen_fds[*listen_fds_size].fd_data = e;
(*listen_fds_size)++;
}
break;
default:
status = NT_STATUS_NOT_SUPPORTED;
break;
}
/* Build binding string again as the endpoint may have changed by
* dcesrv_create_<transport>_socket functions */
TALLOC_FREE(binding);
binding = dcerpc_binding_string(dce_ctx, e->ep_description);
if (binding == NULL) {
return NT_STATUS_NO_MEMORY;
}
if (!NT_STATUS_IS_OK(status)) {
struct dcesrv_if_list *iface = NULL;
DBG_ERR("Failed to create '%s' sockets for ", binding);
for (iface = e->interface_list; iface; iface = iface->next) {
DEBUGADD(DBGLVL_ERR, ("'%s' ", iface->iface->name));
}
DEBUGADD(DBGLVL_ERR, (": %s\n", nt_errstr(status)));
return status;
} else {
struct dcesrv_if_list *iface = NULL;
DBG_INFO("Successfully listening on '%s' for ", binding);
for (iface = e->interface_list; iface; iface = iface->next) {
DEBUGADD(DBGLVL_INFO, ("'%s' ", iface->iface->name));
}
DEBUGADD(DBGLVL_INFO, ("\n"));
}
TALLOC_FREE(binding);
return status;
}
NTSTATUS dcesrv_setup_endpoint_sockets(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx,
struct dcesrv_context *dce_ctx,
struct dcesrv_endpoint *e,
dcerpc_ncacn_termination_fn term_fn,
void *term_data)
{
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport(e->ep_description);
char *binding = NULL;
NTSTATUS status;
binding = dcerpc_binding_string(dce_ctx, e->ep_description);
if (binding == NULL) {
return NT_STATUS_NO_MEMORY;
}
DBG_DEBUG("Setting up endpoint '%s'\n", binding);
switch (transport) {
case NCALRPC:
status = dcesrv_setup_ncalrpc_socket(ev_ctx,
msg_ctx,
dce_ctx,
e,
term_fn,
term_data);
break;
case NCACN_IP_TCP:
status = dcesrv_setup_ncacn_ip_tcp_sockets(ev_ctx,
msg_ctx,
dce_ctx,
e,
term_fn,
term_data);
break;
case NCACN_NP:
status = dcesrv_setup_ncacn_np_socket(ev_ctx,
msg_ctx,
dce_ctx,
e,
term_fn,
term_data);
break;
default:
status = NT_STATUS_NOT_SUPPORTED;
break;
}
/* Build binding string again as the endpoint may have changed by
* dcesrv_create_<transport>_socket functions */
TALLOC_FREE(binding);
binding = dcerpc_binding_string(dce_ctx, e->ep_description);
if (binding == NULL) {
return NT_STATUS_NO_MEMORY;
}
if (!NT_STATUS_IS_OK(status)) {
struct dcesrv_if_list *iface = NULL;
DBG_ERR("Failed to setup '%s' sockets for ", binding);
for (iface = e->interface_list; iface; iface = iface->next) {
DEBUGADD(DBGLVL_ERR, ("'%s' ", iface->iface->name));
}
DEBUGADD(DBGLVL_ERR, (": %s\n", nt_errstr(status)));
return status;
} else {
struct dcesrv_if_list *iface = NULL;
DBG_INFO("Successfully listening on '%s' for ", binding);
for (iface = e->interface_list; iface; iface = iface->next) {
DEBUGADD(DBGLVL_INFO, ("'%s' ", iface->iface->name));
}
DEBUGADD(DBGLVL_INFO, ("\n"));
}
TALLOC_FREE(binding);
return status;
}
static NTSTATUS dcesrv_init_endpoints(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx,
struct dcesrv_context *dce_ctx)
{
struct dcesrv_endpoint *e = NULL;
NTSTATUS status;
for (e = dce_ctx->endpoint_list; e; e = e->next) {
enum dcerpc_transport_t transport =
dcerpc_binding_get_transport(e->ep_description);
if (transport == NCACN_HTTP) {
/*
* We don't support ncacn_http yet
*/
continue;
}
status = dcesrv_setup_endpoint_sockets(ev_ctx,
msg_ctx,
dce_ctx,
e,
NULL,
NULL);
if (!NT_STATUS_IS_OK(status)) {
return status;
}
/* Register only NCACN_NP for embedded services */
if (transport == NCACN_NP) {
struct dcesrv_if_list *ifl = NULL;
for (ifl = e->interface_list; ifl; ifl = ifl->next) {
status = rpc_setup_embedded(ev_ctx,
msg_ctx,
dce_ctx,
ifl->iface);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register embedded "
"interface in endpoint mapper "
": %s", nt_errstr(status));
return status;
}
}
}
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_winreg(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = winreg_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'winreg' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_srvsvc(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = srvsvc_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'srvsvc' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_lsarpc(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
enum rpc_daemon_type_e lsasd_type = rpc_lsasd_daemon();
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = lsarpc_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'lsarpc' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED ||
lsasd_type != RPC_DAEMON_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_samr(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
enum rpc_daemon_type_e lsasd_type = rpc_lsasd_daemon();
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = samr_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'samr' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED ||
lsasd_type != RPC_DAEMON_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_netlogon(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
enum rpc_daemon_type_e lsasd_type = rpc_lsasd_daemon();
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = netlogon_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'netlogon' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED ||
lsasd_type != RPC_DAEMON_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_netdfs(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = netdfs_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'netdfs' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
#ifdef DEVELOPER
static NTSTATUS rpc_setup_rpcecho(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = rpcecho_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'rpcecho' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
#endif
static NTSTATUS rpc_setup_dssetup(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = dssetup_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'dssetup' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_wkssvc(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = wkssvc_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'wkssvc' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server: "
"%s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_spoolss(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
enum rpc_daemon_type_e spoolss_type = rpc_spoolss_daemon();
NTSTATUS status = NT_STATUS_UNSUCCESSFUL;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
if (lp__disable_spoolss()) {
return NT_STATUS_OK;
}
/* Register the endpoint server in DCERPC core */
ep_server = spoolss_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'spoolss' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED ||
spoolss_type != RPC_DAEMON_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server"
": %s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_svcctl(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = svcctl_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'svcctl' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server"
": %s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_ntsvcs(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = ntsvcs_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'ntsvcs' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server"
": %s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_eventlog(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = eventlog_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'eventlog' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint server"
": %s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
static NTSTATUS rpc_setup_initshutdown(struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx)
{
NTSTATUS status;
enum rpc_service_mode_e service_mode;
const struct dcesrv_endpoint_server *ep_server = NULL;
/* Register the endpoint server in DCERPC core */
ep_server = initshutdown_get_ep_server();
if (ep_server == NULL) {
DBG_ERR("Failed to get 'initshutdown' endpoint server\n");
return NT_STATUS_UNSUCCESSFUL;
}
service_mode = rpc_service_mode(ep_server->name);
if (service_mode != RPC_SERVICE_MODE_EMBEDDED) {
return NT_STATUS_OK;
}
status = dcerpc_register_ep_server(ep_server);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to register '%s' endpoint "
"server: %s\n", ep_server->name, nt_errstr(status));
return status;
}
return NT_STATUS_OK;
}
NTSTATUS dcesrv_init(TALLOC_CTX *mem_ctx,
struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx,
struct dcesrv_context *dce_ctx)
{
TALLOC_CTX *tmp_ctx;
bool ok;
init_module_fn *mod_init_fns = NULL;
NTSTATUS status;
tmp_ctx = talloc_stackframe();
if (tmp_ctx == NULL) {
return NT_STATUS_NO_MEMORY;
}
DBG_INFO("Registering DCE/RPC endpoint servers\n");
status = rpc_setup_winreg(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_srvsvc(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_lsarpc(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_samr(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_netlogon(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_netdfs(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
#ifdef DEVELOPER
status = rpc_setup_rpcecho(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
#endif
status = rpc_setup_dssetup(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_wkssvc(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_spoolss(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_svcctl(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_ntsvcs(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_eventlog(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
status = rpc_setup_initshutdown(ev_ctx, msg_ctx);
if (!NT_STATUS_IS_OK(status)) {
goto done;
}
DBG_INFO("Initializing DCE/RPC modules\n");
/* Initialize static subsystems */
static_init_rpc(NULL);
/* Initialize shared modules */
mod_init_fns = load_samba_modules(tmp_ctx, "rpc");
if ((mod_init_fns == NULL) && (errno != ENOENT)) {
/*
* ENOENT means the directory doesn't exist which can happen if
* all modules are static. So ENOENT is ok, everything else is
* not ok.
*/
DBG_ERR("Loading shared DCE/RPC modules failed [%s]\n",
strerror(errno));
status = NT_STATUS_UNSUCCESSFUL;
goto done;
}
ok = run_init_functions(NULL, mod_init_fns);
if (!ok) {
DBG_ERR("Initializing shared DCE/RPC modules failed\n");
status = NT_STATUS_UNSUCCESSFUL;
goto done;
}
/* The RPC module setup function has to register the endpoint server */
ok = setup_rpc_modules(ev_ctx, msg_ctx);
if (!ok) {
DBG_ERR("Shared DCE/RPC modules setup failed\n");
status = NT_STATUS_UNSUCCESSFUL;
goto done;
}
DBG_INFO("Initializing DCE/RPC registered endpoint servers\n");
status = dcesrv_init_registered_ep_servers(dce_ctx);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to init DCE/RPC endpoint servers: %s\n",
nt_errstr(status));
goto done;
}
DBG_INFO("Initializing DCE/RPC connection endpoints\n");
status = dcesrv_init_endpoints(ev_ctx, msg_ctx, dce_ctx);
if (!NT_STATUS_IS_OK(status)) {
DBG_ERR("Failed to init DCE/RPC endpoints: %s\n",
nt_errstr(status));
goto done;
}
status = NT_STATUS_OK;
done:
talloc_free(tmp_ctx);
return status;
}
/* vim: set ts=8 sw=8 noet cindent ft=c.doxygen: */