1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-21 18:04:06 +03:00
samba-mirror/source3/rpc_server/rpc_service_setup.c
Volker Lendecke d4e46cae4c rpc_server: Consolidate transport-specific socket creation
We had the transport switch in two places, put them together into
dcesrv_create_binding_sockets(). This makes the transport-specific
socket creation functions static to rpc_sock_helper.c.

Signed-off-by: Volker Lendecke <vl@samba.org>
Reviewed-by: Jeremy Allison <jra@samba.org>
2021-01-26 00:10:31 +00:00

943 lines
24 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_endpoint *e,
TALLOC_CTX *mem_ctx,
size_t *pnum_fds,
int **pfds)
{
struct dcerpc_binding *b = e->ep_description;
char *binding = NULL;
int *fds = NULL;
size_t num_fds;
NTSTATUS status;
binding = dcerpc_binding_string(mem_ctx, b);
if (binding == NULL) {
return NT_STATUS_NO_MEMORY;
}
DBG_DEBUG("Creating endpoint '%s'\n", binding);
TALLOC_FREE(binding);
status = dcesrv_create_binding_sockets(b, mem_ctx, &num_fds, &fds);
/* Build binding string again as the endpoint may have changed by
* dcesrv_create_<transport>_socket functions */
binding = dcerpc_binding_string(mem_ctx, b);
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);
*pnum_fds = num_fds;
*pfds = fds;
return status;
}
NTSTATUS dcesrv_create_endpoint_list_pf_listen_fds(
struct tevent_context *ev_ctx,
struct messaging_context *msg_ctx,
struct dcesrv_context *dce_ctx,
struct dcesrv_endpoint *e,
TALLOC_CTX *mem_ctx,
size_t *pnum_fds,
struct pf_listen_fd **pfds)
{
struct pf_listen_fd *fds = NULL;
size_t num_fds = 0;
NTSTATUS status;
for (; e != NULL; e = e->next) {
int *ep_fds = NULL;
struct pf_listen_fd *tmp = NULL;
size_t i, num_ep_fds;
status = dcesrv_create_endpoint_sockets(
ev_ctx,
msg_ctx,
e,
mem_ctx,
&num_ep_fds,
&ep_fds);
if (!NT_STATUS_IS_OK(status)) {
char *ep_string = dcerpc_binding_string(
dce_ctx, e->ep_description);
DBG_ERR("Failed to create endpoint '%s': %s\n",
ep_string, nt_errstr(status));
TALLOC_FREE(ep_string);
goto fail;
}
if (num_fds + num_ep_fds < num_fds) {
/* overflow */
status = NT_STATUS_INTEGER_OVERFLOW;
goto fail;
}
tmp = talloc_realloc(
mem_ctx,
fds,
struct pf_listen_fd,
num_fds + num_ep_fds);
if (tmp == NULL) {
status = NT_STATUS_NO_MEMORY;
goto fail;
}
fds = tmp;
for (i=0; i<num_ep_fds; i++) {
fds[num_fds].fd = ep_fds[i];
fds[num_fds].fd_data = e;
num_fds += 1;
}
TALLOC_FREE(ep_fds);
}
*pnum_fds = num_fds;
*pfds = fds;
return NT_STATUS_OK;
fail:
{
size_t i;
for (i=0; i<num_fds; i++) {
close(fds[i].fd);
}
}
TALLOC_FREE(fds);
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)
{
TALLOC_CTX *frame = talloc_stackframe();
struct dcerpc_binding *b = e->ep_description;
char *binding = NULL;
NTSTATUS status = NT_STATUS_NO_MEMORY;
struct dcesrv_if_list *iface = NULL;
int *fds = NULL;
size_t i, num_fds = 0;
struct dcerpc_ncacn_listen_state **listen_states = NULL;
binding = dcerpc_binding_string(frame, b);
if (binding == NULL) {
goto fail;
}
DBG_DEBUG("Setting up endpoint '%s'\n", binding);
TALLOC_FREE(binding);
status = dcesrv_create_binding_sockets(b, frame, &num_fds, &fds);
/* Build binding string again as the endpoint may have changed by
* dcesrv_create_<transport>_socket functions */
binding = dcerpc_binding_string(frame, b);
if (binding == NULL) {
status = NT_STATUS_NO_MEMORY;
goto fail;
}
if (!NT_STATUS_IS_OK(status)) {
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)));
goto fail;
}
listen_states = talloc_array(
frame, struct dcerpc_ncacn_listen_state *, num_fds);
if (listen_states == NULL) {
status = NT_STATUS_NO_MEMORY;
goto fail;
}
for (i=0; i<num_fds; i++) {
int ret = dcesrv_setup_ncacn_listener(
listen_states,
dce_ctx,
ev_ctx,
msg_ctx,
e,
&fds[i],
term_fn,
term_data,
&listen_states[i]);
if (ret != 0) {
DBG_ERR("dcesrv_setup_ncacn_listener failed for "
"socket %d: %s\n",
fds[i],
strerror(ret));
break;
}
}
if (i < num_fds) {
goto fail;
}
for (i=0; i<num_fds; i++) {
/*
* Make the listener states including the tevent_fd's
* talloc children of the endpoint. If the endpoint is
* freed (for example when forked daemons reinit) the
* dcesrv_context, the tevent_fd listener will be
* stopped and the socket closed.
*
* Do this in a loop separate from the one doing the
* dcesrv_setup_ncacn_listener() that can't fail
* anymore.
*/
talloc_move(e, &listen_states[i]);
}
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(frame);
return NT_STATUS_OK;
fail:
for (i=0; i<num_fds; i++) {
if (fds[i] != -1) {
close(fds[i]);
}
}
TALLOC_FREE(frame);
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: */