2 Unix SMB/CIFS implementation.
4 Get a struct wb_dom_info for a domain using DNS, netbios, possibly cldap
7 Copyright (C) Volker Lendecke 2005
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/>.
24 #include "libcli/composite/composite.h"
25 #include "libcli/resolve/resolve.h"
26 #include "libcli/security/security.h"
27 #include "winbind/wb_server.h"
28 #include "smbd/service_task.h"
29 #include "libcli/finddc.h"
30 #include "lib/socket/netif.h"
31 #include "param/param.h"
33 struct get_dom_info_state
{
34 struct composite_context
*ctx
;
35 struct wb_dom_info
*info
;
38 static void get_dom_info_recv_addrs(struct tevent_req
*req
);
40 struct composite_context
*wb_get_dom_info_send(TALLOC_CTX
*mem_ctx
,
41 struct wbsrv_service
*service
,
42 const char *domain_name
,
43 const char *dns_domain_name
,
44 const struct dom_sid
*sid
)
46 struct composite_context
*result
;
47 struct tevent_req
*req
;
48 struct get_dom_info_state
*state
;
49 struct dom_sid
*dom_sid
;
50 struct finddcs finddcs_io
;
52 DEBUG(5, ("wb_get_dom_info_send called\n"));
53 result
= composite_create(mem_ctx
, service
->task
->event_ctx
);
54 if (result
== NULL
) goto failed
;
56 state
= talloc(result
, struct get_dom_info_state
);
57 if (state
== NULL
) goto failed
;
59 result
->private_data
= state
;
61 state
->info
= talloc_zero(state
, struct wb_dom_info
);
62 if (state
->info
== NULL
) goto failed
;
64 state
->info
->name
= talloc_strdup(state
->info
, domain_name
);
65 if (state
->info
->name
== NULL
) goto failed
;
67 state
->info
->sid
= dom_sid_dup(state
->info
, sid
);
68 if (state
->info
->sid
== NULL
) goto failed
;
70 if (dom_sid_equal(sid
, &global_sid_Builtin
) ||
71 ((lpcfg_server_role(service
->task
->lp_ctx
) != ROLE_DOMAIN_MEMBER
) &&
72 dom_sid_equal(sid
, service
->primary_sid
) &&
73 service
->sec_channel_type
!= SEC_CHAN_RODC
)) {
74 struct interface
*ifaces
= NULL
;
76 load_interface_list(state
, service
->task
->lp_ctx
, &ifaces
);
78 state
->info
->dc
= talloc(state
->info
, struct nbt_dc_name
);
80 state
->info
->dc
->address
= talloc_strdup(state
->info
->dc
,
81 iface_list_n_ip(ifaces
, 0));
82 state
->info
->dc
->name
= talloc_strdup(state
->info
->dc
,
83 lpcfg_netbios_name(service
->task
->lp_ctx
));
85 composite_done(state
->ctx
);
89 dom_sid
= dom_sid_dup(mem_ctx
, sid
);
90 if (dom_sid
== NULL
) goto failed
;
92 ZERO_STRUCT(finddcs_io
);
93 finddcs_io
.in
.domain_name
= dns_domain_name
;
94 finddcs_io
.in
.domain_sid
= dom_sid
;
95 finddcs_io
.in
.minimum_dc_flags
= NBT_SERVER_LDAP
| NBT_SERVER_DS
;
96 if (service
->sec_channel_type
== SEC_CHAN_RODC
) {
97 finddcs_io
.in
.minimum_dc_flags
|= NBT_SERVER_WRITABLE
;
100 req
= finddcs_cldap_send(mem_ctx
, &finddcs_io
,
101 lpcfg_resolve_context(service
->task
->lp_ctx
),
102 service
->task
->event_ctx
);
103 if (req
== NULL
) goto failed
;
105 tevent_req_set_callback(req
, get_dom_info_recv_addrs
, state
);
114 static void get_dom_info_recv_addrs(struct tevent_req
*req
)
116 struct get_dom_info_state
*state
= tevent_req_callback_data(req
, struct get_dom_info_state
);
117 struct finddcs finddcs_io
;
119 state
->info
->dc
= talloc(state
->info
, struct nbt_dc_name
);
121 state
->ctx
->status
= finddcs_cldap_recv(req
, state
->info
, &finddcs_io
);
122 if (!composite_is_ok(state
->ctx
)) return;
124 if (finddcs_io
.out
.netlogon
.ntver
!= NETLOGON_NT_VERSION_5EX
) {
125 /* the finddcs code should have mapped the response to
127 DEBUG(0,(__location__
": unexpected ntver 0x%08x in finddcs response\n",
128 finddcs_io
.out
.netlogon
.ntver
));
129 state
->ctx
->status
= NT_STATUS_UNEXPECTED_NETWORK_ERROR
;
130 if (!composite_is_ok(state
->ctx
)) return;
133 state
->info
->dc
->address
= finddcs_io
.out
.address
;
134 state
->info
->dc
->name
= finddcs_io
.out
.netlogon
.data
.nt5_ex
.pdc_dns_name
;
136 composite_done(state
->ctx
);
139 NTSTATUS
wb_get_dom_info_recv(struct composite_context
*ctx
,
141 struct wb_dom_info
**result
)
143 NTSTATUS status
= composite_wait(ctx
);
144 if (NT_STATUS_IS_OK(status
)) {
145 struct get_dom_info_state
*state
=
146 talloc_get_type(ctx
->private_data
,
147 struct get_dom_info_state
);
148 *result
= talloc_steal(mem_ctx
, state
->info
);
154 NTSTATUS
wb_get_dom_info(TALLOC_CTX
*mem_ctx
,
155 struct wbsrv_service
*service
,
156 const char *domain_name
,
157 const char *dns_domain_name
,
158 const struct dom_sid
*sid
,
159 struct wb_dom_info
**result
)
161 struct composite_context
*ctx
=
162 wb_get_dom_info_send(mem_ctx
, service
, domain_name
, dns_domain_name
, sid
);
163 return wb_get_dom_info_recv(ctx
, mem_ctx
, result
);