2 Unix SMB/CIFS implementation.
4 Winbind daemon - miscellaneous other functions
6 Copyright (C) Tim Potter 2000
7 Copyright (C) Andrew Bartlett 2002
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3 of the License, or
12 (at your option) any later version.
14 This program is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with this program. If not, see <http://www.gnu.org/licenses/>.
27 #define DBGC_CLASS DBGC_WINBIND
30 static const struct winbindd_child_dispatch_table locator_dispatch_table[];
32 static struct winbindd_child static_locator_child;
34 void init_locator_child(void)
36 setup_child(&static_locator_child,
37 locator_dispatch_table,
38 "log.winbindd", "locator");
41 struct winbindd_child *locator_child(void)
43 return &static_locator_child;
46 void winbindd_dsgetdcname(struct winbindd_cli_state *state)
48 state->request.domain_name
49 [sizeof(state->request.domain_name)-1] = '\0';
51 DEBUG(3, ("[%5lu]: dsgetdcname for %s\n", (unsigned long)state->pid,
52 state->request.domain_name));
54 sendto_child(state, locator_child());
57 static enum winbindd_result dual_dsgetdcname(struct winbindd_domain *domain,
58 struct winbindd_cli_state *state)
61 struct netr_DsRGetDCNameInfo *info = NULL;
62 const char *dc = NULL;
64 state->request.domain_name
65 [sizeof(state->request.domain_name)-1] = '\0';
67 DEBUG(3, ("[%5lu]: dsgetdcname for %s\n", (unsigned long)state->pid,
68 state->request.domain_name));
70 result = dsgetdcname(state->mem_ctx, state->request.domain_name,
71 NULL, NULL, state->request.flags, &info);
73 if (!NT_STATUS_IS_OK(result)) {
74 return WINBINDD_ERROR;
77 if (info->dc_address) {
78 dc = info->dc_address;
79 if ((dc[0] == '\\') && (dc[1] == '\\')) {
84 if ((!dc || !is_ipaddress_v4(dc)) && info->dc_unc) {
89 return WINBINDD_ERROR;
92 fstrcpy(state->response.data.dc_name, dc);
97 static void ndr_child_get_dc_info_comapt_ds(struct winbindd_domain *domain,
98 struct winbindd_cli_state *state,
99 struct winbind_get_dc_info *r)
102 struct DS_DOMAIN_CONTROLLER_INFO *info = NULL;
103 const char *dc = NULL;
105 DEBUG(3, ("dsgetdcname for '%s'\n", r->in.domain_name));
107 result = dsgetdcname(state->mem_ctx, r->in.domain_name,
108 NULL, NULL, r->in.params.flags, &info);
109 if (!NT_STATUS_IS_OK(result)) {
110 DEBUG(1, ("dsgetdcname failed: %s\n", nt_errstr(result)));
111 r->out.result = WINBIND_STATUS_FOOBAR;
115 if (info->domain_controller_address) {
116 dc = info->domain_controller_address;
117 if ((dc[0] == '\\') && (dc[1] == '\\')) {
122 if ((!dc || !is_ipaddress_v4(dc)) && info->domain_controller_name) {
123 dc = info->domain_controller_name;
128 DEBUG(1, ("DsGetDcName failed: no dc name\n"));
129 r->out.result = WINBIND_STATUS_FOOBAR;
133 r->out.dc_info->name = talloc_strdup(r, dc);
135 if (!r->out.dc_info->name) {
136 r->out.result = WINBIND_STATUS_NO_MEMORY;
139 r->out.result = WINBIND_STATUS_OK;
142 void winbindd_ndr_locator_child_get_dc_info(struct winbindd_domain *domain,
143 struct winbindd_cli_state *state)
145 struct winbind_get_dc_info *r;
147 r = talloc_get_type_abort(state->c.ndr.r,
148 struct winbind_get_dc_info);
150 switch (*r->in.level) {
151 case WINBIND_DC_INFO_LEVEL_COMPAT_NT4:
152 r->out.result = WINBIND_STATUS_INVALID_LEVEL;
155 case WINBIND_DC_INFO_LEVEL_COMPAT_DS:
156 ndr_child_get_dc_info_comapt_ds(domain, state, r);
160 r->out.result = WINBIND_STATUS_UNKNOWN_LEVEL;
164 static const struct winbindd_child_dispatch_table locator_dispatch_table[] = {
166 .name = "DSGETDCNAME",
167 .struct_cmd = WINBINDD_DSGETDCNAME,
168 .struct_fn = dual_dsgetdcname,
170 .name = "NDR_WINBIND_GET_DC_INFO",
171 .ndr_opnum = NDR_WINBIND_GET_DC_INFO,
172 .ndr_fn = winbindd_ndr_locator_child_get_dc_info,